xref: /OK3568_Linux_fs/u-boot/drivers/usb/host/ehci-mxc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright (c) 2009 Daniel Mack <daniel@caiaq.de>
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * SPDX-License-Identifier:	GPL-2.0+
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <common.h>
9*4882a593Smuzhiyun #include <usb.h>
10*4882a593Smuzhiyun #include <asm/io.h>
11*4882a593Smuzhiyun #include <asm/arch/imx-regs.h>
12*4882a593Smuzhiyun #include <usb/ehci-ci.h>
13*4882a593Smuzhiyun #include <errno.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #include "ehci.h"
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #define USBCTRL_OTGBASE_OFFSET	0x600
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun #define MX25_OTG_SIC_SHIFT	29
20*4882a593Smuzhiyun #define MX25_OTG_SIC_MASK	(0x3 << MX25_OTG_SIC_SHIFT)
21*4882a593Smuzhiyun #define MX25_OTG_PM_BIT		(1 << 24)
22*4882a593Smuzhiyun #define MX25_OTG_PP_BIT		(1 << 11)
23*4882a593Smuzhiyun #define MX25_OTG_OCPOL_BIT	(1 << 3)
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #define MX25_H1_SIC_SHIFT	21
26*4882a593Smuzhiyun #define MX25_H1_SIC_MASK	(0x3 << MX25_H1_SIC_SHIFT)
27*4882a593Smuzhiyun #define MX25_H1_PP_BIT		(1 << 18)
28*4882a593Smuzhiyun #define MX25_H1_PM_BIT		(1 << 16)
29*4882a593Smuzhiyun #define MX25_H1_IPPUE_UP_BIT	(1 << 7)
30*4882a593Smuzhiyun #define MX25_H1_IPPUE_DOWN_BIT	(1 << 6)
31*4882a593Smuzhiyun #define MX25_H1_TLL_BIT		(1 << 5)
32*4882a593Smuzhiyun #define MX25_H1_USBTE_BIT	(1 << 4)
33*4882a593Smuzhiyun #define MX25_H1_OCPOL_BIT	(1 << 2)
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun #define MX31_OTG_SIC_SHIFT	29
36*4882a593Smuzhiyun #define MX31_OTG_SIC_MASK	(0x3 << MX31_OTG_SIC_SHIFT)
37*4882a593Smuzhiyun #define MX31_OTG_PM_BIT		(1 << 24)
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun #define MX31_H2_SIC_SHIFT	21
40*4882a593Smuzhiyun #define MX31_H2_SIC_MASK	(0x3 << MX31_H2_SIC_SHIFT)
41*4882a593Smuzhiyun #define MX31_H2_PM_BIT		(1 << 16)
42*4882a593Smuzhiyun #define MX31_H2_DT_BIT		(1 << 5)
43*4882a593Smuzhiyun 
44*4882a593Smuzhiyun #define MX31_H1_SIC_SHIFT	13
45*4882a593Smuzhiyun #define MX31_H1_SIC_MASK	(0x3 << MX31_H1_SIC_SHIFT)
46*4882a593Smuzhiyun #define MX31_H1_PM_BIT		(1 << 8)
47*4882a593Smuzhiyun #define MX31_H1_DT_BIT		(1 << 4)
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun #define MX35_OTG_SIC_SHIFT	29
50*4882a593Smuzhiyun #define MX35_OTG_SIC_MASK	(0x3 << MX35_OTG_SIC_SHIFT)
51*4882a593Smuzhiyun #define MX35_OTG_PM_BIT		(1 << 24)
52*4882a593Smuzhiyun #define MX35_OTG_PP_BIT		(1 << 11)
53*4882a593Smuzhiyun #define MX35_OTG_OCPOL_BIT	(1 << 3)
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun #define MX35_H1_SIC_SHIFT	21
56*4882a593Smuzhiyun #define MX35_H1_SIC_MASK	(0x3 << MX35_H1_SIC_SHIFT)
57*4882a593Smuzhiyun #define MX35_H1_PP_BIT		(1 << 18)
58*4882a593Smuzhiyun #define MX35_H1_PM_BIT		(1 << 16)
59*4882a593Smuzhiyun #define MX35_H1_IPPUE_UP_BIT	(1 << 7)
60*4882a593Smuzhiyun #define MX35_H1_IPPUE_DOWN_BIT	(1 << 6)
61*4882a593Smuzhiyun #define MX35_H1_TLL_BIT		(1 << 5)
62*4882a593Smuzhiyun #define MX35_H1_USBTE_BIT	(1 << 4)
63*4882a593Smuzhiyun #define MX35_H1_OCPOL_BIT	(1 << 2)
64*4882a593Smuzhiyun 
mxc_set_usbcontrol(int port,unsigned int flags)65*4882a593Smuzhiyun static int mxc_set_usbcontrol(int port, unsigned int flags)
66*4882a593Smuzhiyun {
67*4882a593Smuzhiyun 	unsigned int v;
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	v = readl(IMX_USB_BASE + USBCTRL_OTGBASE_OFFSET);
70*4882a593Smuzhiyun #if defined(CONFIG_MX25)
71*4882a593Smuzhiyun 	switch (port) {
72*4882a593Smuzhiyun 	case 0:	/* OTG port */
73*4882a593Smuzhiyun 		v &= ~(MX25_OTG_SIC_MASK | MX25_OTG_PM_BIT | MX25_OTG_PP_BIT |
74*4882a593Smuzhiyun 				MX25_OTG_OCPOL_BIT);
75*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX25_OTG_SIC_SHIFT;
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
78*4882a593Smuzhiyun 			v |= MX25_OTG_PM_BIT;
79*4882a593Smuzhiyun 
80*4882a593Smuzhiyun 		if (flags & MXC_EHCI_PWR_PIN_ACTIVE_HIGH)
81*4882a593Smuzhiyun 			v |= MX25_OTG_PP_BIT;
82*4882a593Smuzhiyun 
83*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_OC_PIN_ACTIVE_LOW))
84*4882a593Smuzhiyun 			v |= MX25_OTG_OCPOL_BIT;
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun 		break;
87*4882a593Smuzhiyun 	case 1: /* H1 port */
88*4882a593Smuzhiyun 		v &= ~(MX25_H1_SIC_MASK | MX25_H1_PM_BIT | MX25_H1_PP_BIT |
89*4882a593Smuzhiyun 				MX25_H1_OCPOL_BIT | MX25_H1_TLL_BIT |
90*4882a593Smuzhiyun 				MX25_H1_USBTE_BIT | MX25_H1_IPPUE_DOWN_BIT |
91*4882a593Smuzhiyun 				MX25_H1_IPPUE_UP_BIT);
92*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX25_H1_SIC_SHIFT;
93*4882a593Smuzhiyun 
94*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
95*4882a593Smuzhiyun 			v |= MX25_H1_PM_BIT;
96*4882a593Smuzhiyun 
97*4882a593Smuzhiyun 		if (flags & MXC_EHCI_PWR_PIN_ACTIVE_HIGH)
98*4882a593Smuzhiyun 			v |= MX25_H1_PP_BIT;
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_OC_PIN_ACTIVE_LOW))
101*4882a593Smuzhiyun 			v |= MX25_H1_OCPOL_BIT;
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_TTL_ENABLED))
104*4882a593Smuzhiyun 			v |= MX25_H1_TLL_BIT;
105*4882a593Smuzhiyun 
106*4882a593Smuzhiyun 		if (flags & MXC_EHCI_INTERNAL_PHY)
107*4882a593Smuzhiyun 			v |= MX25_H1_USBTE_BIT;
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun 		if (flags & MXC_EHCI_IPPUE_DOWN)
110*4882a593Smuzhiyun 			v |= MX25_H1_IPPUE_DOWN_BIT;
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun 		if (flags & MXC_EHCI_IPPUE_UP)
113*4882a593Smuzhiyun 			v |= MX25_H1_IPPUE_UP_BIT;
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun 		break;
116*4882a593Smuzhiyun 	default:
117*4882a593Smuzhiyun 		return -EINVAL;
118*4882a593Smuzhiyun 	}
119*4882a593Smuzhiyun #elif defined(CONFIG_MX31)
120*4882a593Smuzhiyun 	switch (port) {
121*4882a593Smuzhiyun 	case 0:	/* OTG port */
122*4882a593Smuzhiyun 		v &= ~(MX31_OTG_SIC_MASK | MX31_OTG_PM_BIT);
123*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX31_OTG_SIC_SHIFT;
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
126*4882a593Smuzhiyun 			v |= MX31_OTG_PM_BIT;
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun 		break;
129*4882a593Smuzhiyun 	case 1: /* H1 port */
130*4882a593Smuzhiyun 		v &= ~(MX31_H1_SIC_MASK | MX31_H1_PM_BIT | MX31_H1_DT_BIT);
131*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX31_H1_SIC_SHIFT;
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
134*4882a593Smuzhiyun 			v |= MX31_H1_PM_BIT;
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_TTL_ENABLED))
137*4882a593Smuzhiyun 			v |= MX31_H1_DT_BIT;
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun 		break;
140*4882a593Smuzhiyun 	case 2:	/* H2 port */
141*4882a593Smuzhiyun 		v &= ~(MX31_H2_SIC_MASK | MX31_H2_PM_BIT | MX31_H2_DT_BIT);
142*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX31_H2_SIC_SHIFT;
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
145*4882a593Smuzhiyun 			v |= MX31_H2_PM_BIT;
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_TTL_ENABLED))
148*4882a593Smuzhiyun 			v |= MX31_H2_DT_BIT;
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun 		break;
151*4882a593Smuzhiyun 	default:
152*4882a593Smuzhiyun 		return -EINVAL;
153*4882a593Smuzhiyun 	}
154*4882a593Smuzhiyun #elif defined(CONFIG_MX35)
155*4882a593Smuzhiyun 	switch (port) {
156*4882a593Smuzhiyun 	case 0:	/* OTG port */
157*4882a593Smuzhiyun 		v &= ~(MX35_OTG_SIC_MASK | MX35_OTG_PM_BIT | MX35_OTG_PP_BIT |
158*4882a593Smuzhiyun 				MX35_OTG_OCPOL_BIT);
159*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX35_OTG_SIC_SHIFT;
160*4882a593Smuzhiyun 
161*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
162*4882a593Smuzhiyun 			v |= MX35_OTG_PM_BIT;
163*4882a593Smuzhiyun 
164*4882a593Smuzhiyun 		if (flags & MXC_EHCI_PWR_PIN_ACTIVE_HIGH)
165*4882a593Smuzhiyun 			v |= MX35_OTG_PP_BIT;
166*4882a593Smuzhiyun 
167*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_OC_PIN_ACTIVE_LOW))
168*4882a593Smuzhiyun 			v |= MX35_OTG_OCPOL_BIT;
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun 		break;
171*4882a593Smuzhiyun 	case 1: /* H1 port */
172*4882a593Smuzhiyun 		v &= ~(MX35_H1_SIC_MASK | MX35_H1_PM_BIT | MX35_H1_PP_BIT |
173*4882a593Smuzhiyun 				MX35_H1_OCPOL_BIT | MX35_H1_TLL_BIT |
174*4882a593Smuzhiyun 				MX35_H1_USBTE_BIT | MX35_H1_IPPUE_DOWN_BIT |
175*4882a593Smuzhiyun 				MX35_H1_IPPUE_UP_BIT);
176*4882a593Smuzhiyun 		v |= (flags & MXC_EHCI_INTERFACE_MASK) << MX35_H1_SIC_SHIFT;
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_POWER_PINS_ENABLED))
179*4882a593Smuzhiyun 			v |= MX35_H1_PM_BIT;
180*4882a593Smuzhiyun 
181*4882a593Smuzhiyun 		if (flags & MXC_EHCI_PWR_PIN_ACTIVE_HIGH)
182*4882a593Smuzhiyun 			v |= MX35_H1_PP_BIT;
183*4882a593Smuzhiyun 
184*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_OC_PIN_ACTIVE_LOW))
185*4882a593Smuzhiyun 			v |= MX35_H1_OCPOL_BIT;
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun 		if (!(flags & MXC_EHCI_TTL_ENABLED))
188*4882a593Smuzhiyun 			v |= MX35_H1_TLL_BIT;
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun 		if (flags & MXC_EHCI_INTERNAL_PHY)
191*4882a593Smuzhiyun 			v |= MX35_H1_USBTE_BIT;
192*4882a593Smuzhiyun 
193*4882a593Smuzhiyun 		if (flags & MXC_EHCI_IPPUE_DOWN)
194*4882a593Smuzhiyun 			v |= MX35_H1_IPPUE_DOWN_BIT;
195*4882a593Smuzhiyun 
196*4882a593Smuzhiyun 		if (flags & MXC_EHCI_IPPUE_UP)
197*4882a593Smuzhiyun 			v |= MX35_H1_IPPUE_UP_BIT;
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 		break;
200*4882a593Smuzhiyun 	default:
201*4882a593Smuzhiyun 		return -EINVAL;
202*4882a593Smuzhiyun 	}
203*4882a593Smuzhiyun #else
204*4882a593Smuzhiyun #error MXC EHCI USB driver not supported on this platform
205*4882a593Smuzhiyun #endif
206*4882a593Smuzhiyun 	writel(v, IMX_USB_BASE + USBCTRL_OTGBASE_OFFSET);
207*4882a593Smuzhiyun 
208*4882a593Smuzhiyun 	return 0;
209*4882a593Smuzhiyun }
210*4882a593Smuzhiyun 
ehci_hcd_init(int index,enum usb_init_type init,struct ehci_hccr ** hccr,struct ehci_hcor ** hcor)211*4882a593Smuzhiyun int ehci_hcd_init(int index, enum usb_init_type init,
212*4882a593Smuzhiyun 		struct ehci_hccr **hccr, struct ehci_hcor **hcor)
213*4882a593Smuzhiyun {
214*4882a593Smuzhiyun 	struct usb_ehci *ehci;
215*4882a593Smuzhiyun #ifdef CONFIG_MX31
216*4882a593Smuzhiyun 	struct clock_control_regs *sc_regs =
217*4882a593Smuzhiyun 		(struct clock_control_regs *)CCM_BASE;
218*4882a593Smuzhiyun 
219*4882a593Smuzhiyun 	__raw_readl(&sc_regs->ccmr);
220*4882a593Smuzhiyun 	__raw_writel(__raw_readl(&sc_regs->ccmr) | (1 << 9), &sc_regs->ccmr) ;
221*4882a593Smuzhiyun #endif
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun 	udelay(80);
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun 	ehci = (struct usb_ehci *)(IMX_USB_BASE +
226*4882a593Smuzhiyun 			IMX_USB_PORT_OFFSET * CONFIG_MXC_USB_PORT);
227*4882a593Smuzhiyun 	*hccr = (struct ehci_hccr *)((uint32_t)&ehci->caplength);
228*4882a593Smuzhiyun 	*hcor = (struct ehci_hcor *)((uint32_t) *hccr +
229*4882a593Smuzhiyun 			HC_LENGTH(ehci_readl(&(*hccr)->cr_capbase)));
230*4882a593Smuzhiyun 	setbits_le32(&ehci->usbmode, CM_HOST);
231*4882a593Smuzhiyun 	__raw_writel(CONFIG_MXC_USB_PORTSC, &ehci->portsc);
232*4882a593Smuzhiyun 	mxc_set_usbcontrol(CONFIG_MXC_USB_PORT, CONFIG_MXC_USB_FLAGS);
233*4882a593Smuzhiyun #ifdef CONFIG_MX35
234*4882a593Smuzhiyun 	/* Workaround for ENGcm11601 */
235*4882a593Smuzhiyun 	__raw_writel(0, &ehci->sbuscfg);
236*4882a593Smuzhiyun #endif
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun 	udelay(10000);
239*4882a593Smuzhiyun 
240*4882a593Smuzhiyun 	return 0;
241*4882a593Smuzhiyun }
242*4882a593Smuzhiyun 
243*4882a593Smuzhiyun /*
244*4882a593Smuzhiyun  * Destroy the appropriate control structures corresponding
245*4882a593Smuzhiyun  * the the EHCI host controller.
246*4882a593Smuzhiyun  */
ehci_hcd_stop(int index)247*4882a593Smuzhiyun int ehci_hcd_stop(int index)
248*4882a593Smuzhiyun {
249*4882a593Smuzhiyun 	return 0;
250*4882a593Smuzhiyun }
251