On Wed, 25 Jan 2012, Sarah Sharp wrote: > USB 3.0 hubs don't have a port suspend change bit (that bit is now > reserved). Instead, when a host-initiated resume finishes, the hub sets > the port link state change bit. > > When a USB 3.0 device initiates remote wakeup, the parent hubs with > their upstream links in U3 will pass the LFPS up the chain. The first > hub that has an upstream link in U0 (which may be the roothub) will > reflect that LFPS back down the path to the device. > > However, the parent hubs in the resumed path will not set their link > state change bit. Instead, the device that initiated the resume has to > send an asynchronous "Function Wake" Device Notification up to the host > controller. > > Unfortunately, this notification can take up to 2.5 seconds > (tNotification in section 8.13 of the USB3 spec). That means that we > need to prevent the parent hub from being suspended before the > notification has been sent. Do we really need to worry about this? The worst that can happen is the parent hub suspends, then the device requests another wakeup and sends another notification. Also, what you're saying makes no sense. We have no way to know that the parent hub should be prevented from suspending, because we don't realize that anything special has happened until we receive the notification. > First, make the xHCI roothub act like an external USB 3.0 hub and not > pass up the port link state change bit when a device-initiated resume > finishes. Introduce a new xHCI bit field, port_remote_wakeup, so that > we can tell the difference between a port coming out of the U3Exit state > (host-initiated resume) and the RExit state (ending state of > device-initiated resume). > > Since the USB core can't tell whether a port on a hub has resumed by > looking at the Hub Status buffer, we need to introduce a bitfield, > wakeup_bits, that indicates which ports have resumed. When the xHCI > driver notices a port finishing a device-initiated resume, we call into > a new USB core function, usb_super_speed_remote_wakeup(), that will set > the right bit in wakeup_bits, and kick khubd for that hub. Why do we need to do this? Won't the device itself send a wakeup notification? After all, we don't do anything when a non-root hub finishes a device-initiated resume. In fact, we don't even know it has happened until the notification arrives -- the hub itself doesn't tell us about it. Root hubs should behave the same way. > We also call usb_super_speed_remote_wakeup() when the Function Wake > Device Notification is received by the xHCI driver. This covers the > case where the link between the roothub and the first-tier hub is in U0, Or second-tier, or higher. > and the hub reflects the resume signaling back to the device without > giving any indication it has done so until the device sends the Function > Wake notification. > > Change the code in khubd that handles the remote wakeup to look at the > state the USB core thinks the device is in, and handle the remote wakeup > if the port's wakeup bit is set. > > This patch only takes care of the case where the device is attached > directly to the roothub, or the USB 3.0 hub that is attached to the root > hub is the device sending the Function Wake Device Notification (e.g. > because a new USB device was attached). The other cases will be covered > in a second patch. I'm not clear on this. What difference does it make if the device sending the notification is not directly connected to the root hub? > @@ -411,6 +413,34 @@ void usb_kick_khubd(struct usb_device *hdev) > kick_khubd(hub); > } > > +/* > + * USB 3.0 devices are supposed to send a Device Notification Function Wake > + * message after they are finished signaling resume. The USB 3.0 bus > + * specification does not specify the maximum amount of time between the end of > + * resume signaling and the first wake notification. It does specify > + * TNotification, the rate at which the device should send a Function Wake > + * notification if the device has not been accessed. That rate is 2.5 seconds, > + * which is far too long when the hub will be re-suspended by default after > + * 2 seconds. > + * > + * To prevent suspend of hubs while we're waiting for the device notification > + * to be sent, mark the resumed port as woken up and kick khubd for the hub. > + */ This doesn't sound right at all. The purpose of this routine is to let khubd know that this port on this hub has changed to U0 because of a remote wakeup. It has nothing to do with notification rates or resuspends. > +void usb_super_speed_remote_wakeup(struct usb_device *hdev, > + unsigned int portnum) > +{ > + struct usb_hub *hub; > + > + if (!hdev) > + return; > + > + hub = hdev_to_hub(hdev); > + if (hub) { > + set_bit(portnum, hub->wakeup_bits); > + kick_khubd(hub); > + } > +} > +EXPORT_SYMBOL_GPL(usb_super_speed_remote_wakeup); > > /* completion function, fires on port status changes and various faults */ > static void hub_irq(struct urb *urb) > @@ -807,12 +837,6 @@ static void hub_activate(struct usb_hub *hub, enum hub_activation_type type) > clear_port_feature(hub->hdev, port1, > USB_PORT_FEAT_C_ENABLE); > } > - if (portchange & USB_PORT_STAT_C_LINK_STATE) { > - need_debounce_delay = true; > - clear_port_feature(hub->hdev, port1, > - USB_PORT_FEAT_C_PORT_LINK_STATE); > - } Are you sure about this? Won't the hub's interrupt URB fire if the C_PORT_LINK_STATE status-change bit is set? You probably want to leave the clear_port_feature() call and get rid of the need_debounce_delay line. > - > if ((portchange & USB_PORT_STAT_C_BH_RESET) && > hub_is_superspeed(hub->hdev)) { > need_debounce_delay = true; > @@ -3458,11 +3482,18 @@ static int hub_handle_remote_wakeup(struct usb_hub *hub, unsigned int port, > int ret; > > hdev = hub->hdev; > - if (!(portchange & USB_PORT_STAT_C_SUSPEND)) > - return 0; > - clear_port_feature(hdev, port, USB_PORT_FEAT_C_SUSPEND); > - > udev = hdev->children[port-1]; > + if (!hub_is_superspeed(hdev)) { > + if (!(portchange & USB_PORT_STAT_C_SUSPEND)) > + return 0; > + clear_port_feature(hdev, port, USB_PORT_FEAT_C_SUSPEND); > + } else { > + if (!udev || udev->state != USB_STATE_SUSPENDED || > + !test_and_clear_bit(udev->portnum, > + hub->wakeup_bits)) I'm concerned that, being buried at the end of a short-circuit boolean expression like this, the wakeup_bits entry might not get cleared when it should. You can move it to the start of the test expression and replace udev->portnum with port. > + return 0; > + } > + > if (udev) { > /* TRSMRCY = 10 msec */ > msleep(10); > @@ -3648,8 +3680,8 @@ static void hub_events(void) > clear_port_feature(hdev, i, > USB_PORT_FEAT_C_BH_PORT_RESET); > } > - if (portchange & USB_PORT_STAT_C_LINK_STATE) { > - clear_port_feature(hub->hdev, i, > + if (portchange & USB_PORT_FEAT_C_PORT_LINK_STATE) { This should remain the way it was. The values in portchange are status flags, not feature bits. > + clear_port_feature(hdev, i, > USB_PORT_FEAT_C_PORT_LINK_STATE); > } > if (portchange & USB_PORT_STAT_C_CONFIG_ERROR) { > --- a/include/linux/usb/hcd.h > +++ b/include/linux/usb/hcd.h > @@ -412,6 +412,8 @@ extern irqreturn_t usb_hcd_irq(int irq, void *__hcd); > > extern void usb_hc_died(struct usb_hcd *hcd); > extern void usb_hcd_poll_rh_status(struct usb_hcd *hcd); > +extern void usb_super_speed_remote_wakeup(struct usb_device *roothub, This should say hdev, not roothub. > + unsigned int portnum); > > /* The D0/D1 toggle bits ... USE WITH CAUTION (they're almost hcd-internal) $ > #define usb_gettoggle(dev, ep, out) (((dev)->toggle[out] >> (ep)) & 1) Alan Stern -- To unsubscribe from this list: send the line "unsubscribe linux-usb" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html