|  | /***************************************************************************** | 
|  | *  File: drivers/usb/misc/vstusb.c | 
|  | * | 
|  | *  Purpose: Support for the bulk USB Vernier Spectrophotometers | 
|  | * | 
|  | *  Author:     Johnnie Peters | 
|  | *              Axian Consulting | 
|  | *              Beaverton, OR, USA 97005 | 
|  | * | 
|  | *  Modified by:     EQware Engineering, Inc. | 
|  | *                   Oregon City, OR, USA 97045 | 
|  | * | 
|  | *  Copyright:  2007, 2008 | 
|  | *              Vernier Software & Technology | 
|  | *              Beaverton, OR, USA 97005 | 
|  | * | 
|  | *  Web:        www.vernier.com | 
|  | * | 
|  | *  This program is free software; you can redistribute it and/or modify | 
|  | *  it under the terms of the GNU General Public License version 2 as | 
|  | *  published by the Free Software Foundation. | 
|  | * | 
|  | *****************************************************************************/ | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/errno.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/slab.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/mutex.h> | 
|  | #include <linux/uaccess.h> | 
|  | #include <linux/usb.h> | 
|  |  | 
|  | #include <linux/usb/vstusb.h> | 
|  |  | 
|  | #define DRIVER_VERSION "VST USB Driver Version 1.5" | 
|  | #define DRIVER_DESC "Vernier Software Technology Bulk USB Driver" | 
|  |  | 
|  | #ifdef CONFIG_USB_DYNAMIC_MINORS | 
|  | #define VSTUSB_MINOR_BASE	0 | 
|  | #else | 
|  | #define VSTUSB_MINOR_BASE	199 | 
|  | #endif | 
|  |  | 
|  | #define USB_VENDOR_OCEANOPTICS	0x2457 | 
|  | #define USB_VENDOR_VERNIER	0x08F7	/* Vernier Software & Technology */ | 
|  |  | 
|  | #define USB_PRODUCT_USB2000	0x1002 | 
|  | #define USB_PRODUCT_ADC1000_FW	0x1003	/* firmware download (renumerates) */ | 
|  | #define USB_PRODUCT_ADC1000	0x1004 | 
|  | #define USB_PRODUCT_HR2000_FW	0x1009	/* firmware download (renumerates) */ | 
|  | #define USB_PRODUCT_HR2000	0x100A | 
|  | #define USB_PRODUCT_HR4000_FW	0x1011	/* firmware download (renumerates) */ | 
|  | #define USB_PRODUCT_HR4000	0x1012 | 
|  | #define USB_PRODUCT_USB650	0x1014	/* "Red Tide" */ | 
|  | #define USB_PRODUCT_QE65000	0x1018 | 
|  | #define USB_PRODUCT_USB4000	0x1022 | 
|  | #define USB_PRODUCT_USB325	0x1024	/* "Vernier Spectrometer" */ | 
|  |  | 
|  | #define USB_PRODUCT_LABPRO	0x0001 | 
|  | #define USB_PRODUCT_LABQUEST	0x0005 | 
|  |  | 
|  | #define VST_MAXBUFFER		(64*1024) | 
|  |  | 
|  | static struct usb_device_id id_table[] = { | 
|  | { USB_DEVICE(USB_VENDOR_OCEANOPTICS, USB_PRODUCT_USB2000)}, | 
|  | { USB_DEVICE(USB_VENDOR_OCEANOPTICS, USB_PRODUCT_HR4000)}, | 
|  | { USB_DEVICE(USB_VENDOR_OCEANOPTICS, USB_PRODUCT_USB650)}, | 
|  | { USB_DEVICE(USB_VENDOR_OCEANOPTICS, USB_PRODUCT_USB4000)}, | 
|  | { USB_DEVICE(USB_VENDOR_OCEANOPTICS, USB_PRODUCT_USB325)}, | 
|  | { USB_DEVICE(USB_VENDOR_VERNIER, USB_PRODUCT_LABQUEST)}, | 
|  | { USB_DEVICE(USB_VENDOR_VERNIER, USB_PRODUCT_LABPRO)}, | 
|  | {}, | 
|  | }; | 
|  |  | 
|  | MODULE_DEVICE_TABLE(usb, id_table); | 
|  |  | 
|  | struct vstusb_device { | 
|  | struct kref				kref; | 
|  | struct mutex            lock; | 
|  | struct usb_device       *usb_dev; | 
|  | char                    present; | 
|  | char                    isopen; | 
|  | struct usb_anchor       submitted; | 
|  | int                     rd_pipe; | 
|  | int                     rd_timeout_ms; | 
|  | int                     wr_pipe; | 
|  | int                     wr_timeout_ms; | 
|  | }; | 
|  | #define to_vst_dev(d) container_of(d, struct vstusb_device, kref) | 
|  |  | 
|  | static struct usb_driver vstusb_driver; | 
|  |  | 
|  | static void vstusb_delete(struct kref *kref) | 
|  | { | 
|  | struct vstusb_device *vstdev = to_vst_dev(kref); | 
|  |  | 
|  | usb_put_dev(vstdev->usb_dev); | 
|  | kfree(vstdev); | 
|  | } | 
|  |  | 
|  | static int vstusb_open(struct inode *inode, struct file *file) | 
|  | { | 
|  | struct vstusb_device *vstdev; | 
|  | struct usb_interface *interface; | 
|  |  | 
|  | interface = usb_find_interface(&vstusb_driver, iminor(inode)); | 
|  |  | 
|  | if (!interface) { | 
|  | printk(KERN_ERR KBUILD_MODNAME | 
|  | ": %s - error, can't find device for minor %d\n", | 
|  | __func__, iminor(inode)); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | vstdev = usb_get_intfdata(interface); | 
|  |  | 
|  | if (!vstdev) | 
|  | return -ENODEV; | 
|  |  | 
|  | /* lock this device */ | 
|  | mutex_lock(&vstdev->lock); | 
|  |  | 
|  | /* can only open one time */ | 
|  | if ((!vstdev->present) || (vstdev->isopen)) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | return -EBUSY; | 
|  | } | 
|  |  | 
|  | /* increment our usage count */ | 
|  | kref_get(&vstdev->kref); | 
|  |  | 
|  | vstdev->isopen = 1; | 
|  |  | 
|  | /* save device in the file's private structure */ | 
|  | file->private_data = vstdev; | 
|  |  | 
|  | dev_dbg(&vstdev->usb_dev->dev, "%s: opened\n", __func__); | 
|  |  | 
|  | mutex_unlock(&vstdev->lock); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int vstusb_release(struct inode *inode, struct file *file) | 
|  | { | 
|  | struct vstusb_device *vstdev; | 
|  |  | 
|  | vstdev = file->private_data; | 
|  |  | 
|  | if (vstdev == NULL) | 
|  | return -ENODEV; | 
|  |  | 
|  | mutex_lock(&vstdev->lock); | 
|  |  | 
|  | vstdev->isopen = 0; | 
|  |  | 
|  | dev_dbg(&vstdev->usb_dev->dev, "%s: released\n", __func__); | 
|  |  | 
|  | mutex_unlock(&vstdev->lock); | 
|  |  | 
|  | kref_put(&vstdev->kref, vstusb_delete); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void usb_api_blocking_completion(struct urb *urb) | 
|  | { | 
|  | struct completion *completeit = urb->context; | 
|  |  | 
|  | complete(completeit); | 
|  | } | 
|  |  | 
|  | static int vstusb_fill_and_send_urb(struct urb *urb, | 
|  | struct usb_device *usb_dev, | 
|  | unsigned int pipe, void *data, | 
|  | unsigned int len, struct completion *done) | 
|  | { | 
|  | struct usb_host_endpoint *ep; | 
|  | struct usb_host_endpoint **hostep; | 
|  | unsigned int pipend; | 
|  |  | 
|  | int status; | 
|  |  | 
|  | hostep = usb_pipein(pipe) ? usb_dev->ep_in : usb_dev->ep_out; | 
|  | pipend = usb_pipeendpoint(pipe); | 
|  | ep = hostep[pipend]; | 
|  |  | 
|  | if (!ep || (len == 0)) | 
|  | return -EINVAL; | 
|  |  | 
|  | if ((ep->desc.bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) | 
|  | == USB_ENDPOINT_XFER_INT) { | 
|  | pipe = (pipe & ~(3 << 30)) | (PIPE_INTERRUPT << 30); | 
|  | usb_fill_int_urb(urb, usb_dev, pipe, data, len, | 
|  | (usb_complete_t)usb_api_blocking_completion, | 
|  | NULL, ep->desc.bInterval); | 
|  | } else | 
|  | usb_fill_bulk_urb(urb, usb_dev, pipe, data, len, | 
|  | (usb_complete_t)usb_api_blocking_completion, | 
|  | NULL); | 
|  |  | 
|  | init_completion(done); | 
|  | urb->context = done; | 
|  | urb->actual_length = 0; | 
|  | status = usb_submit_urb(urb, GFP_KERNEL); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int vstusb_complete_urb(struct urb *urb, struct completion *done, | 
|  | int timeout, int *actual_length) | 
|  | { | 
|  | unsigned long expire; | 
|  | int status; | 
|  |  | 
|  | expire = timeout ? msecs_to_jiffies(timeout) : MAX_SCHEDULE_TIMEOUT; | 
|  | if (!wait_for_completion_interruptible_timeout(done, expire)) { | 
|  | usb_kill_urb(urb); | 
|  | status = urb->status == -ENOENT ? -ETIMEDOUT : urb->status; | 
|  |  | 
|  | dev_dbg(&urb->dev->dev, | 
|  | "%s timed out on ep%d%s len=%d/%d, urb status = %d\n", | 
|  | current->comm, | 
|  | usb_pipeendpoint(urb->pipe), | 
|  | usb_pipein(urb->pipe) ? "in" : "out", | 
|  | urb->actual_length, | 
|  | urb->transfer_buffer_length, | 
|  | urb->status); | 
|  |  | 
|  | } else { | 
|  | if (signal_pending(current)) { | 
|  | /* if really an error */ | 
|  | if (urb->status && !((urb->status == -ENOENT)     || | 
|  | (urb->status == -ECONNRESET) || | 
|  | (urb->status == -ESHUTDOWN))) { | 
|  | status = -EINTR; | 
|  | usb_kill_urb(urb); | 
|  | } else { | 
|  | status = 0; | 
|  | } | 
|  |  | 
|  | dev_dbg(&urb->dev->dev, | 
|  | "%s: signal pending on ep%d%s len=%d/%d," | 
|  | "urb status = %d\n", | 
|  | current->comm, | 
|  | usb_pipeendpoint(urb->pipe), | 
|  | usb_pipein(urb->pipe) ? "in" : "out", | 
|  | urb->actual_length, | 
|  | urb->transfer_buffer_length, | 
|  | urb->status); | 
|  |  | 
|  | } else { | 
|  | status = urb->status; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (actual_length) | 
|  | *actual_length = urb->actual_length; | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static ssize_t vstusb_read(struct file *file, char __user *buffer, | 
|  | size_t count, loff_t *ppos) | 
|  | { | 
|  | struct vstusb_device *vstdev; | 
|  | int cnt = -1; | 
|  | void *buf; | 
|  | int retval = 0; | 
|  |  | 
|  | struct urb              *urb; | 
|  | struct usb_device       *dev; | 
|  | unsigned int            pipe; | 
|  | int                     timeout; | 
|  |  | 
|  | DECLARE_COMPLETION_ONSTACK(done); | 
|  |  | 
|  | vstdev = file->private_data; | 
|  |  | 
|  | if (vstdev == NULL) | 
|  | return -ENODEV; | 
|  |  | 
|  | /* verify that we actually want to read some data */ | 
|  | if ((count == 0) || (count > VST_MAXBUFFER)) | 
|  | return -EINVAL; | 
|  |  | 
|  | /* lock this object */ | 
|  | if (mutex_lock_interruptible(&vstdev->lock)) | 
|  | return -ERESTARTSYS; | 
|  |  | 
|  | /* anyone home */ | 
|  | if (!vstdev->present) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | printk(KERN_ERR KBUILD_MODNAME | 
|  | ": %s: device not present\n", __func__); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | /* pull out the necessary data */ | 
|  | dev =     vstdev->usb_dev; | 
|  | pipe =    usb_rcvbulkpipe(dev, vstdev->rd_pipe); | 
|  | timeout = vstdev->rd_timeout_ms; | 
|  |  | 
|  | buf = kmalloc(count, GFP_KERNEL); | 
|  | if (buf == NULL) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | urb = usb_alloc_urb(0, GFP_KERNEL); | 
|  | if (!urb) { | 
|  | kfree(buf); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | usb_anchor_urb(urb, &vstdev->submitted); | 
|  | retval = vstusb_fill_and_send_urb(urb, dev, pipe, buf, count, &done); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | if (retval) { | 
|  | usb_unanchor_urb(urb); | 
|  | dev_err(&dev->dev, "%s: error %d filling and sending urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | retval = vstusb_complete_urb(urb, &done, timeout, &cnt); | 
|  | if (retval) { | 
|  | dev_err(&dev->dev, "%s: error %d completing urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | if (copy_to_user(buffer, buf, cnt)) { | 
|  | dev_err(&dev->dev, "%s: can't copy_to_user\n", __func__); | 
|  | retval = -EFAULT; | 
|  | } else { | 
|  | retval = cnt; | 
|  | dev_dbg(&dev->dev, "%s: read %d bytes from pipe %d\n", | 
|  | __func__, cnt, pipe); | 
|  | } | 
|  |  | 
|  | exit: | 
|  | usb_free_urb(urb); | 
|  | kfree(buf); | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static ssize_t vstusb_write(struct file *file, const char __user *buffer, | 
|  | size_t count, loff_t *ppos) | 
|  | { | 
|  | struct vstusb_device *vstdev; | 
|  | int cnt = -1; | 
|  | void *buf; | 
|  | int retval = 0; | 
|  |  | 
|  | struct urb              *urb; | 
|  | struct usb_device       *dev; | 
|  | unsigned int            pipe; | 
|  | int                     timeout; | 
|  |  | 
|  | DECLARE_COMPLETION_ONSTACK(done); | 
|  |  | 
|  | vstdev = file->private_data; | 
|  |  | 
|  | if (vstdev == NULL) | 
|  | return -ENODEV; | 
|  |  | 
|  | /* verify that we actually have some data to write */ | 
|  | if ((count == 0) || (count > VST_MAXBUFFER)) | 
|  | return retval; | 
|  |  | 
|  | /* lock this object */ | 
|  | if (mutex_lock_interruptible(&vstdev->lock)) | 
|  | return -ERESTARTSYS; | 
|  |  | 
|  | /* anyone home */ | 
|  | if (!vstdev->present) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | printk(KERN_ERR KBUILD_MODNAME | 
|  | ": %s: device not present\n", __func__); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | /* pull out the necessary data */ | 
|  | dev =     vstdev->usb_dev; | 
|  | pipe =    usb_sndbulkpipe(dev, vstdev->wr_pipe); | 
|  | timeout = vstdev->wr_timeout_ms; | 
|  |  | 
|  | buf = kmalloc(count, GFP_KERNEL); | 
|  | if (buf == NULL) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | urb = usb_alloc_urb(0, GFP_KERNEL); | 
|  | if (!urb) { | 
|  | kfree(buf); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | if (copy_from_user(buf, buffer, count)) { | 
|  | dev_err(&dev->dev, "%s: can't copy_from_user\n", __func__); | 
|  | retval = -EFAULT; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | usb_anchor_urb(urb, &vstdev->submitted); | 
|  | retval = vstusb_fill_and_send_urb(urb, dev, pipe, buf, count, &done); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | if (retval) { | 
|  | usb_unanchor_urb(urb); | 
|  | dev_err(&dev->dev, "%s: error %d filling and sending urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | retval = vstusb_complete_urb(urb, &done, timeout, &cnt); | 
|  | if (retval) { | 
|  | dev_err(&dev->dev, "%s: error %d completing urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } else { | 
|  | retval = cnt; | 
|  | dev_dbg(&dev->dev, "%s: sent %d bytes to pipe %d\n", | 
|  | __func__, cnt, pipe); | 
|  | } | 
|  |  | 
|  | exit: | 
|  | usb_free_urb(urb); | 
|  | kfree(buf); | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static long vstusb_ioctl(struct file *file, unsigned int cmd, unsigned long arg) | 
|  | { | 
|  | int retval = 0; | 
|  | int cnt = -1; | 
|  | void __user *data = (void __user *)arg; | 
|  | struct vstusb_args usb_data; | 
|  |  | 
|  | struct vstusb_device *vstdev; | 
|  | void *buffer = NULL; /* must be initialized. buffer is | 
|  | *	referenced on exit but not all | 
|  | * ioctls allocate it */ | 
|  |  | 
|  | struct urb              *urb = NULL; /* must be initialized. urb is | 
|  | *	referenced on exit but not all | 
|  | * ioctls allocate it */ | 
|  | struct usb_device       *dev; | 
|  | unsigned int            pipe; | 
|  | int                     timeout; | 
|  |  | 
|  | DECLARE_COMPLETION_ONSTACK(done); | 
|  |  | 
|  | vstdev = file->private_data; | 
|  |  | 
|  | if (_IOC_TYPE(cmd) != VST_IOC_MAGIC) { | 
|  | dev_warn(&vstdev->usb_dev->dev, | 
|  | "%s: ioctl command %x, bad ioctl magic %x, " | 
|  | "expected %x\n", __func__, cmd, | 
|  | _IOC_TYPE(cmd), VST_IOC_MAGIC); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (vstdev == NULL) | 
|  | return -ENODEV; | 
|  |  | 
|  | if (copy_from_user(&usb_data, data, sizeof(struct vstusb_args))) { | 
|  | dev_err(&vstdev->usb_dev->dev, "%s: can't copy_from_user\n", | 
|  | __func__); | 
|  | return -EFAULT; | 
|  | } | 
|  |  | 
|  | /* lock this object */ | 
|  | if (mutex_lock_interruptible(&vstdev->lock)) { | 
|  | retval = -ERESTARTSYS; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | /* anyone home */ | 
|  | if (!vstdev->present) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | dev_err(&vstdev->usb_dev->dev, "%s: device not present\n", | 
|  | __func__); | 
|  | retval = -ENODEV; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | /* pull out the necessary data */ | 
|  | dev = vstdev->usb_dev; | 
|  |  | 
|  | switch (cmd) { | 
|  |  | 
|  | case IOCTL_VSTUSB_CONFIG_RW: | 
|  |  | 
|  | vstdev->rd_pipe = usb_data.rd_pipe; | 
|  | vstdev->rd_timeout_ms = usb_data.rd_timeout_ms; | 
|  | vstdev->wr_pipe = usb_data.wr_pipe; | 
|  | vstdev->wr_timeout_ms = usb_data.wr_timeout_ms; | 
|  |  | 
|  | mutex_unlock(&vstdev->lock); | 
|  |  | 
|  | dev_dbg(&dev->dev, "%s: setting pipes/timeouts, " | 
|  | "rdpipe = %d, rdtimeout = %d, " | 
|  | "wrpipe = %d, wrtimeout = %d\n", __func__, | 
|  | vstdev->rd_pipe, vstdev->rd_timeout_ms, | 
|  | vstdev->wr_pipe, vstdev->wr_timeout_ms); | 
|  | break; | 
|  |  | 
|  | case IOCTL_VSTUSB_SEND_PIPE: | 
|  |  | 
|  | if ((usb_data.count == 0) || (usb_data.count > VST_MAXBUFFER)) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -EINVAL; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | buffer = kmalloc(usb_data.count, GFP_KERNEL); | 
|  | if (buffer == NULL) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -ENOMEM; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | urb = usb_alloc_urb(0, GFP_KERNEL); | 
|  | if (!urb) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -ENOMEM; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | timeout = usb_data.timeout_ms; | 
|  |  | 
|  | pipe = usb_sndbulkpipe(dev, usb_data.pipe); | 
|  |  | 
|  | if (copy_from_user(buffer, usb_data.buffer, usb_data.count)) { | 
|  | dev_err(&dev->dev, "%s: can't copy_from_user\n", | 
|  | __func__); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -EFAULT; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | usb_anchor_urb(urb, &vstdev->submitted); | 
|  | retval = vstusb_fill_and_send_urb(urb, dev, pipe, buffer, | 
|  | usb_data.count, &done); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | if (retval) { | 
|  | usb_unanchor_urb(urb); | 
|  | dev_err(&dev->dev, | 
|  | "%s: error %d filling and sending urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | retval = vstusb_complete_urb(urb, &done, timeout, &cnt); | 
|  | if (retval) { | 
|  | dev_err(&dev->dev, "%s: error %d completing urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | } | 
|  |  | 
|  | break; | 
|  | case IOCTL_VSTUSB_RECV_PIPE: | 
|  |  | 
|  | if ((usb_data.count == 0) || (usb_data.count > VST_MAXBUFFER)) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -EINVAL; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | buffer = kmalloc(usb_data.count, GFP_KERNEL); | 
|  | if (buffer == NULL) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -ENOMEM; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | urb = usb_alloc_urb(0, GFP_KERNEL); | 
|  | if (!urb) { | 
|  | mutex_unlock(&vstdev->lock); | 
|  | retval = -ENOMEM; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | timeout = usb_data.timeout_ms; | 
|  |  | 
|  | pipe = usb_rcvbulkpipe(dev, usb_data.pipe); | 
|  |  | 
|  | usb_anchor_urb(urb, &vstdev->submitted); | 
|  | retval = vstusb_fill_and_send_urb(urb, dev, pipe, buffer, | 
|  | usb_data.count, &done); | 
|  | mutex_unlock(&vstdev->lock); | 
|  | if (retval) { | 
|  | usb_unanchor_urb(urb); | 
|  | dev_err(&dev->dev, | 
|  | "%s: error %d filling and sending urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | retval = vstusb_complete_urb(urb, &done, timeout, &cnt); | 
|  | if (retval) { | 
|  | dev_err(&dev->dev, "%s: error %d completing urb %d\n", | 
|  | __func__, retval, pipe); | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | if (copy_to_user(usb_data.buffer, buffer, cnt)) { | 
|  | dev_err(&dev->dev, "%s: can't copy_to_user\n", | 
|  | __func__); | 
|  | retval = -EFAULT; | 
|  | goto exit; | 
|  | } | 
|  |  | 
|  | usb_data.count = cnt; | 
|  | if (copy_to_user(data, &usb_data, sizeof(struct vstusb_args))) { | 
|  | dev_err(&dev->dev, "%s: can't copy_to_user\n", | 
|  | __func__); | 
|  | retval = -EFAULT; | 
|  | } else { | 
|  | dev_dbg(&dev->dev, "%s: recv %zd bytes from pipe %d\n", | 
|  | __func__, usb_data.count, usb_data.pipe); | 
|  | } | 
|  |  | 
|  | break; | 
|  |  | 
|  | default: | 
|  | mutex_unlock(&vstdev->lock); | 
|  | dev_warn(&dev->dev, "ioctl_vstusb: invalid ioctl cmd %x\n", | 
|  | cmd); | 
|  | return -EINVAL; | 
|  | break; | 
|  | } | 
|  | exit: | 
|  | usb_free_urb(urb); | 
|  | kfree(buffer); | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static const struct file_operations vstusb_fops = { | 
|  | .owner =                THIS_MODULE, | 
|  | .read =                 vstusb_read, | 
|  | .write =                vstusb_write, | 
|  | .unlocked_ioctl =       vstusb_ioctl, | 
|  | .compat_ioctl =         vstusb_ioctl, | 
|  | .open =                 vstusb_open, | 
|  | .release =              vstusb_release, | 
|  | }; | 
|  |  | 
|  | static struct usb_class_driver usb_vstusb_class = { | 
|  | .name =         "usb/vstusb%d", | 
|  | .fops =         &vstusb_fops, | 
|  | .minor_base =   VSTUSB_MINOR_BASE, | 
|  | }; | 
|  |  | 
|  | static int vstusb_probe(struct usb_interface *intf, | 
|  | const struct usb_device_id *id) | 
|  | { | 
|  | struct usb_device *dev = interface_to_usbdev(intf); | 
|  | struct vstusb_device *vstdev; | 
|  | int i; | 
|  | int retval = 0; | 
|  |  | 
|  | /* allocate memory for our device state and intialize it */ | 
|  |  | 
|  | vstdev = kzalloc(sizeof(*vstdev), GFP_KERNEL); | 
|  | if (vstdev == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* must do usb_get_dev() prior to kref_init() since the kref_put() | 
|  | * release function will do a usb_put_dev() */ | 
|  | usb_get_dev(dev); | 
|  | kref_init(&vstdev->kref); | 
|  | mutex_init(&vstdev->lock); | 
|  |  | 
|  | i = dev->descriptor.bcdDevice; | 
|  |  | 
|  | dev_dbg(&intf->dev, "Version %1d%1d.%1d%1d found at address %d\n", | 
|  | (i & 0xF000) >> 12, (i & 0xF00) >> 8, | 
|  | (i & 0xF0) >> 4, (i & 0xF), dev->devnum); | 
|  |  | 
|  | vstdev->present = 1; | 
|  | vstdev->isopen = 0; | 
|  | vstdev->usb_dev = dev; | 
|  | init_usb_anchor(&vstdev->submitted); | 
|  |  | 
|  | usb_set_intfdata(intf, vstdev); | 
|  | retval = usb_register_dev(intf, &usb_vstusb_class); | 
|  | if (retval) { | 
|  | dev_err(&intf->dev, | 
|  | "%s: Not able to get a minor for this device.\n", | 
|  | __func__); | 
|  | usb_set_intfdata(intf, NULL); | 
|  | kref_put(&vstdev->kref, vstusb_delete); | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | /* let the user know what node this device is now attached to */ | 
|  | dev_info(&intf->dev, | 
|  | "VST USB Device #%d now attached to major %d minor %d\n", | 
|  | (intf->minor - VSTUSB_MINOR_BASE), USB_MAJOR, intf->minor); | 
|  |  | 
|  | dev_info(&intf->dev, "%s, %s\n", DRIVER_DESC, DRIVER_VERSION); | 
|  |  | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static void vstusb_disconnect(struct usb_interface *intf) | 
|  | { | 
|  | struct vstusb_device *vstdev = usb_get_intfdata(intf); | 
|  |  | 
|  | usb_deregister_dev(intf, &usb_vstusb_class); | 
|  | usb_set_intfdata(intf, NULL); | 
|  |  | 
|  | if (vstdev) { | 
|  |  | 
|  | mutex_lock(&vstdev->lock); | 
|  | vstdev->present = 0; | 
|  |  | 
|  | usb_kill_anchored_urbs(&vstdev->submitted); | 
|  |  | 
|  | mutex_unlock(&vstdev->lock); | 
|  |  | 
|  | kref_put(&vstdev->kref, vstusb_delete); | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | static int vstusb_suspend(struct usb_interface *intf, pm_message_t message) | 
|  | { | 
|  | struct vstusb_device *vstdev = usb_get_intfdata(intf); | 
|  | int time; | 
|  | if (!vstdev) | 
|  | return 0; | 
|  |  | 
|  | mutex_lock(&vstdev->lock); | 
|  | time = usb_wait_anchor_empty_timeout(&vstdev->submitted, 1000); | 
|  | if (!time) | 
|  | usb_kill_anchored_urbs(&vstdev->submitted); | 
|  | mutex_unlock(&vstdev->lock); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int vstusb_resume(struct usb_interface *intf) | 
|  | { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct usb_driver vstusb_driver = { | 
|  | .name =         "vstusb", | 
|  | .probe =        vstusb_probe, | 
|  | .disconnect =   vstusb_disconnect, | 
|  | .suspend =      vstusb_suspend, | 
|  | .resume =       vstusb_resume, | 
|  | .id_table = id_table, | 
|  | }; | 
|  |  | 
|  | static int __init vstusb_init(void) | 
|  | { | 
|  | int rc; | 
|  |  | 
|  | rc = usb_register(&vstusb_driver); | 
|  | if (rc) | 
|  | printk(KERN_ERR "%s: failed to register (%d)", __func__, rc); | 
|  |  | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | static void __exit vstusb_exit(void) | 
|  | { | 
|  | usb_deregister(&vstusb_driver); | 
|  | } | 
|  |  | 
|  | module_init(vstusb_init); | 
|  | module_exit(vstusb_exit); | 
|  |  | 
|  | MODULE_AUTHOR("Dennis O'Brien/Stephen Ware"); | 
|  | MODULE_DESCRIPTION(DRIVER_VERSION); | 
|  | MODULE_LICENSE("GPL"); |