USB: add flag for HCDs that can't receive wakeup requests (isp1760-hcd)
authorAlan Stern <stern@rowland.harvard.edu>
Thu, 29 Jan 2015 20:05:04 +0000 (15:05 -0500)
committerGreg Kroah-Hartman <greg@kroah.com>
Sat, 31 Jan 2015 17:05:06 +0000 (09:05 -0800)
Currently the USB stack assumes that all host controller drivers are
capable of receiving wakeup requests from downstream devices.
However, this isn't true for the isp1760-hcd driver, which means that
it isn't safe to do a runtime suspend of any device attached to a
root-hub port if the device requires wakeup.

This patch adds a "cant_recv_wakeups" flag to the usb_hcd structure
and sets the flag in isp1760-hcd.  The core is modified to prevent a
direct child of the root hub from being put into runtime suspend with
wakeup enabled if the flag is set.

Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Tested-by: Nicolas Pitre <nico@linaro.org>
CC: <stable@vger.kernel.org>
Signed-off-by: Greg Kroah-Hartman <greg@kroah.com>
drivers/usb/core/driver.c
drivers/usb/host/isp1760-hcd.c
include/linux/usb/hcd.h

index c76ec9758ce301d0b72925aa3cc92b1d7e377225..818369afff63f3a6bc53059fc107937f0d6c0b42 100644 (file)
@@ -1780,6 +1780,18 @@ static int autosuspend_check(struct usb_device *udev)
                dev_dbg(&udev->dev, "remote wakeup needed for autosuspend\n");
                return -EOPNOTSUPP;
        }
+
+       /*
+        * If the device is a direct child of the root hub and the HCD
+        * doesn't handle wakeup requests, don't allow autosuspend when
+        * wakeup is needed.
+        */
+       if (w && udev->parent == udev->bus->root_hub &&
+                       bus_to_hcd(udev->bus)->cant_recv_wakeups) {
+               dev_dbg(&udev->dev, "HCD doesn't handle wakeup requests\n");
+               return -EOPNOTSUPP;
+       }
+
        udev->do_remote_wakeup = w;
        return 0;
 }
index dbba455884f4e0f78ed89e46b17f8d6859a7f43d..cecf39a220e719b101411830b9549ecbe4f768ec 100644 (file)
@@ -2245,6 +2245,9 @@ struct usb_hcd *isp1760_register(phys_addr_t res_start, resource_size_t res_len,
        hcd->rsrc_start = res_start;
        hcd->rsrc_len = res_len;
 
+       /* This driver doesn't support wakeup requests */
+       hcd->cant_recv_wakeups = 1;
+
        ret = usb_add_hcd(hcd, irq, irqflags);
        if (ret)
                goto err_unmap;
index 8968f616e414d6823a9e4c5cbc23b11340c53483..68b1e836dff17b5edad6a651a52e239d48b40651 100644 (file)
@@ -146,6 +146,8 @@ struct usb_hcd {
        unsigned                amd_resume_bug:1; /* AMD remote wakeup quirk */
        unsigned                can_do_streams:1; /* HC supports streams */
        unsigned                tpl_support:1; /* OTG & EH TPL support */
+       unsigned                cant_recv_wakeups:1;
+                       /* wakeup requests from downstream aren't received */
 
        unsigned int            irq;            /* irq allocated */
        void __iomem            *regs;          /* device memory/io */