dwc3: gadget: Take copy of dwc->gadget_driver before releasing lock
gadget_driver can become null if gadget_stop is called while any other gadget callbacks are in progress. As gadget callbacks needs to release spinlock before the callback, store the local copy of gadget_driver to avoid the race with gadget_stop. Change-Id: I7f0cbf9af3e3b286f2826647f08215f29f699de1 Signed-off-by: Mayank Rana <mrana@codeaurora.org> Signed-off-by: Vijayavardhan Vennapusa <vvreddy@codeaurora.org>
This commit is contained in:
parent
1653208bf4
commit
539ae5515f
1 changed files with 18 additions and 4 deletions
|
@ -2674,41 +2674,55 @@ static void dwc3_endpoint_interrupt(struct dwc3 *dwc,
|
|||
|
||||
static void dwc3_disconnect_gadget(struct dwc3 *dwc)
|
||||
{
|
||||
struct usb_gadget_driver *gadget_driver;
|
||||
|
||||
if (dwc->gadget_driver && dwc->gadget_driver->disconnect) {
|
||||
gadget_driver = dwc->gadget_driver;
|
||||
spin_unlock(&dwc->lock);
|
||||
dwc->gadget_driver->disconnect(&dwc->gadget);
|
||||
dbg_event(0xFF, "DISCONNECT", 0);
|
||||
gadget_driver->disconnect(&dwc->gadget);
|
||||
spin_lock(&dwc->lock);
|
||||
}
|
||||
}
|
||||
|
||||
static void dwc3_suspend_gadget(struct dwc3 *dwc)
|
||||
{
|
||||
struct usb_gadget_driver *gadget_driver;
|
||||
|
||||
if (dwc->gadget_driver && dwc->gadget_driver->suspend) {
|
||||
gadget_driver = dwc->gadget_driver;
|
||||
spin_unlock(&dwc->lock);
|
||||
dbg_event(0xFF, "SUSPEND", 0);
|
||||
dwc->gadget_driver->suspend(&dwc->gadget);
|
||||
gadget_driver->suspend(&dwc->gadget);
|
||||
spin_lock(&dwc->lock);
|
||||
}
|
||||
}
|
||||
|
||||
static void dwc3_resume_gadget(struct dwc3 *dwc)
|
||||
{
|
||||
struct usb_gadget_driver *gadget_driver;
|
||||
|
||||
if (dwc->gadget_driver && dwc->gadget_driver->resume) {
|
||||
gadget_driver = dwc->gadget_driver;
|
||||
spin_unlock(&dwc->lock);
|
||||
dbg_event(0xFF, "RESUME", 0);
|
||||
dwc->gadget_driver->resume(&dwc->gadget);
|
||||
gadget_driver->resume(&dwc->gadget);
|
||||
spin_lock(&dwc->lock);
|
||||
}
|
||||
}
|
||||
|
||||
static void dwc3_reset_gadget(struct dwc3 *dwc)
|
||||
{
|
||||
struct usb_gadget_driver *gadget_driver;
|
||||
|
||||
if (!dwc->gadget_driver)
|
||||
return;
|
||||
|
||||
if (dwc->gadget.speed != USB_SPEED_UNKNOWN) {
|
||||
gadget_driver = dwc->gadget_driver;
|
||||
spin_unlock(&dwc->lock);
|
||||
usb_gadget_udc_reset(&dwc->gadget, dwc->gadget_driver);
|
||||
dbg_event(0xFF, "UDC RESET", 0);
|
||||
usb_gadget_udc_reset(&dwc->gadget, gadget_driver);
|
||||
spin_lock(&dwc->lock);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue