Commit c1ad06e8 authored by Alan Stern's avatar Alan Stern Committed by Jiri Slaby

USB: unbind all interfaces before rebinding any

commit 6aec044c upstream.

When a driver doesn't have pre_reset, post_reset, or reset_resume
methods, the USB core unbinds that driver when its device undergoes a
reset or a reset-resume, and then rebinds it afterward.

The existing straightforward implementation can lead to problems,
because each interface gets unbound and rebound before the next
interface is handled.  If a driver claims additional interfaces, the
claim may fail because the old binding instance may still own the
additional interface when the new instance tries to claim it.

This patch fixes the problem by first unbinding all the interfaces
that are marked (i.e., their needs_binding flag is set) and then
rebinding all of them.

The patch also makes the helper functions in driver.c a little more
uniform and adjusts some out-of-date comments.
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Reported-and-tested-by: default avatar"Poulain, Loic" <loic.poulain@intel.com>
Signed-off-by: default avatarJiri Slaby <jslaby@suse.cz>
parent 2c66d33d
...@@ -959,8 +959,7 @@ EXPORT_SYMBOL_GPL(usb_deregister); ...@@ -959,8 +959,7 @@ EXPORT_SYMBOL_GPL(usb_deregister);
* it doesn't support pre_reset/post_reset/reset_resume or * it doesn't support pre_reset/post_reset/reset_resume or
* because it doesn't support suspend/resume. * because it doesn't support suspend/resume.
* *
* The caller must hold @intf's device's lock, but not its pm_mutex * The caller must hold @intf's device's lock, but not @intf's lock.
* and not @intf->dev.sem.
*/ */
void usb_forced_unbind_intf(struct usb_interface *intf) void usb_forced_unbind_intf(struct usb_interface *intf)
{ {
...@@ -973,16 +972,37 @@ void usb_forced_unbind_intf(struct usb_interface *intf) ...@@ -973,16 +972,37 @@ void usb_forced_unbind_intf(struct usb_interface *intf)
intf->needs_binding = 1; intf->needs_binding = 1;
} }
/*
* Unbind drivers for @udev's marked interfaces. These interfaces have
* the needs_binding flag set, for example by usb_resume_interface().
*
* The caller must hold @udev's device lock.
*/
static void unbind_marked_interfaces(struct usb_device *udev)
{
struct usb_host_config *config;
int i;
struct usb_interface *intf;
config = udev->actconfig;
if (config) {
for (i = 0; i < config->desc.bNumInterfaces; ++i) {
intf = config->interface[i];
if (intf->dev.driver && intf->needs_binding)
usb_forced_unbind_intf(intf);
}
}
}
/* Delayed forced unbinding of a USB interface driver and scan /* Delayed forced unbinding of a USB interface driver and scan
* for rebinding. * for rebinding.
* *
* The caller must hold @intf's device's lock, but not its pm_mutex * The caller must hold @intf's device's lock, but not @intf's lock.
* and not @intf->dev.sem.
* *
* Note: Rebinds will be skipped if a system sleep transition is in * Note: Rebinds will be skipped if a system sleep transition is in
* progress and the PM "complete" callback hasn't occurred yet. * progress and the PM "complete" callback hasn't occurred yet.
*/ */
void usb_rebind_intf(struct usb_interface *intf) static void usb_rebind_intf(struct usb_interface *intf)
{ {
int rc; int rc;
...@@ -999,68 +1019,66 @@ void usb_rebind_intf(struct usb_interface *intf) ...@@ -999,68 +1019,66 @@ void usb_rebind_intf(struct usb_interface *intf)
} }
} }
#ifdef CONFIG_PM /*
* Rebind drivers to @udev's marked interfaces. These interfaces have
/* Unbind drivers for @udev's interfaces that don't support suspend/resume * the needs_binding flag set.
* There is no check for reset_resume here because it can be determined
* only during resume whether reset_resume is needed.
* *
* The caller must hold @udev's device lock. * The caller must hold @udev's device lock.
*/ */
static void unbind_no_pm_drivers_interfaces(struct usb_device *udev) static void rebind_marked_interfaces(struct usb_device *udev)
{ {
struct usb_host_config *config; struct usb_host_config *config;
int i; int i;
struct usb_interface *intf; struct usb_interface *intf;
struct usb_driver *drv;
config = udev->actconfig; config = udev->actconfig;
if (config) { if (config) {
for (i = 0; i < config->desc.bNumInterfaces; ++i) { for (i = 0; i < config->desc.bNumInterfaces; ++i) {
intf = config->interface[i]; intf = config->interface[i];
if (intf->needs_binding)
if (intf->dev.driver) { usb_rebind_intf(intf);
drv = to_usb_driver(intf->dev.driver);
if (!drv->suspend || !drv->resume)
usb_forced_unbind_intf(intf);
}
} }
} }
} }
/* Unbind drivers for @udev's interfaces that failed to support reset-resume. /*
* These interfaces have the needs_binding flag set by usb_resume_interface(). * Unbind all of @udev's marked interfaces and then rebind all of them.
* This ordering is necessary because some drivers claim several interfaces
* when they are first probed.
* *
* The caller must hold @udev's device lock. * The caller must hold @udev's device lock.
*/ */
static void unbind_no_reset_resume_drivers_interfaces(struct usb_device *udev) void usb_unbind_and_rebind_marked_interfaces(struct usb_device *udev)
{ {
struct usb_host_config *config; unbind_marked_interfaces(udev);
int i; rebind_marked_interfaces(udev);
struct usb_interface *intf;
config = udev->actconfig;
if (config) {
for (i = 0; i < config->desc.bNumInterfaces; ++i) {
intf = config->interface[i];
if (intf->dev.driver && intf->needs_binding)
usb_forced_unbind_intf(intf);
}
}
} }
static void do_rebind_interfaces(struct usb_device *udev) #ifdef CONFIG_PM
/* Unbind drivers for @udev's interfaces that don't support suspend/resume
* There is no check for reset_resume here because it can be determined
* only during resume whether reset_resume is needed.
*
* The caller must hold @udev's device lock.
*/
static void unbind_no_pm_drivers_interfaces(struct usb_device *udev)
{ {
struct usb_host_config *config; struct usb_host_config *config;
int i; int i;
struct usb_interface *intf; struct usb_interface *intf;
struct usb_driver *drv;
config = udev->actconfig; config = udev->actconfig;
if (config) { if (config) {
for (i = 0; i < config->desc.bNumInterfaces; ++i) { for (i = 0; i < config->desc.bNumInterfaces; ++i) {
intf = config->interface[i]; intf = config->interface[i];
if (intf->needs_binding)
usb_rebind_intf(intf); if (intf->dev.driver) {
drv = to_usb_driver(intf->dev.driver);
if (!drv->suspend || !drv->resume)
usb_forced_unbind_intf(intf);
}
} }
} }
} }
...@@ -1389,7 +1407,7 @@ int usb_resume_complete(struct device *dev) ...@@ -1389,7 +1407,7 @@ int usb_resume_complete(struct device *dev)
* whose needs_binding flag is set * whose needs_binding flag is set
*/ */
if (udev->state != USB_STATE_NOTATTACHED) if (udev->state != USB_STATE_NOTATTACHED)
do_rebind_interfaces(udev); rebind_marked_interfaces(udev);
return 0; return 0;
} }
...@@ -1411,7 +1429,7 @@ int usb_resume(struct device *dev, pm_message_t msg) ...@@ -1411,7 +1429,7 @@ int usb_resume(struct device *dev, pm_message_t msg)
pm_runtime_disable(dev); pm_runtime_disable(dev);
pm_runtime_set_active(dev); pm_runtime_set_active(dev);
pm_runtime_enable(dev); pm_runtime_enable(dev);
unbind_no_reset_resume_drivers_interfaces(udev); unbind_marked_interfaces(udev);
} }
/* Avoid PM error messages for devices disconnected while suspended /* Avoid PM error messages for devices disconnected while suspended
......
...@@ -5308,10 +5308,11 @@ int usb_reset_device(struct usb_device *udev) ...@@ -5308,10 +5308,11 @@ int usb_reset_device(struct usb_device *udev)
else if (cintf->condition == else if (cintf->condition ==
USB_INTERFACE_BOUND) USB_INTERFACE_BOUND)
rebind = 1; rebind = 1;
if (rebind)
cintf->needs_binding = 1;
} }
if (ret == 0 && rebind)
usb_rebind_intf(cintf);
} }
usb_unbind_and_rebind_marked_interfaces(udev);
} }
usb_autosuspend_device(udev); usb_autosuspend_device(udev);
......
...@@ -55,7 +55,7 @@ extern int usb_match_one_id_intf(struct usb_device *dev, ...@@ -55,7 +55,7 @@ extern int usb_match_one_id_intf(struct usb_device *dev,
extern int usb_match_device(struct usb_device *dev, extern int usb_match_device(struct usb_device *dev,
const struct usb_device_id *id); const struct usb_device_id *id);
extern void usb_forced_unbind_intf(struct usb_interface *intf); extern void usb_forced_unbind_intf(struct usb_interface *intf);
extern void usb_rebind_intf(struct usb_interface *intf); extern void usb_unbind_and_rebind_marked_interfaces(struct usb_device *udev);
extern int usb_hub_claim_port(struct usb_device *hdev, unsigned port, extern int usb_hub_claim_port(struct usb_device *hdev, unsigned port,
struct dev_state *owner); struct dev_state *owner);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment