kernel/0125-feat-vdec-support-power-managemnt.patch

730 lines
24 KiB
Diff

From d05b0ddc69628548f431de87e17c9c61e40aa010 Mon Sep 17 00:00:00 2001
From: tangdaoyong <tangdaoyong@eswincomputing.com>
Date: Thu, 18 Jul 2024 13:54:33 +0800
Subject: [PATCH 125/222] feat:vdec support power managemnt
Changelogs:
1. Add generic & runtime power management for vdec & jdec.
Signed-off-by: tangdaoyong <tangdaoyong@eswincomputing.com>
---
drivers/staging/media/eswin/vdec/dts_parser.c | 4 +-
drivers/staging/media/eswin/vdec/dts_parser.h | 3 +-
drivers/staging/media/eswin/vdec/hantro_dec.c | 194 +++++++++++++++---
.../staging/media/eswin/vdec/hantro_vcmd.c | 89 +++++---
drivers/staging/media/eswin/vdec/hantrodec.h | 10 +-
drivers/staging/media/eswin/vdec/subsys.c | 42 ++++
drivers/staging/media/eswin/vdec/subsys.h | 4 +
7 files changed, 283 insertions(+), 63 deletions(-)
diff --git a/drivers/staging/media/eswin/vdec/dts_parser.c b/drivers/staging/media/eswin/vdec/dts_parser.c
index a423612727d0..8d96a39bf1c5 100644
--- a/drivers/staging/media/eswin/vdec/dts_parser.c
+++ b/drivers/staging/media/eswin/vdec/dts_parser.c
@@ -11,6 +11,7 @@
struct SubsysDesc subsys_array[VDEC_MAX_SUBSYS] = {0};
struct CoreDesc core_array[VDEC_MAX_CORE] = {0};
+u8 numa_id_array[4] = {0};
static int vdec_device_node_scan(unsigned char *compatible)
{
@@ -60,7 +61,7 @@ int vdec_device_nodes_check(void)
index++; \
} while (0)
-int vdec_trans_device_nodes(struct platform_device *pdev)
+int vdec_trans_device_nodes(struct platform_device *pdev, u8 numa_id)
{
extern unsigned int vcmd;
int jpeg = 0;
@@ -125,6 +126,7 @@ int vdec_trans_device_nodes(struct platform_device *pdev)
if (jpeg && vcmd)
hw_type = HW_VC8000DJ;
+ numa_id_array[subsys_id] = numa_id;
VDEC_CORE_ARRAY_ASSIGN(core_index, subsys_id, hw_type, (base_addr + vdec_addr[0]), vdec_addr[1], child_irq);
subsys_id++;
}
diff --git a/drivers/staging/media/eswin/vdec/dts_parser.h b/drivers/staging/media/eswin/vdec/dts_parser.h
index 310cfcf8ccd1..d375b2e23cb9 100644
--- a/drivers/staging/media/eswin/vdec/dts_parser.h
+++ b/drivers/staging/media/eswin/vdec/dts_parser.h
@@ -8,6 +8,7 @@
int vdec_device_nodes_check(void);
extern struct SubsysDesc subsys_array[VDEC_MAX_SUBSYS];
extern struct CoreDesc core_array[VDEC_MAX_CORE];
-extern int vdec_trans_device_nodes(struct platform_device *pdev);
+extern u8 numa_id_array[4];
+extern int vdec_trans_device_nodes(struct platform_device *pdev, u8 numa_id);
#endif /* __DTS_PARSER_H__ */
diff --git a/drivers/staging/media/eswin/vdec/hantro_dec.c b/drivers/staging/media/eswin/vdec/hantro_dec.c
index 54b1fbe6ad6f..12d63a5ea985 100644
--- a/drivers/staging/media/eswin/vdec/hantro_dec.c
+++ b/drivers/staging/media/eswin/vdec/hantro_dec.c
@@ -98,6 +98,7 @@
#include <linux/mfd/syscon.h>
#include <linux/eswin-win2030-sid-cfg.h>
#include <dt-bindings/memory/eswin-win2030-sid.h>
+#include <linux/pm_runtime.h>
#include "subsys.h"
#include "hantroaxife.h"
@@ -410,6 +411,9 @@ static void ReleaseIO(void);
static void ResetAsic(hantrodec_t *dev);
+static int vdec_clk_enable(vdec_clk_rst_t *vcrt);
+static int vdec_pm_enable(struct platform_device *pdev);
+
#ifdef HANTRODEC_DEBUG
static void dump_regs(hantrodec_t *dev);
#endif
@@ -2380,7 +2384,7 @@ static long hantrodec_ioctl(struct file *filp, unsigned int cmd,
size_t buf_size = 0;
void *cpu_vaddr = NULL;
struct heap_mem *hmem, *hmem_d1;
- struct dmabuf_priv *db_priv = (struct dmabuf_priv *)filp->private_data;
+ struct filp_priv *fp_priv = (struct filp_priv *)filp->private_data;
if (copy_from_user(&dbcfg, (void __user *)arg, sizeof(struct dmabuf_cfg)) != 0)
return -EFAULT;
@@ -2389,14 +2393,16 @@ static long hantrodec_ioctl(struct file *filp, unsigned int cmd,
/* map the pha to dma addr(iova)*/
/* syscoherent <-> false, flush one time when import; sys,cma <-> true, no fush */
- hmem = common_dmabuf_heap_import_from_user(&db_priv->root, dbcfg.dmabuf_fd);
+ hmem = common_dmabuf_heap_import_from_user(&fp_priv->root, dbcfg.dmabuf_fd);
if(IS_ERR(hmem)) {
LOG_ERR("dmabuf-heap alloc from userspace failed\n");
return -ENOMEM;
}
+ //LOG_INFO("import dmabuf_fd = %d, hmem=%px, filp=%px, platformdev_d1=%px\n", dbcfg.dmabuf_fd, hmem, filp,
+ // platformdev_d1);
if (platformdev_d1) {
- hmem_d1 = common_dmabuf_heap_import_from_user(&db_priv->root_d1, dbcfg.dmabuf_fd);
+ hmem_d1 = common_dmabuf_heap_import_from_user(&fp_priv->root_d1, dbcfg.dmabuf_fd);
if(IS_ERR(hmem_d1)) {
common_dmabuf_heap_release(hmem);
LOG_ERR("dmabuf-heap alloc from userspace failed for d1\n");
@@ -2438,7 +2444,7 @@ static long hantrodec_ioctl(struct file *filp, unsigned int cmd,
common_dmabuf_heap_release(hmem);
if (platformdev_d1)
common_dmabuf_heap_release(hmem_d1);
- LOG_DBG("%s %d: copy_from_user failed, returned %li\n", __func__, __LINE__, tmp);
+ LOG_ERR("%s %d: copy_from_user failed, returned %li\n", __func__, __LINE__, tmp);
return -EFAULT;
}
@@ -2447,7 +2453,7 @@ static long hantrodec_ioctl(struct file *filp, unsigned int cmd,
case HANTRODEC_IOC_DMA_HEAP_PUT_IOVA: {
struct heap_mem *hmem, *hmem_d1;
unsigned int dmabuf_fd;
- struct dmabuf_priv *db_priv = (struct dmabuf_priv *)filp->private_data;
+ struct filp_priv *fp_priv = (struct filp_priv *)filp->private_data;
if (copy_from_user(&dmabuf_fd, (void __user *)arg, sizeof(int)) != 0)
return -EFAULT;
@@ -2455,20 +2461,22 @@ static long hantrodec_ioctl(struct file *filp, unsigned int cmd,
LOG_DBG("release dmabuf_fd = %d\n", dmabuf_fd);
/* find the heap_mem */
- hmem = common_dmabuf_lookup_heapobj_by_fd(&db_priv->root, dmabuf_fd);
+ hmem = common_dmabuf_lookup_heapobj_by_fd(&fp_priv->root, dmabuf_fd);
if (IS_ERR(hmem)) {
- LOG_ERR("cannot find dmabuf-heap for dmabuf_fd %d\n", dmabuf_fd);
+ LOG_ERR("cannot find dmabuf-heap for dmabuf_fd %d, %px\n", dmabuf_fd, hmem);
return -ENOMEM;
}
if (platformdev_d1) {
- hmem_d1 = common_dmabuf_lookup_heapobj_by_fd(&db_priv->root_d1, dmabuf_fd);
+ hmem_d1 = common_dmabuf_lookup_heapobj_by_fd(&fp_priv->root_d1, dmabuf_fd);
if (IS_ERR(hmem_d1)) {
LOG_ERR("cannot find dmabuf-heap for dmabuf_fd %d on d1\n", dmabuf_fd);
return -EFAULT;
}
common_dmabuf_heap_release(hmem_d1);
}
+ //LOG_INFO("release dmabuf_fd = %d, hmem=%px, filp=%px, platformdev_d1=%px\n", dmabuf_fd, hmem, filp,
+ // platformdev_d1);
common_dmabuf_heap_release(hmem);
return 0;
@@ -2535,22 +2543,24 @@ static int hantrodec_mmap(struct file *filp, struct vm_area_struct *vma)
*/
static int hantrodec_open(struct inode *inode, struct file *filp)
{
-#ifdef SUPPORT_DMA_HEAP
- struct dmabuf_priv *db_priv;
+ struct filp_priv *fp_priv;
- db_priv = kzalloc(sizeof(struct dmabuf_priv), GFP_KERNEL);
- if (!db_priv) {
+ fp_priv = kzalloc(sizeof(struct filp_priv), GFP_KERNEL);
+ if (!fp_priv) {
pr_err("%s: alloc failed\n", __func__);
return -ENOMEM;
}
- common_dmabuf_heap_import_init(&db_priv->root, &platformdev->dev);
+#ifdef SUPPORT_DMA_HEAP
+ common_dmabuf_heap_import_init(&fp_priv->root, &platformdev->dev);
if (platformdev_d1) {
- common_dmabuf_heap_import_init(&db_priv->root_d1, &platformdev_d1->dev);
+ common_dmabuf_heap_import_init(&fp_priv->root_d1, &platformdev_d1->dev);
}
-
- filp->private_data = (void *)db_priv;
#endif
+ for (u32 core_id = 0; core_id < DEC_CORE_NUM; core_id ++) {
+ atomic_set(&(fp_priv->core_tasks[core_id]), 0);
+ }
+ filp->private_data = (void *)fp_priv;
LOG_DBG("dev opened\n");
if (vcmd)
@@ -2571,9 +2581,7 @@ static int hantrodec_release(struct inode *inode,
{
int n;
hantrodec_t *dev = &hantrodec_data;
-#ifdef SUPPORT_DMA_HEAP
- struct dmabuf_priv *db_priv = (struct dmabuf_priv *)filp->private_data;
-#endif
+ struct filp_priv *fp_priv = (struct filp_priv *)filp->private_data;
LOG_DBG("closing ...\n");
@@ -2601,12 +2609,18 @@ static int hantrodec_release(struct inode *inode,
end:
#ifdef SUPPORT_DMA_HEAP
- common_dmabuf_heap_import_uninit(&db_priv->root);
+ common_dmabuf_heap_import_uninit(&fp_priv->root);
if (platformdev_d1) {
- common_dmabuf_heap_import_uninit(&db_priv->root_d1);
+ common_dmabuf_heap_import_uninit(&fp_priv->root_d1);
}
- kfree(db_priv);
#endif
+ for (u32 core_id = 0; core_id < DEC_CORE_NUM; core_id ++) {
+ /** clear the tasks for pm*/
+ while (atomic_dec_return(&(fp_priv->core_tasks[core_id])) >= 0) {
+ vdec_pm_runtime_put(core_id);
+ }
+ }
+ kfree(fp_priv);
LOG_DBG("closed\n");
return 0;
@@ -2810,8 +2824,8 @@ static int hantrodec_init(void)
hantrodec_data.async_queue_pp = NULL;
result = register_chrdev(hantrodec_major,
- DEC_DEV_NAME,
- &hantrodec_fops);
+ DEC_DEV_NAME,
+ &hantrodec_fops);
if (result < 0) {
LOG_ERR("unable to get major %d\n", hantrodec_major);
goto err;
@@ -3703,6 +3717,12 @@ static int vdec_sys_clk_enable(vdec_clk_rst_t *vcrt)
LOG_INFO("VD set vd_clk to %ldHZ\n", rate);
}
+ return vdec_clk_enable(vcrt);
+}
+
+static int vdec_clk_enable(vdec_clk_rst_t *vcrt) {
+ int ret;
+
ret = clk_prepare_enable(vcrt->aclk);
if (ret) {
LOG_ERR("Video Decoder: failed to enable aclk: %d\n", ret);
@@ -3873,6 +3893,10 @@ static int hantro_vdec_probe(struct platform_device *pdev)
int ret, vdec_dev_num = 0;
static int pdev_count = 0;
vdec_clk_rst_t *vcrt = devm_kzalloc(&pdev->dev, sizeof(vdec_clk_rst_t), GFP_KERNEL);
+ if (!vcrt) {
+ LOG_ERR("malloc drvdata failed\n");
+ return -ENOMEM;
+ }
// pr_info("[%s]build version: %s\n", DEC_DEV_NAME, ES_VDEC_GIT_VER);
vdec_dev_num = vdec_device_nodes_check();
@@ -3921,7 +3945,7 @@ static int hantro_vdec_probe(struct platform_device *pdev)
d1_clk_reset_init();
}
- if (vdec_trans_device_nodes(pdev)) {
+ if (vdec_trans_device_nodes(pdev, numa_id)) {
LOG_ERR("Translates video decoder dts to subsys failed");
return -1;
}
@@ -3951,17 +3975,36 @@ static int hantro_vdec_probe(struct platform_device *pdev)
LOG_NOTICE("load driver %s failed\n", DEC_DEV_NAME);
} else {
LOG_NOTICE("module inserted. Major = %d\n", hantrodec_major);
+
+ if (platformdev && vdec_pm_enable(platformdev) < 0) {
+ LOG_WARN("enable pm for vdec-die0 failed\n");
+ }
+ if (platformdev_d1 && vdec_pm_enable(platformdev_d1) < 0) {
+ LOG_WARN("enable pm for vdec-die1 failed\n");
+ }
}
return ret;
}
+static int vdec_pm_enable(struct platform_device *pdev) {
+ /* The code below assumes runtime PM to be disabled. */
+ WARN_ON(pm_runtime_enabled(&pdev->dev));
+ pm_runtime_set_autosuspend_delay(&pdev->dev, 1000);
+ pm_runtime_use_autosuspend(&pdev->dev);
+ pm_runtime_set_active(&pdev->dev);
+ pm_runtime_enable(&pdev->dev);
+
+ return 0;
+}
+
static int hantro_vdec_remove(struct platform_device *pdev)
{
#ifdef SUPPORT_DMA_HEAP
int ret;
- vdec_clk_rst_t *vcrt;
#endif
+ vdec_clk_rst_t *vcrt;
+ pm_runtime_disable(&pdev->dev);
hantrodec_cleanup();
#ifdef SUPPORT_DMA_HEAP
ret = win2030_tbu_power(&pdev->dev, false);
@@ -3969,14 +4012,110 @@ static int hantro_vdec_remove(struct platform_device *pdev)
LOG_ERR("vdec tbu power down failed\n");
return -1;
}
+#endif
vcrt = platform_get_drvdata(pdev);
vdec_hardware_reset(vcrt);
vdec_clk_disable(vcrt);
-#endif
return 0;
}
+static int eswin_vdec_runtime_suspend(struct device *dev) {
+ vdec_clk_rst_t *vcrt = NULL;
+ int ret = -1;
+
+ vcrt = dev_get_drvdata(dev);
+ if (vcrt) {
+ ret = win2030_tbu_power(dev, false);
+ if (ret != 0) {
+ LOG_ERR("tbu power up failed, %d\n", __LINE__);
+ return -1;
+ }
+ ret = vdec_clk_disable(vcrt);
+ }
+ return ret;
+}
+
+static int eswin_vdec_runtime_resume(struct device *dev) {
+ vdec_clk_rst_t *vcrt = NULL;
+ int ret = -1;
+
+ vcrt = dev_get_drvdata(dev);
+ if (vcrt) {
+ ret = vdec_clk_enable(vcrt);
+ if (ret) {
+ LOG_ERR("enable sys clk failed, %d\n", __LINE__);
+ return ret;
+ }
+
+ ret = win2030_tbu_power(dev, true);
+ if (ret != 0) {
+ LOG_ERR("tbu power down failed, %d\n", __LINE__);
+ return -1;
+ }
+ }
+ return ret;
+}
+
+/** <TODO> the jd & vd should be seperated as two devices*/
+int vdec_wait_device_idle(struct platform_device *pdev) {
+ int ret;
+
+ if (pdev == platformdev) {
+ ret = hantrovcmd_wait_core_idle(0, msecs_to_jiffies(500));
+ if (ret <= 0) {
+ return ret;
+ }
+ ret = hantrovcmd_wait_core_idle(1, msecs_to_jiffies(500));
+ return ret;
+ }
+ else if (pdev == platformdev_d1) {
+ ret = hantrovcmd_wait_core_idle(2, msecs_to_jiffies(500));
+ if (ret <= 0) {
+ return ret;
+ }
+ ret = hantrovcmd_wait_core_idle(3, msecs_to_jiffies(500));
+ return ret;
+ }
+
+ LOG_ERR("Unknown platform device = %p\n", pdev);
+ return 1;
+}
+
+static int eswin_vdec_suspend(struct device *dev) {
+ int ret = 0;
+ struct platform_device *pdev = NULL;
+
+ if (!pm_runtime_status_suspended(dev)) {
+ pdev = container_of(dev, struct platform_device, dev);
+ ret = vdec_wait_device_idle(pdev);
+ if (!ret) {
+ LOG_ERR("Timeout for vdec_suspend\n");
+ return -ETIMEDOUT;
+ } else if (ret < 0) {
+ LOG_ERR("Interrupt triggered while vdec_suspend\n");
+ return -ERESTARTSYS;
+ }
+
+ ret = eswin_vdec_runtime_suspend(dev);
+ }
+ return ret;
+}
+
+static int eswin_vdec_resume(struct device *dev) {
+ int ret = 0;
+
+ if (!pm_runtime_status_suspended(dev)) {
+ ret = eswin_vdec_runtime_resume(dev);
+ }
+ return ret;
+}
+
+static const struct dev_pm_ops eswin_vdec_dev_pm_ops = {
+ LATE_SYSTEM_SLEEP_PM_OPS(eswin_vdec_suspend, eswin_vdec_resume)
+ RUNTIME_PM_OPS(eswin_vdec_runtime_suspend, eswin_vdec_runtime_resume, NULL)
+};
+
static const struct of_device_id eswin_vdec_match[] = {
{ .compatible = "eswin,video-decoder0", },
{ .compatible = "eswin,video-decoder1", },
@@ -3988,6 +4127,7 @@ static struct platform_driver eswin_vdec_driver = {
.driver = {
.name = DEC_DEV_NAME,
.of_match_table = eswin_vdec_match,
+ .pm = &eswin_vdec_dev_pm_ops,
},
};
diff --git a/drivers/staging/media/eswin/vdec/hantro_vcmd.c b/drivers/staging/media/eswin/vdec/hantro_vcmd.c
index 76a411fdbde2..3e07e4a1f5cb 100644
--- a/drivers/staging/media/eswin/vdec/hantro_vcmd.c
+++ b/drivers/staging/media/eswin/vdec/hantro_vcmd.c
@@ -781,6 +781,7 @@ static int vcmd_type_core_num[MAX_VCMD_TYPE];
#define WORKING_STATE_IDLE 0
#define WORKING_STATE_WORKING 1
+#define WORKING_STATE_STALL 2
#define CMDBUF_EXE_STATUS_OK 0
#define CMDBUF_EXE_STATUS_CMDERR 1
#define CMDBUF_EXE_STATUS_BUSERR 2
@@ -1583,14 +1584,13 @@ static long release_cmdbuf(struct file *filp, u16 cmdbuf_id)
unsigned long flags;
struct hantrovcmd_dev *dev = NULL;
-#ifdef SUPPORT_DMA_HEAP
- struct dmabuf_priv *db_priv = NULL;
+ struct filp_priv *fp_priv = NULL;
if (!filp || !filp->private_data) {
LOG_ERR("invalid filp\n");
return -1;
}
- db_priv = (struct dmabuf_priv *)filp->private_data;
-#endif
+ fp_priv = (struct filp_priv *)filp->private_data;
+
/*get cmdbuf object according to cmdbuf_id*/
new_cmdbuf_node = global_cmdbuf_node[cmdbuf_id];
if (!new_cmdbuf_node) {
@@ -1623,11 +1623,11 @@ static long release_cmdbuf(struct file *filp, u16 cmdbuf_id)
if (new_cmdbuf_node) {
//free node
#ifdef SUPPORT_DMA_HEAP
- spin_lock_irqsave(&db_priv->vcmdlock, flags);
+ spin_lock_irqsave(&fp_priv->vcmdlock, flags);
#endif
global_cmdbuf_node[cmdbuf_obj->cmdbuf_id] = NULL;
#ifdef SUPPORT_DMA_HEAP
- spin_unlock_irqrestore(&db_priv->vcmdlock, flags);
+ spin_unlock_irqrestore(&fp_priv->vcmdlock, flags);
#endif
if (cmdbuf_obj->process_manager_obj) {
spin_lock_irqsave(&cmdbuf_obj->process_manager_obj->spinlock,
@@ -1648,6 +1648,11 @@ static long release_cmdbuf(struct file *filp, u16 cmdbuf_id)
}
//spin_unlock_irqrestore(dev->spinlock, flags);
up(&vcmd_reserve_cmdbuf_sem[module_type]);
+ /** release for the pm*/
+ if (atomic_dec_return(&(fp_priv->core_tasks[dev->core_id])) >= 0) {
+ vdec_pm_runtime_put(dev->core_id);
+ }
+
return 0;
}
@@ -1797,8 +1802,12 @@ static long link_and_run_cmdbuf(struct file *filp,
dev = &hantrovcmd_data[cmdbuf_obj->core_id];
input_para->core_id = cmdbuf_obj->core_id;
- LOG_TRACE("Vdec Allocate cmd buffer [%d] to core [%d]\n", cmdbuf_id,
- input_para->core_id);
+ LOG_TRACE("Vdec Allocate cmd buffer [%d] to core [%d]\n", cmdbuf_id, input_para->core_id);
+ if (filp) {
+ struct filp_priv *fp_priv = (struct filp_priv *)filp->private_data;
+ vdec_pm_runtime_sync(dev->core_id);
+ atomic_inc(&(fp_priv->core_tasks[dev->core_id]));
+ }
//set ddr address for vcmd registers copy.
if (dev->hw_version_id > HW_ID_1_0_C) {
//read vcmd executing register into ddr memory.
@@ -1906,12 +1915,12 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj,
struct hantrovcmd_dev *dev = NULL;
#ifdef SUPPORT_DMA_HEAP
unsigned long flags = 0;
- struct dmabuf_priv *db_priv = NULL;
+ struct filp_priv *fp_priv = NULL;
if (!filp || !filp->private_data) {
LOG_ERR("check mc: invalid filp\n");
return 0;
}
- db_priv = (struct dmabuf_priv *)filp->private_data;
+ fp_priv = (struct filp_priv *)filp->private_data;
#endif
for (k = 0; k < TOTAL_DISCRETE_CMDBUF_NUM; k++) {
@@ -1921,12 +1930,12 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj,
}
#ifdef SUPPORT_DMA_HEAP
- spin_lock_irqsave(&db_priv->vcmdlock, flags);
+ 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(&db_priv->vcmdlock, flags);
+ spin_unlock_irqrestore(&fp_priv->vcmdlock, flags);
#endif
continue;
}
@@ -1934,7 +1943,7 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj,
cmdbuf_obj = (struct cmdbuf_obj *)new_cmdbuf_node->data;
if (!cmdbuf_obj || cmdbuf_obj->filp != filp) {
#ifdef SUPPORT_DMA_HEAP
- spin_unlock_irqrestore(&db_priv->vcmdlock, flags);
+ spin_unlock_irqrestore(&fp_priv->vcmdlock, flags);
#endif
continue;
}
@@ -1945,13 +1954,13 @@ static int check_mc_cmdbuf_irq(struct file *filp, struct cmdbuf_obj *cmdbuf_obj,
*irq_status_ret = cmdbuf_obj->cmdbuf_id;
cmdbuf_obj->waited = 1;
#ifdef SUPPORT_DMA_HEAP
- spin_unlock_irqrestore(&db_priv->vcmdlock, flags);
+ spin_unlock_irqrestore(&fp_priv->vcmdlock, flags);
#endif
return 1;
}
}
#ifdef SUPPORT_DMA_HEAP
- spin_unlock_irqrestore(&db_priv->vcmdlock, flags);
+ spin_unlock_irqrestore(&fp_priv->vcmdlock, flags);
#endif
}
@@ -2616,13 +2625,11 @@ int hantrovcmd_open(struct inode *inode, struct file *filp)
bi_list_node *process_manager_node;
unsigned long flags;
struct process_manager_obj *process_manager_obj = NULL;
-#ifdef SUPPORT_DMA_HEAP
- struct dmabuf_priv *db_priv = (struct dmabuf_priv *)filp->private_data;
- spin_lock_init(&db_priv->vcmdlock);
- db_priv->dev = (void *)dev;
-#else
- filp->private_data = (void *)dev;
-#endif
+ 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();
if (!process_manager_node) {
pr_err("%d: create_process_manager_node failed\n", __LINE__);
@@ -2642,9 +2649,7 @@ int hantrovcmd_open(struct inode *inode, struct file *filp)
int hantrovcmd_release(struct inode *inode, struct file *filp)
{
-#ifdef SUPPORT_DMA_HEAP
- struct dmabuf_priv *db_priv = NULL;
-#endif
+ struct filp_priv *fp_priv = NULL;
struct hantrovcmd_dev *dev = NULL;
u32 core_id = 0;
u32 release_cmdbuf_num = 0;
@@ -2664,12 +2669,8 @@ int hantrovcmd_release(struct inode *inode, struct file *filp)
return EFAULT;
}
-#ifdef SUPPORT_DMA_HEAP
- db_priv = (struct dmabuf_priv *)filp->private_data;
- dev = (struct hantrovcmd_dev *)db_priv->dev;
-#else
- dev = (struct hantrovcmd_dev *)filp->private_data;
-#endif
+ fp_priv = (struct filp_priv *)filp->private_data;
+ dev = (struct hantrovcmd_dev *)fp_priv->dev;
if (down_interruptible(&vcmd_reserve_cmdbuf_sem[dev->vcmd_core_cfg.sub_module_type]))
return -ERESTARTSYS;
@@ -4982,3 +4983,29 @@ static void printk_vcmd_register_debug(const void *hwregs, char *info)
}
#endif
}
+
+static int check_dev_idle(struct hantrovcmd_dev *dev) {
+ int idle = 0;
+
+ u8 vcmd_state = vcmd_get_register_value((const void *)dev->hwregs,
+ dev->reg_mirror, HWIF_VCMD_WORK_STATE);
+ if (WORKING_STATE_STALL != vcmd_state && WORKING_STATE_WORKING != vcmd_state) {
+ idle = 1;
+ } else {
+ // LOG_WARN("check_dev_idle, vcmd_state = %u\n", vcmd_state);
+ }
+ LOG_WARN("check_dev_idle, vcmd_state = %u\n", vcmd_state);
+ return idle;
+}
+
+int hantrovcmd_wait_core_idle(u32 core_id, long timeout) {
+ struct hantrovcmd_dev *dev = NULL;
+
+ if (core_id >= total_vcmd_core_num) {
+ LOG_ERR("invalid core_id = %u, vcmd_core_num = %u\n", core_id, total_vcmd_core_num);
+ return -ERESTARTSYS;
+ }
+
+ dev = &hantrovcmd_data[core_id];
+ return wait_event_interruptible_timeout(*dev->wait_queue, check_dev_idle(dev), timeout);
+}
diff --git a/drivers/staging/media/eswin/vdec/hantrodec.h b/drivers/staging/media/eswin/vdec/hantrodec.h
index e9f356be4f07..9991f4e2ed8e 100644
--- a/drivers/staging/media/eswin/vdec/hantrodec.h
+++ b/drivers/staging/media/eswin/vdec/hantrodec.h
@@ -153,18 +153,22 @@ struct dmabuf_split {
unsigned int offset; /* offset of the buffer corresponding to dmabuf_fd */
unsigned int length; /* size of the splitted buffer start from offset */
};
+#endif
+
+#define DEC_CORE_NUM (4)
#ifdef __KERNEL__
-struct dmabuf_priv {
+struct filp_priv {
+#ifdef SUPPORT_DMA_HEAP
struct heap_root root;
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
-#endif
-
/* Use 'k' as magic number */
#define HANTRODEC_IOC_MAGIC 'k'
diff --git a/drivers/staging/media/eswin/vdec/subsys.c b/drivers/staging/media/eswin/vdec/subsys.c
index 0f68ed405bce..7b8b2a50975c 100644
--- a/drivers/staging/media/eswin/vdec/subsys.c
+++ b/drivers/staging/media/eswin/vdec/subsys.c
@@ -56,6 +56,8 @@
#include "subsys.h"
#include "dts_parser.h"
+#include <linux/pm_runtime.h>
+#include <linux/platform_device.h>
#define LOG_TAG DEC_DEV_NAME ":subs"
#include "vc_drv_log.h"
@@ -210,3 +212,43 @@ void CheckSubsysCoreArray(struct subsys_config *subsys, int *vcmd)
i, multicorebase[i], iosize[i]);
}
}
+
+struct platform_device *vdec_get_platform_device(u32 core_id)
+{
+ struct platform_device *pdev = NULL;
+ u32 core_num = 4;
+ u8 numa_id;
+
+ if (core_id >= core_num) {
+ LOG_ERR("invalid core_id = %u, core_num = %u\n", core_id, core_num);
+ return NULL;
+ }
+ numa_id = numa_id_array[core_id];
+
+ if (0 == numa_id)
+ pdev = platformdev;
+ else if (1 == numa_id)
+ pdev = platformdev_d1;
+
+ return pdev;
+}
+
+int vdec_pm_runtime_sync(u32 core_id) {
+ struct platform_device *pdev = vdec_get_platform_device(core_id);
+
+ if (!pdev) {
+ LOG_ERR("get platform device failed for pm sync, core_id = %u\n", core_id);
+ }
+
+ return pm_runtime_get_sync(&pdev->dev);
+}
+
+int vdec_pm_runtime_put(u32 core_id) {
+ struct platform_device *pdev = vdec_get_platform_device(core_id);
+
+ if (!pdev) {
+ LOG_ERR("get platform device failed for pm put, numa_id = %u\n", core_id);
+ }
+
+ return pm_runtime_put(&pdev->dev);
+}
diff --git a/drivers/staging/media/eswin/vdec/subsys.h b/drivers/staging/media/eswin/vdec/subsys.h
index 39b68c6c1bbd..17a82a6169b0 100644
--- a/drivers/staging/media/eswin/vdec/subsys.h
+++ b/drivers/staging/media/eswin/vdec/subsys.h
@@ -173,6 +173,10 @@ int hantrovcmd_mmap(struct file *filp, struct vm_area_struct *vma);
int hantrovcmd_init(void);
void hantrovcmd_cleanup(void);
+int hantrovcmd_wait_core_idle(u32 core_id, long timeout);
+struct platform_device *vdec_get_platform_device(u32 core_id);
+int vdec_pm_runtime_sync(u32 core_id);
+int vdec_pm_runtime_put(u32 core_id);
/******************************************************************************/
/* MMU */
/******************************************************************************/
--
2.47.0