[Intel-gfx] [PATCH 15/23] drm: Stop updating plane->crtc/fb/old_fb on atomic drivers

Daniel Vetter daniel at ffwll.ch
Mon Mar 26 20:56:45 UTC 2018


On Mon, Mar 26, 2018 at 10:52:58PM +0200, Daniel Vetter wrote:
> On Thu, Mar 22, 2018 at 05:23:05PM +0200, Ville Syrjala wrote:
> > From: Ville Syrjälä <ville.syrjala at linux.intel.com>
> > 
> > Stop playing around with plane->crtc/fb/old_fb with atomic
> > drivers. Make life a lot simpler when we don't have to do the
> > magic old_fb vs. fb dance around plane updates. That way we
> > can't risk plane->fb getting out of sync with plane->state->fb
> > and we're less likely to leak any refcounts as well.
> > 
> > Signed-off-by: Ville Syrjälä <ville.syrjala at linux.intel.com>
> > ---
> >  drivers/gpu/drm/drm_atomic.c        | 55 ++++---------------------------------
> >  drivers/gpu/drm/drm_atomic_helper.c | 15 +---------
> >  drivers/gpu/drm/drm_crtc.c          |  8 ++++--
> >  drivers/gpu/drm/drm_fb_helper.c     |  7 -----
> >  drivers/gpu/drm/drm_framebuffer.c   |  5 ----
> >  drivers/gpu/drm/drm_plane.c         | 14 ++++++----
> >  drivers/gpu/drm/drm_plane_helper.c  |  4 ++-
> >  include/drm/drm_atomic.h            |  3 --
> >  8 files changed, 24 insertions(+), 87 deletions(-)
> > 
> > diff --git a/drivers/gpu/drm/drm_atomic.c b/drivers/gpu/drm/drm_atomic.c
> > index 7d25c42f22db..b16cc37e2adf 100644
> > --- a/drivers/gpu/drm/drm_atomic.c
> > +++ b/drivers/gpu/drm/drm_atomic.c
> > @@ -692,6 +692,11 @@ drm_atomic_get_plane_state(struct drm_atomic_state *state,
> >  
> >  	WARN_ON(!state->acquire_ctx);
> >  
> > +	/* the legacy pointers should never be set */
> > +	WARN_ON(plane->fb);
> > +	WARN_ON(plane->old_fb);
> > +	WARN_ON(plane->crtc);
> 
> I did already review all the users for plane->crtc and found:
> 
> armada + shmob: not atomic, should be fine
> 
> But there's also exynos, msm/mdp5, sti and vc4 doing various silly things
> with setting plane->crtc. I think before you can add this WARN_ON we need
> to clean up that cruft (it looks like 100% cargo culting, so should be
> quit).

Ah, follow-up patches take care of most of this. But note that msm sets
plane->crtc in its _init function, so will trip over your WARN_ON here.

And you seem to have missed sti, which looks at plane->crtc instead of
plane->state->crtc (and appropriate locking) in its debugfs code.
-Daniel

> 
> Going to think about the other patches tomorrow.
> -Daniel
> 
> > +
> >  	plane_state = drm_atomic_get_existing_plane_state(state, plane);
> >  	if (plane_state)
> >  		return plane_state;
> > @@ -2021,45 +2026,6 @@ int drm_atomic_set_property(struct drm_atomic_state *state,
> >  	return ret;
> >  }
> >  
> > -/**
> > - * drm_atomic_clean_old_fb -- Unset old_fb pointers and set plane->fb pointers.
> > - *
> > - * @dev: drm device to check.
> > - * @plane_mask: plane mask for planes that were updated.
> > - * @ret: return value, can be -EDEADLK for a retry.
> > - *
> > - * Before doing an update &drm_plane.old_fb is set to &drm_plane.fb, but before
> > - * dropping the locks old_fb needs to be set to NULL and plane->fb updated. This
> > - * is a common operation for each atomic update, so this call is split off as a
> > - * helper.
> > - */
> > -void drm_atomic_clean_old_fb(struct drm_device *dev,
> > -			     unsigned plane_mask,
> > -			     int ret)
> > -{
> > -	struct drm_plane *plane;
> > -
> > -	/* if succeeded, fixup legacy plane crtc/fb ptrs before dropping
> > -	 * locks (ie. while it is still safe to deref plane->state).  We
> > -	 * need to do this here because the driver entry points cannot
> > -	 * distinguish between legacy and atomic ioctls.
> > -	 */
> > -	drm_for_each_plane_mask(plane, dev, plane_mask) {
> > -		if (ret == 0) {
> > -			struct drm_framebuffer *new_fb = plane->state->fb;
> > -			if (new_fb)
> > -				drm_framebuffer_get(new_fb);
> > -			plane->fb = new_fb;
> > -			plane->crtc = plane->state->crtc;
> > -
> > -			if (plane->old_fb)
> > -				drm_framebuffer_put(plane->old_fb);
> > -		}
> > -		plane->old_fb = NULL;
> > -	}
> > -}
> > -EXPORT_SYMBOL(drm_atomic_clean_old_fb);
> > -
> >  /**
> >   * DOC: explicit fencing properties
> >   *
> > @@ -2280,9 +2246,7 @@ int drm_mode_atomic_ioctl(struct drm_device *dev,
> >  	unsigned int copied_objs, copied_props;
> >  	struct drm_atomic_state *state;
> >  	struct drm_modeset_acquire_ctx ctx;
> > -	struct drm_plane *plane;
> >  	struct drm_out_fence_state *fence_state;
> > -	unsigned plane_mask;
> >  	int ret = 0;
> >  	unsigned int i, j, num_fences;
> >  
> > @@ -2322,7 +2286,6 @@ int drm_mode_atomic_ioctl(struct drm_device *dev,
> >  	state->allow_modeset = !!(arg->flags & DRM_MODE_ATOMIC_ALLOW_MODESET);
> >  
> >  retry:
> > -	plane_mask = 0;
> >  	copied_objs = 0;
> >  	copied_props = 0;
> >  	fence_state = NULL;
> > @@ -2393,12 +2356,6 @@ int drm_mode_atomic_ioctl(struct drm_device *dev,
> >  			copied_props++;
> >  		}
> >  
> > -		if (obj->type == DRM_MODE_OBJECT_PLANE && count_props &&
> > -		    !(arg->flags & DRM_MODE_ATOMIC_TEST_ONLY)) {
> > -			plane = obj_to_plane(obj);
> > -			plane_mask |= (1 << drm_plane_index(plane));
> > -			plane->old_fb = plane->fb;
> > -		}
> >  		drm_mode_object_put(obj);
> >  	}
> >  
> > @@ -2419,8 +2376,6 @@ int drm_mode_atomic_ioctl(struct drm_device *dev,
> >  	}
> >  
> >  out:
> > -	drm_atomic_clean_old_fb(dev, plane_mask, ret);
> > -
> >  	complete_crtc_signaling(dev, state, fence_state, num_fences, !ret);
> >  
> >  	if (ret == -EDEADLK) {
> > diff --git a/drivers/gpu/drm/drm_atomic_helper.c b/drivers/gpu/drm/drm_atomic_helper.c
> > index 0e806f070d00..d42d88b97396 100644
> > --- a/drivers/gpu/drm/drm_atomic_helper.c
> > +++ b/drivers/gpu/drm/drm_atomic_helper.c
> > @@ -2892,7 +2892,6 @@ static int __drm_atomic_helper_disable_all(struct drm_device *dev,
> >  	struct drm_plane *plane;
> >  	struct drm_crtc_state *crtc_state;
> >  	struct drm_crtc *crtc;
> > -	unsigned int plane_mask = 0;
> >  	int ret, i;
> >  
> >  	state = drm_atomic_state_alloc(dev);
> > @@ -2935,17 +2934,10 @@ static int __drm_atomic_helper_disable_all(struct drm_device *dev,
> >  			goto free;
> >  
> >  		drm_atomic_set_fb_for_plane(plane_state, NULL);
> > -
> > -		if (clean_old_fbs) {
> > -			plane->old_fb = plane->fb;
> > -			plane_mask |= BIT(drm_plane_index(plane));
> > -		}
> >  	}
> >  
> >  	ret = drm_atomic_commit(state);
> >  free:
> > -	drm_atomic_clean_old_fb(dev, plane_mask, ret);
> > -
> >  	drm_atomic_state_put(state);
> >  	return ret;
> >  }
> > @@ -3106,13 +3098,8 @@ int drm_atomic_helper_commit_duplicated_state(struct drm_atomic_state *state,
> >  
> >  	state->acquire_ctx = ctx;
> >  
> > -	for_each_new_plane_in_state(state, plane, new_plane_state, i) {
> > -		WARN_ON(plane->crtc != new_plane_state->crtc);
> > -		WARN_ON(plane->fb != new_plane_state->fb);
> > -		WARN_ON(plane->old_fb);
> > -
> > +	for_each_new_plane_in_state(state, plane, new_plane_state, i)
> >  		state->planes[i].old_state = plane->state;
> > -	}
> >  
> >  	for_each_new_crtc_in_state(state, crtc, new_crtc_state, i)
> >  		state->crtcs[i].old_state = crtc->state;
> > diff --git a/drivers/gpu/drm/drm_crtc.c b/drivers/gpu/drm/drm_crtc.c
> > index a231dd5dce16..4e3c1a8d118a 100644
> > --- a/drivers/gpu/drm/drm_crtc.c
> > +++ b/drivers/gpu/drm/drm_crtc.c
> > @@ -474,8 +474,12 @@ static int __drm_mode_set_config_internal(struct drm_mode_set *set,
> >  
> >  	ret = crtc->funcs->set_config(set, ctx);
> >  	if (ret == 0) {
> > -		crtc->primary->crtc = fb ? crtc : NULL;
> > -		crtc->primary->fb = fb;
> > +		struct drm_plane *plane = crtc->primary;
> > +
> > +		if (!plane->state) {
> > +			plane->crtc = fb ? crtc : NULL;
> > +			plane->fb = fb;
> > +		}
> >  	}
> >  
> >  	drm_for_each_crtc(tmp, crtc->dev) {
> > diff --git a/drivers/gpu/drm/drm_fb_helper.c b/drivers/gpu/drm/drm_fb_helper.c
> > index 0646b108030b..5639e804a0cd 100644
> > --- a/drivers/gpu/drm/drm_fb_helper.c
> > +++ b/drivers/gpu/drm/drm_fb_helper.c
> > @@ -368,7 +368,6 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper, bool activ
> >  	struct drm_plane *plane;
> >  	struct drm_atomic_state *state;
> >  	int i, ret;
> > -	unsigned int plane_mask;
> >  	struct drm_modeset_acquire_ctx ctx;
> >  
> >  	drm_modeset_acquire_init(&ctx, 0);
> > @@ -381,7 +380,6 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper, bool activ
> >  
> >  	state->acquire_ctx = &ctx;
> >  retry:
> > -	plane_mask = 0;
> >  	drm_for_each_plane(plane, dev) {
> >  		plane_state = drm_atomic_get_plane_state(state, plane);
> >  		if (IS_ERR(plane_state)) {
> > @@ -391,9 +389,6 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper, bool activ
> >  
> >  		plane_state->rotation = DRM_MODE_ROTATE_0;
> >  
> > -		plane->old_fb = plane->fb;
> > -		plane_mask |= 1 << drm_plane_index(plane);
> > -
> >  		/* disable non-primary: */
> >  		if (plane->type == DRM_PLANE_TYPE_PRIMARY)
> >  			continue;
> > @@ -430,8 +425,6 @@ static int restore_fbdev_mode_atomic(struct drm_fb_helper *fb_helper, bool activ
> >  	ret = drm_atomic_commit(state);
> >  
> >  out_state:
> > -	drm_atomic_clean_old_fb(dev, plane_mask, ret);
> > -
> >  	if (ret == -EDEADLK)
> >  		goto backoff;
> >  
> > diff --git a/drivers/gpu/drm/drm_framebuffer.c b/drivers/gpu/drm/drm_framebuffer.c
> > index ad67203de715..421a77c2a4ac 100644
> > --- a/drivers/gpu/drm/drm_framebuffer.c
> > +++ b/drivers/gpu/drm/drm_framebuffer.c
> > @@ -835,8 +835,6 @@ static int atomic_remove_fb(struct drm_framebuffer *fb)
> >  			goto unlock;
> >  
> >  		plane_mask |= BIT(drm_plane_index(plane));
> > -
> > -		plane->old_fb = plane->fb;
> >  	}
> >  
> >  	/* This list is only filled when disable_crtcs is set. */
> > @@ -851,9 +849,6 @@ static int atomic_remove_fb(struct drm_framebuffer *fb)
> >  		ret = drm_atomic_commit(state);
> >  
> >  unlock:
> > -	if (plane_mask)
> > -		drm_atomic_clean_old_fb(dev, plane_mask, ret);
> > -
> >  	if (ret == -EDEADLK) {
> >  		drm_atomic_state_clear(state);
> >  		drm_modeset_backoff(&ctx);
> > diff --git a/drivers/gpu/drm/drm_plane.c b/drivers/gpu/drm/drm_plane.c
> > index 035054455301..143041666096 100644
> > --- a/drivers/gpu/drm/drm_plane.c
> > +++ b/drivers/gpu/drm/drm_plane.c
> > @@ -650,9 +650,11 @@ static int __setplane_internal(struct drm_plane *plane,
> >  					 crtc_x, crtc_y, crtc_w, crtc_h,
> >  					 src_x, src_y, src_w, src_h, ctx);
> >  	if (!ret) {
> > -		plane->crtc = crtc;
> > -		plane->fb = fb;
> > -		drm_framebuffer_get(plane->fb);
> > +		if (!plane->state) {
> > +			plane->crtc = crtc;
> > +			plane->fb = fb;
> > +			drm_framebuffer_get(plane->fb);
> > +		}
> >  	} else {
> >  		plane->old_fb = NULL;
> >  	}
> > @@ -1092,8 +1094,10 @@ int drm_mode_page_flip_ioctl(struct drm_device *dev,
> >  		/* Keep the old fb, don't unref it. */
> >  		plane->old_fb = NULL;
> >  	} else {
> > -		plane->fb = fb;
> > -		drm_framebuffer_get(fb);
> > +		if (!plane->state) {
> > +			plane->fb = fb;
> > +			drm_framebuffer_get(fb);
> > +		}
> >  	}
> >  
> >  out:
> > diff --git a/drivers/gpu/drm/drm_plane_helper.c b/drivers/gpu/drm/drm_plane_helper.c
> > index f88f68161519..2010794943bc 100644
> > --- a/drivers/gpu/drm/drm_plane_helper.c
> > +++ b/drivers/gpu/drm/drm_plane_helper.c
> > @@ -502,6 +502,7 @@ EXPORT_SYMBOL(drm_plane_helper_update);
> >  int drm_plane_helper_disable(struct drm_plane *plane)
> >  {
> >  	struct drm_plane_state *plane_state;
> > +	struct drm_framebuffer *old_fb;
> >  
> >  	/* crtc helpers love to call disable functions for already disabled hw
> >  	 * functions. So cope with that. */
> > @@ -521,8 +522,9 @@ int drm_plane_helper_disable(struct drm_plane *plane)
> >  	plane_state->plane = plane;
> >  
> >  	plane_state->crtc = NULL;
> > +	old_fb = plane_state->fb;
> >  	drm_atomic_set_fb_for_plane(plane_state, NULL);
> >  
> > -	return drm_plane_helper_commit(plane, plane_state, plane->fb);
> > +	return drm_plane_helper_commit(plane, plane_state, old_fb);
> >  }
> >  EXPORT_SYMBOL(drm_plane_helper_disable);
> > diff --git a/include/drm/drm_atomic.h b/include/drm/drm_atomic.h
> > index a57a8aa90ffb..ca461b6cf71f 100644
> > --- a/include/drm/drm_atomic.h
> > +++ b/include/drm/drm_atomic.h
> > @@ -601,9 +601,6 @@ int __must_check
> >  drm_atomic_add_affected_planes(struct drm_atomic_state *state,
> >  			       struct drm_crtc *crtc);
> >  
> > -void
> > -drm_atomic_clean_old_fb(struct drm_device *dev, unsigned plane_mask, int ret);
> > -
> >  int __must_check drm_atomic_check_only(struct drm_atomic_state *state);
> >  int __must_check drm_atomic_commit(struct drm_atomic_state *state);
> >  int __must_check drm_atomic_nonblocking_commit(struct drm_atomic_state *state);
> > -- 
> > 2.16.1
> > 
> > _______________________________________________
> > Intel-gfx mailing list
> > Intel-gfx at lists.freedesktop.org
> > https://lists.freedesktop.org/mailman/listinfo/intel-gfx
> 
> -- 
> Daniel Vetter
> Software Engineer, Intel Corporation
> http://blog.ffwll.ch

-- 
Daniel Vetter
Software Engineer, Intel Corporation
http://blog.ffwll.ch


More information about the dri-devel mailing list