Commit 78015e27 authored by Imre Deak's avatar Imre Deak

drm/i915/dp: Update the link bpp limits for DSC mode

In non-DSC mode the link bpp can be set in 2*3 bpp steps in the pipe bpp
range, while in DSC mode it can be set in 1/16 bpp steps to any value
up to the maximum pipe bpp. Update the limits accordingly in both modes
to prepare for a follow-up patch which may need to reduce the max link
bpp value and starts to check the link bpp limits in DSC mode as well.

While at it add more detail to the link limit debug print and print it
also for DSC mode.

v2:
- Add to_bpp_frac_dec() instead of open coding it. (Jani)
v3: (Ville)
- Add BPP_X16_FMT / BPP_X16_ARG.
- Add TODO: comment about initializing the DSC link bpp limits earlier.

Cc: Jani Nikula <jani.nikula@linux.intel.com>
Cc: Ville Syrjälä <ville.syrjala@linux.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/20230921195159.2646027-5-imre.deak@intel.com
parent 7d0f2f68
...@@ -2120,6 +2120,14 @@ static inline int to_bpp_int(int bpp_x16) ...@@ -2120,6 +2120,14 @@ static inline int to_bpp_int(int bpp_x16)
return bpp_x16 >> 4; return bpp_x16 >> 4;
} }
static inline int to_bpp_frac(int bpp_x16)
{
return bpp_x16 & 0xf;
}
#define BPP_X16_FMT "%d.%04d"
#define BPP_X16_ARGS(bpp_x16) to_bpp_int(bpp_x16), (to_bpp_frac(bpp_x16) * 625)
static inline int to_bpp_x16(int bpp) static inline int to_bpp_x16(int bpp)
{ {
return bpp << 4; return bpp << 4;
......
...@@ -2190,16 +2190,72 @@ int intel_dp_dsc_compute_config(struct intel_dp *intel_dp, ...@@ -2190,16 +2190,72 @@ int intel_dp_dsc_compute_config(struct intel_dp *intel_dp,
return 0; return 0;
} }
static void /**
intel_dp_compute_config_limits(struct intel_dp *intel_dp, * intel_dp_compute_config_link_bpp_limits - compute output link bpp limits
struct intel_crtc_state *crtc_state, * @intel_dp: intel DP
bool respect_downstream_limits, * @crtc_state: crtc state
* @dsc: DSC compression mode
* @limits: link configuration limits
*
* Calculates the output link min, max bpp values in @limits based on the
* pipe bpp range, @crtc_state and @dsc mode.
*
* Returns %true in case of success.
*/
bool
intel_dp_compute_config_link_bpp_limits(struct intel_dp *intel_dp,
const struct intel_crtc_state *crtc_state,
bool dsc,
struct link_config_limits *limits) struct link_config_limits *limits)
{ {
struct drm_i915_private *i915 = dp_to_i915(intel_dp); struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev);
const struct drm_display_mode *adjusted_mode = const struct drm_display_mode *adjusted_mode =
&crtc_state->hw.adjusted_mode; &crtc_state->hw.adjusted_mode;
const struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
const struct intel_encoder *encoder = &dp_to_dig_port(intel_dp)->base;
int max_link_bpp_x16;
max_link_bpp_x16 = to_bpp_x16(limits->pipe.max_bpp);
if (!dsc) {
max_link_bpp_x16 = rounddown(max_link_bpp_x16, to_bpp_x16(2 * 3));
if (max_link_bpp_x16 < to_bpp_x16(limits->pipe.min_bpp))
return false;
limits->link.min_bpp_x16 = to_bpp_x16(limits->pipe.min_bpp);
} else {
/*
* TODO: set the DSC link limits already here, atm these are
* initialized only later in intel_edp_dsc_compute_pipe_bpp() /
* intel_dp_dsc_compute_pipe_bpp()
*/
limits->link.min_bpp_x16 = 0;
}
limits->link.max_bpp_x16 = max_link_bpp_x16;
drm_dbg_kms(&i915->drm,
"[ENCODER:%d:%s][CRTC:%d:%s] DP link limits: pixel clock %d kHz DSC %s max lanes %d max rate %d max pipe_bpp %d max link_bpp " BPP_X16_FMT "\n",
encoder->base.base.id, encoder->base.name,
crtc->base.base.id, crtc->base.name,
adjusted_mode->crtc_clock,
dsc ? "on" : "off",
limits->max_lane_count,
limits->max_rate,
limits->pipe.max_bpp,
BPP_X16_ARGS(limits->link.max_bpp_x16));
return true;
}
static bool
intel_dp_compute_config_limits(struct intel_dp *intel_dp,
struct intel_crtc_state *crtc_state,
bool respect_downstream_limits,
bool dsc,
struct link_config_limits *limits)
{
limits->min_rate = intel_dp_common_rate(intel_dp, 0); limits->min_rate = intel_dp_common_rate(intel_dp, 0);
limits->max_rate = intel_dp_max_link_rate(intel_dp); limits->max_rate = intel_dp_max_link_rate(intel_dp);
...@@ -2225,13 +2281,10 @@ intel_dp_compute_config_limits(struct intel_dp *intel_dp, ...@@ -2225,13 +2281,10 @@ intel_dp_compute_config_limits(struct intel_dp *intel_dp,
intel_dp_adjust_compliance_config(intel_dp, crtc_state, limits); intel_dp_adjust_compliance_config(intel_dp, crtc_state, limits);
limits->link.min_bpp_x16 = to_bpp_x16(limits->pipe.min_bpp); return intel_dp_compute_config_link_bpp_limits(intel_dp,
limits->link.max_bpp_x16 = to_bpp_x16(limits->pipe.max_bpp); crtc_state,
dsc,
drm_dbg_kms(&i915->drm, "DP link computation with max lane count %i " limits);
"max rate %d max bpp %d pixel clock %iKHz\n",
limits->max_lane_count, limits->max_rate,
to_bpp_int(limits->link.max_bpp_x16), adjusted_mode->crtc_clock);
} }
static int static int
...@@ -2250,9 +2303,6 @@ intel_dp_compute_link_config(struct intel_encoder *encoder, ...@@ -2250,9 +2303,6 @@ intel_dp_compute_link_config(struct intel_encoder *encoder,
bool dsc_needed; bool dsc_needed;
int ret = 0; int ret = 0;
intel_dp_compute_config_limits(intel_dp, pipe_config,
respect_downstream_limits, &limits);
if (intel_dp_need_bigjoiner(intel_dp, adjusted_mode->crtc_hdisplay, if (intel_dp_need_bigjoiner(intel_dp, adjusted_mode->crtc_hdisplay,
adjusted_mode->crtc_clock)) adjusted_mode->crtc_clock))
pipe_config->bigjoiner_pipes = GENMASK(crtc->pipe + 1, crtc->pipe); pipe_config->bigjoiner_pipes = GENMASK(crtc->pipe + 1, crtc->pipe);
...@@ -2264,7 +2314,11 @@ intel_dp_compute_link_config(struct intel_encoder *encoder, ...@@ -2264,7 +2314,11 @@ intel_dp_compute_link_config(struct intel_encoder *encoder,
*/ */
joiner_needs_dsc = DISPLAY_VER(i915) < 13 && pipe_config->bigjoiner_pipes; joiner_needs_dsc = DISPLAY_VER(i915) < 13 && pipe_config->bigjoiner_pipes;
dsc_needed = joiner_needs_dsc || intel_dp->force_dsc_en; dsc_needed = joiner_needs_dsc || intel_dp->force_dsc_en ||
!intel_dp_compute_config_limits(intel_dp, pipe_config,
respect_downstream_limits,
false,
&limits);
if (!dsc_needed) { if (!dsc_needed) {
/* /*
...@@ -2281,6 +2335,13 @@ intel_dp_compute_link_config(struct intel_encoder *encoder, ...@@ -2281,6 +2335,13 @@ intel_dp_compute_link_config(struct intel_encoder *encoder,
drm_dbg_kms(&i915->drm, "Try DSC (fallback=%s, joiner=%s, force=%s)\n", drm_dbg_kms(&i915->drm, "Try DSC (fallback=%s, joiner=%s, force=%s)\n",
str_yes_no(ret), str_yes_no(joiner_needs_dsc), str_yes_no(ret), str_yes_no(joiner_needs_dsc),
str_yes_no(intel_dp->force_dsc_en)); str_yes_no(intel_dp->force_dsc_en));
if (!intel_dp_compute_config_limits(intel_dp, pipe_config,
respect_downstream_limits,
true,
&limits))
return -EINVAL;
ret = intel_dp_dsc_compute_config(intel_dp, pipe_config, ret = intel_dp_dsc_compute_config(intel_dp, pipe_config,
conn_state, &limits, 64, true); conn_state, &limits, 64, true);
if (ret < 0) if (ret < 0)
......
...@@ -156,4 +156,10 @@ void intel_dp_phy_test(struct intel_encoder *encoder); ...@@ -156,4 +156,10 @@ void intel_dp_phy_test(struct intel_encoder *encoder);
void intel_dp_wait_source_oui(struct intel_dp *intel_dp); void intel_dp_wait_source_oui(struct intel_dp *intel_dp);
int intel_dp_output_bpp(enum intel_output_format output_format, int bpp); int intel_dp_output_bpp(enum intel_output_format output_format, int bpp);
bool
intel_dp_compute_config_link_bpp_limits(struct intel_dp *intel_dp,
const struct intel_crtc_state *crtc_state,
bool dsc,
struct link_config_limits *limits);
#endif /* __INTEL_DP_H__ */ #endif /* __INTEL_DP_H__ */
...@@ -295,9 +295,10 @@ static int intel_dp_mst_update_slots(struct intel_encoder *encoder, ...@@ -295,9 +295,10 @@ static int intel_dp_mst_update_slots(struct intel_encoder *encoder,
return 0; return 0;
} }
static void static bool
intel_dp_mst_compute_config_limits(struct intel_dp *intel_dp, intel_dp_mst_compute_config_limits(struct intel_dp *intel_dp,
struct intel_crtc_state *crtc_state, struct intel_crtc_state *crtc_state,
bool dsc,
struct link_config_limits *limits) struct link_config_limits *limits)
{ {
/* /*
...@@ -323,8 +324,10 @@ intel_dp_mst_compute_config_limits(struct intel_dp *intel_dp, ...@@ -323,8 +324,10 @@ intel_dp_mst_compute_config_limits(struct intel_dp *intel_dp,
intel_dp_adjust_compliance_config(intel_dp, crtc_state, limits); intel_dp_adjust_compliance_config(intel_dp, crtc_state, limits);
limits->link.min_bpp_x16 = to_bpp_x16(limits->pipe.min_bpp); return intel_dp_compute_config_link_bpp_limits(intel_dp,
limits->link.max_bpp_x16 = to_bpp_x16(limits->pipe.max_bpp); crtc_state,
dsc,
limits);
} }
static int intel_dp_mst_compute_config(struct intel_encoder *encoder, static int intel_dp_mst_compute_config(struct intel_encoder *encoder,
...@@ -347,9 +350,11 @@ static int intel_dp_mst_compute_config(struct intel_encoder *encoder, ...@@ -347,9 +350,11 @@ static int intel_dp_mst_compute_config(struct intel_encoder *encoder,
pipe_config->output_format = INTEL_OUTPUT_FORMAT_RGB; pipe_config->output_format = INTEL_OUTPUT_FORMAT_RGB;
pipe_config->has_pch_encoder = false; pipe_config->has_pch_encoder = false;
intel_dp_mst_compute_config_limits(intel_dp, pipe_config, &limits); dsc_needed = intel_dp->force_dsc_en ||
!intel_dp_mst_compute_config_limits(intel_dp,
dsc_needed = intel_dp->force_dsc_en; pipe_config,
false,
&limits);
if (!dsc_needed) { if (!dsc_needed) {
ret = intel_dp_mst_compute_link_config(encoder, pipe_config, ret = intel_dp_mst_compute_link_config(encoder, pipe_config,
...@@ -368,6 +373,12 @@ static int intel_dp_mst_compute_config(struct intel_encoder *encoder, ...@@ -368,6 +373,12 @@ static int intel_dp_mst_compute_config(struct intel_encoder *encoder,
str_yes_no(ret), str_yes_no(ret),
str_yes_no(intel_dp->force_dsc_en)); str_yes_no(intel_dp->force_dsc_en));
if (!intel_dp_mst_compute_config_limits(intel_dp,
pipe_config,
true,
&limits))
return -EINVAL;
/* /*
* FIXME: As bpc is hardcoded to 8, as mentioned above, * FIXME: As bpc is hardcoded to 8, as mentioned above,
* WARN and ignore the debug flag force_dsc_bpc for now. * WARN and ignore the debug flag force_dsc_bpc for now.
......
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