Commit a928a96a authored by Alan Stern's avatar Alan Stern Committed by Ben Hutchings

USB: UHCI: fix IRQ race during initialization

commit 0f815a0a upstream.

This patch (as1644) fixes a race that occurs during startup in
uhci-hcd.  If the IRQ line is shared with other devices, it's possible
for the handler routine to be called before the data structures are
fully initialized.

The problem is fixed by adding a check to the IRQ handler routine.  If
the initialization hasn't finished yet, the routine will return
immediately.
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Reported-by: default avatarDon Zickus <dzickus@redhat.com>
Tested-by: default avatar"Huang, Adrian (ISS Linux TW)" <adrian.huang@hp.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 0988130f
...@@ -448,6 +448,10 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd) ...@@ -448,6 +448,10 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd)
return IRQ_NONE; return IRQ_NONE;
uhci_writew(uhci, status, USBSTS); /* Clear it */ uhci_writew(uhci, status, USBSTS); /* Clear it */
spin_lock(&uhci->lock);
if (unlikely(!uhci->is_initialized)) /* not yet configured */
goto done;
if (status & ~(USBSTS_USBINT | USBSTS_ERROR | USBSTS_RD)) { if (status & ~(USBSTS_USBINT | USBSTS_ERROR | USBSTS_RD)) {
if (status & USBSTS_HSE) if (status & USBSTS_HSE)
dev_err(uhci_dev(uhci), "host system error, " dev_err(uhci_dev(uhci), "host system error, "
...@@ -456,7 +460,6 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd) ...@@ -456,7 +460,6 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd)
dev_err(uhci_dev(uhci), "host controller process " dev_err(uhci_dev(uhci), "host controller process "
"error, something bad happened!\n"); "error, something bad happened!\n");
if (status & USBSTS_HCH) { if (status & USBSTS_HCH) {
spin_lock(&uhci->lock);
if (uhci->rh_state >= UHCI_RH_RUNNING) { if (uhci->rh_state >= UHCI_RH_RUNNING) {
dev_err(uhci_dev(uhci), dev_err(uhci_dev(uhci),
"host controller halted, " "host controller halted, "
...@@ -474,15 +477,15 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd) ...@@ -474,15 +477,15 @@ static irqreturn_t uhci_irq(struct usb_hcd *hcd)
* pending unlinks */ * pending unlinks */
mod_timer(&hcd->rh_timer, jiffies); mod_timer(&hcd->rh_timer, jiffies);
} }
spin_unlock(&uhci->lock);
} }
} }
if (status & USBSTS_RD) if (status & USBSTS_RD) {
spin_unlock(&uhci->lock);
usb_hcd_poll_rh_status(hcd); usb_hcd_poll_rh_status(hcd);
else { } else {
spin_lock(&uhci->lock);
uhci_scan_schedule(uhci); uhci_scan_schedule(uhci);
done:
spin_unlock(&uhci->lock); spin_unlock(&uhci->lock);
} }
...@@ -660,9 +663,9 @@ static int uhci_start(struct usb_hcd *hcd) ...@@ -660,9 +663,9 @@ static int uhci_start(struct usb_hcd *hcd)
*/ */
mb(); mb();
spin_lock_irq(&uhci->lock);
configure_hc(uhci); configure_hc(uhci);
uhci->is_initialized = 1; uhci->is_initialized = 1;
spin_lock_irq(&uhci->lock);
start_rh(uhci); start_rh(uhci);
spin_unlock_irq(&uhci->lock); spin_unlock_irq(&uhci->lock);
return 0; return 0;
......
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