Skip to content

Commit 0791971

Browse files
Thadeu Lima de Souza Cascardogregkh
authored andcommitted
usb: allow drivers to use allocated bandwidth until unbound
When using the remove sysfs file, the device configuration is set to -1 (unconfigured). This eventually unbind drivers with the bandwidth_mutex held. Some drivers may call functions that hold said mutex, like usb_reset_device. This is the case for rtl8187, for example. This will lead to the same process holding the mutex twice, which deadlocks. Besides, according to Alan Stern: "The deadlock problem probably could be handled somehow, but there's a separate issue: Until the usb_disable_device call finishes unbinding the drivers, the drivers are free to continue using their allocated bandwidth. We musn't change the bandwidth allocations until after the unbinding is done. So this patch is indeed necessary." Unbinding the driver before holding the bandwidth_mutex solves the problem. If any operation after that fails, drivers are not bound again. But that would be a problem anyway that the user may solve resetting the device configuration to one that works, just like he would need to do in most other failure cases. Signed-off-by: Thadeu Lima de Souza Cascardo <[email protected]> Cc: Alan Stern <[email protected]> Cc: Sarah Sharp <[email protected]> Cc: stable <[email protected]> Signed-off-by: Greg Kroah-Hartman <[email protected]>
1 parent 5b22a32 commit 0791971

File tree

1 file changed

+11
-11
lines changed

1 file changed

+11
-11
lines changed

drivers/usb/core/message.c

Lines changed: 11 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -1724,6 +1724,15 @@ int usb_set_configuration(struct usb_device *dev, int configuration)
17241724
if (ret)
17251725
goto free_interfaces;
17261726

1727+
/* if it's already configured, clear out old state first.
1728+
* getting rid of old interfaces means unbinding their drivers.
1729+
*/
1730+
if (dev->state != USB_STATE_ADDRESS)
1731+
usb_disable_device(dev, 1); /* Skip ep0 */
1732+
1733+
/* Get rid of pending async Set-Config requests for this device */
1734+
cancel_async_set_config(dev);
1735+
17271736
/* Make sure we have bandwidth (and available HCD resources) for this
17281737
* configuration. Remove endpoints from the schedule if we're dropping
17291738
* this configuration to set configuration 0. After this point, the
@@ -1733,20 +1742,11 @@ int usb_set_configuration(struct usb_device *dev, int configuration)
17331742
mutex_lock(&hcd->bandwidth_mutex);
17341743
ret = usb_hcd_alloc_bandwidth(dev, cp, NULL, NULL);
17351744
if (ret < 0) {
1736-
usb_autosuspend_device(dev);
17371745
mutex_unlock(&hcd->bandwidth_mutex);
1746+
usb_autosuspend_device(dev);
17381747
goto free_interfaces;
17391748
}
17401749

1741-
/* if it's already configured, clear out old state first.
1742-
* getting rid of old interfaces means unbinding their drivers.
1743-
*/
1744-
if (dev->state != USB_STATE_ADDRESS)
1745-
usb_disable_device(dev, 1); /* Skip ep0 */
1746-
1747-
/* Get rid of pending async Set-Config requests for this device */
1748-
cancel_async_set_config(dev);
1749-
17501750
ret = usb_control_msg(dev, usb_sndctrlpipe(dev, 0),
17511751
USB_REQ_SET_CONFIGURATION, 0, configuration, 0,
17521752
NULL, 0, USB_CTRL_SET_TIMEOUT);
@@ -1761,8 +1761,8 @@ int usb_set_configuration(struct usb_device *dev, int configuration)
17611761
if (!cp) {
17621762
usb_set_device_state(dev, USB_STATE_ADDRESS);
17631763
usb_hcd_alloc_bandwidth(dev, NULL, NULL, NULL);
1764-
usb_autosuspend_device(dev);
17651764
mutex_unlock(&hcd->bandwidth_mutex);
1765+
usb_autosuspend_device(dev);
17661766
goto free_interfaces;
17671767
}
17681768
mutex_unlock(&hcd->bandwidth_mutex);

0 commit comments

Comments
 (0)