2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* inode.c -- user mode filesystem api for usb gadget controllers
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003-2004 David Brownell
|
|
|
|
* Copyright (C) 2003 Agilent Technologies
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2007-08-02 14:58:22 +08:00
|
|
|
/* #define VERBOSE_DEBUG */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/pagemap.h>
|
|
|
|
#include <linux/uts.h>
|
|
|
|
#include <linux/wait.h>
|
|
|
|
#include <linux/compiler.h>
|
2016-12-25 03:46:01 +08:00
|
|
|
#include <linux/uaccess.h>
|
2009-10-04 20:11:37 +08:00
|
|
|
#include <linux/sched.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/slab.h>
|
2006-08-09 13:23:12 +08:00
|
|
|
#include <linux/poll.h>
|
2013-05-08 07:18:23 +08:00
|
|
|
#include <linux/mmu_context.h>
|
2013-05-08 07:18:33 +08:00
|
|
|
#include <linux/aio.h>
|
2015-02-23 00:58:50 +08:00
|
|
|
#include <linux/uio.h>
|
2017-03-06 22:21:13 +08:00
|
|
|
#include <linux/refcount.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
|
2007-05-15 10:36:41 +08:00
|
|
|
#include <linux/usb/gadgetfs.h>
|
2007-10-05 09:05:17 +08:00
|
|
|
#include <linux/usb/gadget.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The gadgetfs API maps each endpoint to a file descriptor so that you
|
|
|
|
* can use standard synchronous read/write calls for I/O. There's some
|
|
|
|
* O_NONBLOCK and O_ASYNC/FASYNC style i/o support. Example usermode
|
|
|
|
* drivers show how this works in practice. You can also use AIO to
|
|
|
|
* eliminate I/O gaps between requests, to help when streaming data.
|
|
|
|
*
|
|
|
|
* Key parts that must be USB-specific are protocols defining how the
|
|
|
|
* read/write operations relate to the hardware state machines. There
|
|
|
|
* are two types of files. One type is for the device, implementing ep0.
|
|
|
|
* The other type is for each IN or OUT endpoint. In both cases, the
|
|
|
|
* user mode driver must configure the hardware before using it.
|
|
|
|
*
|
|
|
|
* - First, dev_config() is called when /dev/gadget/$CHIP is configured
|
|
|
|
* (by writing configuration and device descriptors). Afterwards it
|
|
|
|
* may serve as a source of device events, used to handle all control
|
|
|
|
* requests other than basic enumeration.
|
|
|
|
*
|
2007-01-16 03:35:01 +08:00
|
|
|
* - Then, after a SET_CONFIGURATION control request, ep_config() is
|
|
|
|
* called when each /dev/gadget/ep* file is configured (by writing
|
|
|
|
* endpoint descriptors). Afterwards these files are used to write()
|
|
|
|
* IN data or to read() OUT data. To halt the endpoint, a "wrong
|
|
|
|
* direction" request is issued (like reading an IN endpoint).
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* Unlike "usbfs" the only ioctl()s are for things that are rare, and maybe
|
|
|
|
* not possible on all hardware. For example, precise fault handling with
|
|
|
|
* respect to data left in endpoint fifos after aborted operations; or
|
|
|
|
* selective clearing of endpoint halts, to implement SET_INTERFACE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define DRIVER_DESC "USB Gadget filesystem"
|
|
|
|
#define DRIVER_VERSION "24 Aug 2004"
|
|
|
|
|
|
|
|
static const char driver_desc [] = DRIVER_DESC;
|
|
|
|
static const char shortname [] = "gadgetfs";
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION (DRIVER_DESC);
|
|
|
|
MODULE_AUTHOR ("David Brownell");
|
|
|
|
MODULE_LICENSE ("GPL");
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
static int ep_open(struct inode *, struct file *);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#define GADGETFS_MAGIC 0xaee71ee7
|
|
|
|
|
|
|
|
/* /dev/gadget/$CHIP represents ep0 and the whole device */
|
|
|
|
enum ep0_state {
|
2017-03-10 08:16:31 +08:00
|
|
|
/* DISABLED is the initial state. */
|
2005-04-17 06:20:36 +08:00
|
|
|
STATE_DEV_DISABLED = 0,
|
|
|
|
|
|
|
|
/* Only one open() of /dev/gadget/$CHIP; only one file tracks
|
|
|
|
* ep0/device i/o modes and binding to the controller. Driver
|
|
|
|
* must always write descriptors to initialize the device, then
|
|
|
|
* the device becomes UNCONNECTED until enumeration.
|
|
|
|
*/
|
2007-01-17 14:51:04 +08:00
|
|
|
STATE_DEV_OPENED,
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* From then on, ep0 fd is in either of two basic modes:
|
|
|
|
* - (UN)CONNECTED: read usb_gadgetfs_event(s) from it
|
|
|
|
* - SETUP: read/write will transfer control data and succeed;
|
|
|
|
* or if "wrong direction", performs protocol stall
|
|
|
|
*/
|
2007-01-17 14:51:04 +08:00
|
|
|
STATE_DEV_UNCONNECTED,
|
|
|
|
STATE_DEV_CONNECTED,
|
|
|
|
STATE_DEV_SETUP,
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* UNBOUND means the driver closed ep0, so the device won't be
|
|
|
|
* accessible again (DEV_DISABLED) until all fds are closed.
|
|
|
|
*/
|
|
|
|
STATE_DEV_UNBOUND,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* enough for the whole queue: most events invalidate others */
|
|
|
|
#define N_EVENT 5
|
|
|
|
|
|
|
|
struct dev_data {
|
|
|
|
spinlock_t lock;
|
2017-03-06 22:21:13 +08:00
|
|
|
refcount_t count;
|
2007-01-17 14:51:04 +08:00
|
|
|
enum ep0_state state; /* P: lock */
|
2005-04-17 06:20:36 +08:00
|
|
|
struct usb_gadgetfs_event event [N_EVENT];
|
|
|
|
unsigned ev_next;
|
|
|
|
struct fasync_struct *fasync;
|
|
|
|
u8 current_config;
|
|
|
|
|
|
|
|
/* drivers reading ep0 MUST handle control requests (SETUP)
|
|
|
|
* reported that way; else the host will time out.
|
|
|
|
*/
|
|
|
|
unsigned usermode_setup : 1,
|
|
|
|
setup_in : 1,
|
|
|
|
setup_can_stall : 1,
|
|
|
|
setup_out_ready : 1,
|
|
|
|
setup_out_error : 1,
|
2016-02-18 15:59:26 +08:00
|
|
|
setup_abort : 1,
|
|
|
|
gadget_registered : 1;
|
2006-01-03 23:30:31 +08:00
|
|
|
unsigned setup_wLength;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* the rest is basically write-once */
|
|
|
|
struct usb_config_descriptor *config, *hs_config;
|
|
|
|
struct usb_device_descriptor *dev;
|
|
|
|
struct usb_request *req;
|
|
|
|
struct usb_gadget *gadget;
|
|
|
|
struct list_head epfiles;
|
|
|
|
void *buf;
|
|
|
|
wait_queue_head_t wait;
|
|
|
|
struct super_block *sb;
|
|
|
|
struct dentry *dentry;
|
|
|
|
|
|
|
|
/* except this scratch i/o buffer for ep0 */
|
|
|
|
u8 rbuf [256];
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline void get_dev (struct dev_data *data)
|
|
|
|
{
|
2017-03-06 22:21:13 +08:00
|
|
|
refcount_inc (&data->count);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void put_dev (struct dev_data *data)
|
|
|
|
{
|
2017-03-06 22:21:13 +08:00
|
|
|
if (likely (!refcount_dec_and_test (&data->count)))
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
/* needs no more cleanup */
|
|
|
|
BUG_ON (waitqueue_active (&data->wait));
|
|
|
|
kfree (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct dev_data *dev_new (void)
|
|
|
|
{
|
|
|
|
struct dev_data *dev;
|
|
|
|
|
2006-02-28 05:34:10 +08:00
|
|
|
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!dev)
|
|
|
|
return NULL;
|
|
|
|
dev->state = STATE_DEV_DISABLED;
|
2017-03-06 22:21:13 +08:00
|
|
|
refcount_set (&dev->count, 1);
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_lock_init (&dev->lock);
|
|
|
|
INIT_LIST_HEAD (&dev->epfiles);
|
|
|
|
init_waitqueue_head (&dev->wait);
|
|
|
|
return dev;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* other /dev/gadget/$ENDPOINT files represent endpoints */
|
|
|
|
enum ep_state {
|
|
|
|
STATE_EP_DISABLED = 0,
|
|
|
|
STATE_EP_READY,
|
|
|
|
STATE_EP_ENABLED,
|
|
|
|
STATE_EP_UNBOUND,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ep_data {
|
2010-01-30 04:38:59 +08:00
|
|
|
struct mutex lock;
|
2005-04-17 06:20:36 +08:00
|
|
|
enum ep_state state;
|
2017-03-06 22:21:14 +08:00
|
|
|
refcount_t count;
|
2005-04-17 06:20:36 +08:00
|
|
|
struct dev_data *dev;
|
|
|
|
/* must hold dev->lock before accessing ep or req */
|
|
|
|
struct usb_ep *ep;
|
|
|
|
struct usb_request *req;
|
|
|
|
ssize_t status;
|
|
|
|
char name [16];
|
|
|
|
struct usb_endpoint_descriptor desc, hs_desc;
|
|
|
|
struct list_head epfiles;
|
|
|
|
wait_queue_head_t wait;
|
|
|
|
struct dentry *dentry;
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline void get_ep (struct ep_data *data)
|
|
|
|
{
|
2017-03-06 22:21:14 +08:00
|
|
|
refcount_inc (&data->count);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void put_ep (struct ep_data *data)
|
|
|
|
{
|
2017-03-06 22:21:14 +08:00
|
|
|
if (likely (!refcount_dec_and_test (&data->count)))
|
2005-04-17 06:20:36 +08:00
|
|
|
return;
|
|
|
|
put_dev (data->dev);
|
|
|
|
/* needs no more cleanup */
|
|
|
|
BUG_ON (!list_empty (&data->epfiles));
|
|
|
|
BUG_ON (waitqueue_active (&data->wait));
|
|
|
|
kfree (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* most "how to use the hardware" policy choices are in userspace:
|
|
|
|
* mapping endpoint roles (which the driver needs) to the capabilities
|
|
|
|
* which the usb controller has. most of those capabilities are exposed
|
|
|
|
* implicitly, starting with the driver name and then endpoint names.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static const char *CHIP;
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* NOTE: don't use dev_printk calls before binding to the gadget
|
|
|
|
* at the end of ep0 configuration, or after unbind.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* too wordy: dev_printk(level , &(d)->gadget->dev , fmt , ## args) */
|
|
|
|
#define xprintk(d,level,fmt,args...) \
|
|
|
|
printk(level "%s: " fmt , shortname , ## args)
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
#define DBG(dev,fmt,args...) \
|
|
|
|
xprintk(dev , KERN_DEBUG , fmt , ## args)
|
|
|
|
#else
|
|
|
|
#define DBG(dev,fmt,args...) \
|
|
|
|
do { } while (0)
|
|
|
|
#endif /* DEBUG */
|
|
|
|
|
2007-08-02 14:58:22 +08:00
|
|
|
#ifdef VERBOSE_DEBUG
|
2005-04-17 06:20:36 +08:00
|
|
|
#define VDEBUG DBG
|
|
|
|
#else
|
|
|
|
#define VDEBUG(dev,fmt,args...) \
|
|
|
|
do { } while (0)
|
|
|
|
#endif /* DEBUG */
|
|
|
|
|
|
|
|
#define ERROR(dev,fmt,args...) \
|
|
|
|
xprintk(dev , KERN_ERR , fmt , ## args)
|
|
|
|
#define INFO(dev,fmt,args...) \
|
|
|
|
xprintk(dev , KERN_INFO , fmt , ## args)
|
|
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* SYNCHRONOUS ENDPOINT OPERATIONS (bulk/intr/iso)
|
|
|
|
*
|
|
|
|
* After opening, configure non-control endpoints. Then use normal
|
|
|
|
* stream read() and write() requests; and maybe ioctl() to get more
|
2005-05-04 09:07:24 +08:00
|
|
|
* precise FIFO status when recovering from cancellation.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
static void epio_complete (struct usb_ep *ep, struct usb_request *req)
|
|
|
|
{
|
|
|
|
struct ep_data *epdata = ep->driver_data;
|
|
|
|
|
|
|
|
if (!req->context)
|
|
|
|
return;
|
|
|
|
if (req->status)
|
|
|
|
epdata->status = req->status;
|
|
|
|
else
|
|
|
|
epdata->status = req->actual;
|
|
|
|
complete ((struct completion *)req->context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* tasklock endpoint, returning when it's connected.
|
|
|
|
* still need dev->lock to use epdata->ep.
|
|
|
|
*/
|
|
|
|
static int
|
2015-03-03 16:39:34 +08:00
|
|
|
get_ready_ep (unsigned f_flags, struct ep_data *epdata, bool is_write)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
int val;
|
|
|
|
|
|
|
|
if (f_flags & O_NONBLOCK) {
|
2010-01-30 04:38:59 +08:00
|
|
|
if (!mutex_trylock(&epdata->lock))
|
2005-04-17 06:20:36 +08:00
|
|
|
goto nonblock;
|
2015-03-03 16:39:34 +08:00
|
|
|
if (epdata->state != STATE_EP_ENABLED &&
|
|
|
|
(!is_write || epdata->state != STATE_EP_READY)) {
|
2010-01-30 04:38:59 +08:00
|
|
|
mutex_unlock(&epdata->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
nonblock:
|
|
|
|
val = -EAGAIN;
|
|
|
|
} else
|
|
|
|
val = 0;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2010-01-30 04:38:59 +08:00
|
|
|
val = mutex_lock_interruptible(&epdata->lock);
|
|
|
|
if (val < 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
return val;
|
2007-01-16 03:35:01 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
switch (epdata->state) {
|
|
|
|
case STATE_EP_ENABLED:
|
2015-03-03 16:39:34 +08:00
|
|
|
return 0;
|
|
|
|
case STATE_EP_READY: /* not configured yet */
|
|
|
|
if (is_write)
|
|
|
|
return 0;
|
|
|
|
// FALLTHRU
|
|
|
|
case STATE_EP_UNBOUND: /* clean disconnect */
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
// case STATE_EP_DISABLED: /* "can't happen" */
|
|
|
|
default: /* error! */
|
|
|
|
pr_debug ("%s: ep %p not available, state %d\n",
|
|
|
|
shortname, epdata, epdata->state);
|
|
|
|
}
|
2015-03-03 16:39:34 +08:00
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -ENODEV;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
ep_io (struct ep_data *epdata, void *buf, unsigned len)
|
|
|
|
{
|
2006-10-01 14:28:10 +08:00
|
|
|
DECLARE_COMPLETION_ONSTACK (done);
|
2005-04-17 06:20:36 +08:00
|
|
|
int value;
|
|
|
|
|
|
|
|
spin_lock_irq (&epdata->dev->lock);
|
|
|
|
if (likely (epdata->ep != NULL)) {
|
|
|
|
struct usb_request *req = epdata->req;
|
|
|
|
|
|
|
|
req->context = &done;
|
|
|
|
req->complete = epio_complete;
|
|
|
|
req->buf = buf;
|
|
|
|
req->length = len;
|
|
|
|
value = usb_ep_queue (epdata->ep, req, GFP_ATOMIC);
|
|
|
|
} else
|
|
|
|
value = -ENODEV;
|
|
|
|
spin_unlock_irq (&epdata->dev->lock);
|
|
|
|
|
|
|
|
if (likely (value == 0)) {
|
|
|
|
value = wait_event_interruptible (done.wait, done.done);
|
|
|
|
if (value != 0) {
|
|
|
|
spin_lock_irq (&epdata->dev->lock);
|
|
|
|
if (likely (epdata->ep != NULL)) {
|
|
|
|
DBG (epdata->dev, "%s i/o interrupted\n",
|
|
|
|
epdata->name);
|
|
|
|
usb_ep_dequeue (epdata->ep, epdata->req);
|
|
|
|
spin_unlock_irq (&epdata->dev->lock);
|
|
|
|
|
|
|
|
wait_event (done.wait, done.done);
|
|
|
|
if (epdata->status == -ECONNRESET)
|
|
|
|
epdata->status = -EINTR;
|
|
|
|
} else {
|
|
|
|
spin_unlock_irq (&epdata->dev->lock);
|
|
|
|
|
|
|
|
DBG (epdata->dev, "endpoint gone\n");
|
|
|
|
epdata->status = -ENODEV;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return epdata->status;
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
ep_release (struct inode *inode, struct file *fd)
|
|
|
|
{
|
|
|
|
struct ep_data *data = fd->private_data;
|
2006-06-26 22:48:00 +08:00
|
|
|
int value;
|
|
|
|
|
2010-01-30 04:38:59 +08:00
|
|
|
value = mutex_lock_interruptible(&data->lock);
|
|
|
|
if (value < 0)
|
2006-06-26 22:48:00 +08:00
|
|
|
return value;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* clean up if this can be reopened */
|
|
|
|
if (data->state != STATE_EP_UNBOUND) {
|
|
|
|
data->state = STATE_EP_DISABLED;
|
|
|
|
data->desc.bDescriptorType = 0;
|
|
|
|
data->hs_desc.bDescriptorType = 0;
|
2005-09-08 00:49:34 +08:00
|
|
|
usb_ep_disable(data->ep);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2010-01-30 04:38:59 +08:00
|
|
|
mutex_unlock(&data->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
put_ep (data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-05-23 05:03:27 +08:00
|
|
|
static long ep_ioctl(struct file *fd, unsigned code, unsigned long value)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct ep_data *data = fd->private_data;
|
|
|
|
int status;
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
if ((status = get_ready_ep (fd->f_flags, data, false)) < 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
return status;
|
|
|
|
|
|
|
|
spin_lock_irq (&data->dev->lock);
|
|
|
|
if (likely (data->ep != NULL)) {
|
|
|
|
switch (code) {
|
|
|
|
case GADGETFS_FIFO_STATUS:
|
|
|
|
status = usb_ep_fifo_status (data->ep);
|
|
|
|
break;
|
|
|
|
case GADGETFS_FIFO_FLUSH:
|
|
|
|
usb_ep_fifo_flush (data->ep);
|
|
|
|
break;
|
|
|
|
case GADGETFS_CLEAR_HALT:
|
|
|
|
status = usb_ep_clear_halt (data->ep);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
status = -ENOTTY;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
status = -ENODEV;
|
|
|
|
spin_unlock_irq (&data->dev->lock);
|
2010-01-30 04:38:59 +08:00
|
|
|
mutex_unlock(&data->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* ASYNCHRONOUS ENDPOINT I/O OPERATIONS (bulk/intr/iso) */
|
|
|
|
|
|
|
|
struct kiocb_priv {
|
|
|
|
struct usb_request *req;
|
|
|
|
struct ep_data *epdata;
|
2013-05-08 07:18:23 +08:00
|
|
|
struct kiocb *iocb;
|
|
|
|
struct mm_struct *mm;
|
|
|
|
struct work_struct work;
|
2005-04-17 06:20:36 +08:00
|
|
|
void *buf;
|
2015-02-07 13:30:23 +08:00
|
|
|
struct iov_iter to;
|
|
|
|
const void *to_free;
|
2005-04-17 06:20:36 +08:00
|
|
|
unsigned actual;
|
|
|
|
};
|
|
|
|
|
2013-05-14 05:45:08 +08:00
|
|
|
static int ep_aio_cancel(struct kiocb *iocb)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct kiocb_priv *priv = iocb->private;
|
|
|
|
struct ep_data *epdata;
|
|
|
|
int value;
|
|
|
|
|
|
|
|
local_irq_disable();
|
|
|
|
epdata = priv->epdata;
|
|
|
|
// spin_lock(&epdata->dev->lock);
|
|
|
|
if (likely(epdata && epdata->ep && priv->req))
|
|
|
|
value = usb_ep_dequeue (epdata->ep, priv->req);
|
|
|
|
else
|
|
|
|
value = -EINVAL;
|
|
|
|
// spin_unlock(&epdata->dev->lock);
|
|
|
|
local_irq_enable();
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2013-05-08 07:18:23 +08:00
|
|
|
static void ep_user_copy_worker(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct kiocb_priv *priv = container_of(work, struct kiocb_priv, work);
|
|
|
|
struct mm_struct *mm = priv->mm;
|
|
|
|
struct kiocb *iocb = priv->iocb;
|
|
|
|
size_t ret;
|
|
|
|
|
|
|
|
use_mm(mm);
|
2015-02-07 13:30:23 +08:00
|
|
|
ret = copy_to_iter(priv->buf, priv->actual, &priv->to);
|
2013-05-08 07:18:23 +08:00
|
|
|
unuse_mm(mm);
|
2015-02-07 13:30:23 +08:00
|
|
|
if (!ret)
|
|
|
|
ret = -EFAULT;
|
2013-05-08 07:18:23 +08:00
|
|
|
|
|
|
|
/* completing the iocb can drop the ctx and mm, don't touch mm after */
|
2015-02-02 21:49:06 +08:00
|
|
|
iocb->ki_complete(iocb, ret, ret);
|
2013-05-08 07:18:23 +08:00
|
|
|
|
2007-01-16 03:30:28 +08:00
|
|
|
kfree(priv->buf);
|
2015-02-07 13:30:23 +08:00
|
|
|
kfree(priv->to_free);
|
2007-01-16 03:30:28 +08:00
|
|
|
kfree(priv);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ep_aio_complete(struct usb_ep *ep, struct usb_request *req)
|
|
|
|
{
|
|
|
|
struct kiocb *iocb = req->context;
|
|
|
|
struct kiocb_priv *priv = iocb->private;
|
|
|
|
struct ep_data *epdata = priv->epdata;
|
|
|
|
|
|
|
|
/* lock against disconnect (and ideally, cancel) */
|
|
|
|
spin_lock(&epdata->dev->lock);
|
|
|
|
priv->req = NULL;
|
|
|
|
priv->epdata = NULL;
|
2007-01-17 15:28:48 +08:00
|
|
|
|
|
|
|
/* if this was a write or a read returning no data then we
|
|
|
|
* don't need to copy anything to userspace, so we can
|
|
|
|
* complete the aio request immediately.
|
|
|
|
*/
|
2015-02-07 13:30:23 +08:00
|
|
|
if (priv->to_free == NULL || unlikely(req->actual == 0)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(req->buf);
|
2015-02-07 13:30:23 +08:00
|
|
|
kfree(priv->to_free);
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree(priv);
|
|
|
|
iocb->private = NULL;
|
|
|
|
/* aio_complete() reports bytes-transferred _and_ faults */
|
2015-02-02 21:49:06 +08:00
|
|
|
|
|
|
|
iocb->ki_complete(iocb, req->actual ? req->actual : req->status,
|
2005-04-17 06:20:36 +08:00
|
|
|
req->status);
|
|
|
|
} else {
|
2013-05-08 07:18:23 +08:00
|
|
|
/* ep_copy_to_user() won't report both; we hide some faults */
|
2005-04-17 06:20:36 +08:00
|
|
|
if (unlikely(0 != req->status))
|
|
|
|
DBG(epdata->dev, "%s fault %d len %d\n",
|
|
|
|
ep->name, req->status, req->actual);
|
|
|
|
|
|
|
|
priv->buf = req->buf;
|
|
|
|
priv->actual = req->actual;
|
2015-02-07 13:30:23 +08:00
|
|
|
INIT_WORK(&priv->work, ep_user_copy_worker);
|
2013-05-08 07:18:23 +08:00
|
|
|
schedule_work(&priv->work);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
spin_unlock(&epdata->dev->lock);
|
|
|
|
|
|
|
|
usb_ep_free_request(ep, req);
|
|
|
|
put_ep(epdata);
|
|
|
|
}
|
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
static ssize_t ep_aio(struct kiocb *iocb,
|
|
|
|
struct kiocb_priv *priv,
|
|
|
|
struct ep_data *epdata,
|
|
|
|
char *buf,
|
|
|
|
size_t len)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2015-02-07 13:30:23 +08:00
|
|
|
struct usb_request *req;
|
|
|
|
ssize_t value;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
iocb->private = priv;
|
2013-05-08 07:18:23 +08:00
|
|
|
priv->iocb = iocb;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-05-08 07:18:49 +08:00
|
|
|
kiocb_set_cancel_fn(iocb, ep_aio_cancel);
|
2005-04-17 06:20:36 +08:00
|
|
|
get_ep(epdata);
|
|
|
|
priv->epdata = epdata;
|
|
|
|
priv->actual = 0;
|
2013-05-08 07:18:23 +08:00
|
|
|
priv->mm = current->mm; /* mm teardown waits for iocbs in exit_aio() */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* each kiocb is coupled to one usb_request, but we can't
|
|
|
|
* allocate or submit those if the host disconnected.
|
|
|
|
*/
|
|
|
|
spin_lock_irq(&epdata->dev->lock);
|
2015-02-07 13:30:23 +08:00
|
|
|
value = -ENODEV;
|
2016-07-15 20:58:41 +08:00
|
|
|
if (unlikely(epdata->ep == NULL))
|
2015-02-07 13:30:23 +08:00
|
|
|
goto fail;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
req = usb_ep_alloc_request(epdata->ep, GFP_ATOMIC);
|
|
|
|
value = -ENOMEM;
|
|
|
|
if (unlikely(!req))
|
|
|
|
goto fail;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
priv->req = req;
|
|
|
|
req->buf = buf;
|
|
|
|
req->length = len;
|
|
|
|
req->complete = ep_aio_complete;
|
|
|
|
req->context = iocb;
|
|
|
|
value = usb_ep_queue(epdata->ep, req, GFP_ATOMIC);
|
|
|
|
if (unlikely(0 != value)) {
|
|
|
|
usb_ep_free_request(epdata->ep, req);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&epdata->dev->lock);
|
|
|
|
return -EIOCBQUEUED;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
spin_unlock_irq(&epdata->dev->lock);
|
|
|
|
kfree(priv->to_free);
|
|
|
|
kfree(priv);
|
|
|
|
put_ep(epdata);
|
2005-04-17 06:20:36 +08:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
2015-02-07 13:30:23 +08:00
|
|
|
ep_read_iter(struct kiocb *iocb, struct iov_iter *to)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2015-02-07 13:30:23 +08:00
|
|
|
struct file *file = iocb->ki_filp;
|
|
|
|
struct ep_data *epdata = file->private_data;
|
|
|
|
size_t len = iov_iter_count(to);
|
|
|
|
ssize_t value;
|
|
|
|
char *buf;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
if ((value = get_ready_ep(file->f_flags, epdata, false)) < 0)
|
2015-02-07 13:30:23 +08:00
|
|
|
return value;
|
2006-10-01 14:28:46 +08:00
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
/* halt any endpoint by doing a "wrong direction" i/o call */
|
|
|
|
if (usb_endpoint_dir_in(&epdata->desc)) {
|
|
|
|
if (usb_endpoint_xfer_isoc(&epdata->desc) ||
|
|
|
|
!is_sync_kiocb(iocb)) {
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
DBG (epdata->dev, "%s halt\n", epdata->name);
|
|
|
|
spin_lock_irq(&epdata->dev->lock);
|
|
|
|
if (likely(epdata->ep != NULL))
|
|
|
|
usb_ep_set_halt(epdata->ep);
|
|
|
|
spin_unlock_irq(&epdata->dev->lock);
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
2006-10-01 14:28:46 +08:00
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
buf = kmalloc(len, GFP_KERNEL);
|
|
|
|
if (unlikely(!buf)) {
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
if (is_sync_kiocb(iocb)) {
|
|
|
|
value = ep_io(epdata, buf, len);
|
2016-07-08 03:22:04 +08:00
|
|
|
if (value >= 0 && (copy_to_iter(buf, value, to) != value))
|
2015-02-07 13:30:23 +08:00
|
|
|
value = -EFAULT;
|
|
|
|
} else {
|
|
|
|
struct kiocb_priv *priv = kzalloc(sizeof *priv, GFP_KERNEL);
|
|
|
|
value = -ENOMEM;
|
|
|
|
if (!priv)
|
|
|
|
goto fail;
|
|
|
|
priv->to_free = dup_iter(&priv->to, to, GFP_KERNEL);
|
|
|
|
if (!priv->to_free) {
|
|
|
|
kfree(priv);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
value = ep_aio(iocb, priv, epdata, buf, len);
|
|
|
|
if (value == -EIOCBQUEUED)
|
|
|
|
buf = NULL;
|
|
|
|
}
|
|
|
|
fail:
|
|
|
|
kfree(buf);
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return value;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
static ssize_t ep_config(struct ep_data *, const char *, size_t);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
static ssize_t
|
2015-02-07 13:30:23 +08:00
|
|
|
ep_write_iter(struct kiocb *iocb, struct iov_iter *from)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2015-02-07 13:30:23 +08:00
|
|
|
struct file *file = iocb->ki_filp;
|
|
|
|
struct ep_data *epdata = file->private_data;
|
|
|
|
size_t len = iov_iter_count(from);
|
2015-03-03 16:39:34 +08:00
|
|
|
bool configured;
|
2015-02-07 13:30:23 +08:00
|
|
|
ssize_t value;
|
|
|
|
char *buf;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
if ((value = get_ready_ep(file->f_flags, epdata, true)) < 0)
|
2015-02-07 13:30:23 +08:00
|
|
|
return value;
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
configured = epdata->state == STATE_EP_ENABLED;
|
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
/* halt any endpoint by doing a "wrong direction" i/o call */
|
2015-03-03 16:39:34 +08:00
|
|
|
if (configured && !usb_endpoint_dir_in(&epdata->desc)) {
|
2015-02-07 13:30:23 +08:00
|
|
|
if (usb_endpoint_xfer_isoc(&epdata->desc) ||
|
|
|
|
!is_sync_kiocb(iocb)) {
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
DBG (epdata->dev, "%s halt\n", epdata->name);
|
|
|
|
spin_lock_irq(&epdata->dev->lock);
|
|
|
|
if (likely(epdata->ep != NULL))
|
|
|
|
usb_ep_set_halt(epdata->ep);
|
|
|
|
spin_unlock_irq(&epdata->dev->lock);
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
2006-10-01 14:28:46 +08:00
|
|
|
|
2015-02-07 13:30:23 +08:00
|
|
|
buf = kmalloc(len, GFP_KERNEL);
|
|
|
|
if (unlikely(!buf)) {
|
|
|
|
mutex_unlock(&epdata->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return -ENOMEM;
|
2015-02-07 13:30:23 +08:00
|
|
|
}
|
2006-10-01 14:28:46 +08:00
|
|
|
|
2016-11-02 10:09:04 +08:00
|
|
|
if (unlikely(!copy_from_iter_full(buf, len, from))) {
|
2015-02-07 13:30:23 +08:00
|
|
|
value = -EFAULT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
if (unlikely(!configured)) {
|
|
|
|
value = ep_config(epdata, buf, len);
|
|
|
|
} else if (is_sync_kiocb(iocb)) {
|
2015-02-07 13:30:23 +08:00
|
|
|
value = ep_io(epdata, buf, len);
|
|
|
|
} else {
|
|
|
|
struct kiocb_priv *priv = kzalloc(sizeof *priv, GFP_KERNEL);
|
|
|
|
value = -ENOMEM;
|
|
|
|
if (priv) {
|
|
|
|
value = ep_aio(iocb, priv, epdata, buf, len);
|
|
|
|
if (value == -EIOCBQUEUED)
|
|
|
|
buf = NULL;
|
2006-10-01 14:28:46 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2015-02-07 13:30:23 +08:00
|
|
|
out:
|
|
|
|
kfree(buf);
|
|
|
|
mutex_unlock(&epdata->lock);
|
|
|
|
return value;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* used after endpoint configuration */
|
2006-08-06 07:37:11 +08:00
|
|
|
static const struct file_operations ep_io_operations = {
|
2005-04-17 06:20:36 +08:00
|
|
|
.owner = THIS_MODULE,
|
|
|
|
|
2015-03-03 16:39:34 +08:00
|
|
|
.open = ep_open,
|
|
|
|
.release = ep_release,
|
|
|
|
.llseek = no_llseek,
|
2008-05-23 05:03:27 +08:00
|
|
|
.unlocked_ioctl = ep_ioctl,
|
2015-02-07 13:30:23 +08:00
|
|
|
.read_iter = ep_read_iter,
|
|
|
|
.write_iter = ep_write_iter,
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* ENDPOINT INITIALIZATION
|
|
|
|
*
|
|
|
|
* fd = open ("/dev/gadget/$ENDPOINT", O_RDWR)
|
|
|
|
* status = write (fd, descriptors, sizeof descriptors)
|
|
|
|
*
|
|
|
|
* That write establishes the endpoint configuration, configuring
|
|
|
|
* the controller to process bulk, interrupt, or isochronous transfers
|
|
|
|
* at the right maxpacket size, and so on.
|
|
|
|
*
|
|
|
|
* The descriptors are message type 1, identified by a host order u32
|
|
|
|
* at the beginning of what's written. Descriptor order is: full/low
|
|
|
|
* speed descriptor, then optional high speed descriptor.
|
|
|
|
*/
|
|
|
|
static ssize_t
|
2015-03-03 16:39:34 +08:00
|
|
|
ep_config (struct ep_data *data, const char *buf, size_t len)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct usb_ep *ep;
|
|
|
|
u32 tag;
|
2006-06-26 22:19:00 +08:00
|
|
|
int value, length = len;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (data->state != STATE_EP_READY) {
|
|
|
|
value = -EL2HLT;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = len;
|
|
|
|
if (len < USB_DT_ENDPOINT_SIZE + 4)
|
|
|
|
goto fail0;
|
|
|
|
|
|
|
|
/* we might need to change message format someday */
|
2015-03-03 16:39:34 +08:00
|
|
|
memcpy(&tag, buf, 4);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (tag != 1) {
|
|
|
|
DBG(data->dev, "config %s, bad tag %d\n", data->name, tag);
|
|
|
|
goto fail0;
|
|
|
|
}
|
|
|
|
buf += 4;
|
|
|
|
len -= 4;
|
|
|
|
|
|
|
|
/* NOTE: audio endpoint extensions not accepted here;
|
|
|
|
* just don't include the extra bytes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* full/low speed descriptor, then high speed */
|
2015-03-03 16:39:34 +08:00
|
|
|
memcpy(&data->desc, buf, USB_DT_ENDPOINT_SIZE);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (data->desc.bLength != USB_DT_ENDPOINT_SIZE
|
|
|
|
|| data->desc.bDescriptorType != USB_DT_ENDPOINT)
|
|
|
|
goto fail0;
|
|
|
|
if (len != USB_DT_ENDPOINT_SIZE) {
|
|
|
|
if (len != 2 * USB_DT_ENDPOINT_SIZE)
|
|
|
|
goto fail0;
|
2015-03-03 16:39:34 +08:00
|
|
|
memcpy(&data->hs_desc, buf + USB_DT_ENDPOINT_SIZE,
|
|
|
|
USB_DT_ENDPOINT_SIZE);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (data->hs_desc.bLength != USB_DT_ENDPOINT_SIZE
|
|
|
|
|| data->hs_desc.bDescriptorType
|
|
|
|
!= USB_DT_ENDPOINT) {
|
|
|
|
DBG(data->dev, "config %s, bad hs length or type\n",
|
|
|
|
data->name);
|
|
|
|
goto fail0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irq (&data->dev->lock);
|
|
|
|
if (data->dev->state == STATE_DEV_UNBOUND) {
|
|
|
|
value = -ENOENT;
|
|
|
|
goto gone;
|
2015-04-30 17:32:54 +08:00
|
|
|
} else {
|
|
|
|
ep = data->ep;
|
|
|
|
if (ep == NULL) {
|
|
|
|
value = -ENODEV;
|
|
|
|
goto gone;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
switch (data->dev->gadget->speed) {
|
|
|
|
case USB_SPEED_LOW:
|
|
|
|
case USB_SPEED_FULL:
|
2011-06-28 21:33:48 +08:00
|
|
|
ep->desc = &data->desc;
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
case USB_SPEED_HIGH:
|
|
|
|
/* fails if caller didn't provide that descriptor... */
|
2011-06-28 21:33:48 +08:00
|
|
|
ep->desc = &data->hs_desc;
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
default:
|
2007-01-16 03:35:01 +08:00
|
|
|
DBG(data->dev, "unconnected, %s init abandoned\n",
|
2005-04-17 06:20:36 +08:00
|
|
|
data->name);
|
2007-01-16 03:35:01 +08:00
|
|
|
value = -EINVAL;
|
2015-03-03 16:39:34 +08:00
|
|
|
goto gone;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2015-03-03 16:39:34 +08:00
|
|
|
value = usb_ep_enable(ep);
|
2006-06-26 22:19:00 +08:00
|
|
|
if (value == 0) {
|
2015-03-03 16:39:34 +08:00
|
|
|
data->state = STATE_EP_ENABLED;
|
2006-06-26 22:19:00 +08:00
|
|
|
value = length;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
gone:
|
|
|
|
spin_unlock_irq (&data->dev->lock);
|
|
|
|
if (value < 0) {
|
|
|
|
fail:
|
|
|
|
data->desc.bDescriptorType = 0;
|
|
|
|
data->hs_desc.bDescriptorType = 0;
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
fail0:
|
|
|
|
value = -EINVAL;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
ep_open (struct inode *inode, struct file *fd)
|
|
|
|
{
|
2006-09-27 16:50:46 +08:00
|
|
|
struct ep_data *data = inode->i_private;
|
2005-04-17 06:20:36 +08:00
|
|
|
int value = -EBUSY;
|
|
|
|
|
2010-01-30 04:38:59 +08:00
|
|
|
if (mutex_lock_interruptible(&data->lock) != 0)
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINTR;
|
|
|
|
spin_lock_irq (&data->dev->lock);
|
|
|
|
if (data->dev->state == STATE_DEV_UNBOUND)
|
|
|
|
value = -ENOENT;
|
|
|
|
else if (data->state == STATE_EP_DISABLED) {
|
|
|
|
value = 0;
|
|
|
|
data->state = STATE_EP_READY;
|
|
|
|
get_ep (data);
|
|
|
|
fd->private_data = data;
|
|
|
|
VDEBUG (data->dev, "%s ready\n", data->name);
|
|
|
|
} else
|
|
|
|
DBG (data->dev, "%s state %d\n",
|
|
|
|
data->name, data->state);
|
|
|
|
spin_unlock_irq (&data->dev->lock);
|
2010-01-30 04:38:59 +08:00
|
|
|
mutex_unlock(&data->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* EP0 IMPLEMENTATION can be partly in userspace.
|
|
|
|
*
|
|
|
|
* Drivers that use this facility receive various events, including
|
|
|
|
* control requests the kernel doesn't handle. Drivers that don't
|
|
|
|
* use this facility may be too simple-minded for real applications.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline void ep0_readable (struct dev_data *dev)
|
|
|
|
{
|
|
|
|
wake_up (&dev->wait);
|
|
|
|
kill_fasync (&dev->fasync, SIGIO, POLL_IN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void clean_req (struct usb_ep *ep, struct usb_request *req)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = ep->driver_data;
|
|
|
|
|
|
|
|
if (req->buf != dev->rbuf) {
|
2007-07-02 02:04:54 +08:00
|
|
|
kfree(req->buf);
|
2005-04-17 06:20:36 +08:00
|
|
|
req->buf = dev->rbuf;
|
|
|
|
}
|
|
|
|
req->complete = epio_complete;
|
|
|
|
dev->setup_out_ready = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ep0_complete (struct usb_ep *ep, struct usb_request *req)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = ep->driver_data;
|
2007-01-17 14:56:26 +08:00
|
|
|
unsigned long flags;
|
2005-04-17 06:20:36 +08:00
|
|
|
int free = 1;
|
|
|
|
|
|
|
|
/* for control OUT, data must still get to userspace */
|
2007-01-17 14:56:26 +08:00
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!dev->setup_in) {
|
|
|
|
dev->setup_out_error = (req->status != 0);
|
|
|
|
if (!dev->setup_out_error)
|
|
|
|
free = 0;
|
|
|
|
dev->setup_out_ready = 1;
|
|
|
|
ep0_readable (dev);
|
2007-01-17 14:51:04 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* clean up as appropriate */
|
|
|
|
if (free && req->buf != &dev->rbuf)
|
|
|
|
clean_req (ep, req);
|
|
|
|
req->complete = epio_complete;
|
2007-01-17 14:56:26 +08:00
|
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int setup_req (struct usb_ep *ep, struct usb_request *req, u16 len)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = ep->driver_data;
|
|
|
|
|
|
|
|
if (dev->setup_out_ready) {
|
|
|
|
DBG (dev, "ep0 request busy!\n");
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
if (len > sizeof (dev->rbuf))
|
2007-07-02 02:04:54 +08:00
|
|
|
req->buf = kmalloc(len, GFP_ATOMIC);
|
2007-07-31 03:31:07 +08:00
|
|
|
if (req->buf == NULL) {
|
2005-04-17 06:20:36 +08:00
|
|
|
req->buf = dev->rbuf;
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
req->complete = ep0_complete;
|
|
|
|
req->length = len;
|
2006-01-03 23:30:31 +08:00
|
|
|
req->zero = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
ep0_read (struct file *fd, char __user *buf, size_t len, loff_t *ptr)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
ssize_t retval;
|
|
|
|
enum ep0_state state;
|
|
|
|
|
|
|
|
spin_lock_irq (&dev->lock);
|
2015-03-04 23:31:50 +08:00
|
|
|
if (dev->state <= STATE_DEV_OPENED) {
|
|
|
|
retval = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* report fd mode change before acting on it */
|
|
|
|
if (dev->setup_abort) {
|
|
|
|
dev->setup_abort = 0;
|
|
|
|
retval = -EIDRM;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* control DATA stage */
|
2007-01-17 14:51:04 +08:00
|
|
|
if ((state = dev->state) == STATE_DEV_SETUP) {
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (dev->setup_in) { /* stall IN */
|
|
|
|
VDEBUG(dev, "ep0in stall\n");
|
|
|
|
(void) usb_ep_set_halt (dev->gadget->ep0);
|
|
|
|
retval = -EL2HLT;
|
2007-01-17 14:51:04 +08:00
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
} else if (len == 0) { /* ack SET_CONFIGURATION etc */
|
|
|
|
struct usb_ep *ep = dev->gadget->ep0;
|
|
|
|
struct usb_request *req = dev->req;
|
|
|
|
|
2016-05-27 00:43:45 +08:00
|
|
|
if ((retval = setup_req (ep, req, 0)) == 0) {
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
retval = usb_ep_queue (ep, req, GFP_KERNEL);
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
}
|
2007-01-17 14:51:04 +08:00
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* assume that was SET_CONFIGURATION */
|
|
|
|
if (dev->current_config) {
|
|
|
|
unsigned power;
|
2007-08-02 14:58:22 +08:00
|
|
|
|
|
|
|
if (gadget_is_dualspeed(dev->gadget)
|
|
|
|
&& (dev->gadget->speed
|
|
|
|
== USB_SPEED_HIGH))
|
2005-04-17 06:20:36 +08:00
|
|
|
power = dev->hs_config->bMaxPower;
|
|
|
|
else
|
|
|
|
power = dev->config->bMaxPower;
|
|
|
|
usb_gadget_vbus_draw(dev->gadget, 2 * power);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else { /* collect OUT data */
|
|
|
|
if ((fd->f_flags & O_NONBLOCK) != 0
|
|
|
|
&& !dev->setup_out_ready) {
|
|
|
|
retval = -EAGAIN;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
retval = wait_event_interruptible (dev->wait,
|
|
|
|
dev->setup_out_ready != 0);
|
|
|
|
|
|
|
|
/* FIXME state could change from under us */
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
if (retval)
|
|
|
|
goto done;
|
2007-01-17 14:56:26 +08:00
|
|
|
|
|
|
|
if (dev->state != STATE_DEV_SETUP) {
|
|
|
|
retval = -ECANCELED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
if (dev->setup_out_error)
|
|
|
|
retval = -EIO;
|
|
|
|
else {
|
|
|
|
len = min (len, (size_t)dev->req->actual);
|
|
|
|
// FIXME don't call this with the spinlock held ...
|
2006-09-02 06:26:27 +08:00
|
|
|
if (copy_to_user (buf, dev->req->buf, len))
|
2005-04-17 06:20:36 +08:00
|
|
|
retval = -EFAULT;
|
2012-03-02 16:41:50 +08:00
|
|
|
else
|
|
|
|
retval = len;
|
2005-04-17 06:20:36 +08:00
|
|
|
clean_req (dev->gadget->ep0, dev->req);
|
|
|
|
/* NOTE userspace can't yet choose to stall */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* else normal: return event data */
|
|
|
|
if (len < sizeof dev->event [0]) {
|
|
|
|
retval = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
len -= len % sizeof (struct usb_gadgetfs_event);
|
|
|
|
dev->usermode_setup = 1;
|
|
|
|
|
|
|
|
scan:
|
|
|
|
/* return queued events right away */
|
|
|
|
if (dev->ev_next != 0) {
|
|
|
|
unsigned i, n;
|
|
|
|
|
|
|
|
n = len / sizeof (struct usb_gadgetfs_event);
|
2007-01-17 14:53:58 +08:00
|
|
|
if (dev->ev_next < n)
|
|
|
|
n = dev->ev_next;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-01-17 14:53:58 +08:00
|
|
|
/* ep0 i/o has special semantics during STATE_DEV_SETUP */
|
2005-04-17 06:20:36 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
if (dev->event [i].type == GADGETFS_SETUP) {
|
2007-01-17 14:53:58 +08:00
|
|
|
dev->state = STATE_DEV_SETUP;
|
|
|
|
n = i + 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock_irq (&dev->lock);
|
2007-01-17 14:53:58 +08:00
|
|
|
len = n * sizeof (struct usb_gadgetfs_event);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (copy_to_user (buf, &dev->event, len))
|
|
|
|
retval = -EFAULT;
|
|
|
|
else
|
|
|
|
retval = len;
|
|
|
|
if (len > 0) {
|
|
|
|
/* NOTE this doesn't guard against broken drivers;
|
|
|
|
* concurrent ep0 readers may lose events.
|
|
|
|
*/
|
|
|
|
spin_lock_irq (&dev->lock);
|
2007-01-17 14:53:58 +08:00
|
|
|
if (dev->ev_next > n) {
|
|
|
|
memmove(&dev->event[0], &dev->event[n],
|
2005-04-17 06:20:36 +08:00
|
|
|
sizeof (struct usb_gadgetfs_event)
|
2007-01-17 14:53:58 +08:00
|
|
|
* (dev->ev_next - n));
|
|
|
|
}
|
|
|
|
dev->ev_next -= n;
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
if (fd->f_flags & O_NONBLOCK) {
|
|
|
|
retval = -EAGAIN;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
default:
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "fail %s, state %d\n", __func__, state);
|
2005-04-17 06:20:36 +08:00
|
|
|
retval = -ESRCH;
|
|
|
|
break;
|
2007-01-17 14:51:04 +08:00
|
|
|
case STATE_DEV_UNCONNECTED:
|
|
|
|
case STATE_DEV_CONNECTED:
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_unlock_irq (&dev->lock);
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "%s wait\n", __func__);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* wait for events */
|
|
|
|
retval = wait_event_interruptible (dev->wait,
|
|
|
|
dev->ev_next != 0);
|
|
|
|
if (retval < 0)
|
|
|
|
return retval;
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
goto scan;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_gadgetfs_event *
|
|
|
|
next_event (struct dev_data *dev, enum usb_gadgetfs_event_type type)
|
|
|
|
{
|
|
|
|
struct usb_gadgetfs_event *event;
|
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
/* these events purge the queue */
|
|
|
|
case GADGETFS_DISCONNECT:
|
2007-01-17 14:51:04 +08:00
|
|
|
if (dev->state == STATE_DEV_SETUP)
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->setup_abort = 1;
|
|
|
|
// FALL THROUGH
|
|
|
|
case GADGETFS_CONNECT:
|
|
|
|
dev->ev_next = 0;
|
|
|
|
break;
|
|
|
|
case GADGETFS_SETUP: /* previous request timed out */
|
|
|
|
case GADGETFS_SUSPEND: /* same effect */
|
|
|
|
/* these events can't be repeated */
|
|
|
|
for (i = 0; i != dev->ev_next; i++) {
|
|
|
|
if (dev->event [i].type != type)
|
|
|
|
continue;
|
2007-01-17 14:53:58 +08:00
|
|
|
DBG(dev, "discard old event[%d] %d\n", i, type);
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->ev_next--;
|
|
|
|
if (i == dev->ev_next)
|
|
|
|
break;
|
|
|
|
/* indices start at zero, for simplicity */
|
|
|
|
memmove (&dev->event [i], &dev->event [i + 1],
|
|
|
|
sizeof (struct usb_gadgetfs_event)
|
|
|
|
* (dev->ev_next - i));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG ();
|
|
|
|
}
|
2007-01-17 14:53:58 +08:00
|
|
|
VDEBUG(dev, "event[%d] = %d\n", dev->ev_next, type);
|
2005-04-17 06:20:36 +08:00
|
|
|
event = &dev->event [dev->ev_next++];
|
|
|
|
BUG_ON (dev->ev_next > N_EVENT);
|
|
|
|
memset (event, 0, sizeof *event);
|
|
|
|
event->type = type;
|
|
|
|
return event;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
ep0_write (struct file *fd, const char __user *buf, size_t len, loff_t *ptr)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
ssize_t retval = -ESRCH;
|
|
|
|
|
|
|
|
/* report fd mode change before acting on it */
|
|
|
|
if (dev->setup_abort) {
|
|
|
|
dev->setup_abort = 0;
|
|
|
|
retval = -EIDRM;
|
|
|
|
|
|
|
|
/* data and/or status stage for control request */
|
2007-01-17 14:51:04 +08:00
|
|
|
} else if (dev->state == STATE_DEV_SETUP) {
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2016-12-10 04:17:46 +08:00
|
|
|
len = min_t(size_t, len, dev->setup_wLength);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (dev->setup_in) {
|
|
|
|
retval = setup_req (dev->gadget->ep0, dev->req, len);
|
|
|
|
if (retval == 0) {
|
2007-01-17 14:56:26 +08:00
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
2005-04-17 06:20:36 +08:00
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
if (copy_from_user (dev->req->buf, buf, len))
|
|
|
|
retval = -EFAULT;
|
2006-01-03 23:30:31 +08:00
|
|
|
else {
|
|
|
|
if (len < dev->setup_wLength)
|
|
|
|
dev->req->zero = 1;
|
2005-04-17 06:20:36 +08:00
|
|
|
retval = usb_ep_queue (
|
|
|
|
dev->gadget->ep0, dev->req,
|
|
|
|
GFP_KERNEL);
|
2006-01-03 23:30:31 +08:00
|
|
|
}
|
2015-12-12 05:13:55 +08:00
|
|
|
spin_lock_irq(&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (retval < 0) {
|
|
|
|
clean_req (dev->gadget->ep0, dev->req);
|
|
|
|
} else
|
|
|
|
retval = len;
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* can stall some OUT transfers */
|
|
|
|
} else if (dev->setup_can_stall) {
|
|
|
|
VDEBUG(dev, "ep0out stall\n");
|
|
|
|
(void) usb_ep_set_halt (dev->gadget->ep0);
|
|
|
|
retval = -EL2HLT;
|
2007-01-17 14:51:04 +08:00
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
|
|
|
DBG(dev, "bogus ep0out stall!\n");
|
|
|
|
}
|
|
|
|
} else
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "fail %s, state %d\n", __func__, dev->state);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
ep0_fasync (int f, struct file *fd, int on)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
// caller must F_SETOWN before signal delivery happens
|
2008-03-04 08:08:34 +08:00
|
|
|
VDEBUG (dev, "%s %s\n", __func__, on ? "on" : "off");
|
2005-04-17 06:20:36 +08:00
|
|
|
return fasync_helper (f, fd, on, &dev->fasync);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_gadget_driver gadgetfs_driver;
|
|
|
|
|
|
|
|
static int
|
|
|
|
dev_release (struct inode *inode, struct file *fd)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
|
|
|
|
/* closing ep0 === shutdown all */
|
|
|
|
|
2017-06-09 01:55:59 +08:00
|
|
|
if (dev->gadget_registered) {
|
2016-02-18 15:59:26 +08:00
|
|
|
usb_gadget_unregister_driver (&gadgetfs_driver);
|
2017-06-09 01:55:59 +08:00
|
|
|
dev->gadget_registered = false;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* at this point "good" hardware has disconnected the
|
|
|
|
* device from USB; the host won't see it any more.
|
|
|
|
* alternatively, all host requests will time out.
|
|
|
|
*/
|
|
|
|
|
|
|
|
kfree (dev->buf);
|
|
|
|
dev->buf = NULL;
|
|
|
|
|
2014-06-05 23:17:06 +08:00
|
|
|
/* other endpoints were all decoupled from this device */
|
|
|
|
spin_lock_irq(&dev->lock);
|
|
|
|
dev->state = STATE_DEV_DISABLED;
|
|
|
|
spin_unlock_irq(&dev->lock);
|
|
|
|
|
|
|
|
put_dev (dev);
|
2005-04-17 06:20:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-08-09 13:23:12 +08:00
|
|
|
static unsigned int
|
|
|
|
ep0_poll (struct file *fd, poll_table *wait)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
int mask = 0;
|
|
|
|
|
2015-03-04 23:31:50 +08:00
|
|
|
if (dev->state <= STATE_DEV_OPENED)
|
|
|
|
return DEFAULT_POLLMASK;
|
|
|
|
|
2006-08-09 13:23:12 +08:00
|
|
|
poll_wait(fd, &dev->wait, wait);
|
|
|
|
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
|
|
|
|
/* report fd mode change before acting on it */
|
|
|
|
if (dev->setup_abort) {
|
|
|
|
dev->setup_abort = 0;
|
|
|
|
mask = POLLHUP;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2007-01-17 14:51:04 +08:00
|
|
|
if (dev->state == STATE_DEV_SETUP) {
|
2006-08-09 13:23:12 +08:00
|
|
|
if (dev->setup_in || dev->setup_can_stall)
|
|
|
|
mask = POLLOUT;
|
|
|
|
} else {
|
|
|
|
if (dev->ev_next != 0)
|
|
|
|
mask = POLLIN;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
spin_unlock_irq(&dev->lock);
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
2008-05-23 05:03:27 +08:00
|
|
|
static long dev_ioctl (struct file *fd, unsigned code, unsigned long value)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
struct usb_gadget *gadget = dev->gadget;
|
2008-05-23 05:03:27 +08:00
|
|
|
long ret = -ENOTTY;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-06-02 05:04:44 +08:00
|
|
|
if (gadget->ops->ioctl)
|
2008-05-23 05:03:27 +08:00
|
|
|
ret = gadget->ops->ioctl (gadget, code, value);
|
2010-06-02 05:04:44 +08:00
|
|
|
|
2008-05-23 05:03:27 +08:00
|
|
|
return ret;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* The in-kernel gadget driver handles most ep0 issues, in particular
|
|
|
|
* enumerating the single configuration (as provided from user space).
|
|
|
|
*
|
|
|
|
* Unrecognized ep0 requests may be handled in user space.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void make_qualifier (struct dev_data *dev)
|
|
|
|
{
|
|
|
|
struct usb_qualifier_descriptor qual;
|
|
|
|
struct usb_device_descriptor *desc;
|
|
|
|
|
|
|
|
qual.bLength = sizeof qual;
|
|
|
|
qual.bDescriptorType = USB_DT_DEVICE_QUALIFIER;
|
2009-02-12 06:11:36 +08:00
|
|
|
qual.bcdUSB = cpu_to_le16 (0x0200);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
desc = dev->dev;
|
|
|
|
qual.bDeviceClass = desc->bDeviceClass;
|
|
|
|
qual.bDeviceSubClass = desc->bDeviceSubClass;
|
|
|
|
qual.bDeviceProtocol = desc->bDeviceProtocol;
|
|
|
|
|
|
|
|
/* assumes ep0 uses the same value for both speeds ... */
|
2011-06-23 20:26:11 +08:00
|
|
|
qual.bMaxPacketSize0 = dev->gadget->ep0->maxpacket;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
qual.bNumConfigurations = 1;
|
|
|
|
qual.bRESERVED = 0;
|
|
|
|
|
|
|
|
memcpy (dev->rbuf, &qual, sizeof qual);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
config_buf (struct dev_data *dev, u8 type, unsigned index)
|
|
|
|
{
|
|
|
|
int len;
|
2007-08-02 14:58:22 +08:00
|
|
|
int hs = 0;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* only one configuration */
|
|
|
|
if (index > 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2007-08-02 14:58:22 +08:00
|
|
|
if (gadget_is_dualspeed(dev->gadget)) {
|
|
|
|
hs = (dev->gadget->speed == USB_SPEED_HIGH);
|
|
|
|
if (type == USB_DT_OTHER_SPEED_CONFIG)
|
|
|
|
hs = !hs;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
if (hs) {
|
|
|
|
dev->req->buf = dev->hs_config;
|
2007-05-26 11:40:14 +08:00
|
|
|
len = le16_to_cpu(dev->hs_config->wTotalLength);
|
2007-08-02 14:58:22 +08:00
|
|
|
} else {
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->req->buf = dev->config;
|
2007-05-26 11:40:14 +08:00
|
|
|
len = le16_to_cpu(dev->config->wTotalLength);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
((u8 *)dev->req->buf) [1] = type;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
gadgetfs_setup (struct usb_gadget *gadget, const struct usb_ctrlrequest *ctrl)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = get_gadget_data (gadget);
|
|
|
|
struct usb_request *req = dev->req;
|
|
|
|
int value = -EOPNOTSUPP;
|
|
|
|
struct usb_gadgetfs_event *event;
|
2005-05-08 04:05:13 +08:00
|
|
|
u16 w_value = le16_to_cpu(ctrl->wValue);
|
|
|
|
u16 w_length = le16_to_cpu(ctrl->wLength);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
spin_lock (&dev->lock);
|
|
|
|
dev->setup_abort = 0;
|
2007-01-17 14:51:04 +08:00
|
|
|
if (dev->state == STATE_DEV_UNCONNECTED) {
|
2007-08-02 14:58:22 +08:00
|
|
|
if (gadget_is_dualspeed(gadget)
|
|
|
|
&& gadget->speed == USB_SPEED_HIGH
|
|
|
|
&& dev->hs_config == NULL) {
|
2007-01-17 15:06:07 +08:00
|
|
|
spin_unlock(&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
ERROR (dev, "no high speed config??\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2007-01-17 15:06:07 +08:00
|
|
|
dev->state = STATE_DEV_CONNECTED;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
INFO (dev, "connected\n");
|
|
|
|
event = next_event (dev, GADGETFS_CONNECT);
|
|
|
|
event->u.speed = gadget->speed;
|
|
|
|
ep0_readable (dev);
|
|
|
|
|
|
|
|
/* host may have given up waiting for response. we can miss control
|
|
|
|
* requests handled lower down (device/endpoint status and features);
|
|
|
|
* then ep0_{read,write} will report the wrong status. controller
|
|
|
|
* driver will have aborted pending i/o.
|
|
|
|
*/
|
2007-01-17 14:51:04 +08:00
|
|
|
} else if (dev->state == STATE_DEV_SETUP)
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->setup_abort = 1;
|
|
|
|
|
|
|
|
req->buf = dev->rbuf;
|
|
|
|
req->context = NULL;
|
|
|
|
value = -EOPNOTSUPP;
|
|
|
|
switch (ctrl->bRequest) {
|
|
|
|
|
|
|
|
case USB_REQ_GET_DESCRIPTOR:
|
|
|
|
if (ctrl->bRequestType != USB_DIR_IN)
|
|
|
|
goto unrecognized;
|
|
|
|
switch (w_value >> 8) {
|
|
|
|
|
|
|
|
case USB_DT_DEVICE:
|
|
|
|
value = min (w_length, (u16) sizeof *dev->dev);
|
2011-06-23 20:26:11 +08:00
|
|
|
dev->dev->bMaxPacketSize0 = dev->gadget->ep0->maxpacket;
|
2005-04-17 06:20:36 +08:00
|
|
|
req->buf = dev->dev;
|
|
|
|
break;
|
|
|
|
case USB_DT_DEVICE_QUALIFIER:
|
|
|
|
if (!dev->hs_config)
|
|
|
|
break;
|
|
|
|
value = min (w_length, (u16)
|
|
|
|
sizeof (struct usb_qualifier_descriptor));
|
|
|
|
make_qualifier (dev);
|
|
|
|
break;
|
|
|
|
case USB_DT_OTHER_SPEED_CONFIG:
|
|
|
|
// FALLTHROUGH
|
|
|
|
case USB_DT_CONFIG:
|
|
|
|
value = config_buf (dev,
|
|
|
|
w_value >> 8,
|
|
|
|
w_value & 0xff);
|
|
|
|
if (value >= 0)
|
|
|
|
value = min (w_length, (u16) value);
|
|
|
|
break;
|
|
|
|
case USB_DT_STRING:
|
|
|
|
goto unrecognized;
|
|
|
|
|
|
|
|
default: // all others are errors
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* currently one config, two speeds */
|
|
|
|
case USB_REQ_SET_CONFIGURATION:
|
|
|
|
if (ctrl->bRequestType != 0)
|
2008-03-13 05:55:31 +08:00
|
|
|
goto unrecognized;
|
2005-04-17 06:20:36 +08:00
|
|
|
if (0 == (u8) w_value) {
|
|
|
|
value = 0;
|
|
|
|
dev->current_config = 0;
|
|
|
|
usb_gadget_vbus_draw(gadget, 8 /* mA */ );
|
|
|
|
// user mode expected to disable endpoints
|
|
|
|
} else {
|
|
|
|
u8 config, power;
|
2007-08-02 14:58:22 +08:00
|
|
|
|
|
|
|
if (gadget_is_dualspeed(gadget)
|
|
|
|
&& gadget->speed == USB_SPEED_HIGH) {
|
2005-04-17 06:20:36 +08:00
|
|
|
config = dev->hs_config->bConfigurationValue;
|
|
|
|
power = dev->hs_config->bMaxPower;
|
2007-08-02 14:58:22 +08:00
|
|
|
} else {
|
2005-04-17 06:20:36 +08:00
|
|
|
config = dev->config->bConfigurationValue;
|
|
|
|
power = dev->config->bMaxPower;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config == (u8) w_value) {
|
|
|
|
value = 0;
|
|
|
|
dev->current_config = config;
|
|
|
|
usb_gadget_vbus_draw(gadget, 2 * power);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* report SET_CONFIGURATION like any other control request,
|
|
|
|
* except that usermode may not stall this. the next
|
|
|
|
* request mustn't be allowed start until this finishes:
|
|
|
|
* endpoints and threads set up, etc.
|
|
|
|
*
|
|
|
|
* NOTE: older PXA hardware (before PXA 255: without UDCCFR)
|
|
|
|
* has bad/racey automagic that prevents synchronizing here.
|
|
|
|
* even kernel mode drivers often miss them.
|
|
|
|
*/
|
|
|
|
if (value == 0) {
|
|
|
|
INFO (dev, "configuration #%d\n", dev->current_config);
|
2014-04-29 13:26:28 +08:00
|
|
|
usb_gadget_set_state(gadget, USB_STATE_CONFIGURED);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (dev->usermode_setup) {
|
|
|
|
dev->setup_can_stall = 0;
|
|
|
|
goto delegate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2014-05-27 05:37:09 +08:00
|
|
|
#ifndef CONFIG_USB_PXA25X
|
2005-04-17 06:20:36 +08:00
|
|
|
/* PXA automagically handles this request too */
|
|
|
|
case USB_REQ_GET_CONFIGURATION:
|
|
|
|
if (ctrl->bRequestType != 0x80)
|
2008-03-13 05:55:31 +08:00
|
|
|
goto unrecognized;
|
2005-04-17 06:20:36 +08:00
|
|
|
*(u8 *)req->buf = dev->current_config;
|
|
|
|
value = min (w_length, (u16) 1);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
|
|
|
unrecognized:
|
|
|
|
VDEBUG (dev, "%s req%02x.%02x v%04x i%04x l%d\n",
|
|
|
|
dev->usermode_setup ? "delegate" : "fail",
|
|
|
|
ctrl->bRequestType, ctrl->bRequest,
|
|
|
|
w_value, le16_to_cpu(ctrl->wIndex), w_length);
|
|
|
|
|
|
|
|
/* if there's an ep0 reader, don't stall */
|
|
|
|
if (dev->usermode_setup) {
|
|
|
|
dev->setup_can_stall = 1;
|
|
|
|
delegate:
|
|
|
|
dev->setup_in = (ctrl->bRequestType & USB_DIR_IN)
|
|
|
|
? 1 : 0;
|
2006-01-03 23:30:31 +08:00
|
|
|
dev->setup_wLength = w_length;
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->setup_out_ready = 0;
|
|
|
|
dev->setup_out_error = 0;
|
|
|
|
value = 0;
|
|
|
|
|
|
|
|
/* read DATA stage for OUT right away */
|
|
|
|
if (unlikely (!dev->setup_in && w_length)) {
|
|
|
|
value = setup_req (gadget->ep0, dev->req,
|
|
|
|
w_length);
|
|
|
|
if (value < 0)
|
|
|
|
break;
|
2016-05-27 00:43:45 +08:00
|
|
|
|
|
|
|
spin_unlock (&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
value = usb_ep_queue (gadget->ep0, dev->req,
|
2016-05-27 00:43:45 +08:00
|
|
|
GFP_KERNEL);
|
|
|
|
spin_lock (&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (value < 0) {
|
|
|
|
clean_req (gadget->ep0, dev->req);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we can't currently stall these */
|
|
|
|
dev->setup_can_stall = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* state changes when reader collects event */
|
|
|
|
event = next_event (dev, GADGETFS_SETUP);
|
|
|
|
event->u.setup = *ctrl;
|
|
|
|
ep0_readable (dev);
|
|
|
|
spin_unlock (&dev->lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* proceed with data transfer and status phases? */
|
2007-01-17 14:51:04 +08:00
|
|
|
if (value >= 0 && dev->state != STATE_DEV_SETUP) {
|
2005-04-17 06:20:36 +08:00
|
|
|
req->length = value;
|
|
|
|
req->zero = value < w_length;
|
2016-05-27 00:43:45 +08:00
|
|
|
|
|
|
|
spin_unlock (&dev->lock);
|
|
|
|
value = usb_ep_queue (gadget->ep0, req, GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (value < 0) {
|
|
|
|
DBG (dev, "ep_queue --> %d\n", value);
|
|
|
|
req->status = 0;
|
|
|
|
}
|
2016-05-27 00:43:45 +08:00
|
|
|
return value;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* device stalls when value < 0 */
|
|
|
|
spin_unlock (&dev->lock);
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_ep_files (struct dev_data *dev)
|
|
|
|
{
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "%s %d\n", __func__, dev->state);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* dev->state must prevent interference */
|
|
|
|
spin_lock_irq (&dev->lock);
|
2012-01-09 05:13:28 +08:00
|
|
|
while (!list_empty(&dev->epfiles)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
struct ep_data *ep;
|
|
|
|
struct inode *parent;
|
|
|
|
struct dentry *dentry;
|
|
|
|
|
|
|
|
/* break link to FS */
|
2012-01-09 05:13:28 +08:00
|
|
|
ep = list_first_entry (&dev->epfiles, struct ep_data, epfiles);
|
2005-04-17 06:20:36 +08:00
|
|
|
list_del_init (&ep->epfiles);
|
|
|
|
dentry = ep->dentry;
|
|
|
|
ep->dentry = NULL;
|
2015-03-18 06:26:12 +08:00
|
|
|
parent = d_inode(dentry->d_parent);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* break link to controller */
|
|
|
|
if (ep->state == STATE_EP_ENABLED)
|
|
|
|
(void) usb_ep_disable (ep->ep);
|
|
|
|
ep->state = STATE_EP_UNBOUND;
|
|
|
|
usb_ep_free_request (ep->ep, ep->req);
|
|
|
|
ep->ep = NULL;
|
|
|
|
wake_up (&ep->wait);
|
|
|
|
put_ep (ep);
|
|
|
|
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
|
|
|
|
/* break link to dcache */
|
2016-01-23 04:40:57 +08:00
|
|
|
inode_lock(parent);
|
2005-04-17 06:20:36 +08:00
|
|
|
d_delete (dentry);
|
|
|
|
dput (dentry);
|
2016-01-23 04:40:57 +08:00
|
|
|
inode_unlock(parent);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2012-01-09 05:13:28 +08:00
|
|
|
spin_lock_irq (&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-04 01:37:56 +08:00
|
|
|
static struct dentry *
|
2005-04-17 06:20:36 +08:00
|
|
|
gadgetfs_create_file (struct super_block *sb, char const *name,
|
2014-09-04 01:37:56 +08:00
|
|
|
void *data, const struct file_operations *fops);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
static int activate_ep_files (struct dev_data *dev)
|
|
|
|
{
|
|
|
|
struct usb_ep *ep;
|
2006-05-23 00:27:38 +08:00
|
|
|
struct ep_data *data;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
gadget_for_each_ep (ep, dev->gadget) {
|
|
|
|
|
2006-02-28 05:34:10 +08:00
|
|
|
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!data)
|
2006-05-23 00:27:38 +08:00
|
|
|
goto enomem0;
|
2005-04-17 06:20:36 +08:00
|
|
|
data->state = STATE_EP_DISABLED;
|
2010-01-30 04:38:59 +08:00
|
|
|
mutex_init(&data->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
init_waitqueue_head (&data->wait);
|
|
|
|
|
|
|
|
strncpy (data->name, ep->name, sizeof (data->name) - 1);
|
2017-03-06 22:21:14 +08:00
|
|
|
refcount_set (&data->count, 1);
|
2005-04-17 06:20:36 +08:00
|
|
|
data->dev = dev;
|
|
|
|
get_dev (dev);
|
|
|
|
|
|
|
|
data->ep = ep;
|
|
|
|
ep->driver_data = data;
|
|
|
|
|
|
|
|
data->req = usb_ep_alloc_request (ep, GFP_KERNEL);
|
|
|
|
if (!data->req)
|
2006-05-23 00:27:38 +08:00
|
|
|
goto enomem1;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2014-09-04 01:37:56 +08:00
|
|
|
data->dentry = gadgetfs_create_file (dev->sb, data->name,
|
2015-03-03 16:39:34 +08:00
|
|
|
data, &ep_io_operations);
|
2014-09-04 01:37:56 +08:00
|
|
|
if (!data->dentry)
|
2006-05-23 00:27:38 +08:00
|
|
|
goto enomem2;
|
2005-04-17 06:20:36 +08:00
|
|
|
list_add_tail (&data->epfiles, &dev->epfiles);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
2006-05-23 00:27:38 +08:00
|
|
|
enomem2:
|
|
|
|
usb_ep_free_request (ep, data->req);
|
|
|
|
enomem1:
|
|
|
|
put_dev (dev);
|
|
|
|
kfree (data);
|
|
|
|
enomem0:
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "%s enomem\n", __func__);
|
2005-04-17 06:20:36 +08:00
|
|
|
destroy_ep_files (dev);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gadgetfs_unbind (struct usb_gadget *gadget)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = get_gadget_data (gadget);
|
|
|
|
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "%s\n", __func__);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
dev->state = STATE_DEV_UNBOUND;
|
|
|
|
spin_unlock_irq (&dev->lock);
|
|
|
|
|
|
|
|
destroy_ep_files (dev);
|
|
|
|
gadget->ep0->driver_data = NULL;
|
|
|
|
set_gadget_data (gadget, NULL);
|
|
|
|
|
|
|
|
/* we've already been disconnected ... no i/o is active */
|
|
|
|
if (dev->req)
|
|
|
|
usb_ep_free_request (gadget->ep0, dev->req);
|
2008-03-04 08:08:34 +08:00
|
|
|
DBG (dev, "%s done\n", __func__);
|
2005-04-17 06:20:36 +08:00
|
|
|
put_dev (dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct dev_data *the_device;
|
|
|
|
|
2012-09-07 15:53:17 +08:00
|
|
|
static int gadgetfs_bind(struct usb_gadget *gadget,
|
|
|
|
struct usb_gadget_driver *driver)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct dev_data *dev = the_device;
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
return -ESRCH;
|
|
|
|
if (0 != strcmp (CHIP, gadget->name)) {
|
2007-11-20 04:58:36 +08:00
|
|
|
pr_err("%s expected %s controller not %s\n",
|
2005-04-17 06:20:36 +08:00
|
|
|
shortname, CHIP, gadget->name);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
set_gadget_data (gadget, dev);
|
|
|
|
dev->gadget = gadget;
|
|
|
|
gadget->ep0->driver_data = dev;
|
|
|
|
|
|
|
|
/* preallocate control response and buffer */
|
|
|
|
dev->req = usb_ep_alloc_request (gadget->ep0, GFP_KERNEL);
|
|
|
|
if (!dev->req)
|
|
|
|
goto enomem;
|
|
|
|
dev->req->context = NULL;
|
|
|
|
dev->req->complete = epio_complete;
|
|
|
|
|
|
|
|
if (activate_ep_files (dev) < 0)
|
|
|
|
goto enomem;
|
|
|
|
|
|
|
|
INFO (dev, "bound to %s driver\n", gadget->name);
|
2007-01-17 14:51:04 +08:00
|
|
|
spin_lock_irq(&dev->lock);
|
|
|
|
dev->state = STATE_DEV_UNCONNECTED;
|
|
|
|
spin_unlock_irq(&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
get_dev (dev);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
enomem:
|
|
|
|
gadgetfs_unbind (gadget);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gadgetfs_disconnect (struct usb_gadget *gadget)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = get_gadget_data (gadget);
|
2011-10-11 00:38:05 +08:00
|
|
|
unsigned long flags;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-10-11 00:38:05 +08:00
|
|
|
spin_lock_irqsave (&dev->lock, flags);
|
2007-01-17 14:51:04 +08:00
|
|
|
if (dev->state == STATE_DEV_UNCONNECTED)
|
2006-06-26 22:46:00 +08:00
|
|
|
goto exit;
|
2007-01-17 14:51:04 +08:00
|
|
|
dev->state = STATE_DEV_UNCONNECTED;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
INFO (dev, "disconnected\n");
|
|
|
|
next_event (dev, GADGETFS_DISCONNECT);
|
|
|
|
ep0_readable (dev);
|
2006-06-26 22:46:00 +08:00
|
|
|
exit:
|
2011-10-11 00:38:05 +08:00
|
|
|
spin_unlock_irqrestore (&dev->lock, flags);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gadgetfs_suspend (struct usb_gadget *gadget)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = get_gadget_data (gadget);
|
USB: gadgetfs, dummy-hcd, net2280: fix locking for callbacks
Using the syzkaller kernel fuzzer, Andrey Konovalov generated the
following error in gadgetfs:
> BUG: KASAN: use-after-free in __lock_acquire+0x3069/0x3690
> kernel/locking/lockdep.c:3246
> Read of size 8 at addr ffff88003a2bdaf8 by task kworker/3:1/903
>
> CPU: 3 PID: 903 Comm: kworker/3:1 Not tainted 4.12.0-rc4+ #35
> Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS Bochs 01/01/2011
> Workqueue: usb_hub_wq hub_event
> Call Trace:
> __dump_stack lib/dump_stack.c:16 [inline]
> dump_stack+0x292/0x395 lib/dump_stack.c:52
> print_address_description+0x78/0x280 mm/kasan/report.c:252
> kasan_report_error mm/kasan/report.c:351 [inline]
> kasan_report+0x230/0x340 mm/kasan/report.c:408
> __asan_report_load8_noabort+0x19/0x20 mm/kasan/report.c:429
> __lock_acquire+0x3069/0x3690 kernel/locking/lockdep.c:3246
> lock_acquire+0x22d/0x560 kernel/locking/lockdep.c:3855
> __raw_spin_lock include/linux/spinlock_api_smp.h:142 [inline]
> _raw_spin_lock+0x2f/0x40 kernel/locking/spinlock.c:151
> spin_lock include/linux/spinlock.h:299 [inline]
> gadgetfs_suspend+0x89/0x130 drivers/usb/gadget/legacy/inode.c:1682
> set_link_state+0x88e/0xae0 drivers/usb/gadget/udc/dummy_hcd.c:455
> dummy_hub_control+0xd7e/0x1fb0 drivers/usb/gadget/udc/dummy_hcd.c:2074
> rh_call_control drivers/usb/core/hcd.c:689 [inline]
> rh_urb_enqueue drivers/usb/core/hcd.c:846 [inline]
> usb_hcd_submit_urb+0x92f/0x20b0 drivers/usb/core/hcd.c:1650
> usb_submit_urb+0x8b2/0x12c0 drivers/usb/core/urb.c:542
> usb_start_wait_urb+0x148/0x5b0 drivers/usb/core/message.c:56
> usb_internal_control_msg drivers/usb/core/message.c:100 [inline]
> usb_control_msg+0x341/0x4d0 drivers/usb/core/message.c:151
> usb_clear_port_feature+0x74/0xa0 drivers/usb/core/hub.c:412
> hub_port_disable+0x123/0x510 drivers/usb/core/hub.c:4177
> hub_port_init+0x1ed/0x2940 drivers/usb/core/hub.c:4648
> hub_port_connect drivers/usb/core/hub.c:4826 [inline]
> hub_port_connect_change drivers/usb/core/hub.c:4999 [inline]
> port_event drivers/usb/core/hub.c:5105 [inline]
> hub_event+0x1ae1/0x3d40 drivers/usb/core/hub.c:5185
> process_one_work+0xc08/0x1bd0 kernel/workqueue.c:2097
> process_scheduled_works kernel/workqueue.c:2157 [inline]
> worker_thread+0xb2b/0x1860 kernel/workqueue.c:2233
> kthread+0x363/0x440 kernel/kthread.c:231
> ret_from_fork+0x2a/0x40 arch/x86/entry/entry_64.S:424
>
> Allocated by task 9958:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_kmalloc+0xad/0xe0 mm/kasan/kasan.c:617
> kmem_cache_alloc_trace+0x87/0x280 mm/slub.c:2745
> kmalloc include/linux/slab.h:492 [inline]
> kzalloc include/linux/slab.h:665 [inline]
> dev_new drivers/usb/gadget/legacy/inode.c:170 [inline]
> gadgetfs_fill_super+0x24f/0x540 drivers/usb/gadget/legacy/inode.c:1993
> mount_single+0xf6/0x160 fs/super.c:1192
> gadgetfs_mount+0x31/0x40 drivers/usb/gadget/legacy/inode.c:2019
> mount_fs+0x9c/0x2d0 fs/super.c:1223
> vfs_kern_mount.part.25+0xcb/0x490 fs/namespace.c:976
> vfs_kern_mount fs/namespace.c:2509 [inline]
> do_new_mount fs/namespace.c:2512 [inline]
> do_mount+0x41b/0x2d90 fs/namespace.c:2834
> SYSC_mount fs/namespace.c:3050 [inline]
> SyS_mount+0xb0/0x120 fs/namespace.c:3027
> entry_SYSCALL_64_fastpath+0x1f/0xbe
>
> Freed by task 9960:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_slab_free+0x72/0xc0 mm/kasan/kasan.c:590
> slab_free_hook mm/slub.c:1357 [inline]
> slab_free_freelist_hook mm/slub.c:1379 [inline]
> slab_free mm/slub.c:2961 [inline]
> kfree+0xed/0x2b0 mm/slub.c:3882
> put_dev+0x124/0x160 drivers/usb/gadget/legacy/inode.c:163
> gadgetfs_kill_sb+0x33/0x60 drivers/usb/gadget/legacy/inode.c:2027
> deactivate_locked_super+0x8d/0xd0 fs/super.c:309
> deactivate_super+0x21e/0x310 fs/super.c:340
> cleanup_mnt+0xb7/0x150 fs/namespace.c:1112
> __cleanup_mnt+0x1b/0x20 fs/namespace.c:1119
> task_work_run+0x1a0/0x280 kernel/task_work.c:116
> exit_task_work include/linux/task_work.h:21 [inline]
> do_exit+0x18a8/0x2820 kernel/exit.c:878
> do_group_exit+0x14e/0x420 kernel/exit.c:982
> get_signal+0x784/0x1780 kernel/signal.c:2318
> do_signal+0xd7/0x2130 arch/x86/kernel/signal.c:808
> exit_to_usermode_loop+0x1ac/0x240 arch/x86/entry/common.c:157
> prepare_exit_to_usermode arch/x86/entry/common.c:194 [inline]
> syscall_return_slowpath+0x3ba/0x410 arch/x86/entry/common.c:263
> entry_SYSCALL_64_fastpath+0xbc/0xbe
>
> The buggy address belongs to the object at ffff88003a2bdae0
> which belongs to the cache kmalloc-1024 of size 1024
> The buggy address is located 24 bytes inside of
> 1024-byte region [ffff88003a2bdae0, ffff88003a2bdee0)
> The buggy address belongs to the page:
> page:ffffea0000e8ae00 count:1 mapcount:0 mapping: (null)
> index:0x0 compound_mapcount: 0
> flags: 0x100000000008100(slab|head)
> raw: 0100000000008100 0000000000000000 0000000000000000 0000000100170017
> raw: ffffea0000ed3020 ffffea0000f5f820 ffff88003e80efc0 0000000000000000
> page dumped because: kasan: bad access detected
>
> Memory state around the buggy address:
> ffff88003a2bd980: fb fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> ffff88003a2bda00: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> >ffff88003a2bda80: fc fc fc fc fc fc fc fc fc fc fc fc fb fb fb fb
> ^
> ffff88003a2bdb00: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ffff88003a2bdb80: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ==================================================================
What this means is that the gadgetfs_suspend() routine was trying to
access dev->lock after it had been deallocated. The root cause is a
race in the dummy_hcd driver; the dummy_udc_stop() routine can race
with the rest of the driver because it contains no locking. And even
when proper locking is added, it can still race with the
set_link_state() function because that function incorrectly drops the
private spinlock before invoking any gadget driver callbacks.
The result of this race, as seen above, is that set_link_state() can
invoke a callback in gadgetfs even after gadgetfs has been unbound
from dummy_hcd's UDC and its private data structures have been
deallocated.
include/linux/usb/gadget.h documents that the ->reset, ->disconnect,
->suspend, and ->resume callbacks may be invoked in interrupt context.
In general this is necessary, to prevent races with gadget driver
removal. This patch fixes dummy_hcd to retain the spinlock across
these calls, and it adds a spinlock acquisition to dummy_udc_stop() to
prevent the race.
The net2280 driver makes the same mistake of dropping the private
spinlock for its ->disconnect and ->reset callback invocations. The
patch fixes it too.
Lastly, since gadgetfs_suspend() may be invoked in interrupt context,
it cannot assume that interrupts are enabled when it runs. It must
use spin_lock_irqsave() instead of spin_lock_irq(). The patch fixes
that bug as well.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Reported-and-tested-by: Andrey Konovalov <andreyknvl@google.com>
CC: <stable@vger.kernel.org>
Acked-by: Felipe Balbi <felipe.balbi@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2017-06-14 03:23:42 +08:00
|
|
|
unsigned long flags;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
INFO (dev, "suspended from state %d\n", dev->state);
|
USB: gadgetfs, dummy-hcd, net2280: fix locking for callbacks
Using the syzkaller kernel fuzzer, Andrey Konovalov generated the
following error in gadgetfs:
> BUG: KASAN: use-after-free in __lock_acquire+0x3069/0x3690
> kernel/locking/lockdep.c:3246
> Read of size 8 at addr ffff88003a2bdaf8 by task kworker/3:1/903
>
> CPU: 3 PID: 903 Comm: kworker/3:1 Not tainted 4.12.0-rc4+ #35
> Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS Bochs 01/01/2011
> Workqueue: usb_hub_wq hub_event
> Call Trace:
> __dump_stack lib/dump_stack.c:16 [inline]
> dump_stack+0x292/0x395 lib/dump_stack.c:52
> print_address_description+0x78/0x280 mm/kasan/report.c:252
> kasan_report_error mm/kasan/report.c:351 [inline]
> kasan_report+0x230/0x340 mm/kasan/report.c:408
> __asan_report_load8_noabort+0x19/0x20 mm/kasan/report.c:429
> __lock_acquire+0x3069/0x3690 kernel/locking/lockdep.c:3246
> lock_acquire+0x22d/0x560 kernel/locking/lockdep.c:3855
> __raw_spin_lock include/linux/spinlock_api_smp.h:142 [inline]
> _raw_spin_lock+0x2f/0x40 kernel/locking/spinlock.c:151
> spin_lock include/linux/spinlock.h:299 [inline]
> gadgetfs_suspend+0x89/0x130 drivers/usb/gadget/legacy/inode.c:1682
> set_link_state+0x88e/0xae0 drivers/usb/gadget/udc/dummy_hcd.c:455
> dummy_hub_control+0xd7e/0x1fb0 drivers/usb/gadget/udc/dummy_hcd.c:2074
> rh_call_control drivers/usb/core/hcd.c:689 [inline]
> rh_urb_enqueue drivers/usb/core/hcd.c:846 [inline]
> usb_hcd_submit_urb+0x92f/0x20b0 drivers/usb/core/hcd.c:1650
> usb_submit_urb+0x8b2/0x12c0 drivers/usb/core/urb.c:542
> usb_start_wait_urb+0x148/0x5b0 drivers/usb/core/message.c:56
> usb_internal_control_msg drivers/usb/core/message.c:100 [inline]
> usb_control_msg+0x341/0x4d0 drivers/usb/core/message.c:151
> usb_clear_port_feature+0x74/0xa0 drivers/usb/core/hub.c:412
> hub_port_disable+0x123/0x510 drivers/usb/core/hub.c:4177
> hub_port_init+0x1ed/0x2940 drivers/usb/core/hub.c:4648
> hub_port_connect drivers/usb/core/hub.c:4826 [inline]
> hub_port_connect_change drivers/usb/core/hub.c:4999 [inline]
> port_event drivers/usb/core/hub.c:5105 [inline]
> hub_event+0x1ae1/0x3d40 drivers/usb/core/hub.c:5185
> process_one_work+0xc08/0x1bd0 kernel/workqueue.c:2097
> process_scheduled_works kernel/workqueue.c:2157 [inline]
> worker_thread+0xb2b/0x1860 kernel/workqueue.c:2233
> kthread+0x363/0x440 kernel/kthread.c:231
> ret_from_fork+0x2a/0x40 arch/x86/entry/entry_64.S:424
>
> Allocated by task 9958:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_kmalloc+0xad/0xe0 mm/kasan/kasan.c:617
> kmem_cache_alloc_trace+0x87/0x280 mm/slub.c:2745
> kmalloc include/linux/slab.h:492 [inline]
> kzalloc include/linux/slab.h:665 [inline]
> dev_new drivers/usb/gadget/legacy/inode.c:170 [inline]
> gadgetfs_fill_super+0x24f/0x540 drivers/usb/gadget/legacy/inode.c:1993
> mount_single+0xf6/0x160 fs/super.c:1192
> gadgetfs_mount+0x31/0x40 drivers/usb/gadget/legacy/inode.c:2019
> mount_fs+0x9c/0x2d0 fs/super.c:1223
> vfs_kern_mount.part.25+0xcb/0x490 fs/namespace.c:976
> vfs_kern_mount fs/namespace.c:2509 [inline]
> do_new_mount fs/namespace.c:2512 [inline]
> do_mount+0x41b/0x2d90 fs/namespace.c:2834
> SYSC_mount fs/namespace.c:3050 [inline]
> SyS_mount+0xb0/0x120 fs/namespace.c:3027
> entry_SYSCALL_64_fastpath+0x1f/0xbe
>
> Freed by task 9960:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_slab_free+0x72/0xc0 mm/kasan/kasan.c:590
> slab_free_hook mm/slub.c:1357 [inline]
> slab_free_freelist_hook mm/slub.c:1379 [inline]
> slab_free mm/slub.c:2961 [inline]
> kfree+0xed/0x2b0 mm/slub.c:3882
> put_dev+0x124/0x160 drivers/usb/gadget/legacy/inode.c:163
> gadgetfs_kill_sb+0x33/0x60 drivers/usb/gadget/legacy/inode.c:2027
> deactivate_locked_super+0x8d/0xd0 fs/super.c:309
> deactivate_super+0x21e/0x310 fs/super.c:340
> cleanup_mnt+0xb7/0x150 fs/namespace.c:1112
> __cleanup_mnt+0x1b/0x20 fs/namespace.c:1119
> task_work_run+0x1a0/0x280 kernel/task_work.c:116
> exit_task_work include/linux/task_work.h:21 [inline]
> do_exit+0x18a8/0x2820 kernel/exit.c:878
> do_group_exit+0x14e/0x420 kernel/exit.c:982
> get_signal+0x784/0x1780 kernel/signal.c:2318
> do_signal+0xd7/0x2130 arch/x86/kernel/signal.c:808
> exit_to_usermode_loop+0x1ac/0x240 arch/x86/entry/common.c:157
> prepare_exit_to_usermode arch/x86/entry/common.c:194 [inline]
> syscall_return_slowpath+0x3ba/0x410 arch/x86/entry/common.c:263
> entry_SYSCALL_64_fastpath+0xbc/0xbe
>
> The buggy address belongs to the object at ffff88003a2bdae0
> which belongs to the cache kmalloc-1024 of size 1024
> The buggy address is located 24 bytes inside of
> 1024-byte region [ffff88003a2bdae0, ffff88003a2bdee0)
> The buggy address belongs to the page:
> page:ffffea0000e8ae00 count:1 mapcount:0 mapping: (null)
> index:0x0 compound_mapcount: 0
> flags: 0x100000000008100(slab|head)
> raw: 0100000000008100 0000000000000000 0000000000000000 0000000100170017
> raw: ffffea0000ed3020 ffffea0000f5f820 ffff88003e80efc0 0000000000000000
> page dumped because: kasan: bad access detected
>
> Memory state around the buggy address:
> ffff88003a2bd980: fb fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> ffff88003a2bda00: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> >ffff88003a2bda80: fc fc fc fc fc fc fc fc fc fc fc fc fb fb fb fb
> ^
> ffff88003a2bdb00: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ffff88003a2bdb80: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ==================================================================
What this means is that the gadgetfs_suspend() routine was trying to
access dev->lock after it had been deallocated. The root cause is a
race in the dummy_hcd driver; the dummy_udc_stop() routine can race
with the rest of the driver because it contains no locking. And even
when proper locking is added, it can still race with the
set_link_state() function because that function incorrectly drops the
private spinlock before invoking any gadget driver callbacks.
The result of this race, as seen above, is that set_link_state() can
invoke a callback in gadgetfs even after gadgetfs has been unbound
from dummy_hcd's UDC and its private data structures have been
deallocated.
include/linux/usb/gadget.h documents that the ->reset, ->disconnect,
->suspend, and ->resume callbacks may be invoked in interrupt context.
In general this is necessary, to prevent races with gadget driver
removal. This patch fixes dummy_hcd to retain the spinlock across
these calls, and it adds a spinlock acquisition to dummy_udc_stop() to
prevent the race.
The net2280 driver makes the same mistake of dropping the private
spinlock for its ->disconnect and ->reset callback invocations. The
patch fixes it too.
Lastly, since gadgetfs_suspend() may be invoked in interrupt context,
it cannot assume that interrupts are enabled when it runs. It must
use spin_lock_irqsave() instead of spin_lock_irq(). The patch fixes
that bug as well.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Reported-and-tested-by: Andrey Konovalov <andreyknvl@google.com>
CC: <stable@vger.kernel.org>
Acked-by: Felipe Balbi <felipe.balbi@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2017-06-14 03:23:42 +08:00
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
2005-04-17 06:20:36 +08:00
|
|
|
switch (dev->state) {
|
2007-01-17 14:51:04 +08:00
|
|
|
case STATE_DEV_SETUP: // VERY odd... host died??
|
|
|
|
case STATE_DEV_CONNECTED:
|
|
|
|
case STATE_DEV_UNCONNECTED:
|
2005-04-17 06:20:36 +08:00
|
|
|
next_event (dev, GADGETFS_SUSPEND);
|
|
|
|
ep0_readable (dev);
|
|
|
|
/* FALLTHROUGH */
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
USB: gadgetfs, dummy-hcd, net2280: fix locking for callbacks
Using the syzkaller kernel fuzzer, Andrey Konovalov generated the
following error in gadgetfs:
> BUG: KASAN: use-after-free in __lock_acquire+0x3069/0x3690
> kernel/locking/lockdep.c:3246
> Read of size 8 at addr ffff88003a2bdaf8 by task kworker/3:1/903
>
> CPU: 3 PID: 903 Comm: kworker/3:1 Not tainted 4.12.0-rc4+ #35
> Hardware name: QEMU Standard PC (i440FX + PIIX, 1996), BIOS Bochs 01/01/2011
> Workqueue: usb_hub_wq hub_event
> Call Trace:
> __dump_stack lib/dump_stack.c:16 [inline]
> dump_stack+0x292/0x395 lib/dump_stack.c:52
> print_address_description+0x78/0x280 mm/kasan/report.c:252
> kasan_report_error mm/kasan/report.c:351 [inline]
> kasan_report+0x230/0x340 mm/kasan/report.c:408
> __asan_report_load8_noabort+0x19/0x20 mm/kasan/report.c:429
> __lock_acquire+0x3069/0x3690 kernel/locking/lockdep.c:3246
> lock_acquire+0x22d/0x560 kernel/locking/lockdep.c:3855
> __raw_spin_lock include/linux/spinlock_api_smp.h:142 [inline]
> _raw_spin_lock+0x2f/0x40 kernel/locking/spinlock.c:151
> spin_lock include/linux/spinlock.h:299 [inline]
> gadgetfs_suspend+0x89/0x130 drivers/usb/gadget/legacy/inode.c:1682
> set_link_state+0x88e/0xae0 drivers/usb/gadget/udc/dummy_hcd.c:455
> dummy_hub_control+0xd7e/0x1fb0 drivers/usb/gadget/udc/dummy_hcd.c:2074
> rh_call_control drivers/usb/core/hcd.c:689 [inline]
> rh_urb_enqueue drivers/usb/core/hcd.c:846 [inline]
> usb_hcd_submit_urb+0x92f/0x20b0 drivers/usb/core/hcd.c:1650
> usb_submit_urb+0x8b2/0x12c0 drivers/usb/core/urb.c:542
> usb_start_wait_urb+0x148/0x5b0 drivers/usb/core/message.c:56
> usb_internal_control_msg drivers/usb/core/message.c:100 [inline]
> usb_control_msg+0x341/0x4d0 drivers/usb/core/message.c:151
> usb_clear_port_feature+0x74/0xa0 drivers/usb/core/hub.c:412
> hub_port_disable+0x123/0x510 drivers/usb/core/hub.c:4177
> hub_port_init+0x1ed/0x2940 drivers/usb/core/hub.c:4648
> hub_port_connect drivers/usb/core/hub.c:4826 [inline]
> hub_port_connect_change drivers/usb/core/hub.c:4999 [inline]
> port_event drivers/usb/core/hub.c:5105 [inline]
> hub_event+0x1ae1/0x3d40 drivers/usb/core/hub.c:5185
> process_one_work+0xc08/0x1bd0 kernel/workqueue.c:2097
> process_scheduled_works kernel/workqueue.c:2157 [inline]
> worker_thread+0xb2b/0x1860 kernel/workqueue.c:2233
> kthread+0x363/0x440 kernel/kthread.c:231
> ret_from_fork+0x2a/0x40 arch/x86/entry/entry_64.S:424
>
> Allocated by task 9958:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_kmalloc+0xad/0xe0 mm/kasan/kasan.c:617
> kmem_cache_alloc_trace+0x87/0x280 mm/slub.c:2745
> kmalloc include/linux/slab.h:492 [inline]
> kzalloc include/linux/slab.h:665 [inline]
> dev_new drivers/usb/gadget/legacy/inode.c:170 [inline]
> gadgetfs_fill_super+0x24f/0x540 drivers/usb/gadget/legacy/inode.c:1993
> mount_single+0xf6/0x160 fs/super.c:1192
> gadgetfs_mount+0x31/0x40 drivers/usb/gadget/legacy/inode.c:2019
> mount_fs+0x9c/0x2d0 fs/super.c:1223
> vfs_kern_mount.part.25+0xcb/0x490 fs/namespace.c:976
> vfs_kern_mount fs/namespace.c:2509 [inline]
> do_new_mount fs/namespace.c:2512 [inline]
> do_mount+0x41b/0x2d90 fs/namespace.c:2834
> SYSC_mount fs/namespace.c:3050 [inline]
> SyS_mount+0xb0/0x120 fs/namespace.c:3027
> entry_SYSCALL_64_fastpath+0x1f/0xbe
>
> Freed by task 9960:
> save_stack_trace+0x1b/0x20 arch/x86/kernel/stacktrace.c:59
> save_stack+0x43/0xd0 mm/kasan/kasan.c:513
> set_track mm/kasan/kasan.c:525 [inline]
> kasan_slab_free+0x72/0xc0 mm/kasan/kasan.c:590
> slab_free_hook mm/slub.c:1357 [inline]
> slab_free_freelist_hook mm/slub.c:1379 [inline]
> slab_free mm/slub.c:2961 [inline]
> kfree+0xed/0x2b0 mm/slub.c:3882
> put_dev+0x124/0x160 drivers/usb/gadget/legacy/inode.c:163
> gadgetfs_kill_sb+0x33/0x60 drivers/usb/gadget/legacy/inode.c:2027
> deactivate_locked_super+0x8d/0xd0 fs/super.c:309
> deactivate_super+0x21e/0x310 fs/super.c:340
> cleanup_mnt+0xb7/0x150 fs/namespace.c:1112
> __cleanup_mnt+0x1b/0x20 fs/namespace.c:1119
> task_work_run+0x1a0/0x280 kernel/task_work.c:116
> exit_task_work include/linux/task_work.h:21 [inline]
> do_exit+0x18a8/0x2820 kernel/exit.c:878
> do_group_exit+0x14e/0x420 kernel/exit.c:982
> get_signal+0x784/0x1780 kernel/signal.c:2318
> do_signal+0xd7/0x2130 arch/x86/kernel/signal.c:808
> exit_to_usermode_loop+0x1ac/0x240 arch/x86/entry/common.c:157
> prepare_exit_to_usermode arch/x86/entry/common.c:194 [inline]
> syscall_return_slowpath+0x3ba/0x410 arch/x86/entry/common.c:263
> entry_SYSCALL_64_fastpath+0xbc/0xbe
>
> The buggy address belongs to the object at ffff88003a2bdae0
> which belongs to the cache kmalloc-1024 of size 1024
> The buggy address is located 24 bytes inside of
> 1024-byte region [ffff88003a2bdae0, ffff88003a2bdee0)
> The buggy address belongs to the page:
> page:ffffea0000e8ae00 count:1 mapcount:0 mapping: (null)
> index:0x0 compound_mapcount: 0
> flags: 0x100000000008100(slab|head)
> raw: 0100000000008100 0000000000000000 0000000000000000 0000000100170017
> raw: ffffea0000ed3020 ffffea0000f5f820 ffff88003e80efc0 0000000000000000
> page dumped because: kasan: bad access detected
>
> Memory state around the buggy address:
> ffff88003a2bd980: fb fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> ffff88003a2bda00: fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc fc
> >ffff88003a2bda80: fc fc fc fc fc fc fc fc fc fc fc fc fb fb fb fb
> ^
> ffff88003a2bdb00: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ffff88003a2bdb80: fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb fb
> ==================================================================
What this means is that the gadgetfs_suspend() routine was trying to
access dev->lock after it had been deallocated. The root cause is a
race in the dummy_hcd driver; the dummy_udc_stop() routine can race
with the rest of the driver because it contains no locking. And even
when proper locking is added, it can still race with the
set_link_state() function because that function incorrectly drops the
private spinlock before invoking any gadget driver callbacks.
The result of this race, as seen above, is that set_link_state() can
invoke a callback in gadgetfs even after gadgetfs has been unbound
from dummy_hcd's UDC and its private data structures have been
deallocated.
include/linux/usb/gadget.h documents that the ->reset, ->disconnect,
->suspend, and ->resume callbacks may be invoked in interrupt context.
In general this is necessary, to prevent races with gadget driver
removal. This patch fixes dummy_hcd to retain the spinlock across
these calls, and it adds a spinlock acquisition to dummy_udc_stop() to
prevent the race.
The net2280 driver makes the same mistake of dropping the private
spinlock for its ->disconnect and ->reset callback invocations. The
patch fixes it too.
Lastly, since gadgetfs_suspend() may be invoked in interrupt context,
it cannot assume that interrupts are enabled when it runs. It must
use spin_lock_irqsave() instead of spin_lock_irq(). The patch fixes
that bug as well.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Reported-and-tested-by: Andrey Konovalov <andreyknvl@google.com>
CC: <stable@vger.kernel.org>
Acked-by: Felipe Balbi <felipe.balbi@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2017-06-14 03:23:42 +08:00
|
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_gadget_driver gadgetfs_driver = {
|
|
|
|
.function = (char *) driver_desc,
|
2012-09-07 02:11:05 +08:00
|
|
|
.bind = gadgetfs_bind,
|
2005-04-17 06:20:36 +08:00
|
|
|
.unbind = gadgetfs_unbind,
|
|
|
|
.setup = gadgetfs_setup,
|
2014-09-09 08:56:51 +08:00
|
|
|
.reset = gadgetfs_disconnect,
|
2005-04-17 06:20:36 +08:00
|
|
|
.disconnect = gadgetfs_disconnect,
|
|
|
|
.suspend = gadgetfs_suspend,
|
|
|
|
|
2007-01-16 03:30:28 +08:00
|
|
|
.driver = {
|
2005-04-17 06:20:36 +08:00
|
|
|
.name = (char *) shortname,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
/* DEVICE INITIALIZATION
|
|
|
|
*
|
|
|
|
* fd = open ("/dev/gadget/$CHIP", O_RDWR)
|
|
|
|
* status = write (fd, descriptors, sizeof descriptors)
|
|
|
|
*
|
|
|
|
* That write establishes the device configuration, so the kernel can
|
|
|
|
* bind to the controller ... guaranteeing it can handle enumeration
|
|
|
|
* at all necessary speeds. Descriptor order is:
|
|
|
|
*
|
|
|
|
* . message tag (u32, host order) ... for now, must be zero; it
|
|
|
|
* would change to support features like multi-config devices
|
|
|
|
* . full/low speed config ... all wTotalLength bytes (with interface,
|
|
|
|
* class, altsetting, endpoint, and other descriptors)
|
|
|
|
* . high speed config ... all descriptors, for high speed operation;
|
2007-01-16 03:30:28 +08:00
|
|
|
* this one's optional except for high-speed hardware
|
2005-04-17 06:20:36 +08:00
|
|
|
* . device descriptor
|
|
|
|
*
|
2007-01-16 03:35:01 +08:00
|
|
|
* Endpoints are not yet enabled. Drivers must wait until device
|
|
|
|
* configuration and interface altsetting changes create
|
2005-04-17 06:20:36 +08:00
|
|
|
* the need to configure (or unconfigure) them.
|
|
|
|
*
|
|
|
|
* After initialization, the device stays active for as long as that
|
2007-01-16 03:35:01 +08:00
|
|
|
* $CHIP file is open. Events must then be read from that descriptor,
|
|
|
|
* such as configuration notifications.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
|
2016-12-10 04:24:24 +08:00
|
|
|
static int is_valid_config(struct usb_config_descriptor *config,
|
|
|
|
unsigned int total)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
return config->bDescriptorType == USB_DT_CONFIG
|
|
|
|
&& config->bLength == USB_DT_CONFIG_SIZE
|
2016-12-10 04:24:24 +08:00
|
|
|
&& total >= USB_DT_CONFIG_SIZE
|
2005-04-17 06:20:36 +08:00
|
|
|
&& config->bConfigurationValue != 0
|
|
|
|
&& (config->bmAttributes & USB_CONFIG_ATT_ONE) != 0
|
|
|
|
&& (config->bmAttributes & USB_CONFIG_ATT_WAKEUP) == 0;
|
|
|
|
/* FIXME if gadget->is_otg, _must_ include an otg descriptor */
|
|
|
|
/* FIXME check lengths: walk to end */
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
dev_config (struct file *fd, const char __user *buf, size_t len, loff_t *ptr)
|
|
|
|
{
|
|
|
|
struct dev_data *dev = fd->private_data;
|
|
|
|
ssize_t value = len, length = len;
|
|
|
|
unsigned total;
|
|
|
|
u32 tag;
|
|
|
|
char *kbuf;
|
|
|
|
|
2015-03-04 23:31:50 +08:00
|
|
|
spin_lock_irq(&dev->lock);
|
|
|
|
if (dev->state > STATE_DEV_OPENED) {
|
|
|
|
value = ep0_write(fd, buf, len, ptr);
|
|
|
|
spin_unlock_irq(&dev->lock);
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&dev->lock);
|
|
|
|
|
2016-12-06 15:36:29 +08:00
|
|
|
if ((len < (USB_DT_CONFIG_SIZE + USB_DT_DEVICE_SIZE + 4)) ||
|
|
|
|
(len > PAGE_SIZE * 4))
|
2005-04-17 06:20:36 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* we might need to change message format someday */
|
|
|
|
if (copy_from_user (&tag, buf, 4))
|
|
|
|
return -EFAULT;
|
|
|
|
if (tag != 0)
|
|
|
|
return -EINVAL;
|
|
|
|
buf += 4;
|
|
|
|
length -= 4;
|
|
|
|
|
2010-05-22 16:26:22 +08:00
|
|
|
kbuf = memdup_user(buf, length);
|
|
|
|
if (IS_ERR(kbuf))
|
|
|
|
return PTR_ERR(kbuf);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
spin_lock_irq (&dev->lock);
|
|
|
|
value = -EINVAL;
|
2017-02-22 05:33:11 +08:00
|
|
|
if (dev->buf) {
|
|
|
|
kfree(kbuf);
|
2005-04-17 06:20:36 +08:00
|
|
|
goto fail;
|
2017-02-22 05:33:11 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
dev->buf = kbuf;
|
|
|
|
|
|
|
|
/* full or low speed config */
|
|
|
|
dev->config = (void *) kbuf;
|
2007-05-26 11:40:14 +08:00
|
|
|
total = le16_to_cpu(dev->config->wTotalLength);
|
2016-12-10 04:24:24 +08:00
|
|
|
if (!is_valid_config(dev->config, total) ||
|
|
|
|
total > length - USB_DT_DEVICE_SIZE)
|
2005-04-17 06:20:36 +08:00
|
|
|
goto fail;
|
|
|
|
kbuf += total;
|
|
|
|
length -= total;
|
|
|
|
|
|
|
|
/* optional high speed config */
|
|
|
|
if (kbuf [1] == USB_DT_CONFIG) {
|
|
|
|
dev->hs_config = (void *) kbuf;
|
2007-05-26 11:40:14 +08:00
|
|
|
total = le16_to_cpu(dev->hs_config->wTotalLength);
|
2016-12-10 04:24:24 +08:00
|
|
|
if (!is_valid_config(dev->hs_config, total) ||
|
|
|
|
total > length - USB_DT_DEVICE_SIZE)
|
2005-04-17 06:20:36 +08:00
|
|
|
goto fail;
|
|
|
|
kbuf += total;
|
|
|
|
length -= total;
|
2016-12-10 04:18:43 +08:00
|
|
|
} else {
|
|
|
|
dev->hs_config = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* could support multiple configs, using another encoding! */
|
|
|
|
|
|
|
|
/* device descriptor (tweaked for paranoia) */
|
|
|
|
if (length != USB_DT_DEVICE_SIZE)
|
|
|
|
goto fail;
|
|
|
|
dev->dev = (void *)kbuf;
|
|
|
|
if (dev->dev->bLength != USB_DT_DEVICE_SIZE
|
|
|
|
|| dev->dev->bDescriptorType != USB_DT_DEVICE
|
|
|
|
|| dev->dev->bNumConfigurations != 1)
|
|
|
|
goto fail;
|
2009-02-12 06:11:36 +08:00
|
|
|
dev->dev->bcdUSB = cpu_to_le16 (0x0200);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* triggers gadgetfs_bind(); then we can enumerate. */
|
|
|
|
spin_unlock_irq (&dev->lock);
|
2012-08-25 02:46:18 +08:00
|
|
|
if (dev->hs_config)
|
|
|
|
gadgetfs_driver.max_speed = USB_SPEED_HIGH;
|
|
|
|
else
|
|
|
|
gadgetfs_driver.max_speed = USB_SPEED_FULL;
|
2012-09-07 02:11:05 +08:00
|
|
|
|
|
|
|
value = usb_gadget_probe_driver(&gadgetfs_driver);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (value != 0) {
|
|
|
|
kfree (dev->buf);
|
|
|
|
dev->buf = NULL;
|
|
|
|
} else {
|
|
|
|
/* at this point "good" hardware has for the first time
|
|
|
|
* let the USB the host see us. alternatively, if users
|
|
|
|
* unplug/replug that will clear all the error state.
|
|
|
|
*
|
|
|
|
* note: everything running before here was guaranteed
|
|
|
|
* to choke driver model style diagnostics. from here
|
|
|
|
* on, they can work ... except in cleanup paths that
|
|
|
|
* kick in after the ep0 descriptor is closed.
|
|
|
|
*/
|
|
|
|
value = len;
|
2016-02-18 15:59:26 +08:00
|
|
|
dev->gadget_registered = true;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
return value;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
spin_unlock_irq (&dev->lock);
|
2017-02-28 06:30:02 +08:00
|
|
|
pr_debug ("%s: %s fail %zd, %p\n", shortname, __func__, value, dev);
|
2005-04-17 06:20:36 +08:00
|
|
|
kfree (dev->buf);
|
|
|
|
dev->buf = NULL;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
dev_open (struct inode *inode, struct file *fd)
|
|
|
|
{
|
2006-09-27 16:50:46 +08:00
|
|
|
struct dev_data *dev = inode->i_private;
|
2005-04-17 06:20:36 +08:00
|
|
|
int value = -EBUSY;
|
|
|
|
|
2007-01-17 14:51:04 +08:00
|
|
|
spin_lock_irq(&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (dev->state == STATE_DEV_DISABLED) {
|
|
|
|
dev->ev_next = 0;
|
2007-01-17 14:51:04 +08:00
|
|
|
dev->state = STATE_DEV_OPENED;
|
2005-04-17 06:20:36 +08:00
|
|
|
fd->private_data = dev;
|
|
|
|
get_dev (dev);
|
|
|
|
value = 0;
|
|
|
|
}
|
2007-01-17 14:51:04 +08:00
|
|
|
spin_unlock_irq(&dev->lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2015-03-04 23:31:50 +08:00
|
|
|
static const struct file_operations ep0_operations = {
|
2005-04-17 06:20:36 +08:00
|
|
|
.llseek = no_llseek,
|
|
|
|
|
|
|
|
.open = dev_open,
|
2015-03-04 23:31:50 +08:00
|
|
|
.read = ep0_read,
|
2005-04-17 06:20:36 +08:00
|
|
|
.write = dev_config,
|
|
|
|
.fasync = ep0_fasync,
|
2015-03-04 23:31:50 +08:00
|
|
|
.poll = ep0_poll,
|
2008-05-23 05:03:27 +08:00
|
|
|
.unlocked_ioctl = dev_ioctl,
|
2005-04-17 06:20:36 +08:00
|
|
|
.release = dev_release,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
/* FILESYSTEM AND SUPERBLOCK OPERATIONS
|
|
|
|
*
|
|
|
|
* Mounting the filesystem creates a controller file, used first for
|
|
|
|
* device configuration then later for event monitoring.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/* FIXME PAM etc could set this security policy without mount options
|
|
|
|
* if epfiles inherited ownership and permissons from ep0 ...
|
|
|
|
*/
|
|
|
|
|
|
|
|
static unsigned default_uid;
|
|
|
|
static unsigned default_gid;
|
|
|
|
static unsigned default_perm = S_IRUSR | S_IWUSR;
|
|
|
|
|
|
|
|
module_param (default_uid, uint, 0644);
|
|
|
|
module_param (default_gid, uint, 0644);
|
|
|
|
module_param (default_perm, uint, 0644);
|
|
|
|
|
|
|
|
|
|
|
|
static struct inode *
|
|
|
|
gadgetfs_make_inode (struct super_block *sb,
|
2006-03-28 17:56:41 +08:00
|
|
|
void *data, const struct file_operations *fops,
|
2005-04-17 06:20:36 +08:00
|
|
|
int mode)
|
|
|
|
{
|
|
|
|
struct inode *inode = new_inode (sb);
|
|
|
|
|
|
|
|
if (inode) {
|
2010-10-23 23:19:54 +08:00
|
|
|
inode->i_ino = get_next_ino();
|
2005-04-17 06:20:36 +08:00
|
|
|
inode->i_mode = mode;
|
2012-02-08 08:32:04 +08:00
|
|
|
inode->i_uid = make_kuid(&init_user_ns, default_uid);
|
|
|
|
inode->i_gid = make_kgid(&init_user_ns, default_gid);
|
2005-04-17 06:20:36 +08:00
|
|
|
inode->i_atime = inode->i_mtime = inode->i_ctime
|
2016-09-14 22:48:04 +08:00
|
|
|
= current_time(inode);
|
2006-09-27 16:50:46 +08:00
|
|
|
inode->i_private = data;
|
2005-04-17 06:20:36 +08:00
|
|
|
inode->i_fop = fops;
|
|
|
|
}
|
|
|
|
return inode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* creates in fs root directory, so non-renamable and non-linkable.
|
|
|
|
* so inode and dentry are paired, until device reconfig.
|
|
|
|
*/
|
2014-09-04 01:37:56 +08:00
|
|
|
static struct dentry *
|
2005-04-17 06:20:36 +08:00
|
|
|
gadgetfs_create_file (struct super_block *sb, char const *name,
|
2014-09-04 01:37:56 +08:00
|
|
|
void *data, const struct file_operations *fops)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
struct dentry *dentry;
|
|
|
|
struct inode *inode;
|
|
|
|
|
|
|
|
dentry = d_alloc_name(sb->s_root, name);
|
|
|
|
if (!dentry)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
inode = gadgetfs_make_inode (sb, data, fops,
|
|
|
|
S_IFREG | (default_perm & S_IRWXUGO));
|
|
|
|
if (!inode) {
|
|
|
|
dput(dentry);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
d_add (dentry, inode);
|
2014-09-04 01:37:56 +08:00
|
|
|
return dentry;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-09-22 08:01:09 +08:00
|
|
|
static const struct super_operations gadget_fs_operations = {
|
2005-04-17 06:20:36 +08:00
|
|
|
.statfs = simple_statfs,
|
|
|
|
.drop_inode = generic_delete_inode,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
gadgetfs_fill_super (struct super_block *sb, void *opts, int silent)
|
|
|
|
{
|
|
|
|
struct inode *inode;
|
|
|
|
struct dev_data *dev;
|
|
|
|
|
|
|
|
if (the_device)
|
|
|
|
return -ESRCH;
|
|
|
|
|
2016-02-18 18:34:43 +08:00
|
|
|
CHIP = usb_get_gadget_udc_name();
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!CHIP)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
/* superblock */
|
mm, fs: get rid of PAGE_CACHE_* and page_cache_{get,release} macros
PAGE_CACHE_{SIZE,SHIFT,MASK,ALIGN} macros were introduced *long* time
ago with promise that one day it will be possible to implement page
cache with bigger chunks than PAGE_SIZE.
This promise never materialized. And unlikely will.
We have many places where PAGE_CACHE_SIZE assumed to be equal to
PAGE_SIZE. And it's constant source of confusion on whether
PAGE_CACHE_* or PAGE_* constant should be used in a particular case,
especially on the border between fs and mm.
Global switching to PAGE_CACHE_SIZE != PAGE_SIZE would cause to much
breakage to be doable.
Let's stop pretending that pages in page cache are special. They are
not.
The changes are pretty straight-forward:
- <foo> << (PAGE_CACHE_SHIFT - PAGE_SHIFT) -> <foo>;
- <foo> >> (PAGE_CACHE_SHIFT - PAGE_SHIFT) -> <foo>;
- PAGE_CACHE_{SIZE,SHIFT,MASK,ALIGN} -> PAGE_{SIZE,SHIFT,MASK,ALIGN};
- page_cache_get() -> get_page();
- page_cache_release() -> put_page();
This patch contains automated changes generated with coccinelle using
script below. For some reason, coccinelle doesn't patch header files.
I've called spatch for them manually.
The only adjustment after coccinelle is revert of changes to
PAGE_CAHCE_ALIGN definition: we are going to drop it later.
There are few places in the code where coccinelle didn't reach. I'll
fix them manually in a separate patch. Comments and documentation also
will be addressed with the separate patch.
virtual patch
@@
expression E;
@@
- E << (PAGE_CACHE_SHIFT - PAGE_SHIFT)
+ E
@@
expression E;
@@
- E >> (PAGE_CACHE_SHIFT - PAGE_SHIFT)
+ E
@@
@@
- PAGE_CACHE_SHIFT
+ PAGE_SHIFT
@@
@@
- PAGE_CACHE_SIZE
+ PAGE_SIZE
@@
@@
- PAGE_CACHE_MASK
+ PAGE_MASK
@@
expression E;
@@
- PAGE_CACHE_ALIGN(E)
+ PAGE_ALIGN(E)
@@
expression E;
@@
- page_cache_get(E)
+ get_page(E)
@@
expression E;
@@
- page_cache_release(E)
+ put_page(E)
Signed-off-by: Kirill A. Shutemov <kirill.shutemov@linux.intel.com>
Acked-by: Michal Hocko <mhocko@suse.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2016-04-01 20:29:47 +08:00
|
|
|
sb->s_blocksize = PAGE_SIZE;
|
|
|
|
sb->s_blocksize_bits = PAGE_SHIFT;
|
2005-04-17 06:20:36 +08:00
|
|
|
sb->s_magic = GADGETFS_MAGIC;
|
|
|
|
sb->s_op = &gadget_fs_operations;
|
|
|
|
sb->s_time_gran = 1;
|
|
|
|
|
|
|
|
/* root inode */
|
|
|
|
inode = gadgetfs_make_inode (sb,
|
|
|
|
NULL, &simple_dir_operations,
|
|
|
|
S_IFDIR | S_IRUGO | S_IXUGO);
|
|
|
|
if (!inode)
|
2012-01-09 04:59:45 +08:00
|
|
|
goto Enomem;
|
2005-04-17 06:20:36 +08:00
|
|
|
inode->i_op = &simple_dir_inode_operations;
|
2012-01-09 11:15:13 +08:00
|
|
|
if (!(sb->s_root = d_make_root (inode)))
|
2012-01-09 04:59:45 +08:00
|
|
|
goto Enomem;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* the ep0 file is named after the controller we expect;
|
|
|
|
* user mode code can use it for sanity checks, like we do.
|
|
|
|
*/
|
|
|
|
dev = dev_new ();
|
|
|
|
if (!dev)
|
2012-01-09 04:59:45 +08:00
|
|
|
goto Enomem;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
dev->sb = sb;
|
2015-03-04 23:31:50 +08:00
|
|
|
dev->dentry = gadgetfs_create_file(sb, CHIP, dev, &ep0_operations);
|
2014-09-04 01:37:56 +08:00
|
|
|
if (!dev->dentry) {
|
2012-01-09 04:59:45 +08:00
|
|
|
put_dev(dev);
|
|
|
|
goto Enomem;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* other endpoint files are available after hardware setup,
|
|
|
|
* from binding to a controller.
|
|
|
|
*/
|
|
|
|
the_device = dev;
|
|
|
|
return 0;
|
2006-05-23 00:27:38 +08:00
|
|
|
|
2012-01-09 04:59:45 +08:00
|
|
|
Enomem:
|
2006-05-23 00:27:38 +08:00
|
|
|
return -ENOMEM;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* "mount -t gadgetfs path /dev/gadget" ends up here */
|
2010-07-25 05:48:30 +08:00
|
|
|
static struct dentry *
|
|
|
|
gadgetfs_mount (struct file_system_type *t, int flags,
|
|
|
|
const char *path, void *opts)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2010-07-25 05:48:30 +08:00
|
|
|
return mount_single (t, flags, opts, gadgetfs_fill_super);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gadgetfs_kill_sb (struct super_block *sb)
|
|
|
|
{
|
|
|
|
kill_litter_super (sb);
|
|
|
|
if (the_device) {
|
|
|
|
put_dev (the_device);
|
|
|
|
the_device = NULL;
|
|
|
|
}
|
2016-02-18 18:34:43 +08:00
|
|
|
kfree(CHIP);
|
|
|
|
CHIP = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
static struct file_system_type gadgetfs_type = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.name = shortname,
|
2010-07-25 05:48:30 +08:00
|
|
|
.mount = gadgetfs_mount,
|
2005-04-17 06:20:36 +08:00
|
|
|
.kill_sb = gadgetfs_kill_sb,
|
|
|
|
};
|
2013-03-03 11:39:14 +08:00
|
|
|
MODULE_ALIAS_FS("gadgetfs");
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
static int __init init (void)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
|
|
|
|
status = register_filesystem (&gadgetfs_type);
|
|
|
|
if (status == 0)
|
|
|
|
pr_info ("%s: %s, version " DRIVER_VERSION "\n",
|
|
|
|
shortname, driver_desc);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
module_init (init);
|
|
|
|
|
|
|
|
static void __exit cleanup (void)
|
|
|
|
{
|
|
|
|
pr_debug ("unregister %s\n", shortname);
|
|
|
|
unregister_filesystem (&gadgetfs_type);
|
|
|
|
}
|
|
|
|
module_exit (cleanup);
|
|
|
|
|