blob: 1b1e12eb852697c74411d1a9931b605ec0719ffc [file] [log] [blame]
Angel Pons7c1d70e2020-04-04 18:51:19 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Julius Werner028cba92014-05-30 18:01:44 -07002
Kyösti Mälkki13f66502019-03-03 08:01:05 +02003#include <device/mmio.h>
Julius Werner028cba92014-05-30 18:01:44 -07004#include <console/console.h>
Julius Werner73d1ed62014-10-20 13:20:49 -07005#include <delay.h>
6#include <soc/clock.h>
7#include <soc/iomap.h>
8#include <soc/usb.h>
Julius Werner028cba92014-05-30 18:01:44 -07009
10#define CRPORT_TX_OVRD_DRV_LO 0x1002
11#define CRPORT_RX_OVRD_IN_HI 0x1006
12#define CRPORT_TX_ALT_BLOCK 0x102d
13
Elyes HAOUAS39303d52018-07-08 12:40:45 +020014static u32 *const tcsr_usb_sel = (void *)0x1a4000b0;
Julius Werner028cba92014-05-30 18:01:44 -070015
16struct usb_qc_phy {
17 u32 ipcat;
18 u32 ctrl;
19 u32 general_cfg;
20 u32 ram1;
21 u32 hs_phy_ctrl;
22 u32 param_ovrd;
23 u32 chrg_det_ctrl;
24 u32 chrg_det_output;
25 u32 alt_irq_en;
26 u32 hs_phy_irq_stat;
27 u32 cgctl;
28 u32 dbg_bus;
29 u32 ss_phy_ctrl;
30 u32 ss_phy_param1;
31 u32 ss_phy_param2;
32 u32 crport_data_in;
33 u32 crport_data_out;
34 u32 crport_cap_addr;
35 u32 crport_cap_data;
36 u32 crport_ack_read;
37 u32 crport_ack_write;
38};
39check_member(usb_qc_phy, crport_ack_write, 0x50);
40
41static struct usb_qc_phy * const usb_host1_phy = (void *)USB_HOST1_PHY_BASE;
42static struct usb_qc_phy * const usb_host2_phy = (void *)USB_HOST2_PHY_BASE;
43
44struct usb_dwc3 {
45 u32 sbuscfg0;
46 u32 sbuscfg1;
47 u32 txthrcfg;
48 u32 rxthrcfg;
49 u32 ctl;
50 u32 evten;
51 u32 sts;
52 u8 reserved0[4];
53 u32 snpsid;
54 u32 gpio;
55 u32 uid;
56 u32 uctl;
57 u64 buserraddr;
58 u64 prtbimap;
59 u8 reserved1[32];
60 u32 dbgfifospace;
61 u32 dbgltssm;
62 u32 dbglnmcc;
63 u32 dbgbmu;
64 u32 dbglspmux;
65 u32 dbglsp;
66 u32 dbgepinfo0;
67 u32 dbgepinfo1;
68 u64 prtbimap_hs;
69 u64 prtbimap_fs;
70 u8 reserved2[112];
71 u32 usb2phycfg;
72 u8 reserved3[60];
73 u32 usb2i2cctl;
74 u8 reserved4[60];
75 u32 usb2phyacc;
76 u8 reserved5[60];
77 u32 usb3pipectl;
78 u8 reserved6[60];
79};
80check_member(usb_dwc3, usb3pipectl, 0x1c0);
81
82static struct usb_dwc3 * const usb_host1_dwc3 = (void *)USB_HOST1_DWC3_BASE;
83static struct usb_dwc3 * const usb_host2_dwc3 = (void *)USB_HOST2_DWC3_BASE;
84
85static void setup_dwc3(struct usb_dwc3 *dwc3)
86{
Julius Werner2f37bd62015-02-19 14:51:15 -080087 write32(&dwc3->usb3pipectl,
Julius Werner94184762015-02-19 20:19:23 -080088 0x1 << 31 | /* assert PHY soft reset */
89 0x1 << 25 | /* (default) U1/U2 exit fail -> recovery? */
90 0x1 << 24 | /* (default) activate PHY low power states */
91 0x1 << 19 | /* (default) PHY low power delay value */
92 0x1 << 18 | /* (default) activate PHY low power delay */
93 0x1 << 1 | /* (default) Tx deemphasis value */
94 0x1 << 0); /* (default) elastic buffer mode */
Julius Werner028cba92014-05-30 18:01:44 -070095
Julius Werner2f37bd62015-02-19 14:51:15 -080096 write32(&dwc3->usb2phycfg,
Julius Werner94184762015-02-19 20:19:23 -080097 0x1 << 31 | /* assert PHY soft reset */
98 0x9 << 10 | /* (default) PHY clock turnaround 8-bit UTMI+ */
99 0x1 << 8 | /* (default) enable PHY sleep in L1 */
100 0x1 << 6); /* (default) enable PHY suspend */
Julius Werner028cba92014-05-30 18:01:44 -0700101
Julius Werner2f37bd62015-02-19 14:51:15 -0800102 write32(&dwc3->ctl,
Julius Werner94184762015-02-19 20:19:23 -0800103 0x2 << 19 | /* (default) suspend clock scaling */
104 0x1 << 16 | /* retry SS three times before HS downgrade */
105 0x1 << 12 | /* port capability HOST */
106 0x1 << 11 | /* assert core soft reset */
107 0x1 << 10 | /* (default) sync ITP to refclk */
108 0x1 << 2); /* U2 exit after 8us LFPS (instead of 248ns) */
Julius Werner028cba92014-05-30 18:01:44 -0700109
Julius Werner94184762015-02-19 20:19:23 -0800110 write32(&dwc3->uctl,
111 0x32 << 22 | /* (default) reference clock period in ns */
112 0x1 << 15 | /* (default) XHCI compliant device addressing */
Elyes HAOUAS05498a22018-05-28 16:26:43 +0200113 0x10 << 0); /* (default) devices time out after 32us */
Julius Werner028cba92014-05-30 18:01:44 -0700114
115 udelay(5);
116
Julius Werner55009af2019-12-02 22:03:27 -0800117 clrbits32(&dwc3->ctl, 0x1 << 11); /* deassert core soft reset */
118 clrbits32(&dwc3->usb2phycfg, 0x1 << 31); /* PHY soft reset */
119 clrbits32(&dwc3->usb3pipectl, 0x1 << 31); /* PHY soft reset */
Julius Werner028cba92014-05-30 18:01:44 -0700120}
121
122static void setup_phy(struct usb_qc_phy *phy)
123{
Julius Werner2f37bd62015-02-19 14:51:15 -0800124 write32(&phy->ss_phy_ctrl,
Julius Werner94184762015-02-19 20:19:23 -0800125 0x1 << 24 | /* Indicate VBUS power present */
126 0x1 << 8 | /* Enable USB3 ref clock to prescaler */
127 0x1 << 7 | /* assert SS PHY reset */
128 0x19 << 0); /* (default) reference clock multiplier */
Julius Werner028cba92014-05-30 18:01:44 -0700129
Julius Werner2f37bd62015-02-19 14:51:15 -0800130 write32(&phy->hs_phy_ctrl,
Julius Werner94184762015-02-19 20:19:23 -0800131 0x1 << 26 | /* (default) unclamp DPSE/DMSE VLS */
132 0x1 << 25 | /* (default) select freeclk for utmi_clk */
133 0x1 << 24 | /* (default) unclamp DMSE VLS */
134 0x1 << 21 | /* (default) enable UTMI clock */
135 0x1 << 20 | /* set OTG VBUS as valid */
136 0x1 << 18 | /* use ref clock from core */
137 0x1 << 17 | /* (default) unclamp DPSE VLS */
138 0x1 << 11 | /* force xo/bias/pll to stay on in suspend */
Elyes HAOUAS05498a22018-05-28 16:26:43 +0200139 0x1 << 9 | /* (default) unclamp IDHV */
Julius Werner94184762015-02-19 20:19:23 -0800140 0x1 << 8 | /* (default) unclamp VLS (again???) */
141 0x1 << 7 | /* (default) unclamp HV VLS */
142 0x7 << 4 | /* select frequency (no idea which one) */
143 0x1 << 1); /* (default) "retention enable" */
Julius Werner028cba92014-05-30 18:01:44 -0700144
Julius Werner2f37bd62015-02-19 14:51:15 -0800145 write32(&phy->ss_phy_param1,
Julius Werner94184762015-02-19 20:19:23 -0800146 0x6e << 20 | /* full TX swing amplitude */
147 0x20 << 14 | /* (default) 6dB TX deemphasis */
148 0x17 << 8 | /* 3.5dB TX deemphasis */
149 0x9 << 3); /* (default) LoS detector level */
Julius Werner028cba92014-05-30 18:01:44 -0700150
Julius Werner2f37bd62015-02-19 14:51:15 -0800151 write32(&phy->general_cfg, 0x1 << 2); /* set XHCI 1.00 compliance */
Julius Werner028cba92014-05-30 18:01:44 -0700152
153 udelay(5);
Julius Werner55009af2019-12-02 22:03:27 -0800154 clrbits32(&phy->ss_phy_ctrl, 0x1 << 7); /* deassert SS PHY reset */
Julius Werner028cba92014-05-30 18:01:44 -0700155}
156
157static void crport_handshake(void *capture_reg, void *acknowledge_bit, u32 data)
158{
159 int usec = 100;
160
161 if (capture_reg)
Julius Werner2f37bd62015-02-19 14:51:15 -0800162 write32(capture_reg, data);
Julius Werner028cba92014-05-30 18:01:44 -0700163
Julius Werner2f37bd62015-02-19 14:51:15 -0800164 write32(acknowledge_bit, 0x1 << 0);
Julius Werner028cba92014-05-30 18:01:44 -0700165 while (read32(acknowledge_bit) && --usec)
166 udelay(1);
167
168 if (!usec)
169 printk(BIOS_ERR, "CRPORT handshake timed out (0x%08x)\n", data);
170}
171
172static void crport_write(struct usb_qc_phy *phy, u16 addr, u16 data)
173{
174 crport_handshake(&phy->crport_data_in, &phy->crport_cap_addr, addr);
175 crport_handshake(&phy->crport_data_in, &phy->crport_cap_data, data);
176 crport_handshake(NULL, &phy->crport_ack_write, 0);
177}
178
179static void tune_phy(struct usb_qc_phy *phy)
180{
181 crport_write(phy, CRPORT_RX_OVRD_IN_HI,
182 0x1 << 11 | /* Set RX_EQ override? */
183 0x4 << 8 | /* Set RX_EQ to 4? */
184 0x1 << 7); /* Enable RX_EQ override */
185 crport_write(phy, CRPORT_TX_OVRD_DRV_LO,
186 0x1 << 14 | /* Enable amplitude (override?) */
187 0x17 << 7 | /* Set TX deemphasis to 23 */
188 0x6e << 0); /* Set amplitude to 110 */
189 crport_write(phy, CRPORT_TX_ALT_BLOCK,
190 0x1 << 7); /* ALT block? ("partial RX reset") */
191}
192
193void setup_usb_host1(void)
194{
195 printk(BIOS_INFO, "Setting up USB HOST1 controller...\n");
Julius Werner55009af2019-12-02 22:03:27 -0800196 setbits32(tcsr_usb_sel, 1 << 0); /* Select DWC3 controller */
Julius Werner028cba92014-05-30 18:01:44 -0700197 setup_phy(usb_host1_phy);
198 setup_dwc3(usb_host1_dwc3);
199 tune_phy(usb_host1_phy);
200}
201
202void setup_usb_host2(void)
203{
204 printk(BIOS_INFO, "Setting up USB HOST2 controller...\n");
Julius Werner55009af2019-12-02 22:03:27 -0800205 setbits32(tcsr_usb_sel, 1 << 1); /* Select DWC3 controller */
Julius Werner028cba92014-05-30 18:01:44 -0700206 setup_phy(usb_host2_phy);
207 setup_dwc3(usb_host2_dwc3);
208 tune_phy(usb_host2_phy);
209}