aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb
diff options
context:
space:
mode:
authorDan Williams <dan.j.williams@intel.com>2014-05-20 21:09:20 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2014-05-27 19:51:50 -0400
commit097a155f05e88dc71184ceb93ad1aab1a13d1e41 (patch)
tree04cd559056bf81a6b53cbba88c41d0247f884f8a /drivers/usb
parentaf376a461cf075de6358255579c8d42bb1246e18 (diff)
usb: synchronize port poweroff and khubd
If a port is powered-off, or in the process of being powered-off, prevent khubd from operating on it. Otherwise, the following sequence of events leading to an unintended disconnect may occur: Events: (0) <set pm_qos_no_poweroff to '0' for port1> (1) hub 2-2:1.0: hub_resume (2) hub 2-2:1.0: port 1: status 0301 change 0000 (3) hub 2-2:1.0: state 7 ports 4 chg 0002 evt 0000 (4) hub 2-2:1.0: port 1, power off status 0000, change 0000, 12 Mb/s (5) usb 2-2.1: USB disconnect, device number 5 Description: (1) hub is resumed before sending a ClearPortFeature request (2) hub_activate() notices the port is connected and sets hub->change_bits for the port (3) hub_events() starts, but at the same time the port suspends (4) hub_connect_change() sees the disabled port and triggers disconnect Acked-by: Alan Stern <stern@rowland.harvard.edu> Signed-off-by: Dan Williams <dan.j.williams@intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb')
-rw-r--r--drivers/usb/core/hub.c21
1 files changed, 20 insertions, 1 deletions
diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c
index 782ce2e31c7f..988f227e796f 100644
--- a/drivers/usb/core/hub.c
+++ b/drivers/usb/core/hub.c
@@ -4784,6 +4784,10 @@ static void port_event(struct usb_hub *hub, int port1)
4784 USB_PORT_FEAT_C_PORT_CONFIG_ERROR); 4784 USB_PORT_FEAT_C_PORT_CONFIG_ERROR);
4785 } 4785 }
4786 4786
4787 /* skip port actions that require the port to be powered on */
4788 if (!pm_runtime_active(&port_dev->dev))
4789 return;
4790
4787 if (hub_handle_remote_wakeup(hub, port1, portstatus, portchange)) 4791 if (hub_handle_remote_wakeup(hub, port1, portstatus, portchange))
4788 connect_change = 1; 4792 connect_change = 1;
4789 4793
@@ -4910,11 +4914,26 @@ static void hub_events(void)
4910 4914
4911 /* deal with port status changes */ 4915 /* deal with port status changes */
4912 for (i = 1; i <= hdev->maxchild; i++) { 4916 for (i = 1; i <= hdev->maxchild; i++) {
4917 struct usb_port *port_dev = hub->ports[i - 1];
4918
4913 if (!test_bit(i, hub->busy_bits) 4919 if (!test_bit(i, hub->busy_bits)
4914 && (test_bit(i, hub->event_bits) 4920 && (test_bit(i, hub->event_bits)
4915 || test_bit(i, hub->change_bits) 4921 || test_bit(i, hub->change_bits)
4916 || test_bit(i, hub->wakeup_bits))) 4922 || test_bit(i, hub->wakeup_bits))) {
4923 /*
4924 * The get_noresume and barrier ensure that if
4925 * the port was in the process of resuming, we
4926 * flush that work and keep the port active for
4927 * the duration of the port_event(). However,
4928 * if the port is runtime pm suspended
4929 * (powered-off), we leave it in that state, run
4930 * an abbreviated port_event(), and move on.
4931 */
4932 pm_runtime_get_noresume(&port_dev->dev);
4933 pm_runtime_barrier(&port_dev->dev);
4917 port_event(hub, i); 4934 port_event(hub, i);
4935 pm_runtime_put_sync(&port_dev->dev);
4936 }
4918 } 4937 }
4919 4938
4920 /* deal with hub status changes */ 4939 /* deal with hub status changes */