]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
dma-buf: add dma_fence_was_initialized function v2
authorChristian König <christian.koenig@amd.com>
Fri, 19 Dec 2025 10:41:54 +0000 (11:41 +0100)
committerChristian König <christian.koenig@amd.com>
Thu, 5 Feb 2026 10:02:56 +0000 (11:02 +0100)
Some driver use fence->ops to test if a fence was initialized or not.
The problem is that this utilizes internal behavior of the dma_fence
implementation.

So better abstract that into a function.

v2: use a flag instead of testing fence->ops, rename the function, move
    to the beginning of the patch set.

Signed-off-by: Christian König <christian.koenig@amd.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@igalia.com>
Link: https://lore.kernel.org/r/20260120105655.7134-2-christian.koenig@amd.com
drivers/dma-buf/dma-fence.c
drivers/gpu/drm/amd/amdgpu/amdgpu_job.c
drivers/gpu/drm/qxl/qxl_release.c
include/linux/dma-fence.h

index 21c5c30b4f34f4889533b162604e38cbce88c8b9..c9a036b0d5923dcc884a4c4f0b9007afca8e8087 100644 (file)
@@ -1054,7 +1054,7 @@ __dma_fence_init(struct dma_fence *fence, const struct dma_fence_ops *ops,
        fence->lock = lock;
        fence->context = context;
        fence->seqno = seqno;
-       fence->flags = flags;
+       fence->flags = flags | BIT(DMA_FENCE_FLAG_INITIALIZED_BIT);
        fence->error = 0;
 
        trace_dma_fence_init(fence);
index aaf5477fcd7ac6a0eb08174fbb9a906c8e56967b..f05683d59f8b9ad327663ff15b441746d304b5e8 100644 (file)
@@ -282,9 +282,10 @@ void amdgpu_job_free_resources(struct amdgpu_job *job)
        unsigned i;
 
        /* Check if any fences were initialized */
-       if (job->base.s_fence && job->base.s_fence->finished.ops)
+       if (job->base.s_fence &&
+           dma_fence_was_initialized(&job->base.s_fence->finished))
                f = &job->base.s_fence->finished;
-       else if (job->hw_fence && job->hw_fence->base.ops)
+       else if (dma_fence_was_initialized(&job->hw_fence->base))
                f = &job->hw_fence->base;
        else
                f = NULL;
@@ -301,11 +302,11 @@ static void amdgpu_job_free_cb(struct drm_sched_job *s_job)
 
        amdgpu_sync_free(&job->explicit_sync);
 
-       if (job->hw_fence->base.ops)
+       if (dma_fence_was_initialized(&job->hw_fence->base))
                dma_fence_put(&job->hw_fence->base);
        else
                kfree(job->hw_fence);
-       if (job->hw_vm_fence->base.ops)
+       if (dma_fence_was_initialized(&job->hw_vm_fence->base))
                dma_fence_put(&job->hw_vm_fence->base);
        else
                kfree(job->hw_vm_fence);
@@ -339,11 +340,11 @@ void amdgpu_job_free(struct amdgpu_job *job)
        if (job->gang_submit != &job->base.s_fence->scheduled)
                dma_fence_put(job->gang_submit);
 
-       if (job->hw_fence->base.ops)
+       if (dma_fence_was_initialized(&job->hw_fence->base))
                dma_fence_put(&job->hw_fence->base);
        else
                kfree(job->hw_fence);
-       if (job->hw_vm_fence->base.ops)
+       if (dma_fence_was_initialized(&job->hw_vm_fence->base))
                dma_fence_put(&job->hw_vm_fence->base);
        else
                kfree(job->hw_vm_fence);
index 7b3c9a6016dbb7d9de59cebc91328973717322aa..06b0b2aa7953f71dd195ce395e3bc7bba2784dea 100644 (file)
@@ -146,7 +146,7 @@ qxl_release_free(struct qxl_device *qdev,
        idr_remove(&qdev->release_idr, release->id);
        spin_unlock(&qdev->release_idr_lock);
 
-       if (release->base.ops) {
+       if (dma_fence_was_initialized(&release->base)) {
                WARN_ON(list_empty(&release->bos));
                qxl_release_free_list(release);
 
index d4c92fd350924759a63fbd00d3c8c545caeb9611..9c4d2528923905c522aea0fccd8b7d2f0031be6d 100644 (file)
@@ -48,6 +48,7 @@ struct seq_file;
  * atomic ops (bit_*), so taking the spinlock will not be needed most
  * of the time.
  *
+ * DMA_FENCE_FLAG_INITIALIZED_BIT - fence was initialized
  * DMA_FENCE_FLAG_SIGNALED_BIT - fence is already signaled
  * DMA_FENCE_FLAG_TIMESTAMP_BIT - timestamp recorded for fence signaling
  * DMA_FENCE_FLAG_ENABLE_SIGNAL_BIT - enable_signaling might have been called
@@ -98,6 +99,7 @@ struct dma_fence {
 };
 
 enum dma_fence_flag_bits {
+       DMA_FENCE_FLAG_INITIALIZED_BIT,
        DMA_FENCE_FLAG_SEQNO64_BIT,
        DMA_FENCE_FLAG_SIGNALED_BIT,
        DMA_FENCE_FLAG_TIMESTAMP_BIT,
@@ -263,6 +265,19 @@ void dma_fence_release(struct kref *kref);
 void dma_fence_free(struct dma_fence *fence);
 void dma_fence_describe(struct dma_fence *fence, struct seq_file *seq);
 
+/**
+ * dma_fence_was_initialized - test if fence was initialized
+ * @fence: fence to test
+ *
+ * Return: True if fence was ever initialized, false otherwise. Works correctly
+ * only when memory backing the fence structure is zero initialized on
+ * allocation.
+ */
+static inline bool dma_fence_was_initialized(struct dma_fence *fence)
+{
+       return fence && test_bit(DMA_FENCE_FLAG_INITIALIZED_BIT, &fence->flags);
+}
+
 /**
  * dma_fence_put - decreases refcount of the fence
  * @fence: fence to reduce refcount of