On 1/1/26 12:42 PM, David Laight wrote:

Hello David,

  static void rcar_mipi_dsi_pll_calc(struct rcar_mipi_dsi *dsi,
                                   unsigned long fin_rate,
                                   unsigned long fout_target,
-                                  struct dsi_setup_info *setup_info)
+                                  struct dsi_setup_info *setup_info,
+                                  u16 vclk_divider)

There is no need for this to be u16, unsigned int will generate better code.

Can you please elaborate on the "better code" part ?

  {
        unsigned int best_err = -1;
        const struct rcar_mipi_dsi_device_info *info = dsi->info;
@@ -360,7 +373,7 @@ static void rcar_mipi_dsi_pll_calc(struct rcar_mipi_dsi 
*dsi,
                        if (fout < info->fout_min || fout > info->fout_max)
                                continue;
- fout = div64_u64(fout, setup_info->vclk_divider);
+                       fout = div64_u64(fout, vclk_divider);

Since vclk_divider is BIT_U32(div [+ 1]) this is just a shift right.
So pass the bit number instead.

While I agree this is a shift in the end, it also makes the code harder to understand. I can add the following change into V2, but I would like input from Tomi/Laurent on whether we should really push the optimization in that direction:

"
diff --git a/drivers/gpu/drm/renesas/rcar-du/rcar_mipi_dsi.c b/drivers/gpu/drm/renesas/rcar-du/rcar_mipi_dsi.c
index 875945bf8255b..c2a0c89b97d93 100644
--- a/drivers/gpu/drm/renesas/rcar-du/rcar_mipi_dsi.c
+++ b/drivers/gpu/drm/renesas/rcar-du/rcar_mipi_dsi.c
@@ -351,7 +351,7 @@ static void rcar_mipi_dsi_pll_calc(struct rcar_mipi_dsi *dsi,
                                   unsigned long fin_rate,
                                   unsigned long fout_target,
                                   struct dsi_setup_info *setup_info,
-                                  u16 vclk_divider)
+                                  unsigned int div)
 {
        unsigned int best_err = -1;
        const struct rcar_mipi_dsi_device_info *info = dsi->info;
@@ -373,7 +373,7 @@ static void rcar_mipi_dsi_pll_calc(struct rcar_mipi_dsi *dsi,
                        if (fout < info->fout_min || fout > info->fout_max)
                                continue;

-                       fout = div64_u64(fout, vclk_divider);
+                       fout >>= div;

                        if (fout < setup_info->clkset->min_freq ||
                            fout > setup_info->clkset->max_freq)
@@ -405,7 +405,6 @@ static void rcar_mipi_dsi_parameters_calc(struct rcar_mipi_dsi *dsi,
        unsigned int i;
        unsigned int div;
        unsigned int err;
-       u16 vclk_divider;

        /*
         * Calculate Fout = dot clock * ColorDepth / (2 * Lane Count)
@@ -429,18 +428,10 @@ static void rcar_mipi_dsi_parameters_calc(struct rcar_mipi_dsi *dsi,

        div = rcar_mipi_dsi_vclk_divider(dsi, setup_info);

-       switch (dsi->info->model) {
-       case RCAR_DSI_V3U:
-       default:
-               vclk_divider = BIT_U32(div);
-               break;
-
-       case RCAR_DSI_V4H:
-               vclk_divider = BIT_U32(div + 1);
-               break;
-       }
+       if (dsi->info->model == RCAR_DSI_V4H)
+               div++;

- rcar_mipi_dsi_pll_calc(dsi, fin_rate, fout_target, setup_info, vclk_divider);
+       rcar_mipi_dsi_pll_calc(dsi, fin_rate, fout_target, setup_info, div);

        /* Find hsfreqrange */
        setup_info->hsfreq = setup_info->fout * 2;
@@ -456,7 +447,7 @@ static void rcar_mipi_dsi_parameters_calc(struct rcar_mipi_dsi *dsi,
        dev_dbg(dsi->dev,
"Fout = %u * %lu / (%u * %u * %u) = %lu (target %lu Hz, error %d.%02u%%)\n",
                setup_info->m, fin_rate, dsi->info->n_mul, setup_info->n,
-               vclk_divider, setup_info->fout, fout_target,
+               BIT_U32(div), setup_info->fout, fout_target,
                err / 100, err % 100);

        dev_dbg(dsi->dev,
"

if (fout < setup_info->clkset->min_freq ||
                            fout > setup_info->clkset->max_freq)
@@ -390,7 +403,9 @@ static void rcar_mipi_dsi_parameters_calc(struct 
rcar_mipi_dsi *dsi,
        unsigned long fout_target;
        unsigned long fin_rate;
        unsigned int i;
+       unsigned int div;
        unsigned int err;
+       u16 vclk_divider;
/*
         * Calculate Fout = dot clock * ColorDepth / (2 * Lane Count)
@@ -412,18 +427,20 @@ static void rcar_mipi_dsi_parameters_calc(struct 
rcar_mipi_dsi *dsi,
fin_rate = clk_get_rate(clk); + div = rcar_mipi_dsi_vclk_divider(dsi, setup_info);
+
        switch (dsi->info->model) {
        case RCAR_DSI_V3U:
        default:
-               setup_info->vclk_divider = 1 << ((clk_cfg->vco_cntrl >> 4) & 
0x3);
+               vclk_divider = BIT_U32(div);
                break;
case RCAR_DSI_V4H:
-               setup_info->vclk_divider = 1 << (((clk_cfg->vco_cntrl >> 3) & 
0x7) + 1);
+               vclk_divider = BIT_U32(div + 1);
                break;
        }
- rcar_mipi_dsi_pll_calc(dsi, fin_rate, fout_target, setup_info);
+       rcar_mipi_dsi_pll_calc(dsi, fin_rate, fout_target, setup_info, 
vclk_divider);
/* Find hsfreqrange */
        setup_info->hsfreq = setup_info->fout * 2;
@@ -439,7 +456,7 @@ static void rcar_mipi_dsi_parameters_calc(struct 
rcar_mipi_dsi *dsi,
        dev_dbg(dsi->dev,
                "Fout = %u * %lu / (%u * %u * %u) = %lu (target %lu Hz, error 
%d.%02u%%)\n",
                setup_info->m, fin_rate, dsi->info->n_mul, setup_info->n,
-               setup_info->vclk_divider, setup_info->fout, fout_target,
+               vclk_divider, setup_info->fout, fout_target,
                err / 100, err % 100);
dev_dbg(dsi->dev,
@@ -653,11 +670,11 @@ static int rcar_mipi_dsi_startup(struct rcar_mipi_dsi 
*dsi,
        switch (dsi->info->model) {
        case RCAR_DSI_V3U:
        default:
-               vclkset |= VCLKSET_DIV_V3U(__ffs(setup_info.vclk_divider));
+               vclkset |= VCLKSET_DIV_V3U(rcar_mipi_dsi_vclk_divider(dsi, 
&setup_info));

What is going on here?
        rcar_mipi_dsi_vclk_divider() is (setup_info->clkset->vco_cntrl >> 4) & 
0x3
        VCLKSET_DIV_V3U(n)              FIELD_PREP(VCLKSET_DIV_V3U_MASK, (n))
        VCLKSET_DIV_V3U_MASK is         GENMASK_U32(5, 4)
Looks like a very complicated way of saying:
                vclkset |= setup_info->clkset->vco_cntrl & VCLKSET_DIV_V3U_MASK;

It might be a semi-accident that the bit numbers match.
But I also suspect it is also semi-deliberate.

The bitfield placement does match here, it does not match for the V4H below anymore. The vco_cntrl is a precalculated value for a bitfield that is written into another register, this bitfield here uses a subset of it and must be kept in sync with the other register. The FIELD_PREP usage here is correct.

                break;
case RCAR_DSI_V4H:
-               vclkset |= VCLKSET_DIV_V4H(__ffs(setup_info.vclk_divider) - 1);
+               vclkset |= VCLKSET_DIV_V4H(rcar_mipi_dsi_vclk_divider(dsi, 
&setup_info));

That one looks like it needs the 'subtract one' done somewhere.
Maybe:
                vclkset |= (setup_info->clkset->vco_cntrl - (1u << 3)) &
                        VCLKSET_DIV_V4U_MASK;
Although you may want to write the '8' in a different (longer) way :-)
Please see hunk "@@ -412,18 +427,20 @@ static void rcar_mipi_dsi_parameters_calc(struct rcar_mipi_dsi *dsi," above, that one removes the +1 part, so the -1 has to be removed here too.

Thanks

Reply via email to