Skip to content

Commit 879d38e

Browse files
AlanSterngregkh
authored andcommitted
USB: fix race between root-hub suspend and remote wakeup
This patch (as1533) fixes a race between root-hub suspend and remote wakeup. If a wakeup event occurs while a root hub is suspending, it might not cause the suspend to fail. Although the host controller drivers check for pending wakeup events at the start of their bus_suspend routines, they generally do not check for wakeup events while the routines are running. In addition, if a wakeup event occurs any time after khubd is frozen and before the root hub is fully suspended, it might not cause a system sleep transition to fail. For example, the host controller drivers do not fail root-hub suspends when a connect-change event is pending. To fix both these issues, this patch causes hcd_bus_suspend() to query the controller driver's hub_status_data method after a root hub is suspended, if the root hub is enabled for wakeup. Any pending status changes will count as wakeup events, causing the root hub to be resumed and the overall suspend to fail with -EBUSY. A significant point is that not all events are reflected immediately in the status bits. Both EHCI and UHCI controllers notify the CPU when remote wakeup begins on a port, but the port's suspend-change status bit doesn't get set until after the port has completed the transition out of the suspend state, some 25 milliseconds later. Consequently, the patch will interpret any nonzero return value from hub_status_data as indicating a pending event, even if none of the status bits are set in the data buffer. Follow-up patches make the necessary changes to ehci-hcd and uhci-hcd. Signed-off-by: Alan Stern <[email protected]> CC: Sarah Sharp <[email protected]> CC: Chen Peter-B29397 <[email protected]> Signed-off-by: Greg Kroah-Hartman <[email protected]>
1 parent c5d703d commit 879d38e

File tree

1 file changed

+12
-0
lines changed

1 file changed

+12
-0
lines changed

drivers/usb/core/hcd.c

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1978,6 +1978,18 @@ int hcd_bus_suspend(struct usb_device *rhdev, pm_message_t msg)
19781978
if (status == 0) {
19791979
usb_set_device_state(rhdev, USB_STATE_SUSPENDED);
19801980
hcd->state = HC_STATE_SUSPENDED;
1981+
1982+
/* Did we race with a root-hub wakeup event? */
1983+
if (rhdev->do_remote_wakeup) {
1984+
char buffer[6];
1985+
1986+
status = hcd->driver->hub_status_data(hcd, buffer);
1987+
if (status != 0) {
1988+
dev_dbg(&rhdev->dev, "suspend raced with wakeup event\n");
1989+
hcd_bus_resume(rhdev, PMSG_AUTO_RESUME);
1990+
status = -EBUSY;
1991+
}
1992+
}
19811993
} else {
19821994
spin_lock_irq(&hcd_root_hub_lock);
19831995
if (!HCD_DEAD(hcd)) {

0 commit comments

Comments
 (0)