]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
drm/amdgpu: Fix eviction fence worker race during fd close
authorJesse.Zhang <Jesse.Zhang@amd.com>
Thu, 15 May 2025 07:02:13 +0000 (15:02 +0800)
committerAlex Deucher <alexander.deucher@amd.com>
Thu, 22 May 2025 16:00:52 +0000 (12:00 -0400)
The current cleanup order during file descriptor close can lead to
a race condition where the eviction fence worker attempts to access
a destroyed mutex from the user queue manager:

[  517.294055] DEBUG_LOCKS_WARN_ON(lock->magic != lock)
[  517.294060] WARNING: CPU: 8 PID: 2030 at kernel/locking/mutex.c:564
[  517.294094] Workqueue: events amdgpu_eviction_fence_suspend_worker [amdgpu]

The issue occurs because:
1. We destroy the user queue manager (including its mutex) first
2. Then try to destroy eviction fences which may have pending work
3. The eviction fence worker may try to access the already-destroyed mutex

Fix this by reordering the cleanup to:
1. First mark the fd as closing and destroy eviction fences,
   which flushes any pending work
2. Then safely destroy the user queue manager after we're certain
   no more fence work will be executed

The copy in amdgpu_driver_postclose_kms() needs to be removed (Christian)

Reviewed-by: Christian König <christian.koenig@amd.com>
Reviewed-by: Prike Liang <Prike.Liang@amd.com>
Reviewed-by: Arvind Yadav <Arvind.Yadav@amd.com>
Signed-off-by: Jesse Zhang <Jesse.Zhang@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/amdgpu/amdgpu_drv.c
drivers/gpu/drm/amd/amdgpu/amdgpu_kms.c

index 4ddd08ce888523498755da25f4f6fcea2c6ff588..4db92e0a60da7b37ad0be30fbc4c6f0d30836153 100644 (file)
@@ -2913,8 +2913,8 @@ static int amdgpu_drm_release(struct inode *inode, struct file *filp)
 
        if (fpriv) {
                fpriv->evf_mgr.fd_closing = true;
-               amdgpu_userq_mgr_fini(&fpriv->userq_mgr);
                amdgpu_eviction_fence_destroy(&fpriv->evf_mgr);
+               amdgpu_userq_mgr_fini(&fpriv->userq_mgr);
        }
 
        return drm_release(inode, filp);
index 9fbb04aee97bcd580146ec39abcda0b20eb1eee6..d2ce7d86dbc8e9ce97f5125e14095cfebced4b7f 100644 (file)
@@ -1502,11 +1502,6 @@ void amdgpu_driver_postclose_kms(struct drm_device *dev,
                amdgpu_bo_unreserve(pd);
        }
 
-       if (!fpriv->evf_mgr.fd_closing) {
-               fpriv->evf_mgr.fd_closing = true;
-               amdgpu_userq_mgr_fini(&fpriv->userq_mgr);
-               amdgpu_eviction_fence_destroy(&fpriv->evf_mgr);
-       }
        amdgpu_ctx_mgr_fini(&fpriv->ctx_mgr);
        amdgpu_vm_fini(adev, &fpriv->vm);