USB: change interface to usb_lock_device_for_reset()
This patch (as1161) changes the interface to usb_lock_device_for_reset(). The existing interface is apparently not very clear, judging from the fact that several of its callers don't use it correctly. The new interface always returns 0 for success and it always requires the caller to unlock the device afterward. The new routine will not return immediately if it is called while the driver's probe method is running. Instead it will wait until the probe is over and the device has been unlocked. This shouldn't cause any problems; I don't know of any cases where drivers call usb_lock_device_for_reset() during probe. Signed-off-by: Alan Stern <stern@rowland.harvard.edu> Cc: Pete Zaitcev <zaitcev@redhat.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
Родитель
857cc4dfb6
Коммит
011b15df46
|
@ -1579,7 +1579,7 @@ static void ub_reset_task(struct work_struct *work)
|
||||||
struct ub_dev *sc = container_of(work, struct ub_dev, reset_work);
|
struct ub_dev *sc = container_of(work, struct ub_dev, reset_work);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct ub_lun *lun;
|
struct ub_lun *lun;
|
||||||
int lkr, rc;
|
int rc;
|
||||||
|
|
||||||
if (!sc->reset) {
|
if (!sc->reset) {
|
||||||
printk(KERN_WARNING "%s: Running reset unrequested\n",
|
printk(KERN_WARNING "%s: Running reset unrequested\n",
|
||||||
|
@ -1597,10 +1597,11 @@ static void ub_reset_task(struct work_struct *work)
|
||||||
} else if (sc->dev->actconfig->desc.bNumInterfaces != 1) {
|
} else if (sc->dev->actconfig->desc.bNumInterfaces != 1) {
|
||||||
;
|
;
|
||||||
} else {
|
} else {
|
||||||
if ((lkr = usb_lock_device_for_reset(sc->dev, sc->intf)) < 0) {
|
rc = usb_lock_device_for_reset(sc->dev, sc->intf);
|
||||||
|
if (rc < 0) {
|
||||||
printk(KERN_NOTICE
|
printk(KERN_NOTICE
|
||||||
"%s: usb_lock_device_for_reset failed (%d)\n",
|
"%s: usb_lock_device_for_reset failed (%d)\n",
|
||||||
sc->name, lkr);
|
sc->name, rc);
|
||||||
} else {
|
} else {
|
||||||
rc = usb_reset_device(sc->dev);
|
rc = usb_reset_device(sc->dev);
|
||||||
if (rc < 0) {
|
if (rc < 0) {
|
||||||
|
@ -1608,9 +1609,7 @@ static void ub_reset_task(struct work_struct *work)
|
||||||
"usb_lock_device_for_reset failed (%d)\n",
|
"usb_lock_device_for_reset failed (%d)\n",
|
||||||
sc->name, rc);
|
sc->name, rc);
|
||||||
}
|
}
|
||||||
|
usb_unlock_device(sc->dev);
|
||||||
if (lkr)
|
|
||||||
usb_unlock_device(sc->dev);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -102,7 +102,7 @@ static void hid_reset(struct work_struct *work)
|
||||||
struct usbhid_device *usbhid =
|
struct usbhid_device *usbhid =
|
||||||
container_of(work, struct usbhid_device, reset_work);
|
container_of(work, struct usbhid_device, reset_work);
|
||||||
struct hid_device *hid = usbhid->hid;
|
struct hid_device *hid = usbhid->hid;
|
||||||
int rc_lock, rc = 0;
|
int rc = 0;
|
||||||
|
|
||||||
if (test_bit(HID_CLEAR_HALT, &usbhid->iofl)) {
|
if (test_bit(HID_CLEAR_HALT, &usbhid->iofl)) {
|
||||||
dev_dbg(&usbhid->intf->dev, "clear halt\n");
|
dev_dbg(&usbhid->intf->dev, "clear halt\n");
|
||||||
|
@ -113,11 +113,10 @@ static void hid_reset(struct work_struct *work)
|
||||||
|
|
||||||
else if (test_bit(HID_RESET_PENDING, &usbhid->iofl)) {
|
else if (test_bit(HID_RESET_PENDING, &usbhid->iofl)) {
|
||||||
dev_dbg(&usbhid->intf->dev, "resetting device\n");
|
dev_dbg(&usbhid->intf->dev, "resetting device\n");
|
||||||
rc = rc_lock = usb_lock_device_for_reset(hid_to_usb_dev(hid), usbhid->intf);
|
rc = usb_lock_device_for_reset(hid_to_usb_dev(hid), usbhid->intf);
|
||||||
if (rc_lock >= 0) {
|
if (rc == 0) {
|
||||||
rc = usb_reset_device(hid_to_usb_dev(hid));
|
rc = usb_reset_device(hid_to_usb_dev(hid));
|
||||||
if (rc_lock)
|
usb_unlock_device(hid_to_usb_dev(hid));
|
||||||
usb_unlock_device(hid_to_usb_dev(hid));
|
|
||||||
}
|
}
|
||||||
clear_bit(HID_RESET_PENDING, &usbhid->iofl);
|
clear_bit(HID_RESET_PENDING, &usbhid->iofl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3655,7 +3655,7 @@ void pvr2_hdw_device_reset(struct pvr2_hdw *hdw)
|
||||||
int ret;
|
int ret;
|
||||||
pvr2_trace(PVR2_TRACE_INIT,"Performing a device reset...");
|
pvr2_trace(PVR2_TRACE_INIT,"Performing a device reset...");
|
||||||
ret = usb_lock_device_for_reset(hdw->usb_dev,NULL);
|
ret = usb_lock_device_for_reset(hdw->usb_dev,NULL);
|
||||||
if (ret == 1) {
|
if (ret == 0) {
|
||||||
ret = usb_reset_device(hdw->usb_dev);
|
ret = usb_reset_device(hdw->usb_dev);
|
||||||
usb_unlock_device(hdw->usb_dev);
|
usb_unlock_device(hdw->usb_dev);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -513,10 +513,7 @@ EXPORT_SYMBOL_GPL(usb_put_intf);
|
||||||
* disconnect; in some drivers (such as usb-storage) the disconnect()
|
* disconnect; in some drivers (such as usb-storage) the disconnect()
|
||||||
* or suspend() method will block waiting for a device reset to complete.
|
* or suspend() method will block waiting for a device reset to complete.
|
||||||
*
|
*
|
||||||
* Returns a negative error code for failure, otherwise 1 or 0 to indicate
|
* Returns a negative error code for failure, otherwise 0.
|
||||||
* that the device will or will not have to be unlocked. (0 can be
|
|
||||||
* returned when an interface is given and is BINDING, because in that
|
|
||||||
* case the driver already owns the device lock.)
|
|
||||||
*/
|
*/
|
||||||
int usb_lock_device_for_reset(struct usb_device *udev,
|
int usb_lock_device_for_reset(struct usb_device *udev,
|
||||||
const struct usb_interface *iface)
|
const struct usb_interface *iface)
|
||||||
|
@ -527,16 +524,9 @@ int usb_lock_device_for_reset(struct usb_device *udev,
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
if (udev->state == USB_STATE_SUSPENDED)
|
if (udev->state == USB_STATE_SUSPENDED)
|
||||||
return -EHOSTUNREACH;
|
return -EHOSTUNREACH;
|
||||||
if (iface) {
|
if (iface && (iface->condition == USB_INTERFACE_UNBINDING ||
|
||||||
switch (iface->condition) {
|
iface->condition == USB_INTERFACE_UNBOUND))
|
||||||
case USB_INTERFACE_BINDING:
|
return -EINTR;
|
||||||
return 0;
|
|
||||||
case USB_INTERFACE_BOUND:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return -EINTR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (usb_trylock_device(udev) != 0) {
|
while (usb_trylock_device(udev) != 0) {
|
||||||
|
|
||||||
|
@ -550,10 +540,11 @@ int usb_lock_device_for_reset(struct usb_device *udev,
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
if (udev->state == USB_STATE_SUSPENDED)
|
if (udev->state == USB_STATE_SUSPENDED)
|
||||||
return -EHOSTUNREACH;
|
return -EHOSTUNREACH;
|
||||||
if (iface && iface->condition != USB_INTERFACE_BOUND)
|
if (iface && (iface->condition == USB_INTERFACE_UNBINDING ||
|
||||||
|
iface->condition == USB_INTERFACE_UNBOUND))
|
||||||
return -EINTR;
|
return -EINTR;
|
||||||
}
|
}
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(usb_lock_device_for_reset);
|
EXPORT_SYMBOL_GPL(usb_lock_device_for_reset);
|
||||||
|
|
||||||
|
|
|
@ -350,17 +350,16 @@ static int mts_scsi_abort(struct scsi_cmnd *srb)
|
||||||
static int mts_scsi_host_reset(struct scsi_cmnd *srb)
|
static int mts_scsi_host_reset(struct scsi_cmnd *srb)
|
||||||
{
|
{
|
||||||
struct mts_desc* desc = (struct mts_desc*)(srb->device->host->hostdata[0]);
|
struct mts_desc* desc = (struct mts_desc*)(srb->device->host->hostdata[0]);
|
||||||
int result, rc;
|
int result;
|
||||||
|
|
||||||
MTS_DEBUG_GOT_HERE();
|
MTS_DEBUG_GOT_HERE();
|
||||||
mts_debug_dump(desc);
|
mts_debug_dump(desc);
|
||||||
|
|
||||||
rc = usb_lock_device_for_reset(desc->usb_dev, desc->usb_intf);
|
result = usb_lock_device_for_reset(desc->usb_dev, desc->usb_intf);
|
||||||
if (rc < 0)
|
if (result == 0) {
|
||||||
return FAILED;
|
result = usb_reset_device(desc->usb_dev);
|
||||||
result = usb_reset_device(desc->usb_dev);
|
|
||||||
if (rc)
|
|
||||||
usb_unlock_device(desc->usb_dev);
|
usb_unlock_device(desc->usb_dev);
|
||||||
|
}
|
||||||
return result ? FAILED : SUCCESS;
|
return result ? FAILED : SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1173,10 +1173,9 @@ int usb_stor_Bulk_reset(struct us_data *us)
|
||||||
*/
|
*/
|
||||||
int usb_stor_port_reset(struct us_data *us)
|
int usb_stor_port_reset(struct us_data *us)
|
||||||
{
|
{
|
||||||
int result, rc_lock;
|
int result;
|
||||||
|
|
||||||
result = rc_lock =
|
result = usb_lock_device_for_reset(us->pusb_dev, us->pusb_intf);
|
||||||
usb_lock_device_for_reset(us->pusb_dev, us->pusb_intf);
|
|
||||||
if (result < 0)
|
if (result < 0)
|
||||||
US_DEBUGP("unable to lock device for reset: %d\n", result);
|
US_DEBUGP("unable to lock device for reset: %d\n", result);
|
||||||
else {
|
else {
|
||||||
|
@ -1189,8 +1188,7 @@ int usb_stor_port_reset(struct us_data *us)
|
||||||
US_DEBUGP("usb_reset_device returns %d\n",
|
US_DEBUGP("usb_reset_device returns %d\n",
|
||||||
result);
|
result);
|
||||||
}
|
}
|
||||||
if (rc_lock)
|
usb_unlock_device(us->pusb_dev);
|
||||||
usb_unlock_device(us->pusb_dev);
|
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
Загрузка…
Ссылка в новой задаче