USB: fix bug in initialization of interface minor numbers
Recent changes in the usbhid layer exposed a bug in usbcore. If CONFIG_USB_DYNAMIC_MINORS is enabled then an interface may be assigned a minor number of 0. However interfaces that aren't registered as USB class devices also have their minor number set to 0, during initialization. As a result usb_find_interface() may return the wrong interface, leading to a crash. This patch (as1418) fixes the problem by initializing every interface's minor number to -1. It also cleans up the usb_register_dev() function, which besides being somewhat awkwardly written, does not unwind completely on all its error paths. Signed-off-by: Alan Stern <stern@rowland.harvard.edu> Tested-by: Philip J. Turmel <philip@turmel.org> Tested-by: Gabriel Craciunescu <nix.or.die@googlemail.com> Tested-by: Alex Riesen <raa.lkml@gmail.com> Tested-by: Matthias Bayer <jackdachef@gmail.com> CC: Jiri Kosina <jkosina@suse.cz> Cc: stable <stable@kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
a850ea3037
commit
0026e00523
2 changed files with 17 additions and 19 deletions
|
@ -159,9 +159,9 @@ void usb_major_cleanup(void)
|
||||||
int usb_register_dev(struct usb_interface *intf,
|
int usb_register_dev(struct usb_interface *intf,
|
||||||
struct usb_class_driver *class_driver)
|
struct usb_class_driver *class_driver)
|
||||||
{
|
{
|
||||||
int retval = -EINVAL;
|
int retval;
|
||||||
int minor_base = class_driver->minor_base;
|
int minor_base = class_driver->minor_base;
|
||||||
int minor = 0;
|
int minor;
|
||||||
char name[20];
|
char name[20];
|
||||||
char *temp;
|
char *temp;
|
||||||
|
|
||||||
|
@ -173,12 +173,17 @@ int usb_register_dev(struct usb_interface *intf,
|
||||||
*/
|
*/
|
||||||
minor_base = 0;
|
minor_base = 0;
|
||||||
#endif
|
#endif
|
||||||
intf->minor = -1;
|
|
||||||
|
|
||||||
dbg ("looking for a minor, starting at %d", minor_base);
|
|
||||||
|
|
||||||
if (class_driver->fops == NULL)
|
if (class_driver->fops == NULL)
|
||||||
goto exit;
|
return -EINVAL;
|
||||||
|
if (intf->minor >= 0)
|
||||||
|
return -EADDRINUSE;
|
||||||
|
|
||||||
|
retval = init_usb_class();
|
||||||
|
if (retval)
|
||||||
|
return retval;
|
||||||
|
|
||||||
|
dev_dbg(&intf->dev, "looking for a minor, starting at %d", minor_base);
|
||||||
|
|
||||||
down_write(&minor_rwsem);
|
down_write(&minor_rwsem);
|
||||||
for (minor = minor_base; minor < MAX_USB_MINORS; ++minor) {
|
for (minor = minor_base; minor < MAX_USB_MINORS; ++minor) {
|
||||||
|
@ -186,20 +191,12 @@ int usb_register_dev(struct usb_interface *intf,
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
usb_minors[minor] = class_driver->fops;
|
usb_minors[minor] = class_driver->fops;
|
||||||
|
intf->minor = minor;
|
||||||
retval = 0;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
up_write(&minor_rwsem);
|
up_write(&minor_rwsem);
|
||||||
|
if (intf->minor < 0)
|
||||||
if (retval)
|
return -EXFULL;
|
||||||
goto exit;
|
|
||||||
|
|
||||||
retval = init_usb_class();
|
|
||||||
if (retval)
|
|
||||||
goto exit;
|
|
||||||
|
|
||||||
intf->minor = minor;
|
|
||||||
|
|
||||||
/* create a usb class device for this usb interface */
|
/* create a usb class device for this usb interface */
|
||||||
snprintf(name, sizeof(name), class_driver->name, minor - minor_base);
|
snprintf(name, sizeof(name), class_driver->name, minor - minor_base);
|
||||||
|
@ -213,11 +210,11 @@ int usb_register_dev(struct usb_interface *intf,
|
||||||
"%s", temp);
|
"%s", temp);
|
||||||
if (IS_ERR(intf->usb_dev)) {
|
if (IS_ERR(intf->usb_dev)) {
|
||||||
down_write(&minor_rwsem);
|
down_write(&minor_rwsem);
|
||||||
usb_minors[intf->minor] = NULL;
|
usb_minors[minor] = NULL;
|
||||||
|
intf->minor = -1;
|
||||||
up_write(&minor_rwsem);
|
up_write(&minor_rwsem);
|
||||||
retval = PTR_ERR(intf->usb_dev);
|
retval = PTR_ERR(intf->usb_dev);
|
||||||
}
|
}
|
||||||
exit:
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(usb_register_dev);
|
EXPORT_SYMBOL_GPL(usb_register_dev);
|
||||||
|
|
|
@ -1802,6 +1802,7 @@ free_interfaces:
|
||||||
intf->dev.groups = usb_interface_groups;
|
intf->dev.groups = usb_interface_groups;
|
||||||
intf->dev.dma_mask = dev->dev.dma_mask;
|
intf->dev.dma_mask = dev->dev.dma_mask;
|
||||||
INIT_WORK(&intf->reset_ws, __usb_queue_reset_device);
|
INIT_WORK(&intf->reset_ws, __usb_queue_reset_device);
|
||||||
|
intf->minor = -1;
|
||||||
device_initialize(&intf->dev);
|
device_initialize(&intf->dev);
|
||||||
dev_set_name(&intf->dev, "%d-%s:%d.%d",
|
dev_set_name(&intf->dev, "%d-%s:%d.%d",
|
||||||
dev->bus->busnum, dev->devpath,
|
dev->bus->busnum, dev->devpath,
|
||||||
|
|
Loading…
Add table
Reference in a new issue