[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <1b783a60-39c7-49b5-8932-e77230f6cddd@linaro.org>
Date: Mon, 20 Oct 2025 16:05:59 +0200
From: neil.armstrong@...aro.org
To: Jessica Zhang <jessica.zhang@....qualcomm.com>,
Rob Clark <robdclark@...il.com>, Dmitry Baryshkov <lumag@...nel.org>,
Sean Paul <sean@...rly.run>, Marijn Suijten <marijn.suijten@...ainline.org>,
David Airlie <airlied@...il.com>, Simona Vetter <simona@...ll.ch>
Cc: Abhinav Kumar <quic_abhinavk@...cinc.com>, linux-arm-msm@...r.kernel.org,
dri-devel@...ts.freedesktop.org, freedreno@...ts.freedesktop.org,
linux-kernel@...r.kernel.org
Subject: Re: [PATCH v2] drm/msm/dpu: Filter modes based on adjusted mode clock
Hi,
On 5/7/25 03:38, Jessica Zhang wrote:
> Filter out modes that have a clock rate greater than the max core clock
> rate when adjusted for the perf clock factor
>
> This is especially important for chipsets such as QCS615 that have lower
> limits for the MDP max core clock.
>
> Since the core CRTC clock is at least the mode clock (adjusted for the
> perf clock factor) [1], the modes supported by the driver should be less
> than the max core clock rate.
>
> [1] https://elixir.bootlin.com/linux/v6.12.4/source/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c#L83
>
> Reviewed-by: Dmitry Baryshkov <dmitry.baryshkov@...aro.org>
> Signed-off-by: Jessica Zhang <jessica.zhang@....qualcomm.com>
This change breaks the T14s OLED display, no modes are reported on the eDP connector.
msm_dpu ae01000.display-controller: [drm:update_display_info.part.0 [drm]] [CONNECTOR:41:eDP-1] Assigning EDID-1.4 digital sink color depth as 10 bpc.
msm_dpu ae01000.display-controller: [drm:update_display_info.part.0 [drm]] [CONNECTOR:41:eDP-1] ELD monitor
msm_dpu ae01000.display-controller: [drm:update_display_info.part.0 [drm]] [CONNECTOR:41:eDP-1] ELD size 20, SAD count 0
[drm:drm_mode_object_put.part.0 [drm]] OBJ ID: 113 (1)
msm_dpu ae01000.display-controller: [drm:drm_mode_prune_invalid [drm]] Rejected mode: "2880x1800": 120 652260 2880 2912 2920 2980 1800 1808 1816 1824 0x48 0x9 (CLOCK_HIGH)
msm_dpu ae01000.display-controller: [drm:drm_mode_prune_invalid [drm]] Rejected mode: "2880x1800": 60 652260 2880 2888 2920 2980 1800 1808 1816 3648 0x40 0x9 (CLOCK_HIGH)
With this reverted on v6.18-rc, display works again.
Neil
> ---
> Changes in v2:
> - *crtc_clock -> *mode_clock (Dmitry)
> - Changed adjusted_mode_clk check to use multiplication (Dmitry)
> - Switch from quic_* email to OSS email
> - Link to v1: https://lore.kernel.org/lkml/20241212-filter-mode-clock-v1-1-f4441988d6aa@quicinc.com/
> ---
> drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c | 35 ++++++++++++++++++---------
> drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h | 3 +++
> drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c | 12 +++++++++
> 3 files changed, 39 insertions(+), 11 deletions(-)
>
> diff --git a/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c b/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c
> index 0fb5789c60d0..13cc658065c5 100644
> --- a/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c
> +++ b/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.c
> @@ -31,6 +31,26 @@ enum dpu_perf_mode {
> DPU_PERF_MODE_MAX
> };
>
> +/**
> + * dpu_core_perf_adjusted_mode_clk - Adjust given mode clock rate according to
> + * the perf clock factor.
> + * @crtc_clk_rate - Unadjusted mode clock rate
> + * @perf_cfg: performance configuration
> + */
> +u64 dpu_core_perf_adjusted_mode_clk(u64 mode_clk_rate,
> + const struct dpu_perf_cfg *perf_cfg)
> +{
> + u32 clk_factor;
> +
> + clk_factor = perf_cfg->clk_inefficiency_factor;
> + if (clk_factor) {
> + mode_clk_rate *= clk_factor;
> + do_div(mode_clk_rate, 100);
> + }
> +
> + return mode_clk_rate;
> +}
> +
> /**
> * _dpu_core_perf_calc_bw() - to calculate BW per crtc
> * @perf_cfg: performance configuration
> @@ -75,28 +95,21 @@ static u64 _dpu_core_perf_calc_clk(const struct dpu_perf_cfg *perf_cfg,
> struct drm_plane *plane;
> struct dpu_plane_state *pstate;
> struct drm_display_mode *mode;
> - u64 crtc_clk;
> - u32 clk_factor;
> + u64 mode_clk;
>
> mode = &state->adjusted_mode;
>
> - crtc_clk = (u64)mode->vtotal * mode->hdisplay * drm_mode_vrefresh(mode);
> + mode_clk = (u64)mode->vtotal * mode->hdisplay * drm_mode_vrefresh(mode);
>
> drm_atomic_crtc_for_each_plane(plane, crtc) {
> pstate = to_dpu_plane_state(plane->state);
> if (!pstate)
> continue;
>
> - crtc_clk = max(pstate->plane_clk, crtc_clk);
> - }
> -
> - clk_factor = perf_cfg->clk_inefficiency_factor;
> - if (clk_factor) {
> - crtc_clk *= clk_factor;
> - do_div(crtc_clk, 100);
> + mode_clk = max(pstate->plane_clk, mode_clk);
> }
>
> - return crtc_clk;
> + return dpu_core_perf_adjusted_mode_clk(mode_clk, perf_cfg);
> }
>
> static struct dpu_kms *_dpu_crtc_get_kms(struct drm_crtc *crtc)
> diff --git a/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h b/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h
> index d2f21d34e501..3740bc97422c 100644
> --- a/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h
> +++ b/drivers/gpu/drm/msm/disp/dpu1/dpu_core_perf.h
> @@ -54,6 +54,9 @@ struct dpu_core_perf {
> u32 fix_core_ab_vote;
> };
>
> +u64 dpu_core_perf_adjusted_mode_clk(u64 clk_rate,
> + const struct dpu_perf_cfg *perf_cfg);
> +
> int dpu_core_perf_crtc_check(struct drm_crtc *crtc,
> struct drm_crtc_state *state);
>
> diff --git a/drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c b/drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c
> index 0714936d8835..5e3c34fed63b 100644
> --- a/drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c
> +++ b/drivers/gpu/drm/msm/disp/dpu1/dpu_crtc.c
> @@ -1501,6 +1501,7 @@ static enum drm_mode_status dpu_crtc_mode_valid(struct drm_crtc *crtc,
> const struct drm_display_mode *mode)
> {
> struct dpu_kms *dpu_kms = _dpu_crtc_get_kms(crtc);
> + u64 adjusted_mode_clk;
>
> /* if there is no 3d_mux block we cannot merge LMs so we cannot
> * split the large layer into 2 LMs, filter out such modes
> @@ -1508,6 +1509,17 @@ static enum drm_mode_status dpu_crtc_mode_valid(struct drm_crtc *crtc,
> if (!dpu_kms->catalog->caps->has_3d_merge &&
> mode->hdisplay > dpu_kms->catalog->caps->max_mixer_width)
> return MODE_BAD_HVALUE;
> +
> + adjusted_mode_clk = dpu_core_perf_adjusted_mode_clk(mode->clock,
> + dpu_kms->perf.perf_cfg);
> +
> + /*
> + * The given mode, adjusted for the perf clock factor, should not exceed
> + * the max core clock rate
> + */
> + if (dpu_kms->perf.max_core_clk_rate < adjusted_mode_clk * 1000)
> + return MODE_CLOCK_HIGH;
> +
> /*
> * max crtc width is equal to the max mixer width * 2 and max height is 4K
> */
>
> ---
> base-commit: db76003ade5953d4a83c2bdc6e15c2d1c33e7350
> change-id: 20250506-filter-modes-c60b4332769f
>
> Best regards,
Powered by blists - more mailing lists