1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
2e82a316dSKuo-Jung Su /*
3e82a316dSKuo-Jung Su * Faraday USB 2.0 EHCI Controller
4e82a316dSKuo-Jung Su *
5e82a316dSKuo-Jung Su * (C) Copyright 2010 Faraday Technology
6e82a316dSKuo-Jung Su * Dante Su <dantesu@faraday-tech.com>
7e82a316dSKuo-Jung Su */
8e82a316dSKuo-Jung Su
9e82a316dSKuo-Jung Su #include <common.h>
10e82a316dSKuo-Jung Su #include <asm/io.h>
11e82a316dSKuo-Jung Su #include <usb.h>
12e82a316dSKuo-Jung Su #include <usb/fusbh200.h>
13e82a316dSKuo-Jung Su #include <usb/fotg210.h>
14e82a316dSKuo-Jung Su
15e82a316dSKuo-Jung Su #include "ehci.h"
16e82a316dSKuo-Jung Su
17e82a316dSKuo-Jung Su #ifndef CONFIG_USB_EHCI_BASE_LIST
18e82a316dSKuo-Jung Su #define CONFIG_USB_EHCI_BASE_LIST { CONFIG_USB_EHCI_BASE }
19e82a316dSKuo-Jung Su #endif
20e82a316dSKuo-Jung Su
21e82a316dSKuo-Jung Su union ehci_faraday_regs {
22e82a316dSKuo-Jung Su struct fusbh200_regs usb;
23e82a316dSKuo-Jung Su struct fotg210_regs otg;
24e82a316dSKuo-Jung Su };
25e82a316dSKuo-Jung Su
ehci_is_fotg2xx(union ehci_faraday_regs * regs)26e82a316dSKuo-Jung Su static inline int ehci_is_fotg2xx(union ehci_faraday_regs *regs)
27e82a316dSKuo-Jung Su {
28e82a316dSKuo-Jung Su return !readl(®s->usb.easstr);
29e82a316dSKuo-Jung Su }
30e82a316dSKuo-Jung Su
faraday_ehci_set_usbmode(struct ehci_ctrl * ctrl)31deb8508cSSimon Glass void faraday_ehci_set_usbmode(struct ehci_ctrl *ctrl)
32deb8508cSSimon Glass {
33deb8508cSSimon Glass /* nothing needs to be done */
34deb8508cSSimon Glass }
35deb8508cSSimon Glass
faraday_ehci_get_port_speed(struct ehci_ctrl * ctrl,uint32_t reg)36deb8508cSSimon Glass int faraday_ehci_get_port_speed(struct ehci_ctrl *ctrl, uint32_t reg)
37deb8508cSSimon Glass {
38deb8508cSSimon Glass int spd, ret = PORTSC_PSPD_HS;
39deb8508cSSimon Glass union ehci_faraday_regs *regs;
40deb8508cSSimon Glass
41deb8508cSSimon Glass ret = (void __iomem *)((ulong)ctrl->hcor - 0x10);
42deb8508cSSimon Glass if (ehci_is_fotg2xx(regs))
43deb8508cSSimon Glass spd = OTGCSR_SPD(readl(®s->otg.otgcsr));
44deb8508cSSimon Glass else
45deb8508cSSimon Glass spd = BMCSR_SPD(readl(®s->usb.bmcsr));
46deb8508cSSimon Glass
47deb8508cSSimon Glass switch (spd) {
48deb8508cSSimon Glass case 0: /* full speed */
49deb8508cSSimon Glass ret = PORTSC_PSPD_FS;
50deb8508cSSimon Glass break;
51deb8508cSSimon Glass case 1: /* low speed */
52deb8508cSSimon Glass ret = PORTSC_PSPD_LS;
53deb8508cSSimon Glass break;
54deb8508cSSimon Glass case 2: /* high speed */
55deb8508cSSimon Glass ret = PORTSC_PSPD_HS;
56deb8508cSSimon Glass break;
57deb8508cSSimon Glass default:
58deb8508cSSimon Glass printf("ehci-faraday: invalid device speed\n");
59deb8508cSSimon Glass break;
60deb8508cSSimon Glass }
61deb8508cSSimon Glass
62deb8508cSSimon Glass return ret;
63deb8508cSSimon Glass }
64deb8508cSSimon Glass
faraday_ehci_get_portsc_register(struct ehci_ctrl * ctrl,int port)65deb8508cSSimon Glass uint32_t *faraday_ehci_get_portsc_register(struct ehci_ctrl *ctrl, int port)
66deb8508cSSimon Glass {
67deb8508cSSimon Glass /* Faraday EHCI has one and only one portsc register */
68deb8508cSSimon Glass if (port) {
69deb8508cSSimon Glass /* Printing the message would cause a scan failure! */
70deb8508cSSimon Glass debug("The request port(%d) is not configured\n", port);
71deb8508cSSimon Glass return NULL;
72deb8508cSSimon Glass }
73deb8508cSSimon Glass
74deb8508cSSimon Glass /* Faraday EHCI PORTSC register offset is 0x20 from hcor */
75deb8508cSSimon Glass return (uint32_t *)((uint8_t *)ctrl->hcor + 0x20);
76deb8508cSSimon Glass }
77deb8508cSSimon Glass
78deb8508cSSimon Glass static const struct ehci_ops faraday_ehci_ops = {
79deb8508cSSimon Glass .set_usb_mode = faraday_ehci_set_usbmode,
80deb8508cSSimon Glass .get_port_speed = faraday_ehci_get_port_speed,
81deb8508cSSimon Glass .get_portsc_register = faraday_ehci_get_portsc_register,
82deb8508cSSimon Glass };
83deb8508cSSimon Glass
84e82a316dSKuo-Jung Su /*
85e82a316dSKuo-Jung Su * Create the appropriate control structures to manage
86e82a316dSKuo-Jung Su * a new EHCI host controller.
87e82a316dSKuo-Jung Su */
ehci_hcd_init(int index,enum usb_init_type init,struct ehci_hccr ** ret_hccr,struct ehci_hcor ** ret_hcor)88127efc4fSTroy Kisky int ehci_hcd_init(int index, enum usb_init_type init,
89127efc4fSTroy Kisky struct ehci_hccr **ret_hccr, struct ehci_hcor **ret_hcor)
90e82a316dSKuo-Jung Su {
91e82a316dSKuo-Jung Su struct ehci_hccr *hccr;
92e82a316dSKuo-Jung Su struct ehci_hcor *hcor;
93e82a316dSKuo-Jung Su union ehci_faraday_regs *regs;
94e82a316dSKuo-Jung Su uint32_t base_list[] = CONFIG_USB_EHCI_BASE_LIST;
95e82a316dSKuo-Jung Su
96e82a316dSKuo-Jung Su if (index < 0 || index >= ARRAY_SIZE(base_list))
97e82a316dSKuo-Jung Su return -1;
98deb8508cSSimon Glass ehci_set_controller_priv(index, NULL, &faraday_ehci_ops);
99e82a316dSKuo-Jung Su regs = (void __iomem *)base_list[index];
100e82a316dSKuo-Jung Su hccr = (struct ehci_hccr *)®s->usb.hccr;
101e82a316dSKuo-Jung Su hcor = (struct ehci_hcor *)®s->usb.hcor;
102e82a316dSKuo-Jung Su
103e82a316dSKuo-Jung Su if (ehci_is_fotg2xx(regs)) {
104e82a316dSKuo-Jung Su /* A-device bus reset */
105e82a316dSKuo-Jung Su /* ... Power off A-device */
106e82a316dSKuo-Jung Su setbits_le32(®s->otg.otgcsr, OTGCSR_A_BUSDROP);
107e82a316dSKuo-Jung Su /* ... Drop vbus and bus traffic */
108e82a316dSKuo-Jung Su clrbits_le32(®s->otg.otgcsr, OTGCSR_A_BUSREQ);
109e82a316dSKuo-Jung Su mdelay(1);
110e82a316dSKuo-Jung Su /* ... Power on A-device */
111e82a316dSKuo-Jung Su clrbits_le32(®s->otg.otgcsr, OTGCSR_A_BUSDROP);
112e82a316dSKuo-Jung Su /* ... Drive vbus and bus traffic */
113e82a316dSKuo-Jung Su setbits_le32(®s->otg.otgcsr, OTGCSR_A_BUSREQ);
114e82a316dSKuo-Jung Su mdelay(1);
115e82a316dSKuo-Jung Su /* Disable OTG & DEV interrupts, triggered at level-high */
116e82a316dSKuo-Jung Su writel(IMR_IRQLH | IMR_OTG | IMR_DEV, ®s->otg.imr);
117e82a316dSKuo-Jung Su /* Clear all interrupt status */
118e82a316dSKuo-Jung Su writel(ISR_HOST | ISR_OTG | ISR_DEV, ®s->otg.isr);
119e82a316dSKuo-Jung Su } else {
120e82a316dSKuo-Jung Su /* Interrupt=level-high */
121e82a316dSKuo-Jung Su setbits_le32(®s->usb.bmcsr, BMCSR_IRQLH);
122e82a316dSKuo-Jung Su /* VBUS on */
123e82a316dSKuo-Jung Su clrbits_le32(®s->usb.bmcsr, BMCSR_VBUS_OFF);
124e82a316dSKuo-Jung Su /* Disable all interrupts */
125e82a316dSKuo-Jung Su writel(0x00, ®s->usb.bmier);
126e82a316dSKuo-Jung Su writel(0x1f, ®s->usb.bmisr);
127e82a316dSKuo-Jung Su }
128e82a316dSKuo-Jung Su
129e82a316dSKuo-Jung Su *ret_hccr = hccr;
130e82a316dSKuo-Jung Su *ret_hcor = hcor;
131e82a316dSKuo-Jung Su
132e82a316dSKuo-Jung Su return 0;
133e82a316dSKuo-Jung Su }
134e82a316dSKuo-Jung Su
135e82a316dSKuo-Jung Su /*
136e82a316dSKuo-Jung Su * Destroy the appropriate control structures corresponding
137e82a316dSKuo-Jung Su * the the EHCI host controller.
138e82a316dSKuo-Jung Su */
ehci_hcd_stop(int index)139e82a316dSKuo-Jung Su int ehci_hcd_stop(int index)
140e82a316dSKuo-Jung Su {
141e82a316dSKuo-Jung Su return 0;
142e82a316dSKuo-Jung Su }
143