]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
drm/i915/wm: Use skl_prefill
authorVille Syrjälä <ville.syrjala@linux.intel.com>
Tue, 14 Oct 2025 19:18:08 +0000 (22:18 +0300)
committerVille Syrjälä <ville.syrjala@linux.intel.com>
Thu, 16 Oct 2025 15:22:18 +0000 (18:22 +0300)
Replace the current ad-hoc prefill calculations with skl_prefill.

v2: cdclk_state no longer needed
    Rename to skl_prefill

Reviewed-by: Uma Shankar <uma.shankar@intel.com> #v1
Signed-off-by: Ville Syrjälä <ville.syrjala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20251014191808.12326-10-ville.syrjala@linux.intel.com
drivers/gpu/drm/i915/display/skl_watermark.c

index 700707a91c704f3769aabd8b39b193d777b496f7..256162da9afc0a6cb348d61ac01af2ab9d717ad2 100644 (file)
@@ -30,6 +30,7 @@
 #include "intel_plane.h"
 #include "intel_vblank.h"
 #include "intel_wm.h"
+#include "skl_prefill.h"
 #include "skl_scaler.h"
 #include "skl_universal_plane_regs.h"
 #include "skl_watermark.h"
@@ -2147,105 +2148,6 @@ static int icl_build_plane_wm(struct intel_crtc_state *crtc_state,
        return 0;
 }
 
-static int
-cdclk_prefill_adjustment(const struct intel_crtc_state *crtc_state)
-{
-       struct intel_display *display = to_intel_display(crtc_state);
-       struct intel_atomic_state *state =
-               to_intel_atomic_state(crtc_state->uapi.state);
-       const struct intel_cdclk_state *cdclk_state;
-
-       cdclk_state = intel_atomic_get_cdclk_state(state);
-       if (IS_ERR(cdclk_state)) {
-               drm_WARN_ON(display->drm, PTR_ERR(cdclk_state));
-               return 1;
-       }
-
-       return min(1, DIV_ROUND_UP(crtc_state->pixel_rate,
-                                  2 * intel_cdclk_logical(cdclk_state)));
-}
-
-static int
-dsc_prefill_latency(const struct intel_crtc_state *crtc_state)
-{
-       struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
-       const struct intel_crtc_scaler_state *scaler_state =
-                                       &crtc_state->scaler_state;
-       int linetime = DIV_ROUND_UP(1000 * crtc_state->hw.adjusted_mode.htotal,
-                                   crtc_state->hw.adjusted_mode.clock);
-       int num_scaler_users = hweight32(scaler_state->scaler_users);
-       int chroma_downscaling_factor =
-               crtc_state->output_format == INTEL_OUTPUT_FORMAT_YCBCR420 ? 2 : 1;
-       u32 dsc_prefill_latency = 0;
-
-       if (!crtc_state->dsc.compression_enable ||
-           !num_scaler_users ||
-           num_scaler_users > crtc->num_scalers)
-               return dsc_prefill_latency;
-
-       dsc_prefill_latency = DIV_ROUND_UP(15 * linetime * chroma_downscaling_factor, 10);
-
-       for (int i = 0; i < num_scaler_users; i++) {
-               u64 hscale_k, vscale_k;
-
-               hscale_k = max(1000, mul_u32_u32(scaler_state->scalers[i].hscale, 1000) >> 16);
-               vscale_k = max(1000, mul_u32_u32(scaler_state->scalers[i].vscale, 1000) >> 16);
-               dsc_prefill_latency = DIV_ROUND_UP_ULL(dsc_prefill_latency * hscale_k * vscale_k,
-                                                      1000000);
-       }
-
-       dsc_prefill_latency *= cdclk_prefill_adjustment(crtc_state);
-
-       return intel_usecs_to_scanlines(&crtc_state->hw.adjusted_mode, dsc_prefill_latency);
-}
-
-static int
-scaler_prefill_latency(const struct intel_crtc_state *crtc_state)
-{
-       const struct intel_crtc_scaler_state *scaler_state =
-                                       &crtc_state->scaler_state;
-       int num_scaler_users = hweight32(scaler_state->scaler_users);
-       int scaler_prefill_latency = 0;
-       int linetime = DIV_ROUND_UP(1000 * crtc_state->hw.adjusted_mode.htotal,
-                                   crtc_state->hw.adjusted_mode.clock);
-
-       if (!num_scaler_users)
-               return scaler_prefill_latency;
-
-       scaler_prefill_latency = 4 * linetime;
-
-       if (num_scaler_users > 1) {
-               u64 hscale_k = max(1000, mul_u32_u32(scaler_state->scalers[0].hscale, 1000) >> 16);
-               u64 vscale_k = max(1000, mul_u32_u32(scaler_state->scalers[0].vscale, 1000) >> 16);
-               int chroma_downscaling_factor =
-                       crtc_state->output_format == INTEL_OUTPUT_FORMAT_YCBCR420 ? 2 : 1;
-               int latency;
-
-               latency = DIV_ROUND_UP_ULL((4 * linetime * hscale_k * vscale_k *
-                                           chroma_downscaling_factor), 1000000);
-               scaler_prefill_latency += latency;
-       }
-
-       scaler_prefill_latency *= cdclk_prefill_adjustment(crtc_state);
-
-       return intel_usecs_to_scanlines(&crtc_state->hw.adjusted_mode, scaler_prefill_latency);
-}
-
-static bool
-skl_is_vblank_too_short(const struct intel_crtc_state *crtc_state,
-                       int wm0_lines, int latency)
-{
-       const struct drm_display_mode *adjusted_mode =
-               &crtc_state->hw.adjusted_mode;
-
-       return crtc_state->framestart_delay +
-               intel_usecs_to_scanlines(adjusted_mode, latency) +
-               scaler_prefill_latency(crtc_state) +
-               dsc_prefill_latency(crtc_state) +
-               wm0_lines >
-               intel_crtc_vblank_length(crtc_state);
-}
-
 unsigned int skl_wm0_prefill_lines_worst(const struct intel_crtc_state *crtc_state)
 {
        struct intel_display *display = to_intel_display(crtc_state);
@@ -2324,9 +2226,10 @@ unsigned int skl_wm0_prefill_lines(const struct intel_crtc_state *crtc_state)
  * the scenario to apply Wa_16025596647.
  */
 static int skl_max_wm_level_for_vblank(struct intel_crtc_state *crtc_state,
-                                      int wm0_lines)
+                                      const struct skl_prefill_ctx *ctx)
 {
        struct intel_display *display = to_intel_display(crtc_state);
+       struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
        int level;
 
        for (level = display->wm.num_levels - 1; level >= 0; level--) {
@@ -2341,10 +2244,13 @@ static int skl_max_wm_level_for_vblank(struct intel_crtc_state *crtc_state,
                if (level == 0)
                        latency = 0;
 
-               if (!skl_is_vblank_too_short(crtc_state, wm0_lines, latency))
+               if (!skl_prefill_vblank_too_short(ctx, crtc_state, latency))
                        return level;
        }
 
+       drm_dbg_kms(display->drm, "[CRTC:%d:%s] Not enough time in vblank for prefill\n",
+                   crtc->base.base.id, crtc->base.name);
+
        return -EINVAL;
 }
 
@@ -2352,14 +2258,15 @@ static int skl_wm_check_vblank(struct intel_crtc_state *crtc_state)
 {
        struct intel_display *display = to_intel_display(crtc_state);
        struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
-       int wm0_lines, level;
+       struct skl_prefill_ctx ctx;
+       int level;
 
        if (!crtc_state->hw.active)
                return 0;
 
-       wm0_lines = skl_max_wm0_lines(crtc_state);
+       skl_prefill_init(&ctx, crtc_state);
 
-       level = skl_max_wm_level_for_vblank(crtc_state, wm0_lines);
+       level = skl_max_wm_level_for_vblank(crtc_state, &ctx);
        if (level < 0)
                return level;
 
@@ -2369,6 +2276,13 @@ static int skl_wm_check_vblank(struct intel_crtc_state *crtc_state)
         */
        crtc_state->wm_level_disabled = level < display->wm.num_levels - 1;
 
+       /*
+        * TODO: assert that we are in fact using the maximum guardband
+        * if we end up disabling any WM levels here. Otherwise we clearly
+        * failed in using a realistic worst case prefill estimate when
+        * determining the guardband size.
+        */
+
        for (level++; level < display->wm.num_levels; level++) {
                enum plane_id plane_id;
 
@@ -2387,8 +2301,8 @@ static int skl_wm_check_vblank(struct intel_crtc_state *crtc_state)
 
        if (DISPLAY_VER(display) >= 12 &&
            display->sagv.block_time_us &&
-           skl_is_vblank_too_short(crtc_state, wm0_lines,
-                                   display->sagv.block_time_us)) {
+           skl_prefill_vblank_too_short(&ctx, crtc_state,
+                                        display->sagv.block_time_us)) {
                enum plane_id plane_id;
 
                for_each_plane_id_on_crtc(crtc, plane_id) {