Files
build/patch/kernel/archive/rockchip64-6.16/rk3328-inno-usb3phy-driver.patch

1157 lines
36 KiB
Diff

From 460b224602414d5a3935663ac57fd891a0d8d576 Mon Sep 17 00:00:00 2001
From: Paolo Sabatino <paolo.sabatino@gmail.com>
Date: Tue, 12 Aug 2025 20:08:01 +0200
Subject: [PATCH] Add rockchip Innosilicon USB3 phy driver
source: https://patchwork.kernel.org/project/linux-rockchip/cover/20250115012628.1035928-1-pgwipeout@gmail.com/
---
.../bindings/phy/rockchip,inno-usb3phy.yaml | 166 ++++
arch/arm64/boot/dts/rockchip/rk3328.dtsi | 39 +
drivers/phy/rockchip/Kconfig | 10 +
drivers/phy/rockchip/Makefile | 1 +
drivers/phy/rockchip/phy-rockchip-inno-usb3.c | 869 ++++++++++++++++++
5 files changed, 1085 insertions(+)
create mode 100644 Documentation/devicetree/bindings/phy/rockchip,inno-usb3phy.yaml
create mode 100644 drivers/phy/rockchip/phy-rockchip-inno-usb3.c
diff --git a/Documentation/devicetree/bindings/phy/rockchip,inno-usb3phy.yaml b/Documentation/devicetree/bindings/phy/rockchip,inno-usb3phy.yaml
new file mode 100644
index 000000000000..cde489ca87ab
--- /dev/null
+++ b/Documentation/devicetree/bindings/phy/rockchip,inno-usb3phy.yaml
@@ -0,0 +1,166 @@
+# SPDX-License-Identifier: GPL-2.0-only
+%YAML 1.2
+---
+$id: http://devicetree.org/schemas/phy/rockchip,inno-usb3phy.yaml#
+$schema: http://devicetree.org/meta-schemas/core.yaml#
+
+title: Rockchip USB 3.0 phy with Innosilicon IP block
+
+maintainers:
+ - Heiko Stuebner <heiko@sntech.de>
+
+properties:
+ compatible:
+ enum:
+ - rockchip,rk3328-usb3phy
+
+ reg:
+ maxItems: 1
+
+ clocks:
+ minItems: 3
+ maxItems: 3
+
+ clock-names:
+ items:
+ - const: refclk-usb3otg
+ - const: usb3phy-otg
+ - const: usb3phy-pipe
+
+ interrupts:
+ minItems: 4
+
+ interrupt-names:
+ items:
+ - const: bvalid
+ - const: id
+ - const: linestate
+ - const: rxdet
+
+ resets:
+ minItems: 6
+
+ reset-names:
+ items:
+ - const: usb3phy-u2-por
+ - const: usb3phy-u3-por
+ - const: usb3phy-pipe-mac
+ - const: usb3phy-utmi-mac
+ - const: usb3phy-utmi-apb
+ - const: usb3phy-pipe-apb
+
+ "#address-cells":
+ const: 2
+
+ "#size-cells":
+ const: 2
+
+ ranges: true
+
+patternProperties:
+
+ utmi-port@[0-9a-f]+$:
+ type: object
+ additionalProperties: false
+
+ properties:
+ compatible:
+ enum:
+ - rockchip,rk3328-usb3phy-utmi
+
+ reg:
+ maxItems: 1
+
+ "#phy-cells":
+ const: 0
+
+ phy-supply:
+ description:
+ Phandle to a regulator that provides power to VBUS.
+ See ./phy-bindings.txt for details.
+
+ required:
+ - compatible
+ - reg
+ - "#phy-cells"
+
+ pipe-port@[0-9a-f]+$:
+ type: object
+ additionalProperties: false
+
+ properties:
+ compatible:
+ enum:
+ - rockchip,rk3328-usb3phy-pipe
+
+ reg:
+ maxItems: 1
+
+ "#phy-cells":
+ const: 0
+
+ phy-supply:
+ description:
+ Phandle to a regulator that provides power to VBUS.
+ See ./phy-bindings.txt for details.
+
+ required:
+ - compatible
+ - reg
+ - "#phy-cells"
+
+required:
+ - compatible
+ - reg
+ - clocks
+ - clock-names
+ - interrupts
+ - interrupt-names
+ - resets
+ - reset-names
+
+additionalProperties: false
+
+examples:
+ - |
+ #include <dt-bindings/clock/rk3328-cru.h>
+ #include <dt-bindings/interrupt-controller/arm-gic.h>
+ #include <dt-bindings/interrupt-controller/irq.h>
+ soc {
+ #address-cells = <2>;
+ #size-cells = <2>;
+
+ usb3phy: usb3-phy@ff460000 {
+ compatible = "rockchip,rk3328-usb3phy";
+ reg = <0x0 0xff460000 0x0 0x10000>;
+ clocks = <&cru SCLK_REF_USB3OTG>, <&cru PCLK_USB3PHY_OTG>, <&cru PCLK_USB3PHY_PIPE>;
+ clock-names = "refclk-usb3otg", "usb3phy-otg", "usb3phy-pipe";
+ interrupts = <GIC_SPI 75 IRQ_TYPE_LEVEL_HIGH>, <GIC_SPI 76 IRQ_TYPE_LEVEL_HIGH>,
+ <GIC_SPI 77 IRQ_TYPE_LEVEL_HIGH>, <GIC_SPI 78 IRQ_TYPE_LEVEL_HIGH>;
+ interrupt-names = "bvalid", "id", "linestate", "rxdet";
+ resets = <&cru SRST_USB3PHY_U2>,
+ <&cru SRST_USB3PHY_U3>,
+ <&cru SRST_USB3PHY_PIPE>,
+ <&cru SRST_USB3OTG_UTMI>,
+ <&cru SRST_USB3PHY_OTG_P>,
+ <&cru SRST_USB3PHY_PIPE_P>;
+ reset-names = "usb3phy-u2-por", "usb3phy-u3-por",
+ "usb3phy-pipe-mac", "usb3phy-utmi-mac",
+ "usb3phy-utmi-apb", "usb3phy-pipe-apb";
+ #address-cells = <2>;
+ #size-cells = <2>;
+ ranges;
+
+ usb3phy_utmi: utmi-port@ff470000 {
+ compatible = "rockchip,rk3328-usb3phy-utmi";
+ reg = <0x0 0xff470000 0x0 0x8000>;
+ #phy-cells = <0>;
+ };
+
+ usb3phy_pipe: pipe-port@ff478000 {
+ compatible = "rockchip,rk3328-usb3phy-pipe";
+ reg = <0x0 0xff478000 0x0 0x8000>;
+ #phy-cells = <0>;
+ };
+ };
+ };
diff --git a/arch/arm64/boot/dts/rockchip/rk3328.dtsi b/arch/arm64/boot/dts/rockchip/rk3328.dtsi
index 7d992c3c01ce..181a900d41f9 100644
--- a/arch/arm64/boot/dts/rockchip/rk3328.dtsi
+++ b/arch/arm64/boot/dts/rockchip/rk3328.dtsi
@@ -903,6 +903,43 @@ u2phy_host: host-port {
};
};
+ usb3phy: usb3-phy@ff460000 {
+ compatible = "rockchip,rk3328-usb3phy";
+ reg = <0x0 0xff460000 0x0 0x10000>;
+ clocks = <&cru SCLK_REF_USB3OTG>, <&cru PCLK_USB3PHY_OTG>, <&cru PCLK_USB3PHY_PIPE>;
+ clock-names = "refclk-usb3otg", "usb3phy-otg", "usb3phy-pipe";
+ interrupts = <GIC_SPI 75 IRQ_TYPE_LEVEL_HIGH>, <GIC_SPI 76 IRQ_TYPE_LEVEL_HIGH>,
+ <GIC_SPI 77 IRQ_TYPE_LEVEL_HIGH>, <GIC_SPI 78 IRQ_TYPE_LEVEL_HIGH>;
+ interrupt-names = "bvalid", "id", "linestate", "rxdet";
+ resets = <&cru SRST_USB3PHY_U2>,
+ <&cru SRST_USB3PHY_U3>,
+ <&cru SRST_USB3PHY_PIPE>,
+ <&cru SRST_USB3OTG_UTMI>,
+ <&cru SRST_USB3PHY_OTG_P>,
+ <&cru SRST_USB3PHY_PIPE_P>;
+ reset-names = "usb3phy-u2-por", "usb3phy-u3-por",
+ "usb3phy-pipe-mac", "usb3phy-utmi-mac",
+ "usb3phy-utmi-apb", "usb3phy-pipe-apb";
+ #address-cells = <2>;
+ #size-cells = <2>;
+ ranges;
+ status = "okay";
+
+ usb3phy_utmi: utmi-port@ff470000 {
+ compatible = "rockchip,rk3328-usb3phy-utmi";
+ reg = <0x0 0xff470000 0x0 0x8000>;
+ #phy-cells = <0>;
+ status = "okay";
+ };
+
+ usb3phy_pipe: pipe-port@ff478000 {
+ compatible = "rockchip,rk3328-usb3phy-pipe";
+ reg = <0x0 0xff478000 0x0 0x8000>;
+ #phy-cells = <0>;
+ status = "okay";
+ };
+ };
+
sdmmc: mmc@ff500000 {
compatible = "rockchip,rk3328-dw-mshc", "rockchip,rk3288-dw-mshc";
reg = <0x0 0xff500000 0x0 0x4000>;
@@ -1067,6 +1104,8 @@ usbdrd3: usb@ff600000 {
clock-names = "ref_clk", "suspend_clk",
"bus_clk";
dr_mode = "otg";
+ phys = <&usb3phy_utmi>, <&usb3phy_pipe>;
+ phy-names = "usb2-phy", "usb3-phy";
phy_type = "utmi_wide";
snps,dis-del-phy-power-chg-quirk;
snps,dis_enblslpm_quirk;
diff --git a/drivers/phy/rockchip/Kconfig b/drivers/phy/rockchip/Kconfig
index 14698571b607..858e451edc5a 100644
--- a/drivers/phy/rockchip/Kconfig
+++ b/drivers/phy/rockchip/Kconfig
@@ -48,6 +48,16 @@ config PHY_ROCKCHIP_INNO_USB2
help
Support for Rockchip USB2.0 PHY with Innosilicon IP block.
+config PHY_ROCKCHIP_INNO_USB3
+ tristate "Rockchip INNO USB3PHY Driver"
+ depends on (ARCH_ROCKCHIP || COMPILE_TEST) && OF
+ depends on COMMON_CLK
+ depends on USB_SUPPORT
+ select GENERIC_PHY
+ select USB_COMMON
+ help
+ Support for Rockchip USB3.0 PHY with Innosilicon IP block.
+
config PHY_ROCKCHIP_INNO_CSIDPHY
tristate "Rockchip Innosilicon MIPI CSI PHY driver"
depends on (ARCH_ROCKCHIP || COMPILE_TEST) && OF
diff --git a/drivers/phy/rockchip/Makefile b/drivers/phy/rockchip/Makefile
index 117aaffd037d..d7b7b090b1e2 100644
--- a/drivers/phy/rockchip/Makefile
+++ b/drivers/phy/rockchip/Makefile
@@ -6,6 +6,7 @@ obj-$(CONFIG_PHY_ROCKCHIP_INNO_CSIDPHY) += phy-rockchip-inno-csidphy.o
obj-$(CONFIG_PHY_ROCKCHIP_INNO_DSIDPHY) += phy-rockchip-inno-dsidphy.o
obj-$(CONFIG_PHY_ROCKCHIP_INNO_HDMI) += phy-rockchip-inno-hdmi.o
obj-$(CONFIG_PHY_ROCKCHIP_INNO_USB2) += phy-rockchip-inno-usb2.o
+obj-$(CONFIG_PHY_ROCKCHIP_INNO_USB3) += phy-rockchip-inno-usb3.o
obj-$(CONFIG_PHY_ROCKCHIP_NANENG_COMBO_PHY) += phy-rockchip-naneng-combphy.o
obj-$(CONFIG_PHY_ROCKCHIP_PCIE) += phy-rockchip-pcie.o
obj-$(CONFIG_PHY_ROCKCHIP_SAMSUNG_DCPHY) += phy-rockchip-samsung-dcphy.o
diff --git a/drivers/phy/rockchip/phy-rockchip-inno-usb3.c b/drivers/phy/rockchip/phy-rockchip-inno-usb3.c
new file mode 100644
index 000000000000..51b9f3b7fbfa
--- /dev/null
+++ b/drivers/phy/rockchip/phy-rockchip-inno-usb3.c
@@ -0,0 +1,869 @@
+// SPDX-License-Identifier: GPL-2.0+
+
+/*
+ * phy-rockchip-inno-usb3.c - USB3 PHY based on Innosilicon IP as
+ * implemented on Rockchip rk3328. Tuning data magic bits are taken as is
+ * from the downstream driver. Downstream driver is located at:
+ * https://github.com/rockchip-linux/kernel/blob/240a5660d7c23841ccf7b7cc489078bf521b9802/drivers/phy/rockchip/phy-rockchip-inno-usb3.c
+ *
+ * Author: Peter Geis <pgwipeout@gmail.com>
+ * TODO:
+ * - Find the rest of the register names / definitions.
+ * - Implement pm functions.
+ * - Implement board specific tuning from dts.
+ * - Implement regulator control.
+ */
+
+#include <linux/clk.h>
+#include <linux/delay.h>
+#include <linux/interrupt.h>
+#include <linux/io.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_address.h>
+#include <linux/of_device.h>
+#include <linux/of_irq.h>
+#include <linux/of_platform.h>
+#include <linux/phy/phy.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+#include <linux/reset.h>
+#include <linux/mfd/syscon.h>
+#include <linux/usb/of.h>
+
+#define REG_WRITE_MASK GENMASK(31, 16)
+#define REG_WRITE_SHIFT 16
+#define DISABLE_BITS 0x0
+
+/* USB3PHY GRF Registers */
+#define USB3PHY_WAKEUP_CON_REG 0x40
+#define USB3PHY_WAKEUP_STAT_REG 0x44
+#define USB3_LINESTATE_IRQ_EN BIT(0)
+#define USB3_RXDET_IRQ_EN BIT(1)
+#define USB3_BVALID_RISE_IRQ_EN BIT(2)
+#define USB3_BVALID_FALL_IRQ_EN BIT(3)
+#define USB3_BVALID_CLEAR_MASK GENMASK(3, 2)
+#define USB3_ID_RISE_IRQ_EN BIT(4)
+#define USB3_ID_FALL_IRQ_EN BIT(5)
+#define USB3_ID_CLEAR_MASK GENMASK(5, 4)
+#define USB3_RXDET_EN BIT(6)
+
+/* PIPE registers */
+/* 0x08 for SSC, default 0x0e */
+#define UNKNOWN_PIPE_REG_000 0x000
+#define UNKNOWN_SSC_000_MASK GENMASK(2, 1)
+#define UNKNOWN_SSC_000_ENABLE (0x00 << 1)
+
+/* 0x83 for 24m, 0x01 for 25m, default 0x86 */
+#define PIPE_REG_020 0x020
+/* RX CDR multiplier high bits [7:6], as P, default 0x2, RX data rate = (2*refclk*P)/Q */
+#define PIPE_RX_CDR_MULT_HIGH_MASK GENMASK(7, 6)
+/* TX PLL divider bits [4:0], as N, default 0x6, TX data rate = (2*refclk*M)/N */
+#define PIPE_TX_PLL_DIV_MASK GENMASK(4, 0)
+
+/* 0x71 for 24m, 0x64 for 25m, default 0x71 */
+#define PIPE_REG_028 0x028
+/* RX CDR multiplier low bits [7:0], as P, default 0x71, RX data rate = (2*refclk*P)/Q */
+#define PIPE_RX_CDR_MULT_LOW_MASK GENMASK(7, 0)
+
+/* 0x26 for 24m?, 0x21 for 25m, default 0x26 */
+#define PIPE_REG_030 0x030
+/* RX CDR divider bits [4:0], as Q, default 0x6, RX data rate = (2*refclk*P)/Q */
+#define PIPE_RX_CDR_DIV_MASK GENMASK(4, 0)
+
+/* 1'b1 Disable bandgap power, default 0x00 */
+#define PIPE_REG_044 0x044
+#define BANDGAP_POWER_DISABLE BIT(4)
+
+/* 0xe0 for rx tune?, default 0xe1 */
+#define PIPE_REG_060 0x060
+#define PIPE_TX_DETECT_BYPASS_DEBUG BIT(4) /* enable to always force detection */
+/* RX CTLE frequency bandwidth response tuning bits [1:0], default 0x1 */
+#define PIPE_RX_CTLE_FREQ_BW_MASK GENMASK(1, 0)
+#define PIPE_RX_CTLE_FREQ_BW_TUNE 0x0
+
+/* default 0x49 */
+#define PIPE_REG_064 0x064
+/* RX equalizer tail current control bits [6:4], default 0x4 */
+#define PIPE_RX_EQ_TAIL_CURR_MASK GENMASK(6, 4)
+
+/* 0x08 for rx tune?, default 0x07 */
+#define PIPE_REG_068 0x068
+/* RX equalizer low frequency gain control bits [7:4], default 0x0 */
+#define PIPE_RX_EQ_LOW_GAIN_MASK GENMASK(7, 4)
+#define PIPE_RX_EQ_LOW_GAIN_TUNE (0x1 << 4)
+/* RX CTLE gain tuning bits [3:0], higher = more gain default 0x7 */
+#define PIPE_RX_CTLE_GAIN_MASK GENMASK(3, 0)
+#define PIPE_RX_CTLE_GAIN_TUNE 0x7 /* 0x5 lowest functional value, 0xf highest */
+
+/* RX ODT manual resistance config, higher = less resistance, depends on REG_1C4 BIT(5) set */
+#define PIPE_REG_06C 0x06c
+/* RX ODT manual resistance high bits [3:0], default 0x0 */
+#define PIPE_RX_ODT_RES_HIGH_MASK GENMASK(3, 0)
+#define PIPE_RX_ODT_RES_HIGH_TUNE 0xf
+
+#define PIPE_REG_070 0x070
+/* RX ODT manual resistance mid bits [7:0], default 0x03 */
+#define PIPE_RX_ODT_RES_MID_MASK GENMASK(7, 0)
+#define PIPE_RX_ODT_RES_MID_TUNE 0xff
+
+#define PIPE_REG_074 0x074
+/* RX ODT manual resistance low bits [7:0], default 0xff */
+#define PIPE_RX_ODT_RES_LOW_MASK GENMASK(7, 0)
+#define PIPE_RX_ODT_RES_LOW_TUNE 0xff
+
+/* default 0x08 */
+#define PIPE_REG_080 0x080
+#define PIPE_TX_COMMON_MODE_DIS BIT(2) /* 1'b1 disable TX common type */
+
+/* default 0x33 */
+#define PIPE_REG_088 0x088
+#define PIPE_TX_DRIVER_PREEMP_EN BIT(4) /* 1'b1 enable pre-emphasis */
+
+/* default 0x18 */
+#define PIPE_REG_0C0 0x0c0
+#define PIPE_RX_CM_EN BIT(3) /* 1'b1 enable RX CM */
+#define PIPE_TX_OBS_EN BIT(4) /* 1'b1 enable TX OBS */
+
+/* 0x12 for rx tune?, default 0x14 */
+#define PIPE_REG_0C8 0x0c8
+/* RX CDR charge pump current bits [3:1], default 0x2 */
+#define PIPE_RX_CDR_CHG_PUMP_MASK GENMASK(3, 1)
+#define PIPE_RX_CDR_CHG_PUMP_TUNE (0x2 << 1)
+
+/* 0x02 for 24m, 0x06 for 25m, default 0x06 */
+#define UNKNOWN_PIPE_REG_108 0x108
+#define UNKNOWN_REFCLK_108_24M 0x02
+
+/* 0x80 for 24m, default 0x00 */
+#define UNKNOWN_PIPE_REG_10C 0x10c
+#define UNKNOWN_REFCLK_10C_24M BIT(7)
+
+/* 0x01 for 24m, 0x00 for 25m, default 0x02 */
+#define PIPE_REG_118 0x118
+/* TX PLL multiplier high bits [3:0], as M, default 0x2, TX data rate = (2*refclk*M)/N */
+#define PIPE_TX_PLL_MUL_HIGH_MASK GENMASK(3, 0)
+
+/* 0x38 for 24m, 0x64 for 25m, default 0x71 */
+#define PIPE_REG_11C 0x11c
+/* TX PLL multiplier low bits [7:0], as M, default 0x71, TX data rate = (2*refclk*M)/N */
+#define PIPE_TX_PLL_MUL_LOW_MASK GENMASK(7, 0)
+
+/* 0x0c for SSC, default 0x1c */
+#define UNKNOWN_PIPE_REG_120 0x120
+#define UNKNOWN_SSC_120_MASK BIT(4)
+#define UNKNOWN_SSC_120_ENABLE (0x0 << 4)
+
+/* default 0x40 */
+#define PIPE_REG_12C 0x12c
+#define PIPE_TX_PLL_ALWAYS_ON BIT(0) /* disable for PLL control by pipe_pd */
+
+/* 0x05 for rx tune, default 0x01 */
+#define PIPE_REG_148 0x148
+#define PIPE_RX_CHG_PUMP_DIV_2 BIT(2) /* RX CDR charge pump div/2, default 0 */
+
+/* 0x70 for rx tune, default 0x72 */
+#define PIPE_REG_150 0x150
+#define PIPE_TX_BIAS_EN BIT(6) /* 1'b1 Enable TX Bias */
+/* RX CDR phase tracking speed bits [3:0], default 0x2 */
+#define PIPE_RX_CDR_SPEED_MASK GENMASK(3, 0)
+#define PIPE_RX_CDR_SPEED_TUNE 0x00
+
+/* default 0xd4 */
+#define PIPE_REG_160
+/* RX common mode voltage strength bits [5:4], default 0x1 */
+#define PIPE_RX_CDR_COM_VOLT_MASK GENMASK(5, 4)
+#define PIPE_RX_CDR_COM_VOLT_TUNE (0x1 << 4)
+
+/* default 0x00 */
+#define PIPE_REG_180 0x180
+/* TX driver bias reference voltage bits [3:2], in mv */
+#define PIPE_TX_BIAS_REF_VOLT_MASK GENMASK(3, 2)
+#define PIPE_TX_BIAS_REF_VOLT_200 (0x0 << 2)
+#define PIPE_TX_BIAS_REF_VOLT_175 (0x1 << 2)
+#define PIPE_TX_BIAS_REF_VOLT_225 (0x2 << 2) /* downstream 5.10 driver setting */
+#define PIPE_TX_BIAS_REF_VOLT_250 (0x3 << 2)
+
+/* default 0x01 */
+#define PIPE_REG_1A8 0x1a8
+#define PIPE_LDO_POWER_DIS BIT(4) /* 1'b1 Disable LDO Power */
+
+/* default 0x07 */
+#define PIPE_REG_1AC 0x1ac
+/* TX driver output common voltage bits [5:4], in mv */
+#define PIPE_TX_COMMON_VOLT_MASK GENMASK(5, 4)
+#define PIPE_TX_COMMON_VOLT_800 (0x0 << 4)
+#define PIPE_TX_COMMON_VOLT_750 (0x1 << 4)
+#define PIPE_TX_COMMON_VOLT_950 (0x2 << 4)
+#define PIPE_TX_COMMON_VOLT_1100 (0x3 << 4)
+
+/* default 0xfb */
+#define PIPE_REG_1B8 0x1b8
+/* TX driver swing strength bits [7:4], range 0x0 to 0xf */
+#define PIPE_TX_DRIVER_SWING_MASK GENMASK(7, 4) /* 0x2 lowest functional value */
+/* TX driver pre-emphasis strength bits [1:0], default 0x3, enabled by REG 088 */
+#define PIPE_TX_DRIVER_PREEMP_STR_MASK GENMASK(1, 0)
+
+/* set to 0xf0 for rx tune?, default 0xd0 */
+#define PIPE_REG_1C4 0x1c4
+#define PIPE_RX_ODT_AUTO_DIS BIT(5) /* Disable RX ODT auto compensation */
+#define PIPE_TX_ODT_AUTO_DIS BIT(3) /* Disable TX ODT auto compensation */
+
+/* UTMI registers */
+/* 0x0f for utmi tune, default 0x09*/
+#define UTMI_REG_030 0x030
+/* {bits[2:0]=111}: always enable pre-emphasis */
+#define UTMI_ENABLE_PRE_EMPH_MASK GENMASK(2, 0)
+#define UTMI_ENABLE_PRE_EMPH 0x07
+
+/* 0x41 for utmi tune, default 0x49 */
+#define UTMI_REG_040 0x040
+/* TX HS pre-emphasis strength bits [5:3], default 0x1*/
+#define UTMI_TX_PRE_EMPH_STR_MASK GENMASK(5, 3)
+#define UTMI_TX_PRE_EMPH_WEAKEST (0x0 << 3)
+
+/* set to 0xb5 for utmi tune, default 0xb5 */
+#define UTMI_REG_11C 0x11c
+/* {bits[4:0]=10101}: odt 45ohm tuning */
+#define UTMI_ODT_45_OHM_MASK GENMASK(4, 0)
+#define UTMI_ODT_45_OHM_TUNE 0x15
+
+enum rockchip_usb3phy_type {
+ USB3PHY_TYPE_USB2,
+ USB3PHY_TYPE_USB3,
+ USB3PHY_TYPE_MAX,
+};
+
+/**
+ * struct rockchip_usb3phy_port - usb-phy port data.
+ * @phy: port usb phy struct.
+ * @regmap: port regmap.
+ * @type: port usb phy type.
+ */
+struct rockchip_usb3phy_port {
+ struct phy *phy;
+ struct regmap *regmap;
+ enum rockchip_usb3phy_type type;
+};
+
+struct rockchip_usb3phy {
+ struct device *dev;
+ struct regmap *regmap;
+ struct clk *clk_pipe;
+ struct clk *clk_otg;
+ struct clk *clk_ref;
+ struct reset_control *u3por_rst;
+ struct reset_control *u2por_rst;
+ struct reset_control *pipe_rst;
+ struct reset_control *utmi_rst;
+ struct reset_control *pipe_apb_rst;
+ struct reset_control *utmi_apb_rst;
+ struct rockchip_usb3phy_port ports[USB3PHY_TYPE_MAX];
+ int bvalid_irq;
+ int id_irq;
+ int ls_irq;
+ int rxdet_irq;
+};
+
+struct usb3phy_config {
+ unsigned int reg;
+ unsigned int mask;
+ u8 def;
+};
+
+static const struct usb3phy_config rk3328_rx_config[] = {
+ { PIPE_REG_150, PIPE_RX_CDR_SPEED_MASK, PIPE_RX_CDR_SPEED_TUNE },
+ { PIPE_REG_0C8, PIPE_RX_CDR_CHG_PUMP_MASK, PIPE_RX_CDR_CHG_PUMP_TUNE },
+ { PIPE_REG_148, PIPE_RX_CHG_PUMP_DIV_2, PIPE_RX_CHG_PUMP_DIV_2 },
+ { PIPE_REG_068, PIPE_RX_CTLE_GAIN_MASK, PIPE_RX_CTLE_GAIN_TUNE },
+// { PIPE_REG_1C4, PIPE_RX_ODT_AUTO_DIS, PIPE_RX_ODT_AUTO_DIS },
+ { PIPE_REG_070, PIPE_RX_ODT_RES_MID_MASK, PIPE_RX_ODT_RES_MID_TUNE },
+ { PIPE_REG_06C, PIPE_RX_ODT_RES_HIGH_MASK, PIPE_RX_ODT_RES_HIGH_TUNE },
+ { PIPE_REG_060, PIPE_RX_CTLE_FREQ_BW_MASK, PIPE_RX_CTLE_FREQ_BW_TUNE },
+ { UNKNOWN_PIPE_REG_10C, UNKNOWN_REFCLK_10C_24M, UNKNOWN_REFCLK_10C_24M },
+ { PIPE_REG_060, PIPE_RX_CTLE_FREQ_BW_MASK, PIPE_RX_CTLE_FREQ_BW_TUNE },
+ { PIPE_REG_068, PIPE_RX_EQ_LOW_GAIN_MASK, PIPE_RX_EQ_LOW_GAIN_TUNE },
+};
+
+static const struct usb3phy_config rk3328_tx_config[] = {
+ { PIPE_REG_180, PIPE_TX_BIAS_REF_VOLT_MASK, PIPE_TX_BIAS_REF_VOLT_250 },
+};
+
+static const struct usb3phy_config rk3328_ssc_config[] = {
+ { UNKNOWN_PIPE_REG_000, UNKNOWN_SSC_000_MASK, UNKNOWN_SSC_000_ENABLE },
+ { UNKNOWN_PIPE_REG_120, UNKNOWN_SSC_120_MASK, UNKNOWN_SSC_120_ENABLE },
+};
+
+static const struct usb3phy_config rk3328_utmi_config[] = {
+ { UTMI_REG_030, UTMI_ENABLE_PRE_EMPH_MASK, UTMI_ENABLE_PRE_EMPH },
+ { UTMI_REG_040, UTMI_TX_PRE_EMPH_STR_MASK, UTMI_TX_PRE_EMPH_WEAKEST },
+ { UTMI_REG_11C, UTMI_ODT_45_OHM_MASK, UTMI_ODT_45_OHM_TUNE },
+};
+
+static const struct usb3phy_config rk3328_pipe_pwr_en_config[] = {
+ { PIPE_REG_1A8, PIPE_LDO_POWER_DIS, DISABLE_BITS },
+ { PIPE_REG_044, BANDGAP_POWER_DISABLE, DISABLE_BITS },
+ { PIPE_REG_150, PIPE_TX_BIAS_EN, PIPE_TX_BIAS_EN },
+ { PIPE_REG_080, PIPE_TX_COMMON_MODE_DIS, DISABLE_BITS },
+ { PIPE_REG_0C0, PIPE_TX_OBS_EN, PIPE_TX_OBS_EN },
+ { PIPE_REG_0C0, PIPE_RX_CM_EN, PIPE_RX_CM_EN },
+};
+
+static int rockchip_usb3phy_reset(struct rockchip_usb3phy *usb3phy,
+ bool reset, enum rockchip_usb3phy_type type)
+{
+ if (reset) {
+ if (type == USB3PHY_TYPE_USB2) {
+ clk_disable_unprepare(usb3phy->clk_otg);
+ reset_control_assert(usb3phy->utmi_rst);
+ reset_control_assert(usb3phy->u2por_rst);
+ }
+ if (type == USB3PHY_TYPE_USB3) {
+ clk_disable_unprepare(usb3phy->clk_pipe);
+ reset_control_assert(usb3phy->pipe_rst);
+ reset_control_assert(usb3phy->u3por_rst);
+ }
+ } else {
+ if (type == USB3PHY_TYPE_USB2) {
+ reset_control_deassert(usb3phy->u2por_rst);
+ fsleep(1000);
+ clk_prepare_enable(usb3phy->clk_otg);
+ fsleep(500);
+ reset_control_deassert(usb3phy->utmi_rst);
+ fsleep(100);
+ }
+ if (type == USB3PHY_TYPE_USB3) {
+ reset_control_deassert(usb3phy->u3por_rst);
+ fsleep(500);
+ clk_prepare_enable(usb3phy->clk_pipe);
+ fsleep(1000);
+ reset_control_deassert(usb3phy->pipe_rst);
+ fsleep(100);
+ }
+ }
+ return 0;
+}
+
+static irqreturn_t rockchip_usb3phy_linestate_irq(int irq, void *data)
+{
+ struct rockchip_usb3phy *usb3phy = data;
+ int tmp;
+
+ /* check if the interrupt is enabled */
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG, &tmp);
+ if (!(tmp & USB3_LINESTATE_IRQ_EN)) {
+ dev_warn(usb3phy->dev, "invalid linestate irq received\n");
+ return IRQ_NONE;
+ }
+
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, &tmp);
+ if (tmp & USB3_LINESTATE_IRQ_EN)
+ dev_dbg_ratelimited(usb3phy->dev, "linestate irq received\n");
+
+ /* clear the interrupt */
+ regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, USB3_LINESTATE_IRQ_EN);
+
+ return IRQ_HANDLED;
+}
+
+static irqreturn_t rockchip_usb3phy_bvalid_irq(int irq, void *data)
+{
+ struct rockchip_usb3phy *usb3phy = data;
+ int tmp;
+
+ /* check if the interrupt is enabled */
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG, &tmp);
+ if (!((tmp & USB3_BVALID_FALL_IRQ_EN) | (tmp & USB3_BVALID_RISE_IRQ_EN))) {
+ dev_warn_ratelimited(usb3phy->dev, "invalid bvalid irq received\n");
+ return IRQ_NONE;
+ }
+
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, &tmp);
+ if (tmp & USB3_BVALID_FALL_IRQ_EN)
+ dev_dbg(usb3phy->dev, "bvalid falling irq received\n");
+ if (tmp & USB3_BVALID_RISE_IRQ_EN)
+ dev_dbg(usb3phy->dev, "bvalid rising irq received\n");
+
+ /* clear the interrupt */
+ regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, USB3_BVALID_CLEAR_MASK);
+
+ return IRQ_HANDLED;
+}
+
+static irqreturn_t rockchip_usb3phy_id_irq(int irq, void *data)
+{
+ struct rockchip_usb3phy *usb3phy = data;
+ int tmp;
+
+ /* check if the interrupt is enabled */
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG, &tmp);
+ if (!((tmp & USB3_ID_FALL_IRQ_EN) | (tmp & USB3_ID_RISE_IRQ_EN))) {
+ dev_warn(usb3phy->dev, "invalid id irq received\n");
+ return IRQ_NONE;
+ }
+
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, &tmp);
+ if (tmp & USB3_ID_FALL_IRQ_EN)
+ dev_dbg(usb3phy->dev, "id falling irq received\n");
+ if (tmp & USB3_ID_RISE_IRQ_EN)
+ dev_dbg(usb3phy->dev, "id rising irq received\n");
+
+ /* clear the interrupt */
+ regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, USB3_ID_CLEAR_MASK);
+
+ return IRQ_HANDLED;
+}
+
+static irqreturn_t rockchip_usb3phy_rxdet_irq(int irq, void *data)
+{
+ struct rockchip_usb3phy *usb3phy = data;
+ int tmp;
+
+ /* check if the interrupt is enabled */
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG, &tmp);
+ if (!(tmp & USB3_RXDET_IRQ_EN)) {
+ dev_warn(usb3phy->dev, "invalid rxdet irq received\n");
+ return IRQ_NONE;
+ }
+
+ regmap_read(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, &tmp);
+ if (tmp & USB3_RXDET_IRQ_EN)
+ dev_dbg_ratelimited(usb3phy->dev, "rxdet irq received\n");
+
+ /* clear the interrupt */
+ regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_STAT_REG, USB3_RXDET_IRQ_EN);
+
+ return IRQ_HANDLED;
+}
+
+static int rockchip_usb3phy_bulk_update(struct rockchip_usb3phy *usb3phy, struct regmap *regmap,
+ const struct usb3phy_config *config, unsigned int size)
+{
+ unsigned int i, val, tmp;
+ int ret = 0;
+
+ for (i = 0; i < size; i++) {
+ ret = regmap_read(regmap, config[i].reg, &val);
+ if (ret < 0) {
+ dev_err(usb3phy->dev, "failed to read addr: 0x%02x\n", config[i].reg);
+ return ret;
+ }
+ tmp = val & ~config[i].mask;
+ tmp |= config[i].def;
+ dev_dbg(usb3phy->dev, "write: 0x%03x old: 0x%02x new: 0x%02x\n",
+ config[i].reg, val, tmp);
+ ret = regmap_write(regmap, config[i].reg, tmp);
+ if (ret < 0) {
+ dev_err(usb3phy->dev, "failed to write addr: 0x%02x\n", config[i].reg);
+ return ret;
+ }
+ }
+
+ return ret;
+}
+
+static int rockchip_usb3phy_calc_rate(struct rockchip_usb3phy *usb3phy, struct regmap *regmap)
+{
+ long rate;
+ unsigned int mul, div, target = (5000000000 / 100000);
+
+ rate = clk_get_rate(usb3phy->clk_ref) / 100000;
+ if (rate < 0) {
+ dev_err(usb3phy->dev, "failed to get refclk, %ld\n", rate);
+ return rate;
+ /* lowest possible supported clock is 4.8MHZ, highest rk3328 can do is 1.6GHZ */
+ } else if ((rate < 48) | (rate > 16000)) {
+ goto error;
+ }
+
+ for (div = 1; div < 32; div++) {
+ for (mul = 1; mul < 1024; mul++) {
+ if (((2 * rate * mul) / div) == target)
+ goto done;
+ if (((2 * rate * mul) / div) > target)
+ break;
+ }
+ }
+
+error:
+ dev_err(usb3phy->dev, "invalid refclock rate, %ld\n", rate * 100000);
+ return -EINVAL;
+
+done:
+ dev_dbg(usb3phy->dev, "refclk rate mul: %x div: %x rate: %ld\n", mul, div, (rate * 100000));
+
+ regmap_write(regmap, PIPE_REG_020, (mul >> 2) & PIPE_RX_CDR_MULT_HIGH_MASK);
+ regmap_write(regmap, PIPE_REG_020, div & PIPE_TX_PLL_DIV_MASK);
+ regmap_write(regmap, PIPE_REG_028, mul & PIPE_RX_CDR_MULT_LOW_MASK);
+ regmap_write(regmap, PIPE_REG_030, div & PIPE_RX_CDR_DIV_MASK);
+ regmap_write(regmap, PIPE_REG_118, (mul >> 8) & PIPE_TX_PLL_MUL_HIGH_MASK);
+ regmap_write(regmap, PIPE_REG_11C, mul & PIPE_TX_PLL_MUL_LOW_MASK);
+
+ return 0;
+}
+
+static int rockchip_usb3phy_init(struct phy *phy)
+{
+ struct rockchip_usb3phy_port *port = phy_get_drvdata(phy);
+ struct rockchip_usb3phy *usb3phy = dev_get_drvdata(phy->dev.parent);
+ int tmp, ret;
+
+ dev_warn(usb3phy->dev, "usb3phy_init %s\n", dev_name(&phy->dev));
+ clk_prepare_enable(usb3phy->clk_ref);
+ rockchip_usb3phy_reset(usb3phy, false, port->type);
+
+ if (port->type == USB3PHY_TYPE_USB2) {
+ /*
+ * "For RK3328 SoC, pre-emphasis and pre-emphasis strength must be
+ * written as one fixed value. The ODT 45ohm value should be tuned
+ * for different boards to adjust HS eye height."
+ */
+ dev_dbg(usb3phy->dev, "tuning UTMI\n");
+ ret = rockchip_usb3phy_bulk_update(usb3phy, port->regmap, rk3328_utmi_config,
+ ARRAY_SIZE(rk3328_utmi_config));
+ }
+
+ if (port->type == USB3PHY_TYPE_USB3) {
+ /* Enable interrupts */
+ tmp = (USB3_LINESTATE_IRQ_EN | USB3_ID_FALL_IRQ_EN | USB3_ID_RISE_IRQ_EN |
+ USB3_RXDET_IRQ_EN | USB3_BVALID_RISE_IRQ_EN | USB3_BVALID_FALL_IRQ_EN);
+ tmp |= (tmp << REG_WRITE_SHIFT);
+
+ ret = regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG, tmp);
+ if (ret < 0) {
+ /* interrupt write determines if we have write access */
+ dev_err(usb3phy->dev, "failed to write interrupts\n");
+ return ret;
+ }
+
+ /* Configure for 24M ref clk */
+ dev_dbg(usb3phy->dev, "setting pipe for 24M refclk\n");
+ if (rockchip_usb3phy_calc_rate(usb3phy, usb3phy->regmap))
+ return -EINVAL;
+
+ /* Enable SSC */
+ udelay(3);
+ dev_dbg(usb3phy->dev, "setting pipe for SSC\n");
+ ret = rockchip_usb3phy_bulk_update(usb3phy, port->regmap, rk3328_ssc_config,
+ ARRAY_SIZE(rk3328_ssc_config));
+
+ /* "Tuning RX for compliance RJTL test" */
+ dev_dbg(usb3phy->dev, "setting pipe for RX tuning\n");
+ ret = rockchip_usb3phy_bulk_update(usb3phy, port->regmap, rk3328_rx_config,
+ ARRAY_SIZE(rk3328_rx_config));
+ if (ret)
+ return ret;
+
+ /*
+ * "Tuning TX to increase the bias current used in TX driver and RX EQ,
+ * it can also increase the voltage of LFPS."
+ */
+ dev_dbg(usb3phy->dev, "setting pipe for TX tuning\n");
+ ret = rockchip_usb3phy_bulk_update(usb3phy, port->regmap,
+ rk3328_tx_config, ARRAY_SIZE(rk3328_tx_config));
+
+ /* Power up the pipe */
+ dev_dbg(usb3phy->dev, "setting pipe power on\n");
+ ret = rockchip_usb3phy_bulk_update(usb3phy, port->regmap, rk3328_pipe_pwr_en_config,
+ ARRAY_SIZE(rk3328_pipe_pwr_en_config));
+ }
+
+ return 0;
+}
+
+static int rockchip_usb3phy_parse_dt(struct rockchip_usb3phy *usb3phy, struct device *dev)
+{
+ int ret;
+
+ usb3phy->clk_pipe = devm_clk_get(dev, "usb3phy-pipe");
+ if (IS_ERR(usb3phy->clk_pipe)) {
+ dev_err(dev, "could not get usb3phy pipe clock\n");
+ return PTR_ERR(usb3phy->clk_pipe);
+ }
+
+ usb3phy->clk_otg = devm_clk_get(dev, "usb3phy-otg");
+ if (IS_ERR(usb3phy->clk_otg)) {
+ dev_err(dev, "could not get usb3phy otg clock\n");
+ return PTR_ERR(usb3phy->clk_otg);
+ }
+
+ usb3phy->clk_ref = devm_clk_get(dev, "refclk-usb3otg");
+ if (IS_ERR(usb3phy->clk_ref)) {
+ dev_err(dev, "could not get usb3phy ref clock\n");
+ return PTR_ERR(usb3phy->clk_ref);
+ }
+
+ usb3phy->u2por_rst = devm_reset_control_get(dev, "usb3phy-u2-por");
+ if (IS_ERR(usb3phy->u2por_rst)) {
+ dev_err(dev, "no usb3phy-u2-por reset control found\n");
+ return PTR_ERR(usb3phy->u2por_rst);
+ }
+
+ usb3phy->u3por_rst = devm_reset_control_get(dev, "usb3phy-u3-por");
+ if (IS_ERR(usb3phy->u3por_rst)) {
+ dev_err(dev, "no usb3phy-u3-por reset control found\n");
+ return PTR_ERR(usb3phy->u3por_rst);
+ }
+
+ usb3phy->pipe_rst = devm_reset_control_get(dev, "usb3phy-pipe-mac");
+ if (IS_ERR(usb3phy->pipe_rst)) {
+ dev_err(dev, "no usb3phy_pipe_mac reset control found\n");
+ return PTR_ERR(usb3phy->pipe_rst);
+ }
+
+ usb3phy->utmi_rst = devm_reset_control_get(dev, "usb3phy-utmi-mac");
+ if (IS_ERR(usb3phy->utmi_rst)) {
+ dev_err(dev, "no usb3phy-utmi-mac reset control found\n");
+ return PTR_ERR(usb3phy->utmi_rst);
+ }
+
+ usb3phy->pipe_apb_rst = devm_reset_control_get(dev, "usb3phy-pipe-apb");
+ if (IS_ERR(usb3phy->pipe_apb_rst)) {
+ dev_err(dev, "no usb3phy-pipe-apb reset control found\n");
+ return PTR_ERR(usb3phy->pipe_apb_rst);
+ }
+
+ usb3phy->utmi_apb_rst = devm_reset_control_get(dev, "usb3phy-utmi-apb");
+ if (IS_ERR(usb3phy->utmi_apb_rst)) {
+ dev_err(dev, "no usb3phy-utmi-apb reset control found\n");
+ return PTR_ERR(usb3phy->utmi_apb_rst);
+ }
+
+ usb3phy->ls_irq = of_irq_get_byname(dev->of_node, "linestate");
+ if (usb3phy->ls_irq < 0) {
+ dev_err(dev, "no linestate irq provided\n");
+ return usb3phy->ls_irq;
+ }
+
+ ret = devm_request_threaded_irq(dev, usb3phy->ls_irq, NULL, rockchip_usb3phy_linestate_irq,
+ IRQF_ONESHOT, "rockchip_usb3phy_ls", usb3phy);
+ if (ret) {
+ dev_err(dev, "failed to request linestate irq handle\n");
+ return ret;
+ }
+
+ usb3phy->bvalid_irq = of_irq_get_byname(dev->of_node, "bvalid");
+ if (usb3phy->bvalid_irq < 0) {
+ dev_err(dev, "no bvalid irq provided\n");
+ return usb3phy->bvalid_irq;
+ }
+
+ ret = devm_request_threaded_irq(dev, usb3phy->bvalid_irq, NULL, rockchip_usb3phy_bvalid_irq,
+ IRQF_ONESHOT, "rockchip_usb3phy_bvalid", usb3phy);
+ if (ret) {
+ dev_err(dev, "failed to request bvalid irq handle\n");
+ return ret;
+ }
+
+ usb3phy->id_irq = of_irq_get_byname(dev->of_node, "id");
+ if (usb3phy->id_irq < 0) {
+ dev_err(dev, "no id irq provided\n");
+ return usb3phy->id_irq;
+ }
+
+ ret = devm_request_threaded_irq(dev, usb3phy->id_irq, NULL, rockchip_usb3phy_id_irq,
+ IRQF_ONESHOT, "rockchip_usb3phy-id", usb3phy);
+ if (ret) {
+ dev_err(dev, "failed to request id irq handle\n");
+ return ret;
+ }
+
+ usb3phy->rxdet_irq = of_irq_get_byname(dev->of_node, "rxdet");
+ if (usb3phy->rxdet_irq < 0) {
+ dev_err(dev, "no rxdet irq provided\n");
+ return usb3phy->rxdet_irq;
+ }
+
+ ret = devm_request_threaded_irq(dev, usb3phy->rxdet_irq, NULL, rockchip_usb3phy_rxdet_irq,
+ IRQF_ONESHOT, "rockchip_usb3phy-rxdet", usb3phy);
+ if (ret) {
+ dev_err(dev, "failed to request rxdet irq handle\n");
+ return ret;
+ }
+
+ return 0;
+}
+
+static int rockchip_usb3phy_exit(struct phy *phy)
+{
+ struct rockchip_usb3phy_port *port = phy_get_drvdata(phy);
+ struct rockchip_usb3phy *usb3phy = dev_get_drvdata(phy->dev.parent);
+
+ dev_dbg(usb3phy->dev, "usb3phy_shutdown\n");
+ rockchip_usb3phy_reset(usb3phy, true, port->type);
+
+ return 0;
+}
+
+static const struct phy_ops rockchip_usb3phy_ops = {
+ .init = rockchip_usb3phy_init,
+ .exit = rockchip_usb3phy_exit,
+ .owner = THIS_MODULE,
+};
+
+static const struct regmap_config rockchip_usb3phy_utmi_port_regmap_config = {
+ .reg_bits = 32,
+ .val_bits = 32,
+ .reg_stride = 4,
+ .max_register = 0x400,
+ .cache_type = REGCACHE_NONE,
+ .fast_io = true,
+ .name = "utmi-port",
+};
+
+static const struct regmap_config rockchip_usb3phy_pipe_port_regmap_config = {
+ .reg_bits = 32,
+ .val_bits = 32,
+ .reg_stride = 4,
+ .max_register = 0x400,
+ .cache_type = REGCACHE_NONE,
+ .fast_io = true,
+ .name = "pipe-port",
+};
+
+static const struct regmap_config rockchip_usb3phy_regmap_config = {
+ .reg_bits = 32,
+ .val_bits = 32,
+ .reg_stride = 4,
+ .max_register = 0x400,
+ .write_flag_mask = REG_WRITE_MASK,
+ .cache_type = REGCACHE_NONE,
+ .fast_io = true,
+};
+
+static int rockchip_usb3phy_probe(struct platform_device *pdev)
+{
+ struct device *dev = &pdev->dev;
+ struct device_node *np = dev->of_node;
+ struct device_node *child_np;
+ struct phy_provider *provider;
+ struct rockchip_usb3phy *usb3phy;
+ const struct regmap_config regmap_config = rockchip_usb3phy_regmap_config;
+ void __iomem *base;
+ int i, ret;
+
+ dev_dbg(dev, "Probe usb3phy main block\n");
+ usb3phy = devm_kzalloc(dev, sizeof(*usb3phy), GFP_KERNEL);
+ if (!usb3phy)
+ return -ENOMEM;
+
+ ret = rockchip_usb3phy_parse_dt(usb3phy, dev);
+ if (ret) {
+ dev_err(dev, "parse dt failed %i\n", ret);
+ return ret;
+ }
+
+ base = devm_of_iomap(dev, np, 0, NULL);
+ if (IS_ERR(base)) {
+ dev_err(dev, "failed port ioremap\n");
+ return PTR_ERR(base);
+ }
+
+ usb3phy->regmap = devm_regmap_init_mmio(dev, base, &regmap_config);
+ if (IS_ERR(usb3phy->regmap)) {
+ dev_err(dev, "regmap init failed\n");
+ return PTR_ERR(usb3phy->regmap);
+ }
+
+ usb3phy->dev = dev;
+ platform_set_drvdata(pdev, usb3phy);
+
+ /* place block in reset */
+ reset_control_assert(usb3phy->pipe_rst);
+ reset_control_assert(usb3phy->utmi_rst);
+ reset_control_assert(usb3phy->u3por_rst);
+ reset_control_assert(usb3phy->u2por_rst);
+ reset_control_assert(usb3phy->pipe_apb_rst);
+ reset_control_assert(usb3phy->utmi_apb_rst);
+
+ fsleep(20);
+
+ /* take apb interface out of reset */
+ reset_control_deassert(usb3phy->utmi_apb_rst);
+ reset_control_deassert(usb3phy->pipe_apb_rst);
+
+ /* enable usb3phy rx detection to fix disconnection issues */
+ regmap_write(usb3phy->regmap, USB3PHY_WAKEUP_CON_REG,
+ (USB3_RXDET_EN | (USB3_RXDET_EN << REG_WRITE_SHIFT)));
+
+ dev_dbg(dev, "Completed usb3phy core probe\n");
+
+ /* probe the actual ports */
+ i = 0;
+ for_each_available_child_of_node(np, child_np) {
+ const struct regmap_config *regmap_port_config;
+ struct rockchip_usb3phy_port *port = &usb3phy->ports[i];
+ struct phy *phy;
+
+ if (of_node_name_eq(child_np, "utmi-port")) {
+ port->type = USB3PHY_TYPE_USB2;
+ regmap_port_config = &rockchip_usb3phy_utmi_port_regmap_config;
+ } else if (of_node_name_eq(child_np, "pipe-port")) {
+ port->type = USB3PHY_TYPE_USB3;
+ regmap_port_config = &rockchip_usb3phy_pipe_port_regmap_config;
+ } else {
+ dev_err(dev, "unknown child node port type %s\n", child_np->name);
+ goto err_port;
+ }
+
+ base = devm_of_iomap(dev, child_np, 0, NULL);
+ if (IS_ERR(base)) {
+ dev_err(dev, "failed port ioremap\n");
+ ret = PTR_ERR(base);
+ goto err_port;
+ }
+
+ port->regmap = devm_regmap_init_mmio(dev, base, regmap_port_config);
+ if (IS_ERR(port->regmap)) {
+ dev_err(dev, "regmap init failed\n");
+ ret = PTR_ERR(port->regmap);
+ goto err_port;
+ }
+
+ phy = devm_phy_create(dev, child_np, &rockchip_usb3phy_ops);
+ if (IS_ERR(phy)) {
+ dev_err_probe(dev, PTR_ERR(phy), "failed to create phy\n");
+ ret = PTR_ERR(phy);
+ goto err_port;
+ }
+
+ port->phy = phy;
+ phy_set_drvdata(port->phy, port);
+
+ /* to prevent out of boundary */
+ if (++i >= USB3PHY_TYPE_MAX) {
+ of_node_put(child_np);
+ break;
+ }
+
+ dev_info(dev, "Completed usb3phy %s port init\n", child_np->name);
+ }
+
+ provider = devm_of_phy_provider_register(dev, of_phy_simple_xlate);
+ return PTR_ERR_OR_ZERO(provider);
+
+err_port:
+ of_node_put(child_np);
+ return ret;
+}
+
+static const struct of_device_id rockchip_usb3phy_dt_ids[] = {
+ { .compatible = "rockchip,rk3328-usb3phy", },
+};
+
+MODULE_DEVICE_TABLE(of, rockchip_usb3phy_dt_ids);
+
+static struct platform_driver rockchip_usb3phy_driver = {
+ .probe = rockchip_usb3phy_probe,
+ .driver = {
+ .name = "rockchip-usb3-phy",
+ .of_match_table = rockchip_usb3phy_dt_ids,
+ },
+};
+
+module_platform_driver(rockchip_usb3phy_driver);
+
+MODULE_AUTHOR("Peter Geis <pgwipeout@gmail.com>");
+MODULE_DESCRIPTION("Rockchip Innosilicon USB3PHY driver");
+MODULE_LICENSE("GPL");
--
2.43.0