aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorAlan Stern <stern@rowland.harvard.edu>2012-07-11 11:23:16 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2012-07-16 19:56:48 -0400
commit43fe3a99d9caf10b25f9c596e9854cdae30db418 (patch)
tree596812dcd61ba9cbffe0e837d53dd3a2232ad214 /drivers
parentc4f3476436f7452b97c8accb5dd7d53219a11a3f (diff)
USB: EHCI: resolve some unlikely races
This patch (as1589) resolves some unlikely races involving system shutdown or controller death in ehci-hcd: Shutdown races with both root-hub resume and controller resume. Controller death races with root-hub suspend. A new bitflag is added to indicate that the controller has been shut down (whether for system shutdown or because it died). Tests are added in the suspend and resume pathways to avoid reactivating the controller after any sort of shutdown. Signed-off-by: Alan Stern <stern@rowland.harvard.edu> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/usb/host/ehci-hcd.c20
-rw-r--r--drivers/usb/host/ehci-hub.c27
-rw-r--r--drivers/usb/host/ehci.h1
3 files changed, 43 insertions, 5 deletions
diff --git a/drivers/usb/host/ehci-hcd.c b/drivers/usb/host/ehci-hcd.c
index ac4c8ddde20a..e44ca5453aa2 100644
--- a/drivers/usb/host/ehci-hcd.c
+++ b/drivers/usb/host/ehci-hcd.c
@@ -343,6 +343,7 @@ static void ehci_shutdown(struct usb_hcd *hcd)
343 struct ehci_hcd *ehci = hcd_to_ehci(hcd); 343 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
344 344
345 spin_lock_irq(&ehci->lock); 345 spin_lock_irq(&ehci->lock);
346 ehci->shutdown = true;
346 ehci->rh_state = EHCI_RH_STOPPING; 347 ehci->rh_state = EHCI_RH_STOPPING;
347 ehci->enabled_hrtimer_events = 0; 348 ehci->enabled_hrtimer_events = 0;
348 spin_unlock_irq(&ehci->lock); 349 spin_unlock_irq(&ehci->lock);
@@ -823,6 +824,7 @@ dead:
823 usb_hc_died(hcd); 824 usb_hc_died(hcd);
824 825
825 /* Don't let the controller do anything more */ 826 /* Don't let the controller do anything more */
827 ehci->shutdown = true;
826 ehci->rh_state = EHCI_RH_STOPPING; 828 ehci->rh_state = EHCI_RH_STOPPING;
827 ehci->command &= ~(CMD_RUN | CMD_ASE | CMD_PSE); 829 ehci->command &= ~(CMD_RUN | CMD_ASE | CMD_PSE);
828 ehci_writel(ehci, ehci->command, &ehci->regs->command); 830 ehci_writel(ehci, ehci->command, &ehci->regs->command);
@@ -1129,6 +1131,9 @@ static int __maybe_unused ehci_resume(struct usb_hcd *hcd, bool hibernated)
1129 /* Mark hardware accessible again as we are back to full power by now */ 1131 /* Mark hardware accessible again as we are back to full power by now */
1130 set_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags); 1132 set_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags);
1131 1133
1134 if (ehci->shutdown)
1135 return 0; /* Controller is dead */
1136
1132 /* 1137 /*
1133 * If CF is still set and we aren't resuming from hibernation 1138 * If CF is still set and we aren't resuming from hibernation
1134 * then we maintained suspend power. 1139 * then we maintained suspend power.
@@ -1139,10 +1144,17 @@ static int __maybe_unused ehci_resume(struct usb_hcd *hcd, bool hibernated)
1139 int mask = INTR_MASK; 1144 int mask = INTR_MASK;
1140 1145
1141 ehci_prepare_ports_for_controller_resume(ehci); 1146 ehci_prepare_ports_for_controller_resume(ehci);
1147
1148 spin_lock_irq(&ehci->lock);
1149 if (ehci->shutdown)
1150 goto skip;
1151
1142 if (!hcd->self.root_hub->do_remote_wakeup) 1152 if (!hcd->self.root_hub->do_remote_wakeup)
1143 mask &= ~STS_PCD; 1153 mask &= ~STS_PCD;
1144 ehci_writel(ehci, mask, &ehci->regs->intr_enable); 1154 ehci_writel(ehci, mask, &ehci->regs->intr_enable);
1145 ehci_readl(ehci, &ehci->regs->intr_enable); 1155 ehci_readl(ehci, &ehci->regs->intr_enable);
1156 skip:
1157 spin_unlock_irq(&ehci->lock);
1146 return 0; 1158 return 0;
1147 } 1159 }
1148 1160
@@ -1154,14 +1166,20 @@ static int __maybe_unused ehci_resume(struct usb_hcd *hcd, bool hibernated)
1154 (void) ehci_halt(ehci); 1166 (void) ehci_halt(ehci);
1155 (void) ehci_reset(ehci); 1167 (void) ehci_reset(ehci);
1156 1168
1169 spin_lock_irq(&ehci->lock);
1170 if (ehci->shutdown)
1171 goto skip;
1172
1157 ehci_writel(ehci, ehci->command, &ehci->regs->command); 1173 ehci_writel(ehci, ehci->command, &ehci->regs->command);
1158 ehci_writel(ehci, FLAG_CF, &ehci->regs->configured_flag); 1174 ehci_writel(ehci, FLAG_CF, &ehci->regs->configured_flag);
1159 ehci_readl(ehci, &ehci->regs->command); /* unblock posted writes */ 1175 ehci_readl(ehci, &ehci->regs->command); /* unblock posted writes */
1160 1176
1177 ehci->rh_state = EHCI_RH_SUSPENDED;
1178 spin_unlock_irq(&ehci->lock);
1179
1161 /* here we "know" root ports should always stay powered */ 1180 /* here we "know" root ports should always stay powered */
1162 ehci_port_power(ehci, 1); 1181 ehci_port_power(ehci, 1);
1163 1182
1164 ehci->rh_state = EHCI_RH_SUSPENDED;
1165 return 1; 1183 return 1;
1166} 1184}
1167 1185
diff --git a/drivers/usb/host/ehci-hub.c b/drivers/usb/host/ehci-hub.c
index ffc5f27df725..c7880223738a 100644
--- a/drivers/usb/host/ehci-hub.c
+++ b/drivers/usb/host/ehci-hub.c
@@ -221,6 +221,8 @@ static int ehci_bus_suspend (struct usb_hcd *hcd)
221 ehci_quiesce(ehci); 221 ehci_quiesce(ehci);
222 222
223 spin_lock_irq (&ehci->lock); 223 spin_lock_irq (&ehci->lock);
224 if (ehci->rh_state < EHCI_RH_RUNNING)
225 goto done;
224 226
225 /* Once the controller is stopped, port resumes that are already 227 /* Once the controller is stopped, port resumes that are already
226 * in progress won't complete. Hence if remote wakeup is enabled 228 * in progress won't complete. Hence if remote wakeup is enabled
@@ -306,6 +308,10 @@ static int ehci_bus_suspend (struct usb_hcd *hcd)
306 ehci_halt (ehci); 308 ehci_halt (ehci);
307 309
308 spin_lock_irq(&ehci->lock); 310 spin_lock_irq(&ehci->lock);
311 if (ehci->enabled_hrtimer_events & BIT(EHCI_HRTIMER_POLL_DEAD))
312 ehci_handle_controller_death(ehci);
313 if (ehci->rh_state != EHCI_RH_RUNNING)
314 goto done;
309 ehci->rh_state = EHCI_RH_SUSPENDED; 315 ehci->rh_state = EHCI_RH_SUSPENDED;
310 316
311 end_unlink_async(ehci); 317 end_unlink_async(ehci);
@@ -320,6 +326,7 @@ static int ehci_bus_suspend (struct usb_hcd *hcd)
320 ehci_writel(ehci, mask, &ehci->regs->intr_enable); 326 ehci_writel(ehci, mask, &ehci->regs->intr_enable);
321 ehci_readl(ehci, &ehci->regs->intr_enable); 327 ehci_readl(ehci, &ehci->regs->intr_enable);
322 328
329 done:
323 ehci->next_statechange = jiffies + msecs_to_jiffies(10); 330 ehci->next_statechange = jiffies + msecs_to_jiffies(10);
324 ehci->enabled_hrtimer_events = 0; 331 ehci->enabled_hrtimer_events = 0;
325 ehci->next_hrtimer_event = EHCI_HRTIMER_NO_EVENT; 332 ehci->next_hrtimer_event = EHCI_HRTIMER_NO_EVENT;
@@ -342,10 +349,8 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
342 if (time_before (jiffies, ehci->next_statechange)) 349 if (time_before (jiffies, ehci->next_statechange))
343 msleep(5); 350 msleep(5);
344 spin_lock_irq (&ehci->lock); 351 spin_lock_irq (&ehci->lock);
345 if (!HCD_HW_ACCESSIBLE(hcd)) { 352 if (!HCD_HW_ACCESSIBLE(hcd) || ehci->shutdown)
346 spin_unlock_irq(&ehci->lock); 353 goto shutdown;
347 return -ESHUTDOWN;
348 }
349 354
350 if (unlikely(ehci->debug)) { 355 if (unlikely(ehci->debug)) {
351 if (!dbgp_reset_prep()) 356 if (!dbgp_reset_prep())
@@ -384,6 +389,8 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
384 spin_unlock_irq(&ehci->lock); 389 spin_unlock_irq(&ehci->lock);
385 msleep(8); 390 msleep(8);
386 spin_lock_irq(&ehci->lock); 391 spin_lock_irq(&ehci->lock);
392 if (ehci->shutdown)
393 goto shutdown;
387 394
388 /* clear phy low-power mode before resume */ 395 /* clear phy low-power mode before resume */
389 if (ehci->bus_suspended && ehci->has_hostpc) { 396 if (ehci->bus_suspended && ehci->has_hostpc) {
@@ -401,6 +408,8 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
401 spin_unlock_irq(&ehci->lock); 408 spin_unlock_irq(&ehci->lock);
402 msleep(5); 409 msleep(5);
403 spin_lock_irq(&ehci->lock); 410 spin_lock_irq(&ehci->lock);
411 if (ehci->shutdown)
412 goto shutdown;
404 } 413 }
405 414
406 /* manually resume the ports we suspended during bus_suspend() */ 415 /* manually resume the ports we suspended during bus_suspend() */
@@ -421,6 +430,8 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
421 spin_unlock_irq(&ehci->lock); 430 spin_unlock_irq(&ehci->lock);
422 msleep(20); 431 msleep(20);
423 spin_lock_irq(&ehci->lock); 432 spin_lock_irq(&ehci->lock);
433 if (ehci->shutdown)
434 goto shutdown;
424 } 435 }
425 436
426 i = HCS_N_PORTS (ehci->hcs_params); 437 i = HCS_N_PORTS (ehci->hcs_params);
@@ -439,10 +450,18 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
439 ehci_handover_companion_ports(ehci); 450 ehci_handover_companion_ports(ehci);
440 451
441 /* Now we can safely re-enable irqs */ 452 /* Now we can safely re-enable irqs */
453 spin_lock_irq(&ehci->lock);
454 if (ehci->shutdown)
455 goto shutdown;
442 ehci_writel(ehci, INTR_MASK, &ehci->regs->intr_enable); 456 ehci_writel(ehci, INTR_MASK, &ehci->regs->intr_enable);
443 (void) ehci_readl(ehci, &ehci->regs->intr_enable); 457 (void) ehci_readl(ehci, &ehci->regs->intr_enable);
458 spin_unlock_irq(&ehci->lock);
444 459
445 return 0; 460 return 0;
461
462 shutdown:
463 spin_unlock_irq(&ehci->lock);
464 return -ESHUTDOWN;
446} 465}
447 466
448#else 467#else
diff --git a/drivers/usb/host/ehci.h b/drivers/usb/host/ehci.h
index 7de58fe52d51..da07d98f7d1d 100644
--- a/drivers/usb/host/ehci.h
+++ b/drivers/usb/host/ehci.h
@@ -118,6 +118,7 @@ struct ehci_hcd { /* one per controller */
118 bool need_rescan:1; 118 bool need_rescan:1;
119 bool intr_unlinking:1; 119 bool intr_unlinking:1;
120 bool async_unlinking:1; 120 bool async_unlinking:1;
121 bool shutdown:1;
121 struct ehci_qh *qh_scan_next; 122 struct ehci_qh *qh_scan_next;
122 123
123 /* async schedule support */ 124 /* async schedule support */