summaryrefslogtreecommitdiff
path: root/src/soc/qualcomm/ipq40xx/usb.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/soc/qualcomm/ipq40xx/usb.c')
-rw-r--r--src/soc/qualcomm/ipq40xx/usb.c430
1 files changed, 251 insertions, 179 deletions
diff --git a/src/soc/qualcomm/ipq40xx/usb.c b/src/soc/qualcomm/ipq40xx/usb.c
index a8d991059f..1ee0e3712f 100644
--- a/src/soc/qualcomm/ipq40xx/usb.c
+++ b/src/soc/qualcomm/ipq40xx/usb.c
@@ -20,203 +20,275 @@
#include <soc/iomap.h>
#include <soc/usb.h>
-#define CRPORT_TX_OVRD_DRV_LO 0x1002
-#define CRPORT_RX_OVRD_IN_HI 0x1006
-#define CRPORT_TX_ALT_BLOCK 0x102d
-
-static u32 * const tcsr_usb_sel = (void *)0x1a4000b0;
-
-struct usb_qc_phy {
- u32 ipcat;
- u32 ctrl;
- u32 general_cfg;
- u32 ram1;
- u32 hs_phy_ctrl;
- u32 param_ovrd;
- u32 chrg_det_ctrl;
- u32 chrg_det_output;
- u32 alt_irq_en;
- u32 hs_phy_irq_stat;
- u32 cgctl;
- u32 dbg_bus;
- u32 ss_phy_ctrl;
- u32 ss_phy_param1;
- u32 ss_phy_param2;
- u32 crport_data_in;
- u32 crport_data_out;
- u32 crport_cap_addr;
- u32 crport_cap_data;
- u32 crport_ack_read;
- u32 crport_ack_write;
-};
-check_member(usb_qc_phy, crport_ack_write, 0x50);
-
-static struct usb_qc_phy * const usb_host1_phy = (void *)USB_HOST1_PHY_BASE;
-static struct usb_qc_phy * const usb_host2_phy = (void *)USB_HOST2_PHY_BASE;
-
-struct usb_dwc3 {
- u32 sbuscfg0;
- u32 sbuscfg1;
- u32 txthrcfg;
- u32 rxthrcfg;
- u32 ctl;
- u32 evten;
- u32 sts;
- u8 reserved0[4];
- u32 snpsid;
- u32 gpio;
- u32 uid;
- u32 uctl;
- u64 buserraddr;
- u64 prtbimap;
- u8 reserved1[32];
- u32 dbgfifospace;
- u32 dbgltssm;
- u32 dbglnmcc;
- u32 dbgbmu;
- u32 dbglspmux;
- u32 dbglsp;
- u32 dbgepinfo0;
- u32 dbgepinfo1;
- u64 prtbimap_hs;
- u64 prtbimap_fs;
- u8 reserved2[112];
- u32 usb2phycfg;
- u8 reserved3[60];
- u32 usb2i2cctl;
- u8 reserved4[60];
- u32 usb2phyacc;
- u8 reserved5[60];
- u32 usb3pipectl;
- u8 reserved6[60];
-};
-check_member(usb_dwc3, usb3pipectl, 0x1c0);
-
-static struct usb_dwc3 * const usb_host1_dwc3 = (void *)USB_HOST1_DWC3_BASE;
-static struct usb_dwc3 * const usb_host2_dwc3 = (void *)USB_HOST2_DWC3_BASE;
-
-static void setup_dwc3(struct usb_dwc3 *dwc3)
-{
- write32(&dwc3->usb3pipectl,
- 0x1 << 31 | /* assert PHY soft reset */
- 0x1 << 25 | /* (default) U1/U2 exit fail -> recovery? */
- 0x1 << 24 | /* (default) activate PHY low power states */
- 0x1 << 19 | /* (default) PHY low power delay value */
- 0x1 << 18 | /* (default) activate PHY low power delay */
- 0x1 << 1 | /* (default) Tx deemphasis value */
- 0x1 << 0); /* (default) elastic buffer mode */
-
- write32(&dwc3->usb2phycfg,
- 0x1 << 31 | /* assert PHY soft reset */
- 0x9 << 10 | /* (default) PHY clock turnaround 8-bit UTMI+ */
- 0x1 << 8 | /* (default) enable PHY sleep in L1 */
- 0x1 << 6); /* (default) enable PHY suspend */
-
- write32(&dwc3->ctl,
- 0x2 << 19 | /* (default) suspend clock scaling */
- 0x1 << 16 | /* retry SS three times before HS downgrade */
- 0x1 << 12 | /* port capability HOST */
- 0x1 << 11 | /* assert core soft reset */
- 0x1 << 10 | /* (default) sync ITP to refclk */
- 0x1 << 2); /* U2 exit after 8us LFPS (instead of 248ns) */
-
- write32(&dwc3->uctl,
- 0x32 << 22 | /* (default) reference clock period in ns */
- 0x1 << 15 | /* (default) XHCI compliant device addressing */
- 0x10 << 0); /* (default) devices time out after 32us */
-
- udelay(5);
-
- clrbits_le32(&dwc3->ctl, 0x1 << 11); /* deassert core soft reset */
- clrbits_le32(&dwc3->usb2phycfg, 0x1 << 31); /* PHY soft reset */
- clrbits_le32(&dwc3->usb3pipectl, 0x1 << 31); /* PHY soft reset */
-}
+/**
+ * USB Hardware registers
+ */
+#define PHY_CTRL0_ADDR 0x000
+#define PHY_CTRL1_ADDR 0x004
+#define PHY_CTRL2_ADDR 0x008
+#define PHY_CTRL3_ADDR 0x00C
+#define PHY_CTRL4_ADDR 0x010
+#define PHY_MISC_ADDR 0x024
+#define PHY_IPG_ADDR 0x030
+
+#define PHY_CTRL0_VAL 0xA4600015
+#define PHY_CTRL1_VAL 0x09500000
+#define PHY_CTRL2_VAL 0x00058180
+#define PHY_CTRL3_VAL 0x6DB6DCD6
+#define PHY_CTRL4_VAL 0x836DB6DB
+#define PHY_MISC_VAL 0x3803FB0C
+#define PHY_IPG_VAL 0x47323232
+
+#define USB_HOST3_PHY_BASE ((void *)0x8a00000)
+#define USB_HOST3_BALDUR_PHY_BASE ((void *)0xa6000)
+#define GCC_USB3_RST_CTRL ((void *)0x0181E038)
+
+#define DWC3_GCTL 0xc110
+#define DWC3_GUSB3PIPECTL(n) (0xc2c0 + (n * 0x04))
+#define DWC3_GUSB2PHYCFG(n) (0xc200 + (n * 0x04))
+
+/* Global USB3 PIPE Control Register */
+#define DWC3_GUSB3PIPECTL_PHYSOFTRST (1 << 31)
+#define DWC3_GUSB3PIPECTL_SUSPHY (1 << 17)
+#define DWC3_GCTL_CORESOFTRESET (1 << 11)
+#define DWC3_GCTL_PRTCAPDIR(n) ((n) << 12)
+#define DWC3_GCTL_PRTCAP_OTG 3
+#define DWC3_DCTL_CSFTRST (1 << 30)
+#define DWC3_GSNPSID 0xc120
+#define DWC3_DCTL 0xc704
+
+
+/* Global USB2 PHY Configuration Register */
+#define DWC3_GUSB2PHYCFG_PHYSOFTRST (1 << 31)
+#define DWC3_GUSB2PHYCFG_SUSPHY (1 << 6)
+#define DWC3_GSNPSID_MASK 0xffff0000
+#define DWC3_GEVTEN 0xc114
+
+
+#define DWC3_GCTL_SCALEDOWN(n) ((n) << 4)
+#define DWC3_GCTL_SCALEDOWN_MASK DWC3_GCTL_SCALEDOWN(3)
+#define DWC3_GCTL_DISSCRAMBLE (1 << 3)
+#define DWC3_GCTL_DSBLCLKGTNG (1 << 0)
+#define DWC3_GCTL_U2RSTECN (1 << 16)
+#define DWC3_REVISION_190A 0x5533190a
+
+#define USB30_HS_PHY_CTRL 0x00000010
+#define SW_SESSVLD (0x01 << 0x1C)
+#define UTMI_OTG_VBUS_VALID (0x01 << 0x14)
+
+#define USB30_SS_PHY_CTRL 0x00000030
+#define LANE0_PWR_PRESENT (0x01 << 0x18)
+
+static void setup_dwc3(void);
-static void setup_phy(struct usb_qc_phy *phy)
+/**
+ * Write register.
+ *
+ * @base - PHY base virtual address.
+ * @offset - register offset.
+ * @val - value to write.
+ */
+static inline void qscratch_write(void *base, u32 offset, u32 val)
{
- write32(&phy->ss_phy_ctrl,
- 0x1 << 24 | /* Indicate VBUS power present */
- 0x1 << 8 | /* Enable USB3 ref clock to prescaler */
- 0x1 << 7 | /* assert SS PHY reset */
- 0x19 << 0); /* (default) reference clock multiplier */
-
- write32(&phy->hs_phy_ctrl,
- 0x1 << 26 | /* (default) unclamp DPSE/DMSE VLS */
- 0x1 << 25 | /* (default) select freeclk for utmi_clk */
- 0x1 << 24 | /* (default) unclamp DMSE VLS */
- 0x1 << 21 | /* (default) enable UTMI clock */
- 0x1 << 20 | /* set OTG VBUS as valid */
- 0x1 << 18 | /* use ref clock from core */
- 0x1 << 17 | /* (default) unclamp DPSE VLS */
- 0x1 << 11 | /* force xo/bias/pll to stay on in suspend */
- 0x1 << 9 | /* (default) unclamp IDHV */
- 0x1 << 8 | /* (default) unclamp VLS (again???) */
- 0x1 << 7 | /* (default) unclamp HV VLS */
- 0x7 << 4 | /* select frequency (no idea which one) */
- 0x1 << 1); /* (default) "retention enable" */
-
- write32(&phy->ss_phy_param1,
- 0x6e << 20 | /* full TX swing amplitude */
- 0x20 << 14 | /* (default) 6dB TX deemphasis */
- 0x17 << 8 | /* 3.5dB TX deemphasis */
- 0x9 << 3); /* (default) LoS detector level */
-
- write32(&phy->general_cfg, 0x1 << 2); /* set XHCI 1.00 compliance */
-
- udelay(5);
- clrbits_le32(&phy->ss_phy_ctrl, 0x1 << 7); /* deassert SS PHY reset */
+ write32(base + offset, val);
}
-static void crport_handshake(void *capture_reg, void *acknowledge_bit, u32 data)
+/**
+ * Write register and read back masked value to confirm it is written
+ *
+ * @base - base virtual address.
+ * @offset - register offset.
+ * @mask - register bitmask specifying what should be updated
+ * @val - value to write.
+ */
+static inline void qscratch_write_readback(void *base, u32 offset,
+ const u32 mask, u32 val)
{
- int usec = 100;
+ u32 write_val, tmp = read32(base + offset);
- if (capture_reg)
- write32(capture_reg, data);
+ tmp &= ~mask; /* retain other bits */
+ write_val = tmp | val;
- write32(acknowledge_bit, 0x1 << 0);
- while (read32(acknowledge_bit) && --usec)
- udelay(1);
+ write32(base + offset, write_val);
- if (!usec)
- printk(BIOS_ERR, "CRPORT handshake timed out (0x%08x)\n", data);
+ /* Read back to see if val was written */
+ tmp = read32(base + offset);
+ tmp &= mask; /* clear other bits */
+
+ if (tmp != val) {
+ printk(BIOS_INFO, "write: %x to QSCRATCH: %x FAILED\n",
+ val, offset);
+ }
}
-static void crport_write(struct usb_qc_phy *phy, u16 addr, u16 data)
+static void dwc3_ipq40xx_enable_vbus_valid(void)
{
- crport_handshake(&phy->crport_data_in, &phy->crport_cap_addr, addr);
- crport_handshake(&phy->crport_data_in, &phy->crport_cap_data, data);
- crport_handshake(NULL, &phy->crport_ack_write, 0);
+ /* Enable VBUS valid for HS PHY*/
+ qscratch_write_readback((void *)0x8af8800, USB30_HS_PHY_CTRL,
+ SW_SESSVLD, SW_SESSVLD);
+ qscratch_write_readback((void *)0x8af8800, USB30_HS_PHY_CTRL,
+ UTMI_OTG_VBUS_VALID, UTMI_OTG_VBUS_VALID);
+
+ /* Enable VBUS valid for SS PHY*/
+ qscratch_write_readback((void *)0x8af8800, USB30_SS_PHY_CTRL,
+ LANE0_PWR_PRESENT, LANE0_PWR_PRESENT);
}
-static void tune_phy(struct usb_qc_phy *phy)
+static void qcom_baldur_hs_phy_init(void)
{
- crport_write(phy, CRPORT_RX_OVRD_IN_HI,
- 0x1 << 11 | /* Set RX_EQ override? */
- 0x4 << 8 | /* Set RX_EQ to 4? */
- 0x1 << 7); /* Enable RX_EQ override */
- crport_write(phy, CRPORT_TX_OVRD_DRV_LO,
- 0x1 << 14 | /* Enable amplitude (override?) */
- 0x17 << 7 | /* Set TX deemphasis to 23 */
- 0x6e << 0); /* Set amplitude to 110 */
- crport_write(phy, CRPORT_TX_ALT_BLOCK,
- 0x1 << 7); /* ALT block? ("partial RX reset") */
+ u32 reg;
+
+ /* assert HS PHY POR reset */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg | 0x10;
+ write32(GCC_USB3_RST_CTRL, reg);
+ mdelay(10);
+
+ /* assert HS PHY SRIF reset */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg | 0x4;
+ write32(GCC_USB3_RST_CTRL, reg);
+ mdelay(10);
+
+ /* deassert HS PHY SRIF reset and program HS PHY registers */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg & ~0x4;
+ write32(GCC_USB3_RST_CTRL, reg);
+
+ mdelay(10);
+
+ /* perform PHY register writes */
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_CTRL0_ADDR, PHY_CTRL0_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_CTRL1_ADDR, PHY_CTRL1_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_CTRL2_ADDR, PHY_CTRL2_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_CTRL3_ADDR, PHY_CTRL3_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_CTRL4_ADDR, PHY_CTRL4_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_MISC_ADDR, PHY_MISC_VAL);
+ write32(USB_HOST3_BALDUR_PHY_BASE + PHY_IPG_ADDR, PHY_IPG_VAL);
+
+ mdelay(10);
+
+ /* de-assert USB3 HS PHY POR reset */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg & ~0x10;
+ write32(GCC_USB3_RST_CTRL, reg);
}
-void setup_usb_host1(void)
+static void qcom_uni_ss_phy_init(void)
+{
+ u32 reg;
+
+ /* assert SS PHY POR reset */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg | 0x20;
+ write32(GCC_USB3_RST_CTRL, reg);
+
+ mdelay(100);
+
+ /* deassert SS PHY POR reset */
+ reg = read32(GCC_USB3_RST_CTRL);
+ reg = reg & ~0x20;
+ write32(GCC_USB3_RST_CTRL, reg);
+}
+
+void setup_dwc3(void)
{
- printk(BIOS_INFO, "Setting up USB HOST1 controller...\n");
- setbits_le32(tcsr_usb_sel, 1 << 0); /* Select DWC3 controller */
- setup_phy(usb_host1_phy);
- setup_dwc3(usb_host1_dwc3);
- tune_phy(usb_host1_phy);
+ u32 reg;
+ u32 revision;
+
+ revision = read32(USB_HOST3_PHY_BASE + DWC3_GSNPSID);
+ /* This should read as U3 followed by revision number */
+ if ((revision & DWC3_GSNPSID_MASK) != 0x55330000)
+ printk(BIOS_INFO, "Error in reading Version\n");
+
+ printk(BIOS_INFO, "Version = %x\n", revision);
+
+ /* issue device SoftReset too */
+ write32(USB_HOST3_PHY_BASE + DWC3_DCTL, DWC3_DCTL_CSFTRST);
+ do {
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_DCTL);
+ if (!(reg & DWC3_DCTL_CSFTRST))
+ break;
+
+ udelay(10);
+ } while (true);
+ printk(BIOS_INFO, "software reset done\n");
+
+ /* Before Resetting PHY, put Core in Reset */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GCTL);
+ reg |= DWC3_GCTL_CORESOFTRESET;
+ write32(USB_HOST3_PHY_BASE + DWC3_GCTL, reg);
+
+ /* Assert USB3 PHY reset */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0));
+ reg |= DWC3_GUSB3PIPECTL_PHYSOFTRST;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0), reg);
+
+ /* Assert USB2 PHY reset */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0));
+ reg |= DWC3_GUSB2PHYCFG_PHYSOFTRST;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0), reg);
+
+ qcom_baldur_hs_phy_init();
+ qcom_uni_ss_phy_init();
+ mdelay(100);
+
+ /* Clear USB3 PHY reset */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0));
+ reg &= ~DWC3_GUSB3PIPECTL_PHYSOFTRST;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0), reg);
+
+ /* Clear USB2 PHY reset */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0));
+ reg &= ~DWC3_GUSB2PHYCFG_PHYSOFTRST;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0), reg);
+
+ mdelay(100);
+
+ /* After PHYs are stable we can take Core out of reset state */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GCTL);
+ reg &= ~DWC3_GCTL_CORESOFTRESET;
+ write32(USB_HOST3_PHY_BASE + DWC3_GCTL, reg);
+
+#if 0
+ /* Enable Suspend USB2.0 HS/FS/LS PHY (SusPHY) */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0));
+ reg |= DWC3_GUSB2PHYCFG_SUSPHY;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB2PHYCFG(0), reg);
+
+ /* Enable Suspend USB3.0 SS PHY (Suspend_en) */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0));
+ reg |= DWC3_GUSB3PIPECTL_SUSPHY;
+ write32(USB_HOST3_PHY_BASE + DWC3_GUSB3PIPECTL(0), reg);
+#endif
+
+ /* configure controller in Host mode */
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GCTL);
+ reg &= ~(DWC3_GCTL_PRTCAPDIR(DWC3_GCTL_PRTCAP_OTG));
+ reg |= DWC3_GCTL_PRTCAPDIR(0x1); /* host mode */
+ write32(USB_HOST3_PHY_BASE + DWC3_GCTL, reg);
+ printk(BIOS_INFO, "USB Host mode reg = %x\n", reg);
+
+ reg = read32(USB_HOST3_PHY_BASE + DWC3_GCTL);
+ reg &= ~DWC3_GCTL_SCALEDOWN_MASK;
+ reg &= ~DWC3_GCTL_DISSCRAMBLE;
+
+ reg &= ~DWC3_GCTL_DSBLCLKGTNG;
+ /*
+ * WORKAROUND: DWC3 revisions <1.90a have a bug
+ * where the device can fail to connect at SuperSpeed
+ * and falls back to high-speed mode which causes
+ * the device to enter a Connect/Disconnect loop
+ */
+ if (revision < DWC3_REVISION_190A)
+ reg |= DWC3_GCTL_U2RSTECN;
+
+ write32(USB_HOST3_PHY_BASE + DWC3_GCTL, reg);
}
-void setup_usb_host2(void)
+void setup_usb_host1(void)
{
- printk(BIOS_INFO, "Setting up USB HOST2 controller...\n");
- setbits_le32(tcsr_usb_sel, 1 << 1); /* Select DWC3 controller */
- setup_phy(usb_host2_phy);
- setup_dwc3(usb_host2_dwc3);
- tune_phy(usb_host2_phy);
+ printk(BIOS_INFO, "Setting up USB HOST1 controller.\n");
+ setup_dwc3();
+ dwc3_ipq40xx_enable_vbus_valid();
}