Hi Shimoda-san,
On Mon, Feb 16, 2015 at 2:52 AM, Yoshihiro Shimoda
<[email protected]> wrote:
> According to the gadget.h, a "complete" function will always be called
> with interrupts disabled. However, sometimes usbhsg_queue_pop() function
> is called with interrupts enabled. So, this function should calls
> local_irq_save() before this calls the usb_gadget_giveback_request().
> Otherwise, there is possible to cause a spinlock suspected in a gadget
> complete function.
I still feel uneasy about adding the call to local_irq_save(), as the need for
this is usually an indicator of another locking problem.
Unfortunately I know next to nothing about the USB gadget subsystem, so some
help from the USB gadget experts would be appreciated.
I had a look at other drivers, and it seems most drivers actually release
and reacquire a spinlock around the call to usb_gadget_giveback_request(),
i.e. they do:
spin_unlock(...);
usb_gadget_giveback_request(...);
spin_lock();
This means they had already acquired the spinlock (and disabled interrupts!)
before, which looks much healthier to me.
There's only one driver that uses local_irq_save() (pxa27x_udc).
> Signed-off-by: Yoshihiro Shimoda <[email protected]>
> ---
> drivers/usb/renesas_usbhs/mod_gadget.c | 11 +++++++++++
> 1 file changed, 11 insertions(+)
>
> diff --git a/drivers/usb/renesas_usbhs/mod_gadget.c
> b/drivers/usb/renesas_usbhs/mod_gadget.c
> index e0384af..104bddf 100644
> --- a/drivers/usb/renesas_usbhs/mod_gadget.c
> +++ b/drivers/usb/renesas_usbhs/mod_gadget.c
> @@ -126,11 +126,22 @@ static void usbhsg_queue_pop(struct usbhsg_uep *uep,
> struct usbhsg_gpriv *gpriv = usbhsg_uep_to_gpriv(uep);
> struct usbhs_pipe *pipe = usbhsg_uep_to_pipe(uep);
> struct device *dev = usbhsg_gpriv_to_dev(gpriv);
> + unsigned long flags;
>
> dev_dbg(dev, "pipe %d : queue pop\n", usbhs_pipe_number(pipe));
>
> ureq->req.status = status;
> + /*
> + * According to the gadget.h, a "complete" function will always be
> + * called with interrupts disabled. However, sometimes this function
> + * is called with interrupts enabled. (e.g. complete a DMAC transfer
> or
> + * write data and done is set immediately when PIO.) So, this function
> + * should calls local_irq_save() before this calls the
> + * usb_gadget_giveback_request().
> + */
> + local_irq_save(flags);
> usb_gadget_giveback_request(&uep->ep, &ureq->req);
> + local_irq_restore(flags);
> }
>
> static void usbhsg_queue_done(struct usbhs_priv *priv, struct usbhs_pkt *pkt)
Gr{oetje,eeting}s,
Geert
--
Geert Uytterhoeven -- There's lots of Linux beyond ia32 -- [email protected]
In personal conversations with technical people, I call myself a hacker. But
when I'm talking to journalists I just say "programmer" or something like that.
-- Linus Torvalds
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to [email protected]
More majordomo info at http://vger.kernel.org/majordomo-info.html