From: Dibin Moolakadan Subrahmanian <dibin.moolakadan.subrahmanian@intel.com>
To: "Shankar, Uma" <uma.shankar@intel.com>,
"intel-gfx@lists.freedesktop.org"
<intel-gfx@lists.freedesktop.org>,
"intel-xe@lists.freedesktop.org" <intel-xe@lists.freedesktop.org>
Cc: "Manna, Animesh" <animesh.manna@intel.com>,
"Kurmi, Suresh Kumar" <suresh.kumar.kurmi@intel.com>,
<imre.deak@intel.com>
Subject: Re: [PATCH v2 07/13] drm/i915/display: Add DC3CO eligibility computation
Date: Mon, 27 Apr 2026 11:36:48 +0530 [thread overview]
Message-ID: <2b8f779c-d2f9-45fa-af12-da0d0d2eac3e@intel.com> (raw)
In-Reply-To: <DM4PR11MB6360EC230948A51A577F63A1F4362@DM4PR11MB6360.namprd11.prod.outlook.com>
On 27-04-2026 08:40, Shankar, Uma wrote:
>
>> -----Original Message-----
>> From: Dibin Moolakadan Subrahmanian
>> <dibin.moolakadan.subrahmanian@intel.com>
>> Sent: Wednesday, April 22, 2026 9:56 PM
>> To: intel-gfx@lists.freedesktop.org; intel-xe@lists.freedesktop.org
>> Cc: Manna, Animesh <animesh.manna@intel.com>; Shankar, Uma
>> <uma.shankar@intel.com>; Kurmi, Suresh Kumar
>> <suresh.kumar.kurmi@intel.com>
>> Subject: [PATCH v2 07/13] drm/i915/display: Add DC3CO eligibility computation
>>
>> Compute DC3CO eligibility during atomic_check based on pipe/port constraints
>> and runtime triggers and store result in display->power.dc3co.
>>
>> When DC3CO is allowed, request DC_STATE_EN_UPTO_DC3CO and reduce the
>> DC entry delay. Otherwise, retain the existing delay and set default
>> DC_STATE_EN_UPTO_DC6 .
>>
>> Changes in v2:
>> - Move dc3co state from intel_atomic_state to display->power (Uma Shankar)
>> - Use #define bitmasks instead of enum for DC3CO triggers (Jani Nikula)
>>
>> BSpec: 75253
>> Signed-off-by: Dibin Moolakadan Subrahmanian
>> <dibin.moolakadan.subrahmanian@intel.com>
>> ---
>> drivers/gpu/drm/i915/display/intel_display.c | 92 ++++++++++++++++++-
>> drivers/gpu/drm/i915/display/intel_display.h | 1 -
>> .../gpu/drm/i915/display/intel_display_core.h | 3 +-
>> .../drm/i915/display/intel_display_power.c | 30 ++++++
>> .../drm/i915/display/intel_display_power.h | 22 +++++
>> 5 files changed, 141 insertions(+), 7 deletions(-)
>>
>> diff --git a/drivers/gpu/drm/i915/display/intel_display.c
>> b/drivers/gpu/drm/i915/display/intel_display.c
>> index 674a4ece6d0f..de493d04a622 100644
>> --- a/drivers/gpu/drm/i915/display/intel_display.c
>> +++ b/drivers/gpu/drm/i915/display/intel_display.c
>> @@ -5870,6 +5870,69 @@ static bool intel_pipes_need_modeset(struct
>> intel_atomic_state *state,
>> return false;
>> }
>>
>> +static bool intel_dc3co_port_pipe_compatible(struct intel_dp *intel_dp,
>> + const struct intel_crtc_state
>> *crtc_state) {
>> + struct intel_digital_port *dig_port = dp_to_dig_port(intel_dp);
>> + enum pipe pipe = to_intel_crtc(crtc_state->uapi.crtc)->pipe;
>> + enum port port = dig_port->base.port;
>> + int num_pipes = intel_crtc_num_joined_pipes(crtc_state);
>> +
>> + return num_pipes == 1 && pipe <= PIPE_B && port <= PORT_B; }
>> +
>> +static void intel_dc3co_compute_state(struct intel_atomic_state *state)
>> +{
>> + struct intel_display *display = to_intel_display(state);
>> + struct intel_crtc *crtc;
>> + struct intel_crtc_state *crtc_state;
>> + struct intel_encoder *encoder;
>> + struct intel_dp *intel_dp;
>> + int active_pipes = 0;
>> + u32 trigger = DC3CO_TRIGGER_NONE;
>> +
>> + if (!HAS_DC3CO(display))
>> + return;
>> +
>> + for_each_intel_crtc(display->drm, crtc) {
>> + trigger = DC3CO_TRIGGER_NONE;
>> + crtc_state = intel_atomic_get_new_crtc_state(state, crtc);
>> + if (!crtc_state)
>> + crtc_state = intel_atomic_get_old_crtc_state(state, crtc);
>> +
>> + if (!crtc_state || !crtc_state->hw.active)
>> + continue;
>> +
>> + active_pipes++;
>> +
>> + if (active_pipes > 1)
>> + goto done;
>> +
>> + for_each_intel_encoder_mask(display->drm, encoder,
>> + crtc_state->uapi.encoder_mask) {
>> + if (encoder->type != INTEL_OUTPUT_EDP)
>> + goto done;
>> +
>> + intel_dp = enc_to_intel_dp(encoder);
>> +
>> + if (!intel_dc3co_port_pipe_compatible(intel_dp,
>> crtc_state))
>> + goto done;
>> + }
>> +
>> + if (crtc_state->has_lobf)
>> + trigger |= DC3CO_TRIGGER_LOBF;
>> + if (crtc_state->has_panel_replay)
>> + trigger |= DC3CO_TRIGGER_PANEL_REPLAY;
>> + if (crtc_state->has_sel_update)
>> + trigger |= DC3CO_TRIGGER_PSR2;
> These values will get updated even if active pipe count is 2. Please check once.
trigger is initialized with DC3CO_TRIGGER_NONE at the beginning for loop.
For the second active pipe, trigger will be set to DC3CO_TRIGGER_NONE
and the loop will break before trigger update here.
>
> With this fixed,
> Reviewed-by: Uma Shankar <uma.shankar@intel.com>
>
> Note: Would be good to get feedback from Imre as well.
ok.
>
>> + }
>> +
>> +done:
>> + intel_display_power_dc3co_update(display, !!trigger, trigger);
>> + drm_dbg_kms(display->drm, "DC3CO allowed=%d trigger=0x%x\n",
>> + !!trigger, trigger);
>> +}
>> +
>> static int intel_atomic_check_joiner(struct intel_atomic_state *state,
>> struct intel_crtc *primary_crtc) { @@ -6544,6
>> +6607,7 @@ int intel_atomic_check(struct drm_device *dev,
>> if (ret)
>> goto fail;
>>
>> + intel_dc3co_compute_state(state);
>> for_each_oldnew_intel_crtc_in_state(state, crtc, old_crtc_state,
>> new_crtc_state, i) {
>> intel_color_assert_luts(new_crtc_state);
>> @@ -7415,6 +7479,7 @@ static void intel_atomic_commit_tail(struct
>> intel_atomic_state *state)
>> struct intel_power_domain_mask put_domains[I915_MAX_PIPES] = {};
>> struct ref_tracker *wakeref = NULL;
>> int i;
>> + int power_async_delay;
>>
>> for_each_new_intel_crtc_in_state(state, crtc, new_crtc_state, i)
>> intel_atomic_dsb_prepare(state, crtc); @@ -7621,11 +7686,28
>> @@ static void intel_atomic_commit_tail(struct intel_atomic_state *state)
>> */
>> intel_uncore_arm_unclaimed_mmio_detection(uncore);
>> }
>> - /*
>> - * Delay re-enabling DC states by 17 ms to avoid the off->on->off
>> - * toggling overhead at and above 60 FPS.
>> - */
>> - intel_display_power_put_async_delay(display,
>> POWER_DOMAIN_DC_OFF, wakeref, 17);
>> +
>> + if (intel_display_power_dc3co_allowed(display) &&
>> + intel_display_power_dc3co_supported(display)) {
>> + intel_display_power_set_target_dc_state(display,
>> DC_STATE_EN_UPTO_DC3CO);
>> + /*
>> + * Use minimal re-enable delay to allow DC3CO entry on
>> + * the next idle frame, unlike the 17ms guard needed to
>> + * prevent DC5/DC6 toggling overhead at 60+ FPS.
>> + */
>> + power_async_delay = 1;
>> + } else {
>> + /*
>> + * Delay re-enabling DC states by 17 ms to avoid the off->on->off
>> + * toggling overhead at and above 60 FPS.
>> + */
>> + intel_display_power_set_target_dc_state(display,
>> DC_STATE_EN_UPTO_DC6);
>> + power_async_delay = 17;
>> + }
>> +
>> + intel_display_power_put_async_delay(display,
>> + POWER_DOMAIN_DC_OFF, wakeref,
>> power_async_delay);
>> +
>> intel_display_rpm_put(display, state->wakeref);
>>
>> /*
>> diff --git a/drivers/gpu/drm/i915/display/intel_display.h
>> b/drivers/gpu/drm/i915/display/intel_display.h
>> index 1e76a455d7c4..2795e4b9e799 100644
>> --- a/drivers/gpu/drm/i915/display/intel_display.h
>> +++ b/drivers/gpu/drm/i915/display/intel_display.h
>> @@ -521,5 +521,4 @@ bool assert_port_valid(struct intel_display *display, enum
>> port port);
>>
>> bool intel_scanout_needs_vtd_wa(struct intel_display *display); int
>> intel_crtc_num_joined_pipes(const struct intel_crtc_state *crtc_state);
>> -
>> #endif
>> diff --git a/drivers/gpu/drm/i915/display/intel_display_core.h
>> b/drivers/gpu/drm/i915/display/intel_display_core.h
>> index c5a07090cba6..13e9b986b6fc 100644
>> --- a/drivers/gpu/drm/i915/display/intel_display_core.h
>> +++ b/drivers/gpu/drm/i915/display/intel_display_core.h
>> @@ -535,7 +535,8 @@ struct intel_display {
>>
>> struct {
>> struct i915_power_domains domains;
>> -
>> + /* DC3CO eligibility state */
>> + struct intel_dc3co_state dc3co;
>> /* Shadow for DISPLAY_PHY_CONTROL which can't be safely
>> read */
>> u32 chv_phy_control;
>>
>> diff --git a/drivers/gpu/drm/i915/display/intel_display_power.c
>> b/drivers/gpu/drm/i915/display/intel_display_power.c
>> index f626803bbd88..ff1915be59c9 100644
>> --- a/drivers/gpu/drm/i915/display/intel_display_power.c
>> +++ b/drivers/gpu/drm/i915/display/intel_display_power.c
>> @@ -372,6 +372,35 @@ bool intel_display_power_dc3co_supported(struct
>> intel_display *display)
>> return (power_domains->allowed_dc_mask &
>> DC_STATE_EN_UPTO_DC3CO) == DC_STATE_EN_UPTO_DC3CO; }
>>
>> +void intel_display_power_dc3co_update(struct intel_display *display,
>> + bool allowed, u32 trigger)
>> +{
>> + struct intel_dc3co_state *dc3co = &display->power.dc3co;
>> +
>> + if (!HAS_DC3CO(display))
>> + return;
>> +
>> + mutex_lock(&dc3co->lock);
>> + dc3co->allowed = allowed;
>> + dc3co->trigger = trigger;
>> + mutex_unlock(&dc3co->lock);
>> +}
>> +
>> +bool intel_display_power_dc3co_allowed(struct intel_display *display) {
>> + struct intel_dc3co_state *dc3co = &display->power.dc3co;
>> + bool allowed;
>> +
>> + if (!HAS_DC3CO(display))
>> + return false;
>> +
>> + mutex_lock(&dc3co->lock);
>> + allowed = dc3co->allowed;
>> + mutex_unlock(&dc3co->lock);
>> +
>> + return allowed;
>> +}
>> +
>> static void __async_put_domains_mask(struct i915_power_domains
>> *power_domains,
>> struct intel_power_domain_mask *mask) {
>> @@ -1051,6 +1080,7 @@ int intel_power_domains_init(struct intel_display
>> *display)
>> sanitize_target_dc_state(display, DC_STATE_EN_UPTO_DC6);
>>
>> mutex_init(&power_domains->lock);
>> + mutex_init(&display->power.dc3co.lock);
>>
>> INIT_DELAYED_WORK(&power_domains->async_put_work,
>> intel_display_power_put_async_work);
>> diff --git a/drivers/gpu/drm/i915/display/intel_display_power.h
>> b/drivers/gpu/drm/i915/display/intel_display_power.h
>> index 05880e9da89f..0b1a06f88ae5 100644
>> --- a/drivers/gpu/drm/i915/display/intel_display_power.h
>> +++ b/drivers/gpu/drm/i915/display/intel_display_power.h
>> @@ -131,6 +131,25 @@ struct intel_power_domain_mask {
>> DECLARE_BITMAP(bits, POWER_DOMAIN_NUM); };
>>
>> +/*
>> + * DC3CO enabling triggers (bitmask).
>> + * DC3CO may be enabled when at least one of these triggers is active.
>> + * Additional constraints may still apply.
>> + */
>> +#define DC3CO_TRIGGER_NONE (0)
>> +#define DC3CO_TRIGGER_PSR2 BIT(0)
>> +#define DC3CO_TRIGGER_LOBF BIT(1)
>> +#define DC3CO_TRIGGER_PANEL_REPLAY BIT(2)
>> +#define DC3CO_TRIGGER_ALL (DC3CO_TRIGGER_PSR2 | \
>> + DC3CO_TRIGGER_LOBF | \
>> + DC3CO_TRIGGER_PANEL_REPLAY)
>> +
>> +struct intel_dc3co_state {
>> + struct mutex lock; /* Protects allowed and trigger fields */
>> + bool allowed; /* DC3CO eligibility result */
>> + u32 trigger; /* Bitmask of active DC3CO triggers */ };
>> +
>> struct i915_power_domains {
>> /*
>> * Power wells needed for initialization at driver init and suspend @@ -
>> 187,6 +206,9 @@ void intel_display_power_set_target_dc_state(struct
>> intel_display *display,
>> u32 state);
>> u32 intel_display_power_get_current_dc_state(struct intel_display *display); bool
>> intel_display_power_dc3co_supported(struct intel_display *display);
>> +void intel_display_power_dc3co_update(struct intel_display *display,
>> + bool allowed, u32 trigger);
>> +bool intel_display_power_dc3co_allowed(struct intel_display *display);
>>
>> bool intel_display_power_is_enabled(struct intel_display *display,
>> enum intel_display_power_domain domain);
>> --
>> 2.43.0
next prev parent reply other threads:[~2026-04-27 6:07 UTC|newest]
Thread overview: 28+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-04-22 16:26 [PATCH v2 00/13] drm/i915/display: Add DC3CO support Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 01/13] drm/i915/display: Remove TGL " Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 02/13] drm/i915/display: Switch DC3Co enable from standalone bit to DC level encoding Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 03/13] drm/i915/display: Use FIELD_PREP() for DC state enable bits Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 04/13] drm/i915/display: Add DC3CO DC_STATE enable/disable support Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 05/13] drm/i915/display: Add DC3CO support check and validate target DC state Dibin Moolakadan Subrahmanian
2026-04-29 5:13 ` Manna, Animesh
2026-04-29 7:31 ` Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 06/13] drm/i915/display: Add HAS_DC3CO() macro Dibin Moolakadan Subrahmanian
2026-04-27 3:00 ` Shankar, Uma
2026-04-22 16:26 ` [PATCH v2 07/13] drm/i915/display: Add DC3CO eligibility computation Dibin Moolakadan Subrahmanian
2026-04-27 3:10 ` Shankar, Uma
2026-04-27 6:06 ` Dibin Moolakadan Subrahmanian [this message]
2026-04-29 5:42 ` Manna, Animesh
2026-04-29 7:05 ` Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 08/13] drm/i915/display: Store DC3CO eligibility in PSR state Dibin Moolakadan Subrahmanian
2026-04-27 3:11 ` Shankar, Uma
2026-04-22 16:26 ` [PATCH v2 09/13] drm/i915/display: PSR2: Set idle_frames to 0 for DC3CO Dibin Moolakadan Subrahmanian
2026-04-27 3:13 ` Shankar, Uma
2026-04-22 16:26 ` [PATCH v2 10/13] drm/i915/display: Enable DC3CO idle protocol in ALPM Dibin Moolakadan Subrahmanian
2026-04-27 3:14 ` Shankar, Uma
2026-04-22 16:26 ` [PATCH v2 11/13] drm/i915/display: PSR Add delayed work to exit DC3CO Dibin Moolakadan Subrahmanian
2026-04-27 3:15 ` Shankar, Uma
2026-04-22 16:26 ` [PATCH v2 12/13] drm/i915/display: Add helper to enable DC counter Dibin Moolakadan Subrahmanian
2026-04-22 16:26 ` [PATCH v2 13/13] drm/i915/display: Add DC3CO count and residency in dmc debugfs Dibin Moolakadan Subrahmanian
2026-04-22 21:27 ` ✓ CI.KUnit: success for drm/i915/display: Add DC3CO support (rev2) Patchwork
2026-04-22 22:50 ` ✓ Xe.CI.BAT: " Patchwork
2026-04-23 6:15 ` ✗ Xe.CI.FULL: failure " 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=2b8f779c-d2f9-45fa-af12-da0d0d2eac3e@intel.com \
--to=dibin.moolakadan.subrahmanian@intel.com \
--cc=animesh.manna@intel.com \
--cc=imre.deak@intel.com \
--cc=intel-gfx@lists.freedesktop.org \
--cc=intel-xe@lists.freedesktop.org \
--cc=suresh.kumar.kurmi@intel.com \
--cc=uma.shankar@intel.com \
/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