Intel-XE Archive on lore.kernel.org
 help / color / mirror / Atom feed
From: Maarten Lankhorst <dev@lankhorst.se>
To: intel-xe@lists.freedesktop.org
Subject: [FOR-CI 09/10] drm/i915/display: Remove dirty_rect from fbc state
Date: Fri,  7 Nov 2025 20:39:44 +0100	[thread overview]
Message-ID: <20251107193946.1075137-10-dev@lankhorst.se> (raw)
In-Reply-To: <20251107193946.1075137-1-dev@lankhorst.se>

It's a terrible idea to serialize against &fbc->mutex during vblank
evasion. All the relevant state is already in plane_state, use
that instead and get rid of a lot of unnecessary bookkeeping.

Reported-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Fixes: af23476af8a9 ("drm/i915/fbc: handle dirty rect coords for the first frame")
Cc: Vinod Govindapillai <vinod.govindapillai@intel.com>
Cc: Ville Syrjälä <ville.syrjala@linux.intel.com>
Cc: <stable@vger.kernel.org> # v6.15+
Signed-off-by: Maarten Lankhorst <dev@lankhorst.se>
---
 drivers/gpu/drm/i915/display/intel_display.c |   3 -
 drivers/gpu/drm/i915/display/intel_fbc.c     | 124 +++++--------------
 drivers/gpu/drm/i915/display/intel_fbc.h     |   3 +-
 drivers/gpu/drm/i915/display/intel_plane.c   |   2 +-
 4 files changed, 31 insertions(+), 101 deletions(-)

diff --git a/drivers/gpu/drm/i915/display/intel_display.c b/drivers/gpu/drm/i915/display/intel_display.c
index 45b210050d8c0..716a2e4825b6d 100644
--- a/drivers/gpu/drm/i915/display/intel_display.c
+++ b/drivers/gpu/drm/i915/display/intel_display.c
@@ -7392,9 +7392,6 @@ static void intel_atomic_commit_tail(struct intel_atomic_state *state)
 
 	intel_atomic_prepare_plane_clear_colors(state);
 
-	for_each_new_intel_crtc_in_state(state, crtc, new_crtc_state, i)
-		intel_fbc_prepare_dirty_rect(state, crtc);
-
 	for_each_new_intel_crtc_in_state(state, crtc, new_crtc_state, i)
 		intel_atomic_dsb_finish(state, crtc);
 
diff --git a/drivers/gpu/drm/i915/display/intel_fbc.c b/drivers/gpu/drm/i915/display/intel_fbc.c
index a1e3083022ee7..dee7ca9eff20d 100644
--- a/drivers/gpu/drm/i915/display/intel_fbc.c
+++ b/drivers/gpu/drm/i915/display/intel_fbc.c
@@ -91,7 +91,6 @@ struct intel_fbc_state {
 	u16 override_cfb_stride;
 	u16 interval;
 	s8 fence_id;
-	struct drm_rect dirty_rect;
 };
 
 struct intel_fbc {
@@ -1272,72 +1271,63 @@ static bool tiling_is_valid(const struct intel_plane_state *plane_state)
 		return i8xx_fbc_tiling_valid(plane_state);
 }
 
-static void
-intel_fbc_invalidate_dirty_rect(struct intel_fbc *fbc)
-{
-	lockdep_assert_held(&fbc->lock);
-
-	fbc->state.dirty_rect = DRM_RECT_INIT(0, 0, 0, 0);
-}
-
 static void
 intel_fbc_program_dirty_rect(struct intel_dsb *dsb, struct intel_fbc *fbc,
-			     const struct drm_rect *fbc_dirty_rect)
+			     const struct drm_rect *dirty_rect_fp)
 {
 	struct intel_display *display = fbc->display;
+	struct drm_rect dirty_rect;
+
+	drm_rect_fp_to_int(&dirty_rect, dirty_rect_fp);
 
-	drm_WARN_ON(display->drm, fbc_dirty_rect->y2 == 0);
+	drm_WARN_ON(display->drm, dirty_rect.y2 == 0);
 
 	intel_de_write_dsb(display, dsb, XE3_FBC_DIRTY_RECT(fbc->id),
-			   FBC_DIRTY_RECT_START_LINE(fbc_dirty_rect->y1) |
-			   FBC_DIRTY_RECT_END_LINE(fbc_dirty_rect->y2 - 1));
+			   FBC_DIRTY_RECT_START_LINE(dirty_rect.y1) |
+			   FBC_DIRTY_RECT_END_LINE(dirty_rect.y2 - 1));
 }
 
