kernel/0172-feat-merge-DSP-DSP-NPU-SDK-driver-to-linux.patch

610 lines
20 KiB
Diff

From e312e399b96609f2bb8fd1e8c3f48ef6baccb95b Mon Sep 17 00:00:00 2001
From: denglei <denglei@eswincomputing.com>
Date: Wed, 18 Sep 2024 16:34:12 +0800
Subject: [PATCH 172/222] feat:merge DSP DSP & NPU SDK driver to linux.
Changelogs:
1.feat: dsp drv support multi op at the same time.
2.feat: Support multi operator tasks.
3.feat:dsp fw and drv add debug info.
4.feat:dsp add cfg_clk and aclk low power.
5.fix:dsp origial submit get task err
Signed-off-by: denglei <denglei@eswincomputing.com>
---
drivers/soc/eswin/ai_driver/dsp/dsp_ioctl.c | 251 ++++++++++++++----
.../soc/eswin/ai_driver/dsp/dsp_ioctl_if.h | 2 +
drivers/soc/eswin/ai_driver/dsp/dsp_main.c | 19 +-
drivers/soc/eswin/ai_driver/dsp/dsp_main.h | 3 +
.../soc/eswin/ai_driver/dsp/dsp_platform.c | 20 +-
.../eswin/ai_driver/include/es_dsp_internal.h | 24 ++
.../soc/eswin/ai_driver/include/hetero_ipc.h | 4 +-
7 files changed, 251 insertions(+), 72 deletions(-)
diff --git a/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl.c b/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl.c
index 5bfd429001bb..1edac591f642 100644
--- a/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl.c
+++ b/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl.c
@@ -345,6 +345,12 @@ static int dsp_set_task_req(struct es_dsp *dsp, dsp_request_t *dsp_req,
dsp_req->allow_eval = 1;
dsp_req->flat_virt = (void *)flat;
dsp_req->prio = task->task.priority;
+ if (dsp_req->prio >= DSP_MAX_PRIO) {
+ dsp_err("%s, %d, dsp request prio = %d is err.\n", __func__,
+ __LINE__, dsp_req->prio);
+ dsp_free_flat_mem(dsp, dma_len, (void *)flat, dma_addr);
+ return -EINVAL;
+ }
dsp_req->flat_size = dma_len;
flat->num_buffer = buffer_count;
@@ -399,8 +405,10 @@ static void dsp_async_task_release(struct khandle *handle)
}
dsp = user_req->es_dsp;
user = user_req->user;
+ if(user_req->need_notify) {
+ module_put(THIS_MODULE);
+ }
kfree(user_req);
- module_put(THIS_MODULE);
dsp_debug("%s, done.\n", __func__);
}
@@ -416,7 +424,7 @@ static void dsp_hw_complete_task(struct device *dev, dsp_request_t *req)
dsp = dsp_file->dsp;
spin_lock_irqsave(&dsp_file->async_ll_lock, flags);
- if (dsp_file->h.fd != INVALID_HANDLE_VALUE) {
+ if (dsp_file->h.fd != INVALID_HANDLE_VALUE && async_task->need_notify) {
list_add_tail(&async_task->async_ll,
&dsp_file->async_ll_complete);
spin_unlock_irqrestore(&dsp_file->async_ll_lock, flags);
@@ -434,51 +442,23 @@ static void dsp_hw_complete_task(struct device *dev, dsp_request_t *req)
}
}
-static long dsp_ioctl_submit_tsk_async(struct file *flip,
- dsp_ioctl_task_s __user *arg)
+static struct dsp_user_req_async *dsp_set_task_info(struct dsp_file *dsp_file,
+ dsp_ioctl_task_s *task,
+ bool need_notify)
{
- struct dsp_file *dsp_file = flip->private_data;
- struct es_dsp *dsp = dsp_file->dsp;
- dsp_ioctl_task_s req;
- dsp_ioctl_task_s *task;
dsp_request_t *dsp_req;
struct dsp_user_req_async *user_req;
int ret;
struct dsp_user *user;
u32 buffer_count;
struct dsp_dma_buf **dma_entry = NULL;
-
- if (copy_from_user(&req, arg, sizeof(dsp_ioctl_task_s))) {
- dsp_err("%s, %d, copy_from_user err.\n", __func__, __LINE__);
- ret = -EINVAL;
- return ret;
- }
- task = &req.task;
-
- // using reserved for op_idx
- dsp->op_idx = task->task.reserved;
- if ((dsp_perf_enable || dsp->perf_enable) &&
- dsp->op_idx < MAX_DSP_TASKS) {
- dsp->op_perf[dsp->op_idx].OpStartCycle =
- 0; //get_perf_timer_cnt();
- dsp->op_perf[dsp->op_idx].Die = dsp->numa_id;
- dsp->op_perf[dsp->op_idx].CoreId = dsp->process_id;
- dsp->op_perf[dsp->op_idx].OpIndex = dsp->op_idx;
- dsp->op_perf[dsp->op_idx].OpType =
- dsp->process_id + 7; // IDX_DSP0
- }
-
- if (!try_module_get(THIS_MODULE)) {
- dsp_err("%s, %d, cannot get module.\n", __func__, __LINE__);
- return -ENODEV;
- }
+ struct es_dsp *dsp = dsp_file->dsp;
user = dsp_find_user_by_fd(dsp_file, task->task.operatorHandle);
if (!user) {
- ret = -EINVAL;
- module_put(THIS_MODULE);
dsp_err("cannot get user.\n");
- return ret;
+ module_put(THIS_MODULE);
+ return NULL;
}
buffer_count = task->task.bufferCntCfg + task->task.bufferCntInput +
@@ -491,8 +471,7 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
kernel_handle_decref(&user->h);
module_put(THIS_MODULE);
dsp_err("kmalloc dsp request struct error.\n");
- ret = -ENOMEM;
- return ret;
+ return NULL;
}
ret = init_kernel_handle(&user_req->handle, dsp_async_task_release,
@@ -502,8 +481,7 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
kernel_handle_decref(&user->h);
kfree(user_req);
module_put(THIS_MODULE);
- ret = -EIO;
- return ret;
+ return NULL;
}
user_req->user = user;
@@ -514,7 +492,6 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
user_req->dsp_file = dsp_file;
user_req->callback = task->task.callback;
user_req->cbarg = task->task.cbArg;
-
INIT_LIST_HEAD(&user_req->async_ll);
dsp_debug("%s, user_req=0x%px.\n", __func__, user_req);
@@ -523,8 +500,8 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
dsp_debug("%s,%d, dsp_req=0x%px.\n", __func__, __LINE__, dsp_req);
ret = dsp_set_task_req(dsp, dsp_req, task);
-
if (ret) {
+ dsp_err("%s, %d, err, ret = %d.\n", __func__, __LINE__, ret);
goto err_req;
}
user_req->req_cpl_handler = dsp_user_async_req_complete;
@@ -532,12 +509,89 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
dsp_req->handle = (u64)(user->op);
ret = dsp_ioctl_set_flat(dsp_file, task, dsp_req->flat_virt, dma_entry);
if (ret != 0) {
- ret = -EINVAL;
+ dsp_err("%s, %d, ret = %d.\n", __func__, __LINE__, ret);
goto err_flat;
}
- kernel_handle_addref(&user_req->handle);
+ user_req->need_notify = need_notify;
+ return user_req;
+err_flat:
+ dsp_free_flat_mem(dsp, dsp_req->flat_size, dsp_req->flat_virt,
+ dsp_req->dsp_flat1_iova);
+err_req:
+ kernel_handle_release_family(&user_req->handle);
+ kernel_handle_decref(&user_req->handle);
+ kernel_handle_decref(&user->h);
+ if (need_notify) {
+ module_put(THIS_MODULE);
+ }
+ return NULL;
+}
+
+static void dsp_free_task(struct dsp_file *dsp_file,
+ struct dsp_user_req_async *user_req)
+{
+ struct dsp_dma_buf **dma_entry = user_req->dma_entry;
+ struct es_dsp *dsp = dsp_file->dsp;
+ u32 buffer_count = user_req->dma_buf_count;
+ dsp_request_t *dsp_req = &user_req->dsp_req;
+ struct dsp_user *user = user_req->user;
+
+ if (dma_entry) {
+ dsp_unmap_dmabuf(dsp_file, dma_entry, buffer_count);
+ }
+ dsp_free_flat_mem(dsp, dsp_req->flat_size, dsp_req->flat_virt,
+ dsp_req->dsp_flat1_iova);
+
+ kernel_handle_release_family(&user_req->handle);
+ kernel_handle_decref(&user_req->handle);
+ kernel_handle_decref(&user->h);
+}
+
+static long dsp_ioctl_submit_tsk_async(struct file *flip,
+ dsp_ioctl_task_s __user *arg)
+{
+ struct dsp_file *dsp_file = flip->private_data;
+ struct es_dsp *dsp = dsp_file->dsp;
+ dsp_ioctl_task_s req;
+ dsp_ioctl_task_s *task;
+ dsp_request_t *dsp_req;
+ struct dsp_user_req_async *user_req;
+ int i, ret;
+
+ if (copy_from_user(&req, arg, sizeof(dsp_ioctl_task_s))) {
+ dsp_err("%s, %d, copy_from_user err.\n", __func__, __LINE__);
+ ret = -EINVAL;
+ return ret;
+ }
+ task = &req;
+
+ // using reserved for op_idx
+ dsp->op_idx = task->task.reserved;
+ if ((dsp_perf_enable || dsp->perf_enable) &&
+ dsp->op_idx < MAX_DSP_TASKS) {
+ dsp->op_perf[dsp->op_idx].OpStartCycle =
+ 0; //get_perf_timer_cnt();
+ dsp->op_perf[dsp->op_idx].Die = dsp->numa_id;
+ dsp->op_perf[dsp->op_idx].CoreId = dsp->process_id;
+ dsp->op_perf[dsp->op_idx].OpIndex = dsp->op_idx;
+ dsp->op_perf[dsp->op_idx].OpType =
+ dsp->process_id + 7; // IDX_DSP0
+ }
+
+ if (!try_module_get(THIS_MODULE)) {
+ dsp_err("%s, %d, cannot get module.\n", __func__, __LINE__);
+ return -ENODEV;
+ }
+
+ user_req = dsp_set_task_info(dsp_file, task, true);
+ if (user_req == NULL) {
+ dsp_err("%s, %d, err\n", __func__, __LINE__);
+ return -EIO;
+ }
req.task.taskHandle = user_req->handle.fd;
- ret = submit_task(dsp->dev, dsp_req);
+
+ kernel_handle_addref(&user_req->handle);
+ ret = submit_task(dsp->dev, &user_req->dsp_req);
if (ret) {
kernel_handle_decref(&user_req->handle);
dsp_err("submit task error.\n");
@@ -547,18 +601,9 @@ static long dsp_ioctl_submit_tsk_async(struct file *flip,
dsp_err("copy to user err.\n");
ret = -EINVAL;
}
- dsp_debug("%s, %d, user refcount=%d.\n\n", __func__, __LINE__,
- kref_read(&user->h.refcount));
return 0;
err_task:
- dsp_unmap_dmabuf(dsp_file, dma_entry, buffer_count);
-err_flat:
- dsp_free_flat_mem(dsp, dsp_req->flat_size, dsp_req->flat_virt,
- dsp_req->dsp_flat1_iova);
-err_req:
- kernel_handle_release_family(&user_req->handle);
- kernel_handle_decref(&user_req->handle);
- kernel_handle_decref(&user->h);
+ dsp_free_task(dsp_file, user_req);
return ret;
}
@@ -961,6 +1006,94 @@ static long dsp_ioctl_get_fw_perf_data(struct file *flip, dsp_fw_perf_t *data)
return ret;
}
+static long dsp_ioctl_multi_tasks_submit(struct file *flip,
+ dsp_ioctl_task_s __user *arg)
+{
+ struct dsp_file *dsp_file = flip->private_data;
+ struct es_dsp *dsp = dsp_file->dsp;
+ dsp_ioctl_task_s req;
+ dsp_ioctl_task_s *tasks;
+ dsp_request_t *dsp_req;
+ struct dsp_user_req_async **user_req;
+ int i, ret;
+ unsigned long flags;
+
+ if (copy_from_user(&req, arg, sizeof(dsp_ioctl_task_s))) {
+ dsp_err("%s, %d, copy_from_user err.\n", __func__, __LINE__);
+ ret = -EINVAL;
+ return ret;
+ }
+
+ if (req.task_num <= 0) {
+ dsp_err("%s, %d, task num below zero, err,\n", __func__,
+ __LINE__);
+ return -EINVAL;
+ }
+ tasks = kzalloc(req.task_num * (sizeof(dsp_ioctl_task_s) +
+ sizeof(struct dsp_user_req_async *)),
+ GFP_KERNEL);
+ if (tasks == NULL) {
+ dsp_err("", __func__, __LINE__);
+ return -ENOMEM;
+ }
+ if (copy_from_user(tasks, arg,
+ req.task_num * sizeof(dsp_ioctl_task_s))) {
+ dsp_err("%s, %d, copy_from_user for multi tasks err.\n",
+ __func__, __LINE__);
+ kfree(tasks);
+ return -EINVAL;
+ }
+ if (!try_module_get(THIS_MODULE)) {
+ dsp_err("%s, %d, cannot get module.\n", __func__, __LINE__);
+ kfree(tasks);
+ return -ENODEV;
+ }
+ user_req = (struct dsp_user_req_async **)(tasks + req.task_num);
+
+ for (i = 0; i < req.task_num; i++) {
+ user_req[i] = dsp_set_task_info(dsp_file, &tasks[i], false);
+ if (user_req[i] == NULL) {
+ dsp_err("%s, %d, and ,i = %d.\n", __func__, __LINE__,
+ i);
+ goto free_task;
+ }
+ tasks[i].task.taskHandle = user_req[i]->handle.fd;
+ }
+ user_req[req.task_num - 1]->need_notify = true;
+
+ if (dsp->off) {
+ dsp_err("es dsp off.\n");
+ ret = -ENODEV;
+ goto free_task;
+ }
+
+ spin_lock_irqsave(&dsp->send_lock, flags);
+ for (i = 0; i < req.task_num; i++) {
+ kernel_handle_addref(&user_req[i]->handle);
+ dsp_req = &user_req[i]->dsp_req;
+ dsp_set_flat_func(dsp_req->flat_virt, dsp_req->handle);
+ __dsp_enqueue_task(dsp, dsp_req);
+ }
+ spin_unlock_irqrestore(&dsp->send_lock, flags);
+
+ dsp_schedule_task(dsp);
+
+ if (copy_to_user(arg, tasks, req.task_num * sizeof(dsp_ioctl_task_s))) {
+ dsp_err("copy to user err.\n");
+ ret = -EINVAL;
+ }
+
+ kfree(tasks);
+ return 0;
+
+free_task:
+ for (i = 0; i < req.task_num; i++) {
+ if (user_req[i] != NULL)
+ dsp_free_task(dsp_file, user_req[i]);
+ }
+ kfree(tasks);
+ return ret;
+}
static long dsp_ioctl(struct file *flip, unsigned int cmd, unsigned long arg)
{
long retval;
@@ -976,6 +1109,11 @@ static long dsp_ioctl(struct file *flip, unsigned int cmd, unsigned long arg)
retval = dsp_ioctl_submit_tsk_async(
flip, (dsp_ioctl_task_s __user *)arg);
break;
+ case DSP_IOCTL_SUBMIT_TSKS_ASYNC:
+ // TODO
+ retval = dsp_ioctl_multi_tasks_submit(
+ flip, (dsp_ioctl_task_s __user *)arg);
+ break;
case DSP_IOCTL_PROCESS_REPORT:
retval = dsp_ioctl_process_complete_task(
flip, (dsp_ioctl_async_process_s __user *)arg);
@@ -1035,7 +1173,8 @@ static int dsp_open(struct inode *inode, struct file *flip)
ret = es_dsp_pm_get_sync(dsp);
if (ret < 0) {
- dsp_err("%s, %d, pm get sync err, ret=%d.\n", __func__, __LINE__, ret);
+ dsp_err("%s, %d, pm get sync err, ret=%d.\n", __func__,
+ __LINE__, ret);
return ret;
}
dsp_file = devm_kzalloc(dsp->dev, sizeof(*dsp_file), GFP_KERNEL);
diff --git a/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl_if.h b/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl_if.h
index 76a88f2399dc..e7ad49893492 100644
--- a/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl_if.h
+++ b/drivers/soc/eswin/ai_driver/dsp/dsp_ioctl_if.h
@@ -36,6 +36,7 @@
#define DSP_IOCTL_ENABLE_PERF _IO(ES_DSP_IOCTL_MAGIC, 22)
#define DSP_IOCTL_GET_PERF_DATA _IO(ES_DSP_IOCTL_MAGIC, 23)
#define DSP_IOCTL_GET_FW_PERF_DATA _IO(ES_DSP_IOCTL_MAGIC, 24)
+#define DSP_IOCTL_SUBMIT_TSKS_ASYNC _IO(ES_DSP_IOCTL_MAGIC, 25)
typedef struct dsp_ioctl_pre_dma_t {
ES_DEV_BUF_S desc;
@@ -54,6 +55,7 @@ typedef struct dsp_ioctl_unload_t {
} __attribute__((packed)) dsp_ioctl_unload_s;
typedef struct dsp_ioctl_task_t {
+ ES_U32 task_num;
ES_DSP_TASK_S task;
ES_S32 result;
} __attribute__((packed)) dsp_ioctl_task_s;
diff --git a/drivers/soc/eswin/ai_driver/dsp/dsp_main.c b/drivers/soc/eswin/ai_driver/dsp/dsp_main.c
index dcfd23a6f205..4a68e4bf3559 100644
--- a/drivers/soc/eswin/ai_driver/dsp/dsp_main.c
+++ b/drivers/soc/eswin/ai_driver/dsp/dsp_main.c
@@ -56,7 +56,7 @@
#define DSP_SUBSYS_HILOAD_CLK 1040000000
#define DSP_SUBSYS_LOWLOAD_CLK 5200000
-#define ES_DSP_DEFAULT_TIMEOUT (100* 6)
+#define ES_DSP_DEFAULT_TIMEOUT (100 * 6)
#ifdef DEBUG
#pragma GCC optimize("O0")
@@ -105,7 +105,7 @@ int es_dsp_exec_cmd_timeout(void)
return fw_timeout;
}
-static void __dsp_enqueue_task(struct es_dsp *dsp, dsp_request_t *req)
+void __dsp_enqueue_task(struct es_dsp *dsp, dsp_request_t *req)
{
struct prio_array *array = &dsp->array;
unsigned long flags;
@@ -175,7 +175,7 @@ static void __dsp_send_task(struct es_dsp *dsp)
es_dsp_send_irq(dsp->hw_arg, (void *)req);
}
-static void dsp_schedule_task(struct es_dsp *dsp)
+void dsp_schedule_task(struct es_dsp *dsp)
{
unsigned long flags;
spin_lock_irqsave(&dsp->send_lock, flags);
@@ -245,6 +245,11 @@ static void dsp_process_expire_work(struct work_struct *work)
dsp_fw_state->exccause, dsp_fw_state->ps, dsp_fw_state->pc,
dsp_fw_state->dsp_task_state, dsp_fw_state->npu_task_state,
dsp_fw_state->func_state);
+
+ if (dsp->stats->last_op_name) {
+ dsp_err("%s, %d, op name = %s.\n", __func__, __LINE__,
+ dsp->stats->last_op_name);
+ }
ret = es_dsp_reboot_core(dsp->hw_arg);
if (ret < 0) {
dsp_err("reboot dsp core failed.\n");
@@ -721,7 +726,7 @@ int __maybe_unused dsp_suspend(struct device *dev)
pm_runtime_mark_last_busy(dsp->dev);
pm_runtime_put_noidle(dsp->dev);
win2030_tbu_power(dsp->dev, false);
- es_dsp_core_clk_disable(dsp);
+ es_dsp_clk_disable(dsp);
dsp_disable_mbox_clock(dsp);
dsp_debug("%s, %d, dsp core%d generic suspend done.\n", __func__,
__LINE__, dsp->process_id);
@@ -743,7 +748,7 @@ int __maybe_unused dsp_resume(struct device *dev)
dsp_err("dsp resume mbox clock err.\n");
return ret;
}
- ret = es_dsp_core_clk_enable(dsp);
+ ret = es_dsp_clk_enable(dsp);
if (ret < 0) {
dev_err(dsp->dev, "couldn't enable DSP\n");
goto out;
@@ -788,7 +793,7 @@ int __maybe_unused dsp_runtime_suspend(struct device *dev)
dsp_debug("%s, dsp core%d runtime suspend.\n", __func__,
dsp->process_id);
win2030_tbu_power(dev, false);
- es_dsp_core_clk_disable(dsp);
+ es_dsp_clk_disable(dsp);
return 0;
}
EXPORT_SYMBOL(dsp_runtime_suspend);
@@ -804,7 +809,7 @@ int __maybe_unused dsp_runtime_resume(struct device *dev)
dsp_debug("%s, dsp core%d runtime resumng.....\n\n", __func__,
dsp->process_id);
- ret = es_dsp_core_clk_enable(dsp);
+ ret = es_dsp_clk_enable(dsp);
if (ret < 0) {
dev_err(dsp->dev, "couldn't enable DSP\n");
goto out;
diff --git a/drivers/soc/eswin/ai_driver/dsp/dsp_main.h b/drivers/soc/eswin/ai_driver/dsp/dsp_main.h
index 308a5b1758fb..d71d1ac3ecaa 100644
--- a/drivers/soc/eswin/ai_driver/dsp/dsp_main.h
+++ b/drivers/soc/eswin/ai_driver/dsp/dsp_main.h
@@ -85,6 +85,7 @@ struct dsp_user_req_async {
/* async_ll, cbarg, callback and dsp_file Can Only use for Lowlevel interface*/
struct list_head async_ll;
struct dsp_file *dsp_file;
+ bool need_notify;
u64 cbarg;
u64 callback;
};
@@ -214,4 +215,6 @@ void dsp_op_release(struct kref *kref);
int es_dsp_exec_cmd_timeout(void);
struct es_dsp *es_proc_get_dsp(int dieid, int dspid);
+void __dsp_enqueue_task(struct es_dsp *dsp, dsp_request_t *req);
+void dsp_schedule_task(struct es_dsp *dsp);
#endif
diff --git a/drivers/soc/eswin/ai_driver/dsp/dsp_platform.c b/drivers/soc/eswin/ai_driver/dsp/dsp_platform.c
index 5051f7585dc8..a2a360d181e9 100644
--- a/drivers/soc/eswin/ai_driver/dsp/dsp_platform.c
+++ b/drivers/soc/eswin/ai_driver/dsp/dsp_platform.c
@@ -255,7 +255,8 @@ int es_dsp_clk_disable(struct es_dsp *dsp)
return ret;
}
- //clk_disable_unprepare(hw->aclk);
+ clk_disable_unprepare(hw->aclk);
+ clk_disable_unprepare(hw->subsys->cfg_clk);
dsp_debug("%s, %d, done.\n", __func__, __LINE__);
return ret;
}
@@ -263,13 +264,18 @@ int es_dsp_clk_enable(struct es_dsp *dsp)
{
struct es_dsp_hw *hw = (struct es_dsp_hw *)dsp->hw_arg;
int ret;
- if (!__clk_is_enabled(hw->aclk)) {
- ret = clk_prepare_enable(hw->aclk);
- if (ret) {
- dev_err(dsp->dev, "failed to enable aclk: %d\n", ret);
- return ret;
- }
+
+ ret = clk_prepare_enable(hw->subsys->cfg_clk);
+ if (ret) {
+ dev_err(dsp->dev, "failed to enable cfg clk, ret=%d.\n", ret);
+ return ret;
}
+ ret = clk_prepare_enable(hw->aclk);
+ if (ret) {
+ dev_err(dsp->dev, "failed to enable aclk: %d\n", ret);
+ return ret;
+ }
+
ret = es_dsp_core_clk_enable(dsp);
if (ret) {
clk_disable_unprepare(hw->aclk);
diff --git a/drivers/soc/eswin/ai_driver/include/es_dsp_internal.h b/drivers/soc/eswin/ai_driver/include/es_dsp_internal.h
index 3fd15afae0c8..2ecbd0e0a0dd 100644
--- a/drivers/soc/eswin/ai_driver/include/es_dsp_internal.h
+++ b/drivers/soc/eswin/ai_driver/include/es_dsp_internal.h
@@ -8,6 +8,7 @@
#ifndef __ESWIN_DSP_INTERNAL_H__
#define __ESWIN_DSP_INTERNAL_H__
#include "es_type.h"
+#include "es_dsp_types.h"
#include "es_dsp_op_types.h"
#ifdef __cplusplus
@@ -183,6 +184,29 @@ typedef struct {
ES_U32 reserved;
} e31_msg_payload_t;
+typedef struct DSP_TASK_DESC_S {
+ ES_S32 dspFd;
+ ES_DSP_TASK_S opTask;
+} __attribute__((packed)) ES_DSP_TASK_DESC_S;
+
+/**
+ * @brief Synchronously submit operator tasks to DSP devices, and only return after all tasks are completed.
+ *
+ * @param[in] tasks: pointer to tasks.
+ * @param[in] numTasks: number of tasks.
+ * @return Returns the execution status code: ES_DSP_SUCCESS for success, others for failure.
+ */
+ES_S32 ES_DSP_LL_SubmitTasks(ES_DSP_TASK_DESC_S *tasks, ES_U32 numTasks);
+
+/**
+ * @brief Asynchronously submit operator tasks to DSP devices, and only return after all tasks are completed.
+ *
+ * @param[in] tasks: pointer to tasks.
+ * @param[in] numTasks: number of tasks.
+ * @return Returns the execution status code: ES_DSP_SUCCESS for success, others for failure.
+ */
+ES_S32 ES_DSP_LL_SubmitAsyncTasks(ES_DSP_TASK_DESC_S *tasks, ES_U32 numTasks);
+
#ifdef __cplusplus
#if __cplusplus
}
diff --git a/drivers/soc/eswin/ai_driver/include/hetero_ipc.h b/drivers/soc/eswin/ai_driver/include/hetero_ipc.h
index 791132cfd645..1a1bf833e668 100644
--- a/drivers/soc/eswin/ai_driver/include/hetero_ipc.h
+++ b/drivers/soc/eswin/ai_driver/include/hetero_ipc.h
@@ -446,14 +446,14 @@ static void messagebox_send_dsp(u32 op_type, u64 payload)
dsp_idx = op_type - IDX_KMD_DSP0;
mbox_base = MAILBOX_E31_TO_DSP_REG[dsp_idx];
mbox_int = MAILBOX_E31_TO_DSP_INT[dsp_idx];
- mbox_lock = (1 << dsp_idx); // must different with dsp driver(8/10/12/14)
+ mbox_lock = BIT1; // must different with dsp driver(using BIT0)
timeout = 0;
// check lock bit and fifo
while (1) {
reg_write(mbox_base + MBOX_NPU_WR_LOCK, mbox_lock);
- if ((reg_read(mbox_base + MBOX_NPU_WR_LOCK) & mbox_lock) ||
+ if ((reg_read(mbox_base + MBOX_NPU_WR_LOCK) & mbox_lock) &&
(reg_read(mbox_base + MBOX_NPU_FIFO_OFFSET) & 0x1) == 0) {
break;
}
--
2.47.0