mirror of
https://github.com/Fishwaldo/Star64_linux.git
synced 2025-06-17 20:29:24 +00:00
bas_gigaset: fix pre_reset handling
The delayed work function int_in_work() may call usb_reset_device() and thus, indirectly, the driver's pre_reset method. Trying to cancel the work synchronously in that situation would deadlock. Fix by avoiding cancel_work_sync() in the pre_reset method. If the reset was NOT initiated by int_in_work() this might cause int_in_work() to run after the post_reset method, with urb_int_in already resubmitted, so handle that case gracefully. Signed-off-by: Tilman Schmidt <tilman@imap.cc> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
910a578f7e
commit
c6fdd8e5d0
1 changed files with 16 additions and 3 deletions
|
@ -617,7 +617,13 @@ static void int_in_work(struct work_struct *work)
|
||||||
if (rc == 0)
|
if (rc == 0)
|
||||||
/* success, resubmit interrupt read URB */
|
/* success, resubmit interrupt read URB */
|
||||||
rc = usb_submit_urb(urb, GFP_ATOMIC);
|
rc = usb_submit_urb(urb, GFP_ATOMIC);
|
||||||
if (rc != 0 && rc != -ENODEV) {
|
|
||||||
|
switch (rc) {
|
||||||
|
case 0: /* success */
|
||||||
|
case -ENODEV: /* device gone */
|
||||||
|
case -EINVAL: /* URB already resubmitted, or terminal badness */
|
||||||
|
break;
|
||||||
|
default: /* failure: try to recover by resetting the device */
|
||||||
dev_err(cs->dev, "clear halt failed: %s\n", get_usb_rcmsg(rc));
|
dev_err(cs->dev, "clear halt failed: %s\n", get_usb_rcmsg(rc));
|
||||||
rc = usb_lock_device_for_reset(ucs->udev, ucs->interface);
|
rc = usb_lock_device_for_reset(ucs->udev, ucs->interface);
|
||||||
if (rc == 0) {
|
if (rc == 0) {
|
||||||
|
@ -2442,7 +2448,9 @@ static void gigaset_disconnect(struct usb_interface *interface)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* gigaset_suspend
|
/* gigaset_suspend
|
||||||
* This function is called before the USB connection is suspended.
|
* This function is called before the USB connection is suspended
|
||||||
|
* or before the USB device is reset.
|
||||||
|
* In the latter case, message == PMSG_ON.
|
||||||
*/
|
*/
|
||||||
static int gigaset_suspend(struct usb_interface *intf, pm_message_t message)
|
static int gigaset_suspend(struct usb_interface *intf, pm_message_t message)
|
||||||
{
|
{
|
||||||
|
@ -2498,6 +2506,11 @@ static int gigaset_suspend(struct usb_interface *intf, pm_message_t message)
|
||||||
del_timer_sync(&ucs->timer_atrdy);
|
del_timer_sync(&ucs->timer_atrdy);
|
||||||
del_timer_sync(&ucs->timer_cmd_in);
|
del_timer_sync(&ucs->timer_cmd_in);
|
||||||
del_timer_sync(&ucs->timer_int_in);
|
del_timer_sync(&ucs->timer_int_in);
|
||||||
|
|
||||||
|
/* don't try to cancel int_in_wq from within reset as it
|
||||||
|
* might be the one requesting the reset
|
||||||
|
*/
|
||||||
|
if (message.event != PM_EVENT_ON)
|
||||||
cancel_work_sync(&ucs->int_in_wq);
|
cancel_work_sync(&ucs->int_in_wq);
|
||||||
|
|
||||||
gig_dbg(DEBUG_SUSPEND, "suspend complete");
|
gig_dbg(DEBUG_SUSPEND, "suspend complete");
|
||||||
|
|
Loading…
Add table
Reference in a new issue