Commit 02cd8571 authored by Alan Stern's avatar Alan Stern Committed by Greg Kroah-Hartman

[PATCH] USB: Remove private khubd semaphore

This patch removes the private semaphore used by the hub driver, and uses
the regular "serialize" semaphore instead.  This will satisfy the general
locking requirements for adding and removing devices attached to the hub.
The only tricky aspect is that now the hub event handler must take a
reference to the hub device while waiting to acquire the semaphore, in
case the hub is disconnected during the wait.

The patch also replaces a few occurrences of spin_lock_irqsave() in
regions where interrupts are known to be enabled with spin_lock_irq().
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Signed-off-by: default avatarGreg Kroah-Hartman <greg@kroah.com>
parent bc1e4d14
...@@ -634,7 +634,6 @@ static void hub_disconnect(struct usb_interface *intf) ...@@ -634,7 +634,6 @@ static void hub_disconnect(struct usb_interface *intf)
{ {
struct usb_hub *hub = usb_get_intfdata (intf); struct usb_hub *hub = usb_get_intfdata (intf);
DECLARE_COMPLETION(urb_complete); DECLARE_COMPLETION(urb_complete);
unsigned long flags;
if (!hub) if (!hub)
return; return;
...@@ -643,16 +642,13 @@ static void hub_disconnect(struct usb_interface *intf) ...@@ -643,16 +642,13 @@ static void hub_disconnect(struct usb_interface *intf)
highspeed_hubs--; highspeed_hubs--;
usb_set_intfdata (intf, NULL); usb_set_intfdata (intf, NULL);
spin_lock_irqsave(&hub_event_lock, flags); spin_lock_irq(&hub_event_lock);
hub->urb_complete = &urb_complete; hub->urb_complete = &urb_complete;
/* Delete it and then reset it */ /* Delete it and then reset it */
list_del_init(&hub->event_list); list_del_init(&hub->event_list);
spin_unlock_irqrestore(&hub_event_lock, flags); spin_unlock_irq(&hub_event_lock);
down(&hub->khubd_sem); /* Wait for khubd to leave this hub alone. */
up(&hub->khubd_sem);
/* assuming we used keventd, it must quiesce too */ /* assuming we used keventd, it must quiesce too */
if (hub->has_indicators) if (hub->has_indicators)
...@@ -738,7 +734,6 @@ static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id) ...@@ -738,7 +734,6 @@ static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id)
INIT_LIST_HEAD(&hub->event_list); INIT_LIST_HEAD(&hub->event_list);
hub->intf = intf; hub->intf = intf;
init_MUTEX(&hub->khubd_sem);
INIT_WORK(&hub->leds, led_work, hub); INIT_WORK(&hub->leds, led_work, hub);
usb_set_intfdata (intf, hub); usb_set_intfdata (intf, hub);
...@@ -1721,7 +1716,6 @@ static void hub_port_connect_change(struct usb_hub *hub, int port, ...@@ -1721,7 +1716,6 @@ static void hub_port_connect_change(struct usb_hub *hub, int port,
static void hub_events(void) static void hub_events(void)
{ {
unsigned long flags;
struct list_head *tmp; struct list_head *tmp;
struct usb_device *hdev; struct usb_device *hdev;
struct usb_hub *hub; struct usb_hub *hub;
...@@ -1740,24 +1734,32 @@ static void hub_events(void) ...@@ -1740,24 +1734,32 @@ static void hub_events(void)
* Not the most efficient, but avoids deadlocks. * Not the most efficient, but avoids deadlocks.
*/ */
while (1) { while (1) {
spin_lock_irqsave(&hub_event_lock, flags);
if (list_empty(&hub_event_list)) /* Grab the first entry at the beginning of the list */
spin_lock_irq(&hub_event_lock);
if (list_empty(&hub_event_list)) {
spin_unlock_irq(&hub_event_lock);
break; break;
}
/* Grab the next entry from the beginning of the list */
tmp = hub_event_list.next; tmp = hub_event_list.next;
list_del_init(tmp);
hub = list_entry(tmp, struct usb_hub, event_list); hub = list_entry(tmp, struct usb_hub, event_list);
hdev = interface_to_usbdev(hub->intf); hdev = interface_to_usbdev(hub->intf);
hub_dev = &hub->intf->dev; hub_dev = &hub->intf->dev;
list_del_init(tmp); usb_get_dev(hdev);
spin_unlock_irq(&hub_event_lock);
if (unlikely(down_trylock(&hub->khubd_sem)))
BUG(); /* never blocks, we were on list */
spin_unlock_irqrestore(&hub_event_lock, flags); /* Lock the device, then check to see if we were
* disconnected while waiting for the lock to succeed. */
down(&hdev->serialize);
if (hdev->state != USB_STATE_CONFIGURED ||
!hdev->actconfig ||
hub != usb_get_intfdata(
hdev->actconfig->interface[0]))
goto loop;
if (hub->error) { if (hub->error) {
dev_dbg (hub_dev, "resetting for error %d\n", dev_dbg (hub_dev, "resetting for error %d\n",
...@@ -1766,9 +1768,8 @@ static void hub_events(void) ...@@ -1766,9 +1768,8 @@ static void hub_events(void)
if (hub_reset(hub)) { if (hub_reset(hub)) {
dev_dbg (hub_dev, dev_dbg (hub_dev,
"can't reset; disconnecting\n"); "can't reset; disconnecting\n");
up(&hub->khubd_sem);
hub_start_disconnect(hdev); hub_start_disconnect(hdev);
continue; goto loop;
} }
hub->nerrors = 0; hub->nerrors = 0;
...@@ -1859,10 +1860,12 @@ static void hub_events(void) ...@@ -1859,10 +1860,12 @@ static void hub_events(void)
hub_power_on(hub); hub_power_on(hub);
} }
} }
up(&hub->khubd_sem);
} /* end while (1) */
spin_unlock_irqrestore(&hub_event_lock, flags); loop:
up(&hdev->serialize);
usb_put_dev(hdev);
} /* end while (1) */
} }
static int hub_thread(void *__unused) static int hub_thread(void *__unused)
......
...@@ -205,7 +205,6 @@ struct usb_hub { ...@@ -205,7 +205,6 @@ struct usb_hub {
struct list_head event_list; /* hubs w/data or errs ready */ struct list_head event_list; /* hubs w/data or errs ready */
struct usb_hub_descriptor *descriptor; /* class descriptor */ struct usb_hub_descriptor *descriptor; /* class descriptor */
struct semaphore khubd_sem;
struct usb_tt tt; /* Transaction Translator */ struct usb_tt tt; /* Transaction Translator */
u8 power_budget; /* in 2mA units; or zero */ u8 power_budget; /* in 2mA units; or zero */
......
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