]> www.infradead.org Git - users/jedix/linux-maple.git/commitdiff
drm/msm: Refcount submits
authorRob Clark <robdclark@chromium.org>
Fri, 23 Oct 2020 16:51:17 +0000 (09:51 -0700)
committerRob Clark <robdclark@chromium.org>
Thu, 5 Nov 2020 00:00:56 +0000 (16:00 -0800)
Before we remove dev->struct_mutex from the retire path, we have to deal
with the situation of a submit retiring before the submit ioctl returns.

To deal with this, ring->submits will hold a reference to the submit,
which is dropped when the submit is retired.  And the submit ioctl path
holds it's own ref, which it drops when it is done with the submit.

Also, add to submit list *after* getting/pinning bo's, to prevent badness
in case the completed fence is corrupted, and retire_worker mistakenly
believes the submit is done too early.

Signed-off-by: Rob Clark <robdclark@chromium.org>
Reviewed-by: Jordan Crouse <jcrouse@codeaurora.org>
Reviewed-by: Kristian H. Kristensen <hoegsberg@google.com>
Signed-off-by: Rob Clark <robdclark@chromium.org>
drivers/gpu/drm/msm/msm_drv.h
drivers/gpu/drm/msm/msm_gem.h
drivers/gpu/drm/msm/msm_gem_submit.c
drivers/gpu/drm/msm/msm_gpu.c

index e3a8fae8c9f6c3f05d17a2fcb93a5ac452ec0641..c1212e021dd46111d57fa0cbe01e8853fd0c089e 100644 (file)
@@ -278,7 +278,6 @@ void msm_unregister_mmu(struct drm_device *dev, struct msm_mmu *mmu);
 
 bool msm_use_mmu(struct drm_device *dev);
 
-void msm_gem_submit_free(struct msm_gem_submit *submit);
 int msm_ioctl_gem_submit(struct drm_device *dev, void *data,
                struct drm_file *file);
 
index f0608d96ef037d1c4089db8e017ff6282bcc1a07..2f289c436dddb4b7a770d974fd20491a3e591b7f 100644 (file)
@@ -213,6 +213,7 @@ void msm_gem_free_work(struct work_struct *work);
  * lasts for the duration of the submit-ioctl.
  */
 struct msm_gem_submit {
+       struct kref ref;
        struct drm_device *dev;
        struct msm_gpu *gpu;
        struct msm_gem_address_space *aspace;
@@ -249,6 +250,18 @@ struct msm_gem_submit {
        } bos[];
 };
 
+void __msm_gem_submit_destroy(struct kref *kref);
+
+static inline void msm_gem_submit_get(struct msm_gem_submit *submit)
+{
+       kref_get(&submit->ref);
+}
+
+static inline void msm_gem_submit_put(struct msm_gem_submit *submit)
+{
+       kref_put(&submit->ref, __msm_gem_submit_destroy);
+}
+
 /* helper to determine of a buffer in submit should be dumped, used for both
  * devcoredump and debugfs cmdstream dumping:
  */
index 24ce4c65429d4514aa1697de5cf1f017de9c1717..d04c349d8112aaa18fc1642d62ddc225437dfeae 100644 (file)
@@ -42,6 +42,7 @@ static struct msm_gem_submit *submit_create(struct drm_device *dev,
        if (!submit)
                return NULL;
 
+       kref_init(&submit->ref);
        submit->dev = dev;
        submit->aspace = queue->ctx->aspace;
        submit->gpu = gpu;
@@ -60,14 +61,13 @@ static struct msm_gem_submit *submit_create(struct drm_device *dev,
        return submit;
 }
 
-void msm_gem_submit_free(struct msm_gem_submit *submit)
+void __msm_gem_submit_destroy(struct kref *kref)
 {
+       struct msm_gem_submit *submit =
+                       container_of(kref, struct msm_gem_submit, ref);
        unsigned i;
 
        dma_fence_put(submit->fence);
-       spin_lock(&submit->ring->submit_lock);
-       list_del(&submit->node);
-       spin_unlock(&submit->ring->submit_lock);
        put_pid(submit->pid);
        msm_submitqueue_put(submit->queue);
 
@@ -847,8 +847,7 @@ out_pre_pm:
        submit_cleanup(submit);
        if (has_ww_ticket)
                ww_acquire_fini(&submit->ticket);
-       if (ret)
-               msm_gem_submit_free(submit);
+       msm_gem_submit_put(submit);
 out_unlock:
        if (ret && (out_fence_fd >= 0))
                put_unused_fd(out_fence_fd);
index 6e6b170b9e5d1a5b427187a94101c52663d5c99c..a69f6cb388380987f70d7f28eda7aa241bec2b6c 100644 (file)
@@ -718,7 +718,12 @@ static void retire_submit(struct msm_gpu *gpu, struct msm_ringbuffer *ring,
 
        pm_runtime_mark_last_busy(&gpu->pdev->dev);
        pm_runtime_put_autosuspend(&gpu->pdev->dev);
-       msm_gem_submit_free(submit);
+
+       spin_lock(&ring->submit_lock);
+       list_del(&submit->node);
+       spin_unlock(&ring->submit_lock);
+
+       msm_gem_submit_put(submit);
 }
 
 static void retire_submits(struct msm_gpu *gpu)
@@ -791,10 +796,6 @@ void msm_gpu_submit(struct msm_gpu *gpu, struct msm_gem_submit *submit)
 
        submit->seqno = ++ring->seqno;
 
-       spin_lock(&ring->submit_lock);
-       list_add_tail(&submit->node, &ring->submits);
-       spin_unlock(&ring->submit_lock);
-
        msm_rd_dump_submit(priv->rd, submit, NULL);
 
        update_sw_cntrs(gpu);
@@ -821,6 +822,16 @@ void msm_gpu_submit(struct msm_gpu *gpu, struct msm_gem_submit *submit)
                msm_gem_active_get(drm_obj, gpu);
        }
 
+       /*
+        * ring->submits holds a ref to the submit, to deal with the case
+        * that a submit completes before msm_ioctl_gem_submit() returns.
+        */
+       msm_gem_submit_get(submit);
+
+       spin_lock(&ring->submit_lock);
+       list_add_tail(&submit->node, &ring->submits);
+       spin_unlock(&ring->submit_lock);
+
        gpu->funcs->submit(gpu, submit);
        priv->lastctx = submit->queue->ctx;