if (enabled)
                drm_crtc_vblank_get(&out->crtc);
 
-       spin_lock_irq(&out->lock);
+       mutex_lock(&out->enabled_lock);
        old_enabled = out->composer_enabled;
        out->composer_enabled = enabled;
-       spin_unlock_irq(&out->lock);
+
+       /* the composition wasn't enabled, so unlock the lock to make sure the lock
+        * will be balanced even if we have a failed commit
+        */
+       if (!out->composer_enabled)
+               mutex_unlock(&out->enabled_lock);
 
        if (old_enabled)
                drm_crtc_vblank_put(&out->crtc);
 
        struct drm_crtc *crtc = &output->crtc;
        struct vkms_crtc_state *state;
        u64 ret_overrun;
-       bool ret, fence_cookie;
+       bool ret, fence_cookie, composer_enabled;
 
        fence_cookie = dma_fence_begin_signalling();
 
        if (ret_overrun != 1)
                pr_warn("%s: vblank timer overrun\n", __func__);
 
-       spin_lock(&output->lock);
        ret = drm_crtc_handle_vblank(crtc);
        if (!ret)
                DRM_ERROR("vkms failure on handling vblank");
 
        state = output->composer_state;
-       spin_unlock(&output->lock);
+       composer_enabled = output->composer_enabled;
+       mutex_unlock(&output->enabled_lock);
 
-       if (state && output->composer_enabled) {
+       if (state && composer_enabled) {
                u64 frame = drm_crtc_accurate_vblank_count(crtc);
 
                /* update frame_start only if a queued vkms_composer_worker()
 
        spin_lock_init(&vkms_out->lock);
        spin_lock_init(&vkms_out->composer_lock);
+       mutex_init(&vkms_out->enabled_lock);
 
        vkms_out->composer_workq = alloc_ordered_workqueue("vkms_composer", 0);
        if (!vkms_out->composer_workq)
 
        struct workqueue_struct *composer_workq;
        /* protects concurrent access to composer */
        spinlock_t lock;
+       /* guarantees that if the composer is enabled, a job will be queued */
+       struct mutex enabled_lock;
 
-       /* protected by @lock */
+       /* protected by @enabled_lock */
        bool composer_enabled;
        struct vkms_crtc_state *composer_state;