blob: b3de8bc6cbeb30761686fe7fb2e152accaaca0fe [file] [log] [blame]
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +04001/*
2 * OHCI HCD (Host Controller Driver) for USB.
3 *
4 * TI DA8xx (OMAP-L1x) Bus Glue
5 *
6 * Derived from: ohci-omap.c and ohci-s3c2410.c
7 * Copyright (C) 2008-2009 MontaVista Software, Inc. <source@mvista.com>
8 *
9 * This file is licensed under the terms of the GNU General Public License
10 * version 2. This program is licensed "as is" without any warranty of any
11 * kind, whether express or implied.
12 */
13
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020014#include <linux/clk.h>
15#include <linux/io.h>
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040016#include <linux/interrupt.h>
17#include <linux/jiffies.h>
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020018#include <linux/kernel.h>
19#include <linux/module.h>
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040020#include <linux/platform_device.h>
David Lechner6110c422016-10-12 20:44:46 -050021#include <linux/phy/phy.h>
Arnd Bergmannec2a0832012-08-24 15:11:34 +020022#include <linux/platform_data/usb-davinci.h>
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020023#include <linux/usb.h>
24#include <linux/usb/hcd.h>
25#include <asm/unaligned.h>
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040026
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020027#include "ohci.h"
28
29#define DRIVER_DESC "DA8XX"
Axel Haslameacae5d2016-11-03 17:03:08 +010030#define DRV_NAME "ohci-da8xx"
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020031
32static struct hc_driver __read_mostly ohci_da8xx_hc_driver;
33
34static int (*orig_ohci_hub_control)(struct usb_hcd *hcd, u16 typeReq,
35 u16 wValue, u16 wIndex, char *buf, u16 wLength);
36static int (*orig_ohci_hub_status_data)(struct usb_hcd *hcd, char *buf);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040037
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040038static struct clk *usb11_clk;
David Lechner6110c422016-10-12 20:44:46 -050039static struct phy *usb11_phy;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040040
41/* Over-current indicator change bitmask */
42static volatile u16 ocic_mask;
43
David Lechner6110c422016-10-12 20:44:46 -050044static int ohci_da8xx_enable(void)
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040045{
David Lechner6110c422016-10-12 20:44:46 -050046 int ret;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040047
David Lechner6110c422016-10-12 20:44:46 -050048 ret = clk_prepare_enable(usb11_clk);
49 if (ret)
50 return ret;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040051
David Lechner6110c422016-10-12 20:44:46 -050052 ret = phy_init(usb11_phy);
53 if (ret)
54 goto err_phy_init;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040055
David Lechner6110c422016-10-12 20:44:46 -050056 ret = phy_power_on(usb11_phy);
57 if (ret)
58 goto err_phy_power_on;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040059
David Lechner6110c422016-10-12 20:44:46 -050060 return 0;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040061
David Lechner6110c422016-10-12 20:44:46 -050062err_phy_power_on:
63 phy_exit(usb11_phy);
64err_phy_init:
65 clk_disable_unprepare(usb11_clk);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040066
David Lechner6110c422016-10-12 20:44:46 -050067 return ret;
68}
69
70static void ohci_da8xx_disable(void)
71{
72 phy_power_off(usb11_phy);
73 phy_exit(usb11_phy);
74 clk_disable_unprepare(usb11_clk);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040075}
76
77/*
78 * Handle the port over-current indicator change.
79 */
80static void ohci_da8xx_ocic_handler(struct da8xx_ohci_root_hub *hub,
81 unsigned port)
82{
83 ocic_mask |= 1 << port;
84
85 /* Once over-current is detected, the port needs to be powered down */
86 if (hub->get_oci(port) > 0)
87 hub->set_power(port, 0);
88}
89
Manjunath Goudar6c21caa2016-10-27 15:52:29 +020090static int ohci_da8xx_reset(struct usb_hcd *hcd)
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040091{
92 struct device *dev = hcd->self.controller;
Jingoo Hand4f09e22013-07-30 19:59:40 +090093 struct da8xx_ohci_root_hub *hub = dev_get_platdata(dev);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +040094 struct ohci_hcd *ohci = hcd_to_ohci(hcd);
95 int result;
96 u32 rh_a;
97
98 dev_dbg(dev, "starting USB controller\n");
99
David Lechner6110c422016-10-12 20:44:46 -0500100 result = ohci_da8xx_enable();
101 if (result < 0)
102 return result;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400103
104 /*
105 * DA8xx only have 1 port connected to the pins but the HC root hub
106 * register A reports 2 ports, thus we'll have to override it...
107 */
108 ohci->num_ports = 1;
109
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200110 result = ohci_setup(hcd);
David Lechner6110c422016-10-12 20:44:46 -0500111 if (result < 0) {
112 ohci_da8xx_disable();
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400113 return result;
David Lechner6110c422016-10-12 20:44:46 -0500114 }
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400115
116 /*
117 * Since we're providing a board-specific root hub port power control
118 * and over-current reporting, we have to override the HC root hub A
119 * register's default value, so that ohci_hub_control() could return
120 * the correct hub descriptor...
121 */
122 rh_a = ohci_readl(ohci, &ohci->regs->roothub.a);
123 if (hub->set_power) {
124 rh_a &= ~RH_A_NPS;
125 rh_a |= RH_A_PSM;
126 }
127 if (hub->get_oci) {
128 rh_a &= ~RH_A_NOCP;
129 rh_a |= RH_A_OCPM;
130 }
131 rh_a &= ~RH_A_POTPGT;
132 rh_a |= hub->potpgt << 24;
133 ohci_writel(ohci, rh_a, &ohci->regs->roothub.a);
134
135 return result;
136}
137
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400138/*
139 * Update the status data from the hub with the over-current indicator change.
140 */
141static int ohci_da8xx_hub_status_data(struct usb_hcd *hcd, char *buf)
142{
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200143 int length = orig_ohci_hub_status_data(hcd, buf);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400144
145 /* See if we have OCIC bit set on port 1 */
146 if (ocic_mask & (1 << 1)) {
147 dev_dbg(hcd->self.controller, "over-current indicator change "
148 "on port 1\n");
149
150 if (!length)
151 length = 1;
152
153 buf[0] |= 1 << 1;
154 }
155 return length;
156}
157
158/*
159 * Look at the control requests to the root hub and see if we need to override.
160 */
161static int ohci_da8xx_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
162 u16 wIndex, char *buf, u16 wLength)
163{
164 struct device *dev = hcd->self.controller;
Jingoo Hand4f09e22013-07-30 19:59:40 +0900165 struct da8xx_ohci_root_hub *hub = dev_get_platdata(dev);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400166 int temp;
167
168 switch (typeReq) {
169 case GetPortStatus:
170 /* Check the port number */
171 if (wIndex != 1)
172 break;
173
174 dev_dbg(dev, "GetPortStatus(%u)\n", wIndex);
175
176 temp = roothub_portstatus(hcd_to_ohci(hcd), wIndex - 1);
177
178 /* The port power status (PPS) bit defaults to 1 */
179 if (hub->get_power && hub->get_power(wIndex) == 0)
180 temp &= ~RH_PS_PPS;
181
182 /* The port over-current indicator (POCI) bit is always 0 */
183 if (hub->get_oci && hub->get_oci(wIndex) > 0)
184 temp |= RH_PS_POCI;
185
186 /* The over-current indicator change (OCIC) bit is 0 too */
187 if (ocic_mask & (1 << wIndex))
188 temp |= RH_PS_OCIC;
189
190 put_unaligned(cpu_to_le32(temp), (__le32 *)buf);
191 return 0;
192 case SetPortFeature:
193 temp = 1;
194 goto check_port;
195 case ClearPortFeature:
196 temp = 0;
197
198check_port:
199 /* Check the port number */
200 if (wIndex != 1)
201 break;
202
203 switch (wValue) {
204 case USB_PORT_FEAT_POWER:
205 dev_dbg(dev, "%sPortFeature(%u): %s\n",
206 temp ? "Set" : "Clear", wIndex, "POWER");
207
208 if (!hub->set_power)
209 return -EPIPE;
210
211 return hub->set_power(wIndex, temp) ? -EPIPE : 0;
212 case USB_PORT_FEAT_C_OVER_CURRENT:
213 dev_dbg(dev, "%sPortFeature(%u): %s\n",
214 temp ? "Set" : "Clear", wIndex,
215 "C_OVER_CURRENT");
216
217 if (temp)
218 ocic_mask |= 1 << wIndex;
219 else
220 ocic_mask &= ~(1 << wIndex);
221 return 0;
222 }
223 }
224
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200225 return orig_ohci_hub_control(hcd, typeReq, wValue,
226 wIndex, buf, wLength);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400227}
228
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400229/*-------------------------------------------------------------------------*/
230
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200231static int ohci_da8xx_probe(struct platform_device *pdev)
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400232{
Jingoo Hand4f09e22013-07-30 19:59:40 +0900233 struct da8xx_ohci_root_hub *hub = dev_get_platdata(&pdev->dev);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400234 struct usb_hcd *hcd;
235 struct resource *mem;
236 int error, irq;
237
238 if (hub == NULL)
239 return -ENODEV;
240
Jingoo Han644db162013-12-11 16:23:39 +0900241 usb11_clk = devm_clk_get(&pdev->dev, "usb11");
David Lechner6110c422016-10-12 20:44:46 -0500242 if (IS_ERR(usb11_clk)) {
243 if (PTR_ERR(usb11_clk) != -EPROBE_DEFER)
244 dev_err(&pdev->dev, "Failed to get clock.\n");
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400245 return PTR_ERR(usb11_clk);
David Lechner6110c422016-10-12 20:44:46 -0500246 }
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400247
David Lechner6110c422016-10-12 20:44:46 -0500248 usb11_phy = devm_phy_get(&pdev->dev, "usb-phy");
249 if (IS_ERR(usb11_phy)) {
250 if (PTR_ERR(usb11_phy) != -EPROBE_DEFER)
251 dev_err(&pdev->dev, "Failed to get phy.\n");
252 return PTR_ERR(usb11_phy);
253 }
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400254
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200255 hcd = usb_create_hcd(&ohci_da8xx_hc_driver, &pdev->dev,
256 dev_name(&pdev->dev));
Jingoo Han644db162013-12-11 16:23:39 +0900257 if (!hcd)
258 return -ENOMEM;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400259
260 mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
Jingoo Han644db162013-12-11 16:23:39 +0900261 hcd->regs = devm_ioremap_resource(&pdev->dev, mem);
262 if (IS_ERR(hcd->regs)) {
263 error = PTR_ERR(hcd->regs);
264 goto err;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400265 }
Varka Bhadram54891d72014-11-04 07:51:09 +0530266 hcd->rsrc_start = mem->start;
267 hcd->rsrc_len = resource_size(mem);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400268
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400269 irq = platform_get_irq(pdev, 0);
270 if (irq < 0) {
271 error = -ENODEV;
Jingoo Han644db162013-12-11 16:23:39 +0900272 goto err;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400273 }
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200274
Yong Zhangb5dd18d2011-09-07 16:10:52 +0800275 error = usb_add_hcd(hcd, irq, 0);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400276 if (error)
Jingoo Han644db162013-12-11 16:23:39 +0900277 goto err;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400278
Peter Chen3c9740a2013-11-05 10:46:02 +0800279 device_wakeup_enable(hcd->self.controller);
280
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400281 if (hub->ocic_notify) {
282 error = hub->ocic_notify(ohci_da8xx_ocic_handler);
283 if (!error)
284 return 0;
285 }
286
287 usb_remove_hcd(hcd);
Jingoo Han644db162013-12-11 16:23:39 +0900288err:
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400289 usb_put_hcd(hcd);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400290 return error;
291}
292
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200293static int ohci_da8xx_remove(struct platform_device *pdev)
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400294{
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200295 struct usb_hcd *hcd = platform_get_drvdata(pdev);
Jingoo Hand4f09e22013-07-30 19:59:40 +0900296 struct da8xx_ohci_root_hub *hub = dev_get_platdata(&pdev->dev);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400297
298 hub->ocic_notify(NULL);
299 usb_remove_hcd(hcd);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400300 usb_put_hcd(hcd);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400301
302 return 0;
303}
304
305#ifdef CONFIG_PM
Majunath Goudar933bb1f2013-11-13 17:40:19 +0530306static int ohci_da8xx_suspend(struct platform_device *pdev,
307 pm_message_t message)
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400308{
Majunath Goudar933bb1f2013-11-13 17:40:19 +0530309 struct usb_hcd *hcd = platform_get_drvdata(pdev);
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400310 struct ohci_hcd *ohci = hcd_to_ohci(hcd);
Majunath Goudar933bb1f2013-11-13 17:40:19 +0530311 bool do_wakeup = device_may_wakeup(&pdev->dev);
312 int ret;
313
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400314
315 if (time_before(jiffies, ohci->next_statechange))
316 msleep(5);
317 ohci->next_statechange = jiffies;
318
Majunath Goudar933bb1f2013-11-13 17:40:19 +0530319 ret = ohci_suspend(hcd, do_wakeup);
320 if (ret)
321 return ret;
322
David Lechner6110c422016-10-12 20:44:46 -0500323 ohci_da8xx_disable();
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400324 hcd->state = HC_STATE_SUSPENDED;
Majunath Goudar933bb1f2013-11-13 17:40:19 +0530325
326 return ret;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400327}
328
329static int ohci_da8xx_resume(struct platform_device *dev)
330{
331 struct usb_hcd *hcd = platform_get_drvdata(dev);
332 struct ohci_hcd *ohci = hcd_to_ohci(hcd);
David Lechner6110c422016-10-12 20:44:46 -0500333 int ret;
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400334
335 if (time_before(jiffies, ohci->next_statechange))
336 msleep(5);
337 ohci->next_statechange = jiffies;
338
David Lechner6110c422016-10-12 20:44:46 -0500339 ret = ohci_da8xx_enable();
340 if (ret)
341 return ret;
342
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400343 dev->dev.power.power_state = PMSG_ON;
344 usb_hcd_resume_root_hub(hcd);
David Lechner6110c422016-10-12 20:44:46 -0500345
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400346 return 0;
347}
348#endif
349
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200350static const struct ohci_driver_overrides da8xx_overrides __initconst = {
351 .reset = ohci_da8xx_reset,
352};
353
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400354/*
355 * Driver definition to register with platform structure.
356 */
357static struct platform_driver ohci_hcd_da8xx_driver = {
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200358 .probe = ohci_da8xx_probe,
359 .remove = ohci_da8xx_remove,
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400360 .shutdown = usb_hcd_platform_shutdown,
361#ifdef CONFIG_PM
362 .suspend = ohci_da8xx_suspend,
363 .resume = ohci_da8xx_resume,
364#endif
365 .driver = {
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200366 .name = DRV_NAME,
Sergei Shtylyovefe7daf2010-02-12 23:52:34 +0400367 },
368};
Jan Luebbeab59ac012012-05-07 10:25:16 +0200369
Manjunath Goudar6c21caa2016-10-27 15:52:29 +0200370static int __init ohci_da8xx_init(void)
371{
372
373 if (usb_disabled())
374 return -ENODEV;
375
376 pr_info("%s: " DRIVER_DESC "\n", DRV_NAME);
377 ohci_init_driver(&ohci_da8xx_hc_driver, &da8xx_overrides);
378
379 /*
380 * The Davinci da8xx HW has some unusual quirks, which require
381 * da8xx-specific workarounds. We override certain hc_driver
382 * functions here to achieve that. We explicitly do not enhance
383 * ohci_driver_overrides to allow this more easily, since this
384 * is an unusual case, and we don't want to encourage others to
385 * override these functions by making it too easy.
386 */
387
388 orig_ohci_hub_control = ohci_da8xx_hc_driver.hub_control;
389 orig_ohci_hub_status_data = ohci_da8xx_hc_driver.hub_status_data;
390
391 ohci_da8xx_hc_driver.hub_status_data = ohci_da8xx_hub_status_data;
392 ohci_da8xx_hc_driver.hub_control = ohci_da8xx_hub_control;
393
394 return platform_driver_register(&ohci_hcd_da8xx_driver);
395}
396module_init(ohci_da8xx_init);
397
398static void __exit ohci_da8xx_exit(void)
399{
400 platform_driver_unregister(&ohci_hcd_da8xx_driver);
401}
402module_exit(ohci_da8xx_exit);
403MODULE_DESCRIPTION(DRIVER_DESC);
404MODULE_LICENSE("GPL");
405MODULE_ALIAS("platform:" DRV_NAME);