On Thu, 21 Feb 2008, Herton Ronaldo Krzesinski wrote:

> I inserted some dev_printk/printk's, I paste here for reference. I went 
> further
> trying to debug, so that's why there is more things, I'll explain:

Your debugging helped.  Here are the key lines:

> [   58.071856] hub 5-0:1.0: trying to enable port power on non-switchable hub
> [   58.172665] ehci_hcd 0000:00:1d.7: GetStatus port 1 status 001803 POWER 
> sig=j CSC CONNECT
> [   58.183620] usb 1-1: finish reset-resume
> [   58.183627] usb 1-1: finish_port_resume: udev->reset_resume enabled
> [   58.234776] ehci_hcd 0000:00:1d.7: port 1 high speed
> [   58.234785] ehci_hcd 0000:00:1d.7: GetStatus port 1 status 001007 POWER 
> sig=se0 PE CSC CONNECT
> [   58.234803] usb 1-1: usb_reset_device ret == -107
> [   58.234807] hub 1-0:1.0: logical disconnect on port 1
> [   58.234815] usb 1-1: gone after usb resume? status -19
> [   58.234819] usb 1-1: can't resume, status -19

> So the child branch is taken and USB_PERSIST && child->persist_enabled
> evaluates as true. Also udev->reset_resume is enabled. The problem here
> is inside usb_reset_device. The devices are redetected because a
> reenumeration is forced, see that usb_reset_device return -ENOTCONN because:
> 
> - usb_reset_device call chain is hub_port_init -> hub_port_reset ->
> hub_port_wait_reset. In hub_port_wait_reset, the call to hub_port_status
> doesn't fail, but portstatus or portchange are respectively
> USB_PORT_STAT_CONNECTION or USB_PORT_STAT_C_CONNECTION
> (Look that no one of the printks in hub_port_wait_reset are printed),
> and -ENOTCONN then from one of the two returns go back until
> usb_reset_device that then takes the re_enumeration label that causes the
> logical disconnect, so the redetection happens.
> 
> I don't know why ehci is returning this portchange/portstatus, but this is
> triggering the redetection.

Yes, the problem is the USB_PORT_STAT_C_CONNECTION.  (Although I'm not 
sure why udev->reset_resume was enabled...)

Anyway, try this patch; it ought to help.

Alan Stern



Index: usb-2.6/drivers/usb/core/hub.c
===================================================================
--- usb-2.6.orig/drivers/usb/core/hub.c
+++ usb-2.6/drivers/usb/core/hub.c
@@ -1903,6 +1903,24 @@ int usb_port_resume(struct usb_device *u
                status = hub_port_status(hub, port1, &portstatus, &portchange);
 
  SuspendCleared:
+               /* Even if the root hub didn't lose power, an unplug-replug
+                * cycle should cause USB_PERSIST to kick in.
+                */
+               dev_dbg(&udev->dev, "persist_enabled %d status %d "
+                               "portstatus %x change %x\n",
+                               udev->persist_enabled, status,
+                               portstatus, portchange);
+               if (USB_PERSIST && udev->persist_enabled && status >= 0 &&
+                               !(portstatus & USB_PORT_STAT_ENABLE)) {
+                       if (portchange & USB_PORT_STAT_C_ENABLE)
+                               clear_port_feature(hub->hdev, port1,
+                                               USB_PORT_FEAT_C_ENABLE);
+                       if (portchange & USB_PORT_STAT_C_CONNECTION)
+                               clear_port_feature(hub->hdev, port1,
+                                               USB_PORT_FEAT_C_CONNECTION);
+                       udev->reset_resume = 1;
+               }
+
                if (USB_PERSIST && udev->reset_resume)
                        want_flags = USB_PORT_STAT_POWER
                                        | USB_PORT_STAT_CONNECTION;

-
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to