-static void
-intel_fbc_dirty_rect_update(struct intel_dsb *dsb, struct intel_fbc *fbc)
-{
-	const struct drm_rect *fbc_dirty_rect = &fbc->state.dirty_rect;
-
-	lockdep_assert_held(&fbc->lock);
-
-	if (!drm_rect_visible(fbc_dirty_rect))
-		return;
-
-	intel_fbc_program_dirty_rect(dsb, fbc, fbc_dirty_rect);
-}
+static inline bool intel_fbc_is_ok(const struct intel_plane_state *plane_state);
 
 void
 intel_fbc_dirty_rect_update_noarm(struct intel_dsb *dsb,
-				  struct intel_plane *plane)
+				  const struct intel_crtc_state *crtc_state,
+				  const struct intel_plane_state *plane_state)
 {
+	struct intel_plane *plane = to_intel_plane(plane_state->uapi.plane);
 	struct intel_display *display = to_intel_display(plane);
 	struct intel_fbc *fbc = plane->fbc;
+	const struct drm_rect *damage = &plane_state->damage;
+	struct drm_rect rect;
 
-	if (!HAS_FBC_DIRTY_RECT(display))
+	if (!HAS_FBC_DIRTY_RECT(display) ||
+	    READ_ONCE(fbc->state.plane) != plane ||
+	    intel_crtc_needs_modeset(crtc_state) ||
+	    !intel_fbc_is_ok(plane_state))
 		return;
 
-	mutex_lock(&fbc->lock);
+	if (!drm_rect_visible(damage)) {
+		int width = drm_rect_width(&plane_state->uapi.src);
+		int y_offset = plane_state->view.color_plane[0].y;
 
-	if (fbc->state.plane == plane)
-		intel_fbc_dirty_rect_update(dsb, fbc);
+		/* dirty rect must cover at least one line */
+		rect = DRM_RECT_INIT(0, y_offset << 16, width, 1 << 16);
+		damage = &rect;
+	}
 
-	mutex_unlock(&fbc->lock);
+	intel_fbc_program_dirty_rect(dsb, fbc, damage);
 }
 
 static void
 intel_fbc_hw_intialize_dirty_rect(struct intel_fbc *fbc,
 				  const struct intel_plane_state *plane_state)
 {
-	struct drm_rect src;
-
 	/*
 	 * Initializing the FBC HW with the whole plane area as the dirty rect.
 	 * This is to ensure that we have valid coords be written to the
 	 * HW as dirty rect.
 	 */
-	drm_rect_fp_to_int(&src, &plane_state->uapi.src);
-
-	intel_fbc_program_dirty_rect(NULL, fbc, &src);
+	intel_fbc_program_dirty_rect(NULL, fbc, &plane_state->uapi.src);
 }
 
 static void intel_fbc_update_state(struct intel_atomic_state *state,
@@ -1355,7 +1345,7 @@ static void intel_fbc_update_state(struct intel_atomic_state *state,
 	WARN_ON(plane_state->no_fbc_reason);
 	WARN_ON(fbc_state->plane && fbc_state->plane != plane);
 
-	fbc_state->plane = plane;
+	WRITE_ONCE(fbc_state->plane, plane);
 
 	/* FBC1 compression interval: arbitrary choice of 1 second */
 	fbc_state->interval = drm_mode_vrefresh(&crtc_state->hw.adjusted_mode);
@@ -1413,62 +1403,6 @@ static bool intel_fbc_is_ok(const struct intel_plane_state *plane_state)
 		intel_fbc_is_cfb_ok(plane_state);
 }
 
-static void
-__intel_fbc_prepare_dirty_rect(const struct intel_plane_state *plane_state,
-			       const struct intel_crtc_state *crtc_state)
-{
-	struct intel_plane *plane = to_intel_plane(plane_state->uapi.plane);
-	struct intel_fbc *fbc = plane->fbc;
-	struct drm_rect *fbc_dirty_rect = &fbc->state.dirty_rect;
-	int width = drm_rect_width(&plane_state->uapi.src) >> 16;
-	const struct drm_rect *damage = &plane_state->damage;
-	int y_offset = plane_state->view.color_plane[0].y;
-
-	lockdep_assert_held(&fbc->lock);
-
-	if (intel_crtc_needs_modeset(crtc_state) ||
-	    !intel_fbc_is_ok(plane_state)) {
-		intel_fbc_invalidate_dirty_rect(fbc);
-		return;
-	}
-
-	if (drm_rect_visible(damage))
-		*fbc_dirty_rect = *damage;
-	else
-		/* dirty rect must cover at least one line */
-		*fbc_dirty_rect = DRM_RECT_INIT(0, y_offset, width, 1);
-}
-
-void
-intel_fbc_prepare_dirty_rect(struct intel_atomic_state *state,
-			     struct intel_crtc *crtc)
-{
-	struct intel_display *display = to_intel_display(state);
-	const struct intel_crtc_state *crtc_state =
-		intel_atomic_get_new_crtc_state(state, crtc);
-	struct intel_plane_state *plane_state;
-	struct intel_plane *plane;
-	int i;
-
-	if (!HAS_FBC_DIRTY_RECT(display))
-		return;
-
-	for_each_new_intel_plane_in_state(state, plane, plane_state, i) {
-		struct intel_fbc *fbc = plane->fbc;
-
-		if (!fbc || plane->pipe != crtc->pipe)
-			continue;
-
-		mutex_lock(&fbc->lock);
-
-		if (fbc->state.plane == plane)
-			__intel_fbc_prepare_dirty_rect(plane_state,
-						       crtc_state);
-
-		mutex_unlock(&fbc->lock);
-	}
-}
-
 static int _intel_fbc_min_cdclk(const struct intel_crtc_state *crtc_state)
 {
 	struct intel_display *display = to_intel_display(crtc_state);
@@ -1771,15 +1705,13 @@ static void __intel_fbc_disable(struct intel_fbc *fbc)
 	drm_dbg_kms(display->drm, "Disabling FBC on [PLANE:%d:%s]\n",
 		    plane->base.base.id, plane->base.name);
 
-	intel_fbc_invalidate_dirty_rect(fbc);
-
 	__intel_fbc_cleanup_cfb(fbc);
 
 	/* wa_18038517565 Enable DPFC clock gating after FBC disable */
 	if (display->platform.dg2 || DISPLAY_VER(display) >= 14)
 		fbc_compressor_clkgate_disable_wa(fbc, false);
 
-	fbc->state.plane = NULL;
+	WRITE_ONCE(fbc->state.plane, NULL);
 	fbc->flip_pending = false;
 	fbc->busy_bits = 0;
 }
diff --git a/drivers/gpu/drm/i915/display/intel_fbc.h b/drivers/gpu/drm/i915/display/intel_fbc.h
index 91424563206a3..65d73cd56bc7b 100644
--- a/drivers/gpu/drm/i915/display/intel_fbc.h
+++ b/drivers/gpu/drm/i915/display/intel_fbc.h
@@ -52,7 +52,8 @@ void intel_fbc_debugfs_register(struct intel_display *display);
 void intel_fbc_prepare_dirty_rect(struct intel_atomic_state *state,
 				  struct intel_crtc *crtc);
 void intel_fbc_dirty_rect_update_noarm(struct intel_dsb *dsb,
-				       struct intel_plane *plane);
+				       const struct intel_crtc_state *crtc_state,
+				       const struct intel_plane_state *plane_state);
 bool
 intel_fbc_is_enable_pixel_normalizer(const struct intel_plane_state *plane_state);
 
