From: "Ville Syrjälä" <ville.syrjala@linux.intel.com>
To: "Nautiyal, Ankit K" <ankit.k.nautiyal@intel.com>
Cc: intel-gfx@lists.freedesktop.org, intel-xe@lists.freedesktop.org,
dri-devel@lists.freedesktop.org
Subject: Re: [PATCH 4/4] drm/i915/panel: Attempt VRR based refresh rate change for !allow_modeset
Date: Thu, 18 Jun 2026 22:10:10 +0300 [thread overview]
Message-ID: <ajRCkjh-kHwg56V4@intel.com> (raw)
In-Reply-To: <e7414943-3902-4209-b372-3cddce601d78@intel.com>
On Mon, Jun 15, 2026 at 10:47:10AM +0530, Nautiyal, Ankit K wrote:
>
> On 6/12/2026 8:12 PM, Ville Syrjala wrote:
> > From: Ville Syrjälä <ville.syrjala@linux.intel.com>
> >
> > Adjust the panel fixed mode selection algorithm to only consider
> > fixed modes that are "VRR compatible" with the old fixed mode
> > when userspace doesn't want to allow full modesets. This will
> > allow a VRR based refresh rate changes (ie. just a change in
> > the vblank length) via the fastset path.
> >
> > When full modesets are allowed, we still use the original algorithm
> > as that may pick a fixed mode with a more optimal dotclock, potentially
> > leading to reduced power consumption.
> >
> > This approach works as long as userspace does the initial
> > allow_modeset=true commit using the highest refresh rate it will
> > want to use. Subsequent commits with allow_modeset=false can then
> > switch between lower refresh rates without blinks.
> >
> > One remaining hurdle we may need to solve is the guardband length.
> > Assuming the highest refresh rate vblank is too short for
> > intel_vrr_compute_optimized_guardband() the intitial guardband will
> > match the highest refresh rate vblank. A subsequent switch to a lower
> > refresh rate will then recompute the guardband and select a value
> > that is higher (since the vblank will be longer). The mismatch in
> > guardband lengths will prevent the fastset. We may either have to
> > preserve the original (sub-optimal) guardband,
>
> I think preserving the original (sub-optimal) guardband makes sense for
> the seamless case, but we will lose out on enabling some power saving
> features like PSR/LOBF for which the sub-otimal guardband would not be
> sufficient.
> So this really comes down to how we want to interpret the
> DRM_MODE_ALLOW_MODESET(state->allow_modeset).
>
> If a lower RR mode is set with allow_modeset = true, then doing a full
> modeset sounds fine.
> In that case we can recompute the guardband and enable the additional
> power saving features (PSR/LOBF) if they are supported.
>
> If the same transition is done without allow_modeset, then we should try
> to keep it seamless.
> In that case using the sub-optimal guardband to avoid a full modeset
> seems like the better choice, even if that means power saving features
> may or may not be enabled, despite being supported at the new RR.
>
> So effectively:
> with allow_modeset -> recompute and get optimal behavior
> without it -> keep things stable, even if sub-optimal
>
> IMO this will make the behavior predictable and lets userspace decide
> when it wants to pay the cost to get those benefits.
>
>
> > or we'll have to
> > revisit the idea of changing the guardband without a full modeset.
> >
> > Note that I'm not 100% happy with this solution because
> > intel_panel_fixed_mode() is no longer fully idempotent, but I wasn't
> > able to come up with anything truly better either :/ The simple
> > solution would be just to always pick the fixed mode with the highest
> > dotclock, but that could lead to increased power consumption even
> > when high refresh rates are never used.
> >
> > Perhaps the proper solution would be to just deprecate this
> > idea of taking in random modes for internal panels and then
> > cooking up a compatible fixed modes. Life would be easier if
> > userspace was required to provide the desired fixed mode directly.
> > But in order to do that we'd need to introduce new uapi properties
> > to control the pfit aspect of this, and we'd probably need a new
> > client cap to select between the old and new userspace behaviour.
> > Something to consider in the future...
> >
> > Signed-off-by: Ville Syrjälä <ville.syrjala@linux.intel.com>
> > ---
> > drivers/gpu/drm/i915/display/intel_panel.c | 55 ++++++++++++++++++++--
> > 1 file changed, 50 insertions(+), 5 deletions(-)
> >
> > diff --git a/drivers/gpu/drm/i915/display/intel_panel.c b/drivers/gpu/drm/i915/display/intel_panel.c
> > index af59fc946fcb..a5fcac1318da 100644
> > --- a/drivers/gpu/drm/i915/display/intel_panel.c
> > +++ b/drivers/gpu/drm/i915/display/intel_panel.c
> > @@ -82,16 +82,37 @@ static bool is_best_fixed_mode(struct intel_connector *connector,
> > abs(drm_mode_vrefresh(best_mode) - vrefresh);
> > }
> >
> > -const struct drm_display_mode *
> > -intel_panel_fixed_mode(struct intel_connector *connector,
> > - const struct drm_display_mode *mode)
> > +static bool is_vrr_compatible(const struct drm_display_mode *mode1,
> > + const struct drm_display_mode *mode2)
> > +{
> > + return drm_mode_match(mode1, mode2,
> > + DRM_MODE_MATCH_CLOCK |
> > + DRM_MODE_MATCH_TIMINGS_VRR |
> > + DRM_MODE_MATCH_FLAGS |
> > + DRM_MODE_MATCH_3D_FLAGS);
> > +}
> > +
> > +static const struct drm_display_mode *
> > +_intel_panel_fixed_mode(struct intel_connector *connector,
> > + const struct drm_display_mode *mode,
> > + const struct drm_display_mode *vrr_ref_mode)
> > {
> > const struct drm_display_mode *fixed_mode, *best_mode = NULL;
> > int vrefresh = drm_mode_vrefresh(mode);
> >
> > + if (vrr_ref_mode &&
> > + (!intel_vrr_is_in_range(connector, vrefresh) ||
> > + !intel_vrr_is_in_range(connector, drm_mode_vrefresh(vrr_ref_mode))))
> > + return NULL;
> > +
> > list_for_each_entry(fixed_mode, &connector->panel.fixed_modes, head) {
> > int fixed_mode_vrefresh = drm_mode_vrefresh(fixed_mode);
> >
> > + if (vrr_ref_mode &&
> > + (!intel_vrr_is_in_range(connector, fixed_mode_vrefresh) ||
> > + !is_vrr_compatible(fixed_mode, vrr_ref_mode)))
> > + continue;
> > +
> > if (is_best_fixed_mode(connector, vrefresh,
> > fixed_mode_vrefresh, best_mode))
>
> This works for all practical purposes, but if we take a hypothetical
> case below, it would not work as expected.
>
> 2 fixed modes:
> with lower RR rate with same clock being first in the modelist:
>
> "3840x2400": 60 1199280 3840 3848 3880 4004 2400 4968 4976 4992
> "3840x2400": 120 1199280 3840 3848 3880 4004 2400 2472 2480 2496
>
> vrr range : 120-40Hz
>
> Scenario: 120Hz is set we want to switch to 80 Hz.
>
> iteration 1
> fixed mode = 60Hz
> best mode = 60 Hz
>
> iteration 2
> fixed mode = 120Hz -> 80 is nearer to 60 than to 120 so best mode
> remains 60
> best mode = 60Hz
>
> We will end up selecting 60Hz mode and try to stretch vtotal based on
> this mode.
I'm thinking that should actually work since it shouldn't really matter
if we end up stretching or shrinking the vblank. But it would feel saner
if we make sure the selected fixed mode does have a higher refresh rate
than the target (ie. final vblank should always end up being >= the
original vblank).
is_best_fixed_mode() does kinda attempt to do that, but badly. I suppose
the correct answer here might be to split the current fixed mode search
to a VRR vs. non-VRR variants, and try the VRR one first. But I think
I'll have to give this one a bit more thought before I can decide if
that approach has any downsides...
>
> So perhaps we should make is_best_fixed_mode() such that the order of
> modes should not affect our fixed mode selection algorithm.
Yeah, I suppose that would be nice. Hmm, perhaps we could just sort the
fixed mode list based on the vrefresh. The actual userspace visible
mode list will anyway be sorted by drm_mode_sort() so it shouldn't
matter for anyone else which order we keep on the fixed modes list.
>
> Note:
>
> 1) As I have mentioned, this is hypothetical case which I have cooked up
> by changing the modes from a real panel, which has highest mode as
> preferred mode and 120 Hz mode being preferred mode.
>
> fixed modes:
>
> "3840x2400": 120 1199280 3840 3848 3880 4004 2400 2472
> 2480 2496 0x48 0xa
>
> "3840x2400": 60 1199280 3840 3848 3880 4004 2400 4968
> 4976 4992 0x40 0xa
>
> 2) This issue will also not be seen with VRR panels when the clocks are
> different but Vtotals are same, the patch should work perfectly in that
> case too.
>
>
> Regards,
>
> Ankit
>
>
> > best_mode = fixed_mode;
> > @@ -100,6 +121,13 @@ intel_panel_fixed_mode(struct intel_connector *connector,
> > return best_mode;
> > }
> >
> > +const struct drm_display_mode *
> > +intel_panel_fixed_mode(struct intel_connector *connector,
> > + const struct drm_display_mode *mode)
> > +{
> > + return _intel_panel_fixed_mode(connector, mode, NULL);
> > +}
> > +
> > static bool is_alt_drrs_mode(const struct drm_display_mode *mode,
> > const struct drm_display_mode *preferred_mode)
> > {
> > @@ -202,11 +230,28 @@ int intel_panel_compute_config(struct intel_atomic_state *state,
> > struct intel_connector *connector)
> > {
> > struct drm_display_mode *adjusted_mode = &crtc_state->hw.adjusted_mode;
> > - const struct drm_display_mode *fixed_mode =
> > - intel_panel_fixed_mode(connector, adjusted_mode);
> > + const struct drm_display_mode *fixed_mode = NULL;
> > int vrefresh, fixed_mode_vrefresh;
> > bool is_vrr;
> >
> > + /*
> > + * Attempt a VRR based refresh rate change if possible
> > + * when userspace has forbidden a full modeset.
> > + */
> > + if (!state->base.allow_modeset) {
> > + struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
> > + const struct intel_crtc_state *old_crtc_state =
> > + intel_atomic_get_old_crtc_state(state, crtc);
> > +
> > + if (old_crtc_state->hw.enable &&
> > + old_crtc_state->uapi.encoder_mask == crtc_state->uapi.encoder_mask)
> > + fixed_mode = _intel_panel_fixed_mode(connector, adjusted_mode,
> > + &old_crtc_state->hw.adjusted_mode);
> > + }
> > +
> > + if (!fixed_mode)
> > + fixed_mode = intel_panel_fixed_mode(connector, adjusted_mode);
> > +
> > if (!fixed_mode)
> > return 0;
> >
--
Ville Syrjälä
Intel
next prev parent reply other threads:[~2026-06-18 19:10 UTC|newest]
Thread overview: 24+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-06-12 14:41 [PATCH 0/4] drm/i915: Work harder to enable VRR based refresh rate changes on eDP Ville Syrjala
2026-06-12 14:42 ` [PATCH 1/4] drm/modes: Add DRM_MODE_MATCH_TIMINGS_VRR Ville Syrjala
2026-06-13 14:19 ` Kandpal, Suraj
2026-06-12 14:42 ` [PATCH 2/4] drm/i915: Pass the full atomic state to .compute_config() Ville Syrjala
2026-06-13 14:23 ` Kandpal, Suraj
2026-06-12 14:42 ` [PATCH 3/4] drm/i915/panel: Adjust intel_panel_compute_config() calling convention Ville Syrjala
2026-06-13 14:25 ` Kandpal, Suraj
2026-06-12 14:42 ` [PATCH 4/4] drm/i915/panel: Attempt VRR based refresh rate change for !allow_modeset Ville Syrjala
2026-06-12 14:56 ` sashiko-bot
2026-06-15 5:17 ` Nautiyal, Ankit K
2026-06-18 19:10 ` Ville Syrjälä [this message]
2026-06-12 14:58 ` ✗ CI.checkpatch: warning for drm/i915: Work harder to enable VRR based refresh rate changes on eDP Patchwork
2026-06-12 14:59 ` ✓ CI.KUnit: success " Patchwork
2026-06-12 15:45 ` ✓ i915.CI.BAT: " Patchwork
2026-06-12 15:54 ` ✓ Xe.CI.BAT: " Patchwork
2026-06-13 7:16 ` ✓ Xe.CI.FULL: " Patchwork
2026-06-13 13:24 ` ✗ i915.CI.Full: failure " Patchwork
2026-06-15 9:06 ` [PATCH 0/4] " Michel Dänzer
2026-06-15 9:08 ` Michel Dänzer
2026-06-15 13:06 ` Ville Syrjälä
2026-06-15 13:30 ` Michel Dänzer
2026-06-15 17:00 ` Ville Syrjälä
2026-06-16 7:21 ` Michel Dänzer
2026-06-18 18:39 ` Ville Syrjälä
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=ajRCkjh-kHwg56V4@intel.com \
--to=ville.syrjala@linux.intel.com \
--cc=ankit.k.nautiyal@intel.com \
--cc=dri-devel@lists.freedesktop.org \
--cc=intel-gfx@lists.freedesktop.org \
--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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.