Commit e3123e4a authored by Al Cooper's avatar Al Cooper Committed by Jiri Slaby

usb: Add connected retry on resume for non SS devices

commit 6b82b122 upstream.

Currently usb_port_resume waits for up to 2 seconds for CONNECT
status for SS devices only. This change will do the same thing for
non-SS devices even though the reason is a little different. This
will fix an issue where VBUS is turned off during system wide
"suspend to ram" and some 2.0 devices take greater than the current
max of 100ms to show connected after VBUS is enabled. This is most
commonly seen on hard drive based devices and USB3.0 devices plugged
into a 2.0 only port.
Signed-off-by: default avatarAl Cooper <alcooperx@gmail.com>
Acked-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarJiri Slaby <jslaby@suse.cz>
parent e1d5059a
...@@ -3167,7 +3167,7 @@ static int finish_port_resume(struct usb_device *udev) ...@@ -3167,7 +3167,7 @@ static int finish_port_resume(struct usb_device *udev)
/* /*
* There are some SS USB devices which take longer time for link training. * There are some SS USB devices which take longer time for link training.
* XHCI specs 4.19.4 says that when Link training is successful, port * XHCI specs 4.19.4 says that when Link training is successful, port
* sets CSC bit to 1. So if SW reads port status before successful link * sets CCS bit to 1. So if SW reads port status before successful link
* training, then it will not find device to be present. * training, then it will not find device to be present.
* USB Analyzer log with such buggy devices show that in some cases * USB Analyzer log with such buggy devices show that in some cases
* device switch on the RX termination after long delay of host enabling * device switch on the RX termination after long delay of host enabling
...@@ -3178,14 +3178,17 @@ static int finish_port_resume(struct usb_device *udev) ...@@ -3178,14 +3178,17 @@ static int finish_port_resume(struct usb_device *udev)
* routine implements a 2000 ms timeout for link training. If in a case * routine implements a 2000 ms timeout for link training. If in a case
* link trains before timeout, loop will exit earlier. * link trains before timeout, loop will exit earlier.
* *
* There are also some 2.0 hard drive based devices and 3.0 thumb
* drives that, when plugged into a 2.0 only port, take a long
* time to set CCS after VBUS enable.
*
* FIXME: If a device was connected before suspend, but was removed * FIXME: If a device was connected before suspend, but was removed
* while system was asleep, then the loop in the following routine will * while system was asleep, then the loop in the following routine will
* only exit at timeout. * only exit at timeout.
* *
* This routine should only be called when persist is enabled for a SS * This routine should only be called when persist is enabled.
* device.
*/ */
static int wait_for_ss_port_enable(struct usb_device *udev, static int wait_for_connected(struct usb_device *udev,
struct usb_hub *hub, int *port1, struct usb_hub *hub, int *port1,
u16 *portchange, u16 *portstatus) u16 *portchange, u16 *portstatus)
{ {
...@@ -3198,6 +3201,7 @@ static int wait_for_ss_port_enable(struct usb_device *udev, ...@@ -3198,6 +3201,7 @@ static int wait_for_ss_port_enable(struct usb_device *udev,
delay_ms += 20; delay_ms += 20;
status = hub_port_status(hub, *port1, portstatus, portchange); status = hub_port_status(hub, *port1, portstatus, portchange);
} }
dev_dbg(&udev->dev, "Waited %dms for CONNECT\n", delay_ms);
return status; return status;
} }
...@@ -3303,8 +3307,8 @@ int usb_port_resume(struct usb_device *udev, pm_message_t msg) ...@@ -3303,8 +3307,8 @@ int usb_port_resume(struct usb_device *udev, pm_message_t msg)
clear_bit(port1, hub->busy_bits); clear_bit(port1, hub->busy_bits);
if (udev->persist_enabled && hub_is_superspeed(hub->hdev)) if (udev->persist_enabled)
status = wait_for_ss_port_enable(udev, hub, &port1, &portchange, status = wait_for_connected(udev, hub, &port1, &portchange,
&portstatus); &portstatus);
status = check_port_resume_type(udev, status = check_port_resume_type(udev,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment