blob: a693381ed3ccc3a840d07eaa64ce78a08abb625b [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * cdc-acm.c
3 *
4 * Copyright (c) 1999 Armin Fuerst <fuerst@in.tum.de>
Pavel Macheka2531292010-07-18 14:27:13 +02005 * Copyright (c) 1999 Pavel Machek <pavel@ucw.cz>
Linus Torvalds1da177e2005-04-16 15:20:36 -07006 * Copyright (c) 1999 Johannes Erdfelt <johannes@erdfelt.com>
7 * Copyright (c) 2000 Vojtech Pavlik <vojtech@suse.cz>
8 * Copyright (c) 2004 Oliver Neukum <oliver@neukum.name>
David Kubicek61a87ad2005-11-01 18:51:34 +01009 * Copyright (c) 2005 David Kubicek <dave@awk.cz>
Linus Torvalds1da177e2005-04-16 15:20:36 -070010 *
11 * USB Abstract Control Model driver for USB modems and ISDN adapters
12 *
13 * Sponsored by SuSE
14 *
Linus Torvalds1da177e2005-04-16 15:20:36 -070015 * This program is free software; you can redistribute it and/or modify
16 * it under the terms of the GNU General Public License as published by
17 * the Free Software Foundation; either version 2 of the License, or
18 * (at your option) any later version.
19 *
20 * This program is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
24 *
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, write to the Free Software
27 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
28 */
29
30#undef DEBUG
David Brownelle5fbab52008-08-06 18:46:10 -070031#undef VERBOSE_DEBUG
Linus Torvalds1da177e2005-04-16 15:20:36 -070032
33#include <linux/kernel.h>
34#include <linux/errno.h>
35#include <linux/init.h>
36#include <linux/slab.h>
37#include <linux/tty.h>
Oliver Neukum7af25b42009-09-08 23:51:28 +020038#include <linux/serial.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070039#include <linux/tty_driver.h>
40#include <linux/tty_flip.h>
41#include <linux/module.h>
Arjan van de Ven4186ecf2006-01-11 15:55:29 +010042#include <linux/mutex.h>
Alan Cox10077d42009-06-11 12:36:09 +010043#include <linux/uaccess.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070044#include <linux/usb.h>
David Brownella8c28f22006-06-13 09:57:47 -070045#include <linux/usb/cdc.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070046#include <asm/byteorder.h>
47#include <asm/unaligned.h>
David Kubicek61a87ad2005-11-01 18:51:34 +010048#include <linux/list.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070049
50#include "cdc-acm.h"
51
David Brownelle5fbab52008-08-06 18:46:10 -070052
53#define ACM_CLOSE_TIMEOUT 15 /* seconds to let writes drain */
54
Johan Hovolda2c7b932011-03-22 11:12:18 +010055
David Kubicek61a87ad2005-11-01 18:51:34 +010056#define DRIVER_AUTHOR "Armin Fuerst, Pavel Machek, Johannes Erdfelt, Vojtech Pavlik, David Kubicek"
Linus Torvalds1da177e2005-04-16 15:20:36 -070057#define DRIVER_DESC "USB Abstract Control Model driver for USB modems and ISDN adapters"
58
59static struct usb_driver acm_driver;
60static struct tty_driver *acm_tty_driver;
61static struct acm *acm_table[ACM_TTY_MINORS];
62
Arjan van de Ven4186ecf2006-01-11 15:55:29 +010063static DEFINE_MUTEX(open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -070064
Alan Cox10077d42009-06-11 12:36:09 +010065#define ACM_READY(acm) (acm && acm->dev && acm->port.count)
Linus Torvalds1da177e2005-04-16 15:20:36 -070066
Alan Cox739e0282009-06-11 12:27:50 +010067static const struct tty_port_operations acm_port_ops = {
68};
69
Linus Torvalds1da177e2005-04-16 15:20:36 -070070/*
71 * Functions for ACM control messages.
72 */
73
Alan Cox6e47e062009-06-11 12:37:06 +010074static int acm_ctrl_msg(struct acm *acm, int request, int value,
75 void *buf, int len)
Linus Torvalds1da177e2005-04-16 15:20:36 -070076{
77 int retval = usb_control_msg(acm->dev, usb_sndctrlpipe(acm->dev, 0),
78 request, USB_RT_ACM, value,
79 acm->control->altsetting[0].desc.bInterfaceNumber,
80 buf, len, 5000);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +010081 dev_dbg(&acm->control->dev,
82 "%s - rq 0x%02x, val %#x, len %#x, result %d\n",
83 __func__, request, value, len, retval);
Linus Torvalds1da177e2005-04-16 15:20:36 -070084 return retval < 0 ? retval : 0;
85}
86
87/* devices aren't required to support these requests.
88 * the cdc acm descriptor tells whether they do...
89 */
90#define acm_set_control(acm, control) \
91 acm_ctrl_msg(acm, USB_CDC_REQ_SET_CONTROL_LINE_STATE, control, NULL, 0)
92#define acm_set_line(acm, line) \
93 acm_ctrl_msg(acm, USB_CDC_REQ_SET_LINE_CODING, 0, line, sizeof *(line))
94#define acm_send_break(acm, ms) \
95 acm_ctrl_msg(acm, USB_CDC_REQ_SEND_BREAK, ms, NULL, 0)
96
97/*
Oliver Neukum884b6002005-04-21 21:28:02 +020098 * Write buffer management.
99 * All of these assume proper locks taken by the caller.
100 */
101
102static int acm_wb_alloc(struct acm *acm)
103{
104 int i, wbn;
105 struct acm_wb *wb;
106
David Engrafe4cf3aa2008-03-20 10:01:34 +0100107 wbn = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200108 i = 0;
109 for (;;) {
110 wb = &acm->wb[wbn];
111 if (!wb->use) {
112 wb->use = 1;
113 return wbn;
114 }
Oliver Neukum86478942006-05-13 22:50:47 +0200115 wbn = (wbn + 1) % ACM_NW;
116 if (++i >= ACM_NW)
Oliver Neukum884b6002005-04-21 21:28:02 +0200117 return -1;
118 }
119}
120
Oliver Neukum884b6002005-04-21 21:28:02 +0200121static int acm_wb_is_avail(struct acm *acm)
122{
123 int i, n;
David Brownelle5fbab52008-08-06 18:46:10 -0700124 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200125
Oliver Neukum86478942006-05-13 22:50:47 +0200126 n = ACM_NW;
David Brownelle5fbab52008-08-06 18:46:10 -0700127 spin_lock_irqsave(&acm->write_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100128 for (i = 0; i < ACM_NW; i++)
Oliver Neukum86478942006-05-13 22:50:47 +0200129 n -= acm->wb[i].use;
David Brownelle5fbab52008-08-06 18:46:10 -0700130 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200131 return n;
132}
133
Oliver Neukum884b6002005-04-21 21:28:02 +0200134/*
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800135 * Finish write. Caller must hold acm->write_lock
Oliver Neukum884b6002005-04-21 21:28:02 +0200136 */
David Engrafe4cf3aa2008-03-20 10:01:34 +0100137static void acm_write_done(struct acm *acm, struct acm_wb *wb)
Oliver Neukum884b6002005-04-21 21:28:02 +0200138{
David Engrafe4cf3aa2008-03-20 10:01:34 +0100139 wb->use = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200140 acm->transmitting--;
Oliver Neukum97d35f92009-12-16 17:05:57 +0100141 usb_autopm_put_interface_async(acm->control);
Oliver Neukum884b6002005-04-21 21:28:02 +0200142}
143
144/*
145 * Poke write.
Oliver Neukum11ea8592008-06-20 11:25:57 +0200146 *
147 * the caller is responsible for locking
Oliver Neukum884b6002005-04-21 21:28:02 +0200148 */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200149
150static int acm_start_wb(struct acm *acm, struct acm_wb *wb)
151{
152 int rc;
153
154 acm->transmitting++;
155
156 wb->urb->transfer_buffer = wb->buf;
157 wb->urb->transfer_dma = wb->dmah;
158 wb->urb->transfer_buffer_length = wb->len;
159 wb->urb->dev = acm->dev;
160
Alan Cox6e47e062009-06-11 12:37:06 +0100161 rc = usb_submit_urb(wb->urb, GFP_ATOMIC);
162 if (rc < 0) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100163 dev_err(&acm->data->dev,
164 "%s - usb_submit_urb(write bulk) failed: %d\n",
165 __func__, rc);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200166 acm_write_done(acm, wb);
167 }
168 return rc;
169}
170
David Engrafe4cf3aa2008-03-20 10:01:34 +0100171static int acm_write_start(struct acm *acm, int wbn)
Oliver Neukum884b6002005-04-21 21:28:02 +0200172{
173 unsigned long flags;
David Brownell934da462008-08-06 18:44:12 -0700174 struct acm_wb *wb = &acm->wb[wbn];
Oliver Neukum884b6002005-04-21 21:28:02 +0200175 int rc;
176
177 spin_lock_irqsave(&acm->write_lock, flags);
178 if (!acm->dev) {
David Brownell934da462008-08-06 18:44:12 -0700179 wb->use = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200180 spin_unlock_irqrestore(&acm->write_lock, flags);
181 return -ENODEV;
182 }
183
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100184 dev_vdbg(&acm->data->dev, "%s - susp_count %d\n", __func__,
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100185 acm->susp_count);
Oliver Neukum97d35f92009-12-16 17:05:57 +0100186 usb_autopm_get_interface_async(acm->control);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200187 if (acm->susp_count) {
Oliver Neukum97d35f92009-12-16 17:05:57 +0100188 if (!acm->delayed_wb)
189 acm->delayed_wb = wb;
190 else
191 usb_autopm_put_interface_async(acm->control);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200192 spin_unlock_irqrestore(&acm->write_lock, flags);
193 return 0; /* A white lie */
194 }
195 usb_mark_last_busy(acm->dev);
196
Oliver Neukum11ea8592008-06-20 11:25:57 +0200197 rc = acm_start_wb(acm, wb);
Oliver Neukum884b6002005-04-21 21:28:02 +0200198 spin_unlock_irqrestore(&acm->write_lock, flags);
199
Oliver Neukum884b6002005-04-21 21:28:02 +0200200 return rc;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200201
Oliver Neukum884b6002005-04-21 21:28:02 +0200202}
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100203/*
204 * attributes exported through sysfs
205 */
206static ssize_t show_caps
207(struct device *dev, struct device_attribute *attr, char *buf)
208{
209 struct usb_interface *intf = to_usb_interface(dev);
210 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum884b6002005-04-21 21:28:02 +0200211
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100212 return sprintf(buf, "%d", acm->ctrl_caps);
213}
214static DEVICE_ATTR(bmCapabilities, S_IRUGO, show_caps, NULL);
215
216static ssize_t show_country_codes
217(struct device *dev, struct device_attribute *attr, char *buf)
218{
219 struct usb_interface *intf = to_usb_interface(dev);
220 struct acm *acm = usb_get_intfdata(intf);
221
222 memcpy(buf, acm->country_codes, acm->country_code_size);
223 return acm->country_code_size;
224}
225
226static DEVICE_ATTR(wCountryCodes, S_IRUGO, show_country_codes, NULL);
227
228static ssize_t show_country_rel_date
229(struct device *dev, struct device_attribute *attr, char *buf)
230{
231 struct usb_interface *intf = to_usb_interface(dev);
232 struct acm *acm = usb_get_intfdata(intf);
233
234 return sprintf(buf, "%d", acm->country_rel_date);
235}
236
237static DEVICE_ATTR(iCountryCodeRelDate, S_IRUGO, show_country_rel_date, NULL);
Oliver Neukum884b6002005-04-21 21:28:02 +0200238/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700239 * Interrupt handlers for various ACM device responses
240 */
241
242/* control interface reports status changes with "interrupt" transfers */
David Howells7d12e782006-10-05 14:55:46 +0100243static void acm_ctrl_irq(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700244{
245 struct acm *acm = urb->context;
246 struct usb_cdc_notification *dr = urb->transfer_buffer;
Alan Cox10077d42009-06-11 12:36:09 +0100247 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700248 unsigned char *data;
249 int newctrl;
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700250 int retval;
251 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700252
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700253 switch (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254 case 0:
255 /* success */
256 break;
257 case -ECONNRESET:
258 case -ENOENT:
259 case -ESHUTDOWN:
260 /* this urb is terminated, clean up */
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100261 dev_dbg(&acm->control->dev,
262 "%s - urb shutting down with status: %d\n",
263 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700264 return;
265 default:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100266 dev_dbg(&acm->control->dev,
267 "%s - nonzero urb status received: %d\n",
268 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700269 goto exit;
270 }
271
272 if (!ACM_READY(acm))
273 goto exit;
274
Johan Hovold7e7797e2011-03-22 11:12:11 +0100275 usb_mark_last_busy(acm->dev);
276
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277 data = (unsigned char *)(dr + 1);
278 switch (dr->bNotificationType) {
Alan Cox6e47e062009-06-11 12:37:06 +0100279 case USB_CDC_NOTIFY_NETWORK_CONNECTION:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100280 dev_dbg(&acm->control->dev, "%s - network connection: %d\n",
281 __func__, dr->wValue);
Alan Cox6e47e062009-06-11 12:37:06 +0100282 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283
Alan Cox6e47e062009-06-11 12:37:06 +0100284 case USB_CDC_NOTIFY_SERIAL_STATE:
285 tty = tty_port_tty_get(&acm->port);
286 newctrl = get_unaligned_le16(data);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700287
Alan Cox6e47e062009-06-11 12:37:06 +0100288 if (tty) {
289 if (!acm->clocal &&
290 (acm->ctrlin & ~newctrl & ACM_CTRL_DCD)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100291 dev_dbg(&acm->control->dev,
292 "%s - calling hangup\n", __func__);
Alan Cox6e47e062009-06-11 12:37:06 +0100293 tty_hangup(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700294 }
Alan Cox6e47e062009-06-11 12:37:06 +0100295 tty_kref_put(tty);
296 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700297
Alan Cox6e47e062009-06-11 12:37:06 +0100298 acm->ctrlin = newctrl;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100300 dev_dbg(&acm->control->dev,
301 "%s - input control lines: dcd%c dsr%c break%c "
302 "ring%c framing%c parity%c overrun%c\n",
303 __func__,
Alan Cox6e47e062009-06-11 12:37:06 +0100304 acm->ctrlin & ACM_CTRL_DCD ? '+' : '-',
305 acm->ctrlin & ACM_CTRL_DSR ? '+' : '-',
306 acm->ctrlin & ACM_CTRL_BRK ? '+' : '-',
307 acm->ctrlin & ACM_CTRL_RI ? '+' : '-',
308 acm->ctrlin & ACM_CTRL_FRAMING ? '+' : '-',
309 acm->ctrlin & ACM_CTRL_PARITY ? '+' : '-',
310 acm->ctrlin & ACM_CTRL_OVERRUN ? '+' : '-');
Linus Torvalds1da177e2005-04-16 15:20:36 -0700311 break;
312
Alan Cox6e47e062009-06-11 12:37:06 +0100313 default:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100314 dev_dbg(&acm->control->dev,
315 "%s - unknown notification %d received: index %d "
316 "len %d data0 %d data1 %d\n",
317 __func__,
Alan Cox6e47e062009-06-11 12:37:06 +0100318 dr->bNotificationType, dr->wIndex,
319 dr->wLength, data[0], data[1]);
320 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700321 }
322exit:
Alan Cox6e47e062009-06-11 12:37:06 +0100323 retval = usb_submit_urb(urb, GFP_ATOMIC);
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700324 if (retval)
Johan Hovold1d9846e2011-03-22 11:12:14 +0100325 dev_err(&acm->control->dev, "%s - usb_submit_urb failed: %d\n",
326 __func__, retval);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700327}
328
329/* data interface returns incoming bytes, or we got unthrottled */
David Howells7d12e782006-10-05 14:55:46 +0100330static void acm_read_bulk(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331{
David Kubicek61a87ad2005-11-01 18:51:34 +0100332 struct acm_rb *buf;
333 struct acm_ru *rcv = urb->context;
334 struct acm *acm = rcv->instance;
Oliver Neukum86478942006-05-13 22:50:47 +0200335 int status = urb->status;
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700336
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100337 dev_vdbg(&acm->data->dev, "%s - status %d\n", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338
Oliver Neukum11ea8592008-06-20 11:25:57 +0200339 if (!ACM_READY(acm)) {
Johan Hovold1d9846e2011-03-22 11:12:14 +0100340 dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200342 }
343 usb_mark_last_busy(acm->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700344
Oliver Neukum86478942006-05-13 22:50:47 +0200345 if (status)
Johan Hovold1d9846e2011-03-22 11:12:14 +0100346 dev_dbg(&acm->data->dev, "%s - non-zero urb status: %d\n",
347 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700348
David Kubicek61a87ad2005-11-01 18:51:34 +0100349 buf = rcv->buffer;
350 buf->size = urb->actual_length;
351
Oliver Neukum86478942006-05-13 22:50:47 +0200352 if (likely(status == 0)) {
353 spin_lock(&acm->read_lock);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200354 acm->processing++;
Oliver Neukum86478942006-05-13 22:50:47 +0200355 list_add_tail(&rcv->list, &acm->spare_read_urbs);
356 list_add_tail(&buf->list, &acm->filled_read_bufs);
357 spin_unlock(&acm->read_lock);
358 } else {
359 /* we drop the buffer due to an error */
360 spin_lock(&acm->read_lock);
361 list_add_tail(&rcv->list, &acm->spare_read_urbs);
362 list_add(&buf->list, &acm->spare_read_bufs);
363 spin_unlock(&acm->read_lock);
364 /* nevertheless the tasklet must be kicked unconditionally
365 so the queue cannot dry up */
366 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200367 if (likely(!acm->susp_count))
368 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700369}
370
371static void acm_rx_tasklet(unsigned long _acm)
372{
373 struct acm *acm = (void *)_acm;
David Kubicek61a87ad2005-11-01 18:51:34 +0100374 struct acm_rb *buf;
Alan Cox10077d42009-06-11 12:36:09 +0100375 struct tty_struct *tty;
David Kubicek61a87ad2005-11-01 18:51:34 +0100376 struct acm_ru *rcv;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200377 unsigned long flags;
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100378 unsigned char throttled;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200379
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100380 dev_vdbg(&acm->data->dev, "%s\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700381
Alan Cox10077d42009-06-11 12:36:09 +0100382 if (!ACM_READY(acm)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100383 dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100384 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200385 }
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100386
Oliver Neukum834dbca2007-03-06 10:47:04 +0100387 spin_lock_irqsave(&acm->throttle_lock, flags);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100388 throttled = acm->throttle;
Oliver Neukum834dbca2007-03-06 10:47:04 +0100389 spin_unlock_irqrestore(&acm->throttle_lock, flags);
Alan Cox10077d42009-06-11 12:36:09 +0100390 if (throttled) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100391 dev_dbg(&acm->data->dev, "%s - throttled\n", __func__);
David Kubicek61a87ad2005-11-01 18:51:34 +0100392 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200393 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100394
Alan Cox10077d42009-06-11 12:36:09 +0100395 tty = tty_port_tty_get(&acm->port);
396
David Kubicek61a87ad2005-11-01 18:51:34 +0100397next_buffer:
Jarek Poplawski762f0072006-10-06 07:23:11 +0200398 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100399 if (list_empty(&acm->filled_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200400 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100401 goto urbs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100403 buf = list_entry(acm->filled_read_bufs.next,
404 struct acm_rb, list);
405 list_del(&buf->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200406 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100407
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100408 dev_vdbg(&acm->data->dev, "%s - processing buf 0x%p, size = %d\n",
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100409 __func__, buf, buf->size);
Alan Cox10077d42009-06-11 12:36:09 +0100410 if (tty) {
411 spin_lock_irqsave(&acm->throttle_lock, flags);
412 throttled = acm->throttle;
413 spin_unlock_irqrestore(&acm->throttle_lock, flags);
414 if (!throttled) {
Alan Cox10077d42009-06-11 12:36:09 +0100415 tty_insert_flip_string(tty, buf->base, buf->size);
416 tty_flip_buffer_push(tty);
417 } else {
418 tty_kref_put(tty);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100419 dev_dbg(&acm->data->dev, "%s - throttling noticed\n",
420 __func__);
Alan Cox10077d42009-06-11 12:36:09 +0100421 spin_lock_irqsave(&acm->read_lock, flags);
422 list_add(&buf->list, &acm->filled_read_bufs);
423 spin_unlock_irqrestore(&acm->read_lock, flags);
424 return;
425 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700427
Jarek Poplawski762f0072006-10-06 07:23:11 +0200428 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100429 list_add(&buf->list, &acm->spare_read_bufs);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200430 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100431 goto next_buffer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432
David Kubicek61a87ad2005-11-01 18:51:34 +0100433urbs:
Alan Cox10077d42009-06-11 12:36:09 +0100434 tty_kref_put(tty);
435
David Kubicek61a87ad2005-11-01 18:51:34 +0100436 while (!list_empty(&acm->spare_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200437 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100438 if (list_empty(&acm->spare_read_urbs)) {
Oliver Neukum11ea8592008-06-20 11:25:57 +0200439 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200440 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100441 return;
442 }
443 rcv = list_entry(acm->spare_read_urbs.next,
444 struct acm_ru, list);
445 list_del(&rcv->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200446 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100447
448 buf = list_entry(acm->spare_read_bufs.next,
449 struct acm_rb, list);
450 list_del(&buf->list);
451
452 rcv->buffer = buf;
453
Oliver Neukumcf7fdd52009-08-04 23:52:09 +0200454 if (acm->is_int_ep)
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +0400455 usb_fill_int_urb(rcv->urb, acm->dev,
456 acm->rx_endpoint,
457 buf->base,
458 acm->readsize,
Oliver Neukumcf7fdd52009-08-04 23:52:09 +0200459 acm_read_bulk, rcv, acm->bInterval);
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +0400460 else
461 usb_fill_bulk_urb(rcv->urb, acm->dev,
462 acm->rx_endpoint,
463 buf->base,
464 acm->readsize,
465 acm_read_bulk, rcv);
David Kubicek61a87ad2005-11-01 18:51:34 +0100466 rcv->urb->transfer_dma = buf->dma;
467 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
468
Alan Cox6e47e062009-06-11 12:37:06 +0100469 /* This shouldn't kill the driver as unsuccessful URBs are
470 returned to the free-urbs-pool and resubmited ASAP */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200471 spin_lock_irqsave(&acm->read_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100472 if (acm->susp_count ||
473 usb_submit_urb(rcv->urb, GFP_ATOMIC) < 0) {
David Kubicek61a87ad2005-11-01 18:51:34 +0100474 list_add(&buf->list, &acm->spare_read_bufs);
David Kubicek61a87ad2005-11-01 18:51:34 +0100475 list_add(&rcv->list, &acm->spare_read_urbs);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200476 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200477 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100478 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200479 } else {
480 spin_unlock_irqrestore(&acm->read_lock, flags);
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100481 dev_vdbg(&acm->data->dev,
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100482 "%s - sending urb 0x%p, rcv 0x%p, buf 0x%p\n",
483 __func__, rcv->urb, rcv, buf);
David Kubicek61a87ad2005-11-01 18:51:34 +0100484 }
485 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200486 spin_lock_irqsave(&acm->read_lock, flags);
487 acm->processing = 0;
488 spin_unlock_irqrestore(&acm->read_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489}
490
491/* data interface wrote those outgoing bytes */
David Howells7d12e782006-10-05 14:55:46 +0100492static void acm_write_bulk(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700493{
Ming Leicdc97792008-02-24 18:41:47 +0800494 struct acm_wb *wb = urb->context;
David Brownelle5fbab52008-08-06 18:46:10 -0700495 struct acm *acm = wb->instance;
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800496 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200497
Johan Hovold4fa46262011-03-22 11:12:16 +0100498 if (urb->status || (urb->actual_length != urb->transfer_buffer_length))
499 dev_vdbg(&acm->data->dev, "%s - len %d/%d, status %d\n",
Johan Hovold1d9846e2011-03-22 11:12:14 +0100500 __func__,
David Brownelle5fbab52008-08-06 18:46:10 -0700501 urb->actual_length,
502 urb->transfer_buffer_length,
503 urb->status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800505 spin_lock_irqsave(&acm->write_lock, flags);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100506 acm_write_done(acm, wb);
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800507 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200508 if (ACM_READY(acm))
509 schedule_work(&acm->work);
David Brownelle5fbab52008-08-06 18:46:10 -0700510 else
511 wake_up_interruptible(&acm->drain_wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512}
513
David Howellsc4028952006-11-22 14:57:56 +0000514static void acm_softint(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700515{
David Howellsc4028952006-11-22 14:57:56 +0000516 struct acm *acm = container_of(work, struct acm, work);
Alan Cox10077d42009-06-11 12:36:09 +0100517 struct tty_struct *tty;
David Brownelle5fbab52008-08-06 18:46:10 -0700518
Johan Hovold1d9846e2011-03-22 11:12:14 +0100519 dev_vdbg(&acm->data->dev, "%s\n", __func__);
520
Linus Torvalds1da177e2005-04-16 15:20:36 -0700521 if (!ACM_READY(acm))
522 return;
Alan Cox10077d42009-06-11 12:36:09 +0100523 tty = tty_port_tty_get(&acm->port);
Johan Hovold15e5bee2011-03-22 11:12:10 +0100524 if (!tty)
525 return;
Alan Cox10077d42009-06-11 12:36:09 +0100526 tty_wakeup(tty);
527 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700528}
529
530/*
531 * TTY handlers
532 */
533
534static int acm_tty_open(struct tty_struct *tty, struct file *filp)
535{
536 struct acm *acm;
Thadeu Lima de Souza Cascardo42dd2aa2009-06-25 14:41:24 +0100537 int rv = -ENODEV;
David Kubicek61a87ad2005-11-01 18:51:34 +0100538 int i;
Arjan van de Ven4186ecf2006-01-11 15:55:29 +0100539
540 mutex_lock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541
542 acm = acm_table[tty->index];
543 if (!acm || !acm->dev)
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100544 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545 else
546 rv = 0;
547
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100548 dev_dbg(&acm->control->dev, "%s\n", __func__);
549
David Engraf28d1dfa2008-03-20 10:53:52 +0100550 set_bit(TTY_NO_WRITE_SPLIT, &tty->flags);
Alan Cox10077d42009-06-11 12:36:09 +0100551
Linus Torvalds1da177e2005-04-16 15:20:36 -0700552 tty->driver_data = acm;
Alan Cox10077d42009-06-11 12:36:09 +0100553 tty_port_tty_set(&acm->port, tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700554
Oliver Neukum94409cc2008-02-11 15:22:29 +0100555 if (usb_autopm_get_interface(acm->control) < 0)
556 goto early_bail;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200557 else
558 acm->control->needs_remote_wakeup = 1;
Oliver Neukum1365baf2007-10-12 17:24:28 +0200559
560 mutex_lock(&acm->mutex);
Alan Cox10077d42009-06-11 12:36:09 +0100561 if (acm->port.count++) {
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100562 mutex_unlock(&acm->mutex);
Oliver Neukum1365baf2007-10-12 17:24:28 +0200563 usb_autopm_put_interface(acm->control);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100564 goto out;
Alan Cox10077d42009-06-11 12:36:09 +0100565 }
Oliver Neukum1365baf2007-10-12 17:24:28 +0200566
Linus Torvalds1da177e2005-04-16 15:20:36 -0700567 acm->ctrlurb->dev = acm->dev;
568 if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100569 dev_err(&acm->control->dev,
570 "%s - usb_submit_urb(ctrl irq) failed\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571 goto bail_out;
572 }
573
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100574 if (0 > acm_set_control(acm, acm->ctrlout = ACM_CTRL_DTR | ACM_CTRL_RTS) &&
575 (acm->ctrl_caps & USB_CDC_CAP_LINE))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 goto full_bailout;
Alan Cox10077d42009-06-11 12:36:09 +0100577
Oliver Neukum11ea8592008-06-20 11:25:57 +0200578 usb_autopm_put_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700579
David Kubicek61a87ad2005-11-01 18:51:34 +0100580 INIT_LIST_HEAD(&acm->spare_read_urbs);
581 INIT_LIST_HEAD(&acm->spare_read_bufs);
582 INIT_LIST_HEAD(&acm->filled_read_bufs);
Alan Cox6e47e062009-06-11 12:37:06 +0100583
584 for (i = 0; i < acm->rx_buflimit; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100585 list_add(&(acm->ru[i].list), &acm->spare_read_urbs);
Alan Cox6e47e062009-06-11 12:37:06 +0100586 for (i = 0; i < acm->rx_buflimit; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100587 list_add(&(acm->rb[i].list), &acm->spare_read_bufs);
David Kubicek61a87ad2005-11-01 18:51:34 +0100588
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100589 acm->throttle = 0;
590
Oliver Neukum7af25b42009-09-08 23:51:28 +0200591 set_bit(ASYNCB_INITIALIZED, &acm->port.flags);
Alan Cox10077d42009-06-11 12:36:09 +0100592 rv = tty_port_block_til_ready(&acm->port, tty, filp);
Henry Gebhardt18a77b52009-11-04 11:19:28 +0100593 tasklet_schedule(&acm->urb_task);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100594
Oliver Neukum1365baf2007-10-12 17:24:28 +0200595 mutex_unlock(&acm->mutex);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100596out:
Oliver Neukum94409cc2008-02-11 15:22:29 +0100597 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700598 return rv;
599
600full_bailout:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601 usb_kill_urb(acm->ctrlurb);
602bail_out:
Alan Cox10077d42009-06-11 12:36:09 +0100603 acm->port.count--;
Oliver Neukum1365baf2007-10-12 17:24:28 +0200604 mutex_unlock(&acm->mutex);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100605 usb_autopm_put_interface(acm->control);
Oliver Neukum94409cc2008-02-11 15:22:29 +0100606early_bail:
607 mutex_unlock(&open_mutex);
Alan Cox10077d42009-06-11 12:36:09 +0100608 tty_port_tty_set(&acm->port, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700609 return -EIO;
610}
611
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700612static void acm_tty_unregister(struct acm *acm)
613{
Alan Cox10077d42009-06-11 12:36:09 +0100614 int i, nr;
David Kubicek61a87ad2005-11-01 18:51:34 +0100615
Oliver Neukum86478942006-05-13 22:50:47 +0200616 nr = acm->rx_buflimit;
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700617 tty_unregister_device(acm_tty_driver, acm->minor);
618 usb_put_intf(acm->control);
619 acm_table[acm->minor] = NULL;
620 usb_free_urb(acm->ctrlurb);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100621 for (i = 0; i < ACM_NW; i++)
622 usb_free_urb(acm->wb[i].urb);
Oliver Neukum86478942006-05-13 22:50:47 +0200623 for (i = 0; i < nr; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100624 usb_free_urb(acm->ru[i].urb);
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100625 kfree(acm->country_codes);
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700626 kfree(acm);
627}
628
David Brownelle5fbab52008-08-06 18:46:10 -0700629static int acm_tty_chars_in_buffer(struct tty_struct *tty);
630
Arnd Bergmann4e608672010-06-01 22:53:04 +0200631static void acm_port_down(struct acm *acm)
Alan Cox10077d42009-06-11 12:36:09 +0100632{
633 int i, nr = acm->rx_buflimit;
634 mutex_lock(&open_mutex);
635 if (acm->dev) {
636 usb_autopm_get_interface(acm->control);
637 acm_set_control(acm, acm->ctrlout = 0);
Alan Cox10077d42009-06-11 12:36:09 +0100638 usb_kill_urb(acm->ctrlurb);
639 for (i = 0; i < ACM_NW; i++)
640 usb_kill_urb(acm->wb[i].urb);
Johan Hovold23b80552011-03-22 11:12:09 +0100641 tasklet_disable(&acm->urb_task);
Alan Cox10077d42009-06-11 12:36:09 +0100642 for (i = 0; i < nr; i++)
643 usb_kill_urb(acm->ru[i].urb);
Johan Hovold23b80552011-03-22 11:12:09 +0100644 tasklet_enable(&acm->urb_task);
Alan Cox10077d42009-06-11 12:36:09 +0100645 acm->control->needs_remote_wakeup = 0;
646 usb_autopm_put_interface(acm->control);
647 }
648 mutex_unlock(&open_mutex);
649}
650
651static void acm_tty_hangup(struct tty_struct *tty)
652{
653 struct acm *acm = tty->driver_data;
654 tty_port_hangup(&acm->port);
Arnd Bergmann4e608672010-06-01 22:53:04 +0200655 acm_port_down(acm);
Alan Cox10077d42009-06-11 12:36:09 +0100656}
657
Linus Torvalds1da177e2005-04-16 15:20:36 -0700658static void acm_tty_close(struct tty_struct *tty, struct file *filp)
659{
660 struct acm *acm = tty->driver_data;
661
Alan Cox10077d42009-06-11 12:36:09 +0100662 /* Perform the closing process and see if we need to do the hardware
663 shutdown */
Francesco Lavra051522b2009-11-03 10:53:07 +0000664 if (!acm)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700665 return;
Francesco Lavra051522b2009-11-03 10:53:07 +0000666 if (tty_port_close_start(&acm->port, tty, filp) == 0) {
667 mutex_lock(&open_mutex);
668 if (!acm->dev) {
669 tty_port_tty_set(&acm->port, NULL);
670 acm_tty_unregister(acm);
671 tty->driver_data = NULL;
672 }
673 mutex_unlock(&open_mutex);
674 return;
675 }
Arnd Bergmann4e608672010-06-01 22:53:04 +0200676 acm_port_down(acm);
Alan Cox10077d42009-06-11 12:36:09 +0100677 tty_port_close_end(&acm->port, tty);
Alan Cox10077d42009-06-11 12:36:09 +0100678 tty_port_tty_set(&acm->port, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700679}
680
Alan Cox6e47e062009-06-11 12:37:06 +0100681static int acm_tty_write(struct tty_struct *tty,
682 const unsigned char *buf, int count)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700683{
684 struct acm *acm = tty->driver_data;
685 int stat;
Oliver Neukum884b6002005-04-21 21:28:02 +0200686 unsigned long flags;
687 int wbn;
688 struct acm_wb *wb;
689
Linus Torvalds1da177e2005-04-16 15:20:36 -0700690 if (!ACM_READY(acm))
691 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700692 if (!count)
693 return 0;
694
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100695 dev_vdbg(&acm->data->dev, "%s - count %d\n", __func__, count);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100696
Oliver Neukum884b6002005-04-21 21:28:02 +0200697 spin_lock_irqsave(&acm->write_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100698 wbn = acm_wb_alloc(acm);
699 if (wbn < 0) {
Oliver Neukum884b6002005-04-21 21:28:02 +0200700 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200701 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700702 }
Oliver Neukum884b6002005-04-21 21:28:02 +0200703 wb = &acm->wb[wbn];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700704
Oliver Neukum884b6002005-04-21 21:28:02 +0200705 count = (count > acm->writesize) ? acm->writesize : count;
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100706 dev_vdbg(&acm->data->dev, "%s - write %d\n", __func__, count);
Oliver Neukum884b6002005-04-21 21:28:02 +0200707 memcpy(wb->buf, buf, count);
708 wb->len = count;
709 spin_unlock_irqrestore(&acm->write_lock, flags);
710
Alan Cox6e47e062009-06-11 12:37:06 +0100711 stat = acm_write_start(acm, wbn);
712 if (stat < 0)
Oliver Neukum884b6002005-04-21 21:28:02 +0200713 return stat;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700714 return count;
715}
716
717static int acm_tty_write_room(struct tty_struct *tty)
718{
719 struct acm *acm = tty->driver_data;
720 if (!ACM_READY(acm))
721 return -EINVAL;
Oliver Neukum884b6002005-04-21 21:28:02 +0200722 /*
723 * Do not let the line discipline to know that we have a reserve,
724 * or it might get too enthusiastic.
725 */
David Brownell934da462008-08-06 18:44:12 -0700726 return acm_wb_is_avail(acm) ? acm->writesize : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727}
728
729static int acm_tty_chars_in_buffer(struct tty_struct *tty)
730{
731 struct acm *acm = tty->driver_data;
732 if (!ACM_READY(acm))
Alan Cox23198fd2009-07-20 16:05:27 +0100733 return 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200734 /*
735 * This is inaccurate (overcounts), but it works.
736 */
Oliver Neukum86478942006-05-13 22:50:47 +0200737 return (ACM_NW - acm_wb_is_avail(acm)) * acm->writesize;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700738}
739
740static void acm_tty_throttle(struct tty_struct *tty)
741{
742 struct acm *acm = tty->driver_data;
743 if (!ACM_READY(acm))
744 return;
745 spin_lock_bh(&acm->throttle_lock);
746 acm->throttle = 1;
747 spin_unlock_bh(&acm->throttle_lock);
748}
749
750static void acm_tty_unthrottle(struct tty_struct *tty)
751{
752 struct acm *acm = tty->driver_data;
753 if (!ACM_READY(acm))
754 return;
755 spin_lock_bh(&acm->throttle_lock);
756 acm->throttle = 0;
757 spin_unlock_bh(&acm->throttle_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +0100758 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700759}
760
Alan Cox9e989662008-07-22 11:18:03 +0100761static int acm_tty_break_ctl(struct tty_struct *tty, int state)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700762{
763 struct acm *acm = tty->driver_data;
Alan Cox9e989662008-07-22 11:18:03 +0100764 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700765 if (!ACM_READY(acm))
Alan Cox9e989662008-07-22 11:18:03 +0100766 return -EINVAL;
767 retval = acm_send_break(acm, state ? 0xffff : 0);
768 if (retval < 0)
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100769 dev_dbg(&acm->control->dev, "%s - send break failed\n",
770 __func__);
Alan Cox9e989662008-07-22 11:18:03 +0100771 return retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700772}
773
Alan Cox60b33c12011-02-14 16:26:14 +0000774static int acm_tty_tiocmget(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700775{
776 struct acm *acm = tty->driver_data;
777
778 if (!ACM_READY(acm))
779 return -EINVAL;
780
781 return (acm->ctrlout & ACM_CTRL_DTR ? TIOCM_DTR : 0) |
782 (acm->ctrlout & ACM_CTRL_RTS ? TIOCM_RTS : 0) |
783 (acm->ctrlin & ACM_CTRL_DSR ? TIOCM_DSR : 0) |
784 (acm->ctrlin & ACM_CTRL_RI ? TIOCM_RI : 0) |
785 (acm->ctrlin & ACM_CTRL_DCD ? TIOCM_CD : 0) |
786 TIOCM_CTS;
787}
788
Alan Cox20b9d172011-02-14 16:26:50 +0000789static int acm_tty_tiocmset(struct tty_struct *tty,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700790 unsigned int set, unsigned int clear)
791{
792 struct acm *acm = tty->driver_data;
793 unsigned int newctrl;
794
795 if (!ACM_READY(acm))
796 return -EINVAL;
797
798 newctrl = acm->ctrlout;
Alan Cox6e47e062009-06-11 12:37:06 +0100799 set = (set & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
800 (set & TIOCM_RTS ? ACM_CTRL_RTS : 0);
801 clear = (clear & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
802 (clear & TIOCM_RTS ? ACM_CTRL_RTS : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700803
804 newctrl = (newctrl & ~clear) | set;
805
806 if (acm->ctrlout == newctrl)
807 return 0;
808 return acm_set_control(acm, acm->ctrlout = newctrl);
809}
810
Alan Cox6caa76b2011-02-14 16:27:22 +0000811static int acm_tty_ioctl(struct tty_struct *tty,
Alan Cox6e47e062009-06-11 12:37:06 +0100812 unsigned int cmd, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700813{
814 struct acm *acm = tty->driver_data;
815
816 if (!ACM_READY(acm))
817 return -EINVAL;
818
819 return -ENOIOCTLCMD;
820}
821
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100822static const __u32 acm_tty_speed[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700823 0, 50, 75, 110, 134, 150, 200, 300, 600,
824 1200, 1800, 2400, 4800, 9600, 19200, 38400,
825 57600, 115200, 230400, 460800, 500000, 576000,
826 921600, 1000000, 1152000, 1500000, 2000000,
827 2500000, 3000000, 3500000, 4000000
828};
829
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100830static const __u8 acm_tty_size[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700831 5, 6, 7, 8
832};
833
Alan Cox6e47e062009-06-11 12:37:06 +0100834static void acm_tty_set_termios(struct tty_struct *tty,
835 struct ktermios *termios_old)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700836{
837 struct acm *acm = tty->driver_data;
Alan Cox606d0992006-12-08 02:38:45 -0800838 struct ktermios *termios = tty->termios;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700839 struct usb_cdc_line_coding newline;
840 int newctrl = acm->ctrlout;
841
842 if (!ACM_READY(acm))
843 return;
844
Alan Cox9b80fee2009-09-19 13:13:23 -0700845 newline.dwDTERate = cpu_to_le32(tty_get_baud_rate(tty));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700846 newline.bCharFormat = termios->c_cflag & CSTOPB ? 2 : 0;
847 newline.bParityType = termios->c_cflag & PARENB ?
Alan Cox6e47e062009-06-11 12:37:06 +0100848 (termios->c_cflag & PARODD ? 1 : 2) +
849 (termios->c_cflag & CMSPAR ? 2 : 0) : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850 newline.bDataBits = acm_tty_size[(termios->c_cflag & CSIZE) >> 4];
Alan Cox6e47e062009-06-11 12:37:06 +0100851 /* FIXME: Needs to clear unsupported bits in the termios */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700852 acm->clocal = ((termios->c_cflag & CLOCAL) != 0);
853
854 if (!newline.dwDTERate) {
855 newline.dwDTERate = acm->line.dwDTERate;
856 newctrl &= ~ACM_CTRL_DTR;
Alan Cox6e47e062009-06-11 12:37:06 +0100857 } else
858 newctrl |= ACM_CTRL_DTR;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700859
860 if (newctrl != acm->ctrlout)
861 acm_set_control(acm, acm->ctrlout = newctrl);
862
863 if (memcmp(&acm->line, &newline, sizeof newline)) {
864 memcpy(&acm->line, &newline, sizeof newline);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100865 dev_dbg(&acm->control->dev, "%s - set line: %d %d %d %d\n",
866 __func__,
867 le32_to_cpu(newline.dwDTERate),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700868 newline.bCharFormat, newline.bParityType,
869 newline.bDataBits);
870 acm_set_line(acm, &acm->line);
871 }
872}
873
874/*
875 * USB probe and disconnect routines.
876 */
877
Oliver Neukum830f4022008-06-25 14:17:16 +0200878/* Little helpers: write/read buffers free */
Oliver Neukum884b6002005-04-21 21:28:02 +0200879static void acm_write_buffers_free(struct acm *acm)
880{
881 int i;
882 struct acm_wb *wb;
Oliver Neukuma496c642008-10-21 10:39:04 +0200883 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
Oliver Neukum884b6002005-04-21 21:28:02 +0200884
Alan Cox6e47e062009-06-11 12:37:06 +0100885 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++)
Daniel Mack997ea582010-04-12 13:17:25 +0200886 usb_free_coherent(usb_dev, acm->writesize, wb->buf, wb->dmah);
Oliver Neukum884b6002005-04-21 21:28:02 +0200887}
888
Oliver Neukum830f4022008-06-25 14:17:16 +0200889static void acm_read_buffers_free(struct acm *acm)
890{
891 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
892 int i, n = acm->rx_buflimit;
893
894 for (i = 0; i < n; i++)
Daniel Mack997ea582010-04-12 13:17:25 +0200895 usb_free_coherent(usb_dev, acm->readsize,
896 acm->rb[i].base, acm->rb[i].dma);
Oliver Neukum830f4022008-06-25 14:17:16 +0200897}
898
Oliver Neukum884b6002005-04-21 21:28:02 +0200899/* Little helper: write buffers allocate */
900static int acm_write_buffers_alloc(struct acm *acm)
901{
902 int i;
903 struct acm_wb *wb;
904
Oliver Neukum86478942006-05-13 22:50:47 +0200905 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++) {
Daniel Mack997ea582010-04-12 13:17:25 +0200906 wb->buf = usb_alloc_coherent(acm->dev, acm->writesize, GFP_KERNEL,
Oliver Neukum884b6002005-04-21 21:28:02 +0200907 &wb->dmah);
908 if (!wb->buf) {
909 while (i != 0) {
910 --i;
911 --wb;
Daniel Mack997ea582010-04-12 13:17:25 +0200912 usb_free_coherent(acm->dev, acm->writesize,
Oliver Neukum884b6002005-04-21 21:28:02 +0200913 wb->buf, wb->dmah);
914 }
915 return -ENOMEM;
916 }
917 }
918 return 0;
919}
920
Alan Cox10077d42009-06-11 12:36:09 +0100921static int acm_probe(struct usb_interface *intf,
922 const struct usb_device_id *id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700923{
924 struct usb_cdc_union_desc *union_header = NULL;
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100925 struct usb_cdc_country_functional_desc *cfd = NULL;
David Brownellc6dbf552008-04-13 14:00:44 -0700926 unsigned char *buffer = intf->altsetting->extra;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700927 int buflen = intf->altsetting->extralen;
928 struct usb_interface *control_interface;
929 struct usb_interface *data_interface;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +0200930 struct usb_endpoint_descriptor *epctrl = NULL;
931 struct usb_endpoint_descriptor *epread = NULL;
932 struct usb_endpoint_descriptor *epwrite = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700933 struct usb_device *usb_dev = interface_to_usbdev(intf);
934 struct acm *acm;
935 int minor;
Alan Cox6e47e062009-06-11 12:37:06 +0100936 int ctrlsize, readsize;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700937 u8 *buf;
938 u8 ac_management_function = 0;
939 u8 call_management_function = 0;
940 int call_interface_num = -1;
941 int data_interface_num;
942 unsigned long quirks;
Oliver Neukum86478942006-05-13 22:50:47 +0200943 int num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +0100944 int i;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +0200945 int combined_interfaces = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700946
Oliver Neukum86478942006-05-13 22:50:47 +0200947 /* normal quirks */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700948 quirks = (unsigned long)id->driver_info;
Oliver Neukum86478942006-05-13 22:50:47 +0200949 num_rx_buf = (quirks == SINGLE_RX_URB) ? 1 : ACM_NR;
950
951 /* handle quirks deadly to normal probing*/
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952 if (quirks == NO_UNION_NORMAL) {
953 data_interface = usb_ifnum_to_if(usb_dev, 1);
954 control_interface = usb_ifnum_to_if(usb_dev, 0);
955 goto skip_normal_probe;
956 }
Alan Cox6e47e062009-06-11 12:37:06 +0100957
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958 /* normal probing*/
959 if (!buffer) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700960 dev_err(&intf->dev, "Weird descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700961 return -EINVAL;
962 }
963
964 if (!buflen) {
Toby Gray577045c2010-09-02 10:46:20 +0100965 if (intf->cur_altsetting->endpoint &&
966 intf->cur_altsetting->endpoint->extralen &&
Alan Cox6e47e062009-06-11 12:37:06 +0100967 intf->cur_altsetting->endpoint->extra) {
968 dev_dbg(&intf->dev,
969 "Seeking extra descriptors on endpoint\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700970 buflen = intf->cur_altsetting->endpoint->extralen;
971 buffer = intf->cur_altsetting->endpoint->extra;
972 } else {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700973 dev_err(&intf->dev,
974 "Zero length descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700975 return -EINVAL;
976 }
977 }
978
979 while (buflen > 0) {
Alan Cox6e47e062009-06-11 12:37:06 +0100980 if (buffer[1] != USB_DT_CS_INTERFACE) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700981 dev_err(&intf->dev, "skipping garbage\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700982 goto next_desc;
983 }
984
Alan Cox6e47e062009-06-11 12:37:06 +0100985 switch (buffer[2]) {
986 case USB_CDC_UNION_TYPE: /* we've found it */
987 if (union_header) {
988 dev_err(&intf->dev, "More than one "
989 "union descriptor, skipping ...\n");
990 goto next_desc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700991 }
Alan Cox6e47e062009-06-11 12:37:06 +0100992 union_header = (struct usb_cdc_union_desc *)buffer;
993 break;
994 case USB_CDC_COUNTRY_TYPE: /* export through sysfs*/
995 cfd = (struct usb_cdc_country_functional_desc *)buffer;
996 break;
997 case USB_CDC_HEADER_TYPE: /* maybe check version */
998 break; /* for now we ignore it */
999 case USB_CDC_ACM_TYPE:
1000 ac_management_function = buffer[3];
1001 break;
1002 case USB_CDC_CALL_MANAGEMENT_TYPE:
1003 call_management_function = buffer[3];
1004 call_interface_num = buffer[4];
Julian Calabyce126642010-01-05 23:58:20 +11001005 if ( (quirks & NOT_A_MODEM) == 0 && (call_management_function & 3) != 3)
Alan Cox6e47e062009-06-11 12:37:06 +01001006 dev_err(&intf->dev, "This device cannot do calls on its own. It is not a modem.\n");
1007 break;
1008 default:
1009 /* there are LOTS more CDC descriptors that
1010 * could legitimately be found here.
1011 */
1012 dev_dbg(&intf->dev, "Ignoring descriptor: "
1013 "type %02x, length %d\n",
1014 buffer[2], buffer[0]);
1015 break;
1016 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001017next_desc:
1018 buflen -= buffer[0];
1019 buffer += buffer[0];
1020 }
1021
1022 if (!union_header) {
1023 if (call_interface_num > 0) {
Alan Cox6e47e062009-06-11 12:37:06 +01001024 dev_dbg(&intf->dev, "No union descriptor, using call management descriptor\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001025 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = call_interface_num));
1026 control_interface = intf;
1027 } else {
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001028 if (intf->cur_altsetting->desc.bNumEndpoints != 3) {
1029 dev_dbg(&intf->dev,"No union descriptor, giving up\n");
1030 return -ENODEV;
1031 } else {
1032 dev_warn(&intf->dev,"No union descriptor, testing for castrated device\n");
1033 combined_interfaces = 1;
1034 control_interface = data_interface = intf;
1035 goto look_for_collapsed_interface;
1036 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001037 }
1038 } else {
1039 control_interface = usb_ifnum_to_if(usb_dev, union_header->bMasterInterface0);
1040 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = union_header->bSlaveInterface0));
1041 if (!control_interface || !data_interface) {
Alan Cox6e47e062009-06-11 12:37:06 +01001042 dev_dbg(&intf->dev, "no interfaces\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001043 return -ENODEV;
1044 }
1045 }
Alan Cox6e47e062009-06-11 12:37:06 +01001046
Linus Torvalds1da177e2005-04-16 15:20:36 -07001047 if (data_interface_num != call_interface_num)
Alan Cox6e47e062009-06-11 12:37:06 +01001048 dev_dbg(&intf->dev, "Separate call control interface. That is not fully supported.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001049
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001050 if (control_interface == data_interface) {
1051 /* some broken devices designed for windows work this way */
1052 dev_warn(&intf->dev,"Control and data interfaces are not separated!\n");
1053 combined_interfaces = 1;
1054 /* a popular other OS doesn't use it */
1055 quirks |= NO_CAP_LINE;
1056 if (data_interface->cur_altsetting->desc.bNumEndpoints != 3) {
1057 dev_err(&intf->dev, "This needs exactly 3 endpoints\n");
1058 return -EINVAL;
1059 }
1060look_for_collapsed_interface:
1061 for (i = 0; i < 3; i++) {
1062 struct usb_endpoint_descriptor *ep;
1063 ep = &data_interface->cur_altsetting->endpoint[i].desc;
1064
1065 if (usb_endpoint_is_int_in(ep))
1066 epctrl = ep;
1067 else if (usb_endpoint_is_bulk_out(ep))
1068 epwrite = ep;
1069 else if (usb_endpoint_is_bulk_in(ep))
1070 epread = ep;
1071 else
1072 return -EINVAL;
1073 }
1074 if (!epctrl || !epread || !epwrite)
1075 return -ENODEV;
1076 else
1077 goto made_compressed_probe;
1078 }
1079
Linus Torvalds1da177e2005-04-16 15:20:36 -07001080skip_normal_probe:
1081
1082 /*workaround for switched interfaces */
Alan Cox6e47e062009-06-11 12:37:06 +01001083 if (data_interface->cur_altsetting->desc.bInterfaceClass
1084 != CDC_DATA_INTERFACE_TYPE) {
1085 if (control_interface->cur_altsetting->desc.bInterfaceClass
1086 == CDC_DATA_INTERFACE_TYPE) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001087 struct usb_interface *t;
Alan Cox6e47e062009-06-11 12:37:06 +01001088 dev_dbg(&intf->dev,
1089 "Your device has switched interfaces.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001090 t = control_interface;
1091 control_interface = data_interface;
1092 data_interface = t;
1093 } else {
1094 return -EINVAL;
1095 }
1096 }
Alan Stern74da5d62007-08-02 13:29:10 -04001097
1098 /* Accept probe requests only for the control interface */
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001099 if (!combined_interfaces && intf != control_interface)
Alan Stern74da5d62007-08-02 13:29:10 -04001100 return -ENODEV;
Alan Cox6e47e062009-06-11 12:37:06 +01001101
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001102 if (!combined_interfaces && usb_interface_claimed(data_interface)) {
1103 /* valid in this context */
Alan Cox6e47e062009-06-11 12:37:06 +01001104 dev_dbg(&intf->dev, "The data interface isn't available\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001105 return -EBUSY;
1106 }
1107
1108
1109 if (data_interface->cur_altsetting->desc.bNumEndpoints < 2)
1110 return -EINVAL;
1111
1112 epctrl = &control_interface->cur_altsetting->endpoint[0].desc;
1113 epread = &data_interface->cur_altsetting->endpoint[0].desc;
1114 epwrite = &data_interface->cur_altsetting->endpoint[1].desc;
1115
1116
1117 /* workaround for switched endpoints */
Luiz Fernando N. Capitulino45aea702006-10-26 13:02:48 -03001118 if (!usb_endpoint_dir_in(epread)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001119 /* descriptors are swapped */
1120 struct usb_endpoint_descriptor *t;
Alan Cox6e47e062009-06-11 12:37:06 +01001121 dev_dbg(&intf->dev,
1122 "The data interface has switched endpoints\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001123 t = epread;
1124 epread = epwrite;
1125 epwrite = t;
1126 }
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001127made_compressed_probe:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +01001128 dev_dbg(&intf->dev, "interfaces are valid\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129 for (minor = 0; minor < ACM_TTY_MINORS && acm_table[minor]; minor++);
1130
1131 if (minor == ACM_TTY_MINORS) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -07001132 dev_err(&intf->dev, "no more free acm devices\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001133 return -ENODEV;
1134 }
1135
Alan Cox6e47e062009-06-11 12:37:06 +01001136 acm = kzalloc(sizeof(struct acm), GFP_KERNEL);
1137 if (acm == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001138 dev_err(&intf->dev, "out of memory (acm kzalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001139 goto alloc_fail;
1140 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001141
1142 ctrlsize = le16_to_cpu(epctrl->wMaxPacketSize);
Alan Cox6e47e062009-06-11 12:37:06 +01001143 readsize = le16_to_cpu(epread->wMaxPacketSize) *
1144 (quirks == SINGLE_RX_URB ? 1 : 2);
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001145 acm->combined_interfaces = combined_interfaces;
David Engrafe4cf3aa2008-03-20 10:01:34 +01001146 acm->writesize = le16_to_cpu(epwrite->wMaxPacketSize) * 20;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001147 acm->control = control_interface;
1148 acm->data = data_interface;
1149 acm->minor = minor;
1150 acm->dev = usb_dev;
1151 acm->ctrl_caps = ac_management_function;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001152 if (quirks & NO_CAP_LINE)
1153 acm->ctrl_caps &= ~USB_CDC_CAP_LINE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001154 acm->ctrlsize = ctrlsize;
1155 acm->readsize = readsize;
Oliver Neukum86478942006-05-13 22:50:47 +02001156 acm->rx_buflimit = num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +01001157 acm->urb_task.func = acm_rx_tasklet;
1158 acm->urb_task.data = (unsigned long) acm;
David Howellsc4028952006-11-22 14:57:56 +00001159 INIT_WORK(&acm->work, acm_softint);
David Brownelle5fbab52008-08-06 18:46:10 -07001160 init_waitqueue_head(&acm->drain_wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161 spin_lock_init(&acm->throttle_lock);
Oliver Neukum884b6002005-04-21 21:28:02 +02001162 spin_lock_init(&acm->write_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +01001163 spin_lock_init(&acm->read_lock);
Oliver Neukum1365baf2007-10-12 17:24:28 +02001164 mutex_init(&acm->mutex);
David Kubicek61a87ad2005-11-01 18:51:34 +01001165 acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress);
Oliver Neukumcf7fdd52009-08-04 23:52:09 +02001166 acm->is_int_ep = usb_endpoint_xfer_int(epread);
1167 if (acm->is_int_ep)
1168 acm->bInterval = epread->bInterval;
Alan Cox739e0282009-06-11 12:27:50 +01001169 tty_port_init(&acm->port);
1170 acm->port.ops = &acm_port_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001171
Daniel Mack997ea582010-04-12 13:17:25 +02001172 buf = usb_alloc_coherent(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001173 if (!buf) {
Johan Hovold255ab562011-03-22 11:12:13 +01001174 dev_err(&intf->dev, "out of memory (ctrl buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001175 goto alloc_fail2;
1176 }
1177 acm->ctrl_buffer = buf;
1178
Oliver Neukum884b6002005-04-21 21:28:02 +02001179 if (acm_write_buffers_alloc(acm) < 0) {
Johan Hovold255ab562011-03-22 11:12:13 +01001180 dev_err(&intf->dev, "out of memory (write buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001181 goto alloc_fail4;
1182 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001183
1184 acm->ctrlurb = usb_alloc_urb(0, GFP_KERNEL);
1185 if (!acm->ctrlurb) {
Johan Hovold255ab562011-03-22 11:12:13 +01001186 dev_err(&intf->dev, "out of memory (ctrlurb kmalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001187 goto alloc_fail5;
1188 }
Oliver Neukum86478942006-05-13 22:50:47 +02001189 for (i = 0; i < num_rx_buf; i++) {
David Kubicek61a87ad2005-11-01 18:51:34 +01001190 struct acm_ru *rcv = &(acm->ru[i]);
1191
Alan Cox6e47e062009-06-11 12:37:06 +01001192 rcv->urb = usb_alloc_urb(0, GFP_KERNEL);
1193 if (rcv->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001194 dev_err(&intf->dev,
Alan Cox6e47e062009-06-11 12:37:06 +01001195 "out of memory (read urbs usb_alloc_urb)\n");
Axel Linc2572b72010-05-31 08:04:47 +08001196 goto alloc_fail6;
David Kubicek61a87ad2005-11-01 18:51:34 +01001197 }
1198
1199 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1200 rcv->instance = acm;
1201 }
Oliver Neukum86478942006-05-13 22:50:47 +02001202 for (i = 0; i < num_rx_buf; i++) {
David Brownell672c4e12008-08-06 18:41:12 -07001203 struct acm_rb *rb = &(acm->rb[i]);
David Kubicek61a87ad2005-11-01 18:51:34 +01001204
Daniel Mack997ea582010-04-12 13:17:25 +02001205 rb->base = usb_alloc_coherent(acm->dev, readsize,
David Brownell672c4e12008-08-06 18:41:12 -07001206 GFP_KERNEL, &rb->dma);
1207 if (!rb->base) {
Johan Hovold255ab562011-03-22 11:12:13 +01001208 dev_err(&intf->dev,
Daniel Mack997ea582010-04-12 13:17:25 +02001209 "out of memory (read bufs usb_alloc_coherent)\n");
David Kubicek61a87ad2005-11-01 18:51:34 +01001210 goto alloc_fail7;
1211 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001212 }
Alan Cox6e47e062009-06-11 12:37:06 +01001213 for (i = 0; i < ACM_NW; i++) {
David Engrafe4cf3aa2008-03-20 10:01:34 +01001214 struct acm_wb *snd = &(acm->wb[i]);
1215
Alan Cox6e47e062009-06-11 12:37:06 +01001216 snd->urb = usb_alloc_urb(0, GFP_KERNEL);
1217 if (snd->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001218 dev_err(&intf->dev,
Johan Hovold59d7fec2011-03-22 11:12:12 +01001219 "out of memory (write urbs usb_alloc_urb)\n");
Axel Linc2572b72010-05-31 08:04:47 +08001220 goto alloc_fail8;
David Engrafe4cf3aa2008-03-20 10:01:34 +01001221 }
1222
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +04001223 if (usb_endpoint_xfer_int(epwrite))
1224 usb_fill_int_urb(snd->urb, usb_dev,
1225 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1226 NULL, acm->writesize, acm_write_bulk, snd, epwrite->bInterval);
1227 else
1228 usb_fill_bulk_urb(snd->urb, usb_dev,
1229 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1230 NULL, acm->writesize, acm_write_bulk, snd);
David Engrafe4cf3aa2008-03-20 10:01:34 +01001231 snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1232 snd->instance = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001233 }
1234
Alan Cox6e47e062009-06-11 12:37:06 +01001235 usb_set_intfdata(intf, acm);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001236
1237 i = device_create_file(&intf->dev, &dev_attr_bmCapabilities);
1238 if (i < 0)
1239 goto alloc_fail8;
1240
1241 if (cfd) { /* export the country data */
1242 acm->country_codes = kmalloc(cfd->bLength - 4, GFP_KERNEL);
1243 if (!acm->country_codes)
1244 goto skip_countries;
1245 acm->country_code_size = cfd->bLength - 4;
Alan Cox6e47e062009-06-11 12:37:06 +01001246 memcpy(acm->country_codes, (u8 *)&cfd->wCountyCode0,
1247 cfd->bLength - 4);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001248 acm->country_rel_date = cfd->iCountryCodeRelDate;
1249
1250 i = device_create_file(&intf->dev, &dev_attr_wCountryCodes);
1251 if (i < 0) {
1252 kfree(acm->country_codes);
1253 goto skip_countries;
1254 }
1255
Alan Cox6e47e062009-06-11 12:37:06 +01001256 i = device_create_file(&intf->dev,
1257 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001258 if (i < 0) {
Axel Linc2572b72010-05-31 08:04:47 +08001259 device_remove_file(&intf->dev, &dev_attr_wCountryCodes);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001260 kfree(acm->country_codes);
1261 goto skip_countries;
1262 }
1263 }
1264
1265skip_countries:
Alan Cox6e47e062009-06-11 12:37:06 +01001266 usb_fill_int_urb(acm->ctrlurb, usb_dev,
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001267 usb_rcvintpipe(usb_dev, epctrl->bEndpointAddress),
1268 acm->ctrl_buffer, ctrlsize, acm_ctrl_irq, acm,
1269 /* works around buggy devices */
1270 epctrl->bInterval ? epctrl->bInterval : 0xff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001271 acm->ctrlurb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1272 acm->ctrlurb->transfer_dma = acm->ctrl_dma;
1273
Linus Torvalds1da177e2005-04-16 15:20:36 -07001274 dev_info(&intf->dev, "ttyACM%d: USB ACM device\n", minor);
1275
1276 acm_set_control(acm, acm->ctrlout);
1277
1278 acm->line.dwDTERate = cpu_to_le32(9600);
1279 acm->line.bDataBits = 8;
1280 acm_set_line(acm, &acm->line);
1281
1282 usb_driver_claim_interface(&acm_driver, data_interface, acm);
David Brownell672c4e12008-08-06 18:41:12 -07001283 usb_set_intfdata(data_interface, acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001284
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001285 usb_get_intf(control_interface);
1286 tty_register_device(acm_tty_driver, minor, &control_interface->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001287
1288 acm_table[minor] = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001289
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001290 return 0;
1291alloc_fail8:
David Engrafe4cf3aa2008-03-20 10:01:34 +01001292 for (i = 0; i < ACM_NW; i++)
1293 usb_free_urb(acm->wb[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001294alloc_fail7:
Oliver Neukum830f4022008-06-25 14:17:16 +02001295 acm_read_buffers_free(acm);
Axel Linc2572b72010-05-31 08:04:47 +08001296alloc_fail6:
Oliver Neukum86478942006-05-13 22:50:47 +02001297 for (i = 0; i < num_rx_buf; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +01001298 usb_free_urb(acm->ru[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001299 usb_free_urb(acm->ctrlurb);
1300alloc_fail5:
Oliver Neukum884b6002005-04-21 21:28:02 +02001301 acm_write_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001302alloc_fail4:
Daniel Mack997ea582010-04-12 13:17:25 +02001303 usb_free_coherent(usb_dev, ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001304alloc_fail2:
1305 kfree(acm);
1306alloc_fail:
1307 return -ENOMEM;
1308}
1309
Oliver Neukum1365baf2007-10-12 17:24:28 +02001310static void stop_data_traffic(struct acm *acm)
1311{
1312 int i;
Johan Hovolda5cc7ef2011-03-22 11:12:15 +01001313
1314 dev_dbg(&acm->control->dev, "%s\n", __func__);
Oliver Neukum1365baf2007-10-12 17:24:28 +02001315
1316 tasklet_disable(&acm->urb_task);
1317
1318 usb_kill_urb(acm->ctrlurb);
Alan Cox6e47e062009-06-11 12:37:06 +01001319 for (i = 0; i < ACM_NW; i++)
David Engrafe4cf3aa2008-03-20 10:01:34 +01001320 usb_kill_urb(acm->wb[i].urb);
Oliver Neukum1365baf2007-10-12 17:24:28 +02001321 for (i = 0; i < acm->rx_buflimit; i++)
1322 usb_kill_urb(acm->ru[i].urb);
1323
Oliver Neukum1365baf2007-10-12 17:24:28 +02001324 tasklet_enable(&acm->urb_task);
1325
1326 cancel_work_sync(&acm->work);
1327}
1328
Linus Torvalds1da177e2005-04-16 15:20:36 -07001329static void acm_disconnect(struct usb_interface *intf)
1330{
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001331 struct acm *acm = usb_get_intfdata(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001332 struct usb_device *usb_dev = interface_to_usbdev(intf);
Alan Cox10077d42009-06-11 12:36:09 +01001333 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001334
David Brownell672c4e12008-08-06 18:41:12 -07001335 /* sibling interface is already cleaning up */
1336 if (!acm)
Oliver Neukum86067eea2006-01-08 12:39:13 +01001337 return;
David Brownell672c4e12008-08-06 18:41:12 -07001338
1339 mutex_lock(&open_mutex);
Alan Cox6e47e062009-06-11 12:37:06 +01001340 if (acm->country_codes) {
Alan Stern74da5d62007-08-02 13:29:10 -04001341 device_remove_file(&acm->control->dev,
1342 &dev_attr_wCountryCodes);
1343 device_remove_file(&acm->control->dev,
1344 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001345 }
Alan Stern74da5d62007-08-02 13:29:10 -04001346 device_remove_file(&acm->control->dev, &dev_attr_bmCapabilities);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001347 acm->dev = NULL;
Oliver Neukum86067eea2006-01-08 12:39:13 +01001348 usb_set_intfdata(acm->control, NULL);
1349 usb_set_intfdata(acm->data, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001350
Oliver Neukum1365baf2007-10-12 17:24:28 +02001351 stop_data_traffic(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001352
Oliver Neukum884b6002005-04-21 21:28:02 +02001353 acm_write_buffers_free(acm);
Daniel Mack997ea582010-04-12 13:17:25 +02001354 usb_free_coherent(usb_dev, acm->ctrlsize, acm->ctrl_buffer,
1355 acm->ctrl_dma);
Oliver Neukum830f4022008-06-25 14:17:16 +02001356 acm_read_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001357
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001358 if (!acm->combined_interfaces)
1359 usb_driver_release_interface(&acm_driver, intf == acm->control ?
Oliver Neukum830f4022008-06-25 14:17:16 +02001360 acm->data : acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001361
Alan Cox10077d42009-06-11 12:36:09 +01001362 if (acm->port.count == 0) {
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001363 acm_tty_unregister(acm);
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001364 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001365 return;
1366 }
1367
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001368 mutex_unlock(&open_mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001369 tty = tty_port_tty_get(&acm->port);
1370 if (tty) {
1371 tty_hangup(tty);
1372 tty_kref_put(tty);
1373 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001374}
1375
Oliver Neukum35758582008-07-01 19:10:08 +02001376#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001377static int acm_suspend(struct usb_interface *intf, pm_message_t message)
1378{
1379 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001380 int cnt;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001381
Alan Stern65bfd292008-11-25 16:39:18 -05001382 if (message.event & PM_EVENT_AUTO) {
Oliver Neukum11ea8592008-06-20 11:25:57 +02001383 int b;
1384
1385 spin_lock_irq(&acm->read_lock);
1386 spin_lock(&acm->write_lock);
1387 b = acm->processing + acm->transmitting;
1388 spin_unlock(&acm->write_lock);
1389 spin_unlock_irq(&acm->read_lock);
1390 if (b)
1391 return -EBUSY;
1392 }
1393
1394 spin_lock_irq(&acm->read_lock);
1395 spin_lock(&acm->write_lock);
1396 cnt = acm->susp_count++;
1397 spin_unlock(&acm->write_lock);
1398 spin_unlock_irq(&acm->read_lock);
1399
1400 if (cnt)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001401 return 0;
1402 /*
1403 we treat opened interfaces differently,
1404 we must guard against open
1405 */
1406 mutex_lock(&acm->mutex);
1407
Alan Cox10077d42009-06-11 12:36:09 +01001408 if (acm->port.count)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001409 stop_data_traffic(acm);
1410
1411 mutex_unlock(&acm->mutex);
1412 return 0;
1413}
1414
1415static int acm_resume(struct usb_interface *intf)
1416{
1417 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001418 struct acm_wb *wb;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001419 int rv = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +02001420 int cnt;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001421
Oliver Neukum11ea8592008-06-20 11:25:57 +02001422 spin_lock_irq(&acm->read_lock);
1423 acm->susp_count -= 1;
1424 cnt = acm->susp_count;
1425 spin_unlock_irq(&acm->read_lock);
1426
1427 if (cnt)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001428 return 0;
1429
1430 mutex_lock(&acm->mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001431 if (acm->port.count) {
Oliver Neukum1365baf2007-10-12 17:24:28 +02001432 rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001433
1434 spin_lock_irq(&acm->write_lock);
1435 if (acm->delayed_wb) {
1436 wb = acm->delayed_wb;
1437 acm->delayed_wb = NULL;
1438 spin_unlock_irq(&acm->write_lock);
Oliver Neukumf0730922010-03-03 00:37:56 +01001439 acm_start_wb(acm, wb);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001440 } else {
1441 spin_unlock_irq(&acm->write_lock);
1442 }
1443
1444 /*
1445 * delayed error checking because we must
1446 * do the write path at all cost
1447 */
Oliver Neukum1365baf2007-10-12 17:24:28 +02001448 if (rv < 0)
Oliver Neukum11ea8592008-06-20 11:25:57 +02001449 goto err_out;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001450
1451 tasklet_schedule(&acm->urb_task);
1452 }
1453
1454err_out:
1455 mutex_unlock(&acm->mutex);
1456 return rv;
1457}
Oliver Neukum35758582008-07-01 19:10:08 +02001458
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001459static int acm_reset_resume(struct usb_interface *intf)
1460{
1461 struct acm *acm = usb_get_intfdata(intf);
1462 struct tty_struct *tty;
1463
1464 mutex_lock(&acm->mutex);
1465 if (acm->port.count) {
1466 tty = tty_port_tty_get(&acm->port);
1467 if (tty) {
1468 tty_hangup(tty);
1469 tty_kref_put(tty);
1470 }
1471 }
1472 mutex_unlock(&acm->mutex);
1473 return acm_resume(intf);
1474}
1475
Oliver Neukum35758582008-07-01 19:10:08 +02001476#endif /* CONFIG_PM */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001477
1478#define NOKIA_PCSUITE_ACM_INFO(x) \
1479 USB_DEVICE_AND_INTERFACE_INFO(0x0421, x, \
1480 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1481 USB_CDC_ACM_PROTO_VENDOR)
1482
Toby Gray4035e452010-09-01 16:01:19 +01001483#define SAMSUNG_PCSUITE_ACM_INFO(x) \
1484 USB_DEVICE_AND_INTERFACE_INFO(0x04e7, x, \
1485 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1486 USB_CDC_ACM_PROTO_VENDOR)
1487
Linus Torvalds1da177e2005-04-16 15:20:36 -07001488/*
1489 * USB driver structure.
1490 */
1491
Németh Márton6ef48522010-01-10 15:33:45 +01001492static const struct usb_device_id acm_ids[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493 /* quirky and broken devices */
1494 { USB_DEVICE(0x0870, 0x0001), /* Metricom GS Modem */
1495 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1496 },
Andrey Arapovb0e2a702007-07-04 17:11:42 +02001497 { USB_DEVICE(0x0e8d, 0x0003), /* FIREFLY, MediaTek Inc; andrey.arapov@gmail.com */
1498 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1499 },
Andrew Lunn0f9c7b42008-12-23 17:31:23 +01001500 { USB_DEVICE(0x0e8d, 0x3329), /* MediaTek Inc GPS */
1501 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1502 },
Masahito Omote8753e652005-07-29 12:17:25 -07001503 { USB_DEVICE(0x0482, 0x0203), /* KYOCERA AH-K3001V */
1504 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1505 },
Chris Malley91a9c922006-10-03 10:08:28 +01001506 { USB_DEVICE(0x079b, 0x000f), /* BT On-Air USB MODEM */
1507 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1508 },
Alan Cox7abcf202009-04-06 17:35:01 +01001509 { USB_DEVICE(0x0ace, 0x1602), /* ZyDAS 56K USB MODEM */
1510 .driver_info = SINGLE_RX_URB,
1511 },
Oliver Neukum86478942006-05-13 22:50:47 +02001512 { USB_DEVICE(0x0ace, 0x1608), /* ZyDAS 56K USB MODEM */
1513 .driver_info = SINGLE_RX_URB, /* firmware bug */
1514 },
Oliver Neukum3dd2ae82006-06-23 09:14:17 +02001515 { USB_DEVICE(0x0ace, 0x1611), /* ZyDAS 56K USB MODEM - new version */
1516 .driver_info = SINGLE_RX_URB, /* firmware bug */
1517 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001518 { USB_DEVICE(0x22b8, 0x7000), /* Motorola Q Phone */
1519 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1520 },
Iain McFarlane6149ed52008-05-04 00:13:49 +01001521 { USB_DEVICE(0x0803, 0x3095), /* Zoom Telephonics Model 3095F USB MODEM */
1522 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1523 },
Eric Sandeenc8fd2c32008-08-14 08:25:40 -05001524 { USB_DEVICE(0x0572, 0x1321), /* Conexant USB MODEM CX93010 */
1525 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1526 },
Alan Coxc89c60e2009-01-11 19:53:10 +00001527 { USB_DEVICE(0x0572, 0x1324), /* Conexant USB MODEM RD02-D400 */
1528 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1529 },
Xiao Kaijiancab98a02009-05-08 00:48:23 +08001530 { USB_DEVICE(0x0572, 0x1328), /* Shiro / Aztech USB MODEM UM-3100 */
1531 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1532 },
Dmitriy Taychenachev155df652009-02-25 12:36:51 +08001533 { USB_DEVICE(0x22b8, 0x6425), /* Motorola MOTOMAGX phones */
1534 },
Adam Richterc332b4e2009-02-18 16:17:15 -08001535 { USB_DEVICE(0x0572, 0x1329), /* Hummingbird huc56s (Conexant) */
1536 .driver_info = NO_UNION_NORMAL, /* union descriptor misplaced on
1537 data interface instead of
1538 communications interface.
1539 Maybe we should define a new
1540 quirk for this. */
1541 },
Kir Kolyshkin1f17c502009-05-28 20:33:58 +04001542 { USB_DEVICE(0x1bbb, 0x0003), /* Alcatel OT-I650 */
1543 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1544 },
Russ Nelsonc3baa192010-04-21 23:07:03 -04001545 { USB_DEVICE(0x1576, 0x03b1), /* Maretron USB100 */
1546 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1547 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001548
Adrian Taylorc1479a92009-11-19 10:35:33 +00001549 /* Nokia S60 phones expose two ACM channels. The first is
1550 * a modem and is picked up by the standard AT-command
1551 * information below. The second is 'vendor-specific' but
1552 * is treated as a serial device at the S60 end, so we want
1553 * to expose it on Linux too. */
1554 { NOKIA_PCSUITE_ACM_INFO(0x042D), }, /* Nokia 3250 */
1555 { NOKIA_PCSUITE_ACM_INFO(0x04D8), }, /* Nokia 5500 Sport */
1556 { NOKIA_PCSUITE_ACM_INFO(0x04C9), }, /* Nokia E50 */
1557 { NOKIA_PCSUITE_ACM_INFO(0x0419), }, /* Nokia E60 */
1558 { NOKIA_PCSUITE_ACM_INFO(0x044D), }, /* Nokia E61 */
1559 { NOKIA_PCSUITE_ACM_INFO(0x0001), }, /* Nokia E61i */
1560 { NOKIA_PCSUITE_ACM_INFO(0x0475), }, /* Nokia E62 */
1561 { NOKIA_PCSUITE_ACM_INFO(0x0508), }, /* Nokia E65 */
1562 { NOKIA_PCSUITE_ACM_INFO(0x0418), }, /* Nokia E70 */
1563 { NOKIA_PCSUITE_ACM_INFO(0x0425), }, /* Nokia N71 */
1564 { NOKIA_PCSUITE_ACM_INFO(0x0486), }, /* Nokia N73 */
1565 { NOKIA_PCSUITE_ACM_INFO(0x04DF), }, /* Nokia N75 */
1566 { NOKIA_PCSUITE_ACM_INFO(0x000e), }, /* Nokia N77 */
1567 { NOKIA_PCSUITE_ACM_INFO(0x0445), }, /* Nokia N80 */
1568 { NOKIA_PCSUITE_ACM_INFO(0x042F), }, /* Nokia N91 & N91 8GB */
1569 { NOKIA_PCSUITE_ACM_INFO(0x048E), }, /* Nokia N92 */
1570 { NOKIA_PCSUITE_ACM_INFO(0x0420), }, /* Nokia N93 */
1571 { NOKIA_PCSUITE_ACM_INFO(0x04E6), }, /* Nokia N93i */
1572 { NOKIA_PCSUITE_ACM_INFO(0x04B2), }, /* Nokia 5700 XpressMusic */
1573 { NOKIA_PCSUITE_ACM_INFO(0x0134), }, /* Nokia 6110 Navigator (China) */
1574 { NOKIA_PCSUITE_ACM_INFO(0x046E), }, /* Nokia 6110 Navigator */
1575 { NOKIA_PCSUITE_ACM_INFO(0x002f), }, /* Nokia 6120 classic & */
1576 { NOKIA_PCSUITE_ACM_INFO(0x0088), }, /* Nokia 6121 classic */
1577 { NOKIA_PCSUITE_ACM_INFO(0x00fc), }, /* Nokia 6124 classic */
1578 { NOKIA_PCSUITE_ACM_INFO(0x0042), }, /* Nokia E51 */
1579 { NOKIA_PCSUITE_ACM_INFO(0x00b0), }, /* Nokia E66 */
1580 { NOKIA_PCSUITE_ACM_INFO(0x00ab), }, /* Nokia E71 */
1581 { NOKIA_PCSUITE_ACM_INFO(0x0481), }, /* Nokia N76 */
1582 { NOKIA_PCSUITE_ACM_INFO(0x0007), }, /* Nokia N81 & N81 8GB */
1583 { NOKIA_PCSUITE_ACM_INFO(0x0071), }, /* Nokia N82 */
1584 { NOKIA_PCSUITE_ACM_INFO(0x04F0), }, /* Nokia N95 & N95-3 NAM */
1585 { NOKIA_PCSUITE_ACM_INFO(0x0070), }, /* Nokia N95 8GB */
1586 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1587 { NOKIA_PCSUITE_ACM_INFO(0x0099), }, /* Nokia 6210 Navigator, RM-367 */
1588 { NOKIA_PCSUITE_ACM_INFO(0x0128), }, /* Nokia 6210 Navigator, RM-419 */
1589 { NOKIA_PCSUITE_ACM_INFO(0x008f), }, /* Nokia 6220 Classic */
1590 { NOKIA_PCSUITE_ACM_INFO(0x00a0), }, /* Nokia 6650 */
1591 { NOKIA_PCSUITE_ACM_INFO(0x007b), }, /* Nokia N78 */
1592 { NOKIA_PCSUITE_ACM_INFO(0x0094), }, /* Nokia N85 */
1593 { NOKIA_PCSUITE_ACM_INFO(0x003a), }, /* Nokia N96 & N96-3 */
1594 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1595 { NOKIA_PCSUITE_ACM_INFO(0x0108), }, /* Nokia 5320 XpressMusic 2G */
1596 { NOKIA_PCSUITE_ACM_INFO(0x01f5), }, /* Nokia N97, RM-505 */
Przemo Firszt83a4eae2010-06-28 21:29:34 +01001597 { NOKIA_PCSUITE_ACM_INFO(0x02e3), }, /* Nokia 5230, RM-588 */
Toby Gray4035e452010-09-01 16:01:19 +01001598 { NOKIA_PCSUITE_ACM_INFO(0x0178), }, /* Nokia E63 */
1599 { NOKIA_PCSUITE_ACM_INFO(0x010e), }, /* Nokia E75 */
1600 { NOKIA_PCSUITE_ACM_INFO(0x02d9), }, /* Nokia 6760 Slide */
1601 { NOKIA_PCSUITE_ACM_INFO(0x01d0), }, /* Nokia E52 */
1602 { NOKIA_PCSUITE_ACM_INFO(0x0223), }, /* Nokia E72 */
1603 { NOKIA_PCSUITE_ACM_INFO(0x0275), }, /* Nokia X6 */
1604 { NOKIA_PCSUITE_ACM_INFO(0x026c), }, /* Nokia N97 Mini */
1605 { NOKIA_PCSUITE_ACM_INFO(0x0154), }, /* Nokia 5800 XpressMusic */
1606 { NOKIA_PCSUITE_ACM_INFO(0x04ce), }, /* Nokia E90 */
1607 { NOKIA_PCSUITE_ACM_INFO(0x01d4), }, /* Nokia E55 */
Arvid Ephraim Picciani721d92f2011-01-25 15:58:40 +01001608 { NOKIA_PCSUITE_ACM_INFO(0x0302), }, /* Nokia N8 */
Toby Gray4035e452010-09-01 16:01:19 +01001609 { SAMSUNG_PCSUITE_ACM_INFO(0x6651), }, /* Samsung GTi8510 (INNOV8) */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001610
1611 /* NOTE: non-Nokia COMM/ACM/0xff is likely MSFT RNDIS... NOT a modem! */
1612
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001613 /* Support Lego NXT using pbLua firmware */
Julian Calabyce126642010-01-05 23:58:20 +11001614 { USB_DEVICE(0x0694, 0xff00),
1615 .driver_info = NOT_A_MODEM,
Otavio Salvador7893afc2010-09-26 23:35:05 -03001616 },
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001617
Philippe Corbes5b239f02010-08-31 19:31:32 +02001618 /* control interfaces without any protocol set */
1619 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1620 USB_CDC_PROTO_NONE) },
1621
Linus Torvalds1da177e2005-04-16 15:20:36 -07001622 /* control interfaces with various AT-command sets */
1623 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1624 USB_CDC_ACM_PROTO_AT_V25TER) },
1625 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1626 USB_CDC_ACM_PROTO_AT_PCCA101) },
1627 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1628 USB_CDC_ACM_PROTO_AT_PCCA101_WAKE) },
1629 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1630 USB_CDC_ACM_PROTO_AT_GSM) },
1631 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
Alan Cox6e47e062009-06-11 12:37:06 +01001632 USB_CDC_ACM_PROTO_AT_3G) },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001633 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1634 USB_CDC_ACM_PROTO_AT_CDMA) },
1635
Linus Torvalds1da177e2005-04-16 15:20:36 -07001636 { }
1637};
1638
Alan Cox6e47e062009-06-11 12:37:06 +01001639MODULE_DEVICE_TABLE(usb, acm_ids);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001640
1641static struct usb_driver acm_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001642 .name = "cdc_acm",
1643 .probe = acm_probe,
1644 .disconnect = acm_disconnect,
Oliver Neukum35758582008-07-01 19:10:08 +02001645#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001646 .suspend = acm_suspend,
1647 .resume = acm_resume,
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001648 .reset_resume = acm_reset_resume,
Oliver Neukum35758582008-07-01 19:10:08 +02001649#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001650 .id_table = acm_ids,
Oliver Neukum35758582008-07-01 19:10:08 +02001651#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001652 .supports_autosuspend = 1,
Oliver Neukum35758582008-07-01 19:10:08 +02001653#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001654};
1655
1656/*
1657 * TTY driver structures.
1658 */
1659
Jeff Dikeb68e31d2006-10-02 02:17:18 -07001660static const struct tty_operations acm_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001661 .open = acm_tty_open,
1662 .close = acm_tty_close,
Alan Cox10077d42009-06-11 12:36:09 +01001663 .hangup = acm_tty_hangup,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001664 .write = acm_tty_write,
1665 .write_room = acm_tty_write_room,
1666 .ioctl = acm_tty_ioctl,
1667 .throttle = acm_tty_throttle,
1668 .unthrottle = acm_tty_unthrottle,
1669 .chars_in_buffer = acm_tty_chars_in_buffer,
1670 .break_ctl = acm_tty_break_ctl,
1671 .set_termios = acm_tty_set_termios,
1672 .tiocmget = acm_tty_tiocmget,
1673 .tiocmset = acm_tty_tiocmset,
1674};
1675
1676/*
1677 * Init / exit.
1678 */
1679
1680static int __init acm_init(void)
1681{
1682 int retval;
1683 acm_tty_driver = alloc_tty_driver(ACM_TTY_MINORS);
1684 if (!acm_tty_driver)
1685 return -ENOMEM;
1686 acm_tty_driver->owner = THIS_MODULE,
1687 acm_tty_driver->driver_name = "acm",
1688 acm_tty_driver->name = "ttyACM",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001689 acm_tty_driver->major = ACM_TTY_MAJOR,
1690 acm_tty_driver->minor_start = 0,
1691 acm_tty_driver->type = TTY_DRIVER_TYPE_SERIAL,
1692 acm_tty_driver->subtype = SERIAL_TYPE_NORMAL,
Greg Kroah-Hartman331b8312005-06-20 21:15:16 -07001693 acm_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001694 acm_tty_driver->init_termios = tty_std_termios;
Alan Cox6e47e062009-06-11 12:37:06 +01001695 acm_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD |
1696 HUPCL | CLOCAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001697 tty_set_operations(acm_tty_driver, &acm_ops);
1698
1699 retval = tty_register_driver(acm_tty_driver);
1700 if (retval) {
1701 put_tty_driver(acm_tty_driver);
1702 return retval;
1703 }
1704
1705 retval = usb_register(&acm_driver);
1706 if (retval) {
1707 tty_unregister_driver(acm_tty_driver);
1708 put_tty_driver(acm_tty_driver);
1709 return retval;
1710 }
1711
Johan Hovolda2c7b932011-03-22 11:12:18 +01001712 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001713
1714 return 0;
1715}
1716
1717static void __exit acm_exit(void)
1718{
1719 usb_deregister(&acm_driver);
1720 tty_unregister_driver(acm_tty_driver);
1721 put_tty_driver(acm_tty_driver);
1722}
1723
1724module_init(acm_init);
1725module_exit(acm_exit);
1726
Alan Cox6e47e062009-06-11 12:37:06 +01001727MODULE_AUTHOR(DRIVER_AUTHOR);
1728MODULE_DESCRIPTION(DRIVER_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001729MODULE_LICENSE("GPL");
Scott James Remnante766aeb2009-04-06 17:33:18 +01001730MODULE_ALIAS_CHARDEV_MAJOR(ACM_TTY_MAJOR);