Re: [PATCH 2/2] reset: bcm4908-usb: add driver for BCM4908 USB reset controller

From: Philipp Zabel
Date: Fri Dec 04 2020 - 11:14:29 EST


Hi Rafał,

On Fri, 2020-12-04 at 10:37 +0100, Rafał Miłecki wrote:
> From: Rafał Miłecki <rafal@xxxxxxxxxx>
>
> This controller is responsible for OHCI, EHCI, XHCI and PHYs setup that
> has to be handled in the proper order.
>
> One unusual thing about this controller is that is provides access to
> the MDIO bus. There are two registers (in the middle of block space)
> responsible for that. For that reason this driver initializes regmap so
> a proper MDIO driver can use them.
>
> Signed-off-by: Rafał Miłecki <rafal@xxxxxxxxxx>

This doesn't look like a reset controller to me, but rather like
something that belongs in drivers/usb.

regards
Philipp

> ---
> drivers/reset/Kconfig | 8 +
> drivers/reset/Makefile | 1 +
> drivers/reset/reset-bcm4908-usb.c | 250 ++++++++++++++++++++++++++++++
> 3 files changed, 259 insertions(+)
> create mode 100644 drivers/reset/reset-bcm4908-usb.c
>
> diff --git a/drivers/reset/Kconfig b/drivers/reset/Kconfig
> index f393f7e17e33..bb4d2bea9e95 100644
> --- a/drivers/reset/Kconfig
> +++ b/drivers/reset/Kconfig
> @@ -35,6 +35,14 @@ config RESET_AXS10X
> help
> This enables the reset controller driver for AXS10x.
>
> +config RESET_BCM4908_USB
> + tristate "Broadcom BCM4908 USB controller"
> + depends on ARM64 || COMPILE_TEST
> + default ARCH_BCM4908
> + help
> + This enables driver for the Broadcom BCM4908 USB controller
> + responsible for initializing OHCI, EHCI, XHCI and PHYs.
> +
> config RESET_BERLIN
> bool "Berlin Reset Driver" if COMPILE_TEST
> default ARCH_BERLIN
> diff --git a/drivers/reset/Makefile b/drivers/reset/Makefile
> index 0dd5d42050dc..f2627bbc7ad4 100644
> --- a/drivers/reset/Makefile
> +++ b/drivers/reset/Makefile
> @@ -6,6 +6,7 @@ obj-$(CONFIG_ARCH_TEGRA) += tegra/
> obj-$(CONFIG_RESET_A10SR) += reset-a10sr.o
> obj-$(CONFIG_RESET_ATH79) += reset-ath79.o
> obj-$(CONFIG_RESET_AXS10X) += reset-axs10x.o
> +obj-$(CONFIG_RESET_BCM4908_USB) += reset-bcm4908-usb.o
> obj-$(CONFIG_RESET_BERLIN) += reset-berlin.o
> obj-$(CONFIG_RESET_BRCM_PMB) += reset-brcm-pmb.o
> obj-$(CONFIG_RESET_BRCMSTB) += reset-brcmstb.o
> diff --git a/drivers/reset/reset-bcm4908-usb.c b/drivers/reset/reset-bcm4908-usb.c
> new file mode 100644
> index 000000000000..e9b7d369c894
> --- /dev/null
> +++ b/drivers/reset/reset-bcm4908-usb.c
> @@ -0,0 +1,250 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * Copyright (c) 2013 Broadcom
> + * Copyright (C) 2020 Rafał Miłecki <rafal@xxxxxxxxxx>
> + */
> +
> +#include <linux/delay.h>
> +#include <linux/io.h>
> +#include <linux/module.h>
> +#include <linux/of.h>
> +#include <linux/of_device.h>
> +#include <linux/phy/phy.h>
> +#include <linux/platform_device.h>
> +#include <linux/regmap.h>
> +#include <linux/reset-controller.h>
> +#include <linux/reset.h>
> +
> +#define BCM4908_USB_RESET_SETUP 0x0000
> +#define BCM4908_USBH_IPP (1<<5)
> +#define BCM4908_USBH_IOC (1<<4)
> +#define BCM4908_USB2_OC_DISABLE_PORT0 (1<<28)
> +#define BCM4908_USB2_OC_DISABLE_PORT1 (1<<29)
> +#define BCM4908_USB3_OC_DISABLE_PORT0 (1<<30)
> +#define BCM4908_USB3_OC_DISABLE_PORT1 (1<<31)
> +#define BCM4908_USB_RESET_PLL_CTL 0x0004
> +#define BCM4908_USB_RESET_FLADJ_VALUE 0x0008
> +#define BCM4908_USB_RESET_BRIDGE_CTL 0x000c
> +#define BCM4908_USB_RESET_SPARE1 0x0010
> +#define BCM4908_USB_RESET_MDIO 0x0014
> +#define BCM4908_USB_RESET_MDIO2 0x0018
> +#define BCM4908_USB_RESET_TEST_PORT_CONTROL 0x001c
> +#define BCM4908_USB_RESET_USB_SIMCTL 0x0020
> +#define BCM4908_USBH_OHCI_MEM_REQ_DIS (1<<1)
> +#define BCM4908_USB_RESET_USB_TESTCTL 0x0024
> +#define BCM4908_USB_RESET_USB_TESTMON 0x0028
> +#define BCM4908_USB_RESET_UTMI_CTL_1 0x002c
> +#define BCM4908_USB_RESET_SPARE2 0x0030
> +#define BCM4908_USB_RESET_USB_PM 0x0034
> +#define BCM4908_XHC_SOFT_RESETB (1<<22)
> +#define BCM4908_USB_PWRDWN (1<<31)
> +#define BCM4908_USB_RESET_USB_PM_STATUS 0x0038
> +#define BCM4908_USB_RESET_SPARE3 0x003c
> +#define BCM4908_USB_RESET_PLL_LDO_CTL 0x0040
> +#define BCM4908_USB_RESET_PLL_LDO_PLLBIAS 0x0044
> +#define BCM4908_USB_RESET_PLL_AFE_BG_CNTL 0x0048
> +#define BCM4908_USB_RESET_AFE_USBIO_TST 0x004c
> +#define BCM4908_USB_RESET_PLL_NDIV_FRAC 0x0050
> +#define BCM4908_USB_RESET_TP_DIAG 0x0054
> +#define BCM4908_USB_RESET_AHB_CAPTURE_FIFO 0x0058
> +#define BCM4908_USB_RESET_SPARE4 0x005c
> +#define BCM4908_USB_RESET_USB30_CTL1 0x0060
> +#define BCM4908_PHY3_PLL_SEQ_START (1<<4)
> +#define BCM4908_USB_RESET_USB30_CTL2 0x0064
> +#define BCM4908_USB_RESET_USB30_CTL3 0x0068
> +#define BCM4908_USB_RESET_USB30_CTL4 0x006c
> +#define BCM4908_USB_RESET_USB30_PCTL 0x0070
> +#define BCM4908_USB_RESET_USB30_CTL5 0x0074
> +#define BCM4908_USB_RESET_SPARE5 0x0078
> +#define BCM4908_USB_RESET_SPARE6 0x007c
> +#define BCM4908_USB_RESET_SPARE7 0x0080
> +#define BCM4908_USB_RESET_USB_DEVICE_CTL1 0x0090
> +#define BCM4908_USB_RESET_USB_DEVICE_CTL2 0x0094
> +#define BCM4908_USB_RESET_USB20_ID 0x0150
> +#define BCM4908_USB_RESET_USB30_ID 0x0154
> +#define BCM4908_USB_RESET_BDC_COREID 0x0158
> +#define BCM4908_USB_RESET_USB_REVID 0x015c
> +
> +struct bcm4908usb {
> + struct device *dev;
> + void __iomem *base;
> + struct regmap *regmap;
> + struct reset_control *reset;
> + struct phy *usb2_phy;
> + struct phy *usb3_phy;
> + struct reset_controller_dev rcdev;
> +};
> +
> +static const struct regmap_config bcm4908_usb_reset_regmap_config = {
> + .reg_bits = 32,
> + .reg_stride = 4,
> + .val_bits = 32,
> + .fast_io = true,
> +};
> +
> +static u32 bcm4908_usb_reset_read(struct bcm4908usb *usb, u32 reg)
> +{
> + return readl(usb->base + reg);
> +}
> +
> +static void bcm4908_usb_reset_write(struct bcm4908usb *usb, u32 reg, u32 value)
> +{
> + writel(value, usb->base + reg);
> +}
> +
> +static void bcm4908_usb_reset_update_bits(struct bcm4908usb *usb, u32 reg, u32 mask, u32 val)
> +{
> + u32 tmp;
> +
> + WARN_ON(val & ~mask);
> +
> + tmp = readl(usb->base + reg);
> + tmp &= ~mask;
> + tmp |= val & mask;
> + writel(tmp, usb->base + reg);
> +}
> +
> +static int bcm4908_usb_deassert(struct reset_controller_dev *rcdev, unsigned long id)
> +{
> + struct bcm4908usb *usb = container_of(rcdev, struct bcm4908usb, rcdev);
> + struct device *dev = usb->dev;
> + u32 val;
> + int err;
> +
> + err = reset_control_deassert(usb->reset);
> + if (err) {
> + dev_err(dev, "failed to deassert");
> + return err;
> + }
> +
> + mdelay(1);
> +
> + /* adjust over current & port power polarity */
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_SETUP,
> + BCM4908_USBH_IOC, BCM4908_USBH_IOC);
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_SETUP,
> + BCM4908_USBH_IPP, BCM4908_USBH_IPP);
> +
> + /* enable USB PHYs */
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_USB_PM,
> + BCM4908_USB_PWRDWN, 0);
> + mdelay(1);
> +
> + err = phy_init(usb->usb3_phy);
> + if (err) {
> + dev_err(usb->dev, "failed to init USB 3.0 PHY: %d\n", err);
> + return err;
> + }
> + mdelay(300);
> +
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_USB30_CTL1,
> + BCM4908_PHY3_PLL_SEQ_START, BCM4908_PHY3_PLL_SEQ_START);
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_USB_PM,
> + BCM4908_XHC_SOFT_RESETB, BCM4908_XHC_SOFT_RESETB);
> +
> + err = phy_init(usb->usb2_phy);
> + if (err) {
> + dev_err(usb->dev, "failed to init USB 2.0 PHY: %d\n", err);
> + return err;
> + }
> +
> + /* no swap for data & descriptors */
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_BRIDGE_CTL, 0xf, 0);
> +
> + /* reset host controllers for possible fake overcurrent indications */
> + val = bcm4908_usb_reset_read(usb, BCM4908_USB_RESET_USB_PM);
> + bcm4908_usb_reset_write(usb, BCM4908_USB_RESET_USB_PM, 0);
> + bcm4908_usb_reset_write(usb, BCM4908_USB_RESET_USB_PM, val);
> + mdelay(1);
> +
> + /* TODO: erdy nump bypass */
> +
> + /* Reduce accesses to DDR during USB idle time when Self-Refresh feature is compiled in */
> + bcm4908_usb_reset_update_bits(usb, BCM4908_USB_RESET_USB_SIMCTL,
> + BCM4908_USBH_OHCI_MEM_REQ_DIS, BCM4908_USBH_OHCI_MEM_REQ_DIS);
> +
> + return 0;
> +}
> +
> +static const struct reset_control_ops bcm4908_usb_reset_control_ops = {
> + .deassert = bcm4908_usb_deassert,
> +};
> +
> +static int bcm4908_usb_reset_xlate(struct reset_controller_dev *rcdev,
> + const struct of_phandle_args *reset_spec)
> +{
> + if (WARN_ON(reset_spec->args_count))
> + return -EINVAL;
> +
> + return 0;
> +}
> +
> +static int bcm4908_usb_reset_probe(struct platform_device *pdev)
> +{
> + struct device *dev = &pdev->dev;
> + struct bcm4908usb *usb;
> + int err;
> +
> + usb = devm_kzalloc(dev, sizeof(*usb), GFP_KERNEL);
> + if (!usb)
> + return -ENOMEM;
> +
> + usb->dev = dev;
> +
> + usb->base = devm_platform_ioremap_resource(pdev, 0);
> + if (IS_ERR(usb->base))
> + return PTR_ERR(usb->base);
> +
> + usb->regmap = devm_regmap_init_mmio(dev, usb->base, &bcm4908_usb_reset_regmap_config);
> + if (IS_ERR(usb->regmap)) {
> + err = PTR_ERR(usb->regmap);
> + dev_err(dev, "failed to init regmap: %d\n", err);
> + return err;
> + }
> +
> + usb->reset = of_reset_control_array_get_optional_exclusive(usb->dev->of_node);
> + if (IS_ERR(usb->reset)) {
> + err = PTR_ERR(usb->reset);
> + return err;
> + }
> +
> + usb->usb2_phy = devm_phy_get(dev, "usb2");
> + if (IS_ERR(usb->usb2_phy)) {
> + err = PTR_ERR(usb->usb2_phy);
> + dev_err(dev, "Failed to get USB 2.0 PHY: %d\n", err);
> + return err;
> + }
> +
> + usb->usb3_phy = devm_phy_get(dev, "usb3");
> + if (IS_ERR(usb->usb3_phy)) {
> + err = PTR_ERR(usb->usb3_phy);
> + dev_err(dev, "Failed to get USB 2.0 PHY: %d\n", err);
> + return err;
> + }
> +
> + usb->rcdev.ops = &bcm4908_usb_reset_control_ops;
> + usb->rcdev.owner = THIS_MODULE;
> + usb->rcdev.of_node = dev->of_node;
> + usb->rcdev.of_reset_n_cells = 0;
> + usb->rcdev.of_xlate = bcm4908_usb_reset_xlate;
> +
> + return devm_reset_controller_register(dev, &usb->rcdev);
> +}
> +
> +static const struct of_device_id bcm4908_usb_reset_of_match[] = {
> + { .compatible = "brcm,bcm4908-usb-reset" },
> + { },
> +};
> +
> +static struct platform_driver bcm4908_usb_reset_driver = {
> + .probe = bcm4908_usb_reset_probe,
> + .driver = {
> + .name = "bcm4908-usb-reset",
> + .of_match_table = bcm4908_usb_reset_of_match,
> + }
> +};
> +module_platform_driver(bcm4908_usb_reset_driver);
> +
> +MODULE_AUTHOR("Rafał Miłecki");
> +MODULE_LICENSE("GPL v2");
> +MODULE_DEVICE_TABLE(of, bcm4908_usb_reset_of_match);