Message ID | 1475885497-6094-6-git-send-email-cpaul@redhat.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Op 08-10-16 om 02:11 schreef Lyude: > Now that we've make skl_wm_levels make a little more sense, we can > remove all of the redundant wm information. Up until now we'd been > storing two copies of all of the skl watermarks: one being the > skl_pipe_wm structs, the other being the global wm struct in > drm_i915_private containing the raw register values. This is confusing > and problematic, since it means we're prone to accidentally letting the > two copies go out of sync. So, get rid of all of the functions > responsible for computing the register values and just use a single > helper, skl_write_wm_level(), to convert and write the new watermarks on > the fly. > > Changes since v1: > - Fixup skl_write_wm_level() > - Fixup skl_wm_level_from_reg_val() > - Don't forget to copy *active to intel_crtc->wm.active.skl > > Signed-off-by: Lyude <cpaul@redhat.com> > Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> > Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> > Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> > --- > drivers/gpu/drm/i915/i915_drv.h | 2 - > drivers/gpu/drm/i915/intel_display.c | 14 ++- > drivers/gpu/drm/i915/intel_drv.h | 6 +- > drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++----------------------- > drivers/gpu/drm/i915/intel_sprite.c | 8 +- > 5 files changed, 90 insertions(+), 144 deletions(-) > > diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h > index 0287c93..76583b2 100644 > --- a/drivers/gpu/drm/i915/i915_drv.h > +++ b/drivers/gpu/drm/i915/i915_drv.h > @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { > struct skl_wm_values { > unsigned dirty_pipes; > struct skl_ddb_allocation ddb; > - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; > - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; > }; > > struct skl_wm_level { > diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c > index a71d05a..39400a0 100644 > --- a/drivers/gpu/drm/i915/intel_display.c > +++ b/drivers/gpu/drm/i915/intel_display.c > @@ -3378,6 +3378,8 @@ static void skylake_update_primary_plane(struct drm_plane *plane, > struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state->base.crtc); > struct drm_framebuffer *fb = plane_state->base.fb; > const struct skl_wm_values *wm = &dev_priv->wm.skl_results; > + const struct skl_plane_wm *p_wm = > + &crtc_state->wm.skl.optimal.planes[0]; > int pipe = intel_crtc->pipe; > u32 plane_ctl; > unsigned int rotation = plane_state->base.rotation; > @@ -3414,7 +3416,7 @@ static void skylake_update_primary_plane(struct drm_plane *plane, > intel_crtc->adjusted_y = src_y; > > if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) > - skl_write_plane_wm(intel_crtc, wm, 0); > + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, 0); > > I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); > I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | src_x); > @@ -3448,6 +3450,8 @@ static void skylake_disable_primary_plane(struct drm_plane *primary, > struct drm_device *dev = crtc->dev; > struct drm_i915_private *dev_priv = to_i915(dev); > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > + struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); > + const struct skl_plane_wm *p_wm = &cstate->wm.skl.optimal.planes[0]; > int pipe = intel_crtc->pipe; > > /* > @@ -3455,7 +3459,8 @@ static void skylake_disable_primary_plane(struct drm_plane *primary, > * plane's visiblity isn't actually changing neither is its watermarks. > */ > if (!crtc->primary->state->visible) > - skl_write_plane_wm(intel_crtc, &dev_priv->wm.skl_results, 0); > + skl_write_plane_wm(intel_crtc, p_wm, > + &dev_priv->wm.skl_results.ddb, 0); > > I915_WRITE(PLANE_CTL(pipe, 0), 0); > I915_WRITE(PLANE_SURF(pipe, 0), 0); > @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct drm_crtc *crtc, u32 base, > struct drm_device *dev = crtc->dev; > struct drm_i915_private *dev_priv = to_i915(dev); > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > + struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); > const struct skl_wm_values *wm = &dev_priv->wm.skl_results; > + const struct skl_plane_wm *p_wm = > + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; > int pipe = intel_crtc->pipe; > uint32_t cntl = 0; > > if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & drm_crtc_mask(crtc)) > - skl_write_cursor_wm(intel_crtc, wm); > + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); > > if (plane_state && plane_state->base.visible) { > cntl = MCURSOR_GAMMA_ENABLE; > diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h > index d684f4f..958dc72 100644 > --- a/drivers/gpu/drm/i915/intel_drv.h > +++ b/drivers/gpu/drm/i915/intel_drv.h > @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const struct skl_ddb_allocation *old, > bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, > struct intel_crtc *intel_crtc); > void skl_write_cursor_wm(struct intel_crtc *intel_crtc, > - const struct skl_wm_values *wm); > + const struct skl_plane_wm *wm, > + const struct skl_ddb_allocation *ddb); > void skl_write_plane_wm(struct intel_crtc *intel_crtc, > - const struct skl_wm_values *wm, > + const struct skl_plane_wm *wm, > + const struct skl_ddb_allocation *ddb, > int plane); > uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state *pipe_config); > bool ilk_disable_lp_wm(struct drm_device *dev); > diff --git a/drivers/gpu/drm/i915/intel_pm.c b/drivers/gpu/drm/i915/intel_pm.c > index 5dbaf12..5cb537c 100644 > --- a/drivers/gpu/drm/i915/intel_pm.c > +++ b/drivers/gpu/drm/i915/intel_pm.c > @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct drm_atomic_state *state) > struct drm_i915_private *dev_priv = to_i915(dev); > struct intel_atomic_state *intel_state = to_intel_atomic_state(state); > struct drm_crtc *crtc; > + struct intel_crtc_state *cstate; > + struct skl_plane_wm *wm; > enum pipe pipe; > int level, plane; > > @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct drm_atomic_state *state) > /* Since we're now guaranteed to only have one active CRTC... */ > pipe = ffs(intel_state->active_crtcs) - 1; > crtc = dev_priv->pipe_to_crtc_mapping[pipe]; > + cstate = intel_atomic_get_crtc_state(state, to_intel_crtc(crtc)); Wrong cstate here, should be crtc->state since it's committed and crtc_state is the old state. ~Maarten
Em Qui, 2016-10-13 às 15:39 +0200, Maarten Lankhorst escreveu: > Op 08-10-16 om 02:11 schreef Lyude: > > > > Now that we've make skl_wm_levels make a little more sense, we can > > remove all of the redundant wm information. Up until now we'd been > > storing two copies of all of the skl watermarks: one being the > > skl_pipe_wm structs, the other being the global wm struct in > > drm_i915_private containing the raw register values. This is > > confusing > > and problematic, since it means we're prone to accidentally letting > > the > > two copies go out of sync. So, get rid of all of the functions > > responsible for computing the register values and just use a single > > helper, skl_write_wm_level(), to convert and write the new > > watermarks on > > the fly. > > > > Changes since v1: > > - Fixup skl_write_wm_level() > > - Fixup skl_wm_level_from_reg_val() > > - Don't forget to copy *active to intel_crtc->wm.active.skl > > > > Signed-off-by: Lyude <cpaul@redhat.com> > > Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> > > Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> > > Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> > > --- > > drivers/gpu/drm/i915/i915_drv.h | 2 - > > drivers/gpu/drm/i915/intel_display.c | 14 ++- > > drivers/gpu/drm/i915/intel_drv.h | 6 +- > > drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++----------- > > ------------ > > drivers/gpu/drm/i915/intel_sprite.c | 8 +- > > 5 files changed, 90 insertions(+), 144 deletions(-) > > > > diff --git a/drivers/gpu/drm/i915/i915_drv.h > > b/drivers/gpu/drm/i915/i915_drv.h > > index 0287c93..76583b2 100644 > > --- a/drivers/gpu/drm/i915/i915_drv.h > > +++ b/drivers/gpu/drm/i915/i915_drv.h > > @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { > > struct skl_wm_values { > > unsigned dirty_pipes; > > struct skl_ddb_allocation ddb; > > - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; > > - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; > > }; > > > > struct skl_wm_level { > > diff --git a/drivers/gpu/drm/i915/intel_display.c > > b/drivers/gpu/drm/i915/intel_display.c > > index a71d05a..39400a0 100644 > > --- a/drivers/gpu/drm/i915/intel_display.c > > +++ b/drivers/gpu/drm/i915/intel_display.c > > @@ -3378,6 +3378,8 @@ static void > > skylake_update_primary_plane(struct drm_plane *plane, > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state- > > >base.crtc); > > struct drm_framebuffer *fb = plane_state->base.fb; > > const struct skl_wm_values *wm = &dev_priv- > > >wm.skl_results; > > + const struct skl_plane_wm *p_wm = > > + &crtc_state->wm.skl.optimal.planes[0]; > > int pipe = intel_crtc->pipe; > > u32 plane_ctl; > > unsigned int rotation = plane_state->base.rotation; > > @@ -3414,7 +3416,7 @@ static void > > skylake_update_primary_plane(struct drm_plane *plane, > > intel_crtc->adjusted_y = src_y; > > > > if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) > > - skl_write_plane_wm(intel_crtc, wm, 0); > > + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, 0); > > > > I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); > > I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | src_x); > > @@ -3448,6 +3450,8 @@ static void > > skylake_disable_primary_plane(struct drm_plane *primary, > > struct drm_device *dev = crtc->dev; > > struct drm_i915_private *dev_priv = to_i915(dev); > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > + struct intel_crtc_state *cstate = > > to_intel_crtc_state(crtc->state); > > + const struct skl_plane_wm *p_wm = &cstate- > > >wm.skl.optimal.planes[0]; > > int pipe = intel_crtc->pipe; > > > > /* > > @@ -3455,7 +3459,8 @@ static void > > skylake_disable_primary_plane(struct drm_plane *primary, > > * plane's visiblity isn't actually changing neither is > > its watermarks. > > */ > > if (!crtc->primary->state->visible) > > - skl_write_plane_wm(intel_crtc, &dev_priv- > > >wm.skl_results, 0); > > + skl_write_plane_wm(intel_crtc, p_wm, > > + &dev_priv->wm.skl_results.ddb, > > 0); > > > > I915_WRITE(PLANE_CTL(pipe, 0), 0); > > I915_WRITE(PLANE_SURF(pipe, 0), 0); > > @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct > > drm_crtc *crtc, u32 base, > > struct drm_device *dev = crtc->dev; > > struct drm_i915_private *dev_priv = to_i915(dev); > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > + struct intel_crtc_state *cstate = > > to_intel_crtc_state(crtc->state); > > const struct skl_wm_values *wm = &dev_priv- > > >wm.skl_results; > > + const struct skl_plane_wm *p_wm = > > + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; > > int pipe = intel_crtc->pipe; > > uint32_t cntl = 0; > > > > if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & > > drm_crtc_mask(crtc)) > > - skl_write_cursor_wm(intel_crtc, wm); > > + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); > > > > if (plane_state && plane_state->base.visible) { > > cntl = MCURSOR_GAMMA_ENABLE; > > diff --git a/drivers/gpu/drm/i915/intel_drv.h > > b/drivers/gpu/drm/i915/intel_drv.h > > index d684f4f..958dc72 100644 > > --- a/drivers/gpu/drm/i915/intel_drv.h > > +++ b/drivers/gpu/drm/i915/intel_drv.h > > @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const struct > > skl_ddb_allocation *old, > > bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, > > struct intel_crtc *intel_crtc); > > void skl_write_cursor_wm(struct intel_crtc *intel_crtc, > > - const struct skl_wm_values *wm); > > + const struct skl_plane_wm *wm, > > + const struct skl_ddb_allocation *ddb); > > void skl_write_plane_wm(struct intel_crtc *intel_crtc, > > - const struct skl_wm_values *wm, > > + const struct skl_plane_wm *wm, > > + const struct skl_ddb_allocation *ddb, > > int plane); > > uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state > > *pipe_config); > > bool ilk_disable_lp_wm(struct drm_device *dev); > > diff --git a/drivers/gpu/drm/i915/intel_pm.c > > b/drivers/gpu/drm/i915/intel_pm.c > > index 5dbaf12..5cb537c 100644 > > --- a/drivers/gpu/drm/i915/intel_pm.c > > +++ b/drivers/gpu/drm/i915/intel_pm.c > > @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct > > drm_atomic_state *state) > > struct drm_i915_private *dev_priv = to_i915(dev); > > struct intel_atomic_state *intel_state = > > to_intel_atomic_state(state); > > struct drm_crtc *crtc; > > + struct intel_crtc_state *cstate; > > + struct skl_plane_wm *wm; > > enum pipe pipe; > > int level, plane; > > > > @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct > > drm_atomic_state *state) > > /* Since we're now guaranteed to only have one active > > CRTC... */ > > pipe = ffs(intel_state->active_crtcs) - 1; > > crtc = dev_priv->pipe_to_crtc_mapping[pipe]; > > + cstate = intel_atomic_get_crtc_state(state, > > to_intel_crtc(crtc)); > Wrong cstate here, should be crtc->state since it's committed and > crtc_state is the old state. Ah... How many instances of the "you're looking at the wrong state struct" have we spotted/fixed ever since the introduction of the atomic code? Can't we try to do something more human-proof in order to avoid these problems from happening again and again? Like for example copying the old states to a specific structure like drm_atomic_state->old_states_which_you_probably_shouldn_be_using, and then setting the CRTC/connector/plane states in the drm_atomic_state struct to NULL so bugs will be much easier to spot? Anyway, with this problem fixed, feel free to add: Reviewed-by: Paulo Zanoni <paulo.r.zanoni@intel.com> > > ~Maarten
Em Qui, 2016-10-13 às 17:04 -0300, Paulo Zanoni escreveu: > Em Qui, 2016-10-13 às 15:39 +0200, Maarten Lankhorst escreveu: > > > > Op 08-10-16 om 02:11 schreef Lyude: > > > > > > > > > Now that we've make skl_wm_levels make a little more sense, we > > > can > > > remove all of the redundant wm information. Up until now we'd > > > been > > > storing two copies of all of the skl watermarks: one being the > > > skl_pipe_wm structs, the other being the global wm struct in > > > drm_i915_private containing the raw register values. This is > > > confusing > > > and problematic, since it means we're prone to accidentally > > > letting > > > the > > > two copies go out of sync. So, get rid of all of the functions > > > responsible for computing the register values and just use a > > > single > > > helper, skl_write_wm_level(), to convert and write the new > > > watermarks on > > > the fly. > > > > > > Changes since v1: > > > - Fixup skl_write_wm_level() > > > - Fixup skl_wm_level_from_reg_val() > > > - Don't forget to copy *active to intel_crtc->wm.active.skl > > > > > > Signed-off-by: Lyude <cpaul@redhat.com> > > > Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com > > > > > > > Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> > > > Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> > > > --- > > > drivers/gpu/drm/i915/i915_drv.h | 2 - > > > drivers/gpu/drm/i915/intel_display.c | 14 ++- > > > drivers/gpu/drm/i915/intel_drv.h | 6 +- > > > drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++--------- > > > -- > > > ------------ > > > drivers/gpu/drm/i915/intel_sprite.c | 8 +- > > > 5 files changed, 90 insertions(+), 144 deletions(-) > > > > > > diff --git a/drivers/gpu/drm/i915/i915_drv.h > > > b/drivers/gpu/drm/i915/i915_drv.h > > > index 0287c93..76583b2 100644 > > > --- a/drivers/gpu/drm/i915/i915_drv.h > > > +++ b/drivers/gpu/drm/i915/i915_drv.h > > > @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { > > > struct skl_wm_values { > > > unsigned dirty_pipes; > > > struct skl_ddb_allocation ddb; > > > - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; > > > - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; > > > }; > > > > > > struct skl_wm_level { > > > diff --git a/drivers/gpu/drm/i915/intel_display.c > > > b/drivers/gpu/drm/i915/intel_display.c > > > index a71d05a..39400a0 100644 > > > --- a/drivers/gpu/drm/i915/intel_display.c > > > +++ b/drivers/gpu/drm/i915/intel_display.c > > > @@ -3378,6 +3378,8 @@ static void > > > skylake_update_primary_plane(struct drm_plane *plane, > > > struct intel_crtc *intel_crtc = > > > to_intel_crtc(crtc_state- > > > > > > > > base.crtc); > > > struct drm_framebuffer *fb = plane_state->base.fb; > > > const struct skl_wm_values *wm = &dev_priv- > > > > > > > > wm.skl_results; > > > + const struct skl_plane_wm *p_wm = > > > + &crtc_state->wm.skl.optimal.planes[0]; > > > int pipe = intel_crtc->pipe; > > > u32 plane_ctl; > > > unsigned int rotation = plane_state->base.rotation; > > > @@ -3414,7 +3416,7 @@ static void > > > skylake_update_primary_plane(struct drm_plane *plane, > > > intel_crtc->adjusted_y = src_y; > > > > > > if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) > > > - skl_write_plane_wm(intel_crtc, wm, 0); > > > + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, > > > 0); > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); > > > I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | > > > src_x); > > > @@ -3448,6 +3450,8 @@ static void > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > struct drm_device *dev = crtc->dev; > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > + struct intel_crtc_state *cstate = > > > to_intel_crtc_state(crtc->state); > > > + const struct skl_plane_wm *p_wm = &cstate- > > > > > > > > wm.skl.optimal.planes[0]; > > > int pipe = intel_crtc->pipe; > > > > > > /* > > > @@ -3455,7 +3459,8 @@ static void > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > * plane's visiblity isn't actually changing neither is > > > its watermarks. > > > */ > > > if (!crtc->primary->state->visible) > > > - skl_write_plane_wm(intel_crtc, &dev_priv- > > > > > > > > wm.skl_results, 0); > > > + skl_write_plane_wm(intel_crtc, p_wm, > > > + &dev_priv- > > > >wm.skl_results.ddb, > > > 0); > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), 0); > > > I915_WRITE(PLANE_SURF(pipe, 0), 0); > > > @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct > > > drm_crtc *crtc, u32 base, > > > struct drm_device *dev = crtc->dev; > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > + struct intel_crtc_state *cstate = > > > to_intel_crtc_state(crtc->state); > > > const struct skl_wm_values *wm = &dev_priv- > > > > > > > > wm.skl_results; > > > + const struct skl_plane_wm *p_wm = > > > + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; > > > int pipe = intel_crtc->pipe; > > > uint32_t cntl = 0; > > > > > > if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & > > > drm_crtc_mask(crtc)) > > > - skl_write_cursor_wm(intel_crtc, wm); > > > + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); > > > > > > if (plane_state && plane_state->base.visible) { > > > cntl = MCURSOR_GAMMA_ENABLE; > > > diff --git a/drivers/gpu/drm/i915/intel_drv.h > > > b/drivers/gpu/drm/i915/intel_drv.h > > > index d684f4f..958dc72 100644 > > > --- a/drivers/gpu/drm/i915/intel_drv.h > > > +++ b/drivers/gpu/drm/i915/intel_drv.h > > > @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const > > > struct > > > skl_ddb_allocation *old, > > > bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, > > > struct intel_crtc *intel_crtc); > > > void skl_write_cursor_wm(struct intel_crtc *intel_crtc, > > > - const struct skl_wm_values *wm); > > > + const struct skl_plane_wm *wm, > > > + const struct skl_ddb_allocation *ddb); > > > void skl_write_plane_wm(struct intel_crtc *intel_crtc, > > > - const struct skl_wm_values *wm, > > > + const struct skl_plane_wm *wm, > > > + const struct skl_ddb_allocation *ddb, > > > int plane); > > > uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state > > > *pipe_config); > > > bool ilk_disable_lp_wm(struct drm_device *dev); > > > diff --git a/drivers/gpu/drm/i915/intel_pm.c > > > b/drivers/gpu/drm/i915/intel_pm.c > > > index 5dbaf12..5cb537c 100644 > > > --- a/drivers/gpu/drm/i915/intel_pm.c > > > +++ b/drivers/gpu/drm/i915/intel_pm.c > > > @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct > > > drm_atomic_state *state) > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_atomic_state *intel_state = > > > to_intel_atomic_state(state); > > > struct drm_crtc *crtc; > > > + struct intel_crtc_state *cstate; > > > + struct skl_plane_wm *wm; > > > enum pipe pipe; > > > int level, plane; > > > > > > @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct > > > drm_atomic_state *state) > > > /* Since we're now guaranteed to only have one active > > > CRTC... */ > > > pipe = ffs(intel_state->active_crtcs) - 1; > > > crtc = dev_priv->pipe_to_crtc_mapping[pipe]; > > > + cstate = intel_atomic_get_crtc_state(state, > > > to_intel_crtc(crtc)); > > Wrong cstate here, should be crtc->state since it's committed and > > crtc_state is the old state. > > Ah... How many instances of the "you're looking at the wrong state > struct" have we spotted/fixed ever since the introduction of the > atomic > code? Can't we try to do something more human-proof in order to avoid > these problems from happening again and again? > > Like for example copying the old states to a specific structure like > drm_atomic_state->old_states_which_you_probably_shouldn_be_using, and > then setting the CRTC/connector/plane states in the drm_atomic_state > struct to NULL so bugs will be much easier to spot? > > Anyway, with this problem fixed, feel free to add: > Reviewed-by: Paulo Zanoni <paulo.r.zanoni@intel.com> Oh, I forgot to mention: this patch has a conflict with my "unconditionally apply the memory workaround" patch, so we may need to decide which one gets merged first. Of course I'd suggest my patch to be merged first since I want it to land in stable, but I'm always biased towards my patches. If you agree, I can even volunteer myself to solve the conflicts of this patch later when applying. > > > > > > > ~Maarten > _______________________________________________ > Intel-gfx mailing list > Intel-gfx@lists.freedesktop.org > https://lists.freedesktop.org/mailman/listinfo/intel-gfx
Your is SAGV related, and when we don't make the SAGV happy people's machines usually hang. So I'm definitely for your patches getting merged first On Thu, 2016-10-13 at 17:07 -0300, Paulo Zanoni wrote: > Em Qui, 2016-10-13 às 17:04 -0300, Paulo Zanoni escreveu: > > > > Em Qui, 2016-10-13 às 15:39 +0200, Maarten Lankhorst escreveu: > > > > > > > > > Op 08-10-16 om 02:11 schreef Lyude: > > > > > > > > > > > > > > > > Now that we've make skl_wm_levels make a little more sense, we > > > > can > > > > remove all of the redundant wm information. Up until now we'd > > > > been > > > > storing two copies of all of the skl watermarks: one being the > > > > skl_pipe_wm structs, the other being the global wm struct in > > > > drm_i915_private containing the raw register values. This is > > > > confusing > > > > and problematic, since it means we're prone to accidentally > > > > letting > > > > the > > > > two copies go out of sync. So, get rid of all of the functions > > > > responsible for computing the register values and just use a > > > > single > > > > helper, skl_write_wm_level(), to convert and write the new > > > > watermarks on > > > > the fly. > > > > > > > > Changes since v1: > > > > - Fixup skl_write_wm_level() > > > > - Fixup skl_wm_level_from_reg_val() > > > > - Don't forget to copy *active to intel_crtc->wm.active.skl > > > > > > > > Signed-off-by: Lyude <cpaul@redhat.com> > > > > Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.c > > > > om > > > > > > > > > > > > > > Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> > > > > Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> > > > > --- > > > > drivers/gpu/drm/i915/i915_drv.h | 2 - > > > > drivers/gpu/drm/i915/intel_display.c | 14 ++- > > > > drivers/gpu/drm/i915/intel_drv.h | 6 +- > > > > drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++------- > > > > -- > > > > -- > > > > ------------ > > > > drivers/gpu/drm/i915/intel_sprite.c | 8 +- > > > > 5 files changed, 90 insertions(+), 144 deletions(-) > > > > > > > > diff --git a/drivers/gpu/drm/i915/i915_drv.h > > > > b/drivers/gpu/drm/i915/i915_drv.h > > > > index 0287c93..76583b2 100644 > > > > --- a/drivers/gpu/drm/i915/i915_drv.h > > > > +++ b/drivers/gpu/drm/i915/i915_drv.h > > > > @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { > > > > struct skl_wm_values { > > > > unsigned dirty_pipes; > > > > struct skl_ddb_allocation ddb; > > > > - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; > > > > - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; > > > > }; > > > > > > > > struct skl_wm_level { > > > > diff --git a/drivers/gpu/drm/i915/intel_display.c > > > > b/drivers/gpu/drm/i915/intel_display.c > > > > index a71d05a..39400a0 100644 > > > > --- a/drivers/gpu/drm/i915/intel_display.c > > > > +++ b/drivers/gpu/drm/i915/intel_display.c > > > > @@ -3378,6 +3378,8 @@ static void > > > > skylake_update_primary_plane(struct drm_plane *plane, > > > > struct intel_crtc *intel_crtc = > > > > to_intel_crtc(crtc_state- > > > > > > > > > > > > > > > base.crtc); > > > > struct drm_framebuffer *fb = plane_state->base.fb; > > > > const struct skl_wm_values *wm = &dev_priv- > > > > > > > > > > > > > > > wm.skl_results; > > > > + const struct skl_plane_wm *p_wm = > > > > + &crtc_state->wm.skl.optimal.planes[0]; > > > > int pipe = intel_crtc->pipe; > > > > u32 plane_ctl; > > > > unsigned int rotation = plane_state->base.rotation; > > > > @@ -3414,7 +3416,7 @@ static void > > > > skylake_update_primary_plane(struct drm_plane *plane, > > > > intel_crtc->adjusted_y = src_y; > > > > > > > > if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc- > > > > >base)) > > > > - skl_write_plane_wm(intel_crtc, wm, 0); > > > > + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, > > > > 0); > > > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); > > > > I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | > > > > src_x); > > > > @@ -3448,6 +3450,8 @@ static void > > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > > struct drm_device *dev = crtc->dev; > > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > > + struct intel_crtc_state *cstate = > > > > to_intel_crtc_state(crtc->state); > > > > + const struct skl_plane_wm *p_wm = &cstate- > > > > > > > > > > > > > > > wm.skl.optimal.planes[0]; > > > > int pipe = intel_crtc->pipe; > > > > > > > > /* > > > > @@ -3455,7 +3459,8 @@ static void > > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > > * plane's visiblity isn't actually changing neither > > > > is > > > > its watermarks. > > > > */ > > > > if (!crtc->primary->state->visible) > > > > - skl_write_plane_wm(intel_crtc, &dev_priv- > > > > > > > > > > > > > > > wm.skl_results, 0); > > > > + skl_write_plane_wm(intel_crtc, p_wm, > > > > + &dev_priv- > > > > > > > > > > wm.skl_results.ddb, > > > > 0); > > > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), 0); > > > > I915_WRITE(PLANE_SURF(pipe, 0), 0); > > > > @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct > > > > drm_crtc *crtc, u32 base, > > > > struct drm_device *dev = crtc->dev; > > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > > + struct intel_crtc_state *cstate = > > > > to_intel_crtc_state(crtc->state); > > > > const struct skl_wm_values *wm = &dev_priv- > > > > > > > > > > > > > > > wm.skl_results; > > > > + const struct skl_plane_wm *p_wm = > > > > + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; > > > > int pipe = intel_crtc->pipe; > > > > uint32_t cntl = 0; > > > > > > > > if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & > > > > drm_crtc_mask(crtc)) > > > > - skl_write_cursor_wm(intel_crtc, wm); > > > > + skl_write_cursor_wm(intel_crtc, p_wm, &wm- > > > > >ddb); > > > > > > > > if (plane_state && plane_state->base.visible) { > > > > cntl = MCURSOR_GAMMA_ENABLE; > > > > diff --git a/drivers/gpu/drm/i915/intel_drv.h > > > > b/drivers/gpu/drm/i915/intel_drv.h > > > > index d684f4f..958dc72 100644 > > > > --- a/drivers/gpu/drm/i915/intel_drv.h > > > > +++ b/drivers/gpu/drm/i915/intel_drv.h > > > > @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const > > > > struct > > > > skl_ddb_allocation *old, > > > > bool skl_ddb_allocation_overlaps(struct drm_atomic_state > > > > *state, > > > > struct intel_crtc > > > > *intel_crtc); > > > > void skl_write_cursor_wm(struct intel_crtc *intel_crtc, > > > > - const struct skl_wm_values *wm); > > > > + const struct skl_plane_wm *wm, > > > > + const struct skl_ddb_allocation > > > > *ddb); > > > > void skl_write_plane_wm(struct intel_crtc *intel_crtc, > > > > - const struct skl_wm_values *wm, > > > > + const struct skl_plane_wm *wm, > > > > + const struct skl_ddb_allocation *ddb, > > > > int plane); > > > > uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state > > > > *pipe_config); > > > > bool ilk_disable_lp_wm(struct drm_device *dev); > > > > diff --git a/drivers/gpu/drm/i915/intel_pm.c > > > > b/drivers/gpu/drm/i915/intel_pm.c > > > > index 5dbaf12..5cb537c 100644 > > > > --- a/drivers/gpu/drm/i915/intel_pm.c > > > > +++ b/drivers/gpu/drm/i915/intel_pm.c > > > > @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct > > > > drm_atomic_state *state) > > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > > struct intel_atomic_state *intel_state = > > > > to_intel_atomic_state(state); > > > > struct drm_crtc *crtc; > > > > + struct intel_crtc_state *cstate; > > > > + struct skl_plane_wm *wm; > > > > enum pipe pipe; > > > > int level, plane; > > > > > > > > @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct > > > > drm_atomic_state *state) > > > > /* Since we're now guaranteed to only have one active > > > > CRTC... */ > > > > pipe = ffs(intel_state->active_crtcs) - 1; > > > > crtc = dev_priv->pipe_to_crtc_mapping[pipe]; > > > > + cstate = intel_atomic_get_crtc_state(state, > > > > to_intel_crtc(crtc)); > > > Wrong cstate here, should be crtc->state since it's committed and > > > crtc_state is the old state. > > > > Ah... How many instances of the "you're looking at the wrong state > > struct" have we spotted/fixed ever since the introduction of the > > atomic > > code? Can't we try to do something more human-proof in order to > > avoid > > these problems from happening again and again? > > > > Like for example copying the old states to a specific structure > > like > > drm_atomic_state->old_states_which_you_probably_shouldn_be_using, > > and > > then setting the CRTC/connector/plane states in the > > drm_atomic_state > > struct to NULL so bugs will be much easier to spot? > > > > Anyway, with this problem fixed, feel free to add: > > Reviewed-by: Paulo Zanoni <paulo.r.zanoni@intel.com> > > Oh, I forgot to mention: this patch has a conflict with my > "unconditionally apply the memory workaround" patch, so we may need > to > decide which one gets merged first. Of course I'd suggest my patch to > be merged first since I want it to land in stable, but I'm always > biased towards my patches. If you agree, I can even volunteer myself > to > solve the conflicts of this patch later when applying. > > > > > > > > > > > > > > > > > ~Maarten > > _______________________________________________ > > Intel-gfx mailing list > > Intel-gfx@lists.freedesktop.org > > https://lists.freedesktop.org/mailman/listinfo/intel-gfx
On Thu, Oct 13, 2016 at 05:04:03PM -0300, Paulo Zanoni wrote: > Em Qui, 2016-10-13 às 15:39 +0200, Maarten Lankhorst escreveu: > > Op 08-10-16 om 02:11 schreef Lyude: > > > > > > Now that we've make skl_wm_levels make a little more sense, we can > > > remove all of the redundant wm information. Up until now we'd been > > > storing two copies of all of the skl watermarks: one being the > > > skl_pipe_wm structs, the other being the global wm struct in > > > drm_i915_private containing the raw register values. This is > > > confusing > > > and problematic, since it means we're prone to accidentally letting > > > the > > > two copies go out of sync. So, get rid of all of the functions > > > responsible for computing the register values and just use a single > > > helper, skl_write_wm_level(), to convert and write the new > > > watermarks on > > > the fly. > > > > > > Changes since v1: > > > - Fixup skl_write_wm_level() > > > - Fixup skl_wm_level_from_reg_val() > > > - Don't forget to copy *active to intel_crtc->wm.active.skl > > > > > > Signed-off-by: Lyude <cpaul@redhat.com> > > > Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> > > > Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> > > > Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> > > > --- > > > drivers/gpu/drm/i915/i915_drv.h | 2 - > > > drivers/gpu/drm/i915/intel_display.c | 14 ++- > > > drivers/gpu/drm/i915/intel_drv.h | 6 +- > > > drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++----------- > > > ------------ > > > drivers/gpu/drm/i915/intel_sprite.c | 8 +- > > > 5 files changed, 90 insertions(+), 144 deletions(-) > > > > > > diff --git a/drivers/gpu/drm/i915/i915_drv.h > > > b/drivers/gpu/drm/i915/i915_drv.h > > > index 0287c93..76583b2 100644 > > > --- a/drivers/gpu/drm/i915/i915_drv.h > > > +++ b/drivers/gpu/drm/i915/i915_drv.h > > > @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { > > > struct skl_wm_values { > > > unsigned dirty_pipes; > > > struct skl_ddb_allocation ddb; > > > - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; > > > - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; > > > }; > > > > > > struct skl_wm_level { > > > diff --git a/drivers/gpu/drm/i915/intel_display.c > > > b/drivers/gpu/drm/i915/intel_display.c > > > index a71d05a..39400a0 100644 > > > --- a/drivers/gpu/drm/i915/intel_display.c > > > +++ b/drivers/gpu/drm/i915/intel_display.c > > > @@ -3378,6 +3378,8 @@ static void > > > skylake_update_primary_plane(struct drm_plane *plane, > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state- > > > >base.crtc); > > > struct drm_framebuffer *fb = plane_state->base.fb; > > > const struct skl_wm_values *wm = &dev_priv- > > > >wm.skl_results; > > > + const struct skl_plane_wm *p_wm = > > > + &crtc_state->wm.skl.optimal.planes[0]; > > > int pipe = intel_crtc->pipe; > > > u32 plane_ctl; > > > unsigned int rotation = plane_state->base.rotation; > > > @@ -3414,7 +3416,7 @@ static void > > > skylake_update_primary_plane(struct drm_plane *plane, > > > intel_crtc->adjusted_y = src_y; > > > > > > if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) > > > - skl_write_plane_wm(intel_crtc, wm, 0); > > > + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, 0); > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); > > > I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | src_x); > > > @@ -3448,6 +3450,8 @@ static void > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > struct drm_device *dev = crtc->dev; > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > + struct intel_crtc_state *cstate = > > > to_intel_crtc_state(crtc->state); > > > + const struct skl_plane_wm *p_wm = &cstate- > > > >wm.skl.optimal.planes[0]; > > > int pipe = intel_crtc->pipe; > > > > > > /* > > > @@ -3455,7 +3459,8 @@ static void > > > skylake_disable_primary_plane(struct drm_plane *primary, > > > * plane's visiblity isn't actually changing neither is > > > its watermarks. > > > */ > > > if (!crtc->primary->state->visible) > > > - skl_write_plane_wm(intel_crtc, &dev_priv- > > > >wm.skl_results, 0); > > > + skl_write_plane_wm(intel_crtc, p_wm, > > > + &dev_priv->wm.skl_results.ddb, > > > 0); > > > > > > I915_WRITE(PLANE_CTL(pipe, 0), 0); > > > I915_WRITE(PLANE_SURF(pipe, 0), 0); > > > @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct > > > drm_crtc *crtc, u32 base, > > > struct drm_device *dev = crtc->dev; > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_crtc *intel_crtc = to_intel_crtc(crtc); > > > + struct intel_crtc_state *cstate = > > > to_intel_crtc_state(crtc->state); > > > const struct skl_wm_values *wm = &dev_priv- > > > >wm.skl_results; > > > + const struct skl_plane_wm *p_wm = > > > + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; > > > int pipe = intel_crtc->pipe; > > > uint32_t cntl = 0; > > > > > > if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & > > > drm_crtc_mask(crtc)) > > > - skl_write_cursor_wm(intel_crtc, wm); > > > + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); > > > > > > if (plane_state && plane_state->base.visible) { > > > cntl = MCURSOR_GAMMA_ENABLE; > > > diff --git a/drivers/gpu/drm/i915/intel_drv.h > > > b/drivers/gpu/drm/i915/intel_drv.h > > > index d684f4f..958dc72 100644 > > > --- a/drivers/gpu/drm/i915/intel_drv.h > > > +++ b/drivers/gpu/drm/i915/intel_drv.h > > > @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const struct > > > skl_ddb_allocation *old, > > > bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, > > > struct intel_crtc *intel_crtc); > > > void skl_write_cursor_wm(struct intel_crtc *intel_crtc, > > > - const struct skl_wm_values *wm); > > > + const struct skl_plane_wm *wm, > > > + const struct skl_ddb_allocation *ddb); > > > void skl_write_plane_wm(struct intel_crtc *intel_crtc, > > > - const struct skl_wm_values *wm, > > > + const struct skl_plane_wm *wm, > > > + const struct skl_ddb_allocation *ddb, > > > int plane); > > > uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state > > > *pipe_config); > > > bool ilk_disable_lp_wm(struct drm_device *dev); > > > diff --git a/drivers/gpu/drm/i915/intel_pm.c > > > b/drivers/gpu/drm/i915/intel_pm.c > > > index 5dbaf12..5cb537c 100644 > > > --- a/drivers/gpu/drm/i915/intel_pm.c > > > +++ b/drivers/gpu/drm/i915/intel_pm.c > > > @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct > > > drm_atomic_state *state) > > > struct drm_i915_private *dev_priv = to_i915(dev); > > > struct intel_atomic_state *intel_state = > > > to_intel_atomic_state(state); > > > struct drm_crtc *crtc; > > > + struct intel_crtc_state *cstate; > > > + struct skl_plane_wm *wm; > > > enum pipe pipe; > > > int level, plane; > > > > > > @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct > > > drm_atomic_state *state) > > > /* Since we're now guaranteed to only have one active > > > CRTC... */ > > > pipe = ffs(intel_state->active_crtcs) - 1; > > > crtc = dev_priv->pipe_to_crtc_mapping[pipe]; > > > + cstate = intel_atomic_get_crtc_state(state, > > > to_intel_crtc(crtc)); > > Wrong cstate here, should be crtc->state since it's committed and > > crtc_state is the old state. > > Ah... How many instances of the "you're looking at the wrong state > struct" have we spotted/fixed ever since the introduction of the atomic > code? Can't we try to do something more human-proof in order to avoid > these problems from happening again and again? > > Like for example copying the old states to a specific structure like > drm_atomic_state->old_states_which_you_probably_shouldn_be_using, and > then setting the CRTC/connector/plane states in the drm_atomic_state > struct to NULL so bugs will be much easier to spot? Maarten has patches somewhere to create explicit old/new state pointers in drm_atomic_state. Not sure where exactly they're stuck. -Daniel
Op 17-10-16 om 08:05 schreef Daniel Vetter: > On Thu, Oct 13, 2016 at 05:04:03PM -0300, Paulo Zanoni wrote: >> Em Qui, 2016-10-13 às 15:39 +0200, Maarten Lankhorst escreveu: >>> Op 08-10-16 om 02:11 schreef Lyude: >>>> Now that we've make skl_wm_levels make a little more sense, we can >>>> remove all of the redundant wm information. Up until now we'd been >>>> storing two copies of all of the skl watermarks: one being the >>>> skl_pipe_wm structs, the other being the global wm struct in >>>> drm_i915_private containing the raw register values. This is >>>> confusing >>>> and problematic, since it means we're prone to accidentally letting >>>> the >>>> two copies go out of sync. So, get rid of all of the functions >>>> responsible for computing the register values and just use a single >>>> helper, skl_write_wm_level(), to convert and write the new >>>> watermarks on >>>> the fly. >>>> >>>> Changes since v1: >>>> - Fixup skl_write_wm_level() >>>> - Fixup skl_wm_level_from_reg_val() >>>> - Don't forget to copy *active to intel_crtc->wm.active.skl >>>> >>>> Signed-off-by: Lyude <cpaul@redhat.com> >>>> Reviewed-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com> >>>> Cc: Ville Syrjälä <ville.syrjala@linux.intel.com> >>>> Cc: Paulo Zanoni <paulo.r.zanoni@intel.com> >>>> --- >>>> drivers/gpu/drm/i915/i915_drv.h | 2 - >>>> drivers/gpu/drm/i915/intel_display.c | 14 ++- >>>> drivers/gpu/drm/i915/intel_drv.h | 6 +- >>>> drivers/gpu/drm/i915/intel_pm.c | 204 ++++++++++++----------- >>>> ------------ >>>> drivers/gpu/drm/i915/intel_sprite.c | 8 +- >>>> 5 files changed, 90 insertions(+), 144 deletions(-) >>>> >>>> diff --git a/drivers/gpu/drm/i915/i915_drv.h >>>> b/drivers/gpu/drm/i915/i915_drv.h >>>> index 0287c93..76583b2 100644 >>>> --- a/drivers/gpu/drm/i915/i915_drv.h >>>> +++ b/drivers/gpu/drm/i915/i915_drv.h >>>> @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { >>>> struct skl_wm_values { >>>> unsigned dirty_pipes; >>>> struct skl_ddb_allocation ddb; >>>> - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; >>>> - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; >>>> }; >>>> >>>> struct skl_wm_level { >>>> diff --git a/drivers/gpu/drm/i915/intel_display.c >>>> b/drivers/gpu/drm/i915/intel_display.c >>>> index a71d05a..39400a0 100644 >>>> --- a/drivers/gpu/drm/i915/intel_display.c >>>> +++ b/drivers/gpu/drm/i915/intel_display.c >>>> @@ -3378,6 +3378,8 @@ static void >>>> skylake_update_primary_plane(struct drm_plane *plane, >>>> struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state- >>>>> base.crtc); >>>> struct drm_framebuffer *fb = plane_state->base.fb; >>>> const struct skl_wm_values *wm = &dev_priv- >>>>> wm.skl_results; >>>> + const struct skl_plane_wm *p_wm = >>>> + &crtc_state->wm.skl.optimal.planes[0]; >>>> int pipe = intel_crtc->pipe; >>>> u32 plane_ctl; >>>> unsigned int rotation = plane_state->base.rotation; >>>> @@ -3414,7 +3416,7 @@ static void >>>> skylake_update_primary_plane(struct drm_plane *plane, >>>> intel_crtc->adjusted_y = src_y; >>>> >>>> if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) >>>> - skl_write_plane_wm(intel_crtc, wm, 0); >>>> + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, 0); >>>> >>>> I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); >>>> I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | src_x); >>>> @@ -3448,6 +3450,8 @@ static void >>>> skylake_disable_primary_plane(struct drm_plane *primary, >>>> struct drm_device *dev = crtc->dev; >>>> struct drm_i915_private *dev_priv = to_i915(dev); >>>> struct intel_crtc *intel_crtc = to_intel_crtc(crtc); >>>> + struct intel_crtc_state *cstate = >>>> to_intel_crtc_state(crtc->state); >>>> + const struct skl_plane_wm *p_wm = &cstate- >>>>> wm.skl.optimal.planes[0]; >>>> int pipe = intel_crtc->pipe; >>>> >>>> /* >>>> @@ -3455,7 +3459,8 @@ static void >>>> skylake_disable_primary_plane(struct drm_plane *primary, >>>> * plane's visiblity isn't actually changing neither is >>>> its watermarks. >>>> */ >>>> if (!crtc->primary->state->visible) >>>> - skl_write_plane_wm(intel_crtc, &dev_priv- >>>>> wm.skl_results, 0); >>>> + skl_write_plane_wm(intel_crtc, p_wm, >>>> + &dev_priv->wm.skl_results.ddb, >>>> 0); >>>> >>>> I915_WRITE(PLANE_CTL(pipe, 0), 0); >>>> I915_WRITE(PLANE_SURF(pipe, 0), 0); >>>> @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct >>>> drm_crtc *crtc, u32 base, >>>> struct drm_device *dev = crtc->dev; >>>> struct drm_i915_private *dev_priv = to_i915(dev); >>>> struct intel_crtc *intel_crtc = to_intel_crtc(crtc); >>>> + struct intel_crtc_state *cstate = >>>> to_intel_crtc_state(crtc->state); >>>> const struct skl_wm_values *wm = &dev_priv- >>>>> wm.skl_results; >>>> + const struct skl_plane_wm *p_wm = >>>> + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; >>>> int pipe = intel_crtc->pipe; >>>> uint32_t cntl = 0; >>>> >>>> if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & >>>> drm_crtc_mask(crtc)) >>>> - skl_write_cursor_wm(intel_crtc, wm); >>>> + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); >>>> >>>> if (plane_state && plane_state->base.visible) { >>>> cntl = MCURSOR_GAMMA_ENABLE; >>>> diff --git a/drivers/gpu/drm/i915/intel_drv.h >>>> b/drivers/gpu/drm/i915/intel_drv.h >>>> index d684f4f..958dc72 100644 >>>> --- a/drivers/gpu/drm/i915/intel_drv.h >>>> +++ b/drivers/gpu/drm/i915/intel_drv.h >>>> @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const struct >>>> skl_ddb_allocation *old, >>>> bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, >>>> struct intel_crtc *intel_crtc); >>>> void skl_write_cursor_wm(struct intel_crtc *intel_crtc, >>>> - const struct skl_wm_values *wm); >>>> + const struct skl_plane_wm *wm, >>>> + const struct skl_ddb_allocation *ddb); >>>> void skl_write_plane_wm(struct intel_crtc *intel_crtc, >>>> - const struct skl_wm_values *wm, >>>> + const struct skl_plane_wm *wm, >>>> + const struct skl_ddb_allocation *ddb, >>>> int plane); >>>> uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state >>>> *pipe_config); >>>> bool ilk_disable_lp_wm(struct drm_device *dev); >>>> diff --git a/drivers/gpu/drm/i915/intel_pm.c >>>> b/drivers/gpu/drm/i915/intel_pm.c >>>> index 5dbaf12..5cb537c 100644 >>>> --- a/drivers/gpu/drm/i915/intel_pm.c >>>> +++ b/drivers/gpu/drm/i915/intel_pm.c >>>> @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct >>>> drm_atomic_state *state) >>>> struct drm_i915_private *dev_priv = to_i915(dev); >>>> struct intel_atomic_state *intel_state = >>>> to_intel_atomic_state(state); >>>> struct drm_crtc *crtc; >>>> + struct intel_crtc_state *cstate; >>>> + struct skl_plane_wm *wm; >>>> enum pipe pipe; >>>> int level, plane; >>>> >>>> @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct >>>> drm_atomic_state *state) >>>> /* Since we're now guaranteed to only have one active >>>> CRTC... */ >>>> pipe = ffs(intel_state->active_crtcs) - 1; >>>> crtc = dev_priv->pipe_to_crtc_mapping[pipe]; >>>> + cstate = intel_atomic_get_crtc_state(state, >>>> to_intel_crtc(crtc)); >>> Wrong cstate here, should be crtc->state since it's committed and >>> crtc_state is the old state. >> Ah... How many instances of the "you're looking at the wrong state >> struct" have we spotted/fixed ever since the introduction of the atomic >> code? Can't we try to do something more human-proof in order to avoid >> these problems from happening again and again? >> >> Like for example copying the old states to a specific structure like >> drm_atomic_state->old_states_which_you_probably_shouldn_be_using, and >> then setting the CRTC/connector/plane states in the drm_atomic_state >> struct to NULL so bugs will be much easier to spot? > Maarten has patches somewhere to create explicit old/new state pointers in > drm_atomic_state. Not sure where exactly they're stuck. > -Daniel Waiting for drm-misc merge window to reopen, but making things more explicit will help a lot. ~Maarten
diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h index 0287c93..76583b2 100644 --- a/drivers/gpu/drm/i915/i915_drv.h +++ b/drivers/gpu/drm/i915/i915_drv.h @@ -1644,8 +1644,6 @@ struct skl_ddb_allocation { struct skl_wm_values { unsigned dirty_pipes; struct skl_ddb_allocation ddb; - uint32_t plane[I915_MAX_PIPES][I915_MAX_PLANES][8]; - uint32_t plane_trans[I915_MAX_PIPES][I915_MAX_PLANES]; }; struct skl_wm_level { diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index a71d05a..39400a0 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c @@ -3378,6 +3378,8 @@ static void skylake_update_primary_plane(struct drm_plane *plane, struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state->base.crtc); struct drm_framebuffer *fb = plane_state->base.fb; const struct skl_wm_values *wm = &dev_priv->wm.skl_results; + const struct skl_plane_wm *p_wm = + &crtc_state->wm.skl.optimal.planes[0]; int pipe = intel_crtc->pipe; u32 plane_ctl; unsigned int rotation = plane_state->base.rotation; @@ -3414,7 +3416,7 @@ static void skylake_update_primary_plane(struct drm_plane *plane, intel_crtc->adjusted_y = src_y; if (wm->dirty_pipes & drm_crtc_mask(&intel_crtc->base)) - skl_write_plane_wm(intel_crtc, wm, 0); + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, 0); I915_WRITE(PLANE_CTL(pipe, 0), plane_ctl); I915_WRITE(PLANE_OFFSET(pipe, 0), (src_y << 16) | src_x); @@ -3448,6 +3450,8 @@ static void skylake_disable_primary_plane(struct drm_plane *primary, struct drm_device *dev = crtc->dev; struct drm_i915_private *dev_priv = to_i915(dev); struct intel_crtc *intel_crtc = to_intel_crtc(crtc); + struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); + const struct skl_plane_wm *p_wm = &cstate->wm.skl.optimal.planes[0]; int pipe = intel_crtc->pipe; /* @@ -3455,7 +3459,8 @@ static void skylake_disable_primary_plane(struct drm_plane *primary, * plane's visiblity isn't actually changing neither is its watermarks. */ if (!crtc->primary->state->visible) - skl_write_plane_wm(intel_crtc, &dev_priv->wm.skl_results, 0); + skl_write_plane_wm(intel_crtc, p_wm, + &dev_priv->wm.skl_results.ddb, 0); I915_WRITE(PLANE_CTL(pipe, 0), 0); I915_WRITE(PLANE_SURF(pipe, 0), 0); @@ -10819,12 +10824,15 @@ static void i9xx_update_cursor(struct drm_crtc *crtc, u32 base, struct drm_device *dev = crtc->dev; struct drm_i915_private *dev_priv = to_i915(dev); struct intel_crtc *intel_crtc = to_intel_crtc(crtc); + struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); const struct skl_wm_values *wm = &dev_priv->wm.skl_results; + const struct skl_plane_wm *p_wm = + &cstate->wm.skl.optimal.planes[PLANE_CURSOR]; int pipe = intel_crtc->pipe; uint32_t cntl = 0; if (INTEL_GEN(dev_priv) >= 9 && wm->dirty_pipes & drm_crtc_mask(crtc)) - skl_write_cursor_wm(intel_crtc, wm); + skl_write_cursor_wm(intel_crtc, p_wm, &wm->ddb); if (plane_state && plane_state->base.visible) { cntl = MCURSOR_GAMMA_ENABLE; diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h index d684f4f..958dc72 100644 --- a/drivers/gpu/drm/i915/intel_drv.h +++ b/drivers/gpu/drm/i915/intel_drv.h @@ -1765,9 +1765,11 @@ bool skl_ddb_allocation_equals(const struct skl_ddb_allocation *old, bool skl_ddb_allocation_overlaps(struct drm_atomic_state *state, struct intel_crtc *intel_crtc); void skl_write_cursor_wm(struct intel_crtc *intel_crtc, - const struct skl_wm_values *wm); + const struct skl_plane_wm *wm, + const struct skl_ddb_allocation *ddb); void skl_write_plane_wm(struct intel_crtc *intel_crtc, - const struct skl_wm_values *wm, + const struct skl_plane_wm *wm, + const struct skl_ddb_allocation *ddb, int plane); uint32_t ilk_pipe_pixel_rate(const struct intel_crtc_state *pipe_config); bool ilk_disable_lp_wm(struct drm_device *dev); diff --git a/drivers/gpu/drm/i915/intel_pm.c b/drivers/gpu/drm/i915/intel_pm.c index 5dbaf12..5cb537c 100644 --- a/drivers/gpu/drm/i915/intel_pm.c +++ b/drivers/gpu/drm/i915/intel_pm.c @@ -3000,6 +3000,8 @@ bool intel_can_enable_sagv(struct drm_atomic_state *state) struct drm_i915_private *dev_priv = to_i915(dev); struct intel_atomic_state *intel_state = to_intel_atomic_state(state); struct drm_crtc *crtc; + struct intel_crtc_state *cstate; + struct skl_plane_wm *wm; enum pipe pipe; int level, plane; @@ -3020,18 +3022,21 @@ bool intel_can_enable_sagv(struct drm_atomic_state *state) /* Since we're now guaranteed to only have one active CRTC... */ pipe = ffs(intel_state->active_crtcs) - 1; crtc = dev_priv->pipe_to_crtc_mapping[pipe]; + cstate = intel_atomic_get_crtc_state(state, to_intel_crtc(crtc)); if (crtc->state->mode.flags & DRM_MODE_FLAG_INTERLACE) return false; for_each_plane(dev_priv, pipe, plane) { + wm = &cstate->wm.skl.optimal.planes[plane]; + /* Skip this plane if it's not enabled */ - if (intel_state->wm_results.plane[pipe][plane][0] == 0) + if (!wm->wm[0].plane_en) continue; /* Find the highest enabled wm level for this plane */ - for (level = ilk_wm_max_level(dev); - intel_state->wm_results.plane[pipe][plane][level] == 0; --level) + for (level = ilk_wm_max_level(dev); !wm->wm[level].plane_en; + --level) { } /* @@ -3779,67 +3784,6 @@ static int skl_build_pipe_wm(struct intel_crtc_state *cstate, return 0; } -static void skl_compute_wm_results(struct drm_device *dev, - struct skl_pipe_wm *p_wm, - struct skl_wm_values *r, - struct intel_crtc *intel_crtc) -{ - int level, max_level = ilk_wm_max_level(dev); - struct skl_plane_wm *plane_wm; - enum pipe pipe = intel_crtc->pipe; - uint32_t temp; - int i; - - for (i = 0; i < intel_num_planes(intel_crtc); i++) { - plane_wm = &p_wm->planes[i]; - - for (level = 0; level <= max_level; level++) { - temp = 0; - - temp |= plane_wm->wm[level].plane_res_l << - PLANE_WM_LINES_SHIFT; - temp |= plane_wm->wm[level].plane_res_b; - if (plane_wm->wm[level].plane_en) - temp |= PLANE_WM_EN; - - r->plane[pipe][i][level] = temp; - } - - } - - for (level = 0; level <= max_level; level++) { - plane_wm = &p_wm->planes[PLANE_CURSOR]; - temp = 0; - temp |= plane_wm->wm[level].plane_res_l << PLANE_WM_LINES_SHIFT; - temp |= plane_wm->wm[level].plane_res_b; - if (plane_wm->wm[level].plane_en) - temp |= PLANE_WM_EN; - - r->plane[pipe][PLANE_CURSOR][level] = temp; - } - - /* transition WMs */ - for (i = 0; i < intel_num_planes(intel_crtc); i++) { - plane_wm = &p_wm->planes[i]; - temp = 0; - temp |= plane_wm->trans_wm.plane_res_l << PLANE_WM_LINES_SHIFT; - temp |= plane_wm->trans_wm.plane_res_b; - if (plane_wm->trans_wm.plane_en) - temp |= PLANE_WM_EN; - - r->plane_trans[pipe][i] = temp; - } - - plane_wm = &p_wm->planes[PLANE_CURSOR]; - temp = 0; - temp |= plane_wm->trans_wm.plane_res_l << PLANE_WM_LINES_SHIFT; - temp |= plane_wm->trans_wm.plane_res_b; - if (plane_wm->trans_wm.plane_en) - temp |= PLANE_WM_EN; - - r->plane_trans[pipe][PLANE_CURSOR] = temp; -} - static void skl_ddb_entry_write(struct drm_i915_private *dev_priv, i915_reg_t reg, const struct skl_ddb_entry *entry) @@ -3850,8 +3794,24 @@ static void skl_ddb_entry_write(struct drm_i915_private *dev_priv, I915_WRITE(reg, 0); } +static void skl_write_wm_level(struct drm_i915_private *dev_priv, + i915_reg_t reg, + const struct skl_wm_level *level) +{ + uint32_t val = 0; + + if (level->plane_en) { + val |= PLANE_WM_EN; + val |= level->plane_res_b; + val |= level->plane_res_l << PLANE_WM_LINES_SHIFT; + } + + I915_WRITE(reg, val); +} + void skl_write_plane_wm(struct intel_crtc *intel_crtc, - const struct skl_wm_values *wm, + const struct skl_plane_wm *wm, + const struct skl_ddb_allocation *ddb, int plane) { struct drm_crtc *crtc = &intel_crtc->base; @@ -3861,19 +3821,21 @@ void skl_write_plane_wm(struct intel_crtc *intel_crtc, enum pipe pipe = intel_crtc->pipe; for (level = 0; level <= max_level; level++) { - I915_WRITE(PLANE_WM(pipe, plane, level), - wm->plane[pipe][plane][level]); + skl_write_wm_level(dev_priv, PLANE_WM(pipe, plane, level), + &wm->wm[level]); } - I915_WRITE(PLANE_WM_TRANS(pipe, plane), wm->plane_trans[pipe][plane]); + skl_write_wm_level(dev_priv, PLANE_WM_TRANS(pipe, plane), + &wm->trans_wm); skl_ddb_entry_write(dev_priv, PLANE_BUF_CFG(pipe, plane), - &wm->ddb.plane[pipe][plane]); + &ddb->plane[pipe][plane]); skl_ddb_entry_write(dev_priv, PLANE_NV12_BUF_CFG(pipe, plane), - &wm->ddb.y_plane[pipe][plane]); + &ddb->y_plane[pipe][plane]); } void skl_write_cursor_wm(struct intel_crtc *intel_crtc, - const struct skl_wm_values *wm) + const struct skl_plane_wm *wm, + const struct skl_ddb_allocation *ddb) { struct drm_crtc *crtc = &intel_crtc->base; struct drm_device *dev = crtc->dev; @@ -3882,13 +3844,13 @@ void skl_write_cursor_wm(struct intel_crtc *intel_crtc, enum pipe pipe = intel_crtc->pipe; for (level = 0; level <= max_level; level++) { - I915_WRITE(CUR_WM(pipe, level), - wm->plane[pipe][PLANE_CURSOR][level]); + skl_write_wm_level(dev_priv, CUR_WM(pipe, level), + &wm->wm[level]); } - I915_WRITE(CUR_WM_TRANS(pipe), wm->plane_trans[pipe][PLANE_CURSOR]); + skl_write_wm_level(dev_priv, CUR_WM_TRANS(pipe), &wm->trans_wm); skl_ddb_entry_write(dev_priv, CUR_BUF_CFG(pipe), - &wm->ddb.plane[pipe][PLANE_CURSOR]); + &ddb->plane[pipe][PLANE_CURSOR]); } static inline bool skl_ddb_entries_overlap(const struct skl_ddb_entry *a, @@ -4072,11 +4034,6 @@ skl_copy_wm_for_pipe(struct skl_wm_values *dst, struct skl_wm_values *src, enum pipe pipe) { - memcpy(dst->plane[pipe], src->plane[pipe], - sizeof(dst->plane[pipe])); - memcpy(dst->plane_trans[pipe], src->plane_trans[pipe], - sizeof(dst->plane_trans[pipe])); - memcpy(dst->ddb.y_plane[pipe], src->ddb.y_plane[pipe], sizeof(dst->ddb.y_plane[pipe])); memcpy(dst->ddb.plane[pipe], src->ddb.plane[pipe], @@ -4125,7 +4082,6 @@ skl_compute_wm(struct drm_atomic_state *state) * no suitable watermark values can be found. */ for_each_crtc_in_state(state, crtc, cstate, i) { - struct intel_crtc *intel_crtc = to_intel_crtc(crtc); struct intel_crtc_state *intel_cstate = to_intel_crtc_state(cstate); @@ -4143,7 +4099,6 @@ skl_compute_wm(struct drm_atomic_state *state) continue; intel_cstate->update_wm_pre = true; - skl_compute_wm_results(crtc->dev, pipe_wm, results, intel_crtc); } return 0; @@ -4177,9 +4132,11 @@ static void skl_update_wm(struct drm_crtc *crtc) int plane; for (plane = 0; plane < intel_num_planes(intel_crtc); plane++) - skl_write_plane_wm(intel_crtc, results, plane); + skl_write_plane_wm(intel_crtc, &pipe_wm->planes[plane], + &results->ddb, plane); - skl_write_cursor_wm(intel_crtc, results); + skl_write_cursor_wm(intel_crtc, &pipe_wm->planes[PLANE_CURSOR], + &results->ddb); } skl_copy_wm_for_pipe(hw_vals, results, pipe); @@ -4264,26 +4221,13 @@ static void ilk_optimize_watermarks(struct intel_crtc_state *cstate) mutex_unlock(&dev_priv->wm.wm_mutex); } -static void skl_pipe_wm_active_state(uint32_t val, - struct skl_pipe_wm *active, - bool is_transwm, - int i, - int level) +static inline void skl_wm_level_from_reg_val(uint32_t val, + struct skl_wm_level *level) { - struct skl_plane_wm *plane_wm = &active->planes[i]; - bool is_enabled = (val & PLANE_WM_EN) != 0; - - if (!is_transwm) { - plane_wm->wm[level].plane_en = is_enabled; - plane_wm->wm[level].plane_res_b = val & PLANE_WM_BLOCKS_MASK; - plane_wm->wm[level].plane_res_l = - (val >> PLANE_WM_LINES_SHIFT) & PLANE_WM_LINES_MASK; - } else { - plane_wm->trans_wm.plane_en = is_enabled; - plane_wm->trans_wm.plane_res_b = val & PLANE_WM_BLOCKS_MASK; - plane_wm->trans_wm.plane_res_l = - (val >> PLANE_WM_LINES_SHIFT) & PLANE_WM_LINES_MASK; - } + level->plane_en = val & PLANE_WM_EN; + level->plane_res_b = val & PLANE_WM_BLOCKS_MASK; + level->plane_res_l = (val >> PLANE_WM_LINES_SHIFT) & + PLANE_WM_LINES_MASK; } static void skl_pipe_wm_get_hw_state(struct drm_crtc *crtc) @@ -4293,49 +4237,39 @@ static void skl_pipe_wm_get_hw_state(struct drm_crtc *crtc) struct skl_wm_values *hw = &dev_priv->wm.skl_hw; struct intel_crtc *intel_crtc = to_intel_crtc(crtc); struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); + struct intel_plane *intel_plane; struct skl_pipe_wm *active = &cstate->wm.skl.optimal; + struct skl_plane_wm *wm; enum pipe pipe = intel_crtc->pipe; - int level, i, max_level; - uint32_t temp; + int level, id, max_level = ilk_wm_max_level(dev); + uint32_t val; - max_level = ilk_wm_max_level(dev); + for_each_intel_plane_on_crtc(dev, intel_crtc, intel_plane) { + id = skl_wm_plane_id(intel_plane); + wm = &cstate->wm.skl.optimal.planes[id]; - for (level = 0; level <= max_level; level++) { - for (i = 0; i < intel_num_planes(intel_crtc); i++) - hw->plane[pipe][i][level] = - I915_READ(PLANE_WM(pipe, i, level)); - hw->plane[pipe][PLANE_CURSOR][level] = I915_READ(CUR_WM(pipe, level)); - } + for (level = 0; level <= max_level; level++) { + if (id != PLANE_CURSOR) + val = I915_READ(PLANE_WM(pipe, id, level)); + else + val = I915_READ(CUR_WM(pipe, level)); - for (i = 0; i < intel_num_planes(intel_crtc); i++) - hw->plane_trans[pipe][i] = I915_READ(PLANE_WM_TRANS(pipe, i)); - hw->plane_trans[pipe][PLANE_CURSOR] = I915_READ(CUR_WM_TRANS(pipe)); + skl_wm_level_from_reg_val(val, &wm->wm[level]); + } + + if (id != PLANE_CURSOR) + val = I915_READ(PLANE_WM_TRANS(pipe, id)); + else + val = I915_READ(CUR_WM_TRANS(pipe)); + + skl_wm_level_from_reg_val(val, &wm->trans_wm); + } if (!intel_crtc->active) return; hw->dirty_pipes |= drm_crtc_mask(crtc); - active->linetime = I915_READ(PIPE_WM_LINETIME(pipe)); - - for (level = 0; level <= max_level; level++) { - for (i = 0; i < intel_num_planes(intel_crtc); i++) { - temp = hw->plane[pipe][i][level]; - skl_pipe_wm_active_state(temp, active, false, i, level); - } - temp = hw->plane[pipe][PLANE_CURSOR][level]; - skl_pipe_wm_active_state(temp, active, false, PLANE_CURSOR, - level); - } - - for (i = 0; i < intel_num_planes(intel_crtc); i++) { - temp = hw->plane_trans[pipe][i]; - skl_pipe_wm_active_state(temp, active, true, i, 0); - } - - temp = hw->plane_trans[pipe][PLANE_CURSOR]; - skl_pipe_wm_active_state(temp, active, true, PLANE_CURSOR, 0); - intel_crtc->wm.active.skl = *active; } diff --git a/drivers/gpu/drm/i915/intel_sprite.c b/drivers/gpu/drm/i915/intel_sprite.c index 73a521f..0fb775b 100644 --- a/drivers/gpu/drm/i915/intel_sprite.c +++ b/drivers/gpu/drm/i915/intel_sprite.c @@ -208,6 +208,8 @@ skl_update_plane(struct drm_plane *drm_plane, struct intel_crtc *intel_crtc = to_intel_crtc(crtc); const int pipe = intel_plane->pipe; const int plane = intel_plane->plane + 1; + const struct skl_plane_wm *p_wm = + &crtc_state->wm.skl.optimal.planes[plane]; u32 plane_ctl; const struct drm_intel_sprite_colorkey *key = &plane_state->ckey; u32 surf_addr = plane_state->main.offset; @@ -232,7 +234,7 @@ skl_update_plane(struct drm_plane *drm_plane, plane_ctl |= skl_plane_ctl_rotation(rotation); if (wm->dirty_pipes & drm_crtc_mask(crtc)) - skl_write_plane_wm(intel_crtc, wm, plane); + skl_write_plane_wm(intel_crtc, p_wm, &wm->ddb, plane); if (key->flags) { I915_WRITE(PLANE_KEYVAL(pipe, plane), key->min_value); @@ -289,6 +291,7 @@ skl_disable_plane(struct drm_plane *dplane, struct drm_crtc *crtc) struct drm_device *dev = dplane->dev; struct drm_i915_private *dev_priv = to_i915(dev); struct intel_plane *intel_plane = to_intel_plane(dplane); + struct intel_crtc_state *cstate = to_intel_crtc_state(crtc->state); const int pipe = intel_plane->pipe; const int plane = intel_plane->plane + 1; @@ -298,7 +301,8 @@ skl_disable_plane(struct drm_plane *dplane, struct drm_crtc *crtc) */ if (!dplane->state->visible) skl_write_plane_wm(to_intel_crtc(crtc), - &dev_priv->wm.skl_results, plane); + &cstate->wm.skl.optimal.planes[plane], + &dev_priv->wm.skl_results.ddb, plane); I915_WRITE(PLANE_CTL(pipe, plane), 0);