usb: phy: msm: Select secondary PHY via TCSR
authorTim Bird <tbird20d@gmail.com>
Mon, 28 Apr 2014 13:34:20 +0000 (16:34 +0300)
committerFelipe Balbi <balbi@ti.com>
Wed, 30 Apr 2014 16:29:51 +0000 (11:29 -0500)
Select the secondary PHY using the TCSR register, if phy-num=1
in the DTS (or phy_number is set in the platform data).  The
SOC has 2 PHYs which can be used with the OTG port, and this
code allows configuring the correct one.

Note: This resolves the problem I was seeing where I couldn't
get the USB driver working at all on a dragonboard, from cold
boot.  This patch depends on patch 5/14 from Ivan's msm USB
patch set.  It does not use DT for the register address, as
there's no evidence that this address changes between SoC
versions.

Signed-off-by: Tim Bird <tim.bird@sonymobile.com>
Signed-off-by: Ivan T. Ivanov <iivanov@mm-sol.com>
Signed-off-by: Felipe Balbi <balbi@ti.com>
drivers/usb/phy/phy-msm-usb.c
include/linux/usb/msm_hsusb_hw.h

index db8d963776202897968b51b11ab902c4e9f5db20..9437bcf8c3678bddbac61301ae1dcee0c578fc2c 100644 (file)
@@ -1489,6 +1489,7 @@ static int msm_otg_probe(struct platform_device *pdev)
        struct resource *res;
        struct msm_otg *motg;
        struct usb_phy *phy;
+       void __iomem *phy_select;
 
        motg = devm_kzalloc(&pdev->dev, sizeof(struct msm_otg), GFP_KERNEL);
        if (!motg) {
@@ -1553,6 +1554,19 @@ static int msm_otg_probe(struct platform_device *pdev)
        if (IS_ERR(motg->regs))
                return PTR_ERR(motg->regs);
 
+       /*
+        * NOTE: The PHYs can be multiplexed between the chipidea controller
+        * and the dwc3 controller, using a single bit. It is important that
+        * the dwc3 driver does not set this bit in an incompatible way.
+        */
+       if (motg->phy_number) {
+               phy_select = devm_ioremap_nocache(&pdev->dev, USB2_PHY_SEL, 4);
+               if (IS_ERR(phy_select))
+                       return PTR_ERR(phy_select);
+               /* Enable second PHY with the OTG port */
+               writel_relaxed(0x1, phy_select);
+       }
+
        dev_info(&pdev->dev, "OTG regs = %p\n", motg->regs);
 
        motg->irq = platform_get_irq(pdev, 0);
index 98d3dd8976e5bf20265b7df95a0e35d67d040e44..a29f6030afb16713fd411ae777e3678b9fdef323 100644 (file)
@@ -16,6 +16,9 @@
 #ifndef __LINUX_USB_GADGET_MSM72K_UDC_H__
 #define __LINUX_USB_GADGET_MSM72K_UDC_H__
 
+/* USB phy selector - in TCSR address range */
+#define USB2_PHY_SEL         0xfd4ab000
+
 #define USB_AHBBURST         (MSM_USB_BASE + 0x0090)
 #define USB_AHBMODE          (MSM_USB_BASE + 0x0098)
 #define USB_CAPLENGTH        (MSM_USB_BASE + 0x0100) /* 8 bit */