[PATCH v3 07/10] usb: synchronize port poweroff and khubd

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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

Signed-off-by: Dan Williams <dan.j.williams@xxxxxxxxx>
---
 drivers/usb/core/hub.c |   38 +++++++++++++++++++++++++++++++++-----
 1 files changed, 33 insertions(+), 5 deletions(-)

diff --git a/drivers/usb/core/hub.c b/drivers/usb/core/hub.c
index d86548edcc36..0fa13a52b52a 100644
--- a/drivers/usb/core/hub.c
+++ b/drivers/usb/core/hub.c
@@ -4654,6 +4654,35 @@ static int hub_handle_remote_wakeup(struct usb_hub *hub, unsigned int port,
 	return connect_change;
 }
 
+static void put_port(struct usb_port *port_dev, int *port1)
+{
+	if (!port_dev)
+		return;
+
+	pm_runtime_put_sync(&port_dev->dev);
+	(*port1)++;
+}
+
+static struct usb_port *next_active_port(struct usb_hub *hub, int *port1)
+{
+	struct usb_device *hdev = hub->hdev;
+
+	while (*port1 <= hdev->maxchild) {
+		struct usb_port *port_dev = hub->ports[*port1 - 1];
+
+		pm_runtime_get_noresume(&port_dev->dev);
+		pm_runtime_barrier(&port_dev->dev);
+		if (pm_runtime_active(&port_dev->dev))
+			return port_dev;
+		put_port(port_dev, port1);
+	}
+
+	return NULL;
+}
+
+#define for_each_pm_active_port(i, p, h) \
+	for (i = 1; (p = next_active_port(h, &i)); put_port(p, &i))
+
 static void hub_events(void)
 {
 	struct list_head *tmp;
@@ -4661,6 +4690,7 @@ static void hub_events(void)
 	struct usb_interface *intf;
 	struct usb_hub *hub;
 	struct device *hub_dev;
+	struct usb_port *port_dev;
 	u16 hubstatus;
 	u16 hubchange;
 	u16 portstatus;
@@ -4739,7 +4769,7 @@ static void hub_events(void)
 		}
 
 		/* deal with port status changes */
-		for (i = 1; i <= hdev->maxchild; i++) {
+		for_each_pm_active_port(i, port_dev, hub) {
 			if (test_bit(i, hub->busy_bits))
 				continue;
 			connect_change = test_bit(i, hub->change_bits);
@@ -4775,8 +4805,7 @@ static void hub_events(void)
 				 * Works at least with mouse driver.
 				 */
 				if (!(portstatus & USB_PORT_STAT_ENABLE)
-				    && !connect_change
-				    && hub->ports[i - 1]->child) {
+				    && !connect_change && port_dev->child) {
 					dev_err (hub_dev,
 					    "port %i "
 					    "disabled by hub (EMI?), "
@@ -4838,8 +4867,7 @@ static void hub_events(void)
 			 */
 			if (hub_port_warm_reset_required(hub, portstatus)) {
 				int status;
-				struct usb_device *udev =
-					hub->ports[i - 1]->child;
+				struct usb_device *udev = port_dev->child;
 
 				dev_dbg(hub_dev, "warm reset port %d\n", i);
 				if (!udev ||

--
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




[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux