From a57e984ff5b8fb6e008bfecdf7602201ef397fb0 Mon Sep 17 00:00:00 2001 From: tangdaoyong Date: Thu, 29 Aug 2024 17:26:10 +0800 Subject: [PATCH 162/219] fix:check_mc_cmdbuf_irq Changelogs: 1. Decoder uses a thread to periodically check the vcmd status for multicore decoding, if vcmd node is released, the driver may be exception. Signed-off-by: tangdaoyong --- .../staging/media/eswin/vdec/hantro_vcmd.c | 66 +++++++++---------- drivers/staging/media/eswin/vdec/hantrodec.h | 1 - 2 files changed, 31 insertions(+), 36 deletions(-) diff --git a/drivers/staging/media/eswin/vdec/hantro_vcmd.c b/drivers/staging/media/eswin/vdec/hantro_vcmd.c index 3e07e4a1f5cb..80b75ffc9d78 100644 --- a/drivers/staging/media/eswin/vdec/hantro_vcmd.c +++ b/drivers/staging/media/eswin/vdec/hantro_vcmd.c @@ -788,6 +788,13 @@ static int vcmd_type_core_num[MAX_VCMD_TYPE]; static struct semaphore vcmd_reserve_cmdbuf_sem[MAX_VCMD_TYPE]; //for reserve +#define CMDBUF_LOCK_NUM 8 +static spinlock_t cmdbuf_lock[CMDBUF_LOCK_NUM]; +#define LOCK_CMDBUF_NODE(cmdbuf_id, flags) \ + spin_lock_irqsave(&cmdbuf_lock[(cmdbuf_id) % CMDBUF_LOCK_NUM], flags) +#define UNLOCK_CMDBUF_NODE(cmdbuf_id, flags) \ + spin_unlock_irqrestore(&cmdbuf_lock[(cmdbuf_id) % CMDBUF_LOCK_NUM], flags) + /* PCI base register address (memalloc) */ //extern unsigned long gBaseDDRHw; //extern unsigned int mmu_enable; @@ -1569,7 +1576,10 @@ static long reserve_cmdbuf(struct file *filp, cmdbuf_obj->process_manager_obj = process_manager_obj; input_para->cmdbuf_id = cmdbuf_obj->cmdbuf_id; + + LOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); global_cmdbuf_node[input_para->cmdbuf_id] = new_cmdbuf_node; + UNLOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); return 0; } @@ -1622,13 +1632,10 @@ static long release_cmdbuf(struct file *filp, u16 cmdbuf_id) list, new_cmdbuf_node); if (new_cmdbuf_node) { //free node -#ifdef SUPPORT_DMA_HEAP - spin_lock_irqsave(&fp_priv->vcmdlock, flags); -#endif + LOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); global_cmdbuf_node[cmdbuf_obj->cmdbuf_id] = NULL; -#ifdef SUPPORT_DMA_HEAP - spin_unlock_irqrestore(&fp_priv->vcmdlock, flags); -#endif + UNLOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); + if (cmdbuf_obj->process_manager_obj) { spin_lock_irqsave(&cmdbuf_obj->process_manager_obj->spinlock, flags); @@ -1660,6 +1667,7 @@ static long release_cmdbuf_node(bi_list *list, bi_list_node *cmdbuf_node) { bi_list_node *new_cmdbuf_node = NULL; struct cmdbuf_obj *cmdbuf_obj = NULL; + unsigned long flags = 0; /*get cmdbuf object according to cmdbuf_id*/ new_cmdbuf_node = cmdbuf_node; if (!new_cmdbuf_node) @@ -1669,7 +1677,9 @@ static long release_cmdbuf_node(bi_list *list, bi_list_node *cmdbuf_node) if (new_cmdbuf_node) { //free node cmdbuf_obj = (struct cmdbuf_obj *)new_cmdbuf_node->data; + LOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); global_cmdbuf_node[cmdbuf_obj->cmdbuf_id] = NULL; + UNLOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); free_cmdbuf_node(new_cmdbuf_node); return 0; } @@ -1680,6 +1690,7 @@ static long release_cmdbuf_node_cleanup(bi_list *list) { bi_list_node *new_cmdbuf_node = NULL; struct cmdbuf_obj *cmdbuf_obj = NULL; + unsigned long flags = 0; while (1) { new_cmdbuf_node = list->head; @@ -1689,7 +1700,9 @@ static long release_cmdbuf_node_cleanup(bi_list *list) bi_list_remove_node(list, new_cmdbuf_node); //free node cmdbuf_obj = (struct cmdbuf_obj *)new_cmdbuf_node->data; + LOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); global_cmdbuf_node[cmdbuf_obj->cmdbuf_id] = NULL; + UNLOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); free_cmdbuf_node(new_cmdbuf_node); } return 0; @@ -1913,38 +1926,19 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj, int k; bi_list_node *new_cmdbuf_node = NULL; struct hantrovcmd_dev *dev = NULL; -#ifdef SUPPORT_DMA_HEAP unsigned long flags = 0; - struct filp_priv *fp_priv = NULL; - if (!filp || !filp->private_data) { - LOG_ERR("check mc: invalid filp\n"); - return 0; - } - fp_priv = (struct filp_priv *)filp->private_data; -#endif for (k = 0; k < TOTAL_DISCRETE_CMDBUF_NUM; k++) { + LOCK_CMDBUF_NODE(k, flags); new_cmdbuf_node = global_cmdbuf_node[k]; if (!new_cmdbuf_node) { - continue; - } - -#ifdef SUPPORT_DMA_HEAP - spin_lock_irqsave(&fp_priv->vcmdlock, flags); -#endif - new_cmdbuf_node = global_cmdbuf_node[k]; - if (!new_cmdbuf_node) { -#ifdef SUPPORT_DMA_HEAP - spin_unlock_irqrestore(&fp_priv->vcmdlock, flags); -#endif + UNLOCK_CMDBUF_NODE(k, flags); continue; } cmdbuf_obj = (struct cmdbuf_obj *)new_cmdbuf_node->data; if (!cmdbuf_obj || cmdbuf_obj->filp != filp) { -#ifdef SUPPORT_DMA_HEAP - spin_unlock_irqrestore(&fp_priv->vcmdlock, flags); -#endif + UNLOCK_CMDBUF_NODE(k, flags); continue; } dev = &hantrovcmd_data[cmdbuf_obj->core_id]; @@ -1953,15 +1947,11 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj, if (!cmdbuf_obj->waited) { *irq_status_ret = cmdbuf_obj->cmdbuf_id; cmdbuf_obj->waited = 1; -#ifdef SUPPORT_DMA_HEAP - spin_unlock_irqrestore(&fp_priv->vcmdlock, flags); -#endif + UNLOCK_CMDBUF_NODE(k, flags); return 1; } } -#ifdef SUPPORT_DMA_HEAP - spin_unlock_irqrestore(&fp_priv->vcmdlock, flags); -#endif + UNLOCK_CMDBUF_NODE(k, flags); } return 0; @@ -2590,6 +2580,7 @@ static void vcmd_delink_rm_cmdbuf(struct hantrovcmd_dev *dev, struct cmdbuf_obj *cmdbuf_obj = (struct cmdbuf_obj *)cmdbuf_node->data; bi_list_node *prev = cmdbuf_node->previous; bi_list_node *next = cmdbuf_node->next; + unsigned long flags = 0; LOG_DBG("Delink and remove cmdbuf [%d] from vcmd list.\n", cmdbuf_obj->cmdbuf_id); @@ -2609,7 +2600,9 @@ static void vcmd_delink_rm_cmdbuf(struct hantrovcmd_dev *dev, #endif bi_list_remove_node(list, cmdbuf_node); + LOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); global_cmdbuf_node[cmdbuf_obj->cmdbuf_id] = NULL; + UNLOCK_CMDBUF_NODE(cmdbuf_obj->cmdbuf_id, flags); free_cmdbuf_node(cmdbuf_node); cmdbuf_update_jmp_cmd(dev->hw_version_id, prev ? prev->data : NULL, @@ -2627,7 +2620,6 @@ int hantrovcmd_open(struct inode *inode, struct file *filp) struct process_manager_obj *process_manager_obj = NULL; struct filp_priv *fp_priv = (struct filp_priv *)filp->private_data; - spin_lock_init(&fp_priv->vcmdlock); fp_priv->dev = (void *)dev; process_manager_node = create_process_manager_node(); @@ -4182,6 +4174,10 @@ int hantrovcmd_init(void) cmdbuf_used[0] = 1; cmdbuf_used_residual -= 1; + for (i = 0; i < CMDBUF_LOCK_NUM; i++) { + spin_lock_init(&cmdbuf_lock[i]); + } + create_kernel_process_manager(); for (i = 0; i < MAX_VCMD_TYPE; i++) { if (vcmd_type_core_num[i] == 0) diff --git a/drivers/staging/media/eswin/vdec/hantrodec.h b/drivers/staging/media/eswin/vdec/hantrodec.h index 9991f4e2ed8e..4f597b740eb2 100644 --- a/drivers/staging/media/eswin/vdec/hantrodec.h +++ b/drivers/staging/media/eswin/vdec/hantrodec.h @@ -164,7 +164,6 @@ struct filp_priv { struct heap_root root_d1; #endif void *dev; - spinlock_t vcmdlock; atomic_t core_tasks[DEC_CORE_NUM]; /** for task count of 4 cores*/ }; #endif -- 2.47.0