diff options
| author | David Brownell <david-b@pacbell.net> | 2006-06-19 17:27:20 -0400 | 
|---|---|---|
| committer | Greg Kroah-Hartman <gregkh@suse.de> | 2006-08-02 19:41:41 -0400 | 
| commit | 0365ee0a8f7450c5e79302930d461e58161a96a1 (patch) | |
| tree | 067ba40e1776b036735e1ac60dfa2a6d27146bc0 | |
| parent | 8b2e76687b39213725207b4a4264e11e8c7b86e6 (diff) | |
USB: AT91 OHCI updates, mostly power management
OHCI updates for AT91 series processors:
 - Get ready for at91sam926x processors (ARMv5tej not ARMv4t)
 - Suspend/resume support now behaves properly
 - In "standby" mode, OHCI can be a source of system wakeup events
   (remote wakeup, device connect/disconnect, etc)
And minor cleanups.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Victor <andrew@sanpeople.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
| -rw-r--r-- | drivers/usb/Kconfig | 2 | ||||
| -rw-r--r-- | drivers/usb/host/ohci-at91.c | 88 | ||||
| -rw-r--r-- | drivers/usb/host/ohci-hcd.c | 3 | 
3 files changed, 62 insertions, 31 deletions
| diff --git a/drivers/usb/Kconfig b/drivers/usb/Kconfig index 2ee742d40c43..005043197527 100644 --- a/drivers/usb/Kconfig +++ b/drivers/usb/Kconfig | |||
| @@ -24,7 +24,7 @@ config USB_ARCH_HAS_OHCI | |||
| 24 | default y if ARCH_S3C2410 | 24 | default y if ARCH_S3C2410 | 
| 25 | default y if PXA27x | 25 | default y if PXA27x | 
| 26 | default y if ARCH_EP93XX | 26 | default y if ARCH_EP93XX | 
| 27 | default y if ARCH_AT91RM9200 | 27 | default y if (ARCH_AT91RM9200 || ARCH_AT91SAM9261) | 
| 28 | # PPC: | 28 | # PPC: | 
| 29 | default y if STB03xxx | 29 | default y if STB03xxx | 
| 30 | default y if PPC_MPC52xx | 30 | default y if PPC_MPC52xx | 
| diff --git a/drivers/usb/host/ohci-at91.c b/drivers/usb/host/ohci-at91.c index cdbafb710000..85cc059705a6 100644 --- a/drivers/usb/host/ohci-at91.c +++ b/drivers/usb/host/ohci-at91.c | |||
| @@ -4,7 +4,7 @@ | |||
| 4 | * Copyright (C) 2004 SAN People (Pty) Ltd. | 4 | * Copyright (C) 2004 SAN People (Pty) Ltd. | 
| 5 | * Copyright (C) 2005 Thibaut VARENE <varenet@parisc-linux.org> | 5 | * Copyright (C) 2005 Thibaut VARENE <varenet@parisc-linux.org> | 
| 6 | * | 6 | * | 
| 7 | * AT91RM9200 Bus Glue | 7 | * AT91 Bus Glue | 
| 8 | * | 8 | * | 
| 9 | * Based on fragments of 2.4 driver by Rick Bronson. | 9 | * Based on fragments of 2.4 driver by Rick Bronson. | 
| 10 | * Based on ohci-omap.c | 10 | * Based on ohci-omap.c | 
| @@ -19,12 +19,13 @@ | |||
| 19 | #include <asm/hardware.h> | 19 | #include <asm/hardware.h> | 
| 20 | #include <asm/arch/board.h> | 20 | #include <asm/arch/board.h> | 
| 21 | 21 | ||
| 22 | #ifndef CONFIG_ARCH_AT91RM9200 | 22 | #ifndef CONFIG_ARCH_AT91 | 
| 23 | #error "CONFIG_ARCH_AT91RM9200 must be defined." | 23 | #error "CONFIG_ARCH_AT91 must be defined." | 
| 24 | #endif | 24 | #endif | 
| 25 | 25 | ||
| 26 | /* interface and function clocks */ | 26 | /* interface and function clocks */ | 
| 27 | static struct clk *iclk, *fclk; | 27 | static struct clk *iclk, *fclk; | 
| 28 | static int clocked; | ||
| 28 | 29 | ||
| 29 | extern int usb_disabled(void); | 30 | extern int usb_disabled(void); | 
| 30 | 31 | ||
| @@ -35,13 +36,14 @@ static void at91_start_hc(struct platform_device *pdev) | |||
| 35 | struct usb_hcd *hcd = platform_get_drvdata(pdev); | 36 | struct usb_hcd *hcd = platform_get_drvdata(pdev); | 
| 36 | struct ohci_regs __iomem *regs = hcd->regs; | 37 | struct ohci_regs __iomem *regs = hcd->regs; | 
| 37 | 38 | ||
| 38 | dev_dbg(&pdev->dev, "starting AT91RM9200 OHCI USB Controller\n"); | 39 | dev_dbg(&pdev->dev, "start\n"); | 
| 39 | 40 | ||
| 40 | /* | 41 | /* | 
| 41 | * Start the USB clocks. | 42 | * Start the USB clocks. | 
| 42 | */ | 43 | */ | 
| 43 | clk_enable(iclk); | 44 | clk_enable(iclk); | 
| 44 | clk_enable(fclk); | 45 | clk_enable(fclk); | 
| 46 | clocked = 1; | ||
| 45 | 47 | ||
| 46 | /* | 48 | /* | 
| 47 | * The USB host controller must remain in reset. | 49 | * The USB host controller must remain in reset. | 
| @@ -54,7 +56,7 @@ static void at91_stop_hc(struct platform_device *pdev) | |||
| 54 | struct usb_hcd *hcd = platform_get_drvdata(pdev); | 56 | struct usb_hcd *hcd = platform_get_drvdata(pdev); | 
| 55 | struct ohci_regs __iomem *regs = hcd->regs; | 57 | struct ohci_regs __iomem *regs = hcd->regs; | 
| 56 | 58 | ||
| 57 | dev_dbg(&pdev->dev, "stopping AT91RM9200 OHCI USB Controller\n"); | 59 | dev_dbg(&pdev->dev, "stop\n"); | 
| 58 | 60 | ||
| 59 | /* | 61 | /* | 
| 60 | * Put the USB host controller into reset. | 62 | * Put the USB host controller into reset. | 
| @@ -66,6 +68,7 @@ static void at91_stop_hc(struct platform_device *pdev) | |||
| 66 | */ | 68 | */ | 
| 67 | clk_disable(fclk); | 69 | clk_disable(fclk); | 
| 68 | clk_disable(iclk); | 70 | clk_disable(iclk); | 
| 71 | clocked = 0; | ||
| 69 | } | 72 | } | 
| 70 | 73 | ||
| 71 | 74 | ||
| @@ -78,14 +81,15 @@ static int usb_hcd_at91_remove (struct usb_hcd *, struct platform_device *); | |||
| 78 | 81 | ||
| 79 | 82 | ||
| 80 | /** | 83 | /** | 
| 81 | * usb_hcd_at91_probe - initialize AT91RM9200-based HCDs | 84 | * usb_hcd_at91_probe - initialize AT91-based HCDs | 
| 82 | * Context: !in_interrupt() | 85 | * Context: !in_interrupt() | 
| 83 | * | 86 | * | 
| 84 | * Allocates basic resources for this USB host controller, and | 87 | * Allocates basic resources for this USB host controller, and | 
| 85 | * then invokes the start() method for the HCD associated with it | 88 | * then invokes the start() method for the HCD associated with it | 
| 86 | * through the hotplug entry's driver_data. | 89 | * through the hotplug entry's driver_data. | 
| 87 | */ | 90 | */ | 
| 88 | int usb_hcd_at91_probe (const struct hc_driver *driver, struct platform_device *pdev) | 91 | static int usb_hcd_at91_probe(const struct hc_driver *driver, | 
| 92 | struct platform_device *pdev) | ||
| 89 | { | 93 | { | 
| 90 | int retval; | 94 | int retval; | 
| 91 | struct usb_hcd *hcd = NULL; | 95 | struct usb_hcd *hcd = NULL; | 
| @@ -95,12 +99,13 @@ int usb_hcd_at91_probe (const struct hc_driver *driver, struct platform_device * | |||
| 95 | return -ENODEV; | 99 | return -ENODEV; | 
| 96 | } | 100 | } | 
| 97 | 101 | ||
| 98 | if ((pdev->resource[0].flags != IORESOURCE_MEM) || (pdev->resource[1].flags != IORESOURCE_IRQ)) { | 102 | if ((pdev->resource[0].flags != IORESOURCE_MEM) | 
| 103 | || (pdev->resource[1].flags != IORESOURCE_IRQ)) { | ||
| 99 | pr_debug("hcd probe: invalid resource type\n"); | 104 | pr_debug("hcd probe: invalid resource type\n"); | 
| 100 | return -ENODEV; | 105 | return -ENODEV; | 
| 101 | } | 106 | } | 
| 102 | 107 | ||
| 103 | hcd = usb_create_hcd(driver, &pdev->dev, "at91rm9200"); | 108 | hcd = usb_create_hcd(driver, &pdev->dev, "at91"); | 
| 104 | if (!hcd) | 109 | if (!hcd) | 
| 105 | return -ENOMEM; | 110 | return -ENOMEM; | 
| 106 | hcd->rsrc_start = pdev->resource[0].start; | 111 | hcd->rsrc_start = pdev->resource[0].start; | 
| @@ -149,21 +154,23 @@ int usb_hcd_at91_probe (const struct hc_driver *driver, struct platform_device * | |||
| 149 | /* may be called with controller, bus, and devices active */ | 154 | /* may be called with controller, bus, and devices active */ | 
| 150 | 155 | ||
| 151 | /** | 156 | /** | 
| 152 | * usb_hcd_at91_remove - shutdown processing for AT91RM9200-based HCDs | 157 | * usb_hcd_at91_remove - shutdown processing for AT91-based HCDs | 
| 153 | * @dev: USB Host Controller being removed | 158 | * @dev: USB Host Controller being removed | 
| 154 | * Context: !in_interrupt() | 159 | * Context: !in_interrupt() | 
| 155 | * | 160 | * | 
| 156 | * Reverses the effect of usb_hcd_at91_probe(), first invoking | 161 | * Reverses the effect of usb_hcd_at91_probe(), first invoking | 
| 157 | * the HCD's stop() method. It is always called from a thread | 162 | * the HCD's stop() method. It is always called from a thread | 
| 158 | * context, normally "rmmod", "apmd", or something similar. | 163 | * context, "rmmod" or something similar. | 
| 159 | * | 164 | * | 
| 160 | */ | 165 | */ | 
| 161 | static int usb_hcd_at91_remove (struct usb_hcd *hcd, struct platform_device *pdev) | 166 | static int usb_hcd_at91_remove(struct usb_hcd *hcd, | 
| 167 | struct platform_device *pdev) | ||
| 162 | { | 168 | { | 
| 163 | usb_remove_hcd(hcd); | 169 | usb_remove_hcd(hcd); | 
| 164 | at91_stop_hc(pdev); | 170 | at91_stop_hc(pdev); | 
| 165 | iounmap(hcd->regs); | 171 | iounmap(hcd->regs); | 
| 166 | release_mem_region(hcd->rsrc_start, hcd->rsrc_len); | 172 | release_mem_region(hcd->rsrc_start, hcd->rsrc_len); | 
| 173 | disable_irq_wake(hcd->irq); | ||
| 167 | 174 | ||
| 168 | clk_put(fclk); | 175 | clk_put(fclk); | 
| 169 | clk_put(iclk); | 176 | clk_put(iclk); | 
| @@ -178,19 +185,21 @@ static int usb_hcd_at91_remove (struct usb_hcd *hcd, struct platform_device *pde | |||
| 178 | static int __devinit | 185 | static int __devinit | 
| 179 | ohci_at91_start (struct usb_hcd *hcd) | 186 | ohci_at91_start (struct usb_hcd *hcd) | 
| 180 | { | 187 | { | 
| 181 | // struct at91_ohci_data *board = hcd->self.controller->platform_data; | 188 | struct at91_usbh_data *board = hcd->self.controller->platform_data; | 
| 182 | struct ohci_hcd *ohci = hcd_to_ohci (hcd); | 189 | struct ohci_hcd *ohci = hcd_to_ohci (hcd); | 
| 190 | struct usb_device *root = hcd->self.root_hub; | ||
| 183 | int ret; | 191 | int ret; | 
| 184 | 192 | ||
| 185 | if ((ret = ohci_init(ohci)) < 0) | 193 | if ((ret = ohci_init(ohci)) < 0) | 
| 186 | return ret; | 194 | return ret; | 
| 187 | 195 | ||
| 196 | root->maxchild = board->ports; | ||
| 197 | |||
| 188 | if ((ret = ohci_run(ohci)) < 0) { | 198 | if ((ret = ohci_run(ohci)) < 0) { | 
| 189 | err("can't start %s", hcd->self.bus_name); | 199 | err("can't start %s", hcd->self.bus_name); | 
| 190 | ohci_stop(hcd); | 200 | ohci_stop(hcd); | 
| 191 | return ret; | 201 | return ret; | 
| 192 | } | 202 | } | 
| 193 | // hcd->self.root_hub->maxchild = board->ports; | ||
| 194 | return 0; | 203 | return 0; | 
| 195 | } | 204 | } | 
| 196 | 205 | ||
| @@ -198,7 +207,7 @@ ohci_at91_start (struct usb_hcd *hcd) | |||
| 198 | 207 | ||
| 199 | static const struct hc_driver ohci_at91_hc_driver = { | 208 | static const struct hc_driver ohci_at91_hc_driver = { | 
| 200 | .description = hcd_name, | 209 | .description = hcd_name, | 
| 201 | .product_desc = "AT91RM9200 OHCI", | 210 | .product_desc = "AT91 OHCI", | 
| 202 | .hcd_priv_size = sizeof(struct ohci_hcd), | 211 | .hcd_priv_size = sizeof(struct ohci_hcd), | 
| 203 | 212 | ||
| 204 | /* | 213 | /* | 
| @@ -240,33 +249,54 @@ static const struct hc_driver ohci_at91_hc_driver = { | |||
| 240 | 249 | ||
| 241 | /*-------------------------------------------------------------------------*/ | 250 | /*-------------------------------------------------------------------------*/ | 
| 242 | 251 | ||
| 243 | static int ohci_hcd_at91_drv_probe(struct platform_device *dev) | 252 | static int ohci_hcd_at91_drv_probe(struct platform_device *pdev) | 
| 244 | { | 253 | { | 
| 245 | return usb_hcd_at91_probe(&ohci_at91_hc_driver, dev); | 254 | device_init_wakeup(&pdev->dev, 1); | 
| 255 | return usb_hcd_at91_probe(&ohci_at91_hc_driver, pdev); | ||
| 246 | } | 256 | } | 
| 247 | 257 | ||
| 248 | static int ohci_hcd_at91_drv_remove(struct platform_device *dev) | 258 | static int ohci_hcd_at91_drv_remove(struct platform_device *pdev) | 
| 249 | { | 259 | { | 
| 250 | return usb_hcd_at91_remove(platform_get_drvdata(dev), dev); | 260 | device_init_wakeup(&pdev->dev, 0); | 
| 261 | return usb_hcd_at91_remove(platform_get_drvdata(pdev), pdev); | ||
| 251 | } | 262 | } | 
| 252 | 263 | ||
| 253 | #ifdef CONFIG_PM | 264 | #ifdef CONFIG_PM | 
| 254 | 265 | ||
| 255 | /* REVISIT suspend/resume look "too" simple here */ | ||
| 256 | |||
| 257 | static int | 266 | static int | 
| 258 | ohci_hcd_at91_drv_suspend(struct platform_device *dev, pm_message_t mesg) | 267 | ohci_hcd_at91_drv_suspend(struct platform_device *pdev, pm_message_t mesg) | 
| 259 | { | 268 | { | 
| 260 | clk_disable(fclk); | 269 | struct usb_hcd *hcd = platform_get_drvdata(pdev); | 
| 261 | clk_disable(iclk); | 270 | struct ohci_hcd *ohci = hcd_to_ohci(hcd); | 
| 271 | |||
| 272 | if (device_may_wakeup(&pdev->dev)) | ||
| 273 | enable_irq_wake(hcd->irq); | ||
| 274 | else | ||
| 275 | disable_irq_wake(hcd->irq); | ||
| 276 | |||
| 277 | /* | ||
| 278 | * The integrated transceivers seem unable to notice disconnect, | ||
| 279 | * reconnect, or wakeup without the 48 MHz clock active. so for | ||
| 280 | * correctness, always discard connection state (using reset). | ||
| 281 | * | ||
| 282 | * REVISIT: some boards will be able to turn VBUS off... | ||
| 283 | */ | ||
| 284 | if (at91_suspend_entering_slow_clock()) { | ||
| 285 | ohci_usb_reset (ohci); | ||
| 286 | clk_disable(fclk); | ||
| 287 | clk_disable(iclk); | ||
| 288 | clocked = 0; | ||
| 289 | } | ||
| 262 | 290 | ||
| 263 | return 0; | 291 | return 0; | 
| 264 | } | 292 | } | 
| 265 | 293 | ||
| 266 | static int ohci_hcd_at91_drv_resume(struct platform_device *dev) | 294 | static int ohci_hcd_at91_drv_resume(struct platform_device *pdev) | 
| 267 | { | 295 | { | 
| 268 | clk_enable(iclk); | 296 | if (!clocked) { | 
| 269 | clk_enable(fclk); | 297 | clk_enable(iclk); | 
| 298 | clk_enable(fclk); | ||
| 299 | } | ||
| 270 | 300 | ||
| 271 | return 0; | 301 | return 0; | 
| 272 | } | 302 | } | 
| @@ -275,7 +305,7 @@ static int ohci_hcd_at91_drv_resume(struct platform_device *dev) | |||
| 275 | #define ohci_hcd_at91_drv_resume NULL | 305 | #define ohci_hcd_at91_drv_resume NULL | 
| 276 | #endif | 306 | #endif | 
| 277 | 307 | ||
| 278 | MODULE_ALIAS("at91rm9200-ohci"); | 308 | MODULE_ALIAS("at91_ohci"); | 
| 279 | 309 | ||
| 280 | static struct platform_driver ohci_hcd_at91_driver = { | 310 | static struct platform_driver ohci_hcd_at91_driver = { | 
| 281 | .probe = ohci_hcd_at91_drv_probe, | 311 | .probe = ohci_hcd_at91_drv_probe, | 
| @@ -283,7 +313,7 @@ static struct platform_driver ohci_hcd_at91_driver = { | |||
| 283 | .suspend = ohci_hcd_at91_drv_suspend, | 313 | .suspend = ohci_hcd_at91_drv_suspend, | 
| 284 | .resume = ohci_hcd_at91_drv_resume, | 314 | .resume = ohci_hcd_at91_drv_resume, | 
| 285 | .driver = { | 315 | .driver = { | 
| 286 | .name = "at91rm9200-ohci", | 316 | .name = "at91_ohci", | 
| 287 | .owner = THIS_MODULE, | 317 | .owner = THIS_MODULE, | 
| 288 | }, | 318 | }, | 
| 289 | }; | 319 | }; | 
| diff --git a/drivers/usb/host/ohci-hcd.c b/drivers/usb/host/ohci-hcd.c index afef5ac35b4a..94d8cf4b36c1 100644 --- a/drivers/usb/host/ohci-hcd.c +++ b/drivers/usb/host/ohci-hcd.c | |||
| @@ -913,7 +913,7 @@ MODULE_LICENSE ("GPL"); | |||
| 913 | #include "ohci-ppc-soc.c" | 913 | #include "ohci-ppc-soc.c" | 
| 914 | #endif | 914 | #endif | 
| 915 | 915 | ||
| 916 | #ifdef CONFIG_ARCH_AT91RM9200 | 916 | #if defined(CONFIG_ARCH_AT91RM9200) || defined(CONFIG_ARCH_AT91SAM9261) | 
| 917 | #include "ohci-at91.c" | 917 | #include "ohci-at91.c" | 
| 918 | #endif | 918 | #endif | 
| 919 | 919 | ||
| @@ -927,6 +927,7 @@ MODULE_LICENSE ("GPL"); | |||
| 927 | || defined (CONFIG_SOC_AU1X00) \ | 927 | || defined (CONFIG_SOC_AU1X00) \ | 
| 928 | || defined (CONFIG_USB_OHCI_HCD_PPC_SOC) \ | 928 | || defined (CONFIG_USB_OHCI_HCD_PPC_SOC) \ | 
| 929 | || defined (CONFIG_ARCH_AT91RM9200) \ | 929 | || defined (CONFIG_ARCH_AT91RM9200) \ | 
| 930 | || defined (CONFIG_ARCH_AT91SAM9261) \ | ||
| 930 | ) | 931 | ) | 
| 931 | #error "missing bus glue for ohci-hcd" | 932 | #error "missing bus glue for ohci-hcd" | 
| 932 | #endif | 933 | #endif | 
