Commit 8315847b authored by Ville Syrjälä's avatar Ville Syrjälä

drm/i915: Clean up skl+ vs. icl+ watermark computation

Make a cleaner split between the skl+ and icl+ ways of computing
watermarks. This way skl_build_pipe_wm() doesn't have to know any
of the gritty details of icl+ master/slave planes.

We can also simplify a bunch of the lower level code by pulling
the plane visibility checks a bit higher up.

v2: WARN_ON(!visible) for the icl+ master plane case (Matt)

Cc: Matt Roper <matthew.d.roper@intel.com>
Reviewed-by: default avatarMaarten Lankhorst <maarten.lankhorst@linux.intel.com>
Reviewed-by: default avatarMatt Roper <matthew.d.roper@intel.com>
Signed-off-by: default avatarVille Syrjälä <ville.syrjala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20181127165726.31122-1-ville.syrjala@linux.intel.com
parent 6a3c910b
...@@ -4630,9 +4630,6 @@ skl_compute_plane_wm_params(const struct drm_i915_private *dev_priv, ...@@ -4630,9 +4630,6 @@ skl_compute_plane_wm_params(const struct drm_i915_private *dev_priv,
to_intel_atomic_state(cstate->base.state); to_intel_atomic_state(cstate->base.state);
bool apply_memory_bw_wa = skl_needs_memory_bw_wa(state); bool apply_memory_bw_wa = skl_needs_memory_bw_wa(state);
if (!intel_wm_plane_visible(cstate, intel_pstate))
return 0;
/* only NV12 format has two planes */ /* only NV12 format has two planes */
if (plane_id == 1 && fb->format->format != DRM_FORMAT_NV12) { if (plane_id == 1 && fb->format->format != DRM_FORMAT_NV12) {
DRM_DEBUG_KMS("Non NV12 format have single plane\n"); DRM_DEBUG_KMS("Non NV12 format have single plane\n");
...@@ -4746,9 +4743,6 @@ static int skl_compute_plane_wm(const struct drm_i915_private *dev_priv, ...@@ -4746,9 +4743,6 @@ static int skl_compute_plane_wm(const struct drm_i915_private *dev_priv,
if (latency == 0) if (latency == 0)
return level == 0 ? -EINVAL : 0; return level == 0 ? -EINVAL : 0;
if (!intel_wm_plane_visible(cstate, intel_pstate))
return 0;
/* Display WA #1141: kbl,cfl */ /* Display WA #1141: kbl,cfl */
if ((IS_KABYLAKE(dev_priv) || IS_COFFEELAKE(dev_priv) || if ((IS_KABYLAKE(dev_priv) || IS_COFFEELAKE(dev_priv) ||
IS_CNL_REVID(dev_priv, CNL_REVID_A0, CNL_REVID_B0)) && IS_CNL_REVID(dev_priv, CNL_REVID_A0, CNL_REVID_B0)) &&
...@@ -4871,21 +4865,16 @@ static int skl_compute_plane_wm(const struct drm_i915_private *dev_priv, ...@@ -4871,21 +4865,16 @@ static int skl_compute_plane_wm(const struct drm_i915_private *dev_priv,
static int static int
skl_compute_wm_levels(const struct drm_i915_private *dev_priv, skl_compute_wm_levels(const struct drm_i915_private *dev_priv,
struct skl_ddb_allocation *ddb,
const struct intel_crtc_state *cstate, const struct intel_crtc_state *cstate,
const struct intel_plane_state *intel_pstate, const struct intel_plane_state *intel_pstate,
uint16_t ddb_blocks, uint16_t ddb_blocks,
const struct skl_wm_params *wm_params, const struct skl_wm_params *wm_params,
struct skl_plane_wm *wm,
struct skl_wm_level *levels) struct skl_wm_level *levels)
{ {
int level, max_level = ilk_wm_max_level(dev_priv); int level, max_level = ilk_wm_max_level(dev_priv);
struct skl_wm_level *result_prev = &levels[0]; struct skl_wm_level *result_prev = &levels[0];
int ret; int ret;
if (WARN_ON(!intel_pstate->base.fb))
return -EINVAL;
for (level = 0; level <= max_level; level++) { for (level = 0; level <= max_level; level++) {
struct skl_wm_level *result = &levels[level]; struct skl_wm_level *result = &levels[level];
...@@ -4903,9 +4892,6 @@ skl_compute_wm_levels(const struct drm_i915_private *dev_priv, ...@@ -4903,9 +4892,6 @@ skl_compute_wm_levels(const struct drm_i915_private *dev_priv,
result_prev = result; result_prev = result;
} }
if (intel_pstate->base.fb->format->format == DRM_FORMAT_NV12)
wm->is_planar = true;
return 0; return 0;
} }
...@@ -4943,9 +4929,6 @@ static void skl_compute_transition_wm(const struct intel_crtc_state *cstate, ...@@ -4943,9 +4929,6 @@ static void skl_compute_transition_wm(const struct intel_crtc_state *cstate,
const uint16_t trans_amount = 10; /* This is configurable amount */ const uint16_t trans_amount = 10; /* This is configurable amount */
uint16_t wm0_sel_res_b, trans_offset_b, res_blocks; uint16_t wm0_sel_res_b, trans_offset_b, res_blocks;
if (!cstate->base.active)
return;
/* Transition WM are not recommended by HW team for GEN9 */ /* Transition WM are not recommended by HW team for GEN9 */
if (INTEL_GEN(dev_priv) <= 9) if (INTEL_GEN(dev_priv) <= 9)
return; return;
...@@ -4994,97 +4977,135 @@ static void skl_compute_transition_wm(const struct intel_crtc_state *cstate, ...@@ -4994,97 +4977,135 @@ static void skl_compute_transition_wm(const struct intel_crtc_state *cstate,
} }
} }
static int __skl_build_plane_wm_single(struct skl_ddb_allocation *ddb, static int skl_build_plane_wm_single(struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm, struct intel_crtc_state *crtc_state,
enum plane_id plane_id, const struct intel_plane_state *plane_state,
const struct intel_crtc_state *cstate, enum plane_id plane_id, int color_plane)
const struct intel_plane_state *pstate,
int color_plane)
{ {
struct drm_i915_private *dev_priv = to_i915(pstate->base.plane->dev); struct intel_plane *plane = to_intel_plane(plane_state->base.plane);
struct skl_plane_wm *wm = &pipe_wm->planes[plane_id]; struct drm_i915_private *dev_priv = to_i915(plane->base.dev);
enum pipe pipe = to_intel_plane(pstate->base.plane)->pipe; struct skl_plane_wm *wm = &crtc_state->wm.skl.optimal.planes[plane_id];
struct skl_wm_params wm_params; struct skl_wm_params wm_params;
enum pipe pipe = plane->pipe;
uint16_t ddb_blocks = skl_ddb_entry_size(&ddb->plane[pipe][plane_id]); uint16_t ddb_blocks = skl_ddb_entry_size(&ddb->plane[pipe][plane_id]);
int ret; int ret;
ret = skl_compute_plane_wm_params(dev_priv, cstate, pstate, ret = skl_compute_plane_wm_params(dev_priv, crtc_state, plane_state,
&wm_params, color_plane); &wm_params, color_plane);
if (ret) if (ret)
return ret; return ret;
ret = skl_compute_wm_levels(dev_priv, ddb, cstate, pstate, ret = skl_compute_wm_levels(dev_priv, crtc_state, plane_state,
ddb_blocks, &wm_params, wm, wm->wm); ddb_blocks, &wm_params, wm->wm);
if (ret) if (ret)
return ret; return ret;
skl_compute_transition_wm(cstate, &wm_params, wm, ddb_blocks); skl_compute_transition_wm(crtc_state, &wm_params, wm, ddb_blocks);
return 0; return 0;
} }
static int skl_build_plane_wm_single(struct skl_ddb_allocation *ddb, static int skl_build_plane_wm_uv(struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm, struct intel_crtc_state *crtc_state,
const struct intel_crtc_state *cstate, const struct intel_plane_state *plane_state,
const struct intel_plane_state *pstate) enum plane_id plane_id)
{
enum plane_id plane_id = to_intel_plane(pstate->base.plane)->id;
return __skl_build_plane_wm_single(ddb, pipe_wm, plane_id, cstate, pstate, 0);
}
static int skl_build_plane_wm_planar(struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm,
const struct intel_crtc_state *cstate,
const struct intel_plane_state *pstate)
{ {
struct intel_plane *plane = to_intel_plane(pstate->base.plane); struct intel_plane *plane = to_intel_plane(plane_state->base.plane);
struct drm_i915_private *dev_priv = to_i915(plane->base.dev); struct drm_i915_private *dev_priv = to_i915(plane->base.dev);
enum plane_id plane_id = plane->id; struct skl_plane_wm *wm = &crtc_state->wm.skl.optimal.planes[plane_id];
struct skl_plane_wm *wm = &pipe_wm->planes[plane_id];
struct skl_wm_params wm_params; struct skl_wm_params wm_params;
enum pipe pipe = plane->pipe; enum pipe pipe = plane->pipe;
uint16_t ddb_blocks = skl_ddb_entry_size(&ddb->plane[pipe][plane_id]); uint16_t ddb_blocks = skl_ddb_entry_size(&ddb->uv_plane[pipe][plane_id]);
int ret; int ret;
ret = __skl_build_plane_wm_single(ddb, pipe_wm, plane_id, cstate, pstate, 0); wm->is_planar = true;
if (ret)
return ret;
/* uv plane watermarks must also be validated for NV12/Planar */ /* uv plane watermarks must also be validated for NV12/Planar */
ddb_blocks = skl_ddb_entry_size(&ddb->uv_plane[pipe][plane_id]); ret = skl_compute_plane_wm_params(dev_priv, crtc_state, plane_state,
&wm_params, 1);
if (ret)
return ret;
ret = skl_compute_plane_wm_params(dev_priv, cstate, pstate, &wm_params, 1); ret = skl_compute_wm_levels(dev_priv, crtc_state, plane_state,
ddb_blocks, &wm_params, wm->uv_wm);
if (ret) if (ret)
return ret; return ret;
return skl_compute_wm_levels(dev_priv, ddb, cstate, pstate, return 0;
ddb_blocks, &wm_params, wm, wm->uv_wm);
} }
static int icl_build_plane_wm_planar(struct skl_ddb_allocation *ddb, static int skl_build_plane_wm(struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm, struct skl_pipe_wm *pipe_wm,
const struct intel_crtc_state *cstate, struct intel_crtc_state *crtc_state,
const struct intel_plane_state *pstate) const struct intel_plane_state *plane_state)
{ {
struct intel_plane *plane = to_intel_plane(plane_state->base.plane);
const struct drm_framebuffer *fb = plane_state->base.fb;
enum plane_id plane_id = plane->id;
int ret; int ret;
enum plane_id y_plane_id = pstate->linked_plane->id;
enum plane_id uv_plane_id = to_intel_plane(pstate->base.plane)->id;
ret = __skl_build_plane_wm_single(ddb, pipe_wm, y_plane_id, if (!intel_wm_plane_visible(crtc_state, plane_state))
cstate, pstate, 0); return 0;
ret = skl_build_plane_wm_single(ddb, crtc_state, plane_state,
plane_id, 0);
if (ret) if (ret)
return ret; return ret;
return __skl_build_plane_wm_single(ddb, pipe_wm, uv_plane_id, if (fb->format->is_yuv && fb->format->num_planes > 1) {
cstate, pstate, 1); ret = skl_build_plane_wm_uv(ddb, crtc_state, plane_state,
plane_id);
if (ret)
return ret;
}
return 0;
}
static int icl_build_plane_wm(struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm,
struct intel_crtc_state *crtc_state,
const struct intel_plane_state *plane_state)
{
enum plane_id plane_id = to_intel_plane(plane_state->base.plane)->id;
int ret;
/* Watermarks calculated in master */
if (plane_state->slave)
return 0;
if (plane_state->linked_plane) {
const struct drm_framebuffer *fb = plane_state->base.fb;
enum plane_id y_plane_id = plane_state->linked_plane->id;
WARN_ON(!intel_wm_plane_visible(crtc_state, plane_state));
WARN_ON(!fb->format->is_yuv ||
fb->format->num_planes == 1);
ret = skl_build_plane_wm_single(ddb, crtc_state, plane_state,
y_plane_id, 0);
if (ret)
return ret;
ret = skl_build_plane_wm_single(ddb, crtc_state, plane_state,
plane_id, 1);
if (ret)
return ret;
} else if (intel_wm_plane_visible(crtc_state, plane_state)) {
ret = skl_build_plane_wm_single(ddb, crtc_state, plane_state,
plane_id, 0);
if (ret)
return ret;
}
return 0;
} }
static int skl_build_pipe_wm(struct intel_crtc_state *cstate, static int skl_build_pipe_wm(struct intel_crtc_state *cstate,
struct skl_ddb_allocation *ddb, struct skl_ddb_allocation *ddb,
struct skl_pipe_wm *pipe_wm) struct skl_pipe_wm *pipe_wm)
{ {
struct drm_i915_private *dev_priv = to_i915(cstate->base.crtc->dev);
struct drm_crtc_state *crtc_state = &cstate->base; struct drm_crtc_state *crtc_state = &cstate->base;
struct drm_plane *plane; struct drm_plane *plane;
const struct drm_plane_state *pstate; const struct drm_plane_state *pstate;
...@@ -5100,18 +5121,12 @@ static int skl_build_pipe_wm(struct intel_crtc_state *cstate, ...@@ -5100,18 +5121,12 @@ static int skl_build_pipe_wm(struct intel_crtc_state *cstate,
const struct intel_plane_state *intel_pstate = const struct intel_plane_state *intel_pstate =
to_intel_plane_state(pstate); to_intel_plane_state(pstate);
/* Watermarks calculated in master */ if (INTEL_GEN(dev_priv) >= 11)
if (intel_pstate->slave) ret = icl_build_plane_wm(ddb, pipe_wm,
continue; cstate, intel_pstate);
if (intel_pstate->linked_plane)
ret = icl_build_plane_wm_planar(ddb, pipe_wm, cstate, intel_pstate);
else if (intel_pstate->base.fb &&
intel_pstate->base.fb->format->format == DRM_FORMAT_NV12)
ret = skl_build_plane_wm_planar(ddb, pipe_wm, cstate, intel_pstate);
else else
ret = skl_build_plane_wm_single(ddb, pipe_wm, cstate, intel_pstate); ret = skl_build_plane_wm(ddb, pipe_wm,
cstate, intel_pstate);
if (ret) if (ret)
return ret; return ret;
} }
......
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