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