diff options
author | Ajay Kaher <ajay.kaher@samsung.com> | 2017-03-28 08:09:32 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2017-05-20 08:28:34 -0400 |
commit | 5ffe717f351ddf63161b6390b1ea90a5c3733994 (patch) | |
tree | acfde2241f73fe309735d10ab37fe3649178e709 | |
parent | c31ff3ceb32616145d16862c7e897a59dc919882 (diff) |
USB: Proper handling of Race Condition when two USB class drivers try to call init_usb_class simultaneously
commit 2f86a96be0ccb1302b7eee7855dbee5ce4dc5dfb upstream.
There is race condition when two USB class drivers try to call
init_usb_class at the same time and leads to crash.
code path: probe->usb_register_dev->init_usb_class
To solve this, mutex locking has been added in init_usb_class() and
destroy_usb_class().
As pointed by Alan, removed "if (usb_class)" test from destroy_usb_class()
because usb_class can never be NULL there.
Signed-off-by: Ajay Kaher <ajay.kaher@samsung.com>
Acked-by: Alan Stern <stern@rowland.harvard.edu>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r-- | drivers/usb/core/file.c | 9 |
1 files changed, 7 insertions, 2 deletions
diff --git a/drivers/usb/core/file.c b/drivers/usb/core/file.c index 822ced9639aa..422ce7b20d73 100644 --- a/drivers/usb/core/file.c +++ b/drivers/usb/core/file.c | |||
@@ -27,6 +27,7 @@ | |||
27 | #define MAX_USB_MINORS 256 | 27 | #define MAX_USB_MINORS 256 |
28 | static const struct file_operations *usb_minors[MAX_USB_MINORS]; | 28 | static const struct file_operations *usb_minors[MAX_USB_MINORS]; |
29 | static DECLARE_RWSEM(minor_rwsem); | 29 | static DECLARE_RWSEM(minor_rwsem); |
30 | static DEFINE_MUTEX(init_usb_class_mutex); | ||
30 | 31 | ||
31 | static int usb_open(struct inode *inode, struct file *file) | 32 | static int usb_open(struct inode *inode, struct file *file) |
32 | { | 33 | { |
@@ -109,8 +110,9 @@ static void release_usb_class(struct kref *kref) | |||
109 | 110 | ||
110 | static void destroy_usb_class(void) | 111 | static void destroy_usb_class(void) |
111 | { | 112 | { |
112 | if (usb_class) | 113 | mutex_lock(&init_usb_class_mutex); |
113 | kref_put(&usb_class->kref, release_usb_class); | 114 | kref_put(&usb_class->kref, release_usb_class); |
115 | mutex_unlock(&init_usb_class_mutex); | ||
114 | } | 116 | } |
115 | 117 | ||
116 | int usb_major_init(void) | 118 | int usb_major_init(void) |
@@ -171,7 +173,10 @@ int usb_register_dev(struct usb_interface *intf, | |||
171 | if (intf->minor >= 0) | 173 | if (intf->minor >= 0) |
172 | return -EADDRINUSE; | 174 | return -EADDRINUSE; |
173 | 175 | ||
176 | mutex_lock(&init_usb_class_mutex); | ||
174 | retval = init_usb_class(); | 177 | retval = init_usb_class(); |
178 | mutex_unlock(&init_usb_class_mutex); | ||
179 | |||
175 | if (retval) | 180 | if (retval) |
176 | return retval; | 181 | return retval; |
177 | 182 | ||