Commit 1dd9d86a authored by Imre Deak's avatar Imre Deak

drm/i915: Fix display bpp limit computation during system resume

The system resume display mode restoration should happen with an output
configuration matching that of the suspend time saved mode. Since the
restored mode configuration is subject to the bpp fallback logic,
starting out with an unlimited bpp and reducing the bpp as required by
any (MST) link BW limit, the resulting bpp will match the one during
suspend only if the BW limit checks during suspend and resume are
applied in an identical way. The latter is not guaranteed at the moment,
since the pre-suspend MST topology may not be in place during resume
(for instance if the MST sink was disconnected while being suspended),
which makes the MST link BW check accept the unlimited bpp mode
configuration unconditionally without ensuring that the required BW fits
into the available MST link BW.

To fix the above, initialize the bpp fallback logic with the max link
bpp / force-FEC limits left behind by the suspend time mode save.
Reviewed-by: default avatarUma Shankar <uma.shankar@intel.com>
Reviewed-by: default avatarVille Syrjälä <ville.syrjala@linux.intel.com>
Signed-off-by: default avatarImre Deak <imre.deak@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20240220211841.448846-4-imre.deak@intel.com
parent 295654f7
...@@ -6253,12 +6253,11 @@ static int intel_atomic_check_config(struct intel_atomic_state *state, ...@@ -6253,12 +6253,11 @@ static int intel_atomic_check_config(struct intel_atomic_state *state,
static int intel_atomic_check_config_and_link(struct intel_atomic_state *state) static int intel_atomic_check_config_and_link(struct intel_atomic_state *state)
{ {
struct drm_i915_private *i915 = to_i915(state->base.dev);
struct intel_link_bw_limits new_limits; struct intel_link_bw_limits new_limits;
struct intel_link_bw_limits old_limits; struct intel_link_bw_limits old_limits;
int ret; int ret;
intel_link_bw_init_limits(i915, &new_limits); intel_link_bw_init_limits(state, &new_limits);
old_limits = new_limits; old_limits = new_limits;
while (true) { while (true) {
......
...@@ -6,6 +6,7 @@ ...@@ -6,6 +6,7 @@
#include "i915_drv.h" #include "i915_drv.h"
#include "intel_atomic.h" #include "intel_atomic.h"
#include "intel_crtc.h"
#include "intel_display_types.h" #include "intel_display_types.h"
#include "intel_dp_mst.h" #include "intel_dp_mst.h"
#include "intel_fdi.h" #include "intel_fdi.h"
...@@ -13,19 +14,32 @@ ...@@ -13,19 +14,32 @@
/** /**
* intel_link_bw_init_limits - initialize BW limits * intel_link_bw_init_limits - initialize BW limits
* @i915: device instance * @state: Atomic state
* @limits: link BW limits * @limits: link BW limits
* *
* Initialize @limits. * Initialize @limits.
*/ */
void intel_link_bw_init_limits(struct drm_i915_private *i915, struct intel_link_bw_limits *limits) void intel_link_bw_init_limits(struct intel_atomic_state *state,
struct intel_link_bw_limits *limits)
{ {
struct drm_i915_private *i915 = to_i915(state->base.dev);
enum pipe pipe; enum pipe pipe;
limits->force_fec_pipes = 0; limits->force_fec_pipes = 0;
limits->bpp_limit_reached_pipes = 0; limits->bpp_limit_reached_pipes = 0;
for_each_pipe(i915, pipe) for_each_pipe(i915, pipe) {
limits->max_bpp_x16[pipe] = INT_MAX; const struct intel_crtc_state *crtc_state =
intel_atomic_get_new_crtc_state(state,
intel_crtc_for_pipe(i915, pipe));
if (state->base.duplicated && crtc_state) {
limits->max_bpp_x16[pipe] = crtc_state->max_link_bpp_x16;
if (crtc_state->fec_enable)
limits->force_fec_pipes |= BIT(pipe);
} else {
limits->max_bpp_x16[pipe] = INT_MAX;
}
}
} }
/** /**
......
...@@ -22,7 +22,7 @@ struct intel_link_bw_limits { ...@@ -22,7 +22,7 @@ struct intel_link_bw_limits {
int max_bpp_x16[I915_MAX_PIPES]; int max_bpp_x16[I915_MAX_PIPES];
}; };
void intel_link_bw_init_limits(struct drm_i915_private *i915, void intel_link_bw_init_limits(struct intel_atomic_state *state,
struct intel_link_bw_limits *limits); struct intel_link_bw_limits *limits);
int intel_link_bw_reduce_bpp(struct intel_atomic_state *state, int intel_link_bw_reduce_bpp(struct intel_atomic_state *state,
struct intel_link_bw_limits *limits, struct intel_link_bw_limits *limits,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment