Quoting David Heidelberg (2026-02-28 22:37:57)
> On 18/01/2026 13:05, Kieran Bingham wrote:
> > Quoting Bryan O'Donoghue (2026-01-17 21:38:17)
> >> On 17/01/2026 15:36, David Heidelberg via B4 Relay wrote:
> >>> From: David Heidelberg <[email protected]>
> >>>
> >>> So far, only D-PHY mode was supported, which uses even bits when enabling
> >>> or masking lanes. For C-PHY configuration, the hardware instead requires
> >>> using the odd bits.
> >>>
> >>> Since there can be unrecognized configuration allow returning failure.
> >>>
> >>> Signed-off-by: David Heidelberg <[email protected]>
> >>> ---
> >>> .../platform/qcom/camss/camss-csiphy-2ph-1-0.c | 8 ++--
> >>> .../platform/qcom/camss/camss-csiphy-3ph-1-0.c | 49
> >>> +++++++++++++++++-----
> >>> drivers/media/platform/qcom/camss/camss-csiphy.c | 4 +-
> >>> drivers/media/platform/qcom/camss/camss-csiphy.h | 6 +--
> >>> 4 files changed, 47 insertions(+), 20 deletions(-)
> >>>
> >>> diff --git a/drivers/media/platform/qcom/camss/camss-csiphy-2ph-1-0.c
> >>> b/drivers/media/platform/qcom/camss/camss-csiphy-2ph-1-0.c
> >>> index 9d67e7fa6366a..bb4b91f69616b 100644
> >>> --- a/drivers/media/platform/qcom/camss/camss-csiphy-2ph-1-0.c
> >>> +++ b/drivers/media/platform/qcom/camss/camss-csiphy-2ph-1-0.c
> >>> @@ -94,9 +94,9 @@ static u8 csiphy_settle_cnt_calc(s64 link_freq, u32
> >>> timer_clk_rate)
> >>> return settle_cnt;
> >>> }
> >>>
> >>> -static void csiphy_lanes_enable(struct csiphy_device *csiphy,
> >>> - struct csiphy_config *cfg,
> >>> - s64 link_freq, u8 lane_mask)
> >>> +static int csiphy_lanes_enable(struct csiphy_device *csiphy,
> >>> + struct csiphy_config *cfg,
> >>> + s64 link_freq, u8 lane_mask)
> >>> {
> >>> struct csiphy_lanes_cfg *c = &cfg->csi2->lane_cfg;
> >>> u8 settle_cnt;
> >>> @@ -132,6 +132,8 @@ static void csiphy_lanes_enable(struct csiphy_device
> >>> *csiphy,
> >>> writel_relaxed(0x3f, csiphy->base +
> >>> CAMSS_CSI_PHY_INTERRUPT_CLEARn(l));
> >>> }
> >>> +
> >>> + return 0;
> >>> }
> >>>
> >>> static void csiphy_lanes_disable(struct csiphy_device *csiphy,
> >>> diff --git a/drivers/media/platform/qcom/camss/camss-csiphy-3ph-1-0.c
> >>> b/drivers/media/platform/qcom/camss/camss-csiphy-3ph-1-0.c
> >>> index 4154832745525..f3a8625511e1e 100644
> >>> --- a/drivers/media/platform/qcom/camss/camss-csiphy-3ph-1-0.c
> >>> +++ b/drivers/media/platform/qcom/camss/camss-csiphy-3ph-1-0.c
> >>> @@ -14,6 +14,7 @@
> >>> #include <linux/delay.h>
> >>> #include <linux/interrupt.h>
> >>> #include <linux/io.h>
> >>> +#include <linux/media-bus-format.h>
> >>>
> >>> #define CSIPHY_3PH_LNn_CFG1(n) (0x000 + 0x100 *
> >>> (n))
> >>> #define CSIPHY_3PH_LNn_CFG1_SWI_REC_DLY_PRG (BIT(7) | BIT(6))
> >>> @@ -993,13 +994,22 @@ static void csiphy_gen2_config_lanes(struct
> >>> csiphy_device *csiphy,
> >>>
> >>> static u8 csiphy_get_lane_mask(struct csiphy_lanes_cfg *lane_cfg)
> >>> {
> >>> - u8 lane_mask;
> >>> - int i;
> >>> + u8 lane_mask = 0;
> >>>
> >>> - lane_mask = CSIPHY_3PH_CMN_CSI_COMMON_CTRL5_CLK_ENABLE;
> >>> + switch (lane_cfg->phy_cfg) {
> >>> + case V4L2_MBUS_CSI2_CPHY:
> >>> + for (int i = 0; i < lane_cfg->num_data; i++)
> >>> + lane_mask |= (1 << lane_cfg->data[i].pos) + 1;
> >>
> >> 1 << anything == BIT(anything)
> >>
> >> I've always disliked the look of this code and now it occurs to me why.
> >>
> >> This code is analogous to:
> >>
> >> lane_mask |= BIT(lane_cfg->data[i].pos) + 1);
> >
> > I see that addition to a bit mask and get a little bit scared.
> >
> > This gives:
> > pos mask
> > 0 0b00000010 (note 0 bit is zero here but 1 on all others)
> > 1 0b00000011
> > 2 0b00000101
> > 3 0b00001001
> > 4 0b00010001
> >
> > Is that expected?
> >
> > Can data[i].pos ever be position 0 ??
> >
> > I assume this starts at position 1 - and the +1 here is to always set
> > the zeroth bit ?
> >
> > Perhapse this might be precise to convey that in such a case?
> >
> > lane_mask |= BIT(pos) | 1;
> >
> > I guess it depends on what this is really being used for which I don't
> > have in my context.
>
> Ok, I started looking again into the lovely downstream code.
>
> D-PHY has bits 0b0D_D_D_D
> C-PHY has bits 0b0_C_C_C_
>
> so for some reason it worked in my usecase without proper lane mask, but
> the original formula should be
>
> - lane_mask |= (1 << lane_cfg->data[i].pos) + 1;
> + lane_mask |= (1 << lane_cfg->data[i].pos + 1);
>
> Thus
>
> BIT(lane_cfg->data[i].pos + 1);
That looks a lot more sane!
--
Kieran
>
> >
> > --
> > Kieran
> >
> >
> >
> >>
> >> but BIT() is less janky and more upstreamy.
> >>
> >> janky/upstreamy - this is the on-point technical argument y'all came
> >> here for :)
> >>
> >>> + break;
> >>> + case V4L2_MBUS_CSI2_DPHY:
> >>> + lane_mask = CSIPHY_3PH_CMN_CSI_COMMON_CTRL5_CLK_ENABLE;
> >>>
> >>> - for (i = 0; i < lane_cfg->num_data; i++)
> >>> - lane_mask |= 1 << lane_cfg->data[i].pos;
> >>> + for (int i = 0; i < lane_cfg->num_data; i++)
> >>> + lane_mask |= 1 << lane_cfg->data[i].pos;
> >>> + break;
> >>> + default:
> >>> + break;
> >>> + }
> >>>
> >>> return lane_mask;
> >>> }
> >>> @@ -1027,10 +1037,11 @@ static bool csiphy_is_gen2(u32 version)
> >>> return ret;
> >>> }
> >>>
> >>> -static void csiphy_lanes_enable(struct csiphy_device *csiphy,
> >>> - struct csiphy_config *cfg,
> >>> - s64 link_freq, u8 lane_mask)
> >>> +static int csiphy_lanes_enable(struct csiphy_device *csiphy,
> >>> + struct csiphy_config *cfg,
> >>> + s64 link_freq, u8 lane_mask)
> >>> {
> >>> + struct device *dev = csiphy->camss->dev;
> >>> struct csiphy_lanes_cfg *c = &cfg->csi2->lane_cfg;
> >>> struct csiphy_device_regs *regs = csiphy->regs;
> >>> u8 settle_cnt;
> >>> @@ -1039,9 +1050,23 @@ static void csiphy_lanes_enable(struct
> >>> csiphy_device *csiphy,
> >>>
> >>> settle_cnt = csiphy_settle_cnt_calc(link_freq,
> >>> csiphy->timer_clk_rate);
> >>>
> >>> - val = CSIPHY_3PH_CMN_CSI_COMMON_CTRL5_CLK_ENABLE;
> >>> - for (i = 0; i < c->num_data; i++)
> >>> - val |= BIT(c->data[i].pos * 2);
> >>> + val = 0;
> >>> +
> >>> + switch (c->phy_cfg) {
> >>> + case V4L2_MBUS_CSI2_CPHY:
> >>> + for (i = 0; i < c->num_data; i++)
> >>> + val |= BIT((c->data[i].pos * 2) + 1);
> >>> + break;
> >>> + case V4L2_MBUS_CSI2_DPHY:
> >>> + val = CSIPHY_3PH_CMN_CSI_COMMON_CTRL5_CLK_ENABLE;
> >>> +
> >>> + for (i = 0; i < c->num_data; i++)
> >>> + val |= BIT(c->data[i].pos * 2);
> >>> + break;
> >>> + default:
> >>> + dev_err(dev, "Unsupported bus type %d\n", c->phy_cfg);
> >>> + return -EINVAL;
> >>> + }
> >>>
> >>> writel_relaxed(val, csiphy->base +
> >>> CSIPHY_3PH_CMN_CSI_COMMON_CTRLn(regs->offset, 5));
> >>> @@ -1068,6 +1093,8 @@ static void csiphy_lanes_enable(struct
> >>> csiphy_device *csiphy,
> >>> writel_relaxed(0, csiphy->base +
> >>>
> >>> CSIPHY_3PH_CMN_CSI_COMMON_CTRLn(regs->offset, i));
> >>> }
> >>> +
> >>> + return 0;
> >>> }
> >>>
> >>> static void csiphy_lanes_disable(struct csiphy_device *csiphy,
> >>> diff --git a/drivers/media/platform/qcom/camss/camss-csiphy.c
> >>> b/drivers/media/platform/qcom/camss/camss-csiphy.c
> >>> index 62623393f4144..08dd238e52799 100644
> >>> --- a/drivers/media/platform/qcom/camss/camss-csiphy.c
> >>> +++ b/drivers/media/platform/qcom/camss/camss-csiphy.c
> >>> @@ -295,9 +295,7 @@ static int csiphy_stream_on(struct csiphy_device
> >>> *csiphy)
> >>> wmb();
> >>> }
> >>>
> >>> - csiphy->res->hw_ops->lanes_enable(csiphy, cfg, link_freq,
> >>> lane_mask);
> >>> -
> >>> - return 0;
> >>> + return csiphy->res->hw_ops->lanes_enable(csiphy, cfg, link_freq,
> >>> lane_mask);
> >>
> >> ick.
> >>
> >> More high brow stuff from bod here but, more seriously this is three
> >> levels of indirection deep and the statement keeps getting longer.
> >>
> >> Could you get a pointer to hw_ops() to reduce this down a bit.
> >>
> >>> }
> >>>
> >>> /*
> >>> diff --git a/drivers/media/platform/qcom/camss/camss-csiphy.h
> >>> b/drivers/media/platform/qcom/camss/camss-csiphy.h
> >>> index d198171700e73..21cf2ce931c1d 100644
> >>> --- a/drivers/media/platform/qcom/camss/camss-csiphy.h
> >>> +++ b/drivers/media/platform/qcom/camss/camss-csiphy.h
> >>> @@ -73,9 +73,9 @@ struct csiphy_hw_ops {
> >>> void (*hw_version_read)(struct csiphy_device *csiphy,
> >>> struct device *dev);
> >>> void (*reset)(struct csiphy_device *csiphy);
> >>> - void (*lanes_enable)(struct csiphy_device *csiphy,
> >>> - struct csiphy_config *cfg,
> >>> - s64 link_freq, u8 lane_mask);
> >>> + int (*lanes_enable)(struct csiphy_device *csiphy,
> >>> + struct csiphy_config *cfg,
> >>> + s64 link_freq, u8 lane_mask);
> >>> void (*lanes_disable)(struct csiphy_device *csiphy,
> >>> struct csiphy_config *cfg);
> >>> irqreturn_t (*isr)(int irq, void *dev);
> >>>
> >>
> >> With those tweaks.
> >>
> >> Reviewed-by: Bryan O'Donoghue <[email protected]>
> >>
> >> ---
> >> bod
>
> --
> David Heidelberg
>