• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* SPDX-License-Identifier: GPL-2.0-only */
2 
3 #include <device/mmio.h>
4 #include <console/console.h>
5 #include <delay.h>
6 #include <soc/clock.h>
7 #include <soc/iomap.h>
8 #include <soc/usb.h>
9 
10 #define CRPORT_TX_OVRD_DRV_LO	0x1002
11 #define CRPORT_RX_OVRD_IN_HI	0x1006
12 #define CRPORT_TX_ALT_BLOCK	0x102d
13 
14 static u32 *const tcsr_usb_sel = (void *)0x1a4000b0;
15 
16 struct 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 };
39 check_member(usb_qc_phy, crport_ack_write, 0x50);
40 
41 static struct usb_qc_phy * const usb_host1_phy = (void *)USB_HOST1_PHY_BASE;
42 static struct usb_qc_phy * const usb_host2_phy = (void *)USB_HOST2_PHY_BASE;
43 
44 struct 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 };
80 check_member(usb_dwc3, usb3pipectl, 0x1c0);
81 
82 static struct usb_dwc3 * const usb_host1_dwc3 = (void *)USB_HOST1_DWC3_BASE;
83 static struct usb_dwc3 * const usb_host2_dwc3 = (void *)USB_HOST2_DWC3_BASE;
84 
setup_dwc3(struct usb_dwc3 * dwc3)85 static void setup_dwc3(struct usb_dwc3 *dwc3)
86 {
87 	write32(&dwc3->usb3pipectl,
88 		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 */
95 
96 	write32(&dwc3->usb2phycfg,
97 		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 */
101 
102 	write32(&dwc3->ctl,
103 		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) */
109 
110 	write32(&dwc3->uctl,
111 		0x32 << 22 |	/* (default) reference clock period in ns */
112 		 0x1 << 15 |	/* (default) XHCI compliant device addressing */
113 		0x10 << 0);	/* (default) devices time out after 32us */
114 
115 	udelay(5);
116 
117 	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 */
120 }
121 
setup_phy(struct usb_qc_phy * phy)122 static void setup_phy(struct usb_qc_phy *phy)
123 {
124 	write32(&phy->ss_phy_ctrl,
125 		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 */
129 
130 	write32(&phy->hs_phy_ctrl,
131 		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 */
139 		0x1 <<  9 |	/* (default) unclamp IDHV */
140 		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" */
144 
145 	write32(&phy->ss_phy_param1,
146 		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 */
150 
151 	write32(&phy->general_cfg, 0x1 << 2);	/* set XHCI 1.00 compliance */
152 
153 	udelay(5);
154 	clrbits32(&phy->ss_phy_ctrl, 0x1 << 7); /* deassert SS PHY reset */
155 }
156 
crport_handshake(void * capture_reg,void * acknowledge_bit,u32 data)157 static void crport_handshake(void *capture_reg, void *acknowledge_bit, u32 data)
158 {
159 	int usec = 100;
160 
161 	if (capture_reg)
162 		write32(capture_reg, data);
163 
164 	write32(acknowledge_bit, 0x1 << 0);
165 	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 
crport_write(struct usb_qc_phy * phy,u16 addr,u16 data)172 static 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 
tune_phy(struct usb_qc_phy * phy)179 static 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 
setup_usb_host1(void)193 void setup_usb_host1(void)
194 {
195 	printk(BIOS_INFO, "Setting up USB HOST1 controller...\n");
196 	setbits32(tcsr_usb_sel, 1 << 0);	/* Select DWC3 controller */
197 	setup_phy(usb_host1_phy);
198 	setup_dwc3(usb_host1_dwc3);
199 	tune_phy(usb_host1_phy);
200 }
201 
setup_usb_host2(void)202 void setup_usb_host2(void)
203 {
204 	printk(BIOS_INFO, "Setting up USB HOST2 controller...\n");
205 	setbits32(tcsr_usb_sel, 1 << 1);	/* Select DWC3 controller */
206 	setup_phy(usb_host2_phy);
207 	setup_dwc3(usb_host2_dwc3);
208 	tune_phy(usb_host2_phy);
209 }
210