]> www.infradead.org Git - users/jedix/linux-maple.git/commitdiff
accel/ivpu: Create priority based command queues
authorWachowski, Karol <karol.wachowski@intel.com>
Mon, 13 May 2024 12:04:22 +0000 (14:04 +0200)
committerJacek Lawrynowicz <jacek.lawrynowicz@linux.intel.com>
Wed, 15 May 2024 05:42:15 +0000 (07:42 +0200)
Create multiple command queues per engine with different priorities.
The cmdqs are created on-demand and they support 4 priority levels.
These priorities will later be used by the HWS (hardware scheduler).

Signed-off-by: Wachowski, Karol <karol.wachowski@intel.com>
Signed-off-by: Jacek Lawrynowicz <jacek.lawrynowicz@linux.intel.com>
Reviewed-by: Jeffrey Hugo <quic_jhugo@quicinc.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20240513120431.3187212-4-jacek.lawrynowicz@linux.intel.com
drivers/accel/ivpu/ivpu_drv.h
drivers/accel/ivpu/ivpu_job.c

index 71b87455e22b0bf2f4a71f5bb314c8ae9c2090d2..aafc5c3e9041e65d847babef35664e6745847c4c 100644 (file)
 #define IVPU_MIN_DB 1
 #define IVPU_MAX_DB 255
 
-#define IVPU_NUM_ENGINES 2
+#define IVPU_NUM_ENGINES       2
+#define IVPU_NUM_PRIORITIES    4
+#define IVPU_NUM_CMDQS_PER_CTX (IVPU_NUM_ENGINES * IVPU_NUM_PRIORITIES)
+
+#define IVPU_CMDQ_INDEX(engine, priority) ((engine) * IVPU_NUM_PRIORITIES + (priority))
 
 #define IVPU_PLATFORM_SILICON 0
 #define IVPU_PLATFORM_SIMICS  2
@@ -149,7 +153,7 @@ struct ivpu_file_priv {
        struct kref ref;
        struct ivpu_device *vdev;
        struct mutex lock; /* Protects cmdq */
-       struct ivpu_cmdq *cmdq[IVPU_NUM_ENGINES];
+       struct ivpu_cmdq *cmdq[IVPU_NUM_CMDQS_PER_CTX];
        struct ivpu_mmu_context ctx;
        bool has_mmu_faults;
        bool bound;
index a49bc9105ed0c98fce6571737096977c6fc4d51b..b56035de1a597b4c511e6bad09cefa8490c612db 100644 (file)
@@ -79,10 +79,12 @@ static void ivpu_cmdq_free(struct ivpu_file_priv *file_priv, struct ivpu_cmdq *c
        kfree(cmdq);
 }
 
-static struct ivpu_cmdq *ivpu_cmdq_acquire(struct ivpu_file_priv *file_priv, u16 engine)
+static struct ivpu_cmdq *ivpu_cmdq_acquire(struct ivpu_file_priv *file_priv, u16 engine,
+                                          u8 priority)
 {
+       int cmdq_idx = IVPU_CMDQ_INDEX(engine, priority);
+       struct ivpu_cmdq *cmdq = file_priv->cmdq[cmdq_idx];
        struct ivpu_device *vdev = file_priv->vdev;
-       struct ivpu_cmdq *cmdq = file_priv->cmdq[engine];
        int ret;
 
        lockdep_assert_held(&file_priv->lock);
@@ -91,7 +93,7 @@ static struct ivpu_cmdq *ivpu_cmdq_acquire(struct ivpu_file_priv *file_priv, u16
                cmdq = ivpu_cmdq_alloc(file_priv, engine);
                if (!cmdq)
                        return NULL;
-               file_priv->cmdq[engine] = cmdq;
+               file_priv->cmdq[cmdq_idx] = cmdq;
        }
 
        if (cmdq->db_registered)
@@ -107,14 +109,15 @@ static struct ivpu_cmdq *ivpu_cmdq_acquire(struct ivpu_file_priv *file_priv, u16
        return cmdq;
 }
 
-static void ivpu_cmdq_release_locked(struct ivpu_file_priv *file_priv, u16 engine)
+static void ivpu_cmdq_release_locked(struct ivpu_file_priv *file_priv, u16 engine, u8 priority)
 {
-       struct ivpu_cmdq *cmdq = file_priv->cmdq[engine];
+       int cmdq_idx = IVPU_CMDQ_INDEX(engine, priority);
+       struct ivpu_cmdq *cmdq = file_priv->cmdq[cmdq_idx];
 
        lockdep_assert_held(&file_priv->lock);
 
        if (cmdq) {
-               file_priv->cmdq[engine] = NULL;
+               file_priv->cmdq[cmdq_idx] = NULL;
                if (cmdq->db_registered)
                        ivpu_jsm_unregister_db(file_priv->vdev, cmdq->db_id);
 
@@ -124,12 +127,14 @@ static void ivpu_cmdq_release_locked(struct ivpu_file_priv *file_priv, u16 engin
 
 void ivpu_cmdq_release_all_locked(struct ivpu_file_priv *file_priv)
 {
-       int i;
+       u16 engine;
+       u8 priority;
 
        lockdep_assert_held(&file_priv->lock);
 
-       for (i = 0; i < IVPU_NUM_ENGINES; i++)
-               ivpu_cmdq_release_locked(file_priv, i);
+       for (engine = 0; engine < IVPU_NUM_ENGINES; engine++)
+               for (priority = 0; priority < IVPU_NUM_PRIORITIES; priority++)
+                       ivpu_cmdq_release_locked(file_priv, engine, priority);
 }
 
 /*
@@ -138,9 +143,10 @@ void ivpu_cmdq_release_all_locked(struct ivpu_file_priv *file_priv)
  * and FW loses job queue state. The next time job queue is used it
  * will be registered again.
  */
-static void ivpu_cmdq_reset_locked(struct ivpu_file_priv *file_priv, u16 engine)
+static void ivpu_cmdq_reset_locked(struct ivpu_file_priv *file_priv, u16 engine, u8 priority)
 {
-       struct ivpu_cmdq *cmdq = file_priv->cmdq[engine];
+       int cmdq_idx = IVPU_CMDQ_INDEX(engine, priority);
+       struct ivpu_cmdq *cmdq = file_priv->cmdq[cmdq_idx];
 
        lockdep_assert_held(&file_priv->lock);
 
@@ -154,12 +160,14 @@ static void ivpu_cmdq_reset_locked(struct ivpu_file_priv *file_priv, u16 engine)
 
 static void ivpu_cmdq_reset_all(struct ivpu_file_priv *file_priv)
 {
-       int i;
+       u16 engine;
+       u8 priority;
 
        mutex_lock(&file_priv->lock);
 
-       for (i = 0; i < IVPU_NUM_ENGINES; i++)
-               ivpu_cmdq_reset_locked(file_priv, i);
+       for (engine = 0; engine < IVPU_NUM_ENGINES; engine++)
+               for (priority = 0; priority < IVPU_NUM_PRIORITIES; priority++)
+                       ivpu_cmdq_reset_locked(file_priv, engine, priority);
 
        mutex_unlock(&file_priv->lock);
 }
@@ -328,7 +336,7 @@ void ivpu_jobs_abort_all(struct ivpu_device *vdev)
                ivpu_job_signal_and_destroy(vdev, id, DRM_IVPU_JOB_STATUS_ABORTED);
 }
 
-static int ivpu_job_submit(struct ivpu_job *job)
+static int ivpu_job_submit(struct ivpu_job *job, u8 priority)
 {
        struct ivpu_file_priv *file_priv = job->file_priv;
        struct ivpu_device *vdev = job->vdev;
@@ -342,10 +350,10 @@ static int ivpu_job_submit(struct ivpu_job *job)
 
        mutex_lock(&file_priv->lock);
 
-       cmdq = ivpu_cmdq_acquire(job->file_priv, job->engine_idx);
+       cmdq = ivpu_cmdq_acquire(job->file_priv, job->engine_idx, priority);
        if (!cmdq) {
-               ivpu_warn_ratelimited(vdev, "Failed get job queue, ctx %d engine %d\n",
-                                     file_priv->ctx.id, job->engine_idx);
+               ivpu_warn_ratelimited(vdev, "Failed to get job queue, ctx %d engine %d prio %d\n",
+                                     file_priv->ctx.id, job->engine_idx, priority);
                ret = -EINVAL;
                goto err_unlock_file_priv;
        }
@@ -375,8 +383,8 @@ static int ivpu_job_submit(struct ivpu_job *job)
                ivpu_cmdq_ring_db(vdev, cmdq);
        }
 
-       ivpu_dbg(vdev, JOB, "Job submitted: id %3u ctx %2d engine %d addr 0x%llx next %d\n",
-                job->job_id, file_priv->ctx.id, job->engine_idx,
+       ivpu_dbg(vdev, JOB, "Job submitted: id %3u ctx %2d engine %d prio %d addr 0x%llx next %d\n",
+                job->job_id, file_priv->ctx.id, job->engine_idx, priority,
                 job->cmd_buf_vpu_addr, cmdq->jobq->header.tail);
 
        xa_unlock(&vdev->submitted_jobs_xa);
@@ -464,6 +472,14 @@ unlock_reservations:
        return ret;
 }
 
+static inline u8 ivpu_job_to_hws_priority(struct ivpu_file_priv *file_priv, u8 priority)
+{
+       if (priority == DRM_IVPU_JOB_PRIORITY_DEFAULT)
+               return DRM_IVPU_JOB_PRIORITY_NORMAL;
+
+       return priority - 1;
+}
+
 int ivpu_submit_ioctl(struct drm_device *dev, void *data, struct drm_file *file)
 {
        struct ivpu_file_priv *file_priv = file->driver_priv;
@@ -472,6 +488,7 @@ int ivpu_submit_ioctl(struct drm_device *dev, void *data, struct drm_file *file)
        struct ivpu_job *job;
        u32 *buf_handles;
        int idx, ret;
+       u8 priority;
 
        if (params->engine > DRM_IVPU_ENGINE_COPY)
                return -EINVAL;
@@ -525,8 +542,10 @@ int ivpu_submit_ioctl(struct drm_device *dev, void *data, struct drm_file *file)
                goto err_destroy_job;
        }
 
+       priority = ivpu_job_to_hws_priority(file_priv, params->priority);
+
        down_read(&vdev->pm->reset_lock);
-       ret = ivpu_job_submit(job);
+       ret = ivpu_job_submit(job, priority);
        up_read(&vdev->pm->reset_lock);
        if (ret)
                goto err_signal_fence;