diff options
Diffstat (limited to 'drivers/gpu/drm/drm_atomic_helper.c')
| -rw-r--r-- | drivers/gpu/drm/drm_atomic_helper.c | 122 | 
1 files changed, 93 insertions, 29 deletions
| diff --git a/drivers/gpu/drm/drm_atomic_helper.c b/drivers/gpu/drm/drm_atomic_helper.c index 130da5195f3b..80be74df7ba6 100644 --- a/drivers/gpu/drm/drm_atomic_helper.c +++ b/drivers/gpu/drm/drm_atomic_helper.c @@ -30,6 +30,7 @@  #include <drm/drm_plane_helper.h>  #include <drm/drm_crtc_helper.h>  #include <drm/drm_atomic_helper.h> +#include <drm/drm_writeback.h>  #include <linux/dma-fence.h>  #include "drm_crtc_helper_internal.h" @@ -120,7 +121,7 @@ static int handle_conflicting_encoders(struct drm_atomic_state *state,  			new_encoder = drm_atomic_helper_best_encoder(connector);  		if (new_encoder) { -			if (encoder_mask & (1 << drm_encoder_index(new_encoder))) { +			if (encoder_mask & drm_encoder_mask(new_encoder)) {  				DRM_DEBUG_ATOMIC("[ENCODER:%d:%s] on [CONNECTOR:%d:%s] already assigned\n",  					new_encoder->base.id, new_encoder->name,  					connector->base.id, connector->name); @@ -128,7 +129,7 @@ static int handle_conflicting_encoders(struct drm_atomic_state *state,  				return -EINVAL;  			} -			encoder_mask |= 1 << drm_encoder_index(new_encoder); +			encoder_mask |= drm_encoder_mask(new_encoder);  		}  	} @@ -154,7 +155,7 @@ static int handle_conflicting_encoders(struct drm_atomic_state *state,  			continue;  		encoder = connector->state->best_encoder; -		if (!encoder || !(encoder_mask & (1 << drm_encoder_index(encoder)))) +		if (!encoder || !(encoder_mask & drm_encoder_mask(encoder)))  			continue;  		if (!disable_conflicting_encoders) { @@ -222,7 +223,7 @@ set_best_encoder(struct drm_atomic_state *state,  			crtc_state = drm_atomic_get_new_crtc_state(state, crtc);  			crtc_state->encoder_mask &= -				~(1 << drm_encoder_index(conn_state->best_encoder)); +				~drm_encoder_mask(conn_state->best_encoder);  		}  	} @@ -233,7 +234,7 @@ set_best_encoder(struct drm_atomic_state *state,  			crtc_state = drm_atomic_get_new_crtc_state(state, crtc);  			crtc_state->encoder_mask |= -				1 << drm_encoder_index(encoder); +				drm_encoder_mask(encoder);  		}  	} @@ -644,7 +645,7 @@ drm_atomic_helper_check_modeset(struct drm_device *dev,  		if (ret)  			return ret; -		connectors_mask += BIT(i); +		connectors_mask |= BIT(i);  	}  	/* @@ -1172,6 +1173,27 @@ void drm_atomic_helper_commit_modeset_disables(struct drm_device *dev,  }  EXPORT_SYMBOL(drm_atomic_helper_commit_modeset_disables); +static void drm_atomic_helper_commit_writebacks(struct drm_device *dev, +						struct drm_atomic_state *old_state) +{ +	struct drm_connector *connector; +	struct drm_connector_state *new_conn_state; +	int i; + +	for_each_new_connector_in_state(old_state, connector, new_conn_state, i) { +		const struct drm_connector_helper_funcs *funcs; + +		funcs = connector->helper_private; +		if (!funcs->atomic_commit) +			continue; + +		if (new_conn_state->writeback_job && new_conn_state->writeback_job->fb) { +			WARN_ON(connector->connector_type != DRM_MODE_CONNECTOR_WRITEBACK); +			funcs->atomic_commit(connector, new_conn_state); +		} +	} +} +  /**   * drm_atomic_helper_commit_modeset_enables - modeset commit to enable outputs   * @dev: DRM device @@ -1251,6 +1273,8 @@ void drm_atomic_helper_commit_modeset_enables(struct drm_device *dev,  		drm_bridge_enable(encoder->bridge);  	} + +	drm_atomic_helper_commit_writebacks(dev, old_state);  }  EXPORT_SYMBOL(drm_atomic_helper_commit_modeset_enables); @@ -1426,6 +1450,8 @@ void drm_atomic_helper_commit_tail(struct drm_atomic_state *old_state)  	drm_atomic_helper_commit_modeset_enables(dev, old_state); +	drm_atomic_helper_fake_vblank(old_state); +  	drm_atomic_helper_commit_hw_done(old_state);  	drm_atomic_helper_wait_for_vblanks(dev, old_state); @@ -1455,6 +1481,8 @@ void drm_atomic_helper_commit_tail_rpm(struct drm_atomic_state *old_state)  	drm_atomic_helper_commit_planes(dev, old_state,  					DRM_PLANE_COMMIT_ACTIVE_ONLY); +	drm_atomic_helper_fake_vblank(old_state); +  	drm_atomic_helper_commit_hw_done(old_state);  	drm_atomic_helper_wait_for_vblanks(dev, old_state); @@ -1510,8 +1538,9 @@ int drm_atomic_helper_async_check(struct drm_device *dev,  {  	struct drm_crtc *crtc;  	struct drm_crtc_state *crtc_state; -	struct drm_plane *plane; -	struct drm_plane_state *old_plane_state, *new_plane_state; +	struct drm_plane *plane = NULL; +	struct drm_plane_state *old_plane_state = NULL; +	struct drm_plane_state *new_plane_state = NULL;  	const struct drm_plane_helper_funcs *funcs;  	int i, n_planes = 0; @@ -1527,7 +1556,8 @@ int drm_atomic_helper_async_check(struct drm_device *dev,  	if (n_planes != 1)  		return -EINVAL; -	if (!new_plane_state->crtc) +	if (!new_plane_state->crtc || +	    old_plane_state->crtc != new_plane_state->crtc)  		return -EINVAL;  	funcs = plane->helper_private; @@ -2030,6 +2060,45 @@ void drm_atomic_helper_wait_for_dependencies(struct drm_atomic_state *old_state)  EXPORT_SYMBOL(drm_atomic_helper_wait_for_dependencies);  /** + * drm_atomic_helper_fake_vblank - fake VBLANK events if needed + * @old_state: atomic state object with old state structures + * + * This function walks all CRTCs and fake VBLANK events on those with + * &drm_crtc_state.no_vblank set to true and &drm_crtc_state.event != NULL. + * The primary use of this function is writeback connectors working in oneshot + * mode and faking VBLANK events. In this case they only fake the VBLANK event + * when a job is queued, and any change to the pipeline that does not touch the + * connector is leading to timeouts when calling + * drm_atomic_helper_wait_for_vblanks() or + * drm_atomic_helper_wait_for_flip_done(). + * + * This is part of the atomic helper support for nonblocking commits, see + * drm_atomic_helper_setup_commit() for an overview. + */ +void drm_atomic_helper_fake_vblank(struct drm_atomic_state *old_state) +{ +	struct drm_crtc_state *new_crtc_state; +	struct drm_crtc *crtc; +	int i; + +	for_each_new_crtc_in_state(old_state, crtc, new_crtc_state, i) { +		unsigned long flags; + +		if (!new_crtc_state->no_vblank) +			continue; + +		spin_lock_irqsave(&old_state->dev->event_lock, flags); +		if (new_crtc_state->event) { +			drm_crtc_send_vblank_event(crtc, +						   new_crtc_state->event); +			new_crtc_state->event = NULL; +		} +		spin_unlock_irqrestore(&old_state->dev->event_lock, flags); +	} +} +EXPORT_SYMBOL(drm_atomic_helper_fake_vblank); + +/**   * drm_atomic_helper_commit_hw_done - setup possible nonblocking commit   * @old_state: atomic state object with old state structures   * @@ -2320,11 +2389,13 @@ drm_atomic_helper_commit_planes_on_crtc(struct drm_crtc_state *old_crtc_state)  	const struct drm_crtc_helper_funcs *crtc_funcs;  	struct drm_crtc *crtc = old_crtc_state->crtc;  	struct drm_atomic_state *old_state = old_crtc_state->state; +	struct drm_crtc_state *new_crtc_state = +		drm_atomic_get_new_crtc_state(old_state, crtc);  	struct drm_plane *plane;  	unsigned plane_mask;  	plane_mask = old_crtc_state->plane_mask; -	plane_mask |= crtc->state->plane_mask; +	plane_mask |= new_crtc_state->plane_mask;  	crtc_funcs = crtc->helper_private;  	if (crtc_funcs && crtc_funcs->atomic_begin) @@ -2333,6 +2404,8 @@ drm_atomic_helper_commit_planes_on_crtc(struct drm_crtc_state *old_crtc_state)  	drm_for_each_plane_mask(plane, crtc->dev, plane_mask) {  		struct drm_plane_state *old_plane_state =  			drm_atomic_get_old_plane_state(old_state, plane); +		struct drm_plane_state *new_plane_state = +			drm_atomic_get_new_plane_state(old_state, plane);  		const struct drm_plane_helper_funcs *plane_funcs;  		plane_funcs = plane->helper_private; @@ -2340,13 +2413,14 @@ drm_atomic_helper_commit_planes_on_crtc(struct drm_crtc_state *old_crtc_state)  		if (!old_plane_state || !plane_funcs)  			continue; -		WARN_ON(plane->state->crtc && plane->state->crtc != crtc); +		WARN_ON(new_plane_state->crtc && +			new_plane_state->crtc != crtc); -		if (drm_atomic_plane_disabling(old_plane_state, plane->state) && +		if (drm_atomic_plane_disabling(old_plane_state, new_plane_state) &&  		    plane_funcs->atomic_disable)  			plane_funcs->atomic_disable(plane, old_plane_state); -		else if (plane->state->crtc || -			 drm_atomic_plane_disabling(old_plane_state, plane->state)) +		else if (new_plane_state->crtc || +			 drm_atomic_plane_disabling(old_plane_state, new_plane_state))  			plane_funcs->atomic_update(plane, old_plane_state);  	} @@ -2795,7 +2869,7 @@ static int update_output_state(struct drm_atomic_state *state,   * resets the "link-status" property to GOOD, to force any link   * re-training. The SETCRTC ioctl does not define whether an update does   * need a full modeset or just a plane update, hence we're allowed to do - * that. See also drm_mode_connector_set_link_status_property(). + * that. See also drm_connector_set_link_status_property().   *   * Returns:   * Returns 0 on success, negative errno numbers on failure. @@ -2914,7 +2988,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 plane_mask = 0;  	int ret, i;  	state = drm_atomic_state_alloc(dev); @@ -2957,17 +3030,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: -	if (plane_mask) -		drm_atomic_clean_old_fb(dev, plane_mask, ret);  	drm_atomic_state_put(state);  	return ret;  } @@ -3129,13 +3195,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; @@ -3660,6 +3721,9 @@ __drm_atomic_helper_connector_duplicate_state(struct drm_connector *connector,  	if (state->crtc)  		drm_connector_get(connector);  	state->commit = NULL; + +	/* Don't copy over a writeback job, they are used only once */ +	state->writeback_job = NULL;  }  EXPORT_SYMBOL(__drm_atomic_helper_connector_duplicate_state); |