On Fri, Sep 12, 2014 at 12:50:23PM -0500, Josh Cartwright wrote: > Hey Andy- > > Mostly cosmetic things below: > > On Fri, Sep 12, 2014 at 12:29:46PM -0500, Andy Gross wrote: > > This patch adds a new driver for the Qualcomm USB 3.0 PHY that exists on some > > Qualcomm platforms. This driver uses the generic PHY framework and will > > interact with the DWC3 controller. > > > > Signed-off-by: Andy Gross <agross@xxxxxxxxxxxxxx> > > --- > > drivers/phy/Kconfig | 11 + > > drivers/phy/Makefile | 1 + > > drivers/phy/phy-qcom-dwc3.c | 500 +++++++++++++++++++++++++++++++++++++++++++ > > 3 files changed, 512 insertions(+) > > create mode 100644 drivers/phy/phy-qcom-dwc3.c > > > > diff --git a/drivers/phy/Kconfig b/drivers/phy/Kconfig > > index 0dd7427..5d56161 100644 > > --- a/drivers/phy/Kconfig > > +++ b/drivers/phy/Kconfig > > @@ -230,4 +230,15 @@ config PHY_XGENE > > help > > This option enables support for APM X-Gene SoC multi-purpose PHY. > > > > +config PHY_QCOM_DWC3 > > + tristate "QCOM DWC3 USB PHY support" > > + depends on ARCH_QCOM > > + depends on HAS_IOMEM > > + depends on OF > > + select GENERIC_PHY > > + help > > + This option enables support for the Synopsis PHYs present inside the > > + Qualcomm USB3.0 DWC3 controller. This driver supports both HS and SS > > + PHY controllers. > > + > > endmenu > > diff --git a/drivers/phy/Makefile b/drivers/phy/Makefile > > index 95c69ed..aa16f30 100644 > > --- a/drivers/phy/Makefile > > +++ b/drivers/phy/Makefile > > @@ -28,3 +28,4 @@ obj-$(CONFIG_PHY_QCOM_IPQ806X_SATA) += phy-qcom-ipq806x-sata.o > > obj-$(CONFIG_PHY_ST_SPEAR1310_MIPHY) += phy-spear1310-miphy.o > > obj-$(CONFIG_PHY_ST_SPEAR1340_MIPHY) += phy-spear1340-miphy.o > > obj-$(CONFIG_PHY_XGENE) += phy-xgene.o > > +obj-$(CONFIG_PHY_QCOM_DWC3) += phy-qcom-dwc3.o > > diff --git a/drivers/phy/phy-qcom-dwc3.c b/drivers/phy/phy-qcom-dwc3.c > > new file mode 100644 > > index 0000000..05b8e1d > > --- /dev/null > > +++ b/drivers/phy/phy-qcom-dwc3.c > [..] > > +/* TX OVRD DRV LO register bits */ > > +#define TX_OVRD_DRV_LO_AMPLITUDE_MASK 0x007F > > +#define TX_OVRD_DRV_LO_PREEMPH_MASK 0x3F80 > > +#define TX_OVRD_DRV_LO_PREEMPH_SHIFT 7 > > +#define TX_OVRD_DRV_LO_EN BIT(14) > > + > > +struct qcom_dwc3_usb_phy; > > Do you need this? No, this was a set of interim changes that should have been squashed. Incidently, this was used when I had a separate ops table for the phys. > > + > > +enum qcom_dwc3_phy_id { > > + QCOM_DWC3_PHY_UTMI = 0, > > + QCOM_DWC3_PHY_PIPE, > > + QCOM_DWC3_PHYS_NUM, > > +}; > > + > > +struct qcom_dwc3_usb_phy { > > + void __iomem *base; > > + struct device *dev; > > + struct phy *phy; > > + enum qcom_dwc3_phy_id index; > > + > > + struct clk *xo_clk; > > + struct clk *ref_clk; > > +}; > > + > > +/** > > + * Write register and read back masked value to confirm it is written > > + * > > + * @base - QCOM DWC3 PHY base virtual address. > > + * @offset - register offset. > > + * @mask - register bitmask specifying what should be updated > > + * @val - value to write. > > + */ > > +static inline void qcom_dwc3_phy_write_readback( > > + struct qcom_dwc3_usb_phy *phy_dwc3, u32 offset, > > + const u32 mask, u32 val) > > +{ > > + u32 write_val, tmp = readl(phy_dwc3->base + offset); > > + > > + tmp &= ~mask; /* retain other bits */ > > + write_val = tmp | val; > > + > > + writel(write_val, phy_dwc3->base + offset); > > + > > + /* Read back to see if val was written */ > > + tmp = readl(phy_dwc3->base + offset); > > + tmp &= mask; /* clear other bits */ > > + > > + if (tmp != val) > > + dev_err(phy_dwc3->dev, "write: %x to QSCRATCH: %x FAILED\n", > > + val, offset); > > +} > > + > > +static int wait_for_latch(void __iomem *addr) > > +{ > > + u32 retry = 10; > > + > > + while (true) { > > + if (!readl(addr)) > > + break; > > + > > + if (--retry == 0) > > + return -ETIMEDOUT; > > + > > + usleep_range(10, 20); > > + } > > + > > + return 0; > > +} > > + > > +/** > > + * Write SSPHY register > > + * > > + * @base - QCOM DWC3 PHY base virtual address. > > + * @addr - SSPHY address to write. > > + * @val - value to write. > > + */ > > +static int qcom_dwc3_ss_write_phycreg(void __iomem *base, u32 addr, u32 val) > > +{ > > + writel(addr, base + CR_PROTOCOL_DATA_IN_REG); > > + writel(0x1, base + CR_PROTOCOL_CAP_ADDR_REG); > > + > > + if (wait_for_latch(base + CR_PROTOCOL_CAP_ADDR_REG)) > > + return 1; > > I'm not too familiar with the GENERIC_PHY infrastructure, does it > understand that '1' is an error case? I'm wondering if you should > instead propogate the -ETIMEDOUT up. Actually, yeah I need to return something < 0. Let me respin to fix that. > > + > > + writel(val, base + CR_PROTOCOL_DATA_IN_REG); > > + writel(0x1, base + CR_PROTOCOL_CAP_DATA_REG); > > + > > + if (wait_for_latch(base + CR_PROTOCOL_CAP_DATA_REG)) > > + return 1; > > + > > + writel(0x1, base + CR_PROTOCOL_WRITE_REG); > > + > > + if (wait_for_latch(base + CR_PROTOCOL_WRITE_REG)) > > + return 1; > > + > > + return 0; > > +} > > + > [..] > > + > > +static struct phy_ops qcom_dwc3_phy_ops = { > > + .init = qcom_dwc3_phy_init, > > + .exit = qcom_dwc3_phy_exit, > > + .power_on = qcom_dwc3_phy_power_on, > > + .power_off = qcom_dwc3_phy_power_off, > > + .owner = THIS_MODULE, > > +}; > > + > > +static const struct of_device_id qcom_dwc3_phy_table[] = { > > + { > > + .compatible = "qcom,dwc3-hs-usb-phy", > > + .data = (void *)QCOM_DWC3_PHY_UTMI, > > + }, > > + { > > + .compatible = "qcom,dwc3-ss-usb-phy", > > + .data = (void *)QCOM_DWC3_PHY_PIPE, > > + }, > > + { /* Sentinel */ } > > +}; > > +MODULE_DEVICE_TABLE(of, qcom_dwc3_phy_table); > > I'd suggest just creating a separate ops table, and have '.data' point > to it. Might cleanup all of the switch() statements above. You could > also probably drop the 'index' member of 'struct qcom_dwc3_usb_phy', and > maybe even 'enum qcom_dwc3_phy_id' altogether. Or just put in two function handlers that I point to in the probe and key off of the compat. -- sent by an employee of the Qualcomm Innovation Center, Inc. The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum, hosted by The Linux Foundation -- To unsubscribe from this list: send the line "unsubscribe devicetree" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html