Re: [PATCH] usb/gadget/pch_udc: Fix ether gadget connect/disconnect issue

From: Tomoya MORINAGA
Date: Wed Jan 11 2012 - 21:27:32 EST


Hi, Balbi

Please see my comment inline.
BTW, I've divided previous patch into six patches.
After this mail, I'll post the patch series soon.

2012/1/9 Felipe Balbi <balbi@xxxxxx>:
> > @@ -1126,7 +1166,23 @@ static int pch_udc_pcd_pullup(struct usb_
gadget *gadget, int is_on)
> >       if (!gadget)
> >               return -EINVAL;
> >       dev = container_of(gadget, struct pch_udc_dev, gadget);
> > -     pch_udc_vbus_session(dev, is_on);
> > +     if (is_on) {
> > +             if (dev->pullup == 1) {
> > +                     pch_udc_clear_disconnect(dev);
> > +             } else {
> > +                     pch_udc_reconnect(dev);
> > +                     dev->pullup = 1;
> > +             }
> > +     } else {
> > +             if (dev->driver && dev->driver->disconnect) {
> > +                     spin_unlock(&dev->lock);
> > +                     dev->driver->disconnect(&dev->gadget);
> > +                     spin_lock(&dev->lock);
>
> this looks very wrong. ->pullup() should not call into the gadget
> driver. Looking over your driver, it seems that one of the issues is
> that you have a race between ->vbus_session() and ->pullup() because
> they both call your pch_udc_vbus_session() without taking care of
> locking those two.

I was not careful about those two conflict. I make the condition simple.
Please see my new patch, and please give me your advice.
Please refer to new 4th patch.

> > @@ -2335,8 +2391,11 @@ static void pch_udc_svc_ur_interrupt(struct pch_udc_dev *dev)
> >               /* Complete request queue */
> >               empty_req_queue(ep);
> >       }
> > -     if (dev->driver && dev->driver->disconnect)
> > +     if (dev->driver && dev->driver->disconnect) {
> > +             spin_unlock(&dev->lock);
> >               dev->driver->disconnect(&dev->gadget);
> > +             spin_lock(&dev->lock);
>
> this needs to be fixed, indeed. And maybe it's the only thing which fits
> under $SUBJECT. Can you describe better the problem you're seeing ?
> Maybe showing any WARN()/BUG() output which showed up ?
>

After a USB cable is connect/disconnected, the system rarely freezes.
I do not have conclusive evidence that this is a root cause.
It was pointed out that this should call spin_unlock() and spin_lock()
by one user.
Please refer to new 1st patch.

> > @@ -2371,6 +2430,11 @@ static void pch_udc_svc_enum_interrupt(struct pch_udc_dev *dev)
> >       pch_udc_set_dma(dev, DMA_DIR_TX);
> >       pch_udc_set_dma(dev, DMA_DIR_RX);
> >       pch_udc_ep_set_rrdy(&(dev->ep[UDC_EP0OUT_IDX]));
> > +
> > +     /* enable device interrupts */
> > +     pch_udc_enable_interrupts(dev, UDC_DEVINT_UR | UDC_DEVINT_US |
> > +                                     UDC_DEVINT_ES | UDC_DEVINT_ENUM |
> > +                                     UDC_DEVINT_SI | UDC_DEVINT_SC);
>
> doesn't fit under $SUBJECT.

This is separated from this $SUBJECT.
Please refer to new 5th patch.

>
> > @@ -2460,11 +2524,15 @@ static void pch_udc_svc_cfg_interrupt(struct pch_udc_dev *dev)
> >  static void pch_udc_dev_isr(struct pch_udc_dev *dev, u32 dev_intr)
> >  {
> >       /* USB Reset Interrupt */
> > -     if (dev_intr & UDC_DEVINT_UR)
> > +     if (dev_intr & UDC_DEVINT_UR) {
> >               pch_udc_svc_ur_interrupt(dev);
> > +             dev_dbg(&dev->pdev->dev, "USB_RESET\n");
>
> all these debugging message are _not_ part of $SUBJECT.
>

This is separated from this $SUBJECT.
Please refer to new 6th patch.

> > @@ -2472,8 +2540,25 @@ static void pch_udc_dev_isr(struct pch_udc_dev *dev, u32 dev_intr)
> >       if (dev_intr & UDC_DEVINT_SC)
> >               pch_udc_svc_cfg_interrupt(dev);
> >       /* USB Suspend interrupt */
> > -     if (dev_intr & UDC_DEVINT_US)
> > +     if (dev_intr & UDC_DEVINT_US) {
> > +             if (dev->gadget.speed != USB_SPEED_UNKNOWN
>
> In fact, I would WARN() if your speed isn't known at this spot. This
> would mean you have a big fat bug on your controller driver.
>

It was not necessary to check speed in this spot.
gadget.speed = USB_SPEED_UNKNOWN is removed.
Please refer to new 3rd patch.


> > +                     && dev->driver
> > +                     && dev->driver->suspend) {
> > +                     spin_unlock(&dev->lock);
> > +                     dev->driver->suspend(&dev->gadget);
> > +                     spin_lock(&dev->lock);
>
> Indeed, this is the right way, but not part of $SUBJECT.

This is separated from this $SUBJECT.
Please refer to new 3rd patch.

> > +             }
> > +
> > +             if ((dev->vbus_session == 0) && (dev->pullup == 1)) {
> > +                     if (dev->driver && dev->driver->disconnect) {
> > +                             spin_unlock(&dev->lock);
> > +                             dev->driver->disconnect(&dev->gadget);
> > +                             spin_lock(&dev->lock);
>
> also correct, and looks part of $SUBJECT.

Please refer to new 4th patch.

> > @@ -2499,6 +2584,14 @@ static irqreturn_t pch_udc_isr(int irq, void *pdev)
> >       dev_intr = pch_udc_read_device_interrupts(dev);
> >       ep_intr = pch_udc_read_ep_interrupts(dev);
> >
> > +     /* For a hot plug, this find that the controller is hung up. */
> > +     if (dev_intr == ep_intr)
> > +             if (dev_intr == pch_udc_readl(dev, UDC_DEVCFG_ADDR)) {
> > +                     dev_dbg(&dev->pdev->dev, "UDC: Hung up\n");
> > +                     /* The controller is reset */
> > +                     pch_udc_writel(dev, UDC_SRST, UDC_SRST_ADDR);
> > +                     return IRQ_HANDLED;
> > +             }
> >       if (dev_intr)
> >               /* Clear device interrupts */
> >               pch_udc_write_device_interrupts(dev, dev_intr);
> > @@ -2725,7 +2818,7 @@ static int pch_udc_start(struct usb_gadget_driver *driver,
> >       pch_udc_setup_ep0(dev);
> >
> >       /* clear SD */
> > -     pch_udc_clear_disconnect(dev);
> > +     pch_udc_pcd_pullup(&dev->gadget, 1);
>
> this rings a bell, why do you need to change this ?

I wanted to set as 1 to dev->pullup.
However, this is not needed anymore.


> > @@ -2912,8 +3005,10 @@ static int pch_udc_probe(struct pci_dev *pdev,
> >       }
> >       pch_udc = dev;
> >       /* initialize the hardware */
> > -     if (pch_udc_pcd_init(dev))
> > +     if (pch_udc_pcd_init(dev)) {
> > +             retval = -ENODEV;
>
> not part of $SUBJECT but how about passing the return value from
> pch_udc_pcd_init() to upper layer instead of changing it here ?
>
> ret = pch_udc_pcd_init(dev);
> if (ret) {
>        ....
>

This is separated from this $ SUBJECT.
Your description is better.
However, I followed a similar description in this function.
Please refer to new 2nd patch.

thanks,
tomoya
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/