Commit 541c7d43 authored by Alan Stern's avatar Alan Stern Committed by Greg Kroah-Hartman

USB: convert usb_hcd bitfields into atomic flags

This patch (as1393) converts several of the single-bit fields in
struct usb_hcd to atomic flags.  This is for safety's sake; not all
CPUs can update bitfield values atomically, and these flags are used
in multiple contexts.

The flag fields that are set only during registration or removal can
remain as they are, since non-atomic accesses at those times will not
cause any problems.

(Strictly speaking, the authorized_default flag should become atomic
as well.  I didn't bother with it because it gets changed only via
sysfs.  It can be done later, if anyone wants.)
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 89ba85d4
...@@ -215,7 +215,7 @@ static int vhci_hub_status(struct usb_hcd *hcd, char *buf) ...@@ -215,7 +215,7 @@ static int vhci_hub_status(struct usb_hcd *hcd, char *buf)
vhci = hcd_to_vhci(hcd); vhci = hcd_to_vhci(hcd);
spin_lock_irqsave(&vhci->lock, flags); spin_lock_irqsave(&vhci->lock, flags);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
usbip_dbg_vhci_rh("hw accessible flag in on?\n"); usbip_dbg_vhci_rh("hw accessible flag in on?\n");
goto done; goto done;
} }
...@@ -269,7 +269,7 @@ static int vhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, ...@@ -269,7 +269,7 @@ static int vhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
u32 prev_port_status[VHCI_NPORTS]; u32 prev_port_status[VHCI_NPORTS];
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
return -ETIMEDOUT; return -ETIMEDOUT;
/* /*
...@@ -1041,7 +1041,7 @@ static int vhci_bus_resume(struct usb_hcd *hcd) ...@@ -1041,7 +1041,7 @@ static int vhci_bus_resume(struct usb_hcd *hcd)
dev_dbg(&hcd->self.root_hub->dev, "%s\n", __func__); dev_dbg(&hcd->self.root_hub->dev, "%s\n", __func__);
spin_lock_irq(&vhci->lock); spin_lock_irq(&vhci->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
} else { } else {
/* vhci->rh_state = DUMMY_RH_RUNNING; /* vhci->rh_state = DUMMY_RH_RUNNING;
......
...@@ -264,7 +264,7 @@ static void c67x00_hcd_irq(struct c67x00_sie *sie, u16 int_status, u16 msg) ...@@ -264,7 +264,7 @@ static void c67x00_hcd_irq(struct c67x00_sie *sie, u16 int_status, u16 msg)
if (unlikely(hcd->state == HC_STATE_HALT)) if (unlikely(hcd->state == HC_STATE_HALT))
return; return;
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
return; return;
/* Handle Start of frame events */ /* Handle Start of frame events */
...@@ -282,7 +282,7 @@ static int c67x00_hcd_start(struct usb_hcd *hcd) ...@@ -282,7 +282,7 @@ static int c67x00_hcd_start(struct usb_hcd *hcd)
{ {
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
hcd->state = HC_STATE_RUNNING; hcd->state = HC_STATE_RUNNING;
hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &hcd->flags);
return 0; return 0;
} }
......
...@@ -679,7 +679,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd) ...@@ -679,7 +679,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd)
spin_lock_irqsave(&hcd_root_hub_lock, flags); spin_lock_irqsave(&hcd_root_hub_lock, flags);
urb = hcd->status_urb; urb = hcd->status_urb;
if (urb) { if (urb) {
hcd->poll_pending = 0; clear_bit(HCD_FLAG_POLL_PENDING, &hcd->flags);
hcd->status_urb = NULL; hcd->status_urb = NULL;
urb->actual_length = length; urb->actual_length = length;
memcpy(urb->transfer_buffer, buffer, length); memcpy(urb->transfer_buffer, buffer, length);
...@@ -690,7 +690,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd) ...@@ -690,7 +690,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd)
spin_lock(&hcd_root_hub_lock); spin_lock(&hcd_root_hub_lock);
} else { } else {
length = 0; length = 0;
hcd->poll_pending = 1; set_bit(HCD_FLAG_POLL_PENDING, &hcd->flags);
} }
spin_unlock_irqrestore(&hcd_root_hub_lock, flags); spin_unlock_irqrestore(&hcd_root_hub_lock, flags);
} }
...@@ -699,7 +699,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd) ...@@ -699,7 +699,7 @@ void usb_hcd_poll_rh_status(struct usb_hcd *hcd)
* exceed that limit if HZ is 100. The math is more clunky than * exceed that limit if HZ is 100. The math is more clunky than
* maybe expected, this is to make sure that all timers for USB devices * maybe expected, this is to make sure that all timers for USB devices
* fire at the same time to give the CPU a break inbetween */ * fire at the same time to give the CPU a break inbetween */
if (hcd->uses_new_polling ? hcd->poll_rh : if (hcd->uses_new_polling ? HCD_POLL_RH(hcd) :
(length == 0 && hcd->status_urb != NULL)) (length == 0 && hcd->status_urb != NULL))
mod_timer (&hcd->rh_timer, (jiffies/(HZ/4) + 1) * (HZ/4)); mod_timer (&hcd->rh_timer, (jiffies/(HZ/4) + 1) * (HZ/4));
} }
...@@ -736,7 +736,7 @@ static int rh_queue_status (struct usb_hcd *hcd, struct urb *urb) ...@@ -736,7 +736,7 @@ static int rh_queue_status (struct usb_hcd *hcd, struct urb *urb)
mod_timer(&hcd->rh_timer, (jiffies/(HZ/4) + 1) * (HZ/4)); mod_timer(&hcd->rh_timer, (jiffies/(HZ/4) + 1) * (HZ/4));
/* If a status change has already occurred, report it ASAP */ /* If a status change has already occurred, report it ASAP */
else if (hcd->poll_pending) else if (HCD_POLL_PENDING(hcd))
mod_timer(&hcd->rh_timer, jiffies); mod_timer(&hcd->rh_timer, jiffies);
retval = 0; retval = 0;
done: done:
...@@ -1150,8 +1150,7 @@ int usb_hcd_check_unlink_urb(struct usb_hcd *hcd, struct urb *urb, ...@@ -1150,8 +1150,7 @@ int usb_hcd_check_unlink_urb(struct usb_hcd *hcd, struct urb *urb,
* finish unlinking the initial failed usb_set_address() * finish unlinking the initial failed usb_set_address()
* or device descriptor fetch. * or device descriptor fetch.
*/ */
if (!test_bit(HCD_FLAG_SAW_IRQ, &hcd->flags) && if (!HCD_SAW_IRQ(hcd) && !is_root_hub(urb->dev)) {
!is_root_hub(urb->dev)) {
dev_warn(hcd->self.controller, "Unlink after no-IRQ? " dev_warn(hcd->self.controller, "Unlink after no-IRQ? "
"Controller is probably using the wrong IRQ.\n"); "Controller is probably using the wrong IRQ.\n");
set_bit(HCD_FLAG_SAW_IRQ, &hcd->flags); set_bit(HCD_FLAG_SAW_IRQ, &hcd->flags);
...@@ -2063,8 +2062,7 @@ irqreturn_t usb_hcd_irq (int irq, void *__hcd) ...@@ -2063,8 +2062,7 @@ irqreturn_t usb_hcd_irq (int irq, void *__hcd)
*/ */
local_irq_save(flags); local_irq_save(flags);
if (unlikely(hcd->state == HC_STATE_HALT || if (unlikely(hcd->state == HC_STATE_HALT || !HCD_HW_ACCESSIBLE(hcd))) {
!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))) {
rc = IRQ_NONE; rc = IRQ_NONE;
} else if (hcd->driver->irq(hcd) == IRQ_NONE) { } else if (hcd->driver->irq(hcd) == IRQ_NONE) {
rc = IRQ_NONE; rc = IRQ_NONE;
...@@ -2098,7 +2096,7 @@ void usb_hc_died (struct usb_hcd *hcd) ...@@ -2098,7 +2096,7 @@ void usb_hc_died (struct usb_hcd *hcd)
spin_lock_irqsave (&hcd_root_hub_lock, flags); spin_lock_irqsave (&hcd_root_hub_lock, flags);
if (hcd->rh_registered) { if (hcd->rh_registered) {
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
/* make khubd clean up old urbs and devices */ /* make khubd clean up old urbs and devices */
usb_set_device_state (hcd->self.root_hub, usb_set_device_state (hcd->self.root_hub,
...@@ -2301,7 +2299,7 @@ int usb_add_hcd(struct usb_hcd *hcd, ...@@ -2301,7 +2299,7 @@ int usb_add_hcd(struct usb_hcd *hcd,
retval); retval);
goto error_create_attr_group; goto error_create_attr_group;
} }
if (hcd->uses_new_polling && hcd->poll_rh) if (hcd->uses_new_polling && HCD_POLL_RH(hcd))
usb_hcd_poll_rh_status(hcd); usb_hcd_poll_rh_status(hcd);
return retval; return retval;
...@@ -2320,11 +2318,11 @@ int usb_add_hcd(struct usb_hcd *hcd, ...@@ -2320,11 +2318,11 @@ int usb_add_hcd(struct usb_hcd *hcd,
mutex_unlock(&usb_bus_list_lock); mutex_unlock(&usb_bus_list_lock);
err_register_root_hub: err_register_root_hub:
hcd->rh_pollable = 0; hcd->rh_pollable = 0;
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
del_timer_sync(&hcd->rh_timer); del_timer_sync(&hcd->rh_timer);
hcd->driver->stop(hcd); hcd->driver->stop(hcd);
hcd->state = HC_STATE_HALT; hcd->state = HC_STATE_HALT;
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
del_timer_sync(&hcd->rh_timer); del_timer_sync(&hcd->rh_timer);
err_hcd_driver_start: err_hcd_driver_start:
if (hcd->irq >= 0) if (hcd->irq >= 0)
...@@ -2380,14 +2378,14 @@ void usb_remove_hcd(struct usb_hcd *hcd) ...@@ -2380,14 +2378,14 @@ void usb_remove_hcd(struct usb_hcd *hcd)
* the hub_status_data() callback. * the hub_status_data() callback.
*/ */
hcd->rh_pollable = 0; hcd->rh_pollable = 0;
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
del_timer_sync(&hcd->rh_timer); del_timer_sync(&hcd->rh_timer);
hcd->driver->stop(hcd); hcd->driver->stop(hcd);
hcd->state = HC_STATE_HALT; hcd->state = HC_STATE_HALT;
/* In case the HCD restarted the timer, stop it again. */ /* In case the HCD restarted the timer, stop it again. */
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
del_timer_sync(&hcd->rh_timer); del_timer_sync(&hcd->rh_timer);
if (hcd->irq >= 0) if (hcd->irq >= 0)
......
...@@ -1542,7 +1542,7 @@ static int dummy_hub_status (struct usb_hcd *hcd, char *buf) ...@@ -1542,7 +1542,7 @@ static int dummy_hub_status (struct usb_hcd *hcd, char *buf)
dum = hcd_to_dummy (hcd); dum = hcd_to_dummy (hcd);
spin_lock_irqsave (&dum->lock, flags); spin_lock_irqsave (&dum->lock, flags);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
goto done; goto done;
if (dum->resuming && time_after_eq (jiffies, dum->re_timeout)) { if (dum->resuming && time_after_eq (jiffies, dum->re_timeout)) {
...@@ -1588,7 +1588,7 @@ static int dummy_hub_control ( ...@@ -1588,7 +1588,7 @@ static int dummy_hub_control (
int retval = 0; int retval = 0;
unsigned long flags; unsigned long flags;
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
return -ETIMEDOUT; return -ETIMEDOUT;
dum = hcd_to_dummy (hcd); dum = hcd_to_dummy (hcd);
...@@ -1739,7 +1739,7 @@ static int dummy_bus_resume (struct usb_hcd *hcd) ...@@ -1739,7 +1739,7 @@ static int dummy_bus_resume (struct usb_hcd *hcd)
dev_dbg (&hcd->self.root_hub->dev, "%s\n", __func__); dev_dbg (&hcd->self.root_hub->dev, "%s\n", __func__);
spin_lock_irq (&dum->lock); spin_lock_irq (&dum->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
} else { } else {
dum->rh_state = DUMMY_RH_RUNNING; dum->rh_state = DUMMY_RH_RUNNING;
......
...@@ -712,7 +712,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf) ...@@ -712,7 +712,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf)
spin_lock_irqsave (&ehci->lock, flags); spin_lock_irqsave (&ehci->lock, flags);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
size = scnprintf (next, size, size = scnprintf (next, size,
"bus %s, device %s\n" "bus %s, device %s\n"
"%s\n" "%s\n"
......
...@@ -642,7 +642,6 @@ static int ehci_run (struct usb_hcd *hcd) ...@@ -642,7 +642,6 @@ static int ehci_run (struct usb_hcd *hcd)
u32 hcc_params; u32 hcc_params;
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
hcd->poll_rh = 0;
/* EHCI spec section 4.1 */ /* EHCI spec section 4.1 */
if ((retval = ehci_reset(ehci)) != 0) { if ((retval = ehci_reset(ehci)) != 0) {
......
...@@ -316,7 +316,7 @@ static int ehci_bus_resume (struct usb_hcd *hcd) ...@@ -316,7 +316,7 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
if (time_before (jiffies, ehci->next_statechange)) if (time_before (jiffies, ehci->next_statechange))
msleep(5); msleep(5);
spin_lock_irq (&ehci->lock); spin_lock_irq (&ehci->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
spin_unlock_irq(&ehci->lock); spin_unlock_irq(&ehci->lock);
return -ESHUTDOWN; return -ESHUTDOWN;
} }
......
...@@ -1126,8 +1126,7 @@ submit_async ( ...@@ -1126,8 +1126,7 @@ submit_async (
#endif #endif
spin_lock_irqsave (&ehci->lock, flags); spin_lock_irqsave (&ehci->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(ehci_to_hcd(ehci)))) {
&ehci_to_hcd(ehci)->flags))) {
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
goto done; goto done;
} }
......
...@@ -880,8 +880,7 @@ static int intr_submit ( ...@@ -880,8 +880,7 @@ static int intr_submit (
spin_lock_irqsave (&ehci->lock, flags); spin_lock_irqsave (&ehci->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(ehci_to_hcd(ehci)))) {
&ehci_to_hcd(ehci)->flags))) {
status = -ESHUTDOWN; status = -ESHUTDOWN;
goto done_not_linked; goto done_not_linked;
} }
...@@ -1815,8 +1814,7 @@ static int itd_submit (struct ehci_hcd *ehci, struct urb *urb, ...@@ -1815,8 +1814,7 @@ static int itd_submit (struct ehci_hcd *ehci, struct urb *urb,
/* schedule ... need to lock */ /* schedule ... need to lock */
spin_lock_irqsave (&ehci->lock, flags); spin_lock_irqsave (&ehci->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(ehci_to_hcd(ehci)))) {
&ehci_to_hcd(ehci)->flags))) {
status = -ESHUTDOWN; status = -ESHUTDOWN;
goto done_not_linked; goto done_not_linked;
} }
...@@ -2201,8 +2199,7 @@ static int sitd_submit (struct ehci_hcd *ehci, struct urb *urb, ...@@ -2201,8 +2199,7 @@ static int sitd_submit (struct ehci_hcd *ehci, struct urb *urb,
/* schedule ... need to lock */ /* schedule ... need to lock */
spin_lock_irqsave (&ehci->lock, flags); spin_lock_irqsave (&ehci->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(ehci_to_hcd(ehci)))) {
&ehci_to_hcd(ehci)->flags))) {
status = -ESHUTDOWN; status = -ESHUTDOWN;
goto done_not_linked; goto done_not_linked;
} }
......
...@@ -159,7 +159,7 @@ static int hwahc_op_start(struct usb_hcd *usb_hcd) ...@@ -159,7 +159,7 @@ static int hwahc_op_start(struct usb_hcd *usb_hcd)
goto error_set_cluster_id; goto error_set_cluster_id;
usb_hcd->uses_new_polling = 1; usb_hcd->uses_new_polling = 1;
usb_hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &usb_hcd->flags);
usb_hcd->state = HC_STATE_RUNNING; usb_hcd->state = HC_STATE_RUNNING;
result = 0; result = 0;
out: out:
...@@ -776,7 +776,7 @@ static int hwahc_probe(struct usb_interface *usb_iface, ...@@ -776,7 +776,7 @@ static int hwahc_probe(struct usb_interface *usb_iface,
goto error_alloc; goto error_alloc;
} }
usb_hcd->wireless = 1; usb_hcd->wireless = 1;
usb_hcd->flags |= HCD_FLAG_SAW_IRQ; set_bit(HCD_FLAG_SAW_IRQ, &usb_hcd->flags);
wusbhc = usb_hcd_to_wusbhc(usb_hcd); wusbhc = usb_hcd_to_wusbhc(usb_hcd);
hwahc = container_of(wusbhc, struct hwahc, wusbhc); hwahc = container_of(wusbhc, struct hwahc, wusbhc);
hwahc_init(hwahc); hwahc_init(hwahc);
......
...@@ -482,7 +482,6 @@ static int isp1760_run(struct usb_hcd *hcd) ...@@ -482,7 +482,6 @@ static int isp1760_run(struct usb_hcd *hcd)
u32 chipid; u32 chipid;
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
hcd->poll_rh = 0;
hcd->state = HC_STATE_RUNNING; hcd->state = HC_STATE_RUNNING;
isp1760_enable_interrupts(hcd); isp1760_enable_interrupts(hcd);
...@@ -1450,7 +1449,7 @@ static int isp1760_prepare_enqueue(struct isp1760_hcd *priv, struct urb *urb, ...@@ -1450,7 +1449,7 @@ static int isp1760_prepare_enqueue(struct isp1760_hcd *priv, struct urb *urb,
epnum = urb->ep->desc.bEndpointAddress; epnum = urb->ep->desc.bEndpointAddress;
spin_lock_irqsave(&priv->lock, flags); spin_lock_irqsave(&priv->lock, flags);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &priv_to_hcd(priv)->flags)) { if (!HCD_HW_ACCESSIBLE(priv_to_hcd(priv))) {
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
goto done; goto done;
} }
......
...@@ -645,7 +645,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf) ...@@ -645,7 +645,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf)
hcd->product_desc, hcd->product_desc,
hcd_name); hcd_name);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
size -= scnprintf (next, size, size -= scnprintf (next, size,
"SUSPENDED (no register access)\n"); "SUSPENDED (no register access)\n");
goto done; goto done;
...@@ -687,7 +687,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf) ...@@ -687,7 +687,7 @@ static ssize_t fill_registers_buffer(struct debug_buffer *buf)
next += temp; next += temp;
temp = scnprintf (next, size, "hub poll timer %s\n", temp = scnprintf (next, size, "hub poll timer %s\n",
ohci_to_hcd(ohci)->poll_rh ? "ON" : "off"); HCD_POLL_RH(ohci_to_hcd(ohci)) ? "ON" : "off");
size -= temp; size -= temp;
next += temp; next += temp;
......
...@@ -212,7 +212,7 @@ static int ohci_urb_enqueue ( ...@@ -212,7 +212,7 @@ static int ohci_urb_enqueue (
spin_lock_irqsave (&ohci->lock, flags); spin_lock_irqsave (&ohci->lock, flags);
/* don't submit to a dead HC */ /* don't submit to a dead HC */
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
retval = -ENODEV; retval = -ENODEV;
goto fail; goto fail;
} }
...@@ -685,7 +685,7 @@ static int ohci_run (struct ohci_hcd *ohci) ...@@ -685,7 +685,7 @@ static int ohci_run (struct ohci_hcd *ohci)
} }
/* use rhsc irqs after khubd is fully initialized */ /* use rhsc irqs after khubd is fully initialized */
hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &hcd->flags);
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
/* start controller operations */ /* start controller operations */
...@@ -822,7 +822,7 @@ static irqreturn_t ohci_irq (struct usb_hcd *hcd) ...@@ -822,7 +822,7 @@ static irqreturn_t ohci_irq (struct usb_hcd *hcd)
else if (ints & OHCI_INTR_RD) { else if (ints & OHCI_INTR_RD) {
ohci_vdbg(ohci, "resume detect\n"); ohci_vdbg(ohci, "resume detect\n");
ohci_writel(ohci, OHCI_INTR_RD, &regs->intrstatus); ohci_writel(ohci, OHCI_INTR_RD, &regs->intrstatus);
hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &hcd->flags);
if (ohci->autostop) { if (ohci->autostop) {
spin_lock (&ohci->lock); spin_lock (&ohci->lock);
ohci_rh_resume (ohci); ohci_rh_resume (ohci);
......
...@@ -284,7 +284,7 @@ static int ohci_bus_suspend (struct usb_hcd *hcd) ...@@ -284,7 +284,7 @@ static int ohci_bus_suspend (struct usb_hcd *hcd)
spin_lock_irq (&ohci->lock); spin_lock_irq (&ohci->lock);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))) if (unlikely(!HCD_HW_ACCESSIBLE(hcd)))
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
else else
rc = ohci_rh_suspend (ohci, 0); rc = ohci_rh_suspend (ohci, 0);
...@@ -302,7 +302,7 @@ static int ohci_bus_resume (struct usb_hcd *hcd) ...@@ -302,7 +302,7 @@ static int ohci_bus_resume (struct usb_hcd *hcd)
spin_lock_irq (&ohci->lock); spin_lock_irq (&ohci->lock);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))) if (unlikely(!HCD_HW_ACCESSIBLE(hcd)))
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
else else
rc = ohci_rh_resume (ohci); rc = ohci_rh_resume (ohci);
...@@ -489,7 +489,7 @@ ohci_hub_status_data (struct usb_hcd *hcd, char *buf) ...@@ -489,7 +489,7 @@ ohci_hub_status_data (struct usb_hcd *hcd, char *buf)
unsigned long flags; unsigned long flags;
spin_lock_irqsave (&ohci->lock, flags); spin_lock_irqsave (&ohci->lock, flags);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
goto done; goto done;
/* undocumented erratum seen on at least rev D */ /* undocumented erratum seen on at least rev D */
...@@ -533,8 +533,12 @@ ohci_hub_status_data (struct usb_hcd *hcd, char *buf) ...@@ -533,8 +533,12 @@ ohci_hub_status_data (struct usb_hcd *hcd, char *buf)
} }
} }
hcd->poll_rh = ohci_root_hub_state_changes(ohci, changed, if (ohci_root_hub_state_changes(ohci, changed,
any_connected, rhsc_status); any_connected, rhsc_status))
set_bit(HCD_FLAG_POLL_RH, &hcd->flags);
else
clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
done: done:
spin_unlock_irqrestore (&ohci->lock, flags); spin_unlock_irqrestore (&ohci->lock, flags);
...@@ -701,7 +705,7 @@ static int ohci_hub_control ( ...@@ -701,7 +705,7 @@ static int ohci_hub_control (
u32 temp; u32 temp;
int retval = 0; int retval = 0;
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))) if (unlikely(!HCD_HW_ACCESSIBLE(hcd)))
return -ESHUTDOWN; return -ESHUTDOWN;
switch (typeReq) { switch (typeReq) {
......
...@@ -1641,8 +1641,7 @@ static int submit_async(struct oxu_hcd *oxu, struct urb *urb, ...@@ -1641,8 +1641,7 @@ static int submit_async(struct oxu_hcd *oxu, struct urb *urb,
#endif #endif
spin_lock_irqsave(&oxu->lock, flags); spin_lock_irqsave(&oxu->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(oxu_to_hcd(oxu)))) {
&oxu_to_hcd(oxu)->flags))) {
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
goto done; goto done;
} }
...@@ -2209,8 +2208,7 @@ static int intr_submit(struct oxu_hcd *oxu, struct urb *urb, ...@@ -2209,8 +2208,7 @@ static int intr_submit(struct oxu_hcd *oxu, struct urb *urb,
spin_lock_irqsave(&oxu->lock, flags); spin_lock_irqsave(&oxu->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, if (unlikely(!HCD_HW_ACCESSIBLE(oxu_to_hcd(oxu)))) {
&oxu_to_hcd(oxu)->flags))) {
status = -ESHUTDOWN; status = -ESHUTDOWN;
goto done; goto done;
} }
...@@ -2715,7 +2713,6 @@ static int oxu_run(struct usb_hcd *hcd) ...@@ -2715,7 +2713,6 @@ static int oxu_run(struct usb_hcd *hcd)
u32 temp, hcc_params; u32 temp, hcc_params;
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
hcd->poll_rh = 0;
/* EHCI spec section 4.1 */ /* EHCI spec section 4.1 */
retval = ehci_reset(oxu); retval = ehci_reset(oxu);
......
...@@ -140,7 +140,7 @@ static void finish_reset(struct uhci_hcd *uhci) ...@@ -140,7 +140,7 @@ static void finish_reset(struct uhci_hcd *uhci)
uhci->rh_state = UHCI_RH_RESET; uhci->rh_state = UHCI_RH_RESET;
uhci->is_stopped = UHCI_IS_STOPPED; uhci->is_stopped = UHCI_IS_STOPPED;
uhci_to_hcd(uhci)->state = HC_STATE_HALT; uhci_to_hcd(uhci)->state = HC_STATE_HALT;
uhci_to_hcd(uhci)->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &uhci_to_hcd(uhci)->flags);
uhci->dead = 0; /* Full reset resurrects the controller */ uhci->dead = 0; /* Full reset resurrects the controller */
} }
...@@ -344,7 +344,10 @@ __acquires(uhci->lock) ...@@ -344,7 +344,10 @@ __acquires(uhci->lock)
/* If interrupts don't work and remote wakeup is enabled then /* If interrupts don't work and remote wakeup is enabled then
* the suspended root hub needs to be polled. * the suspended root hub needs to be polled.
*/ */
uhci_to_hcd(uhci)->poll_rh = (!int_enable && wakeup_enable); if (!int_enable && wakeup_enable)
set_bit(HCD_FLAG_POLL_RH, &uhci_to_hcd(uhci)->flags);
else
clear_bit(HCD_FLAG_POLL_RH, &uhci_to_hcd(uhci)->flags);
uhci_scan_schedule(uhci); uhci_scan_schedule(uhci);
uhci_fsbr_off(uhci); uhci_fsbr_off(uhci);
...@@ -363,7 +366,7 @@ static void start_rh(struct uhci_hcd *uhci) ...@@ -363,7 +366,7 @@ static void start_rh(struct uhci_hcd *uhci)
uhci->io_addr + USBINTR); uhci->io_addr + USBINTR);
mb(); mb();
uhci->rh_state = UHCI_RH_RUNNING; uhci->rh_state = UHCI_RH_RUNNING;
uhci_to_hcd(uhci)->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &uhci_to_hcd(uhci)->flags);
} }
static void wakeup_rh(struct uhci_hcd *uhci) static void wakeup_rh(struct uhci_hcd *uhci)
...@@ -733,7 +736,7 @@ static void uhci_stop(struct usb_hcd *hcd) ...@@ -733,7 +736,7 @@ static void uhci_stop(struct usb_hcd *hcd)
struct uhci_hcd *uhci = hcd_to_uhci(hcd); struct uhci_hcd *uhci = hcd_to_uhci(hcd);
spin_lock_irq(&uhci->lock); spin_lock_irq(&uhci->lock);
if (test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags) && !uhci->dead) if (HCD_HW_ACCESSIBLE(hcd) && !uhci->dead)
uhci_hc_died(uhci); uhci_hc_died(uhci);
uhci_scan_schedule(uhci); uhci_scan_schedule(uhci);
spin_unlock_irq(&uhci->lock); spin_unlock_irq(&uhci->lock);
...@@ -750,7 +753,7 @@ static int uhci_rh_suspend(struct usb_hcd *hcd) ...@@ -750,7 +753,7 @@ static int uhci_rh_suspend(struct usb_hcd *hcd)
int rc = 0; int rc = 0;
spin_lock_irq(&uhci->lock); spin_lock_irq(&uhci->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
else if (uhci->dead) else if (uhci->dead)
; /* Dead controllers tell no tales */ ; /* Dead controllers tell no tales */
...@@ -777,7 +780,7 @@ static int uhci_rh_resume(struct usb_hcd *hcd) ...@@ -777,7 +780,7 @@ static int uhci_rh_resume(struct usb_hcd *hcd)
int rc = 0; int rc = 0;
spin_lock_irq(&uhci->lock); spin_lock_irq(&uhci->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) if (!HCD_HW_ACCESSIBLE(hcd))
rc = -ESHUTDOWN; rc = -ESHUTDOWN;
else if (!uhci->dead) else if (!uhci->dead)
wakeup_rh(uhci); wakeup_rh(uhci);
...@@ -793,7 +796,7 @@ static int uhci_pci_suspend(struct usb_hcd *hcd) ...@@ -793,7 +796,7 @@ static int uhci_pci_suspend(struct usb_hcd *hcd)
dev_dbg(uhci_dev(uhci), "%s\n", __func__); dev_dbg(uhci_dev(uhci), "%s\n", __func__);
spin_lock_irq(&uhci->lock); spin_lock_irq(&uhci->lock);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags) || uhci->dead) if (!HCD_HW_ACCESSIBLE(hcd) || uhci->dead)
goto done_okay; /* Already suspended or dead */ goto done_okay; /* Already suspended or dead */
if (uhci->rh_state > UHCI_RH_SUSPENDED) { if (uhci->rh_state > UHCI_RH_SUSPENDED) {
...@@ -807,7 +810,7 @@ static int uhci_pci_suspend(struct usb_hcd *hcd) ...@@ -807,7 +810,7 @@ static int uhci_pci_suspend(struct usb_hcd *hcd)
*/ */
pci_write_config_word(to_pci_dev(uhci_dev(uhci)), USBLEGSUP, 0); pci_write_config_word(to_pci_dev(uhci_dev(uhci)), USBLEGSUP, 0);
mb(); mb();
hcd->poll_rh = 0; clear_bit(HCD_FLAG_POLL_RH, &hcd->flags);
/* FIXME: Enable non-PME# remote wakeup? */ /* FIXME: Enable non-PME# remote wakeup? */
...@@ -860,7 +863,7 @@ static int uhci_pci_resume(struct usb_hcd *hcd, bool hibernated) ...@@ -860,7 +863,7 @@ static int uhci_pci_resume(struct usb_hcd *hcd, bool hibernated)
* the suspended root hub needs to be polled. * the suspended root hub needs to be polled.
*/ */
if (!uhci->RD_enable && hcd->self.root_hub->do_remote_wakeup) { if (!uhci->RD_enable && hcd->self.root_hub->do_remote_wakeup) {
hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &hcd->flags);
usb_hcd_poll_rh_status(hcd); usb_hcd_poll_rh_status(hcd);
} }
return 0; return 0;
......
...@@ -190,7 +190,7 @@ static int uhci_hub_status_data(struct usb_hcd *hcd, char *buf) ...@@ -190,7 +190,7 @@ static int uhci_hub_status_data(struct usb_hcd *hcd, char *buf)
spin_lock_irqsave(&uhci->lock, flags); spin_lock_irqsave(&uhci->lock, flags);
uhci_scan_schedule(uhci); uhci_scan_schedule(uhci);
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags) || uhci->dead) if (!HCD_HW_ACCESSIBLE(hcd) || uhci->dead)
goto done; goto done;
uhci_check_ports(uhci); uhci_check_ports(uhci);
...@@ -246,7 +246,7 @@ static int uhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, ...@@ -246,7 +246,7 @@ static int uhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
u16 wPortChange, wPortStatus; u16 wPortChange, wPortStatus;
unsigned long flags; unsigned long flags;
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags) || uhci->dead) if (!HCD_HW_ACCESSIBLE(hcd) || uhci->dead)
return -ETIMEDOUT; return -ETIMEDOUT;
spin_lock_irqsave(&uhci->lock, flags); spin_lock_irqsave(&uhci->lock, flags);
......
...@@ -68,7 +68,7 @@ static int whc_start(struct usb_hcd *usb_hcd) ...@@ -68,7 +68,7 @@ static int whc_start(struct usb_hcd *usb_hcd)
whc_write_wusbcmd(whc, WUSBCMD_RUN, WUSBCMD_RUN); whc_write_wusbcmd(whc, WUSBCMD_RUN, WUSBCMD_RUN);
usb_hcd->uses_new_polling = 1; usb_hcd->uses_new_polling = 1;
usb_hcd->poll_rh = 1; set_bit(HCD_FLAG_POLL_RH, &usb_hcd->flags);
usb_hcd->state = HC_STATE_RUNNING; usb_hcd->state = HC_STATE_RUNNING;
out: out:
......
...@@ -427,7 +427,6 @@ int xhci_run(struct usb_hcd *hcd) ...@@ -427,7 +427,6 @@ int xhci_run(struct usb_hcd *hcd)
void (*doorbell)(struct xhci_hcd *) = NULL; void (*doorbell)(struct xhci_hcd *) = NULL;
hcd->uses_new_polling = 1; hcd->uses_new_polling = 1;
hcd->poll_rh = 0;
xhci_dbg(xhci, "xhci_run\n"); xhci_dbg(xhci, "xhci_run\n");
#if 0 /* FIXME: MSI not setup yet */ #if 0 /* FIXME: MSI not setup yet */
...@@ -733,7 +732,7 @@ int xhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, gfp_t mem_flags) ...@@ -733,7 +732,7 @@ int xhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, gfp_t mem_flags)
ret = -EINVAL; ret = -EINVAL;
goto exit; goto exit;
} }
if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) { if (!HCD_HW_ACCESSIBLE(hcd)) {
if (!in_interrupt()) if (!in_interrupt())
xhci_dbg(xhci, "urb submitted during PCI suspend\n"); xhci_dbg(xhci, "urb submitted during PCI suspend\n");
ret = -ESHUTDOWN; ret = -ESHUTDOWN;
......
...@@ -244,7 +244,7 @@ int musb_hub_control( ...@@ -244,7 +244,7 @@ int musb_hub_control(
spin_lock_irqsave(&musb->lock, flags); spin_lock_irqsave(&musb->lock, flags);
if (unlikely(!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))) { if (unlikely(!HCD_HW_ACCESSIBLE(hcd))) {
spin_unlock_irqrestore(&musb->lock, flags); spin_unlock_irqrestore(&musb->lock, flags);
return -ESHUTDOWN; return -ESHUTDOWN;
} }
......
...@@ -89,19 +89,31 @@ struct usb_hcd { ...@@ -89,19 +89,31 @@ struct usb_hcd {
*/ */
const struct hc_driver *driver; /* hw-specific hooks */ const struct hc_driver *driver; /* hw-specific hooks */
/* Flags that need to be manipulated atomically */ /* Flags that need to be manipulated atomically because they can
* change while the host controller is running. Always use
* set_bit() or clear_bit() to change their values.
*/
unsigned long flags; unsigned long flags;
#define HCD_FLAG_HW_ACCESSIBLE 0x00000001 #define HCD_FLAG_HW_ACCESSIBLE 0 /* at full power */
#define HCD_FLAG_SAW_IRQ 0x00000002 #define HCD_FLAG_SAW_IRQ 1
#define HCD_FLAG_POLL_RH 2 /* poll for rh status? */
#define HCD_FLAG_POLL_PENDING 3 /* status has changed? */
/* The flags can be tested using these macros; they are likely to
* be slightly faster than test_bit().
*/
#define HCD_HW_ACCESSIBLE(hcd) ((hcd)->flags & (1U << HCD_FLAG_HW_ACCESSIBLE))
#define HCD_SAW_IRQ(hcd) ((hcd)->flags & (1U << HCD_FLAG_SAW_IRQ))
#define HCD_POLL_RH(hcd) ((hcd)->flags & (1U << HCD_FLAG_POLL_RH))
#define HCD_POLL_PENDING(hcd) ((hcd)->flags & (1U << HCD_FLAG_POLL_PENDING))
/* Flags that get set only during HCD registration or removal. */
unsigned rh_registered:1;/* is root hub registered? */ unsigned rh_registered:1;/* is root hub registered? */
unsigned rh_pollable:1; /* may we poll the root hub? */ unsigned rh_pollable:1; /* may we poll the root hub? */
/* The next flag is a stopgap, to be removed when all the HCDs /* The next flag is a stopgap, to be removed when all the HCDs
* support the new root-hub polling mechanism. */ * support the new root-hub polling mechanism. */
unsigned uses_new_polling:1; unsigned uses_new_polling:1;
unsigned poll_rh:1; /* poll for rh status? */
unsigned poll_pending:1; /* status has changed? */
unsigned wireless:1; /* Wireless USB HCD */ unsigned wireless:1; /* Wireless USB HCD */
unsigned authorized_default:1; unsigned authorized_default:1;
unsigned has_tt:1; /* Integrated TT in root hub */ unsigned has_tt:1; /* Integrated TT in root hub */
......
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