[Intel-gfx] [PATCH v28 4/6] drm/i915: Add TGL+ SAGV support

Lisovskiy, Stanislav stanislav.lisovskiy at intel.com
Tue May 12 13:59:19 UTC 2020


On Tue, May 12, 2020 at 04:50:46PM +0300, Ville Syrjälä wrote:
> On Tue, May 12, 2020 at 04:41:26PM +0300, Lisovskiy, Stanislav wrote:
> > On Tue, May 12, 2020 at 04:38:21PM +0300, Ville Syrjälä wrote:
> > > On Tue, May 12, 2020 at 04:17:34PM +0300, Lisovskiy, Stanislav wrote:
> > > > On Tue, May 12, 2020 at 04:10:46PM +0300, Ville Syrjälä wrote:
> > > > > On Tue, May 12, 2020 at 03:52:27PM +0300, Lisovskiy, Stanislav wrote:
> > > > > > On Tue, May 12, 2020 at 03:03:26PM +0300, Ville Syrjälä wrote:
> > > > > > > On Thu, May 07, 2020 at 05:45:01PM +0300, Stanislav Lisovskiy wrote:
> > > > > > > > Starting from TGL we need to have a separate wm0
> > > > > > > > values for SAGV and non-SAGV which affects
> > > > > > > > how calculations are done.
> > > > > > > > 
> > > > > > > > v2: Remove long lines
> > > > > > > > v3: Removed COLOR_PLANE enum references
> > > > > > > > v4, v5, v6: Fixed rebase conflict
> > > > > > > > v7: - Removed skl_plane_wm_level accessor from skl_allocate_pipe_ddb(Ville)
> > > > > > > >     - Removed sagv_uv_wm0(Ville)
> > > > > > > >     - can_sagv->use_sagv_wm(Ville)
> > > > > > > > 
> > > > > > > > Signed-off-by: Stanislav Lisovskiy <stanislav.lisovskiy at intel.com>
> > > > > > > > ---
> > > > > > > >  drivers/gpu/drm/i915/display/intel_display.c  |   8 +-
> > > > > > > >  .../drm/i915/display/intel_display_types.h    |   2 +
> > > > > > > >  drivers/gpu/drm/i915/intel_pm.c               | 118 +++++++++++++++++-
> > > > > > > >  3 files changed, 121 insertions(+), 7 deletions(-)
> > > > > > > > 
> > > > > > > > diff --git a/drivers/gpu/drm/i915/display/intel_display.c b/drivers/gpu/drm/i915/display/intel_display.c
> > > > > > > > index fd6d63b03489..be5741cb7595 100644
> > > > > > > > --- a/drivers/gpu/drm/i915/display/intel_display.c
> > > > > > > > +++ b/drivers/gpu/drm/i915/display/intel_display.c
> > > > > > > > @@ -13961,7 +13961,9 @@ static void verify_wm_state(struct intel_crtc *crtc,
> > > > > > > >  		/* Watermarks */
> > > > > > > >  		for (level = 0; level <= max_level; level++) {
> > > > > > > >  			if (skl_wm_level_equals(&hw_plane_wm->wm[level],
> > > > > > > > -						&sw_plane_wm->wm[level]))
> > > > > > > > +						&sw_plane_wm->wm[level]) ||
> > > > > > > > +			    (level == 0 && skl_wm_level_equals(&hw_plane_wm->wm[level],
> > > > > > > > +							       &sw_plane_wm->sagv_wm0)))
> > > > > > > >  				continue;
> > > > > > > >  
> > > > > > > >  			drm_err(&dev_priv->drm,
> > > > > > > > @@ -14016,7 +14018,9 @@ static void verify_wm_state(struct intel_crtc *crtc,
> > > > > > > >  		/* Watermarks */
> > > > > > > >  		for (level = 0; level <= max_level; level++) {
> > > > > > > >  			if (skl_wm_level_equals(&hw_plane_wm->wm[level],
> > > > > > > > -						&sw_plane_wm->wm[level]))
> > > > > > > > +						&sw_plane_wm->wm[level]) ||
> > > > > > > > +			    (level == 0 && skl_wm_level_equals(&hw_plane_wm->wm[level],
> > > > > > > > +							       &sw_plane_wm->sagv_wm0)))
> > > > > > > >  				continue;
> > > > > > > >  
> > > > > > > >  			drm_err(&dev_priv->drm,
> > > > > > > > diff --git a/drivers/gpu/drm/i915/display/intel_display_types.h b/drivers/gpu/drm/i915/display/intel_display_types.h
> > > > > > > > index 9488449e4b94..8cede29c9562 100644
> > > > > > > > --- a/drivers/gpu/drm/i915/display/intel_display_types.h
> > > > > > > > +++ b/drivers/gpu/drm/i915/display/intel_display_types.h
> > > > > > > > @@ -688,11 +688,13 @@ struct skl_plane_wm {
> > > > > > > >  	struct skl_wm_level wm[8];
> > > > > > > >  	struct skl_wm_level uv_wm[8];
> > > > > > > >  	struct skl_wm_level trans_wm;
> > > > > > > > +	struct skl_wm_level sagv_wm0;
> > > > > > > >  	bool is_planar;
> > > > > > > >  };
> > > > > > > >  
> > > > > > > >  struct skl_pipe_wm {
> > > > > > > >  	struct skl_plane_wm planes[I915_MAX_PLANES];
> > > > > > > > +	bool use_sagv_wm;
> > > > > > > >  };
> > > > > > > >  
> > > > > > > >  enum vlv_wm_level {
> > > > > > > > diff --git a/drivers/gpu/drm/i915/intel_pm.c b/drivers/gpu/drm/i915/intel_pm.c
> > > > > > > > index db188efee21e..934a686342ad 100644
> > > > > > > > --- a/drivers/gpu/drm/i915/intel_pm.c
> > > > > > > > +++ b/drivers/gpu/drm/i915/intel_pm.c
> > > > > > > > @@ -3863,25 +3863,35 @@ bool intel_can_enable_sagv(struct drm_i915_private *dev_priv,
> > > > > > > >  	return bw_state->pipe_sagv_reject == 0;
> > > > > > > >  }
> > > > > > > >  
> > > > > > > > +static bool
> > > > > > > > +tgl_crtc_can_enable_sagv(const struct intel_crtc_state *crtc_state);
> > > > > > > 
> > > > > > > Just put the function here instead of adding fwd decalrations.
> > > > > > > 
> > > > > > > > +
> > > > > > > >  static int intel_compute_sagv_mask(struct intel_atomic_state *state)
> > > > > > > >  {
> > > > > > > >  	struct drm_i915_private *dev_priv = to_i915(state->base.dev);
> > > > > > > >  	int ret;
> > > > > > > >  	struct intel_crtc *crtc;
> > > > > > > > -	const struct intel_crtc_state *new_crtc_state;
> > > > > > > > +	struct intel_crtc_state *new_crtc_state;
> > > > > > > >  	struct intel_bw_state *new_bw_state = NULL;
> > > > > > > >  	const struct intel_bw_state *old_bw_state = NULL;
> > > > > > > >  	int i;
> > > > > > > >  
> > > > > > > >  	for_each_new_intel_crtc_in_state(state, crtc,
> > > > > > > >  					 new_crtc_state, i) {
> > > > > > > > +		bool can_sagv;
> > > > > > > > +
> > > > > > > >  		new_bw_state = intel_atomic_get_bw_state(state);
> > > > > > > >  		if (IS_ERR(new_bw_state))
> > > > > > > >  			return PTR_ERR(new_bw_state);
> > > > > > > >  
> > > > > > > >  		old_bw_state = intel_atomic_get_old_bw_state(state);
> > > > > > > >  
> > > > > > > > -		if (skl_crtc_can_enable_sagv(new_crtc_state))
> > > > > > > > +		if (INTEL_GEN(dev_priv) >= 12)
> > > > > > > > +			can_sagv = tgl_crtc_can_enable_sagv(new_crtc_state);
> > > > > > > > +		else
> > > > > > > > +			can_sagv = skl_crtc_can_enable_sagv(new_crtc_state);
> > > > > > > > +
> > > > > > > > +		if (can_sagv)
> > > > > > > >  			new_bw_state->pipe_sagv_reject &= ~BIT(crtc->pipe);
> > > > > > > >  		else
> > > > > > > >  			new_bw_state->pipe_sagv_reject |= BIT(crtc->pipe);
> > > > > > > > @@ -3899,6 +3909,24 @@ static int intel_compute_sagv_mask(struct intel_atomic_state *state)
> > > > > > > >  			return ret;
> > > > > > > >  	}
> > > > > > > >  
> > > > > > > > +	for_each_new_intel_crtc_in_state(state, crtc,
> > > > > > > > +					 new_crtc_state, i) {
> > > > > > > > +		struct skl_pipe_wm *pipe_wm = &new_crtc_state->wm.skl.optimal;
> > > > > > > > +
> > > > > > > > +		/*
> > > > > > > > +		 * Due to drm limitation at commit state, when
> > > > > > > > +		 * changes are written the whole atomic state is
> > > > > > > > +		 * zeroed away => which prevents from using it,
> > > > > > > > +		 * so just sticking it into pipe wm state for
> > > > > > > > +		 * keeping it simple - anyway this is related to wm.
> > > > > > > > +		 * Proper way in ideal universe would be of course not
> > > > > > > > +		 * to lose parent atomic state object from child crtc_state,
> > > > > > > > +		 * and stick to OOP programming principles, which had been
> > > > > > > > +		 * scientifically proven to work.
> > > > > > > > +		 */
> > > > > > > 
> > > > > > > More ramblings. Just drop this comment too imo.
> > > > > > 
> > > > > > As I understand what is happening here is rather weird, so I thought 
> > > > > > commenting is good idea.
> > > > > 
> > > > > At least I have no idea what the comment is trying to say.
> > > > > I see nothing weird hapening here, we're just computing the
> > > > > state which is totally standard stuff.
> > > > 
> > > > Well I can remind, this is because there is no way to get parent state
> > > > from crtc_state, because of weird drm atomic behaviour which leaves us
> > > > with NULL parent state. So that we had to stick this boolean to some
> > > > place.
> > > > In normal code universe this wouldn't even be needed if we could
> > > > just get atomic state from crtc_state when we write wm in skl_write_plane_wm.
> > > 
> > > Didn't get that at all from the comment. It talked about zeroing some
> > > whole state which I took as 'memset(state, 0, sizeof(*state))'.
> > > As that is not what's going on I just got confused by the comment.
> > > 
> > > Now that I understand what this is about I think the comment
> > > (if we want to have one) should probably say something more like:
> > > "we store use_sagv_wm in the crtc state rather than relying on
> > >  the bw state since we have no convenient way to get at the
> > >  latter from the plane commit hooks (especially in the legacy
> > >  cursor case)".
> > > 
> > > > 
> > > > However probably OOP principles like parent-child hieararchy is not a thing
> > > > here. That what this comment was trying to say.
> > > > 
> > > > In normal OOP you can't destroy parent object _before_ destroying
> > > > child one.
> > > 
> > > There's no parent-child relationship between the crtc state and atomic
> > > state (which really shouldn't be called a state to begin with, rather
> > > it should be "transaction" or something along those lines).
> > 
> > In practice there is. crtc_state is being aggregated and contained as 
> > part of more general atomic state. That is exactly what parent-child
> > object relation is.
> > If you want to decouple those, this needs to be not aggregation but a reference,
> > i.e atomic state would not contain those state objects, but have a pointer
> > instead, but then you would not be using that container_of magic.
> 
> Pointers is what it has. And once the atomic commit is done the 
> atomic_state (ie. the object used to track the single transaction)
> goes away while the crtc/plane/etc. states remain behind.

If the rest of states are independent there should be sane way
to get those without the atomic state. 

Currently bw_state, cdclk_state and co - all can be retrieved only
using atomic state, which is at some point "gone". 
Also it is actually not even gone, we just zero out a pointer
to it in drm_crtc_state. 

I know why this done as we discussed, however I would 
emphasize that the proper way would be then
to completely decouple from it, so that all required states can
be retrieved without atomic state. Because currently we are
kind of in some "fuzzy" state in between.

Stan

> 
> -- 
> Ville Syrjälä
> Intel


More information about the Intel-gfx mailing list