diff options
Diffstat (limited to 'drivers/gpu/drm/drm_debugfs_crc.c')
| -rw-r--r-- | drivers/gpu/drm/drm_debugfs_crc.c | 59 | 
1 files changed, 44 insertions, 15 deletions
| diff --git a/drivers/gpu/drm/drm_debugfs_crc.c b/drivers/gpu/drm/drm_debugfs_crc.c index 1722d8f21449..f9e26dda56d6 100644 --- a/drivers/gpu/drm/drm_debugfs_crc.c +++ b/drivers/gpu/drm/drm_debugfs_crc.c @@ -136,21 +136,51 @@ static int crtc_crc_data_count(struct drm_crtc_crc *crc)  	return CIRC_CNT(crc->head, crc->tail, DRM_CRC_ENTRIES_NR);  } +static void crtc_crc_cleanup(struct drm_crtc_crc *crc) +{ +	kfree(crc->entries); +	crc->entries = NULL; +	crc->head = 0; +	crc->tail = 0; +	crc->values_cnt = 0; +	crc->opened = false; +} +  static int crtc_crc_open(struct inode *inode, struct file *filep)  {  	struct drm_crtc *crtc = inode->i_private;  	struct drm_crtc_crc *crc = &crtc->crc;  	struct drm_crtc_crc_entry *entries = NULL;  	size_t values_cnt; -	int ret; +	int ret = 0; -	if (crc->opened) -		return -EBUSY; +	if (drm_drv_uses_atomic_modeset(crtc->dev)) { +		ret = drm_modeset_lock_interruptible(&crtc->mutex, NULL); +		if (ret) +			return ret; + +		if (!crtc->state->active) +			ret = -EIO; +		drm_modeset_unlock(&crtc->mutex); + +		if (ret) +			return ret; +	} + +	spin_lock_irq(&crc->lock); +	if (!crc->opened) +		crc->opened = true; +	else +		ret = -EBUSY; +	spin_unlock_irq(&crc->lock); -	ret = crtc->funcs->set_crc_source(crtc, crc->source, &values_cnt);  	if (ret)  		return ret; +	ret = crtc->funcs->set_crc_source(crtc, crc->source, &values_cnt); +	if (ret) +		goto err; +  	if (WARN_ON(values_cnt > DRM_MAX_CRC_NR)) {  		ret = -EINVAL;  		goto err_disable; @@ -170,7 +200,6 @@ static int crtc_crc_open(struct inode *inode, struct file *filep)  	spin_lock_irq(&crc->lock);  	crc->entries = entries;  	crc->values_cnt = values_cnt; -	crc->opened = true;  	/*  	 * Only return once we got a first frame, so userspace doesn't have to @@ -182,12 +211,17 @@ static int crtc_crc_open(struct inode *inode, struct file *filep)  						crc->lock);  	spin_unlock_irq(&crc->lock); -	WARN_ON(ret); +	if (ret) +		goto err_disable;  	return 0;  err_disable:  	crtc->funcs->set_crc_source(crtc, NULL, &values_cnt); +err: +	spin_lock_irq(&crc->lock); +	crtc_crc_cleanup(crc); +	spin_unlock_irq(&crc->lock);  	return ret;  } @@ -197,17 +231,12 @@ static int crtc_crc_release(struct inode *inode, struct file *filep)  	struct drm_crtc_crc *crc = &crtc->crc;  	size_t values_cnt; +	crtc->funcs->set_crc_source(crtc, NULL, &values_cnt); +  	spin_lock_irq(&crc->lock); -	kfree(crc->entries); -	crc->entries = NULL; -	crc->head = 0; -	crc->tail = 0; -	crc->values_cnt = 0; -	crc->opened = false; +	crtc_crc_cleanup(crc);  	spin_unlock_irq(&crc->lock); -	crtc->funcs->set_crc_source(crtc, NULL, &values_cnt); -  	return 0;  } @@ -334,7 +363,7 @@ int drm_crtc_add_crc_entry(struct drm_crtc *crtc, bool has_frame,  	spin_lock(&crc->lock);  	/* Caller may not have noticed yet that userspace has stopped reading */ -	if (!crc->opened) { +	if (!crc->entries) {  		spin_unlock(&crc->lock);  		return -EINVAL;  	} | 