diff --git a/drivers/gpu/drm/i915/display/intel_plane.c b/drivers/gpu/drm/i915/display/intel_plane.c
index 5105e3278bc46..b22edc7e62d6d 100644
--- a/drivers/gpu/drm/i915/display/intel_plane.c
+++ b/drivers/gpu/drm/i915/display/intel_plane.c
@@ -797,7 +797,7 @@ void intel_plane_update_noarm(struct intel_dsb *dsb,
 	trace_intel_plane_update_noarm(plane_state, crtc);
 
 	if (plane->fbc)
-		intel_fbc_dirty_rect_update_noarm(dsb, plane);
+		intel_fbc_dirty_rect_update_noarm(dsb, crtc_state, plane_state);
 
 	if (plane->update_noarm)
 		plane->update_noarm(dsb, plane, crtc_state, plane_state);
-- 
2.51.0


  parent reply	other threads:[~2025-11-07 19:39 UTC|newest]

Thread overview: 19+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2025-11-07 19:39 [FOR-CI 00/10] Fixed PREEMPT_RT CI run Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 01/10] drm/i915/display: Make get_vblank_counter use intel_de_read_fw() Maarten Lankhorst
2025-11-07 20:43   ` Matt Roper
2025-11-08  7:40     ` Maarten Lankhorst
2025-11-10 16:52       ` Matt Roper
2025-11-07 19:39 ` [FOR-CI 02/10] drm/i915/display: Use intel_de_write_fw in intel_pipe_fastset Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 03/10] drm/i915/display: Move vblank put until after critical section Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 04/10] drm/i915/display: Remove locking from intel_vblank_evade " Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 05/10] drm/i915/display: Make icl_dsi_frame_update use _fw too Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 06/10] drm/i915/display: Enable interrupts earlier on PREEMPT_RT Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 07/10] drm/i915: Use preempt_disable/enable_rt() where recommended Maarten Lankhorst
2025-11-07 19:39 ` [FOR-CI 08/10] drm/i915/display: Make set_pipeconf use the fw variants Maarten Lankhorst
2025-11-07 19:39 ` Maarten Lankhorst [this message]
2025-11-07 19:39 ` [FOR-CI 10/10] PREEMPT_RT injection Maarten Lankhorst
2025-11-07 23:51 ` ✗ CI.checkpatch: warning for Fixed PREEMPT_RT CI run Patchwork
2025-11-07 23:52 ` ✓ CI.KUnit: success " Patchwork
2025-11-08  0:10 ` ✗ CI.checksparse: warning " Patchwork
2025-11-08  0:50 ` ✗ Xe.CI.BAT: failure " Patchwork
2025-11-09  6:57 ` ✗ Xe.CI.Full: " Patchwork

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20251107193946.1075137-10-dev@lankhorst.se \
    --to=dev@lankhorst.se \
    --cc=intel-xe@lists.freedesktop.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox