diff options
-rw-r--r-- | drivers/usb/core/hub.c | 34 |
1 files changed, 9 insertions, 25 deletions
diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c index 3e9c4d477d1a..4c259aca7a62 100644 --- a/drivers/usb/core/hub.c +++ b/drivers/usb/core/hub.c | |||
@@ -4886,7 +4886,7 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1, | |||
4886 | static void port_event(struct usb_hub *hub, int port1) | 4886 | static void port_event(struct usb_hub *hub, int port1) |
4887 | __must_hold(&port_dev->status_lock) | 4887 | __must_hold(&port_dev->status_lock) |
4888 | { | 4888 | { |
4889 | int connect_change, reset_device = 0; | 4889 | int connect_change; |
4890 | struct usb_port *port_dev = hub->ports[port1 - 1]; | 4890 | struct usb_port *port_dev = hub->ports[port1 - 1]; |
4891 | struct usb_device *udev = port_dev->child; | 4891 | struct usb_device *udev = port_dev->child; |
4892 | struct usb_device *hdev = hub->hdev; | 4892 | struct usb_device *hdev = hub->hdev; |
@@ -4974,30 +4974,14 @@ static void port_event(struct usb_hub *hub, int port1) | |||
4974 | if (hub_port_reset(hub, port1, NULL, | 4974 | if (hub_port_reset(hub, port1, NULL, |
4975 | HUB_BH_RESET_TIME, true) < 0) | 4975 | HUB_BH_RESET_TIME, true) < 0) |
4976 | hub_port_disable(hub, port1, 1); | 4976 | hub_port_disable(hub, port1, 1); |
4977 | } else | 4977 | } else { |
4978 | reset_device = 1; | 4978 | usb_unlock_port(port_dev); |
4979 | } | 4979 | usb_lock_device(udev); |
4980 | 4980 | usb_reset_device(udev); | |
4981 | /* | 4981 | usb_unlock_device(udev); |
4982 | * On disconnect USB3 protocol ports transit from U0 to | 4982 | usb_lock_port(port_dev); |
4983 | * SS.Inactive to Rx.Detect. If this happens a warm- | 4983 | connect_change = 0; |
4984 | * reset is not needed, but a (re)connect may happen | 4984 | } |
4985 | * before hub_wq runs and sees the disconnect, and the | ||
4986 | * device may be an unknown state. | ||
4987 | * | ||
4988 | * If the port went through SS.Inactive without hub_wq | ||
4989 | * seeing it the C_LINK_STATE change flag will be set, | ||
4990 | * and we reset the dev to put it in a known state. | ||
4991 | */ | ||
4992 | if (reset_device || (udev && hub_is_superspeed(hub->hdev) | ||
4993 | && (portchange & USB_PORT_STAT_C_LINK_STATE) | ||
4994 | && (portstatus & USB_PORT_STAT_CONNECTION))) { | ||
4995 | usb_unlock_port(port_dev); | ||
4996 | usb_lock_device(udev); | ||
4997 | usb_reset_device(udev); | ||
4998 | usb_unlock_device(udev); | ||
4999 | usb_lock_port(port_dev); | ||
5000 | connect_change = 0; | ||
5001 | } | 4985 | } |
5002 | 4986 | ||
5003 | if (connect_change) | 4987 | if (connect_change) |