lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20250701-95_cam-v1-4-c5172bab387b@nxp.com>
Date: Tue, 01 Jul 2025 18:06:09 -0400
From: Frank Li <Frank.Li@....com>
To: Rui Miguel Silva <rmfrfs@...il.com>, 
 Laurent Pinchart <laurent.pinchart@...asonboard.com>, 
 Martin Kepplinger <martink@...teo.de>, Purism Kernel Team <kernel@...i.sm>, 
 Mauro Carvalho Chehab <mchehab@...nel.org>, Rob Herring <robh@...nel.org>, 
 Krzysztof Kozlowski <krzk+dt@...nel.org>, 
 Conor Dooley <conor+dt@...nel.org>, 
 Eugen Hristev <eugen.hristev@...aro.org>, Shawn Guo <shawnguo@...nel.org>, 
 Sascha Hauer <s.hauer@...gutronix.de>, 
 Pengutronix Kernel Team <kernel@...gutronix.de>, 
 Fabio Estevam <festevam@...il.com>, Peng Fan <peng.fan@....com>, 
 Alice Yuan <alice.yuan@....com>, Vinod Koul <vkoul@...nel.org>, 
 Kishon Vijay Abraham I <kishon@...nel.org>, 
 Philipp Zabel <p.zabel@...gutronix.de>
Cc: linux-media@...r.kernel.org, devicetree@...r.kernel.org, 
 linux-kernel@...r.kernel.org, imx@...ts.linux.dev, 
 linux-arm-kernel@...ts.infradead.org, linux-phy@...ts.infradead.org, 
 Frank Li <Frank.Li@....com>, "Guoniu.zhou" <guoniu.zhou@....com>, 
 Jindong Yue <jindong.yue@....com>
Subject: [PATCH 4/7] phy: freescale: add imx93 MIPI CSI2 DPHY support

Add driver i.MX93 MIPI DPHY controller, which is wrapper for Synosys MIPI
CSI2 DPHY module.

Base on
https://github.com/nxp-imx/linux-imx/blob/lf-6.12.y/drivers/phy/freescale/phy-fsl-imx9-dphy-rx.c

Signed-off-by: Guoniu.zhou <guoniu.zhou@....com>
Signed-off-by: Jindong Yue <jindong.yue@....com>
Signed-off-by: Frank Li <Frank.Li@....com>
---
 drivers/phy/freescale/Kconfig                 |  10 +
 drivers/phy/freescale/Makefile                |   1 +
 drivers/phy/freescale/phy-fsl-imx93-dphy-rx.c | 306 ++++++++++++++++++++++++++
 3 files changed, 317 insertions(+)

diff --git a/drivers/phy/freescale/Kconfig b/drivers/phy/freescale/Kconfig
index 81f53564ee156..cb34e151e86c4 100644
--- a/drivers/phy/freescale/Kconfig
+++ b/drivers/phy/freescale/Kconfig
@@ -44,6 +44,16 @@ config PHY_FSL_IMX8QM_HSIO
 	  Enable this to add support for the HSIO PHY as found on
 	  i.MX8QM family of SOCs.
 
+config PHY_FSL_IMX93_DPHY_RX
+	tristate "Freescale i.MX9 DPHY Rx"
+	depends on OF && HAS_IOMEM
+	select GENERIC_PHY
+	select GENERIC_PHY_MIPI_DPHY
+	select REGMAP_MMIO
+	help
+	  Enable this to add support for the Synopsys DW DPHY Rx as found
+	  on NXP's i.MX9 family.
+
 config PHY_FSL_SAMSUNG_HDMI_PHY
 	tristate "Samsung HDMI PHY support"
 	depends on OF && HAS_IOMEM && COMMON_CLK
diff --git a/drivers/phy/freescale/Makefile b/drivers/phy/freescale/Makefile
index 658eac7d0a622..8e122a07695f0 100644
--- a/drivers/phy/freescale/Makefile
+++ b/drivers/phy/freescale/Makefile
@@ -4,5 +4,6 @@ obj-$(CONFIG_PHY_MIXEL_LVDS_PHY)	+= phy-fsl-imx8qm-lvds-phy.o
 obj-$(CONFIG_PHY_MIXEL_MIPI_DPHY)	+= phy-fsl-imx8-mipi-dphy.o
 obj-$(CONFIG_PHY_FSL_IMX8M_PCIE)	+= phy-fsl-imx8m-pcie.o
 obj-$(CONFIG_PHY_FSL_IMX8QM_HSIO)	+= phy-fsl-imx8qm-hsio.o
+obj-$(CONFIG_PHY_FSL_IMX93_DPHY_RX)	+= phy-fsl-imx93-dphy-rx.o
 obj-$(CONFIG_PHY_FSL_LYNX_28G)		+= phy-fsl-lynx-28g.o
 obj-$(CONFIG_PHY_FSL_SAMSUNG_HDMI_PHY)	+= phy-fsl-samsung-hdmi.o
diff --git a/drivers/phy/freescale/phy-fsl-imx93-dphy-rx.c b/drivers/phy/freescale/phy-fsl-imx93-dphy-rx.c
new file mode 100644
index 0000000000000..f5155ae68c50f
--- /dev/null
+++ b/drivers/phy/freescale/phy-fsl-imx93-dphy-rx.c
@@ -0,0 +1,306 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright 2025 NXP
+ */
+
+#include <linux/bitfield.h>
+#include <linux/bits.h>
+#include <linux/clk.h>
+#include <linux/delay.h>
+#include <linux/mfd/syscon.h>
+#include <linux/module.h>
+#include <linux/of_platform.h>
+#include <linux/phy/phy.h>
+#include <linux/phy/phy-mipi-dphy.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+
+#define IMX93_BLK_CSI			0x48
+#define   IMX93_BLK_CSI_CFGCLKFREQRANGE	GENMASK(5, 0)
+#define   IMX93_BLK_CSI_HSFREQRANGE	GENMASK(14, 8)
+
+struct fsl_csi2_phy_drv_data {
+	u32 max_lanes;
+	u32 max_data_rate; /* Mbps */
+};
+
+struct fsl_csi2_phy {
+	struct device *dev;
+	struct regmap *dphy_regmap;
+	struct clk *cfg_clk;
+
+	const struct fsl_csi2_phy_drv_data *drv_data;
+
+	u16 hsfreqrange;
+	u16 cfgclkfreqrange;
+	u16 ddlfreq;
+};
+
+struct dphy_mbps_hsfreqrange_map {
+	u16 mbps;
+	u16 hsfreqrange;
+	u16 ddlfreq;
+};
+
+/*
+ * Data rate to high speed frequency range map table
+ */
+static const struct dphy_mbps_hsfreqrange_map hsfreqrange_table[] = {
+	{ .mbps = 80,  .hsfreqrange = 0x00, .ddlfreq = 489 },
+	{ .mbps = 90,  .hsfreqrange = 0x10, .ddlfreq = 489 },
+	{ .mbps = 100, .hsfreqrange = 0x20, .ddlfreq = 489 },
+	{ .mbps = 110, .hsfreqrange = 0x30, .ddlfreq = 489 },
+	{ .mbps = 120, .hsfreqrange = 0x01, .ddlfreq = 489 },
+	{ .mbps = 130, .hsfreqrange = 0x11, .ddlfreq = 489 },
+	{ .mbps = 140, .hsfreqrange = 0x21, .ddlfreq = 489 },
+	{ .mbps = 150, .hsfreqrange = 0x31, .ddlfreq = 489 },
+	{ .mbps = 160, .hsfreqrange = 0x02, .ddlfreq = 489 },
+	{ .mbps = 170, .hsfreqrange = 0x12, .ddlfreq = 489 },
+	{ .mbps = 180, .hsfreqrange = 0x22, .ddlfreq = 489 },
+	{ .mbps = 190, .hsfreqrange = 0x32, .ddlfreq = 489 },
+	{ .mbps = 205, .hsfreqrange = 0x03, .ddlfreq = 489 },
+	{ .mbps = 220, .hsfreqrange = 0x13, .ddlfreq = 489 },
+	{ .mbps = 235, .hsfreqrange = 0x23, .ddlfreq = 489 },
+	{ .mbps = 250, .hsfreqrange = 0x33, .ddlfreq = 489 },
+	{ .mbps = 275, .hsfreqrange = 0x04, .ddlfreq = 489 },
+	{ .mbps = 300, .hsfreqrange = 0x14, .ddlfreq = 489 },
+	{ .mbps = 325, .hsfreqrange = 0x25, .ddlfreq = 489 },
+	{ .mbps = 350, .hsfreqrange = 0x35, .ddlfreq = 489 },
+	{ .mbps = 400, .hsfreqrange = 0x05, .ddlfreq = 489 },
+	{ .mbps = 450, .hsfreqrange = 0x16, .ddlfreq = 489 },
+	{ .mbps = 500, .hsfreqrange = 0x26, .ddlfreq = 489 },
+	{ .mbps = 550, .hsfreqrange = 0x37, .ddlfreq = 489 },
+	{ .mbps = 600, .hsfreqrange = 0x07, .ddlfreq = 489 },
+	{ .mbps = 650, .hsfreqrange = 0x18, .ddlfreq = 489 },
+	{ .mbps = 700, .hsfreqrange = 0x28, .ddlfreq = 489 },
+	{ .mbps = 750, .hsfreqrange = 0x39, .ddlfreq = 489 },
+	{ .mbps = 800, .hsfreqrange = 0x09, .ddlfreq = 489 },
+	{ .mbps = 850, .hsfreqrange = 0x19, .ddlfreq = 489 },
+	{ .mbps = 900, .hsfreqrange = 0x29, .ddlfreq = 489 },
+	{ .mbps = 950, .hsfreqrange = 0x3a, .ddlfreq = 489 },
+	{ .mbps = 1000, .hsfreqrange = 0x0a, .ddlfreq = 489 },
+	{ .mbps = 1050, .hsfreqrange = 0x1a, .ddlfreq = 489 },
+	{ .mbps = 1100, .hsfreqrange = 0x2a, .ddlfreq = 489 },
+	{ .mbps = 1150, .hsfreqrange = 0x3b, .ddlfreq = 489 },
+	{ .mbps = 1200, .hsfreqrange = 0x0b, .ddlfreq = 489 },
+	{ .mbps = 1250, .hsfreqrange = 0x1b, .ddlfreq = 489 },
+	{ .mbps = 1300, .hsfreqrange = 0x2b, .ddlfreq = 489 },
+	{ .mbps = 1350, .hsfreqrange = 0x3c, .ddlfreq = 489 },
+	{ .mbps = 1400, .hsfreqrange = 0x0c, .ddlfreq = 489 },
+	{ .mbps = 1450, .hsfreqrange = 0x1c, .ddlfreq = 489 },
+	{ .mbps = 1500, .hsfreqrange = 0x2c, .ddlfreq = 489 },
+	{ .mbps = 1550, .hsfreqrange = 0x3d, .ddlfreq = 303 },
+	{ .mbps = 1600, .hsfreqrange = 0x0d, .ddlfreq = 313 },
+	{ .mbps = 1650, .hsfreqrange = 0x1d, .ddlfreq = 323 },
+	{ .mbps = 1700, .hsfreqrange = 0x2e, .ddlfreq = 333 },
+	{ .mbps = 1750, .hsfreqrange = 0x3e, .ddlfreq = 342 },
+	{ .mbps = 1800, .hsfreqrange = 0x0e, .ddlfreq = 352 },
+	{ .mbps = 1850, .hsfreqrange = 0x1e, .ddlfreq = 362 },
+	{ .mbps = 1900, .hsfreqrange = 0x1f, .ddlfreq = 372 },
+	{ .mbps = 1950, .hsfreqrange = 0x3f, .ddlfreq = 381 },
+	{ .mbps = 2000, .hsfreqrange = 0x0f, .ddlfreq = 391 },
+	{ .mbps = 2050, .hsfreqrange = 0x40, .ddlfreq = 401 },
+	{ .mbps = 2100, .hsfreqrange = 0x41, .ddlfreq = 411 },
+	{ .mbps = 2150, .hsfreqrange = 0x42, .ddlfreq = 411 },
+	{ .mbps = 2200, .hsfreqrange = 0x43, .ddlfreq = 411 },
+	{ .mbps = 2250, .hsfreqrange = 0x44, .ddlfreq = 411 },
+	{ .mbps = 2300, .hsfreqrange = 0x45, .ddlfreq = 411 },
+	{ .mbps = 2350, .hsfreqrange = 0x46, .ddlfreq = 411 },
+	{ .mbps = 2400, .hsfreqrange = 0x47, .ddlfreq = 411 },
+	{ .mbps = 2450, .hsfreqrange = 0x48, .ddlfreq = 411 },
+	{ .mbps = 2500, .hsfreqrange = 0x49, .ddlfreq = 411 },
+	{ /* sentinel */ },
+};
+
+static int fsl_csi2_phy_init(struct phy *phy)
+{
+	struct fsl_csi2_phy *priv = phy_get_drvdata(phy);
+
+	return pm_runtime_get_sync(priv->dev);
+}
+
+static int fsl_csi2_phy_exit(struct phy *phy)
+{
+	struct fsl_csi2_phy *priv = phy_get_drvdata(phy);
+
+	return pm_runtime_put(priv->dev);
+}
+
+static int fsl_csi2_phy_power_on(struct phy *phy)
+{
+	struct fsl_csi2_phy *priv = phy_get_drvdata(phy);
+
+	regmap_update_bits(priv->dphy_regmap, IMX93_BLK_CSI,
+			   IMX93_BLK_CSI_CFGCLKFREQRANGE,
+			   FIELD_PREP(IMX93_BLK_CSI_CFGCLKFREQRANGE, priv->cfgclkfreqrange));
+
+	regmap_update_bits(priv->dphy_regmap, IMX93_BLK_CSI,
+			   IMX93_BLK_CSI_HSFREQRANGE,
+			   FIELD_PREP(IMX93_BLK_CSI_HSFREQRANGE, priv->hsfreqrange));
+
+	return 0;
+}
+
+static int set_freqrange_by_mpbs(struct fsl_csi2_phy *priv, u64 mbps)
+{
+	const struct dphy_mbps_hsfreqrange_map *prev_value = NULL;
+	const struct dphy_mbps_hsfreqrange_map *value;
+
+	for (value = hsfreqrange_table; value->mbps; value++) {
+		if (value->mbps >= mbps)
+			break;
+		prev_value = value;
+	}
+
+	if (prev_value &&
+	    ((mbps - prev_value->mbps) <= (value->mbps - mbps)))
+		value = prev_value;
+
+	if (!value->mbps) {
+		pr_err("Unsupported PHY speed (%llu Mbps)", mbps);
+		return -ERANGE;
+	}
+
+	priv->hsfreqrange = value->hsfreqrange;
+	priv->ddlfreq = value->ddlfreq;
+
+	return 0;
+}
+
+static int fsl_csi2_phy_configure(struct phy *phy, union phy_configure_opts *opts)
+{
+	struct fsl_csi2_phy *priv = phy_get_drvdata(phy);
+	const struct fsl_csi2_phy_drv_data *drv_data = priv->drv_data;
+	struct phy_configure_opts_mipi_dphy *config = &opts->mipi_dphy;
+	struct device *dev = priv->dev;
+	u64 data_rate_mbps;
+	int ret;
+
+	if (config->lanes > drv_data->max_lanes) {
+		dev_err(dev, "The number of lanes has exceeded the maximum value\n");
+		return -EINVAL;
+	}
+
+	data_rate_mbps = div_u64(config->hs_clk_rate, 1000 * 1000);
+	if (data_rate_mbps < 80 ||
+	    data_rate_mbps > drv_data->max_data_rate) {
+		dev_err(dev, "Out-of-bound lane rate %llu\n", data_rate_mbps);
+		return -EINVAL;
+	}
+
+	dev_dbg(dev, "Number of lanes: %d, data rate=%llu(Mbps)\n",
+		config->lanes, data_rate_mbps);
+
+	ret = set_freqrange_by_mpbs(priv, data_rate_mbps);
+	if (ret < 0)
+		return ret;
+
+	return 0;
+}
+
+static const struct phy_ops fsl_csi2_phy_ops = {
+	.init = fsl_csi2_phy_init,
+	.exit = fsl_csi2_phy_exit,
+	.power_on = fsl_csi2_phy_power_on,
+	.configure = fsl_csi2_phy_configure,
+	.owner = THIS_MODULE,
+};
+
+static const struct fsl_csi2_phy_drv_data imx93_dphy_drvdata = {
+	.max_lanes = 2,
+	.max_data_rate = 1500,
+};
+
+static int fsl_csi2_runtime_suspend(struct device *dev)
+{
+	struct fsl_csi2_phy *priv = dev_get_drvdata(dev);
+
+	clk_disable_unprepare(priv->cfg_clk);
+
+	return 0;
+}
+
+static int fsl_csi2_runtime_resume(struct device *dev)
+{
+	struct fsl_csi2_phy *priv = dev_get_drvdata(dev);
+	int ret;
+
+	ret = clk_prepare_enable(priv->cfg_clk);
+	if (ret)
+		return ret;
+
+	return 0;
+}
+
+static DEFINE_RUNTIME_DEV_PM_OPS(fsl_csi2_pm_ops, fsl_csi2_runtime_suspend,
+				 fsl_csi2_runtime_resume, NULL);
+
+static const struct of_device_id fsl_csi2_phy_of_match[] = {
+	{ .compatible = "fsl,imx93-dphy-rx", .data = &imx93_dphy_drvdata},
+	{ /* sentinel */ }
+};
+MODULE_DEVICE_TABLE(of, fsl_csi2_phy_of_match);
+
+static int fsl_csi2_phy_probe(struct platform_device *pdev)
+{
+	struct device *dev = &pdev->dev;
+	struct device_node *np = dev->of_node;
+	struct phy_provider *phy_provider;
+	struct fsl_csi2_phy *priv;
+	unsigned long cfg_rate;
+	struct phy *phy;
+
+	if (!dev->parent || !dev->parent->of_node)
+		return -ENODEV;
+
+	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
+	if (!priv)
+		return -ENOMEM;
+
+	priv->dev = dev;
+	priv->drv_data = of_device_get_match_data(dev);
+
+	platform_set_drvdata(pdev, priv);
+
+	priv->dphy_regmap = syscon_node_to_regmap(dev->parent->of_node);
+	if (IS_ERR(priv->dphy_regmap))
+		dev_err_probe(dev, -ENODEV, "Failed to DPHY regmap\n");
+
+	priv->cfg_clk = devm_clk_get(dev, "cfg");
+	if (IS_ERR(priv->cfg_clk))
+		dev_err_probe(dev, PTR_ERR(priv->cfg_clk), "Failed to get DPHY config clock\n");
+
+	/* cfgclkfreqrange[5:0] = round[(cfg_clk(MHz) - 17) * 4] */
+	cfg_rate = clk_get_rate(priv->cfg_clk);
+	if (!cfg_rate)
+		dev_err_probe(dev, -EINVAL, "Failed to get PHY config clock rate\n");
+
+	priv->cfgclkfreqrange = (div_u64(cfg_rate, 1000 * 1000) - 17) * 4;
+
+	phy = devm_phy_create(dev, np, &fsl_csi2_phy_ops);
+	if (IS_ERR(phy))
+		return dev_err_probe(dev, -ENODEV, "Failed to create PHY\n");
+
+	phy_set_drvdata(phy, priv);
+
+	pm_runtime_set_suspended(dev);
+	devm_pm_runtime_enable(dev);
+
+	phy_provider = devm_of_phy_provider_register(dev, of_phy_simple_xlate);
+
+	return PTR_ERR_OR_ZERO(phy_provider);
+}
+
+static struct platform_driver fsl_csi2_phy_driver = {
+	.probe	= fsl_csi2_phy_probe,
+	.driver = {
+		.name = "imx-mipi-dphy-rx",
+		.pm = pm_ptr(&fsl_csi2_pm_ops),
+		.of_match_table	= fsl_csi2_phy_of_match,
+	}
+};
+module_platform_driver(fsl_csi2_phy_driver);
+
+MODULE_DESCRIPTION("i.MX9 Synopsys DesignWare MIPI DPHY Rx wrapper driver");
+MODULE_AUTHOR("NXP Semiconductor");
+MODULE_LICENSE("GPL");

-- 
2.34.1


Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