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 = ▭
+ }
- 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
next prev 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