2005-04-17 00:20:36 +02:00
|
|
|
/*
|
|
|
|
* cdc-acm.c
|
|
|
|
*
|
|
|
|
* Copyright (c) 1999 Armin Fuerst <fuerst@in.tum.de>
|
2010-07-18 14:27:13 +02:00
|
|
|
* Copyright (c) 1999 Pavel Machek <pavel@ucw.cz>
|
2005-04-17 00:20:36 +02:00
|
|
|
* Copyright (c) 1999 Johannes Erdfelt <johannes@erdfelt.com>
|
|
|
|
* Copyright (c) 2000 Vojtech Pavlik <vojtech@suse.cz>
|
|
|
|
* Copyright (c) 2004 Oliver Neukum <oliver@neukum.name>
|
2005-11-01 18:51:34 +01:00
|
|
|
* Copyright (c) 2005 David Kubicek <dave@awk.cz>
|
2005-04-17 00:20:36 +02:00
|
|
|
*
|
|
|
|
* USB Abstract Control Model driver for USB modems and ISDN adapters
|
|
|
|
*
|
|
|
|
* Sponsored by SuSE
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#undef DEBUG
|
2008-08-07 03:46:10 +02:00
|
|
|
#undef VERBOSE_DEBUG
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/tty.h>
|
2009-09-08 23:51:28 +02:00
|
|
|
#include <linux/serial.h>
|
2005-04-17 00:20:36 +02:00
|
|
|
#include <linux/tty_driver.h>
|
|
|
|
#include <linux/tty_flip.h>
|
|
|
|
#include <linux/module.h>
|
2006-01-11 15:55:29 +01:00
|
|
|
#include <linux/mutex.h>
|
2009-06-11 13:36:09 +02:00
|
|
|
#include <linux/uaccess.h>
|
2005-04-17 00:20:36 +02:00
|
|
|
#include <linux/usb.h>
|
2006-06-13 18:57:47 +02:00
|
|
|
#include <linux/usb/cdc.h>
|
2005-04-17 00:20:36 +02:00
|
|
|
#include <asm/byteorder.h>
|
|
|
|
#include <asm/unaligned.h>
|
2005-11-01 18:51:34 +01:00
|
|
|
#include <linux/list.h>
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
#include "cdc-acm.h"
|
|
|
|
|
2008-08-07 03:46:10 +02:00
|
|
|
|
2005-11-01 18:51:34 +01:00
|
|
|
#define DRIVER_AUTHOR "Armin Fuerst, Pavel Machek, Johannes Erdfelt, Vojtech Pavlik, David Kubicek"
|
2005-04-17 00:20:36 +02:00
|
|
|
#define DRIVER_DESC "USB Abstract Control Model driver for USB modems and ISDN adapters"
|
|
|
|
|
|
|
|
static struct usb_driver acm_driver;
|
|
|
|
static struct tty_driver *acm_tty_driver;
|
|
|
|
static struct acm *acm_table[ACM_TTY_MINORS];
|
|
|
|
|
2006-01-11 15:55:29 +01:00
|
|
|
static DEFINE_MUTEX(open_mutex);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
#define ACM_READY(acm) (acm && acm->dev && acm->port.count)
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:27:50 +02:00
|
|
|
static const struct tty_port_operations acm_port_ops = {
|
|
|
|
};
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/*
|
|
|
|
* Functions for ACM control messages.
|
|
|
|
*/
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
static int acm_ctrl_msg(struct acm *acm, int request, int value,
|
|
|
|
void *buf, int len)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
int retval = usb_control_msg(acm->dev, usb_sndctrlpipe(acm->dev, 0),
|
|
|
|
request, USB_RT_ACM, value,
|
|
|
|
acm->control->altsetting[0].desc.bInterfaceNumber,
|
|
|
|
buf, len, 5000);
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - rq 0x%02x, val %#x, len %#x, result %d\n",
|
|
|
|
__func__, request, value, len, retval);
|
2005-04-17 00:20:36 +02:00
|
|
|
return retval < 0 ? retval : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* devices aren't required to support these requests.
|
|
|
|
* the cdc acm descriptor tells whether they do...
|
|
|
|
*/
|
|
|
|
#define acm_set_control(acm, control) \
|
|
|
|
acm_ctrl_msg(acm, USB_CDC_REQ_SET_CONTROL_LINE_STATE, control, NULL, 0)
|
|
|
|
#define acm_set_line(acm, line) \
|
|
|
|
acm_ctrl_msg(acm, USB_CDC_REQ_SET_LINE_CODING, 0, line, sizeof *(line))
|
|
|
|
#define acm_send_break(acm, ms) \
|
|
|
|
acm_ctrl_msg(acm, USB_CDC_REQ_SEND_BREAK, ms, NULL, 0)
|
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
/*
|
|
|
|
* Write buffer management.
|
|
|
|
* All of these assume proper locks taken by the caller.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int acm_wb_alloc(struct acm *acm)
|
|
|
|
{
|
|
|
|
int i, wbn;
|
|
|
|
struct acm_wb *wb;
|
|
|
|
|
2008-03-20 10:01:34 +01:00
|
|
|
wbn = 0;
|
2005-04-21 21:28:02 +02:00
|
|
|
i = 0;
|
|
|
|
for (;;) {
|
|
|
|
wb = &acm->wb[wbn];
|
|
|
|
if (!wb->use) {
|
|
|
|
wb->use = 1;
|
|
|
|
return wbn;
|
|
|
|
}
|
2006-05-13 22:50:47 +02:00
|
|
|
wbn = (wbn + 1) % ACM_NW;
|
|
|
|
if (++i >= ACM_NW)
|
2005-04-21 21:28:02 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int acm_wb_is_avail(struct acm *acm)
|
|
|
|
{
|
|
|
|
int i, n;
|
2008-08-07 03:46:10 +02:00
|
|
|
unsigned long flags;
|
2005-04-21 21:28:02 +02:00
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
n = ACM_NW;
|
2008-08-07 03:46:10 +02:00
|
|
|
spin_lock_irqsave(&acm->write_lock, flags);
|
2009-06-11 13:37:06 +02:00
|
|
|
for (i = 0; i < ACM_NW; i++)
|
2006-05-13 22:50:47 +02:00
|
|
|
n -= acm->wb[i].use;
|
2008-08-07 03:46:10 +02:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
2005-04-21 21:28:02 +02:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2008-11-06 20:19:11 +01:00
|
|
|
* Finish write. Caller must hold acm->write_lock
|
2005-04-21 21:28:02 +02:00
|
|
|
*/
|
2008-03-20 10:01:34 +01:00
|
|
|
static void acm_write_done(struct acm *acm, struct acm_wb *wb)
|
2005-04-21 21:28:02 +02:00
|
|
|
{
|
2008-03-20 10:01:34 +01:00
|
|
|
wb->use = 0;
|
2008-06-20 11:25:57 +02:00
|
|
|
acm->transmitting--;
|
2009-12-16 17:05:57 +01:00
|
|
|
usb_autopm_put_interface_async(acm->control);
|
2005-04-21 21:28:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Poke write.
|
2008-06-20 11:25:57 +02:00
|
|
|
*
|
|
|
|
* the caller is responsible for locking
|
2005-04-21 21:28:02 +02:00
|
|
|
*/
|
2008-06-20 11:25:57 +02:00
|
|
|
|
|
|
|
static int acm_start_wb(struct acm *acm, struct acm_wb *wb)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
acm->transmitting++;
|
|
|
|
|
|
|
|
wb->urb->transfer_buffer = wb->buf;
|
|
|
|
wb->urb->transfer_dma = wb->dmah;
|
|
|
|
wb->urb->transfer_buffer_length = wb->len;
|
|
|
|
wb->urb->dev = acm->dev;
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
rc = usb_submit_urb(wb->urb, GFP_ATOMIC);
|
|
|
|
if (rc < 0) {
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_err(&acm->data->dev,
|
|
|
|
"%s - usb_submit_urb(write bulk) failed: %d\n",
|
|
|
|
__func__, rc);
|
2008-06-20 11:25:57 +02:00
|
|
|
acm_write_done(acm, wb);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2008-03-20 10:01:34 +01:00
|
|
|
static int acm_write_start(struct acm *acm, int wbn)
|
2005-04-21 21:28:02 +02:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
2008-08-07 03:44:12 +02:00
|
|
|
struct acm_wb *wb = &acm->wb[wbn];
|
2005-04-21 21:28:02 +02:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&acm->write_lock, flags);
|
|
|
|
if (!acm->dev) {
|
2008-08-07 03:44:12 +02:00
|
|
|
wb->use = 0;
|
2005-04-21 21:28:02 +02:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s - susp_count %d\n", __func__,
|
2011-03-22 11:12:15 +01:00
|
|
|
acm->susp_count);
|
2009-12-16 17:05:57 +01:00
|
|
|
usb_autopm_get_interface_async(acm->control);
|
2008-06-20 11:25:57 +02:00
|
|
|
if (acm->susp_count) {
|
2009-12-16 17:05:57 +01:00
|
|
|
if (!acm->delayed_wb)
|
|
|
|
acm->delayed_wb = wb;
|
|
|
|
else
|
|
|
|
usb_autopm_put_interface_async(acm->control);
|
2008-06-20 11:25:57 +02:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
|
|
|
return 0; /* A white lie */
|
|
|
|
}
|
|
|
|
usb_mark_last_busy(acm->dev);
|
|
|
|
|
|
|
|
rc = acm_start_wb(acm, wb);
|
2005-04-21 21:28:02 +02:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
|
|
|
|
|
|
|
return rc;
|
2008-06-20 11:25:57 +02:00
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
}
|
2007-02-27 15:28:55 +01:00
|
|
|
/*
|
|
|
|
* attributes exported through sysfs
|
|
|
|
*/
|
|
|
|
static ssize_t show_caps
|
|
|
|
(struct device *dev, struct device_attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
struct usb_interface *intf = to_usb_interface(dev);
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
|
|
|
|
|
|
|
return sprintf(buf, "%d", acm->ctrl_caps);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR(bmCapabilities, S_IRUGO, show_caps, NULL);
|
|
|
|
|
|
|
|
static ssize_t show_country_codes
|
|
|
|
(struct device *dev, struct device_attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
struct usb_interface *intf = to_usb_interface(dev);
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
|
|
|
|
|
|
|
memcpy(buf, acm->country_codes, acm->country_code_size);
|
|
|
|
return acm->country_code_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DEVICE_ATTR(wCountryCodes, S_IRUGO, show_country_codes, NULL);
|
|
|
|
|
|
|
|
static ssize_t show_country_rel_date
|
|
|
|
(struct device *dev, struct device_attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
struct usb_interface *intf = to_usb_interface(dev);
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
|
|
|
|
|
|
|
return sprintf(buf, "%d", acm->country_rel_date);
|
|
|
|
}
|
2005-04-21 21:28:02 +02:00
|
|
|
|
2007-02-27 15:28:55 +01:00
|
|
|
static DEVICE_ATTR(iCountryCodeRelDate, S_IRUGO, show_country_rel_date, NULL);
|
2005-04-17 00:20:36 +02:00
|
|
|
/*
|
|
|
|
* Interrupt handlers for various ACM device responses
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* control interface reports status changes with "interrupt" transfers */
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 15:55:46 +02:00
|
|
|
static void acm_ctrl_irq(struct urb *urb)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = urb->context;
|
|
|
|
struct usb_cdc_notification *dr = urb->transfer_buffer;
|
2009-06-11 13:36:09 +02:00
|
|
|
struct tty_struct *tty;
|
2005-04-17 00:20:36 +02:00
|
|
|
unsigned char *data;
|
|
|
|
int newctrl;
|
2007-07-18 19:58:02 +02:00
|
|
|
int retval;
|
|
|
|
int status = urb->status;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2007-07-18 19:58:02 +02:00
|
|
|
switch (status) {
|
2005-04-17 00:20:36 +02:00
|
|
|
case 0:
|
|
|
|
/* success */
|
|
|
|
break;
|
|
|
|
case -ECONNRESET:
|
|
|
|
case -ENOENT:
|
|
|
|
case -ESHUTDOWN:
|
|
|
|
/* this urb is terminated, clean up */
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - urb shutting down with status: %d\n",
|
|
|
|
__func__, status);
|
2005-04-17 00:20:36 +02:00
|
|
|
return;
|
|
|
|
default:
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - nonzero urb status received: %d\n",
|
|
|
|
__func__, status);
|
2005-04-17 00:20:36 +02:00
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
goto exit;
|
|
|
|
|
2011-03-22 11:12:11 +01:00
|
|
|
usb_mark_last_busy(acm->dev);
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
data = (unsigned char *)(dr + 1);
|
|
|
|
switch (dr->bNotificationType) {
|
2009-06-11 13:37:06 +02:00
|
|
|
case USB_CDC_NOTIFY_NETWORK_CONNECTION:
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev, "%s - network connection: %d\n",
|
|
|
|
__func__, dr->wValue);
|
2009-06-11 13:37:06 +02:00
|
|
|
break;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
case USB_CDC_NOTIFY_SERIAL_STATE:
|
|
|
|
tty = tty_port_tty_get(&acm->port);
|
|
|
|
newctrl = get_unaligned_le16(data);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
if (tty) {
|
|
|
|
if (!acm->clocal &&
|
|
|
|
(acm->ctrlin & ~newctrl & ACM_CTRL_DCD)) {
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - calling hangup\n", __func__);
|
2009-06-11 13:37:06 +02:00
|
|
|
tty_hangup(tty);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
2009-06-11 13:37:06 +02:00
|
|
|
tty_kref_put(tty);
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
acm->ctrlin = newctrl;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - input control lines: dcd%c dsr%c break%c "
|
|
|
|
"ring%c framing%c parity%c overrun%c\n",
|
|
|
|
__func__,
|
2009-06-11 13:37:06 +02:00
|
|
|
acm->ctrlin & ACM_CTRL_DCD ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_DSR ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_BRK ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_RI ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_FRAMING ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_PARITY ? '+' : '-',
|
|
|
|
acm->ctrlin & ACM_CTRL_OVERRUN ? '+' : '-');
|
2005-04-17 00:20:36 +02:00
|
|
|
break;
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
default:
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev,
|
|
|
|
"%s - unknown notification %d received: index %d "
|
|
|
|
"len %d data0 %d data1 %d\n",
|
|
|
|
__func__,
|
2009-06-11 13:37:06 +02:00
|
|
|
dr->bNotificationType, dr->wIndex,
|
|
|
|
dr->wLength, data[0], data[1]);
|
|
|
|
break;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
exit:
|
2009-06-11 13:37:06 +02:00
|
|
|
retval = usb_submit_urb(urb, GFP_ATOMIC);
|
2007-07-18 19:58:02 +02:00
|
|
|
if (retval)
|
2011-03-22 11:12:14 +01:00
|
|
|
dev_err(&acm->control->dev, "%s - usb_submit_urb failed: %d\n",
|
|
|
|
__func__, retval);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* data interface returns incoming bytes, or we got unthrottled */
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 15:55:46 +02:00
|
|
|
static void acm_read_bulk(struct urb *urb)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
2005-11-01 18:51:34 +01:00
|
|
|
struct acm_rb *buf;
|
|
|
|
struct acm_ru *rcv = urb->context;
|
|
|
|
struct acm *acm = rcv->instance;
|
2006-05-13 22:50:47 +02:00
|
|
|
int status = urb->status;
|
2007-07-18 19:58:02 +02:00
|
|
|
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s - status %d\n", __func__, status);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2008-06-20 11:25:57 +02:00
|
|
|
if (!ACM_READY(acm)) {
|
2011-03-22 11:12:14 +01:00
|
|
|
dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
|
2005-04-17 00:20:36 +02:00
|
|
|
return;
|
2008-06-20 11:25:57 +02:00
|
|
|
}
|
|
|
|
usb_mark_last_busy(acm->dev);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
if (status)
|
2011-03-22 11:12:14 +01:00
|
|
|
dev_dbg(&acm->data->dev, "%s - non-zero urb status: %d\n",
|
|
|
|
__func__, status);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2005-11-01 18:51:34 +01:00
|
|
|
buf = rcv->buffer;
|
|
|
|
buf->size = urb->actual_length;
|
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
if (likely(status == 0)) {
|
|
|
|
spin_lock(&acm->read_lock);
|
2008-06-20 11:25:57 +02:00
|
|
|
acm->processing++;
|
2006-05-13 22:50:47 +02:00
|
|
|
list_add_tail(&rcv->list, &acm->spare_read_urbs);
|
|
|
|
list_add_tail(&buf->list, &acm->filled_read_bufs);
|
|
|
|
spin_unlock(&acm->read_lock);
|
|
|
|
} else {
|
|
|
|
/* we drop the buffer due to an error */
|
|
|
|
spin_lock(&acm->read_lock);
|
|
|
|
list_add_tail(&rcv->list, &acm->spare_read_urbs);
|
|
|
|
list_add(&buf->list, &acm->spare_read_bufs);
|
|
|
|
spin_unlock(&acm->read_lock);
|
|
|
|
/* nevertheless the tasklet must be kicked unconditionally
|
|
|
|
so the queue cannot dry up */
|
|
|
|
}
|
2008-06-20 11:25:57 +02:00
|
|
|
if (likely(!acm->susp_count))
|
|
|
|
tasklet_schedule(&acm->urb_task);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void acm_rx_tasklet(unsigned long _acm)
|
|
|
|
{
|
|
|
|
struct acm *acm = (void *)_acm;
|
2005-11-01 18:51:34 +01:00
|
|
|
struct acm_rb *buf;
|
2009-06-11 13:36:09 +02:00
|
|
|
struct tty_struct *tty;
|
2005-11-01 18:51:34 +01:00
|
|
|
struct acm_ru *rcv;
|
2006-10-06 07:23:11 +02:00
|
|
|
unsigned long flags;
|
2007-02-12 08:41:35 +01:00
|
|
|
unsigned char throttled;
|
2008-06-20 11:25:57 +02:00
|
|
|
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s\n", __func__);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
if (!ACM_READY(acm)) {
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
|
2007-02-12 08:41:35 +01:00
|
|
|
return;
|
2008-06-20 11:25:57 +02:00
|
|
|
}
|
2007-02-12 08:41:35 +01:00
|
|
|
|
2007-03-06 10:47:04 +01:00
|
|
|
spin_lock_irqsave(&acm->throttle_lock, flags);
|
2007-02-12 08:41:35 +01:00
|
|
|
throttled = acm->throttle;
|
2007-03-06 10:47:04 +01:00
|
|
|
spin_unlock_irqrestore(&acm->throttle_lock, flags);
|
2009-06-11 13:36:09 +02:00
|
|
|
if (throttled) {
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->data->dev, "%s - throttled\n", __func__);
|
2005-11-01 18:51:34 +01:00
|
|
|
return;
|
2008-06-20 11:25:57 +02:00
|
|
|
}
|
2005-11-01 18:51:34 +01:00
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
tty = tty_port_tty_get(&acm->port);
|
|
|
|
|
2005-11-01 18:51:34 +01:00
|
|
|
next_buffer:
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
if (list_empty(&acm->filled_read_bufs)) {
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
goto urbs;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
2005-11-01 18:51:34 +01:00
|
|
|
buf = list_entry(acm->filled_read_bufs.next,
|
|
|
|
struct acm_rb, list);
|
|
|
|
list_del(&buf->list);
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s - processing buf 0x%p, size = %d\n",
|
2011-03-22 11:12:15 +01:00
|
|
|
__func__, buf, buf->size);
|
2009-06-11 13:36:09 +02:00
|
|
|
if (tty) {
|
|
|
|
spin_lock_irqsave(&acm->throttle_lock, flags);
|
|
|
|
throttled = acm->throttle;
|
|
|
|
spin_unlock_irqrestore(&acm->throttle_lock, flags);
|
|
|
|
if (!throttled) {
|
|
|
|
tty_insert_flip_string(tty, buf->base, buf->size);
|
|
|
|
tty_flip_buffer_push(tty);
|
|
|
|
} else {
|
|
|
|
tty_kref_put(tty);
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->data->dev, "%s - throttling noticed\n",
|
|
|
|
__func__);
|
2009-06-11 13:36:09 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
|
|
|
list_add(&buf->list, &acm->filled_read_bufs);
|
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
list_add(&buf->list, &acm->spare_read_bufs);
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
goto next_buffer;
|
|
|
|
|
|
|
|
urbs:
|
2009-06-11 13:36:09 +02:00
|
|
|
tty_kref_put(tty);
|
|
|
|
|
2005-11-01 18:51:34 +01:00
|
|
|
while (!list_empty(&acm->spare_read_bufs)) {
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
if (list_empty(&acm->spare_read_urbs)) {
|
2008-06-20 11:25:57 +02:00
|
|
|
acm->processing = 0;
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
rcv = list_entry(acm->spare_read_urbs.next,
|
|
|
|
struct acm_ru, list);
|
|
|
|
list_del(&rcv->list);
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
|
|
|
|
buf = list_entry(acm->spare_read_bufs.next,
|
|
|
|
struct acm_rb, list);
|
|
|
|
list_del(&buf->list);
|
|
|
|
|
|
|
|
rcv->buffer = buf;
|
|
|
|
|
2009-08-04 23:52:09 +02:00
|
|
|
if (acm->is_int_ep)
|
2009-07-01 14:27:26 +02:00
|
|
|
usb_fill_int_urb(rcv->urb, acm->dev,
|
|
|
|
acm->rx_endpoint,
|
|
|
|
buf->base,
|
|
|
|
acm->readsize,
|
2009-08-04 23:52:09 +02:00
|
|
|
acm_read_bulk, rcv, acm->bInterval);
|
2009-07-01 14:27:26 +02:00
|
|
|
else
|
|
|
|
usb_fill_bulk_urb(rcv->urb, acm->dev,
|
|
|
|
acm->rx_endpoint,
|
|
|
|
buf->base,
|
|
|
|
acm->readsize,
|
|
|
|
acm_read_bulk, rcv);
|
2005-11-01 18:51:34 +01:00
|
|
|
rcv->urb->transfer_dma = buf->dma;
|
|
|
|
rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
/* This shouldn't kill the driver as unsuccessful URBs are
|
|
|
|
returned to the free-urbs-pool and resubmited ASAP */
|
2008-06-20 11:25:57 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
2009-06-11 13:37:06 +02:00
|
|
|
if (acm->susp_count ||
|
|
|
|
usb_submit_urb(rcv->urb, GFP_ATOMIC) < 0) {
|
2005-11-01 18:51:34 +01:00
|
|
|
list_add(&buf->list, &acm->spare_read_bufs);
|
|
|
|
list_add(&rcv->list, &acm->spare_read_urbs);
|
2008-06-20 11:25:57 +02:00
|
|
|
acm->processing = 0;
|
2006-10-06 07:23:11 +02:00
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-11-01 18:51:34 +01:00
|
|
|
return;
|
2008-06-20 11:25:57 +02:00
|
|
|
} else {
|
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev,
|
2011-03-22 11:12:15 +01:00
|
|
|
"%s - sending urb 0x%p, rcv 0x%p, buf 0x%p\n",
|
|
|
|
__func__, rcv->urb, rcv, buf);
|
2005-11-01 18:51:34 +01:00
|
|
|
}
|
|
|
|
}
|
2008-06-20 11:25:57 +02:00
|
|
|
spin_lock_irqsave(&acm->read_lock, flags);
|
|
|
|
acm->processing = 0;
|
|
|
|
spin_unlock_irqrestore(&acm->read_lock, flags);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* data interface wrote those outgoing bytes */
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 15:55:46 +02:00
|
|
|
static void acm_write_bulk(struct urb *urb)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
2008-02-24 11:41:47 +01:00
|
|
|
struct acm_wb *wb = urb->context;
|
2008-08-07 03:46:10 +02:00
|
|
|
struct acm *acm = wb->instance;
|
2008-11-06 20:19:11 +01:00
|
|
|
unsigned long flags;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2011-03-22 11:12:16 +01:00
|
|
|
if (urb->status || (urb->actual_length != urb->transfer_buffer_length))
|
|
|
|
dev_vdbg(&acm->data->dev, "%s - len %d/%d, status %d\n",
|
2011-03-22 11:12:14 +01:00
|
|
|
__func__,
|
2008-08-07 03:46:10 +02:00
|
|
|
urb->actual_length,
|
|
|
|
urb->transfer_buffer_length,
|
|
|
|
urb->status);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2008-11-06 20:19:11 +01:00
|
|
|
spin_lock_irqsave(&acm->write_lock, flags);
|
2008-03-20 10:01:34 +01:00
|
|
|
acm_write_done(acm, wb);
|
2008-11-06 20:19:11 +01:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
2005-04-21 21:28:02 +02:00
|
|
|
if (ACM_READY(acm))
|
|
|
|
schedule_work(&acm->work);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2006-11-22 15:57:56 +01:00
|
|
|
static void acm_softint(struct work_struct *work)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
2006-11-22 15:57:56 +01:00
|
|
|
struct acm *acm = container_of(work, struct acm, work);
|
2009-06-11 13:36:09 +02:00
|
|
|
struct tty_struct *tty;
|
2008-08-07 03:46:10 +02:00
|
|
|
|
2011-03-22 11:12:14 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s\n", __func__);
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return;
|
2009-06-11 13:36:09 +02:00
|
|
|
tty = tty_port_tty_get(&acm->port);
|
2011-03-22 11:12:10 +01:00
|
|
|
if (!tty)
|
|
|
|
return;
|
2009-06-11 13:36:09 +02:00
|
|
|
tty_wakeup(tty);
|
|
|
|
tty_kref_put(tty);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TTY handlers
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int acm_tty_open(struct tty_struct *tty, struct file *filp)
|
|
|
|
{
|
|
|
|
struct acm *acm;
|
2009-06-25 15:41:24 +02:00
|
|
|
int rv = -ENODEV;
|
2005-11-01 18:51:34 +01:00
|
|
|
int i;
|
2006-01-11 15:55:29 +01:00
|
|
|
|
|
|
|
mutex_lock(&open_mutex);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
acm = acm_table[tty->index];
|
|
|
|
if (!acm || !acm->dev)
|
2010-02-03 17:10:22 +01:00
|
|
|
goto out;
|
2005-04-17 00:20:36 +02:00
|
|
|
else
|
|
|
|
rv = 0;
|
|
|
|
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev, "%s\n", __func__);
|
|
|
|
|
2008-03-20 10:53:52 +01:00
|
|
|
set_bit(TTY_NO_WRITE_SPLIT, &tty->flags);
|
2009-06-11 13:36:09 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
tty->driver_data = acm;
|
2009-06-11 13:36:09 +02:00
|
|
|
tty_port_tty_set(&acm->port, tty);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2008-02-11 15:22:29 +01:00
|
|
|
if (usb_autopm_get_interface(acm->control) < 0)
|
|
|
|
goto early_bail;
|
2008-06-20 11:25:57 +02:00
|
|
|
else
|
|
|
|
acm->control->needs_remote_wakeup = 1;
|
2007-10-12 17:24:28 +02:00
|
|
|
|
|
|
|
mutex_lock(&acm->mutex);
|
2009-06-11 13:36:09 +02:00
|
|
|
if (acm->port.count++) {
|
2010-02-03 17:10:22 +01:00
|
|
|
mutex_unlock(&acm->mutex);
|
2007-10-12 17:24:28 +02:00
|
|
|
usb_autopm_put_interface(acm->control);
|
2010-02-03 17:10:22 +01:00
|
|
|
goto out;
|
2009-06-11 13:36:09 +02:00
|
|
|
}
|
2007-10-12 17:24:28 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->ctrlurb->dev = acm->dev;
|
|
|
|
if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) {
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_err(&acm->control->dev,
|
|
|
|
"%s - usb_submit_urb(ctrl irq) failed\n", __func__);
|
2005-04-17 00:20:36 +02:00
|
|
|
goto bail_out;
|
|
|
|
}
|
|
|
|
|
2007-02-12 08:41:35 +01:00
|
|
|
if (0 > acm_set_control(acm, acm->ctrlout = ACM_CTRL_DTR | ACM_CTRL_RTS) &&
|
|
|
|
(acm->ctrl_caps & USB_CDC_CAP_LINE))
|
2005-04-17 00:20:36 +02:00
|
|
|
goto full_bailout;
|
2009-06-11 13:36:09 +02:00
|
|
|
|
2008-06-20 11:25:57 +02:00
|
|
|
usb_autopm_put_interface(acm->control);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2005-11-01 18:51:34 +01:00
|
|
|
INIT_LIST_HEAD(&acm->spare_read_urbs);
|
|
|
|
INIT_LIST_HEAD(&acm->spare_read_bufs);
|
|
|
|
INIT_LIST_HEAD(&acm->filled_read_bufs);
|
2009-06-11 13:37:06 +02:00
|
|
|
|
|
|
|
for (i = 0; i < acm->rx_buflimit; i++)
|
2005-11-01 18:51:34 +01:00
|
|
|
list_add(&(acm->ru[i].list), &acm->spare_read_urbs);
|
2009-06-11 13:37:06 +02:00
|
|
|
for (i = 0; i < acm->rx_buflimit; i++)
|
2005-11-01 18:51:34 +01:00
|
|
|
list_add(&(acm->rb[i].list), &acm->spare_read_bufs);
|
|
|
|
|
2007-02-12 08:41:35 +01:00
|
|
|
acm->throttle = 0;
|
|
|
|
|
2009-09-08 23:51:28 +02:00
|
|
|
set_bit(ASYNCB_INITIALIZED, &acm->port.flags);
|
2009-06-11 13:36:09 +02:00
|
|
|
rv = tty_port_block_til_ready(&acm->port, tty, filp);
|
2009-11-04 11:19:28 +01:00
|
|
|
tasklet_schedule(&acm->urb_task);
|
2010-02-03 17:10:22 +01:00
|
|
|
|
2007-10-12 17:24:28 +02:00
|
|
|
mutex_unlock(&acm->mutex);
|
2010-02-03 17:10:22 +01:00
|
|
|
out:
|
2008-02-11 15:22:29 +01:00
|
|
|
mutex_unlock(&open_mutex);
|
2005-04-17 00:20:36 +02:00
|
|
|
return rv;
|
|
|
|
|
|
|
|
full_bailout:
|
|
|
|
usb_kill_urb(acm->ctrlurb);
|
|
|
|
bail_out:
|
2009-06-11 13:36:09 +02:00
|
|
|
acm->port.count--;
|
2007-10-12 17:24:28 +02:00
|
|
|
mutex_unlock(&acm->mutex);
|
2010-02-03 17:10:22 +01:00
|
|
|
usb_autopm_put_interface(acm->control);
|
2008-02-11 15:22:29 +01:00
|
|
|
early_bail:
|
|
|
|
mutex_unlock(&open_mutex);
|
2009-06-11 13:36:09 +02:00
|
|
|
tty_port_tty_set(&acm->port, NULL);
|
2005-04-17 00:20:36 +02:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2005-06-30 01:53:29 +02:00
|
|
|
static void acm_tty_unregister(struct acm *acm)
|
|
|
|
{
|
2009-06-11 13:36:09 +02:00
|
|
|
int i, nr;
|
2005-11-01 18:51:34 +01:00
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
nr = acm->rx_buflimit;
|
2005-06-30 01:53:29 +02:00
|
|
|
tty_unregister_device(acm_tty_driver, acm->minor);
|
|
|
|
usb_put_intf(acm->control);
|
|
|
|
acm_table[acm->minor] = NULL;
|
|
|
|
usb_free_urb(acm->ctrlurb);
|
2008-03-20 10:01:34 +01:00
|
|
|
for (i = 0; i < ACM_NW; i++)
|
|
|
|
usb_free_urb(acm->wb[i].urb);
|
2006-05-13 22:50:47 +02:00
|
|
|
for (i = 0; i < nr; i++)
|
2005-11-01 18:51:34 +01:00
|
|
|
usb_free_urb(acm->ru[i].urb);
|
2007-02-27 15:28:55 +01:00
|
|
|
kfree(acm->country_codes);
|
2005-06-30 01:53:29 +02:00
|
|
|
kfree(acm);
|
|
|
|
}
|
|
|
|
|
2010-06-01 22:53:04 +02:00
|
|
|
static void acm_port_down(struct acm *acm)
|
2009-06-11 13:36:09 +02:00
|
|
|
{
|
|
|
|
int i, nr = acm->rx_buflimit;
|
|
|
|
mutex_lock(&open_mutex);
|
|
|
|
if (acm->dev) {
|
|
|
|
usb_autopm_get_interface(acm->control);
|
|
|
|
acm_set_control(acm, acm->ctrlout = 0);
|
|
|
|
usb_kill_urb(acm->ctrlurb);
|
|
|
|
for (i = 0; i < ACM_NW; i++)
|
|
|
|
usb_kill_urb(acm->wb[i].urb);
|
USB: cdc-acm: fix memory corruption / panic
Prevent read urbs from being resubmitted from tasklet after port close.
The receive tasklet was not disabled on port close, which could lead to
corruption of receive lists on consecutive port open. In particular,
read urbs could be re-submitted before port open, added to free list in
open, and then added a second time to the free list in the completion
handler.
cdc-acm.c: Entering acm_tty_open.
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x3 len: 0x0 result: 0
cdc-acm.c: Entering acm_rx_tasklet
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da280, rcv 0xf57fbc24, buf 0xf57fbd64
cdc-acm.c: set line: 115200 0 0 8
cdc-acm.c: acm_control_msg: rq: 0x20 val: 0x0 len: 0x7 result: 7
cdc-acm.c: acm_tty_close
cdc-acm.c: acm_port_down
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x0 len: 0x0 result: 0
cdc-acm.c: acm_ctrl_irq - urb shutting down with status: -2
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da300, rcv 0xf57fbc10, buf 0xf57fbd50
cdc-acm.c: Entering acm_read_bulk with status -2
cdc_acm 4-1:1.1: Aborting, acm not ready
cdc-acm.c: Entering acm_read_bulk with status -2
cdc_acm 4-1:1.1: Aborting, acm not ready
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da380, rcv 0xf57fbbfc, buf 0xf57fbd3c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da400, rcv 0xf57fbbe8, buf 0xf57fbd28
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da480, rcv 0xf57fbbd4, buf 0xf57fbd14
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da900, rcv 0xf57fbbc0, buf 0xf57fbd00
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da980, rcv 0xf57fbbac, buf 0xf57fbcec
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50daa00, rcv 0xf57fbb98, buf 0xf57fbcd8
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50daa80, rcv 0xf57fbb84, buf 0xf57fbcc4
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dab00, rcv 0xf57fbb70, buf 0xf57fbcb0
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dab80, rcv 0xf57fbb5c, buf 0xf57fbc9c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dac00, rcv 0xf57fbb48, buf 0xf57fbc88
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dac80, rcv 0xf57fbb34, buf 0xf57fbc74
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dad00, rcv 0xf57fbb20, buf 0xf57fbc60
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dad80, rcv 0xf57fbb0c, buf 0xf57fbc4c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da880, rcv 0xf57fbaf8, buf 0xf57fbc38
cdc-acm.c: Entering acm_tty_open.
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x3 len: 0x0 result: 0
cdc-acm.c: Entering acm_rx_tasklet
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da280, rcv 0xf57fbc24, buf 0xf57fbd64
cdc-acm.c: Entering acm_tty_write to write 3 bytes,
cdc-acm.c: Get 3 bytes...
cdc-acm.c: acm_write_start susp_count: 0
cdc-acm.c: Entering acm_read_bulk with status 0
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:57 list_del+0x10c/0x120()
Hardware name: Vostro 1520
list_del corruption. next->prev should be f57fbc10, but was f57fbaf8
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Not tainted 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd8ac>] list_del+0x10c/0x120
[<f8051dbf>] acm_rx_tasklet+0xef/0x3e0 [cdc_acm]
[<c135465d>] ? net_rps_action_and_irq_enable+0x6d/0x80
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f0082e ]---
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:57 list_del+0x10c/0x120()
Hardware name: Vostro 1520
list_del corruption. next->prev should be f57fbd50, but was f57fbdb0
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd8ac>] list_del+0x10c/0x120
[<f8051dd6>] acm_rx_tasklet+0x106/0x3e0 [cdc_acm]
[<c135465d>] ? net_rps_action_and_irq_enable+0x6d/0x80
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f0082f ]---
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da300, rcv 0xf57fbc10, buf 0xf57fbd50
cdc-acm.c: disconnected from network
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da380, rcv 0xf57fbbfc, buf 0xf57fbd3c
cdc-acm.c: Entering acm_rx_tasklet
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:48 list_del+0xd5/0x120()
Hardware name: Vostro 1520
list_del corruption, next is LIST_POISON1 (00100100)
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd875>] ? list_del+0xd5/0x120
[<c11dd875>] ? list_del+0xd5/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd875>] list_del+0xd5/0x120
[<f8051fac>] acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f00830 ]---
BUG: unable to handle kernel paging request at 00200200
IP: [<c11dd7bd>] list_del+0x1d/0x120
*pde = 00000000
Oops: 0000 [#1] PREEMPT SMP
last sysfs file: /sys/devices/pci0000:00/0000:00:1a.1/usb4/4-1/4-1:1.0/tty/ttyACM0/uevent
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39 0T816J/Vostro 1520
EIP: 0060:[<c11dd7bd>] EFLAGS: 00010046 CPU: 0
EIP is at list_del+0x1d/0x120
EAX: f57fbd3c EBX: f57fb800 ECX: ffff8000 EDX: 00200200
ESI: f57fbe90 EDI: f57fbd3c EBP: f600bf54 ESP: f600bf3c
DS: 007b ES: 007b FS: 00d8 GS: 0000 SS: 0068
Process ksoftirqd/0 (pid: 3, ti=f600a000 task=f60791c0 task.ti=f6082000)
Stack:
c1527e84 00000030 c1527e54 00100100 f57fb800 f57fbd3c f600bf98 f8051fac
f8053104 f8052b94 f600bf6c c106dbab f600bf80 00000286 f60791c0 c1042b30
f57fbda8 f57f5800 f57fbdb0 f57fbd80 f57fbe7c c1656b04 00000000 f600bfb0
Call Trace:
[<f8051fac>] ? acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] ? tasklet_action+0xe6/0x140
[<c104342f>] ? __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ>
[<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
Code: ff 48 14 e9 57 ff ff ff 90 90 90 90 90 90 55 89 e5 83 ec 18 81 38 00 01 10 00 0f 84 9c 00 00 00 8b 50 04 81 fa 00 02 20 00 74 33 <8b> 12 39 d0 75 5c 8b 10 8b 4a 04 39 c8 0f 85 b5 00 00 00 8b 48
EIP: [<c11dd7bd>] list_del+0x1d/0x120 SS:ESP 0068:f600bf3c
CR2: 0000000000200200
---[ end trace efd9a11434f00831 ]---
Kernel panic - not syncing: Fatal exception in interrupt
Pid: 3, comm: ksoftirqd/0 Tainted: G D W 2.6.37+ #39
Call Trace:
[<c13fede1>] ? printk+0x1d/0x24
[<c13fecce>] panic+0x66/0x15c
[<c10067df>] oops_end+0x8f/0x90
[<c1025476>] no_context+0xc6/0x160
[<c10255a8>] __bad_area_nosemaphore+0x98/0x140
[<c103cf68>] ? release_console_sem+0x1d8/0x210
[<c1025667>] bad_area_nosemaphore+0x17/0x20
[<c1025a49>] do_page_fault+0x279/0x420
[<c1006a8f>] ? show_trace+0x1f/0x30
[<c13fede1>] ? printk+0x1d/0x24
[<c10257d0>] ? do_page_fault+0x0/0x420
[<c140333b>] error_code+0x5f/0x64
[<c103007b>] ? select_task_rq_fair+0x37b/0x6a0
[<c10257d0>] ? do_page_fault+0x0/0x420
[<c11dd7bd>] ? list_del+0x1d/0x120
[<f8051fac>] acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
panic occurred, switching back to text console
------------[ cut here ]------------
Cc: stable <stable@kernel.org>
Signed-off-by: Johan Hovold <jhovold@gmail.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-03-22 11:12:09 +01:00
|
|
|
tasklet_disable(&acm->urb_task);
|
2009-06-11 13:36:09 +02:00
|
|
|
for (i = 0; i < nr; i++)
|
|
|
|
usb_kill_urb(acm->ru[i].urb);
|
USB: cdc-acm: fix memory corruption / panic
Prevent read urbs from being resubmitted from tasklet after port close.
The receive tasklet was not disabled on port close, which could lead to
corruption of receive lists on consecutive port open. In particular,
read urbs could be re-submitted before port open, added to free list in
open, and then added a second time to the free list in the completion
handler.
cdc-acm.c: Entering acm_tty_open.
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x3 len: 0x0 result: 0
cdc-acm.c: Entering acm_rx_tasklet
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da280, rcv 0xf57fbc24, buf 0xf57fbd64
cdc-acm.c: set line: 115200 0 0 8
cdc-acm.c: acm_control_msg: rq: 0x20 val: 0x0 len: 0x7 result: 7
cdc-acm.c: acm_tty_close
cdc-acm.c: acm_port_down
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x0 len: 0x0 result: 0
cdc-acm.c: acm_ctrl_irq - urb shutting down with status: -2
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da300, rcv 0xf57fbc10, buf 0xf57fbd50
cdc-acm.c: Entering acm_read_bulk with status -2
cdc_acm 4-1:1.1: Aborting, acm not ready
cdc-acm.c: Entering acm_read_bulk with status -2
cdc_acm 4-1:1.1: Aborting, acm not ready
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da380, rcv 0xf57fbbfc, buf 0xf57fbd3c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da400, rcv 0xf57fbbe8, buf 0xf57fbd28
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da480, rcv 0xf57fbbd4, buf 0xf57fbd14
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da900, rcv 0xf57fbbc0, buf 0xf57fbd00
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da980, rcv 0xf57fbbac, buf 0xf57fbcec
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50daa00, rcv 0xf57fbb98, buf 0xf57fbcd8
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50daa80, rcv 0xf57fbb84, buf 0xf57fbcc4
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dab00, rcv 0xf57fbb70, buf 0xf57fbcb0
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dab80, rcv 0xf57fbb5c, buf 0xf57fbc9c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dac00, rcv 0xf57fbb48, buf 0xf57fbc88
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dac80, rcv 0xf57fbb34, buf 0xf57fbc74
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dad00, rcv 0xf57fbb20, buf 0xf57fbc60
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50dad80, rcv 0xf57fbb0c, buf 0xf57fbc4c
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da880, rcv 0xf57fbaf8, buf 0xf57fbc38
cdc-acm.c: Entering acm_tty_open.
cdc-acm.c: acm_control_msg: rq: 0x22 val: 0x3 len: 0x0 result: 0
cdc-acm.c: Entering acm_rx_tasklet
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da280, rcv 0xf57fbc24, buf 0xf57fbd64
cdc-acm.c: Entering acm_tty_write to write 3 bytes,
cdc-acm.c: Get 3 bytes...
cdc-acm.c: acm_write_start susp_count: 0
cdc-acm.c: Entering acm_read_bulk with status 0
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:57 list_del+0x10c/0x120()
Hardware name: Vostro 1520
list_del corruption. next->prev should be f57fbc10, but was f57fbaf8
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Not tainted 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd8ac>] list_del+0x10c/0x120
[<f8051dbf>] acm_rx_tasklet+0xef/0x3e0 [cdc_acm]
[<c135465d>] ? net_rps_action_and_irq_enable+0x6d/0x80
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f0082e ]---
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:57 list_del+0x10c/0x120()
Hardware name: Vostro 1520
list_del corruption. next->prev should be f57fbd50, but was f57fbdb0
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c11dd8ac>] ? list_del+0x10c/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd8ac>] list_del+0x10c/0x120
[<f8051dd6>] acm_rx_tasklet+0x106/0x3e0 [cdc_acm]
[<c135465d>] ? net_rps_action_and_irq_enable+0x6d/0x80
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f0082f ]---
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da300, rcv 0xf57fbc10, buf 0xf57fbd50
cdc-acm.c: disconnected from network
cdc-acm.c: acm_rx_tasklet: sending urb 0xf50da380, rcv 0xf57fbbfc, buf 0xf57fbd3c
cdc-acm.c: Entering acm_rx_tasklet
------------[ cut here ]------------
WARNING: at /home/johan/src/linux/linux-2.6/lib/list_debug.c:48 list_del+0xd5/0x120()
Hardware name: Vostro 1520
list_del corruption, next is LIST_POISON1 (00100100)
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39
Call Trace:
[<c103c7e2>] warn_slowpath_common+0x72/0xa0
[<c11dd875>] ? list_del+0xd5/0x120
[<c11dd875>] ? list_del+0xd5/0x120
[<c103c8b3>] warn_slowpath_fmt+0x33/0x40
[<c11dd875>] list_del+0xd5/0x120
[<f8051fac>] acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
---[ end trace efd9a11434f00830 ]---
BUG: unable to handle kernel paging request at 00200200
IP: [<c11dd7bd>] list_del+0x1d/0x120
*pde = 00000000
Oops: 0000 [#1] PREEMPT SMP
last sysfs file: /sys/devices/pci0000:00/0000:00:1a.1/usb4/4-1/4-1:1.0/tty/ttyACM0/uevent
Modules linked in: cdc_acm
Pid: 3, comm: ksoftirqd/0 Tainted: G W 2.6.37+ #39 0T816J/Vostro 1520
EIP: 0060:[<c11dd7bd>] EFLAGS: 00010046 CPU: 0
EIP is at list_del+0x1d/0x120
EAX: f57fbd3c EBX: f57fb800 ECX: ffff8000 EDX: 00200200
ESI: f57fbe90 EDI: f57fbd3c EBP: f600bf54 ESP: f600bf3c
DS: 007b ES: 007b FS: 00d8 GS: 0000 SS: 0068
Process ksoftirqd/0 (pid: 3, ti=f600a000 task=f60791c0 task.ti=f6082000)
Stack:
c1527e84 00000030 c1527e54 00100100 f57fb800 f57fbd3c f600bf98 f8051fac
f8053104 f8052b94 f600bf6c c106dbab f600bf80 00000286 f60791c0 c1042b30
f57fbda8 f57f5800 f57fbdb0 f57fbd80 f57fbe7c c1656b04 00000000 f600bfb0
Call Trace:
[<f8051fac>] ? acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] ? tasklet_action+0xe6/0x140
[<c104342f>] ? __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ>
[<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
Code: ff 48 14 e9 57 ff ff ff 90 90 90 90 90 90 55 89 e5 83 ec 18 81 38 00 01 10 00 0f 84 9c 00 00 00 8b 50 04 81 fa 00 02 20 00 74 33 <8b> 12 39 d0 75 5c 8b 10 8b 4a 04 39 c8 0f 85 b5 00 00 00 8b 48
EIP: [<c11dd7bd>] list_del+0x1d/0x120 SS:ESP 0068:f600bf3c
CR2: 0000000000200200
---[ end trace efd9a11434f00831 ]---
Kernel panic - not syncing: Fatal exception in interrupt
Pid: 3, comm: ksoftirqd/0 Tainted: G D W 2.6.37+ #39
Call Trace:
[<c13fede1>] ? printk+0x1d/0x24
[<c13fecce>] panic+0x66/0x15c
[<c10067df>] oops_end+0x8f/0x90
[<c1025476>] no_context+0xc6/0x160
[<c10255a8>] __bad_area_nosemaphore+0x98/0x140
[<c103cf68>] ? release_console_sem+0x1d8/0x210
[<c1025667>] bad_area_nosemaphore+0x17/0x20
[<c1025a49>] do_page_fault+0x279/0x420
[<c1006a8f>] ? show_trace+0x1f/0x30
[<c13fede1>] ? printk+0x1d/0x24
[<c10257d0>] ? do_page_fault+0x0/0x420
[<c140333b>] error_code+0x5f/0x64
[<c103007b>] ? select_task_rq_fair+0x37b/0x6a0
[<c10257d0>] ? do_page_fault+0x0/0x420
[<c11dd7bd>] ? list_del+0x1d/0x120
[<f8051fac>] acm_rx_tasklet+0x2dc/0x3e0 [cdc_acm]
[<c106dbab>] ? trace_hardirqs_on+0xb/0x10
[<c1042b30>] ? tasklet_action+0x60/0x140
[<c1042bb6>] tasklet_action+0xe6/0x140
[<c104342f>] __do_softirq+0xaf/0x210
[<c1043380>] ? __do_softirq+0x0/0x210
<IRQ> [<c1042c9a>] ? run_ksoftirqd+0x8a/0x1c0
[<c1042c10>] ? run_ksoftirqd+0x0/0x1c0
[<c105ac24>] ? kthread+0x74/0x80
[<c105abb0>] ? kthread+0x0/0x80
[<c100337a>] ? kernel_thread_helper+0x6/0x10
panic occurred, switching back to text console
------------[ cut here ]------------
Cc: stable <stable@kernel.org>
Signed-off-by: Johan Hovold <jhovold@gmail.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-03-22 11:12:09 +01:00
|
|
|
tasklet_enable(&acm->urb_task);
|
2009-06-11 13:36:09 +02:00
|
|
|
acm->control->needs_remote_wakeup = 0;
|
|
|
|
usb_autopm_put_interface(acm->control);
|
|
|
|
}
|
|
|
|
mutex_unlock(&open_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void acm_tty_hangup(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
tty_port_hangup(&acm->port);
|
2010-06-01 22:53:04 +02:00
|
|
|
acm_port_down(acm);
|
2009-06-11 13:36:09 +02:00
|
|
|
}
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
static void acm_tty_close(struct tty_struct *tty, struct file *filp)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
/* Perform the closing process and see if we need to do the hardware
|
|
|
|
shutdown */
|
USB: cdc_acm: Fix memory leak after hangup
Am Donnerstag, 10. September 2009 15:43:53 schrieb Dietmar Hilbrich:
> Hello,
>
> i have the following problem with the cdc-acm - driver:
>
> I'm using the driver with an "Ericsson F3507G" on a Thinkpad T400.
>
> If a disable the device (with the RFKill-Switch) while it is used by a
> programm like ppp, the driver doesn't seem to correctly clean up the tty,
> even after the program has been closed)
>
> The tty is still active (e.g. there still exists an entry in
> /sys/dev/char/166:0 if ttyACM0 was used) and if a reacticate the device,
> this device entry will be skipped and the Device-Nodes ttyACM1, ttyACM2
> and ttyACM3 will be used.
>
> This problem was introduced with the commit
> 10077d4a6674f535abdbe25cdecb1202af7948f1 (before 2.6.31-rc1) and still
> exists in 2.6.31.
>
> I was able the fix this problem with the following patch:
>
> diff --git a/drivers/usb/class/cdc-acm.c b/drivers/usb/class/cdc-acm.c
> index 2bfc41e..0970d2f 100644
> --- a/drivers/usb/class/cdc-acm.c
> +++ b/drivers/usb/class/cdc-acm.c
> @@ -676,6 +676,7 @@ static void acm_tty_hangup(struct tty_struct *tty)
> struct acm *acm = tty->driver_data;
> tty_port_hangup(&acm->port);
> acm_port_down(acm, 0);
> + acm_tty_unregister(acm);
> }
I have the same problem with cdc-acm (I'm using a Samsung SGH-U900): when I
unplug it from the USB port during a PPP connection, the ppp daemon gets the
hangup correctly (and closes the device), but the struct acm corresponding to
the device disconnected is not freed. Hence reconnecting the device results in
creation of /dev/ttyACM(x+1). The same happens when the system is hibernated
during a PPP connection.
This memory leak is due to the fact that when the tty is hung up,
tty_port_close_start() returns always zero, and acm_tty_close() never reaches
the point where acm_tty_unregister() is called.
Here is a fix for this.
Signed-off-by: Francesco Lavra <francescolavra@interfree.it>
Acked-by: Oliver Neukum <oliver@neukum.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2009-11-03 11:53:07 +01:00
|
|
|
if (!acm)
|
2005-04-17 00:20:36 +02:00
|
|
|
return;
|
USB: cdc_acm: Fix memory leak after hangup
Am Donnerstag, 10. September 2009 15:43:53 schrieb Dietmar Hilbrich:
> Hello,
>
> i have the following problem with the cdc-acm - driver:
>
> I'm using the driver with an "Ericsson F3507G" on a Thinkpad T400.
>
> If a disable the device (with the RFKill-Switch) while it is used by a
> programm like ppp, the driver doesn't seem to correctly clean up the tty,
> even after the program has been closed)
>
> The tty is still active (e.g. there still exists an entry in
> /sys/dev/char/166:0 if ttyACM0 was used) and if a reacticate the device,
> this device entry will be skipped and the Device-Nodes ttyACM1, ttyACM2
> and ttyACM3 will be used.
>
> This problem was introduced with the commit
> 10077d4a6674f535abdbe25cdecb1202af7948f1 (before 2.6.31-rc1) and still
> exists in 2.6.31.
>
> I was able the fix this problem with the following patch:
>
> diff --git a/drivers/usb/class/cdc-acm.c b/drivers/usb/class/cdc-acm.c
> index 2bfc41e..0970d2f 100644
> --- a/drivers/usb/class/cdc-acm.c
> +++ b/drivers/usb/class/cdc-acm.c
> @@ -676,6 +676,7 @@ static void acm_tty_hangup(struct tty_struct *tty)
> struct acm *acm = tty->driver_data;
> tty_port_hangup(&acm->port);
> acm_port_down(acm, 0);
> + acm_tty_unregister(acm);
> }
I have the same problem with cdc-acm (I'm using a Samsung SGH-U900): when I
unplug it from the USB port during a PPP connection, the ppp daemon gets the
hangup correctly (and closes the device), but the struct acm corresponding to
the device disconnected is not freed. Hence reconnecting the device results in
creation of /dev/ttyACM(x+1). The same happens when the system is hibernated
during a PPP connection.
This memory leak is due to the fact that when the tty is hung up,
tty_port_close_start() returns always zero, and acm_tty_close() never reaches
the point where acm_tty_unregister() is called.
Here is a fix for this.
Signed-off-by: Francesco Lavra <francescolavra@interfree.it>
Acked-by: Oliver Neukum <oliver@neukum.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2009-11-03 11:53:07 +01:00
|
|
|
if (tty_port_close_start(&acm->port, tty, filp) == 0) {
|
|
|
|
mutex_lock(&open_mutex);
|
|
|
|
if (!acm->dev) {
|
|
|
|
tty_port_tty_set(&acm->port, NULL);
|
|
|
|
acm_tty_unregister(acm);
|
|
|
|
tty->driver_data = NULL;
|
|
|
|
}
|
|
|
|
mutex_unlock(&open_mutex);
|
|
|
|
return;
|
|
|
|
}
|
2010-06-01 22:53:04 +02:00
|
|
|
acm_port_down(acm);
|
2009-06-11 13:36:09 +02:00
|
|
|
tty_port_close_end(&acm->port, tty);
|
|
|
|
tty_port_tty_set(&acm->port, NULL);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
static int acm_tty_write(struct tty_struct *tty,
|
|
|
|
const unsigned char *buf, int count)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
int stat;
|
2005-04-21 21:28:02 +02:00
|
|
|
unsigned long flags;
|
|
|
|
int wbn;
|
|
|
|
struct acm_wb *wb;
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return -EINVAL;
|
|
|
|
if (!count)
|
|
|
|
return 0;
|
|
|
|
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s - count %d\n", __func__, count);
|
2011-03-22 11:12:15 +01:00
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
spin_lock_irqsave(&acm->write_lock, flags);
|
2009-06-11 13:37:06 +02:00
|
|
|
wbn = acm_wb_alloc(acm);
|
|
|
|
if (wbn < 0) {
|
2005-04-21 21:28:02 +02:00
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
wb = &acm->wb[wbn];
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
count = (count > acm->writesize) ? acm->writesize : count;
|
2011-03-22 11:12:17 +01:00
|
|
|
dev_vdbg(&acm->data->dev, "%s - write %d\n", __func__, count);
|
2005-04-21 21:28:02 +02:00
|
|
|
memcpy(wb->buf, buf, count);
|
|
|
|
wb->len = count;
|
|
|
|
spin_unlock_irqrestore(&acm->write_lock, flags);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
stat = acm_write_start(acm, wbn);
|
|
|
|
if (stat < 0)
|
2005-04-17 00:20:36 +02:00
|
|
|
return stat;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int acm_tty_write_room(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return -EINVAL;
|
2005-04-21 21:28:02 +02:00
|
|
|
/*
|
|
|
|
* Do not let the line discipline to know that we have a reserve,
|
|
|
|
* or it might get too enthusiastic.
|
|
|
|
*/
|
2008-08-07 03:44:12 +02:00
|
|
|
return acm_wb_is_avail(acm) ? acm->writesize : 0;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int acm_tty_chars_in_buffer(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
if (!ACM_READY(acm))
|
2009-07-20 17:05:27 +02:00
|
|
|
return 0;
|
2005-04-21 21:28:02 +02:00
|
|
|
/*
|
|
|
|
* This is inaccurate (overcounts), but it works.
|
|
|
|
*/
|
2006-05-13 22:50:47 +02:00
|
|
|
return (ACM_NW - acm_wb_is_avail(acm)) * acm->writesize;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void acm_tty_throttle(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return;
|
|
|
|
spin_lock_bh(&acm->throttle_lock);
|
|
|
|
acm->throttle = 1;
|
|
|
|
spin_unlock_bh(&acm->throttle_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void acm_tty_unthrottle(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return;
|
|
|
|
spin_lock_bh(&acm->throttle_lock);
|
|
|
|
acm->throttle = 0;
|
|
|
|
spin_unlock_bh(&acm->throttle_lock);
|
2005-11-01 18:51:34 +01:00
|
|
|
tasklet_schedule(&acm->urb_task);
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2008-07-22 12:18:03 +02:00
|
|
|
static int acm_tty_break_ctl(struct tty_struct *tty, int state)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
2008-07-22 12:18:03 +02:00
|
|
|
int retval;
|
2005-04-17 00:20:36 +02:00
|
|
|
if (!ACM_READY(acm))
|
2008-07-22 12:18:03 +02:00
|
|
|
return -EINVAL;
|
|
|
|
retval = acm_send_break(acm, state ? 0xffff : 0);
|
|
|
|
if (retval < 0)
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev, "%s - send break failed\n",
|
|
|
|
__func__);
|
2008-07-22 12:18:03 +02:00
|
|
|
return retval;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2011-02-14 17:26:14 +01:00
|
|
|
static int acm_tty_tiocmget(struct tty_struct *tty)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return (acm->ctrlout & ACM_CTRL_DTR ? TIOCM_DTR : 0) |
|
|
|
|
(acm->ctrlout & ACM_CTRL_RTS ? TIOCM_RTS : 0) |
|
|
|
|
(acm->ctrlin & ACM_CTRL_DSR ? TIOCM_DSR : 0) |
|
|
|
|
(acm->ctrlin & ACM_CTRL_RI ? TIOCM_RI : 0) |
|
|
|
|
(acm->ctrlin & ACM_CTRL_DCD ? TIOCM_CD : 0) |
|
|
|
|
TIOCM_CTS;
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:26:50 +01:00
|
|
|
static int acm_tty_tiocmset(struct tty_struct *tty,
|
2005-04-17 00:20:36 +02:00
|
|
|
unsigned int set, unsigned int clear)
|
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
unsigned int newctrl;
|
|
|
|
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
newctrl = acm->ctrlout;
|
2009-06-11 13:37:06 +02:00
|
|
|
set = (set & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
|
|
|
|
(set & TIOCM_RTS ? ACM_CTRL_RTS : 0);
|
|
|
|
clear = (clear & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
|
|
|
|
(clear & TIOCM_RTS ? ACM_CTRL_RTS : 0);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
newctrl = (newctrl & ~clear) | set;
|
|
|
|
|
|
|
|
if (acm->ctrlout == newctrl)
|
|
|
|
return 0;
|
|
|
|
return acm_set_control(acm, acm->ctrlout = newctrl);
|
|
|
|
}
|
|
|
|
|
2011-02-14 17:27:22 +01:00
|
|
|
static int acm_tty_ioctl(struct tty_struct *tty,
|
2009-06-11 13:37:06 +02:00
|
|
|
unsigned int cmd, unsigned long arg)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
|
|
|
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return -ENOIOCTLCMD;
|
|
|
|
}
|
|
|
|
|
2005-11-29 09:43:42 +01:00
|
|
|
static const __u32 acm_tty_speed[] = {
|
2005-04-17 00:20:36 +02:00
|
|
|
0, 50, 75, 110, 134, 150, 200, 300, 600,
|
|
|
|
1200, 1800, 2400, 4800, 9600, 19200, 38400,
|
|
|
|
57600, 115200, 230400, 460800, 500000, 576000,
|
|
|
|
921600, 1000000, 1152000, 1500000, 2000000,
|
|
|
|
2500000, 3000000, 3500000, 4000000
|
|
|
|
};
|
|
|
|
|
2005-11-29 09:43:42 +01:00
|
|
|
static const __u8 acm_tty_size[] = {
|
2005-04-17 00:20:36 +02:00
|
|
|
5, 6, 7, 8
|
|
|
|
};
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
static void acm_tty_set_termios(struct tty_struct *tty,
|
|
|
|
struct ktermios *termios_old)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct acm *acm = tty->driver_data;
|
2006-12-08 11:38:45 +01:00
|
|
|
struct ktermios *termios = tty->termios;
|
2005-04-17 00:20:36 +02:00
|
|
|
struct usb_cdc_line_coding newline;
|
|
|
|
int newctrl = acm->ctrlout;
|
|
|
|
|
|
|
|
if (!ACM_READY(acm))
|
|
|
|
return;
|
|
|
|
|
2009-09-19 22:13:23 +02:00
|
|
|
newline.dwDTERate = cpu_to_le32(tty_get_baud_rate(tty));
|
2005-04-17 00:20:36 +02:00
|
|
|
newline.bCharFormat = termios->c_cflag & CSTOPB ? 2 : 0;
|
|
|
|
newline.bParityType = termios->c_cflag & PARENB ?
|
2009-06-11 13:37:06 +02:00
|
|
|
(termios->c_cflag & PARODD ? 1 : 2) +
|
|
|
|
(termios->c_cflag & CMSPAR ? 2 : 0) : 0;
|
2005-04-17 00:20:36 +02:00
|
|
|
newline.bDataBits = acm_tty_size[(termios->c_cflag & CSIZE) >> 4];
|
2009-06-11 13:37:06 +02:00
|
|
|
/* FIXME: Needs to clear unsupported bits in the termios */
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->clocal = ((termios->c_cflag & CLOCAL) != 0);
|
|
|
|
|
|
|
|
if (!newline.dwDTERate) {
|
|
|
|
newline.dwDTERate = acm->line.dwDTERate;
|
|
|
|
newctrl &= ~ACM_CTRL_DTR;
|
2009-06-11 13:37:06 +02:00
|
|
|
} else
|
|
|
|
newctrl |= ACM_CTRL_DTR;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
if (newctrl != acm->ctrlout)
|
|
|
|
acm_set_control(acm, acm->ctrlout = newctrl);
|
|
|
|
|
|
|
|
if (memcmp(&acm->line, &newline, sizeof newline)) {
|
|
|
|
memcpy(&acm->line, &newline, sizeof newline);
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&acm->control->dev, "%s - set line: %d %d %d %d\n",
|
|
|
|
__func__,
|
|
|
|
le32_to_cpu(newline.dwDTERate),
|
2005-04-17 00:20:36 +02:00
|
|
|
newline.bCharFormat, newline.bParityType,
|
|
|
|
newline.bDataBits);
|
|
|
|
acm_set_line(acm, &acm->line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* USB probe and disconnect routines.
|
|
|
|
*/
|
|
|
|
|
2008-06-25 14:17:16 +02:00
|
|
|
/* Little helpers: write/read buffers free */
|
2005-04-21 21:28:02 +02:00
|
|
|
static void acm_write_buffers_free(struct acm *acm)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct acm_wb *wb;
|
2008-10-21 10:39:04 +02:00
|
|
|
struct usb_device *usb_dev = interface_to_usbdev(acm->control);
|
2005-04-21 21:28:02 +02:00
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++)
|
2010-04-12 13:17:25 +02:00
|
|
|
usb_free_coherent(usb_dev, acm->writesize, wb->buf, wb->dmah);
|
2005-04-21 21:28:02 +02:00
|
|
|
}
|
|
|
|
|
2008-06-25 14:17:16 +02:00
|
|
|
static void acm_read_buffers_free(struct acm *acm)
|
|
|
|
{
|
|
|
|
struct usb_device *usb_dev = interface_to_usbdev(acm->control);
|
|
|
|
int i, n = acm->rx_buflimit;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++)
|
2010-04-12 13:17:25 +02:00
|
|
|
usb_free_coherent(usb_dev, acm->readsize,
|
|
|
|
acm->rb[i].base, acm->rb[i].dma);
|
2008-06-25 14:17:16 +02:00
|
|
|
}
|
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
/* Little helper: write buffers allocate */
|
|
|
|
static int acm_write_buffers_alloc(struct acm *acm)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct acm_wb *wb;
|
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++) {
|
2010-04-12 13:17:25 +02:00
|
|
|
wb->buf = usb_alloc_coherent(acm->dev, acm->writesize, GFP_KERNEL,
|
2005-04-21 21:28:02 +02:00
|
|
|
&wb->dmah);
|
|
|
|
if (!wb->buf) {
|
|
|
|
while (i != 0) {
|
|
|
|
--i;
|
|
|
|
--wb;
|
2010-04-12 13:17:25 +02:00
|
|
|
usb_free_coherent(acm->dev, acm->writesize,
|
2005-04-21 21:28:02 +02:00
|
|
|
wb->buf, wb->dmah);
|
|
|
|
}
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
static int acm_probe(struct usb_interface *intf,
|
|
|
|
const struct usb_device_id *id)
|
2005-04-17 00:20:36 +02:00
|
|
|
{
|
|
|
|
struct usb_cdc_union_desc *union_header = NULL;
|
2007-02-27 15:28:55 +01:00
|
|
|
struct usb_cdc_country_functional_desc *cfd = NULL;
|
2008-04-13 23:00:44 +02:00
|
|
|
unsigned char *buffer = intf->altsetting->extra;
|
2005-04-17 00:20:36 +02:00
|
|
|
int buflen = intf->altsetting->extralen;
|
|
|
|
struct usb_interface *control_interface;
|
|
|
|
struct usb_interface *data_interface;
|
2009-05-16 21:13:19 +02:00
|
|
|
struct usb_endpoint_descriptor *epctrl = NULL;
|
|
|
|
struct usb_endpoint_descriptor *epread = NULL;
|
|
|
|
struct usb_endpoint_descriptor *epwrite = NULL;
|
2005-04-17 00:20:36 +02:00
|
|
|
struct usb_device *usb_dev = interface_to_usbdev(intf);
|
|
|
|
struct acm *acm;
|
|
|
|
int minor;
|
2009-06-11 13:37:06 +02:00
|
|
|
int ctrlsize, readsize;
|
2005-04-17 00:20:36 +02:00
|
|
|
u8 *buf;
|
|
|
|
u8 ac_management_function = 0;
|
|
|
|
u8 call_management_function = 0;
|
|
|
|
int call_interface_num = -1;
|
|
|
|
int data_interface_num;
|
|
|
|
unsigned long quirks;
|
2006-05-13 22:50:47 +02:00
|
|
|
int num_rx_buf;
|
2005-11-01 18:51:34 +01:00
|
|
|
int i;
|
2009-05-16 21:13:19 +02:00
|
|
|
int combined_interfaces = 0;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2006-05-13 22:50:47 +02:00
|
|
|
/* normal quirks */
|
2005-04-17 00:20:36 +02:00
|
|
|
quirks = (unsigned long)id->driver_info;
|
2006-05-13 22:50:47 +02:00
|
|
|
num_rx_buf = (quirks == SINGLE_RX_URB) ? 1 : ACM_NR;
|
|
|
|
|
|
|
|
/* handle quirks deadly to normal probing*/
|
2005-04-17 00:20:36 +02:00
|
|
|
if (quirks == NO_UNION_NORMAL) {
|
|
|
|
data_interface = usb_ifnum_to_if(usb_dev, 1);
|
|
|
|
control_interface = usb_ifnum_to_if(usb_dev, 0);
|
|
|
|
goto skip_normal_probe;
|
|
|
|
}
|
2009-06-11 13:37:06 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/* normal probing*/
|
|
|
|
if (!buffer) {
|
2008-08-14 18:37:34 +02:00
|
|
|
dev_err(&intf->dev, "Weird descriptor references\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!buflen) {
|
2010-09-02 11:46:20 +02:00
|
|
|
if (intf->cur_altsetting->endpoint &&
|
|
|
|
intf->cur_altsetting->endpoint->extralen &&
|
2009-06-11 13:37:06 +02:00
|
|
|
intf->cur_altsetting->endpoint->extra) {
|
|
|
|
dev_dbg(&intf->dev,
|
|
|
|
"Seeking extra descriptors on endpoint\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
buflen = intf->cur_altsetting->endpoint->extralen;
|
|
|
|
buffer = intf->cur_altsetting->endpoint->extra;
|
|
|
|
} else {
|
2008-08-14 18:37:34 +02:00
|
|
|
dev_err(&intf->dev,
|
|
|
|
"Zero length descriptor references\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (buflen > 0) {
|
2009-06-11 13:37:06 +02:00
|
|
|
if (buffer[1] != USB_DT_CS_INTERFACE) {
|
2008-08-14 18:37:34 +02:00
|
|
|
dev_err(&intf->dev, "skipping garbage\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
goto next_desc;
|
|
|
|
}
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
switch (buffer[2]) {
|
|
|
|
case USB_CDC_UNION_TYPE: /* we've found it */
|
|
|
|
if (union_header) {
|
|
|
|
dev_err(&intf->dev, "More than one "
|
|
|
|
"union descriptor, skipping ...\n");
|
|
|
|
goto next_desc;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
2009-06-11 13:37:06 +02:00
|
|
|
union_header = (struct usb_cdc_union_desc *)buffer;
|
|
|
|
break;
|
|
|
|
case USB_CDC_COUNTRY_TYPE: /* export through sysfs*/
|
|
|
|
cfd = (struct usb_cdc_country_functional_desc *)buffer;
|
|
|
|
break;
|
|
|
|
case USB_CDC_HEADER_TYPE: /* maybe check version */
|
|
|
|
break; /* for now we ignore it */
|
|
|
|
case USB_CDC_ACM_TYPE:
|
|
|
|
ac_management_function = buffer[3];
|
|
|
|
break;
|
|
|
|
case USB_CDC_CALL_MANAGEMENT_TYPE:
|
|
|
|
call_management_function = buffer[3];
|
|
|
|
call_interface_num = buffer[4];
|
2010-01-05 13:58:20 +01:00
|
|
|
if ( (quirks & NOT_A_MODEM) == 0 && (call_management_function & 3) != 3)
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_err(&intf->dev, "This device cannot do calls on its own. It is not a modem.\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* there are LOTS more CDC descriptors that
|
|
|
|
* could legitimately be found here.
|
|
|
|
*/
|
|
|
|
dev_dbg(&intf->dev, "Ignoring descriptor: "
|
|
|
|
"type %02x, length %d\n",
|
|
|
|
buffer[2], buffer[0]);
|
|
|
|
break;
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
next_desc:
|
|
|
|
buflen -= buffer[0];
|
|
|
|
buffer += buffer[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!union_header) {
|
|
|
|
if (call_interface_num > 0) {
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev, "No union descriptor, using call management descriptor\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = call_interface_num));
|
|
|
|
control_interface = intf;
|
|
|
|
} else {
|
2009-05-16 21:13:19 +02:00
|
|
|
if (intf->cur_altsetting->desc.bNumEndpoints != 3) {
|
|
|
|
dev_dbg(&intf->dev,"No union descriptor, giving up\n");
|
|
|
|
return -ENODEV;
|
|
|
|
} else {
|
|
|
|
dev_warn(&intf->dev,"No union descriptor, testing for castrated device\n");
|
|
|
|
combined_interfaces = 1;
|
|
|
|
control_interface = data_interface = intf;
|
|
|
|
goto look_for_collapsed_interface;
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
control_interface = usb_ifnum_to_if(usb_dev, union_header->bMasterInterface0);
|
|
|
|
data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = union_header->bSlaveInterface0));
|
|
|
|
if (!control_interface || !data_interface) {
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev, "no interfaces\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
}
|
2009-06-11 13:37:06 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
if (data_interface_num != call_interface_num)
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev, "Separate call control interface. That is not fully supported.\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-05-16 21:13:19 +02:00
|
|
|
if (control_interface == data_interface) {
|
|
|
|
/* some broken devices designed for windows work this way */
|
|
|
|
dev_warn(&intf->dev,"Control and data interfaces are not separated!\n");
|
|
|
|
combined_interfaces = 1;
|
|
|
|
/* a popular other OS doesn't use it */
|
|
|
|
quirks |= NO_CAP_LINE;
|
|
|
|
if (data_interface->cur_altsetting->desc.bNumEndpoints != 3) {
|
|
|
|
dev_err(&intf->dev, "This needs exactly 3 endpoints\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
look_for_collapsed_interface:
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
struct usb_endpoint_descriptor *ep;
|
|
|
|
ep = &data_interface->cur_altsetting->endpoint[i].desc;
|
|
|
|
|
|
|
|
if (usb_endpoint_is_int_in(ep))
|
|
|
|
epctrl = ep;
|
|
|
|
else if (usb_endpoint_is_bulk_out(ep))
|
|
|
|
epwrite = ep;
|
|
|
|
else if (usb_endpoint_is_bulk_in(ep))
|
|
|
|
epread = ep;
|
|
|
|
else
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (!epctrl || !epread || !epwrite)
|
|
|
|
return -ENODEV;
|
|
|
|
else
|
|
|
|
goto made_compressed_probe;
|
|
|
|
}
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
skip_normal_probe:
|
|
|
|
|
|
|
|
/*workaround for switched interfaces */
|
2009-06-11 13:37:06 +02:00
|
|
|
if (data_interface->cur_altsetting->desc.bInterfaceClass
|
|
|
|
!= CDC_DATA_INTERFACE_TYPE) {
|
|
|
|
if (control_interface->cur_altsetting->desc.bInterfaceClass
|
|
|
|
== CDC_DATA_INTERFACE_TYPE) {
|
2005-04-17 00:20:36 +02:00
|
|
|
struct usb_interface *t;
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev,
|
|
|
|
"Your device has switched interfaces.\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
t = control_interface;
|
|
|
|
control_interface = data_interface;
|
|
|
|
data_interface = t;
|
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
2007-08-02 19:29:10 +02:00
|
|
|
|
|
|
|
/* Accept probe requests only for the control interface */
|
2009-05-16 21:13:19 +02:00
|
|
|
if (!combined_interfaces && intf != control_interface)
|
2007-08-02 19:29:10 +02:00
|
|
|
return -ENODEV;
|
2009-06-11 13:37:06 +02:00
|
|
|
|
2009-05-16 21:13:19 +02:00
|
|
|
if (!combined_interfaces && usb_interface_claimed(data_interface)) {
|
|
|
|
/* valid in this context */
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev, "The data interface isn't available\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (data_interface->cur_altsetting->desc.bNumEndpoints < 2)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
epctrl = &control_interface->cur_altsetting->endpoint[0].desc;
|
|
|
|
epread = &data_interface->cur_altsetting->endpoint[0].desc;
|
|
|
|
epwrite = &data_interface->cur_altsetting->endpoint[1].desc;
|
|
|
|
|
|
|
|
|
|
|
|
/* workaround for switched endpoints */
|
2006-10-26 18:02:48 +02:00
|
|
|
if (!usb_endpoint_dir_in(epread)) {
|
2005-04-17 00:20:36 +02:00
|
|
|
/* descriptors are swapped */
|
|
|
|
struct usb_endpoint_descriptor *t;
|
2009-06-11 13:37:06 +02:00
|
|
|
dev_dbg(&intf->dev,
|
|
|
|
"The data interface has switched endpoints\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
t = epread;
|
|
|
|
epread = epwrite;
|
|
|
|
epwrite = t;
|
|
|
|
}
|
2009-05-16 21:13:19 +02:00
|
|
|
made_compressed_probe:
|
2011-03-22 11:12:15 +01:00
|
|
|
dev_dbg(&intf->dev, "interfaces are valid\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
for (minor = 0; minor < ACM_TTY_MINORS && acm_table[minor]; minor++);
|
|
|
|
|
|
|
|
if (minor == ACM_TTY_MINORS) {
|
2008-08-14 18:37:34 +02:00
|
|
|
dev_err(&intf->dev, "no more free acm devices\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
acm = kzalloc(sizeof(struct acm), GFP_KERNEL);
|
|
|
|
if (acm == NULL) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev, "out of memory (acm kzalloc)\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
goto alloc_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctrlsize = le16_to_cpu(epctrl->wMaxPacketSize);
|
2009-06-11 13:37:06 +02:00
|
|
|
readsize = le16_to_cpu(epread->wMaxPacketSize) *
|
|
|
|
(quirks == SINGLE_RX_URB ? 1 : 2);
|
2009-05-16 21:13:19 +02:00
|
|
|
acm->combined_interfaces = combined_interfaces;
|
2008-03-20 10:01:34 +01:00
|
|
|
acm->writesize = le16_to_cpu(epwrite->wMaxPacketSize) * 20;
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->control = control_interface;
|
|
|
|
acm->data = data_interface;
|
|
|
|
acm->minor = minor;
|
|
|
|
acm->dev = usb_dev;
|
|
|
|
acm->ctrl_caps = ac_management_function;
|
2009-05-16 21:13:19 +02:00
|
|
|
if (quirks & NO_CAP_LINE)
|
|
|
|
acm->ctrl_caps &= ~USB_CDC_CAP_LINE;
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->ctrlsize = ctrlsize;
|
|
|
|
acm->readsize = readsize;
|
2006-05-13 22:50:47 +02:00
|
|
|
acm->rx_buflimit = num_rx_buf;
|
2005-11-01 18:51:34 +01:00
|
|
|
acm->urb_task.func = acm_rx_tasklet;
|
|
|
|
acm->urb_task.data = (unsigned long) acm;
|
2006-11-22 15:57:56 +01:00
|
|
|
INIT_WORK(&acm->work, acm_softint);
|
2005-04-17 00:20:36 +02:00
|
|
|
spin_lock_init(&acm->throttle_lock);
|
2005-04-21 21:28:02 +02:00
|
|
|
spin_lock_init(&acm->write_lock);
|
2005-11-01 18:51:34 +01:00
|
|
|
spin_lock_init(&acm->read_lock);
|
2007-10-12 17:24:28 +02:00
|
|
|
mutex_init(&acm->mutex);
|
2005-11-01 18:51:34 +01:00
|
|
|
acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress);
|
2009-08-04 23:52:09 +02:00
|
|
|
acm->is_int_ep = usb_endpoint_xfer_int(epread);
|
|
|
|
if (acm->is_int_ep)
|
|
|
|
acm->bInterval = epread->bInterval;
|
2009-06-11 13:27:50 +02:00
|
|
|
tty_port_init(&acm->port);
|
|
|
|
acm->port.ops = &acm_port_ops;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2010-04-12 13:17:25 +02:00
|
|
|
buf = usb_alloc_coherent(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma);
|
2005-04-17 00:20:36 +02:00
|
|
|
if (!buf) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev, "out of memory (ctrl buffer alloc)\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
goto alloc_fail2;
|
|
|
|
}
|
|
|
|
acm->ctrl_buffer = buf;
|
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
if (acm_write_buffers_alloc(acm) < 0) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev, "out of memory (write buffer alloc)\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
goto alloc_fail4;
|
|
|
|
}
|
|
|
|
|
|
|
|
acm->ctrlurb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
|
if (!acm->ctrlurb) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev, "out of memory (ctrlurb kmalloc)\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
goto alloc_fail5;
|
|
|
|
}
|
2006-05-13 22:50:47 +02:00
|
|
|
for (i = 0; i < num_rx_buf; i++) {
|
2005-11-01 18:51:34 +01:00
|
|
|
struct acm_ru *rcv = &(acm->ru[i]);
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
rcv->urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
|
if (rcv->urb == NULL) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev,
|
2009-06-11 13:37:06 +02:00
|
|
|
"out of memory (read urbs usb_alloc_urb)\n");
|
2010-05-31 02:04:47 +02:00
|
|
|
goto alloc_fail6;
|
2005-11-01 18:51:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
|
rcv->instance = acm;
|
|
|
|
}
|
2006-05-13 22:50:47 +02:00
|
|
|
for (i = 0; i < num_rx_buf; i++) {
|
2008-08-07 03:41:12 +02:00
|
|
|
struct acm_rb *rb = &(acm->rb[i]);
|
2005-11-01 18:51:34 +01:00
|
|
|
|
2010-04-12 13:17:25 +02:00
|
|
|
rb->base = usb_alloc_coherent(acm->dev, readsize,
|
2008-08-07 03:41:12 +02:00
|
|
|
GFP_KERNEL, &rb->dma);
|
|
|
|
if (!rb->base) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev,
|
2010-04-12 13:17:25 +02:00
|
|
|
"out of memory (read bufs usb_alloc_coherent)\n");
|
2005-11-01 18:51:34 +01:00
|
|
|
goto alloc_fail7;
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
2009-06-11 13:37:06 +02:00
|
|
|
for (i = 0; i < ACM_NW; i++) {
|
2008-03-20 10:01:34 +01:00
|
|
|
struct acm_wb *snd = &(acm->wb[i]);
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
snd->urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
|
if (snd->urb == NULL) {
|
2011-03-22 11:12:13 +01:00
|
|
|
dev_err(&intf->dev,
|
2011-03-22 11:12:12 +01:00
|
|
|
"out of memory (write urbs usb_alloc_urb)\n");
|
2010-05-31 02:04:47 +02:00
|
|
|
goto alloc_fail8;
|
2008-03-20 10:01:34 +01:00
|
|
|
}
|
|
|
|
|
2009-07-01 14:27:26 +02:00
|
|
|
if (usb_endpoint_xfer_int(epwrite))
|
|
|
|
usb_fill_int_urb(snd->urb, usb_dev,
|
|
|
|
usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
|
|
|
|
NULL, acm->writesize, acm_write_bulk, snd, epwrite->bInterval);
|
|
|
|
else
|
|
|
|
usb_fill_bulk_urb(snd->urb, usb_dev,
|
|
|
|
usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
|
|
|
|
NULL, acm->writesize, acm_write_bulk, snd);
|
2008-03-20 10:01:34 +01:00
|
|
|
snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
|
snd->instance = acm;
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
usb_set_intfdata(intf, acm);
|
2007-02-27 15:28:55 +01:00
|
|
|
|
|
|
|
i = device_create_file(&intf->dev, &dev_attr_bmCapabilities);
|
|
|
|
if (i < 0)
|
|
|
|
goto alloc_fail8;
|
|
|
|
|
|
|
|
if (cfd) { /* export the country data */
|
|
|
|
acm->country_codes = kmalloc(cfd->bLength - 4, GFP_KERNEL);
|
|
|
|
if (!acm->country_codes)
|
|
|
|
goto skip_countries;
|
|
|
|
acm->country_code_size = cfd->bLength - 4;
|
2009-06-11 13:37:06 +02:00
|
|
|
memcpy(acm->country_codes, (u8 *)&cfd->wCountyCode0,
|
|
|
|
cfd->bLength - 4);
|
2007-02-27 15:28:55 +01:00
|
|
|
acm->country_rel_date = cfd->iCountryCodeRelDate;
|
|
|
|
|
|
|
|
i = device_create_file(&intf->dev, &dev_attr_wCountryCodes);
|
|
|
|
if (i < 0) {
|
|
|
|
kfree(acm->country_codes);
|
|
|
|
goto skip_countries;
|
|
|
|
}
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
i = device_create_file(&intf->dev,
|
|
|
|
&dev_attr_iCountryCodeRelDate);
|
2007-02-27 15:28:55 +01:00
|
|
|
if (i < 0) {
|
2010-05-31 02:04:47 +02:00
|
|
|
device_remove_file(&intf->dev, &dev_attr_wCountryCodes);
|
2007-02-27 15:28:55 +01:00
|
|
|
kfree(acm->country_codes);
|
|
|
|
goto skip_countries;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
skip_countries:
|
2009-06-11 13:37:06 +02:00
|
|
|
usb_fill_int_urb(acm->ctrlurb, usb_dev,
|
2009-05-16 21:13:19 +02:00
|
|
|
usb_rcvintpipe(usb_dev, epctrl->bEndpointAddress),
|
|
|
|
acm->ctrl_buffer, ctrlsize, acm_ctrl_irq, acm,
|
|
|
|
/* works around buggy devices */
|
|
|
|
epctrl->bInterval ? epctrl->bInterval : 0xff);
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->ctrlurb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
|
acm->ctrlurb->transfer_dma = acm->ctrl_dma;
|
|
|
|
|
|
|
|
dev_info(&intf->dev, "ttyACM%d: USB ACM device\n", minor);
|
|
|
|
|
|
|
|
acm_set_control(acm, acm->ctrlout);
|
|
|
|
|
|
|
|
acm->line.dwDTERate = cpu_to_le32(9600);
|
|
|
|
acm->line.bDataBits = 8;
|
|
|
|
acm_set_line(acm, &acm->line);
|
|
|
|
|
|
|
|
usb_driver_claim_interface(&acm_driver, data_interface, acm);
|
2008-08-07 03:41:12 +02:00
|
|
|
usb_set_intfdata(data_interface, acm);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2005-06-30 01:53:29 +02:00
|
|
|
usb_get_intf(control_interface);
|
|
|
|
tty_register_device(acm_tty_driver, minor, &control_interface->dev);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
acm_table[minor] = acm;
|
|
|
|
|
2007-02-27 15:28:55 +01:00
|
|
|
return 0;
|
|
|
|
alloc_fail8:
|
2008-03-20 10:01:34 +01:00
|
|
|
for (i = 0; i < ACM_NW; i++)
|
|
|
|
usb_free_urb(acm->wb[i].urb);
|
2005-04-17 00:20:36 +02:00
|
|
|
alloc_fail7:
|
2008-06-25 14:17:16 +02:00
|
|
|
acm_read_buffers_free(acm);
|
2010-05-31 02:04:47 +02:00
|
|
|
alloc_fail6:
|
2006-05-13 22:50:47 +02:00
|
|
|
for (i = 0; i < num_rx_buf; i++)
|
2005-11-01 18:51:34 +01:00
|
|
|
usb_free_urb(acm->ru[i].urb);
|
2005-04-17 00:20:36 +02:00
|
|
|
usb_free_urb(acm->ctrlurb);
|
|
|
|
alloc_fail5:
|
2005-04-21 21:28:02 +02:00
|
|
|
acm_write_buffers_free(acm);
|
2005-04-17 00:20:36 +02:00
|
|
|
alloc_fail4:
|
2010-04-12 13:17:25 +02:00
|
|
|
usb_free_coherent(usb_dev, ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
|
2005-04-17 00:20:36 +02:00
|
|
|
alloc_fail2:
|
|
|
|
kfree(acm);
|
|
|
|
alloc_fail:
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2007-10-12 17:24:28 +02:00
|
|
|
static void stop_data_traffic(struct acm *acm)
|
|
|
|
{
|
|
|
|
int i;
|
2011-03-22 11:12:15 +01:00
|
|
|
|
|
|
|
dev_dbg(&acm->control->dev, "%s\n", __func__);
|
2007-10-12 17:24:28 +02:00
|
|
|
|
|
|
|
tasklet_disable(&acm->urb_task);
|
|
|
|
|
|
|
|
usb_kill_urb(acm->ctrlurb);
|
2009-06-11 13:37:06 +02:00
|
|
|
for (i = 0; i < ACM_NW; i++)
|
2008-03-20 10:01:34 +01:00
|
|
|
usb_kill_urb(acm->wb[i].urb);
|
2007-10-12 17:24:28 +02:00
|
|
|
for (i = 0; i < acm->rx_buflimit; i++)
|
|
|
|
usb_kill_urb(acm->ru[i].urb);
|
|
|
|
|
|
|
|
tasklet_enable(&acm->urb_task);
|
|
|
|
|
|
|
|
cancel_work_sync(&acm->work);
|
|
|
|
}
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
static void acm_disconnect(struct usb_interface *intf)
|
|
|
|
{
|
2007-02-27 15:28:55 +01:00
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
2005-04-17 00:20:36 +02:00
|
|
|
struct usb_device *usb_dev = interface_to_usbdev(intf);
|
2009-06-11 13:36:09 +02:00
|
|
|
struct tty_struct *tty;
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2008-08-07 03:41:12 +02:00
|
|
|
/* sibling interface is already cleaning up */
|
|
|
|
if (!acm)
|
2006-01-08 12:39:13 +01:00
|
|
|
return;
|
2008-08-07 03:41:12 +02:00
|
|
|
|
|
|
|
mutex_lock(&open_mutex);
|
2009-06-11 13:37:06 +02:00
|
|
|
if (acm->country_codes) {
|
2007-08-02 19:29:10 +02:00
|
|
|
device_remove_file(&acm->control->dev,
|
|
|
|
&dev_attr_wCountryCodes);
|
|
|
|
device_remove_file(&acm->control->dev,
|
|
|
|
&dev_attr_iCountryCodeRelDate);
|
2007-02-27 15:28:55 +01:00
|
|
|
}
|
2007-08-02 19:29:10 +02:00
|
|
|
device_remove_file(&acm->control->dev, &dev_attr_bmCapabilities);
|
2005-04-17 00:20:36 +02:00
|
|
|
acm->dev = NULL;
|
2006-01-08 12:39:13 +01:00
|
|
|
usb_set_intfdata(acm->control, NULL);
|
|
|
|
usb_set_intfdata(acm->data, NULL);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2007-10-12 17:24:28 +02:00
|
|
|
stop_data_traffic(acm);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2005-04-21 21:28:02 +02:00
|
|
|
acm_write_buffers_free(acm);
|
2010-04-12 13:17:25 +02:00
|
|
|
usb_free_coherent(usb_dev, acm->ctrlsize, acm->ctrl_buffer,
|
|
|
|
acm->ctrl_dma);
|
2008-06-25 14:17:16 +02:00
|
|
|
acm_read_buffers_free(acm);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-05-16 21:13:19 +02:00
|
|
|
if (!acm->combined_interfaces)
|
|
|
|
usb_driver_release_interface(&acm_driver, intf == acm->control ?
|
2008-06-25 14:17:16 +02:00
|
|
|
acm->data : acm->control);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
if (acm->port.count == 0) {
|
2005-06-30 01:53:29 +02:00
|
|
|
acm_tty_unregister(acm);
|
2006-01-11 15:55:29 +01:00
|
|
|
mutex_unlock(&open_mutex);
|
2005-04-17 00:20:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-01-11 15:55:29 +01:00
|
|
|
mutex_unlock(&open_mutex);
|
2009-06-11 13:36:09 +02:00
|
|
|
tty = tty_port_tty_get(&acm->port);
|
|
|
|
if (tty) {
|
|
|
|
tty_hangup(tty);
|
|
|
|
tty_kref_put(tty);
|
|
|
|
}
|
2005-04-17 00:20:36 +02:00
|
|
|
}
|
|
|
|
|
2008-07-01 19:10:08 +02:00
|
|
|
#ifdef CONFIG_PM
|
2007-10-12 17:24:28 +02:00
|
|
|
static int acm_suspend(struct usb_interface *intf, pm_message_t message)
|
|
|
|
{
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
2008-06-20 11:25:57 +02:00
|
|
|
int cnt;
|
|
|
|
|
2008-11-25 22:39:18 +01:00
|
|
|
if (message.event & PM_EVENT_AUTO) {
|
2008-06-20 11:25:57 +02:00
|
|
|
int b;
|
|
|
|
|
|
|
|
spin_lock_irq(&acm->read_lock);
|
|
|
|
spin_lock(&acm->write_lock);
|
|
|
|
b = acm->processing + acm->transmitting;
|
|
|
|
spin_unlock(&acm->write_lock);
|
|
|
|
spin_unlock_irq(&acm->read_lock);
|
|
|
|
if (b)
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irq(&acm->read_lock);
|
|
|
|
spin_lock(&acm->write_lock);
|
|
|
|
cnt = acm->susp_count++;
|
|
|
|
spin_unlock(&acm->write_lock);
|
|
|
|
spin_unlock_irq(&acm->read_lock);
|
2007-10-12 17:24:28 +02:00
|
|
|
|
2008-06-20 11:25:57 +02:00
|
|
|
if (cnt)
|
2007-10-12 17:24:28 +02:00
|
|
|
return 0;
|
|
|
|
/*
|
|
|
|
we treat opened interfaces differently,
|
|
|
|
we must guard against open
|
|
|
|
*/
|
|
|
|
mutex_lock(&acm->mutex);
|
|
|
|
|
2009-06-11 13:36:09 +02:00
|
|
|
if (acm->port.count)
|
2007-10-12 17:24:28 +02:00
|
|
|
stop_data_traffic(acm);
|
|
|
|
|
|
|
|
mutex_unlock(&acm->mutex);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int acm_resume(struct usb_interface *intf)
|
|
|
|
{
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
2009-12-16 17:05:57 +01:00
|
|
|
struct acm_wb *wb;
|
2007-10-12 17:24:28 +02:00
|
|
|
int rv = 0;
|
2008-06-20 11:25:57 +02:00
|
|
|
int cnt;
|
2007-10-12 17:24:28 +02:00
|
|
|
|
2008-06-20 11:25:57 +02:00
|
|
|
spin_lock_irq(&acm->read_lock);
|
|
|
|
acm->susp_count -= 1;
|
|
|
|
cnt = acm->susp_count;
|
|
|
|
spin_unlock_irq(&acm->read_lock);
|
|
|
|
|
|
|
|
if (cnt)
|
2007-10-12 17:24:28 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
mutex_lock(&acm->mutex);
|
2009-06-11 13:36:09 +02:00
|
|
|
if (acm->port.count) {
|
2007-10-12 17:24:28 +02:00
|
|
|
rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
|
2009-12-16 17:05:57 +01:00
|
|
|
|
|
|
|
spin_lock_irq(&acm->write_lock);
|
|
|
|
if (acm->delayed_wb) {
|
|
|
|
wb = acm->delayed_wb;
|
|
|
|
acm->delayed_wb = NULL;
|
|
|
|
spin_unlock_irq(&acm->write_lock);
|
2010-03-03 00:37:56 +01:00
|
|
|
acm_start_wb(acm, wb);
|
2009-12-16 17:05:57 +01:00
|
|
|
} else {
|
|
|
|
spin_unlock_irq(&acm->write_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* delayed error checking because we must
|
|
|
|
* do the write path at all cost
|
|
|
|
*/
|
2007-10-12 17:24:28 +02:00
|
|
|
if (rv < 0)
|
2008-06-20 11:25:57 +02:00
|
|
|
goto err_out;
|
2007-10-12 17:24:28 +02:00
|
|
|
|
|
|
|
tasklet_schedule(&acm->urb_task);
|
|
|
|
}
|
|
|
|
|
|
|
|
err_out:
|
|
|
|
mutex_unlock(&acm->mutex);
|
|
|
|
return rv;
|
|
|
|
}
|
2008-07-01 19:10:08 +02:00
|
|
|
|
2009-12-08 09:54:11 +01:00
|
|
|
static int acm_reset_resume(struct usb_interface *intf)
|
|
|
|
{
|
|
|
|
struct acm *acm = usb_get_intfdata(intf);
|
|
|
|
struct tty_struct *tty;
|
|
|
|
|
|
|
|
mutex_lock(&acm->mutex);
|
|
|
|
if (acm->port.count) {
|
|
|
|
tty = tty_port_tty_get(&acm->port);
|
|
|
|
if (tty) {
|
|
|
|
tty_hangup(tty);
|
|
|
|
tty_kref_put(tty);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&acm->mutex);
|
|
|
|
return acm_resume(intf);
|
|
|
|
}
|
|
|
|
|
2008-07-01 19:10:08 +02:00
|
|
|
#endif /* CONFIG_PM */
|
2009-11-19 11:35:33 +01:00
|
|
|
|
|
|
|
#define NOKIA_PCSUITE_ACM_INFO(x) \
|
|
|
|
USB_DEVICE_AND_INTERFACE_INFO(0x0421, x, \
|
|
|
|
USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
|
|
|
|
USB_CDC_ACM_PROTO_VENDOR)
|
|
|
|
|
2010-09-01 17:01:19 +02:00
|
|
|
#define SAMSUNG_PCSUITE_ACM_INFO(x) \
|
|
|
|
USB_DEVICE_AND_INTERFACE_INFO(0x04e7, x, \
|
|
|
|
USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
|
|
|
|
USB_CDC_ACM_PROTO_VENDOR)
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/*
|
|
|
|
* USB driver structure.
|
|
|
|
*/
|
|
|
|
|
2010-01-10 15:33:45 +01:00
|
|
|
static const struct usb_device_id acm_ids[] = {
|
2005-04-17 00:20:36 +02:00
|
|
|
/* quirky and broken devices */
|
|
|
|
{ USB_DEVICE(0x0870, 0x0001), /* Metricom GS Modem */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2007-07-04 17:11:42 +02:00
|
|
|
{ USB_DEVICE(0x0e8d, 0x0003), /* FIREFLY, MediaTek Inc; andrey.arapov@gmail.com */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2008-12-23 17:31:23 +01:00
|
|
|
{ USB_DEVICE(0x0e8d, 0x3329), /* MediaTek Inc GPS */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2005-07-29 21:17:25 +02:00
|
|
|
{ USB_DEVICE(0x0482, 0x0203), /* KYOCERA AH-K3001V */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2006-10-03 11:08:28 +02:00
|
|
|
{ USB_DEVICE(0x079b, 0x000f), /* BT On-Air USB MODEM */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2009-04-06 18:35:01 +02:00
|
|
|
{ USB_DEVICE(0x0ace, 0x1602), /* ZyDAS 56K USB MODEM */
|
|
|
|
.driver_info = SINGLE_RX_URB,
|
|
|
|
},
|
2006-05-13 22:50:47 +02:00
|
|
|
{ USB_DEVICE(0x0ace, 0x1608), /* ZyDAS 56K USB MODEM */
|
|
|
|
.driver_info = SINGLE_RX_URB, /* firmware bug */
|
|
|
|
},
|
2006-06-23 09:14:17 +02:00
|
|
|
{ USB_DEVICE(0x0ace, 0x1611), /* ZyDAS 56K USB MODEM - new version */
|
|
|
|
.driver_info = SINGLE_RX_URB, /* firmware bug */
|
|
|
|
},
|
2007-02-12 08:50:03 +01:00
|
|
|
{ USB_DEVICE(0x22b8, 0x7000), /* Motorola Q Phone */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2008-05-04 01:13:49 +02:00
|
|
|
{ USB_DEVICE(0x0803, 0x3095), /* Zoom Telephonics Model 3095F USB MODEM */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2008-08-14 15:25:40 +02:00
|
|
|
{ USB_DEVICE(0x0572, 0x1321), /* Conexant USB MODEM CX93010 */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2009-01-11 20:53:10 +01:00
|
|
|
{ USB_DEVICE(0x0572, 0x1324), /* Conexant USB MODEM RD02-D400 */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2009-05-07 18:48:23 +02:00
|
|
|
{ USB_DEVICE(0x0572, 0x1328), /* Shiro / Aztech USB MODEM UM-3100 */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* has no union descriptor */
|
|
|
|
},
|
2009-02-25 05:36:51 +01:00
|
|
|
{ USB_DEVICE(0x22b8, 0x6425), /* Motorola MOTOMAGX phones */
|
|
|
|
},
|
2009-02-19 01:17:15 +01:00
|
|
|
{ USB_DEVICE(0x0572, 0x1329), /* Hummingbird huc56s (Conexant) */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* union descriptor misplaced on
|
|
|
|
data interface instead of
|
|
|
|
communications interface.
|
|
|
|
Maybe we should define a new
|
|
|
|
quirk for this. */
|
|
|
|
},
|
2009-05-28 18:33:58 +02:00
|
|
|
{ USB_DEVICE(0x1bbb, 0x0003), /* Alcatel OT-I650 */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
|
|
|
|
},
|
2010-04-22 05:07:03 +02:00
|
|
|
{ USB_DEVICE(0x1576, 0x03b1), /* Maretron USB100 */
|
|
|
|
.driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
|
|
|
|
},
|
2007-02-12 08:50:03 +01:00
|
|
|
|
2009-11-19 11:35:33 +01:00
|
|
|
/* Nokia S60 phones expose two ACM channels. The first is
|
|
|
|
* a modem and is picked up by the standard AT-command
|
|
|
|
* information below. The second is 'vendor-specific' but
|
|
|
|
* is treated as a serial device at the S60 end, so we want
|
|
|
|
* to expose it on Linux too. */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x042D), }, /* Nokia 3250 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04D8), }, /* Nokia 5500 Sport */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04C9), }, /* Nokia E50 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0419), }, /* Nokia E60 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x044D), }, /* Nokia E61 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0001), }, /* Nokia E61i */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0475), }, /* Nokia E62 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0508), }, /* Nokia E65 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0418), }, /* Nokia E70 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0425), }, /* Nokia N71 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0486), }, /* Nokia N73 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04DF), }, /* Nokia N75 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x000e), }, /* Nokia N77 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0445), }, /* Nokia N80 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x042F), }, /* Nokia N91 & N91 8GB */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x048E), }, /* Nokia N92 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0420), }, /* Nokia N93 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04E6), }, /* Nokia N93i */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04B2), }, /* Nokia 5700 XpressMusic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0134), }, /* Nokia 6110 Navigator (China) */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x046E), }, /* Nokia 6110 Navigator */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x002f), }, /* Nokia 6120 classic & */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0088), }, /* Nokia 6121 classic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00fc), }, /* Nokia 6124 classic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0042), }, /* Nokia E51 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00b0), }, /* Nokia E66 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00ab), }, /* Nokia E71 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0481), }, /* Nokia N76 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0007), }, /* Nokia N81 & N81 8GB */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0071), }, /* Nokia N82 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04F0), }, /* Nokia N95 & N95-3 NAM */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0070), }, /* Nokia N95 8GB */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0099), }, /* Nokia 6210 Navigator, RM-367 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0128), }, /* Nokia 6210 Navigator, RM-419 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x008f), }, /* Nokia 6220 Classic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00a0), }, /* Nokia 6650 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x007b), }, /* Nokia N78 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0094), }, /* Nokia N85 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x003a), }, /* Nokia N96 & N96-3 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0108), }, /* Nokia 5320 XpressMusic 2G */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x01f5), }, /* Nokia N97, RM-505 */
|
2010-06-28 22:29:34 +02:00
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x02e3), }, /* Nokia 5230, RM-588 */
|
2010-09-01 17:01:19 +02:00
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0178), }, /* Nokia E63 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x010e), }, /* Nokia E75 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x02d9), }, /* Nokia 6760 Slide */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x01d0), }, /* Nokia E52 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0223), }, /* Nokia E72 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0275), }, /* Nokia X6 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x026c), }, /* Nokia N97 Mini */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0154), }, /* Nokia 5800 XpressMusic */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x04ce), }, /* Nokia E90 */
|
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x01d4), }, /* Nokia E55 */
|
2011-01-25 15:58:40 +01:00
|
|
|
{ NOKIA_PCSUITE_ACM_INFO(0x0302), }, /* Nokia N8 */
|
2010-09-01 17:01:19 +02:00
|
|
|
{ SAMSUNG_PCSUITE_ACM_INFO(0x6651), }, /* Samsung GTi8510 (INNOV8) */
|
2009-11-19 11:35:33 +01:00
|
|
|
|
|
|
|
/* NOTE: non-Nokia COMM/ACM/0xff is likely MSFT RNDIS... NOT a modem! */
|
|
|
|
|
2010-01-05 13:57:46 +01:00
|
|
|
/* Support Lego NXT using pbLua firmware */
|
2010-01-05 13:58:20 +01:00
|
|
|
{ USB_DEVICE(0x0694, 0xff00),
|
|
|
|
.driver_info = NOT_A_MODEM,
|
2010-09-27 04:35:05 +02:00
|
|
|
},
|
2010-01-05 13:57:46 +01:00
|
|
|
|
2010-08-31 19:31:32 +02:00
|
|
|
/* control interfaces without any protocol set */
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_PROTO_NONE) },
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/* control interfaces with various AT-command sets */
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_ACM_PROTO_AT_V25TER) },
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_ACM_PROTO_AT_PCCA101) },
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_ACM_PROTO_AT_PCCA101_WAKE) },
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_ACM_PROTO_AT_GSM) },
|
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
2009-06-11 13:37:06 +02:00
|
|
|
USB_CDC_ACM_PROTO_AT_3G) },
|
2005-04-17 00:20:36 +02:00
|
|
|
{ USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
|
|
|
|
USB_CDC_ACM_PROTO_AT_CDMA) },
|
|
|
|
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
MODULE_DEVICE_TABLE(usb, acm_ids);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
static struct usb_driver acm_driver = {
|
|
|
|
.name = "cdc_acm",
|
|
|
|
.probe = acm_probe,
|
|
|
|
.disconnect = acm_disconnect,
|
2008-07-01 19:10:08 +02:00
|
|
|
#ifdef CONFIG_PM
|
2007-10-12 17:24:28 +02:00
|
|
|
.suspend = acm_suspend,
|
|
|
|
.resume = acm_resume,
|
2009-12-08 09:54:11 +01:00
|
|
|
.reset_resume = acm_reset_resume,
|
2008-07-01 19:10:08 +02:00
|
|
|
#endif
|
2005-04-17 00:20:36 +02:00
|
|
|
.id_table = acm_ids,
|
2008-07-01 19:10:08 +02:00
|
|
|
#ifdef CONFIG_PM
|
2007-10-12 17:24:28 +02:00
|
|
|
.supports_autosuspend = 1,
|
2008-07-01 19:10:08 +02:00
|
|
|
#endif
|
2005-04-17 00:20:36 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TTY driver structures.
|
|
|
|
*/
|
|
|
|
|
2006-10-02 11:17:18 +02:00
|
|
|
static const struct tty_operations acm_ops = {
|
2005-04-17 00:20:36 +02:00
|
|
|
.open = acm_tty_open,
|
|
|
|
.close = acm_tty_close,
|
2009-06-11 13:36:09 +02:00
|
|
|
.hangup = acm_tty_hangup,
|
2005-04-17 00:20:36 +02:00
|
|
|
.write = acm_tty_write,
|
|
|
|
.write_room = acm_tty_write_room,
|
|
|
|
.ioctl = acm_tty_ioctl,
|
|
|
|
.throttle = acm_tty_throttle,
|
|
|
|
.unthrottle = acm_tty_unthrottle,
|
|
|
|
.chars_in_buffer = acm_tty_chars_in_buffer,
|
|
|
|
.break_ctl = acm_tty_break_ctl,
|
|
|
|
.set_termios = acm_tty_set_termios,
|
|
|
|
.tiocmget = acm_tty_tiocmget,
|
|
|
|
.tiocmset = acm_tty_tiocmset,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Init / exit.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int __init acm_init(void)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
acm_tty_driver = alloc_tty_driver(ACM_TTY_MINORS);
|
|
|
|
if (!acm_tty_driver)
|
|
|
|
return -ENOMEM;
|
|
|
|
acm_tty_driver->owner = THIS_MODULE,
|
|
|
|
acm_tty_driver->driver_name = "acm",
|
|
|
|
acm_tty_driver->name = "ttyACM",
|
|
|
|
acm_tty_driver->major = ACM_TTY_MAJOR,
|
|
|
|
acm_tty_driver->minor_start = 0,
|
|
|
|
acm_tty_driver->type = TTY_DRIVER_TYPE_SERIAL,
|
|
|
|
acm_tty_driver->subtype = SERIAL_TYPE_NORMAL,
|
2005-06-21 06:15:16 +02:00
|
|
|
acm_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
|
2005-04-17 00:20:36 +02:00
|
|
|
acm_tty_driver->init_termios = tty_std_termios;
|
2009-06-11 13:37:06 +02:00
|
|
|
acm_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD |
|
|
|
|
HUPCL | CLOCAL;
|
2005-04-17 00:20:36 +02:00
|
|
|
tty_set_operations(acm_tty_driver, &acm_ops);
|
|
|
|
|
|
|
|
retval = tty_register_driver(acm_tty_driver);
|
|
|
|
if (retval) {
|
|
|
|
put_tty_driver(acm_tty_driver);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
retval = usb_register(&acm_driver);
|
|
|
|
if (retval) {
|
|
|
|
tty_unregister_driver(acm_tty_driver);
|
|
|
|
put_tty_driver(acm_tty_driver);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2011-03-22 11:12:18 +01:00
|
|
|
printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_DESC "\n");
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit acm_exit(void)
|
|
|
|
{
|
|
|
|
usb_deregister(&acm_driver);
|
|
|
|
tty_unregister_driver(acm_tty_driver);
|
|
|
|
put_tty_driver(acm_tty_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(acm_init);
|
|
|
|
module_exit(acm_exit);
|
|
|
|
|
2009-06-11 13:37:06 +02:00
|
|
|
MODULE_AUTHOR(DRIVER_AUTHOR);
|
|
|
|
MODULE_DESCRIPTION(DRIVER_DESC);
|
2005-04-17 00:20:36 +02:00
|
|
|
MODULE_LICENSE("GPL");
|
2009-04-06 18:33:18 +02:00
|
|
|
MODULE_ALIAS_CHARDEV_MAJOR(ACM_TTY_MAJOR);
|