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;
+       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;
        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
                                                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;
 }
 
        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;
 }
 
        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;
        }