845 lines
23 KiB
Diff
845 lines
23 KiB
Diff
From 08607d6cb0ff303706da7275968b76e18de93bf9 Mon Sep 17 00:00:00 2001
|
||
From: donghuawei <donghuawei@eswincomputing.com>
|
||
Date: Fri, 24 May 2024 13:59:55 +0800
|
||
Subject: [PATCH 032/222] refactor:khandle, dsp subsys drv
|
||
|
||
Changelogs:
|
||
adaptor khandle, dsp subsys drv for linux-6.6
|
||
---
|
||
drivers/soc/Kconfig | 1 +
|
||
drivers/soc/Makefile | 1 +
|
||
drivers/soc/eswin/Kconfig | 20 ++
|
||
drivers/soc/eswin/Makefile | 4 +
|
||
drivers/soc/eswin/dsp_subsys.c | 348 +++++++++++++++++++++++++++
|
||
drivers/soc/eswin/eswin-dsp-subsys.h | 44 ++++
|
||
drivers/soc/eswin/eswin-khandle.c | 242 +++++++++++++++++++
|
||
drivers/soc/eswin/eswin-khandle.h | 84 +++++++
|
||
drivers/soc/eswin/eswin_timer.h | 6 +
|
||
9 files changed, 750 insertions(+)
|
||
create mode 100644 drivers/soc/eswin/Kconfig
|
||
create mode 100644 drivers/soc/eswin/Makefile
|
||
create mode 100644 drivers/soc/eswin/dsp_subsys.c
|
||
create mode 100644 drivers/soc/eswin/eswin-dsp-subsys.h
|
||
create mode 100644 drivers/soc/eswin/eswin-khandle.c
|
||
create mode 100644 drivers/soc/eswin/eswin-khandle.h
|
||
create mode 100644 drivers/soc/eswin/eswin_timer.h
|
||
|
||
diff --git a/drivers/soc/Kconfig b/drivers/soc/Kconfig
|
||
index d21e75d69294..bc857765a996 100644
|
||
--- a/drivers/soc/Kconfig
|
||
+++ b/drivers/soc/Kconfig
|
||
@@ -24,6 +24,7 @@ source "drivers/soc/renesas/Kconfig"
|
||
source "drivers/soc/rockchip/Kconfig"
|
||
source "drivers/soc/samsung/Kconfig"
|
||
source "drivers/soc/sifive/Kconfig"
|
||
+source "drivers/soc/eswin/Kconfig"
|
||
source "drivers/soc/starfive/Kconfig"
|
||
source "drivers/soc/sunxi/Kconfig"
|
||
source "drivers/soc/tegra/Kconfig"
|
||
diff --git a/drivers/soc/Makefile b/drivers/soc/Makefile
|
||
index 0706a27d13be..5dbaca5b7bac 100644
|
||
--- a/drivers/soc/Makefile
|
||
+++ b/drivers/soc/Makefile
|
||
@@ -29,6 +29,7 @@ obj-y += renesas/
|
||
obj-y += rockchip/
|
||
obj-$(CONFIG_SOC_SAMSUNG) += samsung/
|
||
obj-y += sifive/
|
||
+obj-y += eswin/
|
||
obj-y += sunxi/
|
||
obj-$(CONFIG_ARCH_TEGRA) += tegra/
|
||
obj-y += ti/
|
||
diff --git a/drivers/soc/eswin/Kconfig b/drivers/soc/eswin/Kconfig
|
||
new file mode 100644
|
||
index 000000000000..54179b920cc8
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/Kconfig
|
||
@@ -0,0 +1,20 @@
|
||
+if SOC_SIFIVE || SOC_STARFIVE
|
||
+
|
||
+config ESWIN_KHANDLE
|
||
+ bool "eswin kernel khandle functions"
|
||
+ default y
|
||
+ help
|
||
+ eswin realize this khandle, and mainly use for user process resource
|
||
+ mangement.
|
||
+
|
||
+config ESWIN_DSP_SUBSYS
|
||
+ tristate "Eswin dsp subsys"
|
||
+ default y
|
||
+ help
|
||
+ This is hardware-specific DSP subsys kernel driver for the eswin
|
||
+ hardware. It should be enabled to support dsp on eswin
|
||
+ platform.
|
||
+
|
||
+ If unsure, say N.
|
||
+
|
||
+endif
|
||
diff --git a/drivers/soc/eswin/Makefile b/drivers/soc/eswin/Makefile
|
||
new file mode 100644
|
||
index 000000000000..290bd185817f
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/Makefile
|
||
@@ -0,0 +1,4 @@
|
||
+obj-$(CONFIG_ESWIN_KHANDLE) += eswin-khandle.o
|
||
+obj-$(CONFIG_ESWIN_DSP_SUBSYS) += dsp_subsys.o
|
||
+
|
||
+
|
||
diff --git a/drivers/soc/eswin/dsp_subsys.c b/drivers/soc/eswin/dsp_subsys.c
|
||
new file mode 100644
|
||
index 000000000000..acf33631926a
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/dsp_subsys.c
|
||
@@ -0,0 +1,348 @@
|
||
+/*
|
||
+ * Program's name, and a brief idea of what it does(One line).
|
||
+ * Copyright 20XX, Beijing ESWIN Computing Technology Co., Ltd.. All rights reserved.
|
||
+ * SPDX-License-Identifier: GPL-2.0-only
|
||
+ *
|
||
+ * This program is free software: you can redistribute it and/or modify
|
||
+ * it under the terms of the GNU General Public License as published by
|
||
+ * the Free Software Foundation, version 2.
|
||
+ *
|
||
+ * This program is distributed in the hope that it will be useful,
|
||
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
+ * GNU General Public License for more details.
|
||
+ *
|
||
+ * You should have received a copy of the GNU General Public License
|
||
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
+ */
|
||
+
|
||
+#include <linux/delay.h>
|
||
+#include <linux/io.h>
|
||
+#include <linux/module.h>
|
||
+#include <linux/of_address.h>
|
||
+#include <linux/of_device.h>
|
||
+#include <linux/platform_device.h>
|
||
+#include <linux/regmap.h>
|
||
+#include <linux/reset.h>
|
||
+#include <linux/clk.h>
|
||
+#include <linux/win2030_noc.h>
|
||
+#include <dt-bindings/interconnect/eswin,win2030.h>
|
||
+#include "eswin-dsp-subsys.h"
|
||
+
|
||
+#define DRIVER_NAME "eswin-dsp-subsys"
|
||
+
|
||
+/**
|
||
+ * dsp_subsys_status - query the dsp subsys transaction status
|
||
+ *
|
||
+ * @void
|
||
+ *
|
||
+ * return: module transaction status on success , 1 if idle, 0 if busy.
|
||
+ * negative for error
|
||
+ * if can't get idle status in 3 seconds, return current status.
|
||
+ */
|
||
+static int dsp_subsys_status(void)
|
||
+{
|
||
+ unsigned long deadline = jiffies + 3 * HZ;
|
||
+ int status = 0;
|
||
+
|
||
+ do {
|
||
+ status = win2030_noc_sideband_mgr_query(SBM_DSPT_SNOC);
|
||
+ status |= win2030_noc_sideband_mgr_query(SBM_CNOC_DSPT);
|
||
+ if (0 != status) {
|
||
+ break;
|
||
+ }
|
||
+ schedule();
|
||
+ } while (time_before(jiffies, deadline));
|
||
+
|
||
+ return status;
|
||
+}
|
||
+
|
||
+static inline int dsp_subsys_clk_init(struct platform_device *pdev,
|
||
+ struct es_dsp_subsys *subsys)
|
||
+{
|
||
+ int ret;
|
||
+
|
||
+ subsys->cfg_clk = devm_clk_get(&pdev->dev, "cfg_clk");
|
||
+ if (IS_ERR(subsys->cfg_clk)) {
|
||
+ ret = PTR_ERR(subsys->cfg_clk);
|
||
+ dev_err(&pdev->dev, "failed to get cfg_clk: %d\n", ret);
|
||
+ return ret;
|
||
+ }
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_reset(struct es_dsp_subsys *subsys)
|
||
+{
|
||
+ int ret;
|
||
+
|
||
+ /*reset dsp bus*/
|
||
+ ret = reset_control_reset(subsys->rstc_axi);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ ret = reset_control_reset(subsys->rstc_div4);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ /*reset dsp cfg*/
|
||
+ ret = reset_control_reset(subsys->rstc_cfg);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ /*reset dsp core clk div*/
|
||
+ ret = reset_control_reset(subsys->rstc_div_0);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ ret = reset_control_reset(subsys->rstc_div_1);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ ret = reset_control_reset(subsys->rstc_div_2);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ ret = reset_control_reset(subsys->rstc_div_3);
|
||
+ WARN_ON(0 != ret);
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_clk_enable(struct es_dsp_subsys *subsys)
|
||
+{
|
||
+ int ret;
|
||
+
|
||
+ ret = clk_prepare_enable(subsys->cfg_clk);
|
||
+ if (ret) {
|
||
+ dev_err(&subsys->pdev->dev, "failed to enable cfg_clk: %d\n", ret);
|
||
+ return ret;
|
||
+ }
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_reset_init(struct platform_device *pdev,
|
||
+ struct es_dsp_subsys *subsys)
|
||
+{
|
||
+ subsys->rstc_axi = devm_reset_control_get_optional(&pdev->dev, "axi");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_axi)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to get axi reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+
|
||
+ subsys->rstc_div4 = devm_reset_control_get_optional(&pdev->dev, "div4");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_div4)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to div4 reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+
|
||
+ subsys->rstc_cfg = devm_reset_control_get_optional(&pdev->dev, "cfg");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_cfg)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to get cfg reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+
|
||
+ subsys->rstc_div_0 = devm_reset_control_get_optional(&pdev->dev, "div_0");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_div_0)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to div_0 reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+ subsys->rstc_div_1 = devm_reset_control_get_optional(&pdev->dev, "div_1");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_div_1)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to div_1 reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+ subsys->rstc_div_2 = devm_reset_control_get_optional(&pdev->dev, "div_2");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_div_2)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to div_2 reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+ subsys->rstc_div_3 = devm_reset_control_get_optional(&pdev->dev, "div_3");
|
||
+ if (IS_ERR_OR_NULL(subsys->rstc_div_3)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to div_3 reset handle\n");
|
||
+ return -EFAULT;
|
||
+ }
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_reg_read(void *context, unsigned int reg,
|
||
+ unsigned int *val)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys = context;
|
||
+
|
||
+ *val = readl_relaxed(subsys->reg_base + reg);
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_reg_write(void *context, unsigned int reg,
|
||
+ unsigned int val)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys = context;
|
||
+
|
||
+ writel_relaxed(val, subsys->reg_base + reg);
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_con_reg_read(void *context, unsigned int reg,
|
||
+ unsigned int *val)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys = context;
|
||
+
|
||
+ *val = readl_relaxed(subsys->con_reg_base + reg);
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int dsp_subsys_con_reg_write(void *context, unsigned int reg,
|
||
+ unsigned int val)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys = context;
|
||
+
|
||
+ writel_relaxed(val, subsys->con_reg_base + reg);
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+/**
|
||
+ * dsp_subsys_init_regmap() - Initialize registers map
|
||
+ *
|
||
+ * Autodetects needed register access mode and creates the regmap with
|
||
+ * corresponding read/write callbacks. This must be called before doing any
|
||
+ * other register access.
|
||
+ */
|
||
+static int dsp_subsys_init_regmap(struct es_dsp_subsys *subsys)
|
||
+{
|
||
+ struct regmap_config map_cfg = {
|
||
+ .reg_bits = 32,
|
||
+ .val_bits = 32,
|
||
+ .reg_stride = 4,
|
||
+ .use_hwlock = true,
|
||
+ .cache_type = REGCACHE_NONE,
|
||
+ .can_sleep = false,
|
||
+ .reg_read = dsp_subsys_reg_read,
|
||
+ .reg_write = dsp_subsys_reg_write,
|
||
+ };
|
||
+ struct regmap_config con_map_cfg = {
|
||
+ .reg_bits = 32,
|
||
+ .val_bits = 32,
|
||
+ .reg_stride = 4,
|
||
+ .use_hwlock = true,
|
||
+ .cache_type = REGCACHE_NONE,
|
||
+ .can_sleep = false,
|
||
+ .reg_read = dsp_subsys_con_reg_read,
|
||
+ .reg_write = dsp_subsys_con_reg_write,
|
||
+ };
|
||
+
|
||
+ /*
|
||
+ * Note we'll check the return value of the regmap IO accessors only
|
||
+ * at the probe stage. The rest of the code won't do this because
|
||
+ * basically we have MMIO-based regmap so non of the read/write methods
|
||
+ * can fail.
|
||
+ */
|
||
+ subsys->map = devm_regmap_init(&subsys->pdev->dev, NULL, subsys, &map_cfg);
|
||
+ if (IS_ERR(subsys->map)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to init the registers map\n");
|
||
+ return PTR_ERR(subsys->map);
|
||
+ }
|
||
+
|
||
+ subsys->con_map = devm_regmap_init(&subsys->pdev->dev, NULL, subsys, &con_map_cfg);
|
||
+ if (IS_ERR(subsys->con_map)) {
|
||
+ dev_err(&subsys->pdev->dev, "Failed to init the con registers map\n");
|
||
+ return PTR_ERR(subsys->con_map);
|
||
+ }
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int es_dsp_subsys_probe(struct platform_device *pdev)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys;
|
||
+ int ret;
|
||
+ struct resource *res;
|
||
+
|
||
+ dev_info(&pdev->dev, "%s\n", __func__);
|
||
+ subsys = devm_kzalloc(&pdev->dev, sizeof(*subsys), GFP_KERNEL);
|
||
+ if (!subsys) {
|
||
+ return -ENOMEM;
|
||
+ }
|
||
+
|
||
+ dev_set_drvdata(&pdev->dev, subsys);
|
||
+
|
||
+ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||
+ if (!res)
|
||
+ return -ENODEV;
|
||
+
|
||
+ subsys->reg_base = devm_ioremap_resource(&pdev->dev, res);
|
||
+ if (IS_ERR_OR_NULL(subsys->reg_base)) {
|
||
+ return PTR_ERR(subsys->reg_base);
|
||
+ }
|
||
+
|
||
+ res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
|
||
+ if (!res)
|
||
+ return -ENODEV;
|
||
+
|
||
+ subsys->con_reg_base = devm_ioremap_resource(&pdev->dev, res);
|
||
+ if (IS_ERR_OR_NULL(subsys->con_reg_base)) {
|
||
+ return PTR_ERR(subsys->con_reg_base);
|
||
+ }
|
||
+
|
||
+ subsys->pdev = pdev;
|
||
+
|
||
+ ret = dsp_subsys_init_regmap(subsys);
|
||
+ if (0 != ret) {
|
||
+ return -ENODEV;
|
||
+ }
|
||
+
|
||
+ ret = dsp_subsys_reset_init(pdev, subsys);
|
||
+ if (0 != ret) {
|
||
+ return ret;
|
||
+ }
|
||
+
|
||
+ ret = dsp_subsys_clk_init(pdev, subsys);
|
||
+ if (0 != ret) {
|
||
+ return ret;
|
||
+ }
|
||
+
|
||
+ ret = dsp_subsys_clk_enable(subsys);
|
||
+ if (0 != ret) {
|
||
+ return ret;
|
||
+ }
|
||
+
|
||
+ ret = dsp_subsys_reset(subsys);
|
||
+ if (0 != ret) {
|
||
+ return ret;
|
||
+ }
|
||
+
|
||
+ subsys->dsp_subsys_status = dsp_subsys_status;
|
||
+
|
||
+ /* enable qos */
|
||
+ // win2030_noc_qos_set("DSPT");
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+static int es_dsp_subsys_remove(struct platform_device *pdev)
|
||
+{
|
||
+ struct es_dsp_subsys *subsys = dev_get_drvdata(&pdev->dev);
|
||
+ if (subsys) {
|
||
+ clk_disable_unprepare(subsys->cfg_clk);
|
||
+ }
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+
|
||
+#ifdef CONFIG_OF
|
||
+static const struct of_device_id es_dsp_subsys_match[] = {
|
||
+ {
|
||
+ .compatible = "es-dsp-subsys",
|
||
+ },
|
||
+ {},
|
||
+};
|
||
+MODULE_DEVICE_TABLE(of, es_dsp_subsys_match);
|
||
+#endif
|
||
+
|
||
+static struct platform_driver es_dsp_subsys_driver = {
|
||
+ .probe = es_dsp_subsys_probe,
|
||
+ .remove = es_dsp_subsys_remove,
|
||
+ .driver = {
|
||
+ .name = DRIVER_NAME,
|
||
+ .of_match_table = of_match_ptr(es_dsp_subsys_match),
|
||
+ },
|
||
+};
|
||
+
|
||
+module_platform_driver(es_dsp_subsys_driver);
|
||
+
|
||
+MODULE_AUTHOR("Eswin");
|
||
+MODULE_DESCRIPTION("DSP: Low Level Device Driver For Eswin DSP");
|
||
+MODULE_LICENSE("Dual MIT/GPL");
|
||
diff --git a/drivers/soc/eswin/eswin-dsp-subsys.h b/drivers/soc/eswin/eswin-dsp-subsys.h
|
||
new file mode 100644
|
||
index 000000000000..4933411fde94
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/eswin-dsp-subsys.h
|
||
@@ -0,0 +1,44 @@
|
||
+/*
|
||
+ * Program's name, and a brief idea of what it does(One line).
|
||
+ * Copyright 20XX, Beijing ESWIN Computing Technology Co., Ltd.. All rights reserved.
|
||
+ * SPDX-License-Identifier: GPL-2.0-only
|
||
+ *
|
||
+ * This program is free software: you can redistribute it and/or modify
|
||
+ * it under the terms of the GNU General Public License as published by
|
||
+ * the Free Software Foundation, version 2.
|
||
+ *
|
||
+ * This program is distributed in the hope that it will be useful,
|
||
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
+ * GNU General Public License for more details.
|
||
+ *
|
||
+ * You should have received a copy of the GNU General Public License
|
||
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
+ */
|
||
+
|
||
+#ifndef __ESWIN_DSP_SUBSYS_H__
|
||
+#define __ESWIN_DSP_SUBSYS_H__
|
||
+#include <linux/reset.h>
|
||
+#include <linux/regmap.h>
|
||
+#include <linux/platform_device.h>
|
||
+
|
||
+typedef int (*dsp_subsys_status_pfunc)(void);
|
||
+
|
||
+struct es_dsp_subsys {
|
||
+ void __iomem *reg_base;
|
||
+ void __iomem *con_reg_base;
|
||
+ struct regmap *map;
|
||
+ struct regmap *con_map;
|
||
+ struct platform_device *pdev;
|
||
+
|
||
+ struct reset_control *rstc_axi;
|
||
+ struct reset_control *rstc_cfg;
|
||
+ struct reset_control *rstc_div4;
|
||
+ struct reset_control *rstc_div_0;
|
||
+ struct reset_control *rstc_div_1;
|
||
+ struct reset_control *rstc_div_2;
|
||
+ struct reset_control *rstc_div_3;
|
||
+ struct clk *cfg_clk;
|
||
+ dsp_subsys_status_pfunc dsp_subsys_status;
|
||
+};
|
||
+#endif
|
||
\ No newline at end of file
|
||
diff --git a/drivers/soc/eswin/eswin-khandle.c b/drivers/soc/eswin/eswin-khandle.c
|
||
new file mode 100644
|
||
index 000000000000..055615a17b7a
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/eswin-khandle.c
|
||
@@ -0,0 +1,242 @@
|
||
+// SPDX-License-Identifier: GPL-2.0
|
||
+
|
||
+#include "eswin-khandle.h"
|
||
+#include <linux/vmalloc.h>
|
||
+#include <asm/atomic.h>
|
||
+#include <linux/xarray.h>
|
||
+
|
||
+struct fd_pool_desc {
|
||
+ spinlock_t lock;
|
||
+ struct xarray fd_array;
|
||
+};
|
||
+
|
||
+static void init_fd_pool(void *fd_pool)
|
||
+{
|
||
+ struct fd_pool_desc *pool = (struct fd_pool_desc *)fd_pool;
|
||
+ pr_debug("%s, %d, pool=0x%px", __func__, __LINE__, pool);
|
||
+ xa_init_flags(&pool->fd_array, XA_FLAGS_ALLOC);
|
||
+ spin_lock_init(&pool->lock);
|
||
+}
|
||
+
|
||
+static int alloc_fd(void *fd_pool, struct khandle *h)
|
||
+{
|
||
+ struct fd_pool_desc *pool = (struct fd_pool_desc *)fd_pool;
|
||
+ int ret;
|
||
+ u32 fd;
|
||
+
|
||
+ pr_debug("%s, %d, pool=0x%px", __func__, __LINE__, pool);
|
||
+ ret = xa_alloc(&pool->fd_array, &fd, h, xa_limit_32b, GFP_ATOMIC);
|
||
+ if (ret < 0) {
|
||
+ pr_err("%s, %d, ret=%d.\n", __func__, __LINE__, ret);
|
||
+ return ret;
|
||
+ }
|
||
+ pr_debug("%s, %d, pool=0x%px, fd=%d.\n", __func__, __LINE__, pool, fd);
|
||
+ return fd;
|
||
+}
|
||
+
|
||
+static void release_fd(void *fd_pool, int fd)
|
||
+{
|
||
+ unsigned long flags;
|
||
+ struct khandle *h;
|
||
+ struct fd_pool_desc *pool = (struct fd_pool_desc *)fd_pool;
|
||
+
|
||
+ pr_debug("%s, %d, pool=0x%px, fd=%d.\n", __func__, __LINE__, pool, fd);
|
||
+
|
||
+ spin_lock_irqsave(&pool->lock, flags);
|
||
+ h = xa_load(&pool->fd_array, fd);
|
||
+ if (!h) {
|
||
+ spin_unlock_irqrestore(&pool->lock, flags);
|
||
+ return;
|
||
+ }
|
||
+ xa_erase(&pool->fd_array, fd);
|
||
+ spin_unlock_irqrestore(&pool->lock, flags);
|
||
+}
|
||
+
|
||
+static struct khandle *find_khandle_by_fd(void *fd_pool, int fd)
|
||
+{
|
||
+ unsigned long flags;
|
||
+ struct khandle *h;
|
||
+ struct fd_pool_desc *pool = (struct fd_pool_desc *)fd_pool;
|
||
+
|
||
+ spin_lock_irqsave(&pool->lock, flags);
|
||
+ h = xa_load(&pool->fd_array, fd);
|
||
+ if (h == NULL) {
|
||
+ spin_unlock_irqrestore(&pool->lock, flags);
|
||
+ return NULL;
|
||
+ }
|
||
+ kref_get(&h->refcount);
|
||
+ spin_unlock_irqrestore(&pool->lock, flags);
|
||
+ return h;
|
||
+}
|
||
+
|
||
+static void kref_khandle_fn(struct kref *kref)
|
||
+{
|
||
+ unsigned long flags;
|
||
+ struct khandle *h = container_of(kref, struct khandle, refcount);
|
||
+ struct khandle *parent;
|
||
+ struct fd_pool_desc *pool;
|
||
+
|
||
+ pr_debug("%s, h address=0x%px.\n", __func__, h);
|
||
+ BUG_ON(h == NULL);
|
||
+ BUG_ON(h->fd != INVALID_HANDLE_VALUE);
|
||
+
|
||
+ pr_debug("%s, k->fd=%d, refcount=%d.\n", __func__, h->fd,
|
||
+ kref_read(kref));
|
||
+
|
||
+ parent = h->parent;
|
||
+
|
||
+ if (parent == NULL) {
|
||
+ pool = h->fd_pool;
|
||
+ xa_destroy(&pool->fd_array);
|
||
+ vfree(h->fd_pool);
|
||
+ } else {
|
||
+ spin_lock_irqsave(&parent->lock, flags);
|
||
+ list_del_init(&h->entry);
|
||
+ spin_unlock_irqrestore(&parent->lock, flags);
|
||
+ }
|
||
+
|
||
+ if (h->fn != NULL) {
|
||
+ h->fn(h);
|
||
+ }
|
||
+
|
||
+ if (parent != NULL) {
|
||
+ kref_put(&parent->refcount, kref_khandle_fn);
|
||
+ }
|
||
+}
|
||
+
|
||
+void kernel_handle_addref(struct khandle *h)
|
||
+{
|
||
+ BUG_ON(h == NULL);
|
||
+
|
||
+ kref_get(&h->refcount);
|
||
+ pr_debug("%s, h addr=0x%px, fd=%d, refcount=%d.\n", __func__, h, h->fd,
|
||
+ kref_read(&h->refcount));
|
||
+}
|
||
+EXPORT_SYMBOL(kernel_handle_addref);
|
||
+
|
||
+void kernel_handle_decref(struct khandle *h)
|
||
+{
|
||
+ BUG_ON(h == NULL);
|
||
+
|
||
+ kref_put(&h->refcount, kref_khandle_fn);
|
||
+ pr_debug("%s, done.\n", __func__);
|
||
+}
|
||
+EXPORT_SYMBOL(kernel_handle_decref);
|
||
+
|
||
+static struct list_head *capture_next_khandle_node(struct list_head *head,
|
||
+ struct list_head *cur)
|
||
+{
|
||
+ struct khandle *h;
|
||
+
|
||
+ while (true) {
|
||
+ cur = cur->next;
|
||
+ if (cur == head) {
|
||
+ return cur;
|
||
+ }
|
||
+
|
||
+ /* Protect child not released until return of kernel_handle_release_family. */
|
||
+ h = container_of(cur, struct khandle, entry);
|
||
+ if (kref_get_unless_zero(&h->refcount) != 0) {
|
||
+ return cur;
|
||
+ }
|
||
+ }
|
||
+}
|
||
+
|
||
+void kernel_handle_release_family(struct khandle *h)
|
||
+{
|
||
+ unsigned long flags;
|
||
+ struct list_head *child;
|
||
+ struct khandle *child_khandle;
|
||
+
|
||
+ BUG_ON(h == NULL);
|
||
+ spin_lock_irqsave(&h->lock, flags);
|
||
+ if (h->fd == INVALID_HANDLE_VALUE) {
|
||
+ spin_unlock_irqrestore(&h->lock, flags);
|
||
+ return;
|
||
+ }
|
||
+
|
||
+ release_fd(h->fd_pool, h->fd);
|
||
+ h->fd = INVALID_HANDLE_VALUE;
|
||
+ child = capture_next_khandle_node(&h->head, &h->head);
|
||
+ while (child != &h->head) {
|
||
+ child_khandle = container_of(child, struct khandle, entry);
|
||
+ child = capture_next_khandle_node(&h->head, child);
|
||
+ spin_unlock_irqrestore(&h->lock, flags);
|
||
+ kernel_handle_release_family(child_khandle);
|
||
+ kernel_handle_decref(child_khandle);
|
||
+ spin_lock_irqsave(&h->lock, flags);
|
||
+ }
|
||
+
|
||
+ spin_unlock_irqrestore(&h->lock, flags);
|
||
+ kref_put(&h->refcount, kref_khandle_fn);
|
||
+ pr_debug("%s, done.\n", __func__);
|
||
+}
|
||
+EXPORT_SYMBOL(kernel_handle_release_family);
|
||
+
|
||
+int init_kernel_handle(struct khandle *h, release_khandle_fn fn, int magic,
|
||
+ struct khandle *parent)
|
||
+{
|
||
+ unsigned long flags;
|
||
+ void *fd_pool;
|
||
+
|
||
+ BUG_ON(h == NULL);
|
||
+ kref_init(&h->refcount);
|
||
+ kref_get(&h->refcount);
|
||
+
|
||
+ if ((h->parent = parent) == NULL) {
|
||
+ fd_pool = vmalloc(sizeof(struct fd_pool_desc));
|
||
+ init_fd_pool(fd_pool);
|
||
+ if (fd_pool == NULL) {
|
||
+ return -ENOMEM;
|
||
+ }
|
||
+ } else {
|
||
+ fd_pool = parent->fd_pool;
|
||
+ }
|
||
+ h->fd_pool = fd_pool;
|
||
+
|
||
+ if ((h->fd = alloc_fd(fd_pool, h)) == INVALID_HANDLE_VALUE) {
|
||
+ BUG_ON(parent == NULL);
|
||
+ return -EINVAL;
|
||
+ }
|
||
+
|
||
+ pr_debug("%s, hfile addr=%u.\n", __func__, h->fd);
|
||
+ h->fn = fn;
|
||
+ h->magic = magic;
|
||
+ spin_lock_init(&h->lock);
|
||
+
|
||
+ INIT_LIST_HEAD(&h->head);
|
||
+ INIT_LIST_HEAD(&h->entry);
|
||
+
|
||
+ if (parent != NULL) {
|
||
+ spin_lock_irqsave(&parent->lock, flags);
|
||
+ if (parent->fd == INVALID_HANDLE_VALUE) {
|
||
+ spin_unlock_irqrestore(&parent->lock, flags);
|
||
+ release_fd(fd_pool, h->fd);
|
||
+ return -EINVAL;
|
||
+ }
|
||
+
|
||
+ list_add_tail(&h->entry, &parent->head);
|
||
+ kref_get(&parent->refcount);
|
||
+ spin_unlock_irqrestore(&parent->lock, flags);
|
||
+ }
|
||
+
|
||
+ return 0;
|
||
+}
|
||
+EXPORT_SYMBOL(init_kernel_handle);
|
||
+
|
||
+struct khandle *find_kernel_handle(struct khandle *ancestor, int fd, int magic)
|
||
+{
|
||
+ struct khandle *h;
|
||
+
|
||
+ h = find_khandle_by_fd(ancestor->fd_pool, fd);
|
||
+ if (h == NULL) {
|
||
+ return NULL;
|
||
+ }
|
||
+
|
||
+ if (h->magic != magic) {
|
||
+ kref_put(&h->refcount, kref_khandle_fn);
|
||
+ return NULL;
|
||
+ }
|
||
+ return h;
|
||
+}
|
||
+EXPORT_SYMBOL(find_kernel_handle);
|
||
diff --git a/drivers/soc/eswin/eswin-khandle.h b/drivers/soc/eswin/eswin-khandle.h
|
||
new file mode 100644
|
||
index 000000000000..7e086388890a
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/eswin-khandle.h
|
||
@@ -0,0 +1,84 @@
|
||
+/* SPDX-License-Identifier: GPL-2.0 */
|
||
+#ifndef __ESWIN_KHANDLE_H_
|
||
+#define __ESWIN_KHANDLE_H_
|
||
+
|
||
+#include <linux/kref.h>
|
||
+#include <linux/types.h>
|
||
+#include <linux/list.h>
|
||
+#include <linux/mutex.h>
|
||
+
|
||
+#define INVALID_HANDLE_VALUE (-1)
|
||
+
|
||
+struct khandle;
|
||
+typedef void (*release_khandle_fn)(struct khandle *h);
|
||
+struct khandle {
|
||
+ int fd;
|
||
+ int magic;
|
||
+ spinlock_t lock;
|
||
+ release_khandle_fn fn;
|
||
+ struct kref refcount;
|
||
+ struct list_head entry;
|
||
+ struct list_head head;
|
||
+ struct khandle *parent;
|
||
+ void *fd_pool;
|
||
+};
|
||
+
|
||
+/**
|
||
+ * @brief Remove the family relations hierachy. This function also actively
|
||
+ * free the fd of this kernel object and its descendants.
|
||
+ *
|
||
+ * @param o: This is the kernel object.
|
||
+ */
|
||
+void kernel_handle_release_family(struct khandle *o);
|
||
+
|
||
+/**
|
||
+ * @brief Decrease the reference of kernel object `o`. If reference reaches 0,
|
||
+ * the release delegation function is called.
|
||
+ *
|
||
+ * @param o: This is the kernel object.
|
||
+ */
|
||
+void kernel_handle_decref(struct khandle *o);
|
||
+
|
||
+
|
||
+/**
|
||
+ * @brief Increase the reference of kernel object `o`.
|
||
+ *
|
||
+ * @param o: This is the kernel object.
|
||
+ */
|
||
+void kernel_handle_addref(struct khandle *o);
|
||
+
|
||
+
|
||
+/**
|
||
+ * @brief This function intialize an kernel object in the memory specified by
|
||
+ * `o`. It returns zero on success or a Linux error code. Note this function
|
||
+ * should only be called in IOCtl context. The initial reference is set to 1.
|
||
+ *
|
||
+ * @param o: This specifies an memory for holding kernel object.
|
||
+ * @param fn: This points to a callback delegation function. When the
|
||
+ * reference of `o` reaches 0, this callback function is called. It
|
||
+ * is intended for releasing resources associated with this kernel
|
||
+ * object.
|
||
+ * @param magic: This is a magic number for determining the type of kernel
|
||
+ * object.
|
||
+ * @param parent: Points to the parent of this kernel object.
|
||
+ * @return It returns zero on success or a Linux error code.
|
||
+ *
|
||
+ * when use khandle, host structure release must use kernel_handle_decref function.
|
||
+ */
|
||
+int init_kernel_handle(struct khandle *o, release_khandle_fn fn, int magic,
|
||
+ struct khandle *parent);
|
||
+
|
||
+
|
||
+/**
|
||
+ * @brief This function is used to find the kernel object associated with fd.
|
||
+ * Note the khandle object has one additional reference so user should dereference
|
||
+ * it if not needed.
|
||
+ *
|
||
+ * @param ancestor: This is one ancestor of kernel object that matches fd.
|
||
+ * @param fd: This is the fd associated with a specific kernel object.
|
||
+ * @param magic: This is the magic associated with a specific kernel object.
|
||
+ * @return It returns the kernel object on success or NULL if the given fd
|
||
+ * is invalid.
|
||
+ */
|
||
+struct khandle *find_kernel_handle(struct khandle *ancestor, int fd, int magic);
|
||
+#endif
|
||
diff --git a/drivers/soc/eswin/eswin_timer.h b/drivers/soc/eswin/eswin_timer.h
|
||
new file mode 100644
|
||
index 000000000000..a7fac84ac613
|
||
--- /dev/null
|
||
+++ b/drivers/soc/eswin/eswin_timer.h
|
||
@@ -0,0 +1,6 @@
|
||
+#ifndef __ESWIN_TIMER_H_
|
||
+#define __ESWIN_TIMER_H_
|
||
+
|
||
+extern u32 get_perf_timer_cnt(void);
|
||
+
|
||
+#endif
|
||
--
|
||
2.47.0
|
||
|