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