aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWenyou Yang <wenyou.yang@atmel.com>2016-08-22 21:05:29 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2016-08-30 13:23:22 -0400
commit2e2aa1bc7eff90ecc1dddfc593aef07c57e539d0 (patch)
tree6f7e1220af52e3dd70ef316403f8d07ec448484f
parentfc8b690d5da86a4b309b74fb706cc1eb75d003e6 (diff)
usb: ohci-at91: Forcibly suspend ports while USB suspend
The usb controller does not manage correctly the suspend mode for the ehci. In echi mode, there is no way to suspend without any device connected to it. This is why this specific control is added to fix this issue. Since the suspend mode works in ohci mode, this specific control works by suspend the usb controller in ohci mode. This specific control is by setting the SUSPEND_A/B/C fields of SFR_OHCIICR(OHCI Interrupt Configuration Register) in the SFR while the OHCI USB suspend. This set operation must be done before the USB clock disabled, clear operation after the USB clock enabled. Signed-off-by: Wenyou Yang <wenyou.yang@atmel.com> Reviewed-by: Alexandre Belloni <alexandre.belloni@free-electrons.com> Acked-by: Nicolas Ferre <nicolas.ferre@atmel.com> Acked-by: Alan Stern <stern@rowland.harvard.edu> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--drivers/usb/host/ohci-at91.c69
-rw-r--r--include/soc/at91/atmel-sfr.h14
2 files changed, 82 insertions, 1 deletions
diff --git a/drivers/usb/host/ohci-at91.c b/drivers/usb/host/ohci-at91.c
index d177372bb357..31102170c7a0 100644
--- a/drivers/usb/host/ohci-at91.c
+++ b/drivers/usb/host/ohci-at91.c
@@ -21,8 +21,11 @@
21#include <linux/io.h> 21#include <linux/io.h>
22#include <linux/kernel.h> 22#include <linux/kernel.h>
23#include <linux/module.h> 23#include <linux/module.h>
24#include <linux/mfd/syscon.h>
25#include <linux/regmap.h>
24#include <linux/usb.h> 26#include <linux/usb.h>
25#include <linux/usb/hcd.h> 27#include <linux/usb/hcd.h>
28#include <soc/at91/atmel-sfr.h>
26 29
27#include "ohci.h" 30#include "ohci.h"
28 31
@@ -51,6 +54,7 @@ struct ohci_at91_priv {
51 struct clk *hclk; 54 struct clk *hclk;
52 bool clocked; 55 bool clocked;
53 bool wakeup; /* Saved wake-up state for resume */ 56 bool wakeup; /* Saved wake-up state for resume */
57 struct regmap *sfr_regmap;
54}; 58};
55/* interface and function clocks; sometimes also an AHB clock */ 59/* interface and function clocks; sometimes also an AHB clock */
56 60
@@ -134,6 +138,17 @@ static void at91_stop_hc(struct platform_device *pdev)
134 138
135static void usb_hcd_at91_remove (struct usb_hcd *, struct platform_device *); 139static void usb_hcd_at91_remove (struct usb_hcd *, struct platform_device *);
136 140
141struct regmap *at91_dt_syscon_sfr(void)
142{
143 struct regmap *regmap;
144
145 regmap = syscon_regmap_lookup_by_compatible("atmel,sama5d2-sfr");
146 if (IS_ERR(regmap))
147 regmap = NULL;
148
149 return regmap;
150}
151
137/* configure so an HC device and id are always provided */ 152/* configure so an HC device and id are always provided */
138/* always called with process context; sleeping is OK */ 153/* always called with process context; sleeping is OK */
139 154
@@ -197,6 +212,10 @@ static int usb_hcd_at91_probe(const struct hc_driver *driver,
197 goto err; 212 goto err;
198 } 213 }
199 214
215 ohci_at91->sfr_regmap = at91_dt_syscon_sfr();
216 if (!ohci_at91->sfr_regmap)
217 dev_warn(dev, "failed to find sfr node\n");
218
200 board = hcd->self.controller->platform_data; 219 board = hcd->self.controller->platform_data;
201 ohci = hcd_to_ohci(hcd); 220 ohci = hcd_to_ohci(hcd);
202 ohci->num_ports = board->ports; 221 ohci->num_ports = board->ports;
@@ -282,6 +301,28 @@ static int ohci_at91_hub_status_data(struct usb_hcd *hcd, char *buf)
282 return length; 301 return length;
283} 302}
284 303
304static int ohci_at91_port_suspend(struct regmap *regmap, u8 set)
305{
306 u32 regval;
307 int ret;
308
309 if (!regmap)
310 return 0;
311
312 ret = regmap_read(regmap, AT91_SFR_OHCIICR, &regval);
313 if (ret)
314 return ret;
315
316 if (set)
317 regval |= AT91_OHCIICR_USB_SUSPEND;
318 else
319 regval &= ~AT91_OHCIICR_USB_SUSPEND;
320
321 regmap_write(regmap, AT91_SFR_OHCIICR, regval);
322
323 return 0;
324}
325
285/* 326/*
286 * Look at the control requests to the root hub and see if we need to override. 327 * Look at the control requests to the root hub and see if we need to override.
287 */ 328 */
@@ -289,6 +330,7 @@ static int ohci_at91_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
289 u16 wIndex, char *buf, u16 wLength) 330 u16 wIndex, char *buf, u16 wLength)
290{ 331{
291 struct at91_usbh_data *pdata = dev_get_platdata(hcd->self.controller); 332 struct at91_usbh_data *pdata = dev_get_platdata(hcd->self.controller);
333 struct ohci_at91_priv *ohci_at91 = hcd_to_ohci_at91_priv(hcd);
292 struct usb_hub_descriptor *desc; 334 struct usb_hub_descriptor *desc;
293 int ret = -EINVAL; 335 int ret = -EINVAL;
294 u32 *data = (u32 *)buf; 336 u32 *data = (u32 *)buf;
@@ -301,7 +343,8 @@ static int ohci_at91_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
301 343
302 switch (typeReq) { 344 switch (typeReq) {
303 case SetPortFeature: 345 case SetPortFeature:
304 if (wValue == USB_PORT_FEAT_POWER) { 346 switch (wValue) {
347 case USB_PORT_FEAT_POWER:
305 dev_dbg(hcd->self.controller, "SetPortFeat: POWER\n"); 348 dev_dbg(hcd->self.controller, "SetPortFeat: POWER\n");
306 if (valid_port(wIndex)) { 349 if (valid_port(wIndex)) {
307 ohci_at91_usb_set_power(pdata, wIndex, 1); 350 ohci_at91_usb_set_power(pdata, wIndex, 1);
@@ -309,6 +352,15 @@ static int ohci_at91_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
309 } 352 }
310 353
311 goto out; 354 goto out;
355
356 case USB_PORT_FEAT_SUSPEND:
357 dev_dbg(hcd->self.controller, "SetPortFeat: SUSPEND\n");
358 if (valid_port(wIndex)) {
359 ohci_at91_port_suspend(ohci_at91->sfr_regmap,
360 1);
361 return 0;
362 }
363 break;
312 } 364 }
313 break; 365 break;
314 366
@@ -342,6 +394,16 @@ static int ohci_at91_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
342 ohci_at91_usb_set_power(pdata, wIndex, 0); 394 ohci_at91_usb_set_power(pdata, wIndex, 0);
343 return 0; 395 return 0;
344 } 396 }
397 break;
398
399 case USB_PORT_FEAT_SUSPEND:
400 dev_dbg(hcd->self.controller, "ClearPortFeature: SUSPEND\n");
401 if (valid_port(wIndex)) {
402 ohci_at91_port_suspend(ohci_at91->sfr_regmap,
403 0);
404 return 0;
405 }
406 break;
345 } 407 }
346 break; 408 break;
347 } 409 }
@@ -599,6 +661,8 @@ ohci_hcd_at91_drv_suspend(struct device *dev)
599 if (ohci_at91->wakeup) 661 if (ohci_at91->wakeup)
600 enable_irq_wake(hcd->irq); 662 enable_irq_wake(hcd->irq);
601 663
664 ohci_at91_port_suspend(ohci_at91->sfr_regmap, 1);
665
602 ret = ohci_suspend(hcd, ohci_at91->wakeup); 666 ret = ohci_suspend(hcd, ohci_at91->wakeup);
603 if (ret) { 667 if (ret) {
604 if (ohci_at91->wakeup) 668 if (ohci_at91->wakeup)
@@ -638,6 +702,9 @@ ohci_hcd_at91_drv_resume(struct device *dev)
638 at91_start_clock(ohci_at91); 702 at91_start_clock(ohci_at91);
639 703
640 ohci_resume(hcd, false); 704 ohci_resume(hcd, false);
705
706 ohci_at91_port_suspend(ohci_at91->sfr_regmap, 0);
707
641 return 0; 708 return 0;
642} 709}
643 710
diff --git a/include/soc/at91/atmel-sfr.h b/include/soc/at91/atmel-sfr.h
index 2f9bb984a4df..506ea8ffda19 100644
--- a/include/soc/at91/atmel-sfr.h
+++ b/include/soc/at91/atmel-sfr.h
@@ -13,6 +13,20 @@
13#ifndef _LINUX_MFD_SYSCON_ATMEL_SFR_H 13#ifndef _LINUX_MFD_SYSCON_ATMEL_SFR_H
14#define _LINUX_MFD_SYSCON_ATMEL_SFR_H 14#define _LINUX_MFD_SYSCON_ATMEL_SFR_H
15 15
16#define AT91_SFR_DDRCFG 0x04 /* DDR Configuration Register */
17/* 0x08 ~ 0x0c: Reserved */
18#define AT91_SFR_OHCIICR 0x10 /* OHCI INT Configuration Register */
19#define AT91_SFR_OHCIISR 0x14 /* OHCI INT Status Register */
16#define AT91_SFR_I2SCLKSEL 0x90 /* I2SC Register */ 20#define AT91_SFR_I2SCLKSEL 0x90 /* I2SC Register */
17 21
22/* Field definitions */
23#define AT91_OHCIICR_SUSPEND_A BIT(8)
24#define AT91_OHCIICR_SUSPEND_B BIT(9)
25#define AT91_OHCIICR_SUSPEND_C BIT(10)
26
27#define AT91_OHCIICR_USB_SUSPEND (AT91_OHCIICR_SUSPEND_A | \
28 AT91_OHCIICR_SUSPEND_B | \
29 AT91_OHCIICR_SUSPEND_C)
30
31
18#endif /* _LINUX_MFD_SYSCON_ATMEL_SFR_H */ 32#endif /* _LINUX_MFD_SYSCON_ATMEL_SFR_H */