On 5/20/26 18:04, Dmitry Osipenko wrote:
> On 5/19/26 11:22, Deepanshu Kartikey wrote:
>> virtio_gpu_cursor_plane_update() and virtio_gpu_resource_flush() lock
>> the framebuffer BO's dma_resv via virtio_gpu_array_lock_resv() and
>> ignore its return value. The function can fail with -EINTR from
>> dma_resv_lock_interruptible() (signal during lock wait) or with
>> -ENOMEM from dma_resv_reserve_fences() (fence slot allocation),
>> leaving the resv lock not held. The queue path then walks the object
>> array and calls dma_resv_add_fence(), which requires the lock held;
>> with lockdep enabled this trips dma_resv_assert_held():
>>
>>   WARNING: drivers/dma-buf/dma-resv.c:296 at dma_resv_add_fence+0x71e/0x840
>>   Call Trace:
>>    virtio_gpu_array_add_fence
>>    virtio_gpu_queue_ctrl_sgs
>>    virtio_gpu_queue_fenced_ctrl_buffer
>>    virtio_gpu_cursor_plane_update
>>    drm_atomic_helper_commit_planes
>>    drm_atomic_helper_commit_tail
>>    commit_tail
>>    drm_atomic_helper_commit
>>    drm_atomic_commit
>>    drm_atomic_helper_update_plane
>>    __setplane_atomic
>>    drm_mode_cursor_universal
>>    drm_mode_cursor_common
>>    drm_mode_cursor_ioctl
>>    drm_ioctl
>>    __x64_sys_ioctl
>>
>> Beyond the WARN, mutating the dma_resv fence list without the lock
>> races with concurrent readers/writers and can corrupt the list.
>>
>> Both call sites run inside the .atomic_update plane callback, which
>> DRM atomic helpers do not allow to fail (by the time it runs, the
>> commit has been signed off to userspace and there is no clean
>> rollback path). Moving the lock acquisition to .prepare_fb was
>> rejected because the broader lock scope deadlocks against other BO
>> locking paths in the same atomic commit.
>>
>> Introduce virtio_gpu_lock_one_resv_uninterruptible() that uses
>> dma_resv_lock() instead of dma_resv_lock_interruptible(). This
>> eliminates the -EINTR failure mode -- the realistic syzbot trigger
>> -- without extending the lock hold across the commit. The helper
>> locks a single BO and rejects nents > 1 with -EINVAL; both fix
>> sites lock exactly one BO.
>>
>> Use it from virtio_gpu_cursor_plane_update() and
>> virtio_gpu_resource_flush(); check the return value to handle the
>> remaining -ENOMEM case from dma_resv_reserve_fences() by freeing
>> the objs and skipping the plane update for that frame. The
>> framebuffer BOs touched here are not shared with other contexts
>> and lock contention is expected to be brief, so the loss of
>> signal-interruptibility is acceptable.
>>
>> Other callers of virtio_gpu_array_lock_resv() (the ioctl paths)
>> continue to use the interruptible variant.
>>
>> The bug was reported by syzbot, triggered via fault injection
>> (fail_nth) on the DRM_IOCTL_MODE_CURSOR path, which forces the
>> -ENOMEM branch in dma_resv_reserve_fences().
>>
>> Reported-by: [email protected]
>> Closes: https://syzkaller.appspot.com/bug?extid=72bd3dd3a5d5f39a0271
>> Fixes: 5cfd31c5b3a3 ("drm/virtio: fix virtio_gpu_cursor_plane_update().")
>> Cc: [email protected]
>> Signed-off-by: Deepanshu Kartikey <[email protected]>
>> ---
>> v4: Rename the helper to virtio_gpu_lock_one_resv_uninterruptible()
>>     and reject objs->nents > 1 with -EINVAL. The v3 helper's
>>     multi-object branch used drm_gem_lock_reservations(), which is
>>     interruptible, contradicting the "uninterruptible" name; both
>>     fix sites lock a single BO so the multi-object path is dropped.
>>     (Dmitry Osipenko)
>> v3: Drop the prepare_fb/cleanup_fb approach from v2 (it deadlocked
>>     against virtio_gpu_resource_flush(), which also locks the BO in
>>     the same atomic commit). Instead add an uninterruptible variant
>>     of the resv lock helper and use it in both
>>     virtio_gpu_cursor_plane_update() and virtio_gpu_resource_flush().
>>     (Dmitry Osipenko)
>> v2: Move resv lock acquisition from .atomic_update (which must not
>>     fail) to .prepare_fb (which may), per maintainer review of v1.
>>     The v1 approach of silently skipping the cursor update on lock
>>     failure violated the atomic-commit contract with userspace.
>> ---
>>  drivers/gpu/drm/virtio/virtgpu_drv.h   |  1 +
>>  drivers/gpu/drm/virtio/virtgpu_gem.c   | 17 +++++++++++++++++
>>  drivers/gpu/drm/virtio/virtgpu_plane.c | 10 ++++++++--
>>  3 files changed, 26 insertions(+), 2 deletions(-)
>>
>> diff --git a/drivers/gpu/drm/virtio/virtgpu_drv.h 
>> b/drivers/gpu/drm/virtio/virtgpu_drv.h
>> index f17660a71a3e..2f3531950aa4 100644
>> --- a/drivers/gpu/drm/virtio/virtgpu_drv.h
>> +++ b/drivers/gpu/drm/virtio/virtgpu_drv.h
>> @@ -317,6 +317,7 @@ virtio_gpu_array_from_handles(struct drm_file *drm_file, 
>> u32 *handles, u32 nents
>>  void virtio_gpu_array_add_obj(struct virtio_gpu_object_array *objs,
>>                            struct drm_gem_object *obj);
>>  int virtio_gpu_array_lock_resv(struct virtio_gpu_object_array *objs);
>> +int virtio_gpu_lock_one_resv_uninterruptible(struct virtio_gpu_object_array 
>> *objs);
>>  void virtio_gpu_array_unlock_resv(struct virtio_gpu_object_array *objs);
>>  void virtio_gpu_array_add_fence(struct virtio_gpu_object_array *objs,
>>                              struct dma_fence *fence);
>> diff --git a/drivers/gpu/drm/virtio/virtgpu_gem.c 
>> b/drivers/gpu/drm/virtio/virtgpu_gem.c
>> index f22dc5c21cd4..435d37d36034 100644
>> --- a/drivers/gpu/drm/virtio/virtgpu_gem.c
>> +++ b/drivers/gpu/drm/virtio/virtgpu_gem.c
>> @@ -238,6 +238,23 @@ int virtio_gpu_array_lock_resv(struct 
>> virtio_gpu_object_array *objs)
>>      return ret;
>>  }
>>  
>> +int virtio_gpu_lock_one_resv_uninterruptible(struct virtio_gpu_object_array 
>> *objs)
>> +{
>> +    int ret;
>> +
>> +    if (objs->nents != 1)
>> +            return -EINVAL;
>> +
>> +    dma_resv_lock(objs->objs[0]->resv, NULL);
>> +
>> +    ret = dma_resv_reserve_fences(objs->objs[0]->resv, 1);
>> +    if (ret) {
>> +            virtio_gpu_array_unlock_resv(objs);
>> +            return ret;
>> +    }
>> +    return 0;
>> +}
>> +
>>  void virtio_gpu_array_unlock_resv(struct virtio_gpu_object_array *objs)
>>  {
>>      if (objs->nents == 1) {
>> diff --git a/drivers/gpu/drm/virtio/virtgpu_plane.c 
>> b/drivers/gpu/drm/virtio/virtgpu_plane.c
>> index a126d1b25f46..652352424744 100644
>> --- a/drivers/gpu/drm/virtio/virtgpu_plane.c
>> +++ b/drivers/gpu/drm/virtio/virtgpu_plane.c
>> @@ -215,7 +215,10 @@ static void virtio_gpu_resource_flush(struct drm_plane 
>> *plane,
>>              if (!objs)
>>                      return;
>>              virtio_gpu_array_add_obj(objs, vgfb->base.obj[0]);
>> -            virtio_gpu_array_lock_resv(objs);
>> +            if (virtio_gpu_lock_one_resv_uninterruptible(objs)) {
>> +                    virtio_gpu_array_put_free(objs);
>> +                    return;
>> +            }
>>              virtio_gpu_cmd_resource_flush(vgdev, bo->hw_res_handle, x, y,
>>                                            width, height, objs,
>>                                            vgplane_st->fence);
>> @@ -459,7 +462,10 @@ static void virtio_gpu_cursor_plane_update(struct 
>> drm_plane *plane,
>>              if (!objs)
>>                      return;
>>              virtio_gpu_array_add_obj(objs, vgfb->base.obj[0]);
>> -            virtio_gpu_array_lock_resv(objs);
>> +            if (virtio_gpu_lock_one_resv_uninterruptible(objs)) {
>> +                    virtio_gpu_array_put_free(objs);
>> +                    return;
>> +            }
>>              virtio_gpu_cmd_transfer_to_host_2d
>>                      (vgdev, 0,
>>                       plane->state->crtc_w,
> 
> Applied to misc-next, thanks

Realized patche should go to -fixes, applied to misc-fixes too

-- 
Best regards,
Dmitry

Reply via email to