diff options
Diffstat (limited to 'drivers/usb/host/uhci-hcd.c')
-rw-r--r-- | drivers/usb/host/uhci-hcd.c | 69 |
1 files changed, 16 insertions, 53 deletions
diff --git a/drivers/usb/host/uhci-hcd.c b/drivers/usb/host/uhci-hcd.c index 8b6c87ef486b..c17bd7ebc021 100644 --- a/drivers/usb/host/uhci-hcd.c +++ b/drivers/usb/host/uhci-hcd.c | |||
@@ -100,22 +100,15 @@ static void uhci_get_current_frame_number(struct uhci_hcd *uhci); | |||
100 | /* to make sure it doesn't hog all of the bandwidth */ | 100 | /* to make sure it doesn't hog all of the bandwidth */ |
101 | #define DEPTH_INTERVAL 5 | 101 | #define DEPTH_INTERVAL 5 |
102 | 102 | ||
103 | static inline void restart_timer(struct uhci_hcd *uhci) | ||
104 | { | ||
105 | mod_timer(&uhci->stall_timer, jiffies + msecs_to_jiffies(100)); | ||
106 | } | ||
107 | |||
103 | #include "uhci-hub.c" | 108 | #include "uhci-hub.c" |
104 | #include "uhci-debug.c" | 109 | #include "uhci-debug.c" |
105 | #include "uhci-q.c" | 110 | #include "uhci-q.c" |
106 | 111 | ||
107 | static int ports_active(struct uhci_hcd *uhci) | ||
108 | { | ||
109 | unsigned long io_addr = uhci->io_addr; | ||
110 | int connection = 0; | ||
111 | int i; | ||
112 | |||
113 | for (i = 0; i < uhci->rh_numports; i++) | ||
114 | connection |= (inw(io_addr + USBPORTSC1 + i * 2) & USBPORTSC_CCS); | ||
115 | |||
116 | return connection; | ||
117 | } | ||
118 | |||
119 | static int suspend_allowed(struct uhci_hcd *uhci) | 112 | static int suspend_allowed(struct uhci_hcd *uhci) |
120 | { | 113 | { |
121 | unsigned long io_addr = uhci->io_addr; | 114 | unsigned long io_addr = uhci->io_addr; |
@@ -270,14 +263,14 @@ static void hc_state_transitions(struct uhci_hcd *uhci) | |||
270 | case UHCI_RUNNING: | 263 | case UHCI_RUNNING: |
271 | 264 | ||
272 | /* global suspend if nothing connected for 1 second */ | 265 | /* global suspend if nothing connected for 1 second */ |
273 | if (!ports_active(uhci) && suspend_allowed(uhci)) { | 266 | if (!any_ports_active(uhci) && suspend_allowed(uhci)) { |
274 | uhci->state = UHCI_SUSPENDING_GRACE; | 267 | uhci->state = UHCI_SUSPENDING_GRACE; |
275 | uhci->state_end = jiffies + HZ; | 268 | uhci->state_end = jiffies + HZ; |
276 | } | 269 | } |
277 | break; | 270 | break; |
278 | 271 | ||
279 | case UHCI_SUSPENDING_GRACE: | 272 | case UHCI_SUSPENDING_GRACE: |
280 | if (ports_active(uhci)) | 273 | if (any_ports_active(uhci)) |
281 | uhci->state = UHCI_RUNNING; | 274 | uhci->state = UHCI_RUNNING; |
282 | else if (time_after_eq(jiffies, uhci->state_end)) | 275 | else if (time_after_eq(jiffies, uhci->state_end)) |
283 | suspend_hc(uhci); | 276 | suspend_hc(uhci); |
@@ -302,58 +295,24 @@ static void hc_state_transitions(struct uhci_hcd *uhci) | |||
302 | } | 295 | } |
303 | } | 296 | } |
304 | 297 | ||
305 | static int init_stall_timer(struct usb_hcd *hcd); | 298 | static void stall_callback(unsigned long _uhci) |
306 | |||
307 | static void stall_callback(unsigned long ptr) | ||
308 | { | 299 | { |
309 | struct usb_hcd *hcd = (struct usb_hcd *)ptr; | 300 | struct uhci_hcd *uhci = (struct uhci_hcd *) _uhci; |
310 | struct uhci_hcd *uhci = hcd_to_uhci(hcd); | ||
311 | struct urb_priv *up; | ||
312 | unsigned long flags; | 301 | unsigned long flags; |
313 | 302 | ||
314 | spin_lock_irqsave(&uhci->lock, flags); | 303 | spin_lock_irqsave(&uhci->lock, flags); |
315 | uhci_scan_schedule(uhci, NULL); | 304 | uhci_scan_schedule(uhci, NULL); |
316 | 305 | check_fsbr(uhci); | |
317 | list_for_each_entry(up, &uhci->urb_list, urb_list) { | ||
318 | struct urb *u = up->urb; | ||
319 | |||
320 | spin_lock(&u->lock); | ||
321 | |||
322 | /* Check if the FSBR timed out */ | ||
323 | if (up->fsbr && !up->fsbr_timeout && time_after_eq(jiffies, up->fsbrtime + IDLE_TIMEOUT)) | ||
324 | uhci_fsbr_timeout(uhci, u); | ||
325 | |||
326 | spin_unlock(&u->lock); | ||
327 | } | ||
328 | |||
329 | /* Really disable FSBR */ | ||
330 | if (!uhci->fsbr && uhci->fsbrtimeout && time_after_eq(jiffies, uhci->fsbrtimeout)) { | ||
331 | uhci->fsbrtimeout = 0; | ||
332 | uhci->skel_term_qh->link = UHCI_PTR_TERM; | ||
333 | } | ||
334 | 306 | ||
335 | /* Poll for and perform state transitions */ | 307 | /* Poll for and perform state transitions */ |
336 | hc_state_transitions(uhci); | 308 | hc_state_transitions(uhci); |
337 | if (unlikely(uhci->suspended_ports && uhci->state != UHCI_SUSPENDED)) | 309 | if (unlikely(uhci->suspended_ports && uhci->state != UHCI_SUSPENDED)) |
338 | uhci_check_ports(uhci); | 310 | uhci_check_ports(uhci); |
339 | 311 | ||
340 | init_stall_timer(hcd); | 312 | restart_timer(uhci); |
341 | spin_unlock_irqrestore(&uhci->lock, flags); | 313 | spin_unlock_irqrestore(&uhci->lock, flags); |
342 | } | 314 | } |
343 | 315 | ||
344 | static int init_stall_timer(struct usb_hcd *hcd) | ||
345 | { | ||
346 | struct uhci_hcd *uhci = hcd_to_uhci(hcd); | ||
347 | |||
348 | init_timer(&uhci->stall_timer); | ||
349 | uhci->stall_timer.function = stall_callback; | ||
350 | uhci->stall_timer.data = (unsigned long)hcd; | ||
351 | uhci->stall_timer.expires = jiffies + msecs_to_jiffies(100); | ||
352 | add_timer(&uhci->stall_timer); | ||
353 | |||
354 | return 0; | ||
355 | } | ||
356 | |||
357 | static irqreturn_t uhci_irq(struct usb_hcd *hcd, struct pt_regs *regs) | 316 | static irqreturn_t uhci_irq(struct usb_hcd *hcd, struct pt_regs *regs) |
358 | { | 317 | { |
359 | struct uhci_hcd *uhci = hcd_to_uhci(hcd); | 318 | struct uhci_hcd *uhci = hcd_to_uhci(hcd); |
@@ -509,6 +468,10 @@ static int uhci_start(struct usb_hcd *hcd) | |||
509 | 468 | ||
510 | init_waitqueue_head(&uhci->waitqh); | 469 | init_waitqueue_head(&uhci->waitqh); |
511 | 470 | ||
471 | init_timer(&uhci->stall_timer); | ||
472 | uhci->stall_timer.function = stall_callback; | ||
473 | uhci->stall_timer.data = (unsigned long) uhci; | ||
474 | |||
512 | uhci->fl = dma_alloc_coherent(uhci_dev(uhci), sizeof(*uhci->fl), | 475 | uhci->fl = dma_alloc_coherent(uhci_dev(uhci), sizeof(*uhci->fl), |
513 | &dma_handle, 0); | 476 | &dma_handle, 0); |
514 | if (!uhci->fl) { | 477 | if (!uhci->fl) { |
@@ -646,7 +609,7 @@ static int uhci_start(struct usb_hcd *hcd) | |||
646 | if ((retval = start_hc(uhci)) != 0) | 609 | if ((retval = start_hc(uhci)) != 0) |
647 | goto err_alloc_skelqh; | 610 | goto err_alloc_skelqh; |
648 | 611 | ||
649 | init_stall_timer(hcd); | 612 | restart_timer(uhci); |
650 | 613 | ||
651 | udev->speed = USB_SPEED_FULL; | 614 | udev->speed = USB_SPEED_FULL; |
652 | 615 | ||