blob: ca425e8099ea816096d1962713a5c4eecfced49c [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*****************************************************************************/
2
3/*
4 * devio.c -- User space communication with USB devices.
5 *
6 * Copyright (C) 1999-2000 Thomas Sailer (sailer@ife.ee.ethz.ch)
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 *
Linus Torvalds1da177e2005-04-16 15:20:36 -070022 * This file implements the usbfs/x/y files, where
23 * x is the bus number and y the device number.
24 *
25 * It allows user space programs/"drivers" to communicate directly
26 * with USB devices without intervening kernel driver.
27 *
28 * Revision history
29 * 22.12.1999 0.1 Initial release (split from proc_usb.c)
30 * 04.01.2000 0.2 Turned into its own filesystem
Harald Welte46113832005-10-10 19:44:29 +020031 * 30.09.2005 0.3 Fix user-triggerable oops in async URB delivery
32 * (CAN-2005-3055)
Linus Torvalds1da177e2005-04-16 15:20:36 -070033 */
34
35/*****************************************************************************/
36
37#include <linux/fs.h>
38#include <linux/mm.h>
39#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070040#include <linux/signal.h>
41#include <linux/poll.h>
42#include <linux/module.h>
Chen Gangb11b2e12013-02-02 15:57:53 +080043#include <linux/string.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070044#include <linux/usb.h>
45#include <linux/usbdevice_fs.h>
Eric Lescouet27729aa2010-04-24 23:21:52 +020046#include <linux/usb/hcd.h> /* for usbcore internals */
Kay Sieversfbf82fd2005-07-31 01:05:53 +020047#include <linux/cdev.h>
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -070048#include <linux/notifier.h>
David Quigley7a019552006-06-30 01:55:48 -070049#include <linux/security.h>
Serge Hallynd178bc32011-09-26 10:45:18 -050050#include <linux/user_namespace.h>
Hans de Goede3d97ff62012-07-04 09:18:03 +020051#include <linux/scatterlist.h>
Tülin İzere6889b32013-05-17 10:12:34 +030052#include <linux/uaccess.h>
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +010053#include <linux/dma-mapping.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070054#include <asm/byteorder.h>
55#include <linux/moduleparam.h>
56
Linus Torvalds1da177e2005-04-16 15:20:36 -070057#include "usb.h"
58
Kay Sieversfbf82fd2005-07-31 01:05:53 +020059#define USB_MAXBUS 64
Tülin İzerfa86ad02013-05-17 10:13:24 +030060#define USB_DEVICE_MAX (USB_MAXBUS * 128)
Hans de Goede3d97ff62012-07-04 09:18:03 +020061#define USB_SG_SIZE 16384 /* split-size for large txs */
Kay Sieversfbf82fd2005-07-31 01:05:53 +020062
Alan Stern4a2a8a22006-07-01 22:05:01 -040063/* Mutual exclusion for removal, open, and release */
64DEFINE_MUTEX(usbfs_mutex);
65
Valentina Manea9b6f0c42014-03-10 10:36:40 +020066struct usb_dev_state {
Alan Sterncd9f0372008-06-24 14:47:04 -040067 struct list_head list; /* state list */
68 struct usb_device *dev;
69 struct file *file;
70 spinlock_t lock; /* protects the async urb lists */
71 struct list_head async_pending;
72 struct list_head async_completed;
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +010073 struct list_head memory_list;
Alan Sterncd9f0372008-06-24 14:47:04 -040074 wait_queue_head_t wait; /* wake up if a request completed */
75 unsigned int discsignr;
76 struct pid *disc_pid;
Serge Hallynd178bc32011-09-26 10:45:18 -050077 const struct cred *cred;
Alan Sterncd9f0372008-06-24 14:47:04 -040078 void __user *disccontext;
79 unsigned long ifclaimed;
80 u32 secid;
Alan Stern01c64602009-09-01 11:09:56 -040081 u32 disabled_bulk_eps;
Reilly Grantd883f522016-02-21 18:38:01 -030082 bool privileges_dropped;
83 unsigned long interface_allowed_mask;
Alan Sterncd9f0372008-06-24 14:47:04 -040084};
85
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +010086struct usb_memory {
87 struct list_head memlist;
88 int vma_use_count;
89 int urb_use_count;
90 u32 size;
91 void *mem;
92 dma_addr_t dma_handle;
93 unsigned long vm_start;
94 struct usb_dev_state *ps;
95};
96
Linus Torvalds1da177e2005-04-16 15:20:36 -070097struct async {
98 struct list_head asynclist;
Valentina Manea9b6f0c42014-03-10 10:36:40 +020099 struct usb_dev_state *ps;
Eric W. Biederman2425c082006-10-02 02:17:28 -0700100 struct pid *pid;
Serge Hallynd178bc32011-09-26 10:45:18 -0500101 const struct cred *cred;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700102 unsigned int signr;
103 unsigned int ifnum;
104 void __user *userbuffer;
105 void __user *userurb;
106 struct urb *urb;
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100107 struct usb_memory *usbm;
Alan Sternadd1aae2011-11-17 16:41:25 -0500108 unsigned int mem_usage;
Alan Sterne0152682007-08-24 15:42:52 -0400109 int status;
David Quigley7a019552006-06-30 01:55:48 -0700110 u32 secid;
Alan Stern01c64602009-09-01 11:09:56 -0400111 u8 bulk_addr;
112 u8 bulk_status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700113};
114
Rusty Russell90ab5ee2012-01-13 09:32:20 +1030115static bool usbfs_snoop;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800116module_param(usbfs_snoop, bool, S_IRUGO | S_IWUSR);
117MODULE_PARM_DESC(usbfs_snoop, "true to log all usbfs traffic");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118
Alan Stern0290cc92015-11-20 13:53:22 -0500119static unsigned usbfs_snoop_max = 65536;
120module_param(usbfs_snoop_max, uint, S_IRUGO | S_IWUSR);
121MODULE_PARM_DESC(usbfs_snoop_max,
122 "maximum number of bytes to print while snooping");
123
Linus Torvalds1da177e2005-04-16 15:20:36 -0700124#define snoop(dev, format, arg...) \
125 do { \
126 if (usbfs_snoop) \
Kris Borerf355e832015-08-07 07:22:44 -0400127 dev_info(dev, format, ## arg); \
Linus Torvalds1da177e2005-04-16 15:20:36 -0700128 } while (0)
129
Alan Stern4c6e8972009-06-29 11:02:04 -0400130enum snoop_when {
131 SUBMIT, COMPLETE
132};
133
Alan Sternfad21bd2005-08-10 15:15:57 -0400134#define USB_DEVICE_DEV MKDEV(USB_DEVICE_MAJOR, 0)
135
Alan Sternadd1aae2011-11-17 16:41:25 -0500136/* Limit on the total amount of memory we can allocate for transfers */
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800137static u32 usbfs_memory_mb = 16;
Alan Stern3f5eb8d2011-11-17 16:41:35 -0500138module_param(usbfs_memory_mb, uint, 0644);
139MODULE_PARM_DESC(usbfs_memory_mb,
140 "maximum MB allowed for usbfs buffers (0 = no limit)");
141
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800142static atomic64_t usbfs_memory_usage; /* Total memory currently allocated */
Alan Sternadd1aae2011-11-17 16:41:25 -0500143
144/* Check whether it's okay to allocate more memory for a transfer */
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800145static int usbfs_increase_memory_usage(u64 amount)
Alan Sternadd1aae2011-11-17 16:41:25 -0500146{
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800147 u64 lim;
Alan Stern3f5eb8d2011-11-17 16:41:35 -0500148
Alan Stern3f5eb8d2011-11-17 16:41:35 -0500149 lim = ACCESS_ONCE(usbfs_memory_mb);
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800150 lim <<= 20;
Alan Stern3f5eb8d2011-11-17 16:41:35 -0500151
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800152 atomic64_add(amount, &usbfs_memory_usage);
153
154 if (lim > 0 && atomic64_read(&usbfs_memory_usage) > lim) {
155 atomic64_sub(amount, &usbfs_memory_usage);
156 return -ENOMEM;
157 }
158
159 return 0;
Alan Sternadd1aae2011-11-17 16:41:25 -0500160}
161
162/* Memory for a transfer is being deallocated */
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800163static void usbfs_decrease_memory_usage(u64 amount)
Alan Sternadd1aae2011-11-17 16:41:25 -0500164{
Mateusz Berezecki1129d272016-12-21 09:19:14 -0800165 atomic64_sub(amount, &usbfs_memory_usage);
Alan Sternadd1aae2011-11-17 16:41:25 -0500166}
Alan Stern4c6e8972009-06-29 11:02:04 -0400167
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200168static int connected(struct usb_dev_state *ps)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700169{
Alan Stern349710c2006-07-01 22:05:56 -0400170 return (!list_empty(&ps->list) &&
171 ps->dev->state != USB_STATE_NOTATTACHED);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172}
173
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100174static void dec_usb_memory_use_count(struct usb_memory *usbm, int *count)
175{
176 struct usb_dev_state *ps = usbm->ps;
177 unsigned long flags;
178
179 spin_lock_irqsave(&ps->lock, flags);
180 --*count;
181 if (usbm->urb_use_count == 0 && usbm->vma_use_count == 0) {
182 list_del(&usbm->memlist);
183 spin_unlock_irqrestore(&ps->lock, flags);
184
185 usb_free_coherent(ps->dev, usbm->size, usbm->mem,
186 usbm->dma_handle);
187 usbfs_decrease_memory_usage(
188 usbm->size + sizeof(struct usb_memory));
189 kfree(usbm);
190 } else {
191 spin_unlock_irqrestore(&ps->lock, flags);
192 }
193}
194
195static void usbdev_vm_open(struct vm_area_struct *vma)
196{
197 struct usb_memory *usbm = vma->vm_private_data;
198 unsigned long flags;
199
200 spin_lock_irqsave(&usbm->ps->lock, flags);
201 ++usbm->vma_use_count;
202 spin_unlock_irqrestore(&usbm->ps->lock, flags);
203}
204
205static void usbdev_vm_close(struct vm_area_struct *vma)
206{
207 struct usb_memory *usbm = vma->vm_private_data;
208
209 dec_usb_memory_use_count(usbm, &usbm->vma_use_count);
210}
211
Michele Curti10871c12016-04-27 21:23:07 +0200212static struct vm_operations_struct usbdev_vm_ops = {
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100213 .open = usbdev_vm_open,
214 .close = usbdev_vm_close
215};
216
217static int usbdev_mmap(struct file *file, struct vm_area_struct *vma)
218{
219 struct usb_memory *usbm = NULL;
220 struct usb_dev_state *ps = file->private_data;
221 size_t size = vma->vm_end - vma->vm_start;
222 void *mem;
223 unsigned long flags;
224 dma_addr_t dma_handle;
225 int ret;
226
227 ret = usbfs_increase_memory_usage(size + sizeof(struct usb_memory));
228 if (ret)
229 goto error;
230
231 usbm = kzalloc(sizeof(struct usb_memory), GFP_KERNEL);
232 if (!usbm) {
233 ret = -ENOMEM;
234 goto error_decrease_mem;
235 }
236
Jiri Slaby70f7ca92016-06-15 15:56:11 +0200237 mem = usb_alloc_coherent(ps->dev, size, GFP_USER | __GFP_NOWARN,
238 &dma_handle);
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100239 if (!mem) {
240 ret = -ENOMEM;
241 goto error_free_usbm;
242 }
243
244 memset(mem, 0, size);
245
246 usbm->mem = mem;
247 usbm->dma_handle = dma_handle;
248 usbm->size = size;
249 usbm->ps = ps;
250 usbm->vm_start = vma->vm_start;
251 usbm->vma_use_count = 1;
252 INIT_LIST_HEAD(&usbm->memlist);
253
254 if (remap_pfn_range(vma, vma->vm_start,
255 virt_to_phys(usbm->mem) >> PAGE_SHIFT,
256 size, vma->vm_page_prot) < 0) {
257 dec_usb_memory_use_count(usbm, &usbm->vma_use_count);
258 return -EAGAIN;
259 }
260
261 vma->vm_flags |= VM_IO;
262 vma->vm_flags |= (VM_DONTEXPAND | VM_DONTDUMP);
263 vma->vm_ops = &usbdev_vm_ops;
264 vma->vm_private_data = usbm;
265
266 spin_lock_irqsave(&ps->lock, flags);
267 list_add_tail(&usbm->memlist, &ps->memory_list);
268 spin_unlock_irqrestore(&ps->lock, flags);
269
270 return 0;
271
272error_free_usbm:
273 kfree(usbm);
274error_decrease_mem:
275 usbfs_decrease_memory_usage(size + sizeof(struct usb_memory));
276error:
277 return ret;
278}
279
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800280static ssize_t usbdev_read(struct file *file, char __user *buf, size_t nbytes,
281 loff_t *ppos)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282{
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200283 struct usb_dev_state *ps = file->private_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700284 struct usb_device *dev = ps->dev;
285 ssize_t ret = 0;
286 unsigned len;
287 loff_t pos;
288 int i;
289
290 pos = *ppos;
291 usb_lock_device(dev);
Alan Stern349710c2006-07-01 22:05:56 -0400292 if (!connected(ps)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700293 ret = -ENODEV;
294 goto err;
295 } else if (pos < 0) {
296 ret = -EINVAL;
297 goto err;
298 }
299
300 if (pos < sizeof(struct usb_device_descriptor)) {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800301 /* 18 bytes - fits on the stack */
302 struct usb_device_descriptor temp_desc;
Oliver Neukum8781ba02006-01-06 21:24:25 +0100303
304 memcpy(&temp_desc, &dev->descriptor, sizeof(dev->descriptor));
Andrew Morton9fcd5c32006-01-18 23:55:07 -0800305 le16_to_cpus(&temp_desc.bcdUSB);
306 le16_to_cpus(&temp_desc.idVendor);
307 le16_to_cpus(&temp_desc.idProduct);
308 le16_to_cpus(&temp_desc.bcdDevice);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700309
310 len = sizeof(struct usb_device_descriptor) - pos;
311 if (len > nbytes)
312 len = nbytes;
Oliver Neukum8781ba02006-01-06 21:24:25 +0100313 if (copy_to_user(buf, ((char *)&temp_desc) + pos, len)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700314 ret = -EFAULT;
315 goto err;
316 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700317
318 *ppos += len;
319 buf += len;
320 nbytes -= len;
321 ret += len;
322 }
323
324 pos = sizeof(struct usb_device_descriptor);
325 for (i = 0; nbytes && i < dev->descriptor.bNumConfigurations; i++) {
326 struct usb_config_descriptor *config =
327 (struct usb_config_descriptor *)dev->rawdescriptors[i];
328 unsigned int length = le16_to_cpu(config->wTotalLength);
329
330 if (*ppos < pos + length) {
331
332 /* The descriptor may claim to be longer than it
333 * really is. Here is the actual allocated length. */
334 unsigned alloclen =
335 le16_to_cpu(dev->config[i].desc.wTotalLength);
336
337 len = length - (*ppos - pos);
338 if (len > nbytes)
339 len = nbytes;
340
341 /* Simply don't write (skip over) unallocated parts */
342 if (alloclen > (*ppos - pos)) {
343 alloclen -= (*ppos - pos);
344 if (copy_to_user(buf,
345 dev->rawdescriptors[i] + (*ppos - pos),
346 min(len, alloclen))) {
347 ret = -EFAULT;
348 goto err;
349 }
350 }
351
352 *ppos += len;
353 buf += len;
354 nbytes -= len;
355 ret += len;
356 }
357
358 pos += length;
359 }
360
361err:
362 usb_unlock_device(dev);
363 return ret;
364}
365
366/*
367 * async list handling
368 */
369
370static struct async *alloc_async(unsigned int numisoframes)
371{
Pete Zaitcevdd95b812008-01-05 02:01:07 -0800372 struct async *as;
Alan Stern0a1ef3b2005-10-24 15:38:24 -0400373
Pete Zaitcevdd95b812008-01-05 02:01:07 -0800374 as = kzalloc(sizeof(struct async), GFP_KERNEL);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800375 if (!as)
376 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377 as->urb = usb_alloc_urb(numisoframes, GFP_KERNEL);
378 if (!as->urb) {
379 kfree(as);
380 return NULL;
381 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800382 return as;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383}
384
385static void free_async(struct async *as)
386{
Hans de Goede3d97ff62012-07-04 09:18:03 +0200387 int i;
388
Eric W. Biederman2425c082006-10-02 02:17:28 -0700389 put_pid(as->pid);
Sarah Sharp1b41c832011-12-16 11:26:30 -0800390 if (as->cred)
391 put_cred(as->cred);
Hans de Goede3d97ff62012-07-04 09:18:03 +0200392 for (i = 0; i < as->urb->num_sgs; i++) {
393 if (sg_page(&as->urb->sg[i]))
394 kfree(sg_virt(&as->urb->sg[i]));
395 }
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100396
Hans de Goede3d97ff62012-07-04 09:18:03 +0200397 kfree(as->urb->sg);
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +0100398 if (as->usbm == NULL)
399 kfree(as->urb->transfer_buffer);
400 else
401 dec_usb_memory_use_count(as->usbm, &as->usbm->urb_use_count);
402
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -0700403 kfree(as->urb->setup_packet);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404 usb_free_urb(as->urb);
Alan Sternadd1aae2011-11-17 16:41:25 -0500405 usbfs_decrease_memory_usage(as->mem_usage);
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -0700406 kfree(as);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407}
408
Alan Sternd34d9722009-03-09 13:44:48 -0400409static void async_newpending(struct async *as)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700410{
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200411 struct usb_dev_state *ps = as->ps;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800412 unsigned long flags;
413
414 spin_lock_irqsave(&ps->lock, flags);
415 list_add_tail(&as->asynclist, &ps->async_pending);
416 spin_unlock_irqrestore(&ps->lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700417}
418
Alan Sternd34d9722009-03-09 13:44:48 -0400419static void async_removepending(struct async *as)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700420{
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200421 struct usb_dev_state *ps = as->ps;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800422 unsigned long flags;
423
424 spin_lock_irqsave(&ps->lock, flags);
425 list_del_init(&as->asynclist);
426 spin_unlock_irqrestore(&ps->lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700427}
428
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200429static struct async *async_getcompleted(struct usb_dev_state *ps)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700430{
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800431 unsigned long flags;
432 struct async *as = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700433
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800434 spin_lock_irqsave(&ps->lock, flags);
435 if (!list_empty(&ps->async_completed)) {
436 as = list_entry(ps->async_completed.next, struct async,
437 asynclist);
438 list_del_init(&as->asynclist);
439 }
440 spin_unlock_irqrestore(&ps->lock, flags);
441 return as;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700442}
443
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200444static struct async *async_getpending(struct usb_dev_state *ps,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800445 void __user *userurb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446{
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800447 struct async *as;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700448
Linus Torvalds1da177e2005-04-16 15:20:36 -0700449 list_for_each_entry(as, &ps->async_pending, asynclist)
450 if (as->userurb == userurb) {
451 list_del_init(&as->asynclist);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700452 return as;
453 }
Huajun Li4e09dcf2012-05-18 20:12:51 +0800454
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800455 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700456}
457
Alan Stern4c6e8972009-06-29 11:02:04 -0400458static void snoop_urb(struct usb_device *udev,
459 void __user *userurb, int pipe, unsigned length,
Chris Frey0880aef2010-01-26 17:07:29 -0500460 int timeout_or_status, enum snoop_when when,
461 unsigned char *data, unsigned data_len)
Greg Kroah-Hartmane639dd32005-06-20 21:15:16 -0700462{
Alan Stern4c6e8972009-06-29 11:02:04 -0400463 static const char *types[] = {"isoc", "int", "ctrl", "bulk"};
464 static const char *dirs[] = {"out", "in"};
465 int ep;
466 const char *t, *d;
Greg Kroah-Hartmane639dd32005-06-20 21:15:16 -0700467
468 if (!usbfs_snoop)
469 return;
470
Alan Stern4c6e8972009-06-29 11:02:04 -0400471 ep = usb_pipeendpoint(pipe);
472 t = types[usb_pipetype(pipe)];
473 d = dirs[!!usb_pipein(pipe)];
474
475 if (userurb) { /* Async */
476 if (when == SUBMIT)
477 dev_info(&udev->dev, "userurb %p, ep%d %s-%s, "
478 "length %u\n",
479 userurb, ep, t, d, length);
480 else
481 dev_info(&udev->dev, "userurb %p, ep%d %s-%s, "
482 "actual_length %u status %d\n",
483 userurb, ep, t, d, length,
484 timeout_or_status);
485 } else {
486 if (when == SUBMIT)
487 dev_info(&udev->dev, "ep%d %s-%s, length %u, "
488 "timeout %d\n",
489 ep, t, d, length, timeout_or_status);
490 else
491 dev_info(&udev->dev, "ep%d %s-%s, actual_length %u, "
492 "status %d\n",
493 ep, t, d, length, timeout_or_status);
494 }
Chris Frey0880aef2010-01-26 17:07:29 -0500495
Alan Stern0290cc92015-11-20 13:53:22 -0500496 data_len = min(data_len, usbfs_snoop_max);
Chris Frey0880aef2010-01-26 17:07:29 -0500497 if (data && data_len > 0) {
498 print_hex_dump(KERN_DEBUG, "data: ", DUMP_PREFIX_NONE, 32, 1,
499 data, data_len, 1);
500 }
Greg Kroah-Hartmane639dd32005-06-20 21:15:16 -0700501}
502
Hans de Goede3d97ff62012-07-04 09:18:03 +0200503static void snoop_urb_data(struct urb *urb, unsigned len)
504{
505 int i, size;
506
Alan Stern0290cc92015-11-20 13:53:22 -0500507 len = min(len, usbfs_snoop_max);
508 if (!usbfs_snoop || len == 0)
Hans de Goede3d97ff62012-07-04 09:18:03 +0200509 return;
510
511 if (urb->num_sgs == 0) {
512 print_hex_dump(KERN_DEBUG, "data: ", DUMP_PREFIX_NONE, 32, 1,
513 urb->transfer_buffer, len, 1);
514 return;
515 }
516
517 for (i = 0; i < urb->num_sgs && len; i++) {
518 size = (len > USB_SG_SIZE) ? USB_SG_SIZE : len;
519 print_hex_dump(KERN_DEBUG, "data: ", DUMP_PREFIX_NONE, 32, 1,
520 sg_virt(&urb->sg[i]), size, 1);
521 len -= size;
522 }
523}
524
525static int copy_urb_data_to_user(u8 __user *userbuffer, struct urb *urb)
526{
527 unsigned i, len, size;
528
529 if (urb->number_of_packets > 0) /* Isochronous */
530 len = urb->transfer_buffer_length;
531 else /* Non-Isoc */
532 len = urb->actual_length;
533
534 if (urb->num_sgs == 0) {
535 if (copy_to_user(userbuffer, urb->transfer_buffer, len))
536 return -EFAULT;
537 return 0;
538 }
539
540 for (i = 0; i < urb->num_sgs && len; i++) {
541 size = (len > USB_SG_SIZE) ? USB_SG_SIZE : len;
542 if (copy_to_user(userbuffer, sg_virt(&urb->sg[i]), size))
543 return -EFAULT;
544 userbuffer += size;
545 len -= size;
546 }
547
548 return 0;
549}
550
Alan Stern01c64602009-09-01 11:09:56 -0400551#define AS_CONTINUATION 1
552#define AS_UNLINK 2
553
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200554static void cancel_bulk_urbs(struct usb_dev_state *ps, unsigned bulk_addr)
Alan Stern01c64602009-09-01 11:09:56 -0400555__releases(ps->lock)
556__acquires(ps->lock)
557{
Huajun Li4e09dcf2012-05-18 20:12:51 +0800558 struct urb *urb;
Alan Stern01c64602009-09-01 11:09:56 -0400559 struct async *as;
560
561 /* Mark all the pending URBs that match bulk_addr, up to but not
562 * including the first one without AS_CONTINUATION. If such an
563 * URB is encountered then a new transfer has already started so
564 * the endpoint doesn't need to be disabled; otherwise it does.
565 */
566 list_for_each_entry(as, &ps->async_pending, asynclist) {
567 if (as->bulk_addr == bulk_addr) {
568 if (as->bulk_status != AS_CONTINUATION)
569 goto rescan;
570 as->bulk_status = AS_UNLINK;
571 as->bulk_addr = 0;
572 }
573 }
574 ps->disabled_bulk_eps |= (1 << bulk_addr);
575
576 /* Now carefully unlink all the marked pending URBs */
577 rescan:
578 list_for_each_entry(as, &ps->async_pending, asynclist) {
579 if (as->bulk_status == AS_UNLINK) {
580 as->bulk_status = 0; /* Only once */
Huajun Li4e09dcf2012-05-18 20:12:51 +0800581 urb = as->urb;
582 usb_get_urb(urb);
Alan Stern01c64602009-09-01 11:09:56 -0400583 spin_unlock(&ps->lock); /* Allow completions */
Huajun Li4e09dcf2012-05-18 20:12:51 +0800584 usb_unlink_urb(urb);
585 usb_put_urb(urb);
Alan Stern01c64602009-09-01 11:09:56 -0400586 spin_lock(&ps->lock);
587 goto rescan;
588 }
589 }
590}
591
David Howells7d12e782006-10-05 14:55:46 +0100592static void async_completed(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593{
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800594 struct async *as = urb->context;
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200595 struct usb_dev_state *ps = as->ps;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700596 struct siginfo sinfo;
Oliver Neukum516a1a02009-07-08 19:09:23 +0200597 struct pid *pid = NULL;
Oliver Neukum516a1a02009-07-08 19:09:23 +0200598 u32 secid = 0;
Serge Hallynd178bc32011-09-26 10:45:18 -0500599 const struct cred *cred = NULL;
Oliver Neukum516a1a02009-07-08 19:09:23 +0200600 int signr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800602 spin_lock(&ps->lock);
603 list_move_tail(&as->asynclist, &ps->async_completed);
Alan Sterne0152682007-08-24 15:42:52 -0400604 as->status = urb->status;
Oliver Neukum516a1a02009-07-08 19:09:23 +0200605 signr = as->signr;
606 if (signr) {
Alan Sternf0c2b682015-02-13 10:54:53 -0500607 memset(&sinfo, 0, sizeof(sinfo));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700608 sinfo.si_signo = as->signr;
Alan Sterne0152682007-08-24 15:42:52 -0400609 sinfo.si_errno = as->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700610 sinfo.si_code = SI_ASYNCIO;
611 sinfo.si_addr = as->userurb;
Serge Hallynaec01c52011-09-26 10:18:29 -0500612 pid = get_pid(as->pid);
Serge Hallynd178bc32011-09-26 10:45:18 -0500613 cred = get_cred(as->cred);
Oliver Neukum516a1a02009-07-08 19:09:23 +0200614 secid = as->secid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700615 }
Greg Kroah-Hartmane639dd32005-06-20 21:15:16 -0700616 snoop(&urb->dev->dev, "urb complete\n");
Alan Stern4c6e8972009-06-29 11:02:04 -0400617 snoop_urb(urb->dev, as->userurb, urb->pipe, urb->actual_length,
Hans de Goede3d97ff62012-07-04 09:18:03 +0200618 as->status, COMPLETE, NULL, 0);
Dan Carpenter83ed07c2015-05-18 15:29:51 +0300619 if ((urb->transfer_flags & URB_DIR_MASK) == URB_DIR_IN)
Hans de Goede3d97ff62012-07-04 09:18:03 +0200620 snoop_urb_data(urb, urb->actual_length);
621
Alan Stern01c64602009-09-01 11:09:56 -0400622 if (as->status < 0 && as->bulk_addr && as->status != -ECONNRESET &&
623 as->status != -ENOENT)
624 cancel_bulk_urbs(ps, as->bulk_addr);
Oliver Neukum516a1a02009-07-08 19:09:23 +0200625 spin_unlock(&ps->lock);
626
Serge Hallynaec01c52011-09-26 10:18:29 -0500627 if (signr) {
Serge Hallynd178bc32011-09-26 10:45:18 -0500628 kill_pid_info_as_cred(sinfo.si_signo, &sinfo, pid, cred, secid);
Serge Hallynaec01c52011-09-26 10:18:29 -0500629 put_pid(pid);
Serge Hallynd178bc32011-09-26 10:45:18 -0500630 put_cred(cred);
Serge Hallynaec01c52011-09-26 10:18:29 -0500631 }
Oliver Neukum516a1a02009-07-08 19:09:23 +0200632
Greg Kroah-Hartmane639dd32005-06-20 21:15:16 -0700633 wake_up(&ps->wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700634}
635
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200636static void destroy_async(struct usb_dev_state *ps, struct list_head *list)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700637{
Huajun Li4e09dcf2012-05-18 20:12:51 +0800638 struct urb *urb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700639 struct async *as;
640 unsigned long flags;
641
642 spin_lock_irqsave(&ps->lock, flags);
643 while (!list_empty(list)) {
644 as = list_entry(list->next, struct async, asynclist);
645 list_del_init(&as->asynclist);
Huajun Li4e09dcf2012-05-18 20:12:51 +0800646 urb = as->urb;
647 usb_get_urb(urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700648
649 /* drop the spinlock so the completion handler can run */
650 spin_unlock_irqrestore(&ps->lock, flags);
Huajun Li4e09dcf2012-05-18 20:12:51 +0800651 usb_kill_urb(urb);
652 usb_put_urb(urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700653 spin_lock_irqsave(&ps->lock, flags);
654 }
655 spin_unlock_irqrestore(&ps->lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656}
657
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200658static void destroy_async_on_interface(struct usb_dev_state *ps,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800659 unsigned int ifnum)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700660{
661 struct list_head *p, *q, hitlist;
662 unsigned long flags;
663
664 INIT_LIST_HEAD(&hitlist);
665 spin_lock_irqsave(&ps->lock, flags);
666 list_for_each_safe(p, q, &ps->async_pending)
667 if (ifnum == list_entry(p, struct async, asynclist)->ifnum)
668 list_move_tail(p, &hitlist);
669 spin_unlock_irqrestore(&ps->lock, flags);
670 destroy_async(ps, &hitlist);
671}
672
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200673static void destroy_all_async(struct usb_dev_state *ps)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700674{
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800675 destroy_async(ps, &ps->async_pending);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700676}
677
678/*
679 * interface claims are made only at the request of user level code,
680 * which can also release them (explicitly or by closing files).
681 * they're also undone when devices disconnect.
682 */
683
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800684static int driver_probe(struct usb_interface *intf,
685 const struct usb_device_id *id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700686{
687 return -ENODEV;
688}
689
690static void driver_disconnect(struct usb_interface *intf)
691{
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200692 struct usb_dev_state *ps = usb_get_intfdata(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700693 unsigned int ifnum = intf->altsetting->desc.bInterfaceNumber;
694
695 if (!ps)
696 return;
697
698 /* NOTE: this relies on usbcore having canceled and completed
699 * all pending I/O requests; 2.6 does that.
700 */
701
702 if (likely(ifnum < 8*sizeof(ps->ifclaimed)))
703 clear_bit(ifnum, &ps->ifclaimed);
704 else
Greg Kroah-Hartman3b6004f2008-08-14 09:37:34 -0700705 dev_warn(&intf->dev, "interface number %u out of range\n",
706 ifnum);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700707
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800708 usb_set_intfdata(intf, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700709
710 /* force async requests to complete */
711 destroy_async_on_interface(ps, ifnum);
712}
713
Alan Stern2e2eb832007-12-04 14:35:15 -0500714/* The following routines are merely placeholders. There is no way
715 * to inform a user task about suspend or resumes.
716 */
717static int driver_suspend(struct usb_interface *intf, pm_message_t msg)
718{
719 return 0;
720}
721
722static int driver_resume(struct usb_interface *intf)
723{
724 return 0;
725}
726
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727struct usb_driver usbfs_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728 .name = "usbfs",
729 .probe = driver_probe,
730 .disconnect = driver_disconnect,
Alan Stern2e2eb832007-12-04 14:35:15 -0500731 .suspend = driver_suspend,
732 .resume = driver_resume,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700733};
734
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200735static int claimintf(struct usb_dev_state *ps, unsigned int ifnum)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736{
737 struct usb_device *dev = ps->dev;
738 struct usb_interface *intf;
739 int err;
740
741 if (ifnum >= 8*sizeof(ps->ifclaimed))
742 return -EINVAL;
743 /* already claimed */
744 if (test_bit(ifnum, &ps->ifclaimed))
745 return 0;
746
Reilly Grantd883f522016-02-21 18:38:01 -0300747 if (ps->privileges_dropped &&
748 !test_bit(ifnum, &ps->interface_allowed_mask))
749 return -EACCES;
750
Linus Torvalds1da177e2005-04-16 15:20:36 -0700751 intf = usb_ifnum_to_if(dev, ifnum);
752 if (!intf)
753 err = -ENOENT;
754 else
755 err = usb_driver_claim_interface(&usbfs_driver, intf, ps);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756 if (err == 0)
757 set_bit(ifnum, &ps->ifclaimed);
758 return err;
759}
760
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200761static int releaseintf(struct usb_dev_state *ps, unsigned int ifnum)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700762{
763 struct usb_device *dev;
764 struct usb_interface *intf;
765 int err;
766
767 err = -EINVAL;
768 if (ifnum >= 8*sizeof(ps->ifclaimed))
769 return err;
770 dev = ps->dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700771 intf = usb_ifnum_to_if(dev, ifnum);
772 if (!intf)
773 err = -ENOENT;
774 else if (test_and_clear_bit(ifnum, &ps->ifclaimed)) {
775 usb_driver_release_interface(&usbfs_driver, intf);
776 err = 0;
777 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700778 return err;
779}
780
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200781static int checkintf(struct usb_dev_state *ps, unsigned int ifnum)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700782{
783 if (ps->dev->state != USB_STATE_CONFIGURED)
784 return -EHOSTUNREACH;
785 if (ifnum >= 8*sizeof(ps->ifclaimed))
786 return -EINVAL;
787 if (test_bit(ifnum, &ps->ifclaimed))
788 return 0;
789 /* if not yet claimed, claim it for the driver */
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800790 dev_warn(&ps->dev->dev, "usbfs: process %d (%s) did not claim "
791 "interface %u before use\n", task_pid_nr(current),
792 current->comm, ifnum);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700793 return claimintf(ps, ifnum);
794}
795
796static int findintfep(struct usb_device *dev, unsigned int ep)
797{
798 unsigned int i, j, e;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800799 struct usb_interface *intf;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700800 struct usb_host_interface *alts;
801 struct usb_endpoint_descriptor *endpt;
802
803 if (ep & ~(USB_DIR_IN|0xf))
804 return -EINVAL;
805 if (!dev->actconfig)
806 return -ESRCH;
807 for (i = 0; i < dev->actconfig->desc.bNumInterfaces; i++) {
808 intf = dev->actconfig->interface[i];
809 for (j = 0; j < intf->num_altsetting; j++) {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800810 alts = &intf->altsetting[j];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700811 for (e = 0; e < alts->desc.bNumEndpoints; e++) {
812 endpt = &alts->endpoint[e].desc;
813 if (endpt->bEndpointAddress == ep)
814 return alts->desc.bInterfaceNumber;
815 }
816 }
817 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800818 return -ENOENT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819}
820
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200821static int check_ctrlrecip(struct usb_dev_state *ps, unsigned int requesttype,
Matthias Dellweg393cbb52011-09-25 14:26:25 +0200822 unsigned int request, unsigned int index)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700823{
824 int ret = 0;
Matthias Dellweg393cbb52011-09-25 14:26:25 +0200825 struct usb_host_interface *alt_setting;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700826
David Vrabel6da9c992009-02-18 14:43:47 +0000827 if (ps->dev->state != USB_STATE_UNAUTHENTICATED
828 && ps->dev->state != USB_STATE_ADDRESS
Horst Schirmeier24f8b112006-03-11 00:16:55 -0800829 && ps->dev->state != USB_STATE_CONFIGURED)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700830 return -EHOSTUNREACH;
831 if (USB_TYPE_VENDOR == (USB_TYPE_MASK & requesttype))
832 return 0;
833
Matthias Dellweg393cbb52011-09-25 14:26:25 +0200834 /*
835 * check for the special corner case 'get_device_id' in the printer
Hans de Goede5dc50c32013-07-12 10:04:18 +0200836 * class specification, which we always want to allow as it is used
837 * to query things like ink level, etc.
Matthias Dellweg393cbb52011-09-25 14:26:25 +0200838 */
839 if (requesttype == 0xa1 && request == 0) {
840 alt_setting = usb_find_alt_setting(ps->dev->actconfig,
841 index >> 8, index & 0xff);
842 if (alt_setting
843 && alt_setting->desc.bInterfaceClass == USB_CLASS_PRINTER)
Hans de Goede5dc50c32013-07-12 10:04:18 +0200844 return 0;
Matthias Dellweg393cbb52011-09-25 14:26:25 +0200845 }
846
Linus Torvalds1da177e2005-04-16 15:20:36 -0700847 index &= 0xff;
848 switch (requesttype & USB_RECIP_MASK) {
849 case USB_RECIP_ENDPOINT:
Hans de Goede1361bf4b2013-04-16 11:08:33 +0200850 if ((index & ~USB_DIR_IN) == 0)
851 return 0;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800852 ret = findintfep(ps->dev, index);
Kurt Garloff831abf72013-09-24 14:13:48 +0200853 if (ret < 0) {
854 /*
855 * Some not fully compliant Win apps seem to get
856 * index wrong and have the endpoint number here
857 * rather than the endpoint address (with the
858 * correct direction). Win does let this through,
859 * so we'll not reject it here but leave it to
860 * the device to not break KVM. But we warn.
861 */
862 ret = findintfep(ps->dev, index ^ 0x80);
863 if (ret >= 0)
864 dev_info(&ps->dev->dev,
865 "%s: process %i (%s) requesting ep %02x but needs %02x\n",
866 __func__, task_pid_nr(current),
867 current->comm, index, index ^ 0x80);
868 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800869 if (ret >= 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700870 ret = checkintf(ps, ret);
871 break;
872
873 case USB_RECIP_INTERFACE:
874 ret = checkintf(ps, index);
875 break;
876 }
877 return ret;
878}
879
Hans de Goede2fec32b2013-10-09 17:19:30 +0200880static struct usb_host_endpoint *ep_to_host_endpoint(struct usb_device *dev,
881 unsigned char ep)
882{
883 if (ep & USB_ENDPOINT_DIR_MASK)
884 return dev->ep_in[ep & USB_ENDPOINT_NUMBER_MASK];
885 else
886 return dev->ep_out[ep & USB_ENDPOINT_NUMBER_MASK];
887}
888
Linus Torvalds3e75c6d2014-04-01 17:06:09 -0700889static int parse_usbdevfs_streams(struct usb_dev_state *ps,
Hans de Goedebcf7f6e2013-10-09 17:19:31 +0200890 struct usbdevfs_streams __user *streams,
891 unsigned int *num_streams_ret,
892 unsigned int *num_eps_ret,
893 struct usb_host_endpoint ***eps_ret,
894 struct usb_interface **intf_ret)
895{
896 unsigned int i, num_streams, num_eps;
897 struct usb_host_endpoint **eps;
898 struct usb_interface *intf = NULL;
899 unsigned char ep;
900 int ifnum, ret;
901
902 if (get_user(num_streams, &streams->num_streams) ||
903 get_user(num_eps, &streams->num_eps))
904 return -EFAULT;
905
906 if (num_eps < 1 || num_eps > USB_MAXENDPOINTS)
907 return -EINVAL;
908
909 /* The XHCI controller allows max 2 ^ 16 streams */
910 if (num_streams_ret && (num_streams < 2 || num_streams > 65536))
911 return -EINVAL;
912
913 eps = kmalloc(num_eps * sizeof(*eps), GFP_KERNEL);
914 if (!eps)
915 return -ENOMEM;
916
917 for (i = 0; i < num_eps; i++) {
918 if (get_user(ep, &streams->eps[i])) {
919 ret = -EFAULT;
920 goto error;
921 }
922 eps[i] = ep_to_host_endpoint(ps->dev, ep);
923 if (!eps[i]) {
924 ret = -EINVAL;
925 goto error;
926 }
927
928 /* usb_alloc/free_streams operate on an usb_interface */
929 ifnum = findintfep(ps->dev, ep);
930 if (ifnum < 0) {
931 ret = ifnum;
932 goto error;
933 }
934
935 if (i == 0) {
936 ret = checkintf(ps, ifnum);
937 if (ret < 0)
938 goto error;
939 intf = usb_ifnum_to_if(ps->dev, ifnum);
940 } else {
941 /* Verify all eps belong to the same interface */
942 if (ifnum != intf->altsetting->desc.bInterfaceNumber) {
943 ret = -EINVAL;
944 goto error;
945 }
946 }
947 }
948
949 if (num_streams_ret)
950 *num_streams_ret = num_streams;
951 *num_eps_ret = num_eps;
952 *eps_ret = eps;
953 *intf_ret = intf;
954
955 return 0;
956
957error:
958 kfree(eps);
959 return ret;
960}
961
Alan Stern61ad04a2008-06-24 14:47:12 -0400962static int match_devt(struct device *dev, void *data)
Greg Kroah-Hartman4592bf52005-06-20 21:15:16 -0700963{
David Howellsa80d5ff2008-07-02 12:28:55 +0100964 return dev->devt == (dev_t) (unsigned long) data;
Kay Sievers9f8b17e2007-03-13 15:59:31 +0100965}
Greg Kroah-Hartman4592bf52005-06-20 21:15:16 -0700966
Alan Stern61ad04a2008-06-24 14:47:12 -0400967static struct usb_device *usbdev_lookup_by_devt(dev_t devt)
Kay Sievers9f8b17e2007-03-13 15:59:31 +0100968{
969 struct device *dev;
970
David Howellsa80d5ff2008-07-02 12:28:55 +0100971 dev = bus_find_device(&usb_bus_type, NULL,
972 (void *) (unsigned long) devt, match_devt);
Kay Sievers9f8b17e2007-03-13 15:59:31 +0100973 if (!dev)
974 return NULL;
Geliang Tang69ab55d2015-12-23 21:26:50 +0800975 return to_usb_device(dev);
Kay Sievers9f8b17e2007-03-13 15:59:31 +0100976}
Greg Kroah-Hartman4592bf52005-06-20 21:15:16 -0700977
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978/*
979 * file operations
980 */
981static int usbdev_open(struct inode *inode, struct file *file)
982{
Kay Sieversfbf82fd2005-07-31 01:05:53 +0200983 struct usb_device *dev = NULL;
Valentina Manea9b6f0c42014-03-10 10:36:40 +0200984 struct usb_dev_state *ps;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700985 int ret;
986
Linus Torvalds1da177e2005-04-16 15:20:36 -0700987 ret = -ENOMEM;
Reilly Grantd883f522016-02-21 18:38:01 -0300988 ps = kzalloc(sizeof(struct usb_dev_state), GFP_KERNEL);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -0800989 if (!ps)
Alan Stern62e299e2010-01-08 12:56:19 -0500990 goto out_free_ps;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700991
Alan Stern01105a22009-07-30 15:28:14 -0400992 ret = -ENODEV;
Alan Stern61ad04a2008-06-24 14:47:12 -0400993
Alan Stern62e299e2010-01-08 12:56:19 -0500994 /* Protect against simultaneous removal or release */
995 mutex_lock(&usbfs_mutex);
996
Kay Sievers9f8b17e2007-03-13 15:59:31 +0100997 /* usbdev device-node */
Kay Sieversfbf82fd2005-07-31 01:05:53 +0200998 if (imajor(inode) == USB_DEVICE_MAJOR)
Alan Stern61ad04a2008-06-24 14:47:12 -0400999 dev = usbdev_lookup_by_devt(inode->i_rdev);
Alan Stern62e299e2010-01-08 12:56:19 -05001000
Alan Stern62e299e2010-01-08 12:56:19 -05001001 mutex_unlock(&usbfs_mutex);
1002
1003 if (!dev)
1004 goto out_free_ps;
1005
1006 usb_lock_device(dev);
1007 if (dev->state == USB_STATE_NOTATTACHED)
1008 goto out_unlock_device;
1009
Alan Stern94fcda12006-11-20 11:38:46 -05001010 ret = usb_autoresume_device(dev);
Alan Stern01d883d2006-08-30 15:47:18 -04001011 if (ret)
Alan Stern62e299e2010-01-08 12:56:19 -05001012 goto out_unlock_device;
Alan Stern01d883d2006-08-30 15:47:18 -04001013
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014 ps->dev = dev;
1015 ps->file = file;
Reilly Grantd883f522016-02-21 18:38:01 -03001016 ps->interface_allowed_mask = 0xFFFFFFFF; /* 32 bits */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001017 spin_lock_init(&ps->lock);
Dan Carpenter316547f2006-12-13 00:03:38 -08001018 INIT_LIST_HEAD(&ps->list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001019 INIT_LIST_HEAD(&ps->async_pending);
1020 INIT_LIST_HEAD(&ps->async_completed);
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001021 INIT_LIST_HEAD(&ps->memory_list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001022 init_waitqueue_head(&ps->wait);
Eric W. Biederman2425c082006-10-02 02:17:28 -07001023 ps->disc_pid = get_pid(task_pid(current));
Serge Hallynd178bc32011-09-26 10:45:18 -05001024 ps->cred = get_current_cred();
David Quigley7a019552006-06-30 01:55:48 -07001025 security_task_getsecid(current, &ps->secid);
Oliver Neukum527660a2007-04-20 20:50:48 +02001026 smp_wmb();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001027 list_add_tail(&ps->list, &dev->filelist);
1028 file->private_data = ps;
Alan Stern62e299e2010-01-08 12:56:19 -05001029 usb_unlock_device(dev);
Alan Stern2da41d52008-10-06 11:24:26 -04001030 snoop(&dev->dev, "opened by process %d: %s\n", task_pid_nr(current),
1031 current->comm);
Alan Stern62e299e2010-01-08 12:56:19 -05001032 return ret;
1033
1034 out_unlock_device:
1035 usb_unlock_device(dev);
1036 usb_put_dev(dev);
1037 out_free_ps:
1038 kfree(ps);
Alan Stern4a2a8a22006-07-01 22:05:01 -04001039 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001040}
1041
1042static int usbdev_release(struct inode *inode, struct file *file)
1043{
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001044 struct usb_dev_state *ps = file->private_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001045 struct usb_device *dev = ps->dev;
1046 unsigned int ifnum;
Alan Stern6ff10462009-03-09 13:44:02 -04001047 struct async *as;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048
1049 usb_lock_device(dev);
Alan Stern7cbe5dc2009-06-29 10:56:54 -04001050 usb_hub_release_all_ports(dev, ps);
Alan Stern4a2a8a22006-07-01 22:05:01 -04001051
Linus Torvalds1da177e2005-04-16 15:20:36 -07001052 list_del_init(&ps->list);
Alan Stern4a2a8a22006-07-01 22:05:01 -04001053
Linus Torvalds1da177e2005-04-16 15:20:36 -07001054 for (ifnum = 0; ps->ifclaimed && ifnum < 8*sizeof(ps->ifclaimed);
1055 ifnum++) {
1056 if (test_bit(ifnum, &ps->ifclaimed))
1057 releaseintf(ps, ifnum);
1058 }
1059 destroy_all_async(ps);
Alan Stern94fcda12006-11-20 11:38:46 -05001060 usb_autosuspend_device(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001061 usb_unlock_device(dev);
1062 usb_put_dev(dev);
Eric W. Biederman2425c082006-10-02 02:17:28 -07001063 put_pid(ps->disc_pid);
Serge Hallynd178bc32011-09-26 10:45:18 -05001064 put_cred(ps->cred);
Alan Stern6ff10462009-03-09 13:44:02 -04001065
1066 as = async_getcompleted(ps);
1067 while (as) {
1068 free_async(as);
1069 as = async_getcompleted(ps);
1070 }
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001071
Linus Torvalds1da177e2005-04-16 15:20:36 -07001072 kfree(ps);
Alan Stern4a2a8a22006-07-01 22:05:01 -04001073 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001074}
1075
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001076static int proc_control(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001077{
1078 struct usb_device *dev = ps->dev;
1079 struct usbdevfs_ctrltransfer ctrl;
1080 unsigned int tmo;
1081 unsigned char *tbuf;
Andrew Mortonff66e3c2008-03-12 13:32:24 -07001082 unsigned wLength;
Alan Stern4c6e8972009-06-29 11:02:04 -04001083 int i, pipe, ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001084
1085 if (copy_from_user(&ctrl, arg, sizeof(ctrl)))
1086 return -EFAULT;
Matthias Dellweg393cbb52011-09-25 14:26:25 +02001087 ret = check_ctrlrecip(ps, ctrl.bRequestType, ctrl.bRequest,
1088 ctrl.wIndex);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001089 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001090 return ret;
Andrew Mortonff66e3c2008-03-12 13:32:24 -07001091 wLength = ctrl.wLength; /* To suppress 64k PAGE_SIZE warning */
1092 if (wLength > PAGE_SIZE)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001093 return -EINVAL;
Alan Sternadd1aae2011-11-17 16:41:25 -05001094 ret = usbfs_increase_memory_usage(PAGE_SIZE + sizeof(struct urb) +
1095 sizeof(struct usb_ctrlrequest));
1096 if (ret)
1097 return ret;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001098 tbuf = (unsigned char *)__get_free_page(GFP_KERNEL);
Alan Sternadd1aae2011-11-17 16:41:25 -05001099 if (!tbuf) {
1100 ret = -ENOMEM;
1101 goto done;
1102 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001103 tmo = ctrl.timeout;
Chris Frey0880aef2010-01-26 17:07:29 -05001104 snoop(&dev->dev, "control urb: bRequestType=%02x "
1105 "bRequest=%02x wValue=%04x "
1106 "wIndex=%04x wLength=%04x\n",
Xenia Ragiadakouc8f2efc2013-08-31 18:09:14 +03001107 ctrl.bRequestType, ctrl.bRequest, ctrl.wValue,
1108 ctrl.wIndex, ctrl.wLength);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001109 if (ctrl.bRequestType & 0x80) {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001110 if (ctrl.wLength && !access_ok(VERIFY_WRITE, ctrl.data,
1111 ctrl.wLength)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001112 ret = -EINVAL;
1113 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001114 }
Alan Stern4c6e8972009-06-29 11:02:04 -04001115 pipe = usb_rcvctrlpipe(dev, 0);
Chris Frey0880aef2010-01-26 17:07:29 -05001116 snoop_urb(dev, NULL, pipe, ctrl.wLength, tmo, SUBMIT, NULL, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001117
1118 usb_unlock_device(dev);
Alan Stern4c6e8972009-06-29 11:02:04 -04001119 i = usb_control_msg(dev, pipe, ctrl.bRequest,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001120 ctrl.bRequestType, ctrl.wValue, ctrl.wIndex,
1121 tbuf, ctrl.wLength, tmo);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001122 usb_lock_device(dev);
Chris Frey0880aef2010-01-26 17:07:29 -05001123 snoop_urb(dev, NULL, pipe, max(i, 0), min(i, 0), COMPLETE,
Michal Sojka9d02b422011-03-15 16:41:47 +01001124 tbuf, max(i, 0));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001125 if ((i > 0) && ctrl.wLength) {
Alan Sternfe0410c2005-07-29 12:16:58 -07001126 if (copy_to_user(ctrl.data, tbuf, i)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001127 ret = -EFAULT;
1128 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129 }
1130 }
1131 } else {
1132 if (ctrl.wLength) {
1133 if (copy_from_user(tbuf, ctrl.data, ctrl.wLength)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001134 ret = -EFAULT;
1135 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001136 }
1137 }
Alan Stern4c6e8972009-06-29 11:02:04 -04001138 pipe = usb_sndctrlpipe(dev, 0);
Chris Frey0880aef2010-01-26 17:07:29 -05001139 snoop_urb(dev, NULL, pipe, ctrl.wLength, tmo, SUBMIT,
1140 tbuf, ctrl.wLength);
Alan Stern4c6e8972009-06-29 11:02:04 -04001141
Linus Torvalds1da177e2005-04-16 15:20:36 -07001142 usb_unlock_device(dev);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001143 i = usb_control_msg(dev, usb_sndctrlpipe(dev, 0), ctrl.bRequest,
1144 ctrl.bRequestType, ctrl.wValue, ctrl.wIndex,
1145 tbuf, ctrl.wLength, tmo);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001146 usb_lock_device(dev);
Chris Frey0880aef2010-01-26 17:07:29 -05001147 snoop_urb(dev, NULL, pipe, max(i, 0), min(i, 0), COMPLETE, NULL, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001149 if (i < 0 && i != -EPIPE) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001150 dev_printk(KERN_DEBUG, &dev->dev, "usbfs: USBDEVFS_CONTROL "
1151 "failed cmd %s rqt %u rq %u len %u ret %d\n",
1152 current->comm, ctrl.bRequestType, ctrl.bRequest,
1153 ctrl.wLength, i);
1154 }
Alan Stern52fb7432011-11-17 16:41:14 -05001155 ret = i;
1156 done:
1157 free_page((unsigned long) tbuf);
Alan Sternadd1aae2011-11-17 16:41:25 -05001158 usbfs_decrease_memory_usage(PAGE_SIZE + sizeof(struct urb) +
1159 sizeof(struct usb_ctrlrequest));
Alan Stern52fb7432011-11-17 16:41:14 -05001160 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161}
1162
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001163static int proc_bulk(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001164{
1165 struct usb_device *dev = ps->dev;
1166 struct usbdevfs_bulktransfer bulk;
1167 unsigned int tmo, len1, pipe;
1168 int len2;
1169 unsigned char *tbuf;
Alan Stern4c6e8972009-06-29 11:02:04 -04001170 int i, ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001171
1172 if (copy_from_user(&bulk, arg, sizeof(bulk)))
1173 return -EFAULT;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001174 ret = findintfep(ps->dev, bulk.ep);
1175 if (ret < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001176 return ret;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001177 ret = checkintf(ps, ret);
1178 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001179 return ret;
1180 if (bulk.ep & USB_DIR_IN)
1181 pipe = usb_rcvbulkpipe(dev, bulk.ep & 0x7f);
1182 else
1183 pipe = usb_sndbulkpipe(dev, bulk.ep & 0x7f);
1184 if (!usb_maxpacket(dev, pipe, !(bulk.ep & USB_DIR_IN)))
1185 return -EINVAL;
1186 len1 = bulk.len;
Mateusz Berezecki1129d272016-12-21 09:19:14 -08001187 if (len1 >= (INT_MAX - sizeof(struct urb)))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001188 return -EINVAL;
Alan Sternadd1aae2011-11-17 16:41:25 -05001189 ret = usbfs_increase_memory_usage(len1 + sizeof(struct urb));
1190 if (ret)
1191 return ret;
Kris Borer135551e2015-08-04 08:39:31 -04001192 tbuf = kmalloc(len1, GFP_KERNEL);
1193 if (!tbuf) {
Alan Sternadd1aae2011-11-17 16:41:25 -05001194 ret = -ENOMEM;
1195 goto done;
1196 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197 tmo = bulk.timeout;
1198 if (bulk.ep & 0x80) {
1199 if (len1 && !access_ok(VERIFY_WRITE, bulk.data, len1)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001200 ret = -EINVAL;
1201 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001202 }
Chris Frey0880aef2010-01-26 17:07:29 -05001203 snoop_urb(dev, NULL, pipe, len1, tmo, SUBMIT, NULL, 0);
Alan Stern4c6e8972009-06-29 11:02:04 -04001204
Linus Torvalds1da177e2005-04-16 15:20:36 -07001205 usb_unlock_device(dev);
1206 i = usb_bulk_msg(dev, pipe, tbuf, len1, &len2, tmo);
1207 usb_lock_device(dev);
Chris Frey0880aef2010-01-26 17:07:29 -05001208 snoop_urb(dev, NULL, pipe, len2, i, COMPLETE, tbuf, len2);
Alan Stern4c6e8972009-06-29 11:02:04 -04001209
Linus Torvalds1da177e2005-04-16 15:20:36 -07001210 if (!i && len2) {
1211 if (copy_to_user(bulk.data, tbuf, len2)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001212 ret = -EFAULT;
1213 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001214 }
1215 }
1216 } else {
1217 if (len1) {
1218 if (copy_from_user(tbuf, bulk.data, len1)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001219 ret = -EFAULT;
1220 goto done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001221 }
1222 }
Chris Frey0880aef2010-01-26 17:07:29 -05001223 snoop_urb(dev, NULL, pipe, len1, tmo, SUBMIT, tbuf, len1);
Alan Stern4c6e8972009-06-29 11:02:04 -04001224
Linus Torvalds1da177e2005-04-16 15:20:36 -07001225 usb_unlock_device(dev);
1226 i = usb_bulk_msg(dev, pipe, tbuf, len1, &len2, tmo);
1227 usb_lock_device(dev);
Chris Frey0880aef2010-01-26 17:07:29 -05001228 snoop_urb(dev, NULL, pipe, len2, i, COMPLETE, NULL, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001229 }
Alan Stern52fb7432011-11-17 16:41:14 -05001230 ret = (i < 0 ? i : len2);
1231 done:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001232 kfree(tbuf);
Alan Sternadd1aae2011-11-17 16:41:25 -05001233 usbfs_decrease_memory_usage(len1 + sizeof(struct urb));
Alan Stern52fb7432011-11-17 16:41:14 -05001234 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001235}
1236
Alan Sternf080a512014-02-20 10:49:30 -05001237static void check_reset_of_active_ep(struct usb_device *udev,
1238 unsigned int epnum, char *ioctl_name)
1239{
1240 struct usb_host_endpoint **eps;
1241 struct usb_host_endpoint *ep;
1242
1243 eps = (epnum & USB_DIR_IN) ? udev->ep_in : udev->ep_out;
1244 ep = eps[epnum & 0x0f];
1245 if (ep && !list_empty(&ep->urb_list))
1246 dev_warn(&udev->dev, "Process %d (%s) called USBDEVFS_%s for active endpoint 0x%02x\n",
1247 task_pid_nr(current), current->comm,
1248 ioctl_name, epnum);
1249}
1250
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001251static int proc_resetep(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001252{
1253 unsigned int ep;
1254 int ret;
1255
1256 if (get_user(ep, (unsigned int __user *)arg))
1257 return -EFAULT;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001258 ret = findintfep(ps->dev, ep);
1259 if (ret < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001260 return ret;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001261 ret = checkintf(ps, ret);
1262 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001263 return ret;
Alan Sternf080a512014-02-20 10:49:30 -05001264 check_reset_of_active_ep(ps->dev, ep, "RESETEP");
David Vrabel3444b262009-04-08 17:36:28 +00001265 usb_reset_endpoint(ps->dev, ep);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001266 return 0;
1267}
1268
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001269static int proc_clearhalt(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001270{
1271 unsigned int ep;
1272 int pipe;
1273 int ret;
1274
1275 if (get_user(ep, (unsigned int __user *)arg))
1276 return -EFAULT;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001277 ret = findintfep(ps->dev, ep);
1278 if (ret < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001279 return ret;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001280 ret = checkintf(ps, ret);
1281 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001282 return ret;
Alan Sternf080a512014-02-20 10:49:30 -05001283 check_reset_of_active_ep(ps->dev, ep, "CLEAR_HALT");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001284 if (ep & USB_DIR_IN)
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001285 pipe = usb_rcvbulkpipe(ps->dev, ep & 0x7f);
1286 else
1287 pipe = usb_sndbulkpipe(ps->dev, ep & 0x7f);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001288
1289 return usb_clear_halt(ps->dev, pipe);
1290}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001291
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001292static int proc_getdriver(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001293{
1294 struct usbdevfs_getdriver gd;
1295 struct usb_interface *intf;
1296 int ret;
1297
1298 if (copy_from_user(&gd, arg, sizeof(gd)))
1299 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001300 intf = usb_ifnum_to_if(ps->dev, gd.interface);
1301 if (!intf || !intf->dev.driver)
1302 ret = -ENODATA;
1303 else {
Chen Gangb11b2e12013-02-02 15:57:53 +08001304 strlcpy(gd.driver, intf->dev.driver->name,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001305 sizeof(gd.driver));
1306 ret = (copy_to_user(arg, &gd, sizeof(gd)) ? -EFAULT : 0);
1307 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001308 return ret;
1309}
1310
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001311static int proc_connectinfo(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001312{
Kangjie Lu681fef82016-05-03 16:32:16 -04001313 struct usbdevfs_connectinfo ci;
1314
1315 memset(&ci, 0, sizeof(ci));
1316 ci.devnum = ps->dev->devnum;
1317 ci.slow = ps->dev->speed == USB_SPEED_LOW;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001318
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319 if (copy_to_user(arg, &ci, sizeof(ci)))
1320 return -EFAULT;
1321 return 0;
1322}
1323
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001324static int proc_resetdevice(struct usb_dev_state *ps)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001325{
Reilly Grantd883f522016-02-21 18:38:01 -03001326 struct usb_host_config *actconfig = ps->dev->actconfig;
1327 struct usb_interface *interface;
1328 int i, number;
1329
1330 /* Don't allow a device reset if the process has dropped the
1331 * privilege to do such things and any of the interfaces are
1332 * currently claimed.
1333 */
1334 if (ps->privileges_dropped && actconfig) {
1335 for (i = 0; i < actconfig->desc.bNumInterfaces; ++i) {
1336 interface = actconfig->interface[i];
1337 number = interface->cur_altsetting->desc.bInterfaceNumber;
1338 if (usb_interface_claimed(interface) &&
1339 !test_bit(number, &ps->ifclaimed)) {
1340 dev_warn(&ps->dev->dev,
1341 "usbfs: interface %d claimed by %s while '%s' resets device\n",
1342 number, interface->dev.driver->name, current->comm);
1343 return -EACCES;
1344 }
1345 }
1346 }
1347
Ming Lei742120c2008-06-18 22:00:29 +08001348 return usb_reset_device(ps->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001349}
1350
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001351static int proc_setintf(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001352{
1353 struct usbdevfs_setinterface setintf;
1354 int ret;
1355
1356 if (copy_from_user(&setintf, arg, sizeof(setintf)))
1357 return -EFAULT;
Kris Borer135551e2015-08-04 08:39:31 -04001358 ret = checkintf(ps, setintf.interface);
1359 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001360 return ret;
Hans de Goede5ec9c172013-10-09 17:19:27 +02001361
1362 destroy_async_on_interface(ps, setintf.interface);
1363
Linus Torvalds1da177e2005-04-16 15:20:36 -07001364 return usb_set_interface(ps->dev, setintf.interface,
1365 setintf.altsetting);
1366}
1367
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001368static int proc_setconfig(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001369{
Alan Stern3f141e22007-02-08 16:40:43 -05001370 int u;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001371 int status = 0;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001372 struct usb_host_config *actconfig;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001373
Alan Stern3f141e22007-02-08 16:40:43 -05001374 if (get_user(u, (int __user *)arg))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001375 return -EFAULT;
1376
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001377 actconfig = ps->dev->actconfig;
1378
1379 /* Don't touch the device if any interfaces are claimed.
1380 * It could interfere with other drivers' operations, and if
Linus Torvalds1da177e2005-04-16 15:20:36 -07001381 * an interface is claimed by usbfs it could easily deadlock.
1382 */
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001383 if (actconfig) {
1384 int i;
1385
1386 for (i = 0; i < actconfig->desc.bNumInterfaces; ++i) {
1387 if (usb_interface_claimed(actconfig->interface[i])) {
1388 dev_warn(&ps->dev->dev,
David Brownell72ebddb2005-04-11 18:34:17 -07001389 "usbfs: interface %d claimed by %s "
Linus Torvalds1da177e2005-04-16 15:20:36 -07001390 "while '%s' sets config #%d\n",
1391 actconfig->interface[i]
1392 ->cur_altsetting
1393 ->desc.bInterfaceNumber,
David Brownell72ebddb2005-04-11 18:34:17 -07001394 actconfig->interface[i]
1395 ->dev.driver->name,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001396 current->comm, u);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001397 status = -EBUSY;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001398 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001399 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001400 }
1401 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001402
1403 /* SET_CONFIGURATION is often abused as a "cheap" driver reset,
1404 * so avoid usb_set_configuration()'s kick to sysfs
1405 */
1406 if (status == 0) {
1407 if (actconfig && actconfig->desc.bConfigurationValue == u)
1408 status = usb_reset_configuration(ps->dev);
1409 else
1410 status = usb_set_configuration(ps->dev, u);
1411 }
1412
1413 return status;
1414}
1415
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001416static struct usb_memory *
1417find_memory_area(struct usb_dev_state *ps, const struct usbdevfs_urb *uurb)
1418{
1419 struct usb_memory *usbm = NULL, *iter;
1420 unsigned long flags;
1421 unsigned long uurb_start = (unsigned long)uurb->buffer;
1422
1423 spin_lock_irqsave(&ps->lock, flags);
1424 list_for_each_entry(iter, &ps->memory_list, memlist) {
1425 if (uurb_start >= iter->vm_start &&
1426 uurb_start < iter->vm_start + iter->size) {
1427 if (uurb->buffer_length > iter->vm_start + iter->size -
1428 uurb_start) {
1429 usbm = ERR_PTR(-EINVAL);
1430 } else {
1431 usbm = iter;
1432 usbm->urb_use_count++;
1433 }
1434 break;
1435 }
1436 }
1437 spin_unlock_irqrestore(&ps->lock, flags);
1438 return usbm;
1439}
1440
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001441static int proc_do_submiturb(struct usb_dev_state *ps, struct usbdevfs_urb *uurb,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001442 struct usbdevfs_iso_packet_desc __user *iso_frame_desc,
1443 void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001444{
1445 struct usbdevfs_iso_packet_desc *isopkt = NULL;
1446 struct usb_host_endpoint *ep;
Alan Stern52fb7432011-11-17 16:41:14 -05001447 struct async *as = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001448 struct usb_ctrlrequest *dr = NULL;
1449 unsigned int u, totlen, isofrmlen;
Hans de Goede3d97ff62012-07-04 09:18:03 +02001450 int i, ret, is_in, num_sgs = 0, ifnum = -1;
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001451 int number_of_packets = 0;
Hans de Goede948cd8c2013-10-09 17:19:29 +02001452 unsigned int stream_id = 0;
Hans de Goede3d97ff62012-07-04 09:18:03 +02001453 void *buf;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001454
Alan Stern14722ef2008-04-17 10:18:11 -04001455 if (uurb->flags & ~(USBDEVFS_URB_ISO_ASAP |
1456 USBDEVFS_URB_SHORT_NOT_OK |
Alan Stern01c64602009-09-01 11:09:56 -04001457 USBDEVFS_URB_BULK_CONTINUATION |
Alan Stern14722ef2008-04-17 10:18:11 -04001458 USBDEVFS_URB_NO_FSBR |
1459 USBDEVFS_URB_ZERO_PACKET |
1460 USBDEVFS_URB_NO_INTERRUPT))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001461 return -EINVAL;
Alan Stern91801352009-06-29 11:04:54 -04001462 if (uurb->buffer_length > 0 && !uurb->buffer)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001463 return -EINVAL;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001464 if (!(uurb->type == USBDEVFS_URB_TYPE_CONTROL &&
1465 (uurb->endpoint & ~USB_ENDPOINT_DIR_MASK) == 0)) {
1466 ifnum = findintfep(ps->dev, uurb->endpoint);
1467 if (ifnum < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001468 return ifnum;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001469 ret = checkintf(ps, ifnum);
1470 if (ret)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001471 return ret;
1472 }
Hans de Goede2fec32b2013-10-09 17:19:30 +02001473 ep = ep_to_host_endpoint(ps->dev, uurb->endpoint);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001474 if (!ep)
1475 return -ENOENT;
Hans de Goede2fec32b2013-10-09 17:19:30 +02001476 is_in = (uurb->endpoint & USB_ENDPOINT_DIR_MASK) != 0;
Alan Sternadd1aae2011-11-17 16:41:25 -05001477
1478 u = 0;
Kris Borerf355e832015-08-07 07:22:44 -04001479 switch (uurb->type) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001480 case USBDEVFS_URB_TYPE_CONTROL:
Alan Stern93cf9b92007-07-30 17:09:28 -04001481 if (!usb_endpoint_xfer_control(&ep->desc))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001482 return -EINVAL;
Alan Sternadd1aae2011-11-17 16:41:25 -05001483 /* min 8 byte setup packet */
1484 if (uurb->buffer_length < 8)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001485 return -EINVAL;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001486 dr = kmalloc(sizeof(struct usb_ctrlrequest), GFP_KERNEL);
1487 if (!dr)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001488 return -ENOMEM;
1489 if (copy_from_user(dr, uurb->buffer, 8)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001490 ret = -EFAULT;
1491 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001492 }
1493 if (uurb->buffer_length < (le16_to_cpup(&dr->wLength) + 8)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001494 ret = -EINVAL;
1495 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001496 }
Matthias Dellweg393cbb52011-09-25 14:26:25 +02001497 ret = check_ctrlrecip(ps, dr->bRequestType, dr->bRequest,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001498 le16_to_cpup(&dr->wIndex));
Alan Stern52fb7432011-11-17 16:41:14 -05001499 if (ret)
1500 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001501 uurb->buffer_length = le16_to_cpup(&dr->wLength);
1502 uurb->buffer += 8;
Alan Stern93cf9b92007-07-30 17:09:28 -04001503 if ((dr->bRequestType & USB_DIR_IN) && uurb->buffer_length) {
1504 is_in = 1;
1505 uurb->endpoint |= USB_DIR_IN;
1506 } else {
1507 is_in = 0;
1508 uurb->endpoint &= ~USB_DIR_IN;
1509 }
Chris Frey0880aef2010-01-26 17:07:29 -05001510 snoop(&ps->dev->dev, "control urb: bRequestType=%02x "
1511 "bRequest=%02x wValue=%04x "
1512 "wIndex=%04x wLength=%04x\n",
1513 dr->bRequestType, dr->bRequest,
1514 __le16_to_cpup(&dr->wValue),
1515 __le16_to_cpup(&dr->wIndex),
1516 __le16_to_cpup(&dr->wLength));
Alan Sternadd1aae2011-11-17 16:41:25 -05001517 u = sizeof(struct usb_ctrlrequest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001518 break;
1519
1520 case USBDEVFS_URB_TYPE_BULK:
Alan Stern93cf9b92007-07-30 17:09:28 -04001521 switch (usb_endpoint_type(&ep->desc)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001522 case USB_ENDPOINT_XFER_CONTROL:
1523 case USB_ENDPOINT_XFER_ISOC:
1524 return -EINVAL;
Alan Sternf661c6f2009-12-11 16:20:20 -05001525 case USB_ENDPOINT_XFER_INT:
1526 /* allow single-shot interrupt transfers */
1527 uurb->type = USBDEVFS_URB_TYPE_INTERRUPT;
1528 goto interrupt_urb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001529 }
Hans de Goede3d97ff62012-07-04 09:18:03 +02001530 num_sgs = DIV_ROUND_UP(uurb->buffer_length, USB_SG_SIZE);
1531 if (num_sgs == 1 || num_sgs > ps->dev->bus->sg_tablesize)
1532 num_sgs = 0;
Hans de Goede948cd8c2013-10-09 17:19:29 +02001533 if (ep->streams)
1534 stream_id = uurb->stream_id;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001535 break;
1536
Alan Sternf661c6f2009-12-11 16:20:20 -05001537 case USBDEVFS_URB_TYPE_INTERRUPT:
1538 if (!usb_endpoint_xfer_int(&ep->desc))
1539 return -EINVAL;
1540 interrupt_urb:
Alan Sternf661c6f2009-12-11 16:20:20 -05001541 break;
1542
Linus Torvalds1da177e2005-04-16 15:20:36 -07001543 case USBDEVFS_URB_TYPE_ISO:
1544 /* arbitrary limit */
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001545 if (uurb->number_of_packets < 1 ||
1546 uurb->number_of_packets > 128)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001547 return -EINVAL;
Alan Stern93cf9b92007-07-30 17:09:28 -04001548 if (!usb_endpoint_xfer_isoc(&ep->desc))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001549 return -EINVAL;
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001550 number_of_packets = uurb->number_of_packets;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001551 isofrmlen = sizeof(struct usbdevfs_iso_packet_desc) *
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001552 number_of_packets;
Rahul Pathak73a02d32015-12-11 05:40:51 +00001553 isopkt = memdup_user(iso_frame_desc, isofrmlen);
1554 if (IS_ERR(isopkt)) {
1555 ret = PTR_ERR(isopkt);
1556 isopkt = NULL;
Alan Stern52fb7432011-11-17 16:41:14 -05001557 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001558 }
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001559 for (totlen = u = 0; u < number_of_packets; u++) {
Federico Manzane2e2f0e2013-05-24 18:18:48 +02001560 /*
1561 * arbitrary limit need for USB 3.0
1562 * bMaxBurst (0~15 allowed, 1~16 packets)
1563 * bmAttributes (bit 1:0, mult 0~2, 1~3 packets)
1564 * sizemax: 1024 * 16 * 3 = 49152
1565 */
1566 if (isopkt[u].length > 49152) {
Alan Stern52fb7432011-11-17 16:41:14 -05001567 ret = -EINVAL;
1568 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001569 }
1570 totlen += isopkt[u].length;
1571 }
Alan Sternadd1aae2011-11-17 16:41:25 -05001572 u *= sizeof(struct usb_iso_packet_descriptor);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001573 uurb->buffer_length = totlen;
1574 break;
1575
Linus Torvalds1da177e2005-04-16 15:20:36 -07001576 default:
1577 return -EINVAL;
1578 }
Alan Sternadd1aae2011-11-17 16:41:25 -05001579
Alan Stern91801352009-06-29 11:04:54 -04001580 if (uurb->buffer_length > 0 &&
1581 !access_ok(is_in ? VERIFY_WRITE : VERIFY_READ,
1582 uurb->buffer, uurb->buffer_length)) {
Alan Stern52fb7432011-11-17 16:41:14 -05001583 ret = -EFAULT;
1584 goto error;
Alan Stern91801352009-06-29 11:04:54 -04001585 }
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001586 as = alloc_async(number_of_packets);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001587 if (!as) {
Alan Stern52fb7432011-11-17 16:41:14 -05001588 ret = -ENOMEM;
1589 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001590 }
Hans de Goede3d97ff62012-07-04 09:18:03 +02001591
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001592 as->usbm = find_memory_area(ps, uurb);
1593 if (IS_ERR(as->usbm)) {
1594 ret = PTR_ERR(as->usbm);
1595 as->usbm = NULL;
1596 goto error;
1597 }
1598
1599 /* do not use SG buffers when memory mapped segments
1600 * are in use
1601 */
1602 if (as->usbm)
1603 num_sgs = 0;
1604
Hans de Goede3d97ff62012-07-04 09:18:03 +02001605 u += sizeof(struct async) + sizeof(struct urb) + uurb->buffer_length +
1606 num_sgs * sizeof(struct scatterlist);
Alan Sternadd1aae2011-11-17 16:41:25 -05001607 ret = usbfs_increase_memory_usage(u);
1608 if (ret)
1609 goto error;
1610 as->mem_usage = u;
1611
Hans de Goede3d97ff62012-07-04 09:18:03 +02001612 if (num_sgs) {
1613 as->urb->sg = kmalloc(num_sgs * sizeof(struct scatterlist),
1614 GFP_KERNEL);
1615 if (!as->urb->sg) {
1616 ret = -ENOMEM;
1617 goto error;
1618 }
1619 as->urb->num_sgs = num_sgs;
1620 sg_init_table(as->urb->sg, as->urb->num_sgs);
1621
1622 totlen = uurb->buffer_length;
1623 for (i = 0; i < as->urb->num_sgs; i++) {
1624 u = (totlen > USB_SG_SIZE) ? USB_SG_SIZE : totlen;
1625 buf = kmalloc(u, GFP_KERNEL);
1626 if (!buf) {
1627 ret = -ENOMEM;
1628 goto error;
1629 }
1630 sg_set_buf(&as->urb->sg[i], buf, u);
1631
1632 if (!is_in) {
1633 if (copy_from_user(buf, uurb->buffer, u)) {
1634 ret = -EFAULT;
1635 goto error;
1636 }
Henrik Rydberg01463902012-10-13 12:20:36 +02001637 uurb->buffer += u;
Hans de Goede3d97ff62012-07-04 09:18:03 +02001638 }
1639 totlen -= u;
1640 }
1641 } else if (uurb->buffer_length > 0) {
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001642 if (as->usbm) {
1643 unsigned long uurb_start = (unsigned long)uurb->buffer;
Hans de Goede3d97ff62012-07-04 09:18:03 +02001644
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001645 as->urb->transfer_buffer = as->usbm->mem +
1646 (uurb_start - as->usbm->vm_start);
1647 } else {
1648 as->urb->transfer_buffer = kmalloc(uurb->buffer_length,
1649 GFP_KERNEL);
1650 if (!as->urb->transfer_buffer) {
1651 ret = -ENOMEM;
Hans de Goede3d97ff62012-07-04 09:18:03 +02001652 goto error;
1653 }
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001654 if (!is_in) {
1655 if (copy_from_user(as->urb->transfer_buffer,
1656 uurb->buffer,
1657 uurb->buffer_length)) {
1658 ret = -EFAULT;
1659 goto error;
1660 }
1661 } else if (uurb->type == USBDEVFS_URB_TYPE_ISO) {
1662 /*
1663 * Isochronous input data may end up being
1664 * discontiguous if some of the packets are
1665 * short. Clear the buffer so that the gaps
1666 * don't leak kernel data to userspace.
1667 */
1668 memset(as->urb->transfer_buffer, 0,
1669 uurb->buffer_length);
1670 }
Hans de Goede3d97ff62012-07-04 09:18:03 +02001671 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001672 }
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001673 as->urb->dev = ps->dev;
1674 as->urb->pipe = (uurb->type << 30) |
Alan Stern93cf9b92007-07-30 17:09:28 -04001675 __create_pipe(ps->dev, uurb->endpoint & 0xf) |
1676 (uurb->endpoint & USB_DIR_IN);
Alan Stern14722ef2008-04-17 10:18:11 -04001677
1678 /* This tedious sequence is necessary because the URB_* flags
1679 * are internal to the kernel and subject to change, whereas
1680 * the USBDEVFS_URB_* flags are a user API and must not be changed.
1681 */
1682 u = (is_in ? URB_DIR_IN : URB_DIR_OUT);
1683 if (uurb->flags & USBDEVFS_URB_ISO_ASAP)
1684 u |= URB_ISO_ASAP;
Oliver Neukumd310d052014-08-01 09:55:20 +02001685 if (uurb->flags & USBDEVFS_URB_SHORT_NOT_OK && is_in)
Alan Stern14722ef2008-04-17 10:18:11 -04001686 u |= URB_SHORT_NOT_OK;
1687 if (uurb->flags & USBDEVFS_URB_NO_FSBR)
1688 u |= URB_NO_FSBR;
1689 if (uurb->flags & USBDEVFS_URB_ZERO_PACKET)
1690 u |= URB_ZERO_PACKET;
1691 if (uurb->flags & USBDEVFS_URB_NO_INTERRUPT)
1692 u |= URB_NO_INTERRUPT;
1693 as->urb->transfer_flags = u;
1694
Linus Torvalds1da177e2005-04-16 15:20:36 -07001695 as->urb->transfer_buffer_length = uurb->buffer_length;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001696 as->urb->setup_packet = (unsigned char *)dr;
Alan Stern52fb7432011-11-17 16:41:14 -05001697 dr = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001698 as->urb->start_frame = uurb->start_frame;
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001699 as->urb->number_of_packets = number_of_packets;
Hans de Goede948cd8c2013-10-09 17:19:29 +02001700 as->urb->stream_id = stream_id;
Alan Stern53e5f362016-08-23 15:32:51 -04001701
1702 if (ep->desc.bInterval) {
1703 if (uurb->type == USBDEVFS_URB_TYPE_ISO ||
1704 ps->dev->speed == USB_SPEED_HIGH ||
1705 ps->dev->speed >= USB_SPEED_SUPER)
1706 as->urb->interval = 1 <<
1707 min(15, ep->desc.bInterval - 1);
1708 else
1709 as->urb->interval = ep->desc.bInterval;
1710 }
1711
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001712 as->urb->context = as;
1713 as->urb->complete = async_completed;
Hans de Goedeb2d03eb2013-10-09 17:19:28 +02001714 for (totlen = u = 0; u < number_of_packets; u++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001715 as->urb->iso_frame_desc[u].offset = totlen;
1716 as->urb->iso_frame_desc[u].length = isopkt[u].length;
1717 totlen += isopkt[u].length;
1718 }
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -07001719 kfree(isopkt);
Alan Stern52fb7432011-11-17 16:41:14 -05001720 isopkt = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001721 as->ps = ps;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001722 as->userurb = arg;
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001723 if (as->usbm) {
1724 unsigned long uurb_start = (unsigned long)uurb->buffer;
1725
1726 as->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1727 as->urb->transfer_dma = as->usbm->dma_handle +
1728 (uurb_start - as->usbm->vm_start);
1729 } else if (is_in && uurb->buffer_length > 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001730 as->userbuffer = uurb->buffer;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001731 as->signr = uurb->signr;
1732 as->ifnum = ifnum;
Eric W. Biederman2425c082006-10-02 02:17:28 -07001733 as->pid = get_pid(task_pid(current));
Serge Hallynd178bc32011-09-26 10:45:18 -05001734 as->cred = get_current_cred();
David Quigley7a019552006-06-30 01:55:48 -07001735 security_task_getsecid(current, &as->secid);
Alan Stern4c6e8972009-06-29 11:02:04 -04001736 snoop_urb(ps->dev, as->userurb, as->urb->pipe,
Chris Frey0880aef2010-01-26 17:07:29 -05001737 as->urb->transfer_buffer_length, 0, SUBMIT,
Hans de Goede3d97ff62012-07-04 09:18:03 +02001738 NULL, 0);
1739 if (!is_in)
1740 snoop_urb_data(as->urb, as->urb->transfer_buffer_length);
1741
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001742 async_newpending(as);
Alan Stern01c64602009-09-01 11:09:56 -04001743
1744 if (usb_endpoint_xfer_bulk(&ep->desc)) {
1745 spin_lock_irq(&ps->lock);
1746
1747 /* Not exactly the endpoint address; the direction bit is
1748 * shifted to the 0x10 position so that the value will be
1749 * between 0 and 31.
1750 */
1751 as->bulk_addr = usb_endpoint_num(&ep->desc) |
1752 ((ep->desc.bEndpointAddress & USB_ENDPOINT_DIR_MASK)
1753 >> 3);
1754
1755 /* If this bulk URB is the start of a new transfer, re-enable
1756 * the endpoint. Otherwise mark it as a continuation URB.
1757 */
1758 if (uurb->flags & USBDEVFS_URB_BULK_CONTINUATION)
1759 as->bulk_status = AS_CONTINUATION;
1760 else
1761 ps->disabled_bulk_eps &= ~(1 << as->bulk_addr);
1762
1763 /* Don't accept continuation URBs if the endpoint is
1764 * disabled because of an earlier error.
1765 */
1766 if (ps->disabled_bulk_eps & (1 << as->bulk_addr))
1767 ret = -EREMOTEIO;
1768 else
1769 ret = usb_submit_urb(as->urb, GFP_ATOMIC);
1770 spin_unlock_irq(&ps->lock);
1771 } else {
1772 ret = usb_submit_urb(as->urb, GFP_KERNEL);
1773 }
1774
1775 if (ret) {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001776 dev_printk(KERN_DEBUG, &ps->dev->dev,
1777 "usbfs: usb_submit_urb returned %d\n", ret);
Alan Stern4c6e8972009-06-29 11:02:04 -04001778 snoop_urb(ps->dev, as->userurb, as->urb->pipe,
Chris Frey0880aef2010-01-26 17:07:29 -05001779 0, ret, COMPLETE, NULL, 0);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001780 async_removepending(as);
Alan Stern52fb7432011-11-17 16:41:14 -05001781 goto error;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001782 }
1783 return 0;
Alan Stern52fb7432011-11-17 16:41:14 -05001784
1785 error:
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01001786 if (as && as->usbm)
1787 dec_usb_memory_use_count(as->usbm, &as->usbm->urb_use_count);
Alan Stern52fb7432011-11-17 16:41:14 -05001788 kfree(isopkt);
1789 kfree(dr);
1790 if (as)
1791 free_async(as);
1792 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001793}
1794
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001795static int proc_submiturb(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001796{
1797 struct usbdevfs_urb uurb;
1798
1799 if (copy_from_user(&uurb, arg, sizeof(uurb)))
1800 return -EFAULT;
1801
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001802 return proc_do_submiturb(ps, &uurb,
1803 (((struct usbdevfs_urb __user *)arg)->iso_frame_desc),
1804 arg);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001805}
1806
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001807static int proc_unlinkurb(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001808{
Huajun Li4e09dcf2012-05-18 20:12:51 +08001809 struct urb *urb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001810 struct async *as;
Huajun Li4e09dcf2012-05-18 20:12:51 +08001811 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001812
Huajun Li4e09dcf2012-05-18 20:12:51 +08001813 spin_lock_irqsave(&ps->lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001814 as = async_getpending(ps, arg);
Huajun Li4e09dcf2012-05-18 20:12:51 +08001815 if (!as) {
1816 spin_unlock_irqrestore(&ps->lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001817 return -EINVAL;
Huajun Li4e09dcf2012-05-18 20:12:51 +08001818 }
1819
1820 urb = as->urb;
1821 usb_get_urb(urb);
1822 spin_unlock_irqrestore(&ps->lock, flags);
1823
1824 usb_kill_urb(urb);
1825 usb_put_urb(urb);
1826
Linus Torvalds1da177e2005-04-16 15:20:36 -07001827 return 0;
1828}
1829
1830static int processcompl(struct async *as, void __user * __user *arg)
1831{
1832 struct urb *urb = as->urb;
1833 struct usbdevfs_urb __user *userurb = as->userurb;
1834 void __user *addr = as->userurb;
1835 unsigned int i;
1836
Alan Stern7152b592010-03-06 15:04:03 -05001837 if (as->userbuffer && urb->actual_length) {
Hans de Goede3d97ff62012-07-04 09:18:03 +02001838 if (copy_urb_data_to_user(as->userbuffer, urb))
Oliver Neukumd794a022009-06-28 23:34:14 +02001839 goto err_out;
Alan Stern7152b592010-03-06 15:04:03 -05001840 }
Alan Sterne0152682007-08-24 15:42:52 -04001841 if (put_user(as->status, &userurb->status))
Oliver Neukumd794a022009-06-28 23:34:14 +02001842 goto err_out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001843 if (put_user(urb->actual_length, &userurb->actual_length))
Oliver Neukumd794a022009-06-28 23:34:14 +02001844 goto err_out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001845 if (put_user(urb->error_count, &userurb->error_count))
Oliver Neukumd794a022009-06-28 23:34:14 +02001846 goto err_out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001847
Alan Stern93cf9b92007-07-30 17:09:28 -04001848 if (usb_endpoint_xfer_isoc(&urb->ep->desc)) {
Christopher Li668a9542005-04-18 17:39:26 -07001849 for (i = 0; i < urb->number_of_packets; i++) {
1850 if (put_user(urb->iso_frame_desc[i].actual_length,
1851 &userurb->iso_frame_desc[i].actual_length))
Oliver Neukumd794a022009-06-28 23:34:14 +02001852 goto err_out;
Christopher Li668a9542005-04-18 17:39:26 -07001853 if (put_user(urb->iso_frame_desc[i].status,
1854 &userurb->iso_frame_desc[i].status))
Oliver Neukumd794a022009-06-28 23:34:14 +02001855 goto err_out;
Christopher Li668a9542005-04-18 17:39:26 -07001856 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001857 }
1858
Linus Torvalds1da177e2005-04-16 15:20:36 -07001859 if (put_user(addr, (void __user * __user *)arg))
1860 return -EFAULT;
1861 return 0;
Oliver Neukumd794a022009-06-28 23:34:14 +02001862
1863err_out:
Oliver Neukumd794a022009-06-28 23:34:14 +02001864 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001865}
1866
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001867static struct async *reap_as(struct usb_dev_state *ps)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001868{
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001869 DECLARE_WAITQUEUE(wait, current);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001870 struct async *as = NULL;
1871 struct usb_device *dev = ps->dev;
1872
1873 add_wait_queue(&ps->wait, &wait);
1874 for (;;) {
1875 __set_current_state(TASK_INTERRUPTIBLE);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001876 as = async_getcompleted(ps);
Alan Stern3f2cee72015-01-29 11:29:13 -05001877 if (as || !connected(ps))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001878 break;
1879 if (signal_pending(current))
1880 break;
1881 usb_unlock_device(dev);
1882 schedule();
1883 usb_lock_device(dev);
1884 }
1885 remove_wait_queue(&ps->wait, &wait);
1886 set_current_state(TASK_RUNNING);
1887 return as;
1888}
1889
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001890static int proc_reapurb(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001891{
1892 struct async *as = reap_as(ps);
Alan Sterna016a812015-11-20 13:53:35 -05001893
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001894 if (as) {
Alan Sterna016a812015-11-20 13:53:35 -05001895 int retval;
1896
1897 snoop(&ps->dev->dev, "reap %p\n", as->userurb);
1898 retval = processcompl(as, (void __user * __user *)arg);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001899 free_async(as);
1900 return retval;
1901 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001902 if (signal_pending(current))
1903 return -EINTR;
Alan Stern3f2cee72015-01-29 11:29:13 -05001904 return -ENODEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001905}
1906
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001907static int proc_reapurbnonblock(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001908{
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001909 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001910 struct async *as;
1911
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001912 as = async_getcompleted(ps);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001913 if (as) {
Alan Sterna016a812015-11-20 13:53:35 -05001914 snoop(&ps->dev->dev, "reap %p\n", as->userurb);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001915 retval = processcompl(as, (void __user * __user *)arg);
1916 free_async(as);
Alan Stern3f2cee72015-01-29 11:29:13 -05001917 } else {
1918 retval = (connected(ps) ? -EAGAIN : -ENODEV);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08001919 }
1920 return retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001921}
1922
1923#ifdef CONFIG_COMPAT
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001924static int proc_control_compat(struct usb_dev_state *ps,
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001925 struct usbdevfs_ctrltransfer32 __user *p32)
1926{
Matthias Beyer5b32c382013-10-14 21:46:36 +02001927 struct usbdevfs_ctrltransfer __user *p;
1928 __u32 udata;
1929 p = compat_alloc_user_space(sizeof(*p));
1930 if (copy_in_user(p, p32, (sizeof(*p32) - sizeof(compat_caddr_t))) ||
1931 get_user(udata, &p32->data) ||
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001932 put_user(compat_ptr(udata), &p->data))
1933 return -EFAULT;
Matthias Beyer5b32c382013-10-14 21:46:36 +02001934 return proc_control(ps, p);
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001935}
1936
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001937static int proc_bulk_compat(struct usb_dev_state *ps,
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001938 struct usbdevfs_bulktransfer32 __user *p32)
1939{
Matthias Beyer06793f22013-10-14 21:46:37 +02001940 struct usbdevfs_bulktransfer __user *p;
1941 compat_uint_t n;
1942 compat_caddr_t addr;
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001943
Matthias Beyer06793f22013-10-14 21:46:37 +02001944 p = compat_alloc_user_space(sizeof(*p));
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001945
Matthias Beyer06793f22013-10-14 21:46:37 +02001946 if (get_user(n, &p32->ep) || put_user(n, &p->ep) ||
1947 get_user(n, &p32->len) || put_user(n, &p->len) ||
1948 get_user(n, &p32->timeout) || put_user(n, &p->timeout) ||
1949 get_user(addr, &p32->data) || put_user(compat_ptr(addr), &p->data))
1950 return -EFAULT;
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001951
Matthias Beyer06793f22013-10-14 21:46:37 +02001952 return proc_bulk(ps, p);
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001953}
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001954static int proc_disconnectsignal_compat(struct usb_dev_state *ps, void __user *arg)
Arnd Bergmann637e8a62009-11-14 02:28:05 +01001955{
1956 struct usbdevfs_disconnectsignal32 ds;
1957
1958 if (copy_from_user(&ds, arg, sizeof(ds)))
1959 return -EFAULT;
1960 ps->discsignr = ds.signr;
1961 ps->disccontext = compat_ptr(ds.context);
1962 return 0;
1963}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001964
1965static int get_urb32(struct usbdevfs_urb *kurb,
1966 struct usbdevfs_urb32 __user *uurb)
1967{
1968 __u32 uptr;
Michael Buesch18753eb2009-07-29 11:39:03 +02001969 if (!access_ok(VERIFY_READ, uurb, sizeof(*uurb)) ||
1970 __get_user(kurb->type, &uurb->type) ||
Linus Torvalds1da177e2005-04-16 15:20:36 -07001971 __get_user(kurb->endpoint, &uurb->endpoint) ||
1972 __get_user(kurb->status, &uurb->status) ||
1973 __get_user(kurb->flags, &uurb->flags) ||
1974 __get_user(kurb->buffer_length, &uurb->buffer_length) ||
1975 __get_user(kurb->actual_length, &uurb->actual_length) ||
1976 __get_user(kurb->start_frame, &uurb->start_frame) ||
1977 __get_user(kurb->number_of_packets, &uurb->number_of_packets) ||
1978 __get_user(kurb->error_count, &uurb->error_count) ||
1979 __get_user(kurb->signr, &uurb->signr))
1980 return -EFAULT;
1981
1982 if (__get_user(uptr, &uurb->buffer))
1983 return -EFAULT;
1984 kurb->buffer = compat_ptr(uptr);
Mark Lorded0c7722009-01-02 02:48:24 -05001985 if (__get_user(uptr, &uurb->usercontext))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001986 return -EFAULT;
1987 kurb->usercontext = compat_ptr(uptr);
1988
1989 return 0;
1990}
1991
Valentina Manea9b6f0c42014-03-10 10:36:40 +02001992static int proc_submiturb_compat(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001993{
1994 struct usbdevfs_urb uurb;
1995
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001996 if (get_urb32(&uurb, (struct usbdevfs_urb32 __user *)arg))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001997 return -EFAULT;
1998
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08001999 return proc_do_submiturb(ps, &uurb,
2000 ((struct usbdevfs_urb32 __user *)arg)->iso_frame_desc,
2001 arg);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002002}
2003
2004static int processcompl_compat(struct async *as, void __user * __user *arg)
2005{
2006 struct urb *urb = as->urb;
2007 struct usbdevfs_urb32 __user *userurb = as->userurb;
2008 void __user *addr = as->userurb;
2009 unsigned int i;
2010
Hans de Goede2102e062012-07-04 09:18:01 +02002011 if (as->userbuffer && urb->actual_length) {
Hans de Goede3d97ff62012-07-04 09:18:03 +02002012 if (copy_urb_data_to_user(as->userbuffer, urb))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002013 return -EFAULT;
Hans de Goede2102e062012-07-04 09:18:01 +02002014 }
Alan Sterne0152682007-08-24 15:42:52 -04002015 if (put_user(as->status, &userurb->status))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002016 return -EFAULT;
2017 if (put_user(urb->actual_length, &userurb->actual_length))
2018 return -EFAULT;
2019 if (put_user(urb->error_count, &userurb->error_count))
2020 return -EFAULT;
2021
Alan Stern93cf9b92007-07-30 17:09:28 -04002022 if (usb_endpoint_xfer_isoc(&urb->ep->desc)) {
Christopher Li668a9542005-04-18 17:39:26 -07002023 for (i = 0; i < urb->number_of_packets; i++) {
2024 if (put_user(urb->iso_frame_desc[i].actual_length,
2025 &userurb->iso_frame_desc[i].actual_length))
2026 return -EFAULT;
2027 if (put_user(urb->iso_frame_desc[i].status,
2028 &userurb->iso_frame_desc[i].status))
2029 return -EFAULT;
2030 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002031 }
2032
Al Viroc714de52006-10-10 22:45:37 +01002033 if (put_user(ptr_to_compat(addr), (u32 __user *)arg))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002034 return -EFAULT;
2035 return 0;
2036}
2037
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002038static int proc_reapurb_compat(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002039{
2040 struct async *as = reap_as(ps);
Alan Sterna016a812015-11-20 13:53:35 -05002041
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002042 if (as) {
Alan Sterna016a812015-11-20 13:53:35 -05002043 int retval;
2044
2045 snoop(&ps->dev->dev, "reap %p\n", as->userurb);
2046 retval = processcompl_compat(as, (void __user * __user *)arg);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002047 free_async(as);
2048 return retval;
2049 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002050 if (signal_pending(current))
2051 return -EINTR;
Alan Stern3f2cee72015-01-29 11:29:13 -05002052 return -ENODEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002053}
2054
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002055static int proc_reapurbnonblock_compat(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002056{
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002057 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002058 struct async *as;
2059
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002060 as = async_getcompleted(ps);
2061 if (as) {
Alan Sterna016a812015-11-20 13:53:35 -05002062 snoop(&ps->dev->dev, "reap %p\n", as->userurb);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002063 retval = processcompl_compat(as, (void __user * __user *)arg);
2064 free_async(as);
Alan Stern3f2cee72015-01-29 11:29:13 -05002065 } else {
2066 retval = (connected(ps) ? -EAGAIN : -ENODEV);
Linus Torvaldsddeee0b2010-02-16 12:35:07 -08002067 }
2068 return retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002069}
2070
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002071
Linus Torvalds1da177e2005-04-16 15:20:36 -07002072#endif
2073
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002074static int proc_disconnectsignal(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002075{
2076 struct usbdevfs_disconnectsignal ds;
2077
2078 if (copy_from_user(&ds, arg, sizeof(ds)))
2079 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002080 ps->discsignr = ds.signr;
2081 ps->disccontext = ds.context;
2082 return 0;
2083}
2084
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002085static int proc_claiminterface(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002086{
2087 unsigned int ifnum;
2088
2089 if (get_user(ifnum, (unsigned int __user *)arg))
2090 return -EFAULT;
2091 return claimintf(ps, ifnum);
2092}
2093
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002094static int proc_releaseinterface(struct usb_dev_state *ps, void __user *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002095{
2096 unsigned int ifnum;
2097 int ret;
2098
2099 if (get_user(ifnum, (unsigned int __user *)arg))
2100 return -EFAULT;
Kris Borer135551e2015-08-04 08:39:31 -04002101 ret = releaseintf(ps, ifnum);
2102 if (ret < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002103 return ret;
Chase Metzger64f10ed2015-12-22 20:19:45 -08002104 destroy_async_on_interface(ps, ifnum);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002105 return 0;
2106}
2107
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002108static int proc_ioctl(struct usb_dev_state *ps, struct usbdevfs_ioctl *ctl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002109{
Linus Torvalds1da177e2005-04-16 15:20:36 -07002110 int size;
2111 void *buf = NULL;
2112 int retval = 0;
2113 struct usb_interface *intf = NULL;
2114 struct usb_driver *driver = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002115
Reilly Grantd883f522016-02-21 18:38:01 -03002116 if (ps->privileges_dropped)
2117 return -EACCES;
2118
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002119 /* alloc buffer */
Kris Borer135551e2015-08-04 08:39:31 -04002120 size = _IOC_SIZE(ctl->ioctl_code);
2121 if (size > 0) {
Tülin İzer4baf0df2013-05-17 10:13:58 +03002122 buf = kmalloc(size, GFP_KERNEL);
2123 if (buf == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002124 return -ENOMEM;
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002125 if ((_IOC_DIR(ctl->ioctl_code) & _IOC_WRITE)) {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002126 if (copy_from_user(buf, ctl->data, size)) {
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -07002127 kfree(buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002128 return -EFAULT;
2129 }
2130 } else {
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002131 memset(buf, 0, size);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002132 }
2133 }
2134
Alan Stern349710c2006-07-01 22:05:56 -04002135 if (!connected(ps)) {
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -07002136 kfree(buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002137 return -ENODEV;
2138 }
2139
2140 if (ps->dev->state != USB_STATE_CONFIGURED)
2141 retval = -EHOSTUNREACH;
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002142 else if (!(intf = usb_ifnum_to_if(ps->dev, ctl->ifno)))
2143 retval = -EINVAL;
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002144 else switch (ctl->ioctl_code) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002145
2146 /* disconnect kernel driver from interface */
2147 case USBDEVFS_DISCONNECT:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002148 if (intf->dev.driver) {
2149 driver = to_usb_driver(intf->dev.driver);
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002150 dev_dbg(&intf->dev, "disconnect by usbfs\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002151 usb_driver_release_interface(driver, intf);
2152 } else
2153 retval = -ENODATA;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002154 break;
2155
2156 /* let kernel drivers try to (re)bind to the interface */
2157 case USBDEVFS_CONNECT:
Alan Stern885e9742007-12-03 15:42:10 -05002158 if (!intf->dev.driver)
2159 retval = device_attach(&intf->dev);
2160 else
2161 retval = -EBUSY;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002162 break;
2163
2164 /* talk directly to the interface's driver */
2165 default:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002166 if (intf->dev.driver)
2167 driver = to_usb_driver(intf->dev.driver);
Andi Kleenc532b292010-06-01 23:04:41 +02002168 if (driver == NULL || driver->unlocked_ioctl == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002169 retval = -ENOTTY;
2170 } else {
Andi Kleenc532b292010-06-01 23:04:41 +02002171 retval = driver->unlocked_ioctl(intf, ctl->ioctl_code, buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002172 if (retval == -ENOIOCTLCMD)
2173 retval = -ENOTTY;
2174 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002175 }
2176
2177 /* cleanup and return */
2178 if (retval >= 0
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002179 && (_IOC_DIR(ctl->ioctl_code) & _IOC_READ) != 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002180 && size > 0
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002181 && copy_to_user(ctl->data, buf, size) != 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002182 retval = -EFAULT;
Jesper Juhl6fd19f4b2005-04-18 17:39:33 -07002183
2184 kfree(buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002185 return retval;
2186}
2187
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002188static int proc_ioctl_default(struct usb_dev_state *ps, void __user *arg)
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002189{
2190 struct usbdevfs_ioctl ctrl;
2191
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002192 if (copy_from_user(&ctrl, arg, sizeof(ctrl)))
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002193 return -EFAULT;
2194 return proc_ioctl(ps, &ctrl);
2195}
2196
2197#ifdef CONFIG_COMPAT
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002198static int proc_ioctl_compat(struct usb_dev_state *ps, compat_uptr_t arg)
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002199{
2200 struct usbdevfs_ioctl32 __user *uioc;
2201 struct usbdevfs_ioctl ctrl;
2202 u32 udata;
2203
Andrew Morton058120d2005-11-17 09:47:49 -08002204 uioc = compat_ptr((long)arg);
Michael Buesch18753eb2009-07-29 11:39:03 +02002205 if (!access_ok(VERIFY_READ, uioc, sizeof(*uioc)) ||
2206 __get_user(ctrl.ifno, &uioc->ifno) ||
2207 __get_user(ctrl.ioctl_code, &uioc->ioctl_code) ||
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002208 __get_user(udata, &uioc->data))
2209 return -EFAULT;
2210 ctrl.data = compat_ptr(udata);
2211
2212 return proc_ioctl(ps, &ctrl);
2213}
2214#endif
2215
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002216static int proc_claim_port(struct usb_dev_state *ps, void __user *arg)
Alan Stern7cbe5dc2009-06-29 10:56:54 -04002217{
2218 unsigned portnum;
2219 int rc;
2220
2221 if (get_user(portnum, (unsigned __user *) arg))
2222 return -EFAULT;
2223 rc = usb_hub_claim_port(ps->dev, portnum, ps);
2224 if (rc == 0)
2225 snoop(&ps->dev->dev, "port %d claimed by process %d: %s\n",
2226 portnum, task_pid_nr(current), current->comm);
2227 return rc;
2228}
2229
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002230static int proc_release_port(struct usb_dev_state *ps, void __user *arg)
Alan Stern7cbe5dc2009-06-29 10:56:54 -04002231{
2232 unsigned portnum;
2233
2234 if (get_user(portnum, (unsigned __user *) arg))
2235 return -EFAULT;
2236 return usb_hub_release_port(ps->dev, portnum, ps);
2237}
2238
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002239static int proc_get_capabilities(struct usb_dev_state *ps, void __user *arg)
Hans de Goede19181bc2012-07-04 09:18:02 +02002240{
2241 __u32 caps;
2242
Alan Stern3f2cee72015-01-29 11:29:13 -05002243 caps = USBDEVFS_CAP_ZERO_PACKET | USBDEVFS_CAP_NO_PACKET_SIZE_LIM |
Reilly Grantd883f522016-02-21 18:38:01 -03002244 USBDEVFS_CAP_REAP_AFTER_DISCONNECT | USBDEVFS_CAP_MMAP |
2245 USBDEVFS_CAP_DROP_PRIVILEGES;
Hans de Goede19181bc2012-07-04 09:18:02 +02002246 if (!ps->dev->bus->no_stop_on_short)
2247 caps |= USBDEVFS_CAP_BULK_CONTINUATION;
Hans de Goede3d97ff62012-07-04 09:18:03 +02002248 if (ps->dev->bus->sg_tablesize)
2249 caps |= USBDEVFS_CAP_BULK_SCATTER_GATHER;
Hans de Goede19181bc2012-07-04 09:18:02 +02002250
2251 if (put_user(caps, (__u32 __user *)arg))
2252 return -EFAULT;
2253
2254 return 0;
2255}
2256
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002257static int proc_disconnect_claim(struct usb_dev_state *ps, void __user *arg)
Hans de Goede0837e7e2012-09-08 20:02:05 +02002258{
2259 struct usbdevfs_disconnect_claim dc;
2260 struct usb_interface *intf;
2261
2262 if (copy_from_user(&dc, arg, sizeof(dc)))
2263 return -EFAULT;
2264
2265 intf = usb_ifnum_to_if(ps->dev, dc.interface);
2266 if (!intf)
2267 return -EINVAL;
2268
2269 if (intf->dev.driver) {
2270 struct usb_driver *driver = to_usb_driver(intf->dev.driver);
2271
Reilly Grantd883f522016-02-21 18:38:01 -03002272 if (ps->privileges_dropped)
2273 return -EACCES;
2274
Hans de Goede0837e7e2012-09-08 20:02:05 +02002275 if ((dc.flags & USBDEVFS_DISCONNECT_CLAIM_IF_DRIVER) &&
2276 strncmp(dc.driver, intf->dev.driver->name,
2277 sizeof(dc.driver)) != 0)
2278 return -EBUSY;
2279
2280 if ((dc.flags & USBDEVFS_DISCONNECT_CLAIM_EXCEPT_DRIVER) &&
2281 strncmp(dc.driver, intf->dev.driver->name,
2282 sizeof(dc.driver)) == 0)
2283 return -EBUSY;
2284
2285 dev_dbg(&intf->dev, "disconnect by usbfs\n");
2286 usb_driver_release_interface(driver, intf);
2287 }
2288
2289 return claimintf(ps, dc.interface);
2290}
2291
Linus Torvalds3e75c6d2014-04-01 17:06:09 -07002292static int proc_alloc_streams(struct usb_dev_state *ps, void __user *arg)
Hans de Goedebcf7f6e2013-10-09 17:19:31 +02002293{
2294 unsigned num_streams, num_eps;
2295 struct usb_host_endpoint **eps;
2296 struct usb_interface *intf;
2297 int r;
2298
2299 r = parse_usbdevfs_streams(ps, arg, &num_streams, &num_eps,
2300 &eps, &intf);
2301 if (r)
2302 return r;
2303
2304 destroy_async_on_interface(ps,
2305 intf->altsetting[0].desc.bInterfaceNumber);
2306
2307 r = usb_alloc_streams(intf, eps, num_eps, num_streams, GFP_KERNEL);
2308 kfree(eps);
2309 return r;
2310}
2311
Linus Torvalds3e75c6d2014-04-01 17:06:09 -07002312static int proc_free_streams(struct usb_dev_state *ps, void __user *arg)
Hans de Goedebcf7f6e2013-10-09 17:19:31 +02002313{
2314 unsigned num_eps;
2315 struct usb_host_endpoint **eps;
2316 struct usb_interface *intf;
2317 int r;
2318
2319 r = parse_usbdevfs_streams(ps, arg, NULL, &num_eps, &eps, &intf);
2320 if (r)
2321 return r;
2322
2323 destroy_async_on_interface(ps,
2324 intf->altsetting[0].desc.bInterfaceNumber);
2325
2326 r = usb_free_streams(intf, eps, num_eps, GFP_KERNEL);
2327 kfree(eps);
2328 return r;
2329}
2330
Reilly Grantd883f522016-02-21 18:38:01 -03002331static int proc_drop_privileges(struct usb_dev_state *ps, void __user *arg)
2332{
2333 u32 data;
2334
2335 if (copy_from_user(&data, arg, sizeof(data)))
2336 return -EFAULT;
2337
Masahiro Yamada0f5e1552017-02-27 14:28:52 -08002338 /* This is a one way operation. Once privileges are
Reilly Grantd883f522016-02-21 18:38:01 -03002339 * dropped, you cannot regain them. You may however reissue
2340 * this ioctl to shrink the allowed interfaces mask.
2341 */
2342 ps->interface_allowed_mask &= data;
2343 ps->privileges_dropped = true;
2344
2345 return 0;
2346}
2347
Linus Torvalds1da177e2005-04-16 15:20:36 -07002348/*
2349 * NOTE: All requests here that have interface numbers as parameters
2350 * are assuming that somehow the configuration has been prevented from
2351 * changing. But there's no mechanism to ensure that...
2352 */
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002353static long usbdev_do_ioctl(struct file *file, unsigned int cmd,
2354 void __user *p)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002355{
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002356 struct usb_dev_state *ps = file->private_data;
Al Viro496ad9a2013-01-23 17:07:38 -05002357 struct inode *inode = file_inode(file);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002358 struct usb_device *dev = ps->dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002359 int ret = -ENOTTY;
2360
2361 if (!(file->f_mode & FMODE_WRITE))
2362 return -EPERM;
Oliver Neukum01412a22010-01-13 15:33:43 +01002363
Linus Torvalds1da177e2005-04-16 15:20:36 -07002364 usb_lock_device(dev);
Alan Stern3f2cee72015-01-29 11:29:13 -05002365
2366 /* Reap operations are allowed even after disconnection */
2367 switch (cmd) {
2368 case USBDEVFS_REAPURB:
2369 snoop(&dev->dev, "%s: REAPURB\n", __func__);
2370 ret = proc_reapurb(ps, p);
2371 goto done;
2372
2373 case USBDEVFS_REAPURBNDELAY:
2374 snoop(&dev->dev, "%s: REAPURBNDELAY\n", __func__);
2375 ret = proc_reapurbnonblock(ps, p);
2376 goto done;
2377
2378#ifdef CONFIG_COMPAT
2379 case USBDEVFS_REAPURB32:
2380 snoop(&dev->dev, "%s: REAPURB32\n", __func__);
2381 ret = proc_reapurb_compat(ps, p);
2382 goto done;
2383
2384 case USBDEVFS_REAPURBNDELAY32:
2385 snoop(&dev->dev, "%s: REAPURBNDELAY32\n", __func__);
2386 ret = proc_reapurbnonblock_compat(ps, p);
2387 goto done;
2388#endif
2389 }
2390
Alan Stern349710c2006-07-01 22:05:56 -04002391 if (!connected(ps)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002392 usb_unlock_device(dev);
2393 return -ENODEV;
2394 }
2395
2396 switch (cmd) {
2397 case USBDEVFS_CONTROL:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002398 snoop(&dev->dev, "%s: CONTROL\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002399 ret = proc_control(ps, p);
2400 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002401 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002402 break;
2403
2404 case USBDEVFS_BULK:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002405 snoop(&dev->dev, "%s: BULK\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002406 ret = proc_bulk(ps, p);
2407 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002408 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002409 break;
2410
2411 case USBDEVFS_RESETEP:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002412 snoop(&dev->dev, "%s: RESETEP\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002413 ret = proc_resetep(ps, p);
2414 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002415 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002416 break;
2417
2418 case USBDEVFS_RESET:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002419 snoop(&dev->dev, "%s: RESET\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002420 ret = proc_resetdevice(ps);
2421 break;
2422
2423 case USBDEVFS_CLEAR_HALT:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002424 snoop(&dev->dev, "%s: CLEAR_HALT\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002425 ret = proc_clearhalt(ps, p);
2426 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002427 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002428 break;
2429
2430 case USBDEVFS_GETDRIVER:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002431 snoop(&dev->dev, "%s: GETDRIVER\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002432 ret = proc_getdriver(ps, p);
2433 break;
2434
2435 case USBDEVFS_CONNECTINFO:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002436 snoop(&dev->dev, "%s: CONNECTINFO\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002437 ret = proc_connectinfo(ps, p);
2438 break;
2439
2440 case USBDEVFS_SETINTERFACE:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002441 snoop(&dev->dev, "%s: SETINTERFACE\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002442 ret = proc_setintf(ps, p);
2443 break;
2444
2445 case USBDEVFS_SETCONFIGURATION:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002446 snoop(&dev->dev, "%s: SETCONFIGURATION\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002447 ret = proc_setconfig(ps, p);
2448 break;
2449
2450 case USBDEVFS_SUBMITURB:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002451 snoop(&dev->dev, "%s: SUBMITURB\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002452 ret = proc_submiturb(ps, p);
2453 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002454 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002455 break;
2456
2457#ifdef CONFIG_COMPAT
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002458 case USBDEVFS_CONTROL32:
2459 snoop(&dev->dev, "%s: CONTROL32\n", __func__);
2460 ret = proc_control_compat(ps, p);
2461 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002462 inode->i_mtime = current_time(inode);
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002463 break;
2464
2465 case USBDEVFS_BULK32:
2466 snoop(&dev->dev, "%s: BULK32\n", __func__);
2467 ret = proc_bulk_compat(ps, p);
2468 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002469 inode->i_mtime = current_time(inode);
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002470 break;
2471
2472 case USBDEVFS_DISCSIGNAL32:
2473 snoop(&dev->dev, "%s: DISCSIGNAL32\n", __func__);
2474 ret = proc_disconnectsignal_compat(ps, p);
2475 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002476
2477 case USBDEVFS_SUBMITURB32:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002478 snoop(&dev->dev, "%s: SUBMITURB32\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002479 ret = proc_submiturb_compat(ps, p);
2480 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002481 inode->i_mtime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002482 break;
2483
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002484 case USBDEVFS_IOCTL32:
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002485 snoop(&dev->dev, "%s: IOCTL32\n", __func__);
Al Viroc714de52006-10-10 22:45:37 +01002486 ret = proc_ioctl_compat(ps, ptr_to_compat(p));
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002487 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002488#endif
2489
2490 case USBDEVFS_DISCARDURB:
Alan Sterna016a812015-11-20 13:53:35 -05002491 snoop(&dev->dev, "%s: DISCARDURB %p\n", __func__, p);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002492 ret = proc_unlinkurb(ps, p);
2493 break;
2494
Linus Torvalds1da177e2005-04-16 15:20:36 -07002495 case USBDEVFS_DISCSIGNAL:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002496 snoop(&dev->dev, "%s: DISCSIGNAL\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002497 ret = proc_disconnectsignal(ps, p);
2498 break;
2499
2500 case USBDEVFS_CLAIMINTERFACE:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002501 snoop(&dev->dev, "%s: CLAIMINTERFACE\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002502 ret = proc_claiminterface(ps, p);
2503 break;
2504
2505 case USBDEVFS_RELEASEINTERFACE:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002506 snoop(&dev->dev, "%s: RELEASEINTERFACE\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002507 ret = proc_releaseinterface(ps, p);
2508 break;
2509
2510 case USBDEVFS_IOCTL:
Harvey Harrison441b62c2008-03-03 16:08:34 -08002511 snoop(&dev->dev, "%s: IOCTL\n", __func__);
Pete Zaitcevc36fc882005-10-17 18:15:54 -07002512 ret = proc_ioctl_default(ps, p);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002513 break;
Alan Stern7cbe5dc2009-06-29 10:56:54 -04002514
2515 case USBDEVFS_CLAIM_PORT:
2516 snoop(&dev->dev, "%s: CLAIM_PORT\n", __func__);
2517 ret = proc_claim_port(ps, p);
2518 break;
2519
2520 case USBDEVFS_RELEASE_PORT:
2521 snoop(&dev->dev, "%s: RELEASE_PORT\n", __func__);
2522 ret = proc_release_port(ps, p);
2523 break;
Hans de Goede19181bc2012-07-04 09:18:02 +02002524 case USBDEVFS_GET_CAPABILITIES:
2525 ret = proc_get_capabilities(ps, p);
2526 break;
Hans de Goede0837e7e2012-09-08 20:02:05 +02002527 case USBDEVFS_DISCONNECT_CLAIM:
2528 ret = proc_disconnect_claim(ps, p);
2529 break;
Hans de Goedebcf7f6e2013-10-09 17:19:31 +02002530 case USBDEVFS_ALLOC_STREAMS:
2531 ret = proc_alloc_streams(ps, p);
2532 break;
2533 case USBDEVFS_FREE_STREAMS:
2534 ret = proc_free_streams(ps, p);
2535 break;
Reilly Grantd883f522016-02-21 18:38:01 -03002536 case USBDEVFS_DROP_PRIVILEGES:
2537 ret = proc_drop_privileges(ps, p);
2538 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002539 }
Alan Stern3f2cee72015-01-29 11:29:13 -05002540
2541 done:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002542 usb_unlock_device(dev);
2543 if (ret >= 0)
Deepa Dinamani078cd822016-09-14 07:48:04 -07002544 inode->i_atime = current_time(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002545 return ret;
2546}
2547
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002548static long usbdev_ioctl(struct file *file, unsigned int cmd,
2549 unsigned long arg)
2550{
2551 int ret;
2552
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002553 ret = usbdev_do_ioctl(file, cmd, (void __user *)arg);
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002554
2555 return ret;
2556}
2557
2558#ifdef CONFIG_COMPAT
2559static long usbdev_compat_ioctl(struct file *file, unsigned int cmd,
2560 unsigned long arg)
2561{
2562 int ret;
2563
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002564 ret = usbdev_do_ioctl(file, cmd, compat_ptr(arg));
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002565
2566 return ret;
2567}
2568#endif
2569
Linus Torvalds1da177e2005-04-16 15:20:36 -07002570/* No kernel lock - fine */
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002571static unsigned int usbdev_poll(struct file *file,
2572 struct poll_table_struct *wait)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002573{
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002574 struct usb_dev_state *ps = file->private_data;
Tobias Klauserec17cf12006-09-13 21:38:41 +02002575 unsigned int mask = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002576
2577 poll_wait(file, &ps->wait, wait);
2578 if (file->f_mode & FMODE_WRITE && !list_empty(&ps->async_completed))
2579 mask |= POLLOUT | POLLWRNORM;
Alan Stern349710c2006-07-01 22:05:56 -04002580 if (!connected(ps))
Alan Stern5cce4382016-06-10 14:42:55 -04002581 mask |= POLLHUP;
2582 if (list_empty(&ps->list))
2583 mask |= POLLERR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002584 return mask;
2585}
2586
Kay Sievers9f8b17e2007-03-13 15:59:31 +01002587const struct file_operations usbdev_file_operations = {
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002588 .owner = THIS_MODULE,
Al Virob25472f2015-12-05 22:04:48 -05002589 .llseek = no_seek_end_llseek,
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002590 .read = usbdev_read,
2591 .poll = usbdev_poll,
2592 .unlocked_ioctl = usbdev_ioctl,
2593#ifdef CONFIG_COMPAT
2594 .compat_ioctl = usbdev_compat_ioctl,
2595#endif
Steinar H. Gundersonf7d34b42016-02-03 22:58:26 +01002596 .mmap = usbdev_mmap,
Arnd Bergmann637e8a62009-11-14 02:28:05 +01002597 .open = usbdev_open,
2598 .release = usbdev_release,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002599};
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002600
Alan Stern501950d2009-01-13 11:33:42 -05002601static void usbdev_remove(struct usb_device *udev)
Alan Sterncd9f0372008-06-24 14:47:04 -04002602{
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002603 struct usb_dev_state *ps;
Alan Sterncd9f0372008-06-24 14:47:04 -04002604 struct siginfo sinfo;
2605
2606 while (!list_empty(&udev->filelist)) {
Valentina Manea9b6f0c42014-03-10 10:36:40 +02002607 ps = list_entry(udev->filelist.next, struct usb_dev_state, list);
Alan Sterncd9f0372008-06-24 14:47:04 -04002608 destroy_all_async(ps);
2609 wake_up_all(&ps->wait);
2610 list_del_init(&ps->list);
2611 if (ps->discsignr) {
Alan Sternf0c2b682015-02-13 10:54:53 -05002612 memset(&sinfo, 0, sizeof(sinfo));
Alan Sterncd9f0372008-06-24 14:47:04 -04002613 sinfo.si_signo = ps->discsignr;
2614 sinfo.si_errno = EPIPE;
2615 sinfo.si_code = SI_ASYNCIO;
2616 sinfo.si_addr = ps->disccontext;
Serge Hallynd178bc32011-09-26 10:45:18 -05002617 kill_pid_info_as_cred(ps->discsignr, &sinfo,
2618 ps->disc_pid, ps->cred, ps->secid);
Alan Sterncd9f0372008-06-24 14:47:04 -04002619 }
2620 }
2621}
2622
Alan Stern501950d2009-01-13 11:33:42 -05002623static int usbdev_notify(struct notifier_block *self,
Kay Sievers9f8b17e2007-03-13 15:59:31 +01002624 unsigned long action, void *dev)
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002625{
2626 switch (action) {
2627 case USB_DEVICE_ADD:
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002628 break;
2629 case USB_DEVICE_REMOVE:
Alan Stern501950d2009-01-13 11:33:42 -05002630 usbdev_remove(dev);
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002631 break;
2632 }
2633 return NOTIFY_OK;
2634}
2635
2636static struct notifier_block usbdev_nb = {
Chase Metzger00fe52d2015-04-09 21:41:52 -07002637 .notifier_call = usbdev_notify,
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002638};
2639
Greg Kroah-Hartman7e7654a2007-09-12 15:06:57 -07002640static struct cdev usb_device_cdev;
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002641
Kay Sievers9f8b17e2007-03-13 15:59:31 +01002642int __init usb_devio_init(void)
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002643{
2644 int retval;
2645
Alan Sternfad21bd2005-08-10 15:15:57 -04002646 retval = register_chrdev_region(USB_DEVICE_DEV, USB_DEVICE_MAX,
Greg Kroah-Hartman04e482ff2008-01-30 15:21:33 -08002647 "usb_device");
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002648 if (retval) {
Greg Kroah-Hartman69a85942008-08-14 09:37:34 -07002649 printk(KERN_ERR "Unable to register minors for usb_device\n");
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002650 goto out;
2651 }
Kay Sievers9f8b17e2007-03-13 15:59:31 +01002652 cdev_init(&usb_device_cdev, &usbdev_file_operations);
Alan Sternfad21bd2005-08-10 15:15:57 -04002653 retval = cdev_add(&usb_device_cdev, USB_DEVICE_DEV, USB_DEVICE_MAX);
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002654 if (retval) {
Greg Kroah-Hartman69a85942008-08-14 09:37:34 -07002655 printk(KERN_ERR "Unable to get usb_device major %d\n",
2656 USB_DEVICE_MAJOR);
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002657 goto error_cdev;
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002658 }
Alan Stern501950d2009-01-13 11:33:42 -05002659 usb_register_notify(&usbdev_nb);
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002660out:
2661 return retval;
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002662
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002663error_cdev:
2664 unregister_chrdev_region(USB_DEVICE_DEV, USB_DEVICE_MAX);
2665 goto out;
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002666}
2667
Kay Sievers9f8b17e2007-03-13 15:59:31 +01002668void usb_devio_cleanup(void)
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002669{
Greg Kroah-Hartmana7b986b2005-06-20 21:15:16 -07002670 usb_unregister_notify(&usbdev_nb);
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002671 cdev_del(&usb_device_cdev);
Alan Sternfad21bd2005-08-10 15:15:57 -04002672 unregister_chrdev_region(USB_DEVICE_DEV, USB_DEVICE_MAX);
Kay Sieversfbf82fd2005-07-31 01:05:53 +02002673}