On Wednesday 05 December 2007, Alan Stern wrote:
> On Wed, 5 Dec 2007, David Brownell wrote:
> 
> > On Wednesday 05 December 2007, Alan Stern wrote:
> > > But it will be easy enough to add IRQF_DISABLED in.  Only a few drivers 
> > > omit it.
> > 
> > I forgot that the flags are passed in to usb_add_hcd()... if we
> > go that route, the drivers that don't pass it should be updated
> > and so should hcd-pci.c; that seems the safest option to me.
> 
> The proposed patch is below.

OK by me ... ack.  And schedule merge for 2.6.24-final ...

Re Pete's complaint, I can sort of understand not wanting IRQF_DISABLED;
but that's why I commented that this option seems "safest".  Alternative
solutions to this little surprise would need more investigation, on more
varied hardware, than I think we can deliver just now.  This way has the
advantage of being "obviously correct".


> > This looks like a longstanding bug/oversight.  Color me surprised.
> 
> It's possible that it has caused undiagnosed lockups in the past.  The 
> recent changes for urb->status removal must have made them much more 
> likely.  For instance, the urb_unlink() routine formerly in hcd.c 
> used spin_lock_irqsave() but its replacement
> usb_hcd_unlink_urb_from_ep() uses only spin_lock().  That particular 
> spinlock is what led to the bug report.

Exactly.

- Dave


 
> Alan Stern
> 
> 
> Host controller IRQs are supposed to be serviced with interrupts
> disabled.  This patch (as1025) adds an IRQF_DISABLED flag to all the
> controller drivers that lack it.  It also replaces the
> spin_lock_irqsave() and spin_unlock_irqrestore() calls in uhci_irq()
> with simple spin_lock() and spin_unlock().
> 
> This fixes Bugzilla #9335.
> 
> Signed-off-by: Alan Stern <[EMAIL PROTECTED]>
> 
> ---
> 
> Index: usb-2.6/drivers/usb/core/hcd-pci.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/core/hcd-pci.c
> +++ usb-2.6/drivers/usb/core/hcd-pci.c
> @@ -125,7 +125,7 @@ int usb_hcd_pci_probe (struct pci_dev *d
>  
>       pci_set_master (dev);
>  
> -     retval = usb_add_hcd (hcd, dev->irq, IRQF_SHARED);
> +     retval = usb_add_hcd(hcd, dev->irq, IRQF_DISABLED | IRQF_SHARED);
>       if (retval != 0)
>               goto err4;
>       return retval;
> Index: usb-2.6/drivers/usb/host/ehci-fsl.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/host/ehci-fsl.c
> +++ usb-2.6/drivers/usb/host/ehci-fsl.c
> @@ -122,7 +122,7 @@ int usb_hcd_fsl_probe(const struct hc_dr
>       temp = in_le32(hcd->regs + 0x1a8);
>       out_le32(hcd->regs + 0x1a8, temp | 0x3);
>  
> -     retval = usb_add_hcd(hcd, irq, IRQF_SHARED);
> +     retval = usb_add_hcd(hcd, irq, IRQF_DISABLED | IRQF_SHARED);
>       if (retval != 0)
>               goto err4;
>       return retval;
> Index: usb-2.6/drivers/usb/host/ohci-ppc-of.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/host/ohci-ppc-of.c
> +++ usb-2.6/drivers/usb/host/ohci-ppc-of.c
> @@ -142,7 +142,7 @@ ohci_hcd_ppc_of_probe(struct of_device *
>  
>       ohci_hcd_init(ohci);
>  
> -     rv = usb_add_hcd(hcd, irq, 0);
> +     rv = usb_add_hcd(hcd, irq, IRQF_DISABLED);
>       if (rv == 0)
>               return 0;
>  
> Index: usb-2.6/drivers/usb/host/ohci-ssb.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/host/ohci-ssb.c
> +++ usb-2.6/drivers/usb/host/ohci-ssb.c
> @@ -160,7 +160,7 @@ static int ssb_ohci_attach(struct ssb_de
>       hcd->regs = ioremap_nocache(hcd->rsrc_start, hcd->rsrc_len);
>       if (!hcd->regs)
>               goto err_put_hcd;
> -     err = usb_add_hcd(hcd, dev->irq, IRQF_SHARED);
> +     err = usb_add_hcd(hcd, dev->irq, IRQF_DISABLED | IRQF_SHARED);
>       if (err)
>               goto err_iounmap;
>  
> Index: usb-2.6/drivers/usb/host/r8a66597-hcd.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/host/r8a66597-hcd.c
> +++ usb-2.6/drivers/usb/host/r8a66597-hcd.c
> @@ -2197,7 +2197,7 @@ static int __init r8a66597_probe(struct 
>       INIT_LIST_HEAD(&r8a66597->child_device);
>  
>       hcd->rsrc_start = res->start;
> -     ret = usb_add_hcd(hcd, irq, 0);
> +     ret = usb_add_hcd(hcd, irq, IRQF_DISABLED);
>       if (ret != 0) {
>               err("Failed to add hcd");
>               goto clean_up;
> Index: usb-2.6/drivers/usb/host/uhci-hcd.c
> ===================================================================
> --- usb-2.6.orig/drivers/usb/host/uhci-hcd.c
> +++ usb-2.6/drivers/usb/host/uhci-hcd.c
> @@ -378,7 +378,6 @@ static irqreturn_t uhci_irq(struct usb_h
>  {
>       struct uhci_hcd *uhci = hcd_to_uhci(hcd);
>       unsigned short status;
> -     unsigned long flags;
>  
>       /*
>        * Read the interrupt status, and write it back to clear the
> @@ -398,7 +397,7 @@ static irqreturn_t uhci_irq(struct usb_h
>                       dev_err(uhci_dev(uhci), "host controller process "
>                                       "error, something bad happened!\n");
>               if (status & USBSTS_HCH) {
> -                     spin_lock_irqsave(&uhci->lock, flags);
> +                     spin_lock(&uhci->lock);
>                       if (uhci->rh_state >= UHCI_RH_RUNNING) {
>                               dev_err(uhci_dev(uhci),
>                                       "host controller halted, "
> @@ -415,16 +414,16 @@ static irqreturn_t uhci_irq(struct usb_h
>                                * pending unlinks */
>                               mod_timer(&hcd->rh_timer, jiffies);
>                       }
> -                     spin_unlock_irqrestore(&uhci->lock, flags);
> +                     spin_unlock(&uhci->lock);
>               }
>       }
>  
>       if (status & USBSTS_RD)
>               usb_hcd_poll_rh_status(hcd);
>       else {
> -             spin_lock_irqsave(&uhci->lock, flags);
> +             spin_lock(&uhci->lock);
>               uhci_scan_schedule(uhci);
> -             spin_unlock_irqrestore(&uhci->lock, flags);
> +             spin_unlock(&uhci->lock);
>       }
>  
>       return IRQ_HANDLED;
> 


-
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to