blob: 0a69c0977e3f3f372f337b996732e341d07637e3 [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>
5 * Copyright (c) 1999 Pavel Machek <pavel@suse.cz>
6 * 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 *
15 * ChangeLog:
16 * v0.9 - thorough cleaning, URBification, almost a rewrite
17 * v0.10 - some more cleanups
18 * v0.11 - fixed flow control, read error doesn't stop reads
19 * v0.12 - added TIOCM ioctls, added break handling, made struct acm kmalloced
20 * v0.13 - added termios, added hangup
21 * v0.14 - sized down struct acm
22 * v0.15 - fixed flow control again - characters could be lost
23 * v0.16 - added code for modems with swapped data and control interfaces
24 * v0.17 - added new style probing
25 * v0.18 - fixed new style probing for devices with more configurations
26 * v0.19 - fixed CLOCAL handling (thanks to Richard Shih-Ping Chan)
27 * v0.20 - switched to probing on interface (rather than device) class
28 * v0.21 - revert to probing on device for devices with multiple configs
29 * v0.22 - probe only the control interface. if usbcore doesn't choose the
30 * config we want, sysadmin changes bConfigurationValue in sysfs.
31 * v0.23 - use softirq for rx processing, as needed by tty layer
32 * v0.24 - change probe method to evaluate CDC union descriptor
David Kubicek61a87ad2005-11-01 18:51:34 +010033 * v0.25 - downstream tasks paralelized to maximize throughput
David Engrafe4cf3aa2008-03-20 10:01:34 +010034 * v0.26 - multiple write urbs, writesize increased
Linus Torvalds1da177e2005-04-16 15:20:36 -070035 */
36
37/*
38 * This program is free software; you can redistribute it and/or modify
39 * it under the terms of the GNU General Public License as published by
40 * the Free Software Foundation; either version 2 of the License, or
41 * (at your option) any later version.
42 *
43 * This program is distributed in the hope that it will be useful,
44 * but WITHOUT ANY WARRANTY; without even the implied warranty of
45 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
46 * GNU General Public License for more details.
47 *
48 * You should have received a copy of the GNU General Public License
49 * along with this program; if not, write to the Free Software
50 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
51 */
52
53#undef DEBUG
David Brownelle5fbab52008-08-06 18:46:10 -070054#undef VERBOSE_DEBUG
Linus Torvalds1da177e2005-04-16 15:20:36 -070055
56#include <linux/kernel.h>
57#include <linux/errno.h>
58#include <linux/init.h>
59#include <linux/slab.h>
60#include <linux/tty.h>
61#include <linux/tty_driver.h>
62#include <linux/tty_flip.h>
63#include <linux/module.h>
Arjan van de Ven4186ecf2006-01-11 15:55:29 +010064#include <linux/mutex.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070065#include <asm/uaccess.h>
66#include <linux/usb.h>
David Brownella8c28f22006-06-13 09:57:47 -070067#include <linux/usb/cdc.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070068#include <asm/byteorder.h>
69#include <asm/unaligned.h>
David Kubicek61a87ad2005-11-01 18:51:34 +010070#include <linux/list.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070071
72#include "cdc-acm.h"
73
David Brownelle5fbab52008-08-06 18:46:10 -070074
75#define ACM_CLOSE_TIMEOUT 15 /* seconds to let writes drain */
76
Linus Torvalds1da177e2005-04-16 15:20:36 -070077/*
78 * Version Information
79 */
David Engrafe4cf3aa2008-03-20 10:01:34 +010080#define DRIVER_VERSION "v0.26"
David Kubicek61a87ad2005-11-01 18:51:34 +010081#define DRIVER_AUTHOR "Armin Fuerst, Pavel Machek, Johannes Erdfelt, Vojtech Pavlik, David Kubicek"
Linus Torvalds1da177e2005-04-16 15:20:36 -070082#define DRIVER_DESC "USB Abstract Control Model driver for USB modems and ISDN adapters"
83
84static struct usb_driver acm_driver;
85static struct tty_driver *acm_tty_driver;
86static struct acm *acm_table[ACM_TTY_MINORS];
87
Arjan van de Ven4186ecf2006-01-11 15:55:29 +010088static DEFINE_MUTEX(open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -070089
90#define ACM_READY(acm) (acm && acm->dev && acm->used)
91
David Brownelle5fbab52008-08-06 18:46:10 -070092#ifdef VERBOSE_DEBUG
93#define verbose 1
94#else
95#define verbose 0
96#endif
97
Linus Torvalds1da177e2005-04-16 15:20:36 -070098/*
99 * Functions for ACM control messages.
100 */
101
102static int acm_ctrl_msg(struct acm *acm, int request, int value, void *buf, int len)
103{
104 int retval = usb_control_msg(acm->dev, usb_sndctrlpipe(acm->dev, 0),
105 request, USB_RT_ACM, value,
106 acm->control->altsetting[0].desc.bInterfaceNumber,
107 buf, len, 5000);
108 dbg("acm_control_msg: rq: 0x%02x val: %#x len: %#x result: %d", request, value, len, retval);
109 return retval < 0 ? retval : 0;
110}
111
112/* devices aren't required to support these requests.
113 * the cdc acm descriptor tells whether they do...
114 */
115#define acm_set_control(acm, control) \
116 acm_ctrl_msg(acm, USB_CDC_REQ_SET_CONTROL_LINE_STATE, control, NULL, 0)
117#define acm_set_line(acm, line) \
118 acm_ctrl_msg(acm, USB_CDC_REQ_SET_LINE_CODING, 0, line, sizeof *(line))
119#define acm_send_break(acm, ms) \
120 acm_ctrl_msg(acm, USB_CDC_REQ_SEND_BREAK, ms, NULL, 0)
121
122/*
Oliver Neukum884b6002005-04-21 21:28:02 +0200123 * Write buffer management.
124 * All of these assume proper locks taken by the caller.
125 */
126
127static int acm_wb_alloc(struct acm *acm)
128{
129 int i, wbn;
130 struct acm_wb *wb;
131
David Engrafe4cf3aa2008-03-20 10:01:34 +0100132 wbn = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200133 i = 0;
134 for (;;) {
135 wb = &acm->wb[wbn];
136 if (!wb->use) {
137 wb->use = 1;
138 return wbn;
139 }
Oliver Neukum86478942006-05-13 22:50:47 +0200140 wbn = (wbn + 1) % ACM_NW;
141 if (++i >= ACM_NW)
Oliver Neukum884b6002005-04-21 21:28:02 +0200142 return -1;
143 }
144}
145
Oliver Neukum884b6002005-04-21 21:28:02 +0200146static int acm_wb_is_avail(struct acm *acm)
147{
148 int i, n;
David Brownelle5fbab52008-08-06 18:46:10 -0700149 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200150
Oliver Neukum86478942006-05-13 22:50:47 +0200151 n = ACM_NW;
David Brownelle5fbab52008-08-06 18:46:10 -0700152 spin_lock_irqsave(&acm->write_lock, flags);
Oliver Neukum86478942006-05-13 22:50:47 +0200153 for (i = 0; i < ACM_NW; i++) {
154 n -= acm->wb[i].use;
Oliver Neukum884b6002005-04-21 21:28:02 +0200155 }
David Brownelle5fbab52008-08-06 18:46:10 -0700156 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200157 return n;
158}
159
Oliver Neukum884b6002005-04-21 21:28:02 +0200160/*
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800161 * Finish write. Caller must hold acm->write_lock
Oliver Neukum884b6002005-04-21 21:28:02 +0200162 */
David Engrafe4cf3aa2008-03-20 10:01:34 +0100163static void acm_write_done(struct acm *acm, struct acm_wb *wb)
Oliver Neukum884b6002005-04-21 21:28:02 +0200164{
David Engrafe4cf3aa2008-03-20 10:01:34 +0100165 wb->use = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200166 acm->transmitting--;
Oliver Neukum884b6002005-04-21 21:28:02 +0200167}
168
169/*
170 * Poke write.
Oliver Neukum11ea8592008-06-20 11:25:57 +0200171 *
172 * the caller is responsible for locking
Oliver Neukum884b6002005-04-21 21:28:02 +0200173 */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200174
175static int acm_start_wb(struct acm *acm, struct acm_wb *wb)
176{
177 int rc;
178
179 acm->transmitting++;
180
181 wb->urb->transfer_buffer = wb->buf;
182 wb->urb->transfer_dma = wb->dmah;
183 wb->urb->transfer_buffer_length = wb->len;
184 wb->urb->dev = acm->dev;
185
186 if ((rc = usb_submit_urb(wb->urb, GFP_ATOMIC)) < 0) {
187 dbg("usb_submit_urb(write bulk) failed: %d", rc);
188 acm_write_done(acm, wb);
189 }
190 return rc;
191}
192
David Engrafe4cf3aa2008-03-20 10:01:34 +0100193static int acm_write_start(struct acm *acm, int wbn)
Oliver Neukum884b6002005-04-21 21:28:02 +0200194{
195 unsigned long flags;
David Brownell934da462008-08-06 18:44:12 -0700196 struct acm_wb *wb = &acm->wb[wbn];
Oliver Neukum884b6002005-04-21 21:28:02 +0200197 int rc;
198
199 spin_lock_irqsave(&acm->write_lock, flags);
200 if (!acm->dev) {
David Brownell934da462008-08-06 18:44:12 -0700201 wb->use = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200202 spin_unlock_irqrestore(&acm->write_lock, flags);
203 return -ENODEV;
204 }
205
Oliver Neukum11ea8592008-06-20 11:25:57 +0200206 dbg("%s susp_count: %d", __func__, acm->susp_count);
207 if (acm->susp_count) {
Oliver Neukum11ea8592008-06-20 11:25:57 +0200208 acm->delayed_wb = wb;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200209 schedule_work(&acm->waker);
210 spin_unlock_irqrestore(&acm->write_lock, flags);
211 return 0; /* A white lie */
212 }
213 usb_mark_last_busy(acm->dev);
214
Oliver Neukum11ea8592008-06-20 11:25:57 +0200215 rc = acm_start_wb(acm, wb);
Oliver Neukum884b6002005-04-21 21:28:02 +0200216 spin_unlock_irqrestore(&acm->write_lock, flags);
217
Oliver Neukum884b6002005-04-21 21:28:02 +0200218 return rc;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200219
Oliver Neukum884b6002005-04-21 21:28:02 +0200220}
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100221/*
222 * attributes exported through sysfs
223 */
224static ssize_t show_caps
225(struct device *dev, struct device_attribute *attr, char *buf)
226{
227 struct usb_interface *intf = to_usb_interface(dev);
228 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum884b6002005-04-21 21:28:02 +0200229
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100230 return sprintf(buf, "%d", acm->ctrl_caps);
231}
232static DEVICE_ATTR(bmCapabilities, S_IRUGO, show_caps, NULL);
233
234static ssize_t show_country_codes
235(struct device *dev, struct device_attribute *attr, char *buf)
236{
237 struct usb_interface *intf = to_usb_interface(dev);
238 struct acm *acm = usb_get_intfdata(intf);
239
240 memcpy(buf, acm->country_codes, acm->country_code_size);
241 return acm->country_code_size;
242}
243
244static DEVICE_ATTR(wCountryCodes, S_IRUGO, show_country_codes, NULL);
245
246static ssize_t show_country_rel_date
247(struct device *dev, struct device_attribute *attr, char *buf)
248{
249 struct usb_interface *intf = to_usb_interface(dev);
250 struct acm *acm = usb_get_intfdata(intf);
251
252 return sprintf(buf, "%d", acm->country_rel_date);
253}
254
255static DEVICE_ATTR(iCountryCodeRelDate, S_IRUGO, show_country_rel_date, NULL);
Oliver Neukum884b6002005-04-21 21:28:02 +0200256/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 * Interrupt handlers for various ACM device responses
258 */
259
260/* control interface reports status changes with "interrupt" transfers */
David Howells7d12e782006-10-05 14:55:46 +0100261static void acm_ctrl_irq(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700262{
263 struct acm *acm = urb->context;
264 struct usb_cdc_notification *dr = urb->transfer_buffer;
265 unsigned char *data;
266 int newctrl;
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700267 int retval;
268 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700269
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700270 switch (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700271 case 0:
272 /* success */
273 break;
274 case -ECONNRESET:
275 case -ENOENT:
276 case -ESHUTDOWN:
277 /* this urb is terminated, clean up */
Harvey Harrison441b62c2008-03-03 16:08:34 -0800278 dbg("%s - urb shutting down with status: %d", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700279 return;
280 default:
Harvey Harrison441b62c2008-03-03 16:08:34 -0800281 dbg("%s - nonzero urb status received: %d", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 goto exit;
283 }
284
285 if (!ACM_READY(acm))
286 goto exit;
287
288 data = (unsigned char *)(dr + 1);
289 switch (dr->bNotificationType) {
290
291 case USB_CDC_NOTIFY_NETWORK_CONNECTION:
292
293 dbg("%s network", dr->wValue ? "connected to" : "disconnected from");
294 break;
295
296 case USB_CDC_NOTIFY_SERIAL_STATE:
297
Harvey Harrisona5abdea2008-04-29 01:03:40 -0700298 newctrl = get_unaligned_le16(data);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299
300 if (acm->tty && !acm->clocal && (acm->ctrlin & ~newctrl & ACM_CTRL_DCD)) {
301 dbg("calling hangup");
302 tty_hangup(acm->tty);
303 }
304
305 acm->ctrlin = newctrl;
306
307 dbg("input control lines: dcd%c dsr%c break%c ring%c framing%c parity%c overrun%c",
308 acm->ctrlin & ACM_CTRL_DCD ? '+' : '-', acm->ctrlin & ACM_CTRL_DSR ? '+' : '-',
309 acm->ctrlin & ACM_CTRL_BRK ? '+' : '-', acm->ctrlin & ACM_CTRL_RI ? '+' : '-',
310 acm->ctrlin & ACM_CTRL_FRAMING ? '+' : '-', acm->ctrlin & ACM_CTRL_PARITY ? '+' : '-',
311 acm->ctrlin & ACM_CTRL_OVERRUN ? '+' : '-');
312
313 break;
314
315 default:
316 dbg("unknown notification %d received: index %d len %d data0 %d data1 %d",
317 dr->bNotificationType, dr->wIndex,
318 dr->wLength, data[0], data[1]);
319 break;
320 }
321exit:
Oliver Neukum11ea8592008-06-20 11:25:57 +0200322 usb_mark_last_busy(acm->dev);
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700323 retval = usb_submit_urb (urb, GFP_ATOMIC);
324 if (retval)
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700325 dev_err(&urb->dev->dev, "%s - usb_submit_urb failed with "
326 "result %d", __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
337 dbg("Entering acm_read_bulk with status %d", status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338
Oliver Neukum11ea8592008-06-20 11:25:57 +0200339 if (!ACM_READY(acm)) {
340 dev_dbg(&acm->data->dev, "Aborting, acm not ready");
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)
Joe Perches898eb712007-10-18 03:06:30 -0700346 dev_dbg(&acm->data->dev, "bulk rx status %d\n", status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700347
David Kubicek61a87ad2005-11-01 18:51:34 +0100348 buf = rcv->buffer;
349 buf->size = urb->actual_length;
350
Oliver Neukum86478942006-05-13 22:50:47 +0200351 if (likely(status == 0)) {
352 spin_lock(&acm->read_lock);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200353 acm->processing++;
Oliver Neukum86478942006-05-13 22:50:47 +0200354 list_add_tail(&rcv->list, &acm->spare_read_urbs);
355 list_add_tail(&buf->list, &acm->filled_read_bufs);
356 spin_unlock(&acm->read_lock);
357 } else {
358 /* we drop the buffer due to an error */
359 spin_lock(&acm->read_lock);
360 list_add_tail(&rcv->list, &acm->spare_read_urbs);
361 list_add(&buf->list, &acm->spare_read_bufs);
362 spin_unlock(&acm->read_lock);
363 /* nevertheless the tasklet must be kicked unconditionally
364 so the queue cannot dry up */
365 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200366 if (likely(!acm->susp_count))
367 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700368}
369
370static void acm_rx_tasklet(unsigned long _acm)
371{
372 struct acm *acm = (void *)_acm;
David Kubicek61a87ad2005-11-01 18:51:34 +0100373 struct acm_rb *buf;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374 struct tty_struct *tty = acm->tty;
David Kubicek61a87ad2005-11-01 18:51:34 +0100375 struct acm_ru *rcv;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200376 unsigned long flags;
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100377 unsigned char throttled;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200378
Linus Torvalds1da177e2005-04-16 15:20:36 -0700379 dbg("Entering acm_rx_tasklet");
380
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100381 if (!ACM_READY(acm))
Oliver Neukum11ea8592008-06-20 11:25:57 +0200382 {
383 dbg("acm_rx_tasklet: ACM not ready");
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);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100390 if (throttled)
Oliver Neukum11ea8592008-06-20 11:25:57 +0200391 {
392 dbg("acm_rx_tasklet: throttled");
David Kubicek61a87ad2005-11-01 18:51:34 +0100393 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200394 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100395
396next_buffer:
Jarek Poplawski762f0072006-10-06 07:23:11 +0200397 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100398 if (list_empty(&acm->filled_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200399 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100400 goto urbs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700401 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100402 buf = list_entry(acm->filled_read_bufs.next,
403 struct acm_rb, list);
404 list_del(&buf->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200405 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100406
Oliver Neukum3dd2ae82006-06-23 09:14:17 +0200407 dbg("acm_rx_tasklet: procesing buf 0x%p, size = %d", buf, buf->size);
David Kubicek61a87ad2005-11-01 18:51:34 +0100408
Alan Cox33f0f882006-01-09 20:54:13 -0800409 tty_buffer_request_room(tty, buf->size);
Oliver Neukum834dbca2007-03-06 10:47:04 +0100410 spin_lock_irqsave(&acm->throttle_lock, flags);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100411 throttled = acm->throttle;
Oliver Neukum834dbca2007-03-06 10:47:04 +0100412 spin_unlock_irqrestore(&acm->throttle_lock, flags);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100413 if (!throttled)
Alan Cox33f0f882006-01-09 20:54:13 -0800414 tty_insert_flip_string(tty, buf->base, buf->size);
David Kubicek61a87ad2005-11-01 18:51:34 +0100415 tty_flip_buffer_push(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100417 if (throttled) {
418 dbg("Throttling noticed");
Jarek Poplawski762f0072006-10-06 07:23:11 +0200419 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100420 list_add(&buf->list, &acm->filled_read_bufs);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200421 spin_unlock_irqrestore(&acm->read_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422 return;
423 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700424
Jarek Poplawski762f0072006-10-06 07:23:11 +0200425 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100426 list_add(&buf->list, &acm->spare_read_bufs);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200427 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100428 goto next_buffer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700429
David Kubicek61a87ad2005-11-01 18:51:34 +0100430urbs:
431 while (!list_empty(&acm->spare_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200432 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100433 if (list_empty(&acm->spare_read_urbs)) {
Oliver Neukum11ea8592008-06-20 11:25:57 +0200434 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200435 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100436 return;
437 }
438 rcv = list_entry(acm->spare_read_urbs.next,
439 struct acm_ru, list);
440 list_del(&rcv->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200441 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100442
443 buf = list_entry(acm->spare_read_bufs.next,
444 struct acm_rb, list);
445 list_del(&buf->list);
446
447 rcv->buffer = buf;
448
449 usb_fill_bulk_urb(rcv->urb, acm->dev,
450 acm->rx_endpoint,
451 buf->base,
452 acm->readsize,
453 acm_read_bulk, rcv);
454 rcv->urb->transfer_dma = buf->dma;
455 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
456
David Kubicek61a87ad2005-11-01 18:51:34 +0100457 /* This shouldn't kill the driver as unsuccessful URBs are returned to the
458 free-urbs-pool and resubmited ASAP */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200459 spin_lock_irqsave(&acm->read_lock, flags);
460 if (acm->susp_count || usb_submit_urb(rcv->urb, GFP_ATOMIC) < 0) {
David Kubicek61a87ad2005-11-01 18:51:34 +0100461 list_add(&buf->list, &acm->spare_read_bufs);
David Kubicek61a87ad2005-11-01 18:51:34 +0100462 list_add(&rcv->list, &acm->spare_read_urbs);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200463 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200464 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100465 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200466 } else {
467 spin_unlock_irqrestore(&acm->read_lock, flags);
468 dbg("acm_rx_tasklet: sending urb 0x%p, rcv 0x%p, buf 0x%p", rcv->urb, rcv, buf);
David Kubicek61a87ad2005-11-01 18:51:34 +0100469 }
470 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200471 spin_lock_irqsave(&acm->read_lock, flags);
472 acm->processing = 0;
473 spin_unlock_irqrestore(&acm->read_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474}
475
476/* data interface wrote those outgoing bytes */
David Howells7d12e782006-10-05 14:55:46 +0100477static void acm_write_bulk(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700478{
Ming Leicdc97792008-02-24 18:41:47 +0800479 struct acm_wb *wb = urb->context;
David Brownelle5fbab52008-08-06 18:46:10 -0700480 struct acm *acm = wb->instance;
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800481 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200482
David Brownelle5fbab52008-08-06 18:46:10 -0700483 if (verbose || urb->status
484 || (urb->actual_length != urb->transfer_buffer_length))
485 dev_dbg(&acm->data->dev, "tx %d/%d bytes -- > %d\n",
486 urb->actual_length,
487 urb->transfer_buffer_length,
488 urb->status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800490 spin_lock_irqsave(&acm->write_lock, flags);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100491 acm_write_done(acm, wb);
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800492 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200493 if (ACM_READY(acm))
494 schedule_work(&acm->work);
David Brownelle5fbab52008-08-06 18:46:10 -0700495 else
496 wake_up_interruptible(&acm->drain_wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497}
498
David Howellsc4028952006-11-22 14:57:56 +0000499static void acm_softint(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700500{
David Howellsc4028952006-11-22 14:57:56 +0000501 struct acm *acm = container_of(work, struct acm, work);
David Brownelle5fbab52008-08-06 18:46:10 -0700502
503 dev_vdbg(&acm->data->dev, "tx work\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504 if (!ACM_READY(acm))
505 return;
506 tty_wakeup(acm->tty);
507}
508
Oliver Neukum11ea8592008-06-20 11:25:57 +0200509static void acm_waker(struct work_struct *waker)
510{
511 struct acm *acm = container_of(waker, struct acm, waker);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200512 int rv;
513
514 rv = usb_autopm_get_interface(acm->control);
515 if (rv < 0) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700516 dev_err(&acm->dev->dev, "Autopm failure in %s\n", __func__);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200517 return;
518 }
519 if (acm->delayed_wb) {
520 acm_start_wb(acm, acm->delayed_wb);
521 acm->delayed_wb = NULL;
522 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200523 usb_autopm_put_interface(acm->control);
524}
525
Linus Torvalds1da177e2005-04-16 15:20:36 -0700526/*
527 * TTY handlers
528 */
529
530static int acm_tty_open(struct tty_struct *tty, struct file *filp)
531{
532 struct acm *acm;
533 int rv = -EINVAL;
David Kubicek61a87ad2005-11-01 18:51:34 +0100534 int i;
Oliver Neukum3dd2ae82006-06-23 09:14:17 +0200535 dbg("Entering acm_tty_open.");
Arjan van de Ven4186ecf2006-01-11 15:55:29 +0100536
537 mutex_lock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538
539 acm = acm_table[tty->index];
540 if (!acm || !acm->dev)
541 goto err_out;
542 else
543 rv = 0;
544
David Engraf28d1dfa2008-03-20 10:53:52 +0100545 set_bit(TTY_NO_WRITE_SPLIT, &tty->flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546 tty->driver_data = acm;
547 acm->tty = tty;
548
Oliver Neukum94409cc2008-02-11 15:22:29 +0100549 if (usb_autopm_get_interface(acm->control) < 0)
550 goto early_bail;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200551 else
552 acm->control->needs_remote_wakeup = 1;
Oliver Neukum1365baf72007-10-12 17:24:28 +0200553
554 mutex_lock(&acm->mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700555 if (acm->used++) {
Oliver Neukum1365baf72007-10-12 17:24:28 +0200556 usb_autopm_put_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700557 goto done;
558 }
559
Oliver Neukum1365baf72007-10-12 17:24:28 +0200560
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561 acm->ctrlurb->dev = acm->dev;
562 if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) {
563 dbg("usb_submit_urb(ctrl irq) failed");
564 goto bail_out;
565 }
566
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100567 if (0 > acm_set_control(acm, acm->ctrlout = ACM_CTRL_DTR | ACM_CTRL_RTS) &&
568 (acm->ctrl_caps & USB_CDC_CAP_LINE))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700569 goto full_bailout;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200570 usb_autopm_put_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571
David Kubicek61a87ad2005-11-01 18:51:34 +0100572 INIT_LIST_HEAD(&acm->spare_read_urbs);
573 INIT_LIST_HEAD(&acm->spare_read_bufs);
574 INIT_LIST_HEAD(&acm->filled_read_bufs);
Oliver Neukum86478942006-05-13 22:50:47 +0200575 for (i = 0; i < acm->rx_buflimit; i++) {
David Kubicek61a87ad2005-11-01 18:51:34 +0100576 list_add(&(acm->ru[i].list), &acm->spare_read_urbs);
577 }
Oliver Neukum86478942006-05-13 22:50:47 +0200578 for (i = 0; i < acm->rx_buflimit; i++) {
David Kubicek61a87ad2005-11-01 18:51:34 +0100579 list_add(&(acm->rb[i].list), &acm->spare_read_bufs);
580 }
581
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100582 acm->throttle = 0;
583
David Kubicek61a87ad2005-11-01 18:51:34 +0100584 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700585
586done:
Oliver Neukum1365baf72007-10-12 17:24:28 +0200587 mutex_unlock(&acm->mutex);
Alexey Dobriyan74573ee2008-08-20 16:56:04 -0700588err_out:
Oliver Neukum94409cc2008-02-11 15:22:29 +0100589 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700590 return rv;
591
592full_bailout:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593 usb_kill_urb(acm->ctrlurb);
594bail_out:
Oliver Neukum1365baf72007-10-12 17:24:28 +0200595 usb_autopm_put_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700596 acm->used--;
Oliver Neukum1365baf72007-10-12 17:24:28 +0200597 mutex_unlock(&acm->mutex);
Oliver Neukum94409cc2008-02-11 15:22:29 +0100598early_bail:
599 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700600 return -EIO;
601}
602
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700603static void acm_tty_unregister(struct acm *acm)
604{
Oliver Neukum86478942006-05-13 22:50:47 +0200605 int i,nr;
David Kubicek61a87ad2005-11-01 18:51:34 +0100606
Oliver Neukum86478942006-05-13 22:50:47 +0200607 nr = acm->rx_buflimit;
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700608 tty_unregister_device(acm_tty_driver, acm->minor);
609 usb_put_intf(acm->control);
610 acm_table[acm->minor] = NULL;
611 usb_free_urb(acm->ctrlurb);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100612 for (i = 0; i < ACM_NW; i++)
613 usb_free_urb(acm->wb[i].urb);
Oliver Neukum86478942006-05-13 22:50:47 +0200614 for (i = 0; i < nr; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100615 usb_free_urb(acm->ru[i].urb);
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100616 kfree(acm->country_codes);
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700617 kfree(acm);
618}
619
David Brownelle5fbab52008-08-06 18:46:10 -0700620static int acm_tty_chars_in_buffer(struct tty_struct *tty);
621
Linus Torvalds1da177e2005-04-16 15:20:36 -0700622static void acm_tty_close(struct tty_struct *tty, struct file *filp)
623{
624 struct acm *acm = tty->driver_data;
Oliver Neukum86478942006-05-13 22:50:47 +0200625 int i,nr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700626
627 if (!acm || !acm->used)
628 return;
629
Oliver Neukum86478942006-05-13 22:50:47 +0200630 nr = acm->rx_buflimit;
Arjan van de Ven4186ecf2006-01-11 15:55:29 +0100631 mutex_lock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700632 if (!--acm->used) {
633 if (acm->dev) {
Oliver Neukum11ea8592008-06-20 11:25:57 +0200634 usb_autopm_get_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700635 acm_set_control(acm, acm->ctrlout = 0);
David Brownelle5fbab52008-08-06 18:46:10 -0700636
637 /* try letting the last writes drain naturally */
638 wait_event_interruptible_timeout(acm->drain_wait,
639 (ACM_NW == acm_wb_is_avail(acm))
640 || !acm->dev,
641 ACM_CLOSE_TIMEOUT * HZ);
642
Linus Torvalds1da177e2005-04-16 15:20:36 -0700643 usb_kill_urb(acm->ctrlurb);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100644 for (i = 0; i < ACM_NW; i++)
645 usb_kill_urb(acm->wb[i].urb);
Oliver Neukum86478942006-05-13 22:50:47 +0200646 for (i = 0; i < nr; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100647 usb_kill_urb(acm->ru[i].urb);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200648 acm->control->needs_remote_wakeup = 0;
Oliver Neukum1365baf72007-10-12 17:24:28 +0200649 usb_autopm_put_interface(acm->control);
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700650 } else
651 acm_tty_unregister(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700652 }
Arjan van de Ven4186ecf2006-01-11 15:55:29 +0100653 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700654}
655
656static int acm_tty_write(struct tty_struct *tty, const unsigned char *buf, int count)
657{
658 struct acm *acm = tty->driver_data;
659 int stat;
Oliver Neukum884b6002005-04-21 21:28:02 +0200660 unsigned long flags;
661 int wbn;
662 struct acm_wb *wb;
663
Oliver Neukum3dd2ae82006-06-23 09:14:17 +0200664 dbg("Entering acm_tty_write to write %d bytes,", count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700665
666 if (!ACM_READY(acm))
667 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700668 if (!count)
669 return 0;
670
Oliver Neukum884b6002005-04-21 21:28:02 +0200671 spin_lock_irqsave(&acm->write_lock, flags);
672 if ((wbn = acm_wb_alloc(acm)) < 0) {
673 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200674 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700675 }
Oliver Neukum884b6002005-04-21 21:28:02 +0200676 wb = &acm->wb[wbn];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700677
Oliver Neukum884b6002005-04-21 21:28:02 +0200678 count = (count > acm->writesize) ? acm->writesize : count;
679 dbg("Get %d bytes...", count);
680 memcpy(wb->buf, buf, count);
681 wb->len = count;
682 spin_unlock_irqrestore(&acm->write_lock, flags);
683
David Engrafe4cf3aa2008-03-20 10:01:34 +0100684 if ((stat = acm_write_start(acm, wbn)) < 0)
Oliver Neukum884b6002005-04-21 21:28:02 +0200685 return stat;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700686 return count;
687}
688
689static int acm_tty_write_room(struct tty_struct *tty)
690{
691 struct acm *acm = tty->driver_data;
692 if (!ACM_READY(acm))
693 return -EINVAL;
Oliver Neukum884b6002005-04-21 21:28:02 +0200694 /*
695 * Do not let the line discipline to know that we have a reserve,
696 * or it might get too enthusiastic.
697 */
David Brownell934da462008-08-06 18:44:12 -0700698 return acm_wb_is_avail(acm) ? acm->writesize : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700699}
700
701static int acm_tty_chars_in_buffer(struct tty_struct *tty)
702{
703 struct acm *acm = tty->driver_data;
704 if (!ACM_READY(acm))
705 return -EINVAL;
Oliver Neukum884b6002005-04-21 21:28:02 +0200706 /*
707 * This is inaccurate (overcounts), but it works.
708 */
Oliver Neukum86478942006-05-13 22:50:47 +0200709 return (ACM_NW - acm_wb_is_avail(acm)) * acm->writesize;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700710}
711
712static void acm_tty_throttle(struct tty_struct *tty)
713{
714 struct acm *acm = tty->driver_data;
715 if (!ACM_READY(acm))
716 return;
717 spin_lock_bh(&acm->throttle_lock);
718 acm->throttle = 1;
719 spin_unlock_bh(&acm->throttle_lock);
720}
721
722static void acm_tty_unthrottle(struct tty_struct *tty)
723{
724 struct acm *acm = tty->driver_data;
725 if (!ACM_READY(acm))
726 return;
727 spin_lock_bh(&acm->throttle_lock);
728 acm->throttle = 0;
729 spin_unlock_bh(&acm->throttle_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +0100730 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700731}
732
Alan Cox9e98966c2008-07-22 11:18:03 +0100733static int acm_tty_break_ctl(struct tty_struct *tty, int state)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700734{
735 struct acm *acm = tty->driver_data;
Alan Cox9e98966c2008-07-22 11:18:03 +0100736 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700737 if (!ACM_READY(acm))
Alan Cox9e98966c2008-07-22 11:18:03 +0100738 return -EINVAL;
739 retval = acm_send_break(acm, state ? 0xffff : 0);
740 if (retval < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700741 dbg("send break failed");
Alan Cox9e98966c2008-07-22 11:18:03 +0100742 return retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700743}
744
745static int acm_tty_tiocmget(struct tty_struct *tty, struct file *file)
746{
747 struct acm *acm = tty->driver_data;
748
749 if (!ACM_READY(acm))
750 return -EINVAL;
751
752 return (acm->ctrlout & ACM_CTRL_DTR ? TIOCM_DTR : 0) |
753 (acm->ctrlout & ACM_CTRL_RTS ? TIOCM_RTS : 0) |
754 (acm->ctrlin & ACM_CTRL_DSR ? TIOCM_DSR : 0) |
755 (acm->ctrlin & ACM_CTRL_RI ? TIOCM_RI : 0) |
756 (acm->ctrlin & ACM_CTRL_DCD ? TIOCM_CD : 0) |
757 TIOCM_CTS;
758}
759
760static int acm_tty_tiocmset(struct tty_struct *tty, struct file *file,
761 unsigned int set, unsigned int clear)
762{
763 struct acm *acm = tty->driver_data;
764 unsigned int newctrl;
765
766 if (!ACM_READY(acm))
767 return -EINVAL;
768
769 newctrl = acm->ctrlout;
770 set = (set & TIOCM_DTR ? ACM_CTRL_DTR : 0) | (set & TIOCM_RTS ? ACM_CTRL_RTS : 0);
771 clear = (clear & TIOCM_DTR ? ACM_CTRL_DTR : 0) | (clear & TIOCM_RTS ? ACM_CTRL_RTS : 0);
772
773 newctrl = (newctrl & ~clear) | set;
774
775 if (acm->ctrlout == newctrl)
776 return 0;
777 return acm_set_control(acm, acm->ctrlout = newctrl);
778}
779
780static int acm_tty_ioctl(struct tty_struct *tty, struct file *file, unsigned int cmd, unsigned long arg)
781{
782 struct acm *acm = tty->driver_data;
783
784 if (!ACM_READY(acm))
785 return -EINVAL;
786
787 return -ENOIOCTLCMD;
788}
789
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100790static const __u32 acm_tty_speed[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700791 0, 50, 75, 110, 134, 150, 200, 300, 600,
792 1200, 1800, 2400, 4800, 9600, 19200, 38400,
793 57600, 115200, 230400, 460800, 500000, 576000,
794 921600, 1000000, 1152000, 1500000, 2000000,
795 2500000, 3000000, 3500000, 4000000
796};
797
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100798static const __u8 acm_tty_size[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700799 5, 6, 7, 8
800};
801
Alan Cox606d0992006-12-08 02:38:45 -0800802static void acm_tty_set_termios(struct tty_struct *tty, struct ktermios *termios_old)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700803{
804 struct acm *acm = tty->driver_data;
Alan Cox606d0992006-12-08 02:38:45 -0800805 struct ktermios *termios = tty->termios;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700806 struct usb_cdc_line_coding newline;
807 int newctrl = acm->ctrlout;
808
809 if (!ACM_READY(acm))
810 return;
811
812 newline.dwDTERate = cpu_to_le32p(acm_tty_speed +
813 (termios->c_cflag & CBAUD & ~CBAUDEX) + (termios->c_cflag & CBAUDEX ? 15 : 0));
814 newline.bCharFormat = termios->c_cflag & CSTOPB ? 2 : 0;
815 newline.bParityType = termios->c_cflag & PARENB ?
816 (termios->c_cflag & PARODD ? 1 : 2) + (termios->c_cflag & CMSPAR ? 2 : 0) : 0;
817 newline.bDataBits = acm_tty_size[(termios->c_cflag & CSIZE) >> 4];
818
819 acm->clocal = ((termios->c_cflag & CLOCAL) != 0);
820
821 if (!newline.dwDTERate) {
822 newline.dwDTERate = acm->line.dwDTERate;
823 newctrl &= ~ACM_CTRL_DTR;
824 } else newctrl |= ACM_CTRL_DTR;
825
826 if (newctrl != acm->ctrlout)
827 acm_set_control(acm, acm->ctrlout = newctrl);
828
829 if (memcmp(&acm->line, &newline, sizeof newline)) {
830 memcpy(&acm->line, &newline, sizeof newline);
831 dbg("set line: %d %d %d %d", le32_to_cpu(newline.dwDTERate),
832 newline.bCharFormat, newline.bParityType,
833 newline.bDataBits);
834 acm_set_line(acm, &acm->line);
835 }
836}
837
838/*
839 * USB probe and disconnect routines.
840 */
841
Oliver Neukum830f4022008-06-25 14:17:16 +0200842/* Little helpers: write/read buffers free */
Oliver Neukum884b6002005-04-21 21:28:02 +0200843static void acm_write_buffers_free(struct acm *acm)
844{
845 int i;
846 struct acm_wb *wb;
Oliver Neukuma496c642008-10-21 10:39:04 +0200847 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
Oliver Neukum884b6002005-04-21 21:28:02 +0200848
Oliver Neukum86478942006-05-13 22:50:47 +0200849 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++) {
Oliver Neukuma496c642008-10-21 10:39:04 +0200850 usb_buffer_free(usb_dev, acm->writesize, wb->buf, wb->dmah);
Oliver Neukum884b6002005-04-21 21:28:02 +0200851 }
852}
853
Oliver Neukum830f4022008-06-25 14:17:16 +0200854static void acm_read_buffers_free(struct acm *acm)
855{
856 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
857 int i, n = acm->rx_buflimit;
858
859 for (i = 0; i < n; i++)
860 usb_buffer_free(usb_dev, acm->readsize, acm->rb[i].base, acm->rb[i].dma);
861}
862
Oliver Neukum884b6002005-04-21 21:28:02 +0200863/* Little helper: write buffers allocate */
864static int acm_write_buffers_alloc(struct acm *acm)
865{
866 int i;
867 struct acm_wb *wb;
868
Oliver Neukum86478942006-05-13 22:50:47 +0200869 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++) {
Oliver Neukum884b6002005-04-21 21:28:02 +0200870 wb->buf = usb_buffer_alloc(acm->dev, acm->writesize, GFP_KERNEL,
871 &wb->dmah);
872 if (!wb->buf) {
873 while (i != 0) {
874 --i;
875 --wb;
876 usb_buffer_free(acm->dev, acm->writesize,
877 wb->buf, wb->dmah);
878 }
879 return -ENOMEM;
880 }
881 }
882 return 0;
883}
884
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885static int acm_probe (struct usb_interface *intf,
886 const struct usb_device_id *id)
887{
888 struct usb_cdc_union_desc *union_header = NULL;
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100889 struct usb_cdc_country_functional_desc *cfd = NULL;
David Brownellc6dbf552008-04-13 14:00:44 -0700890 unsigned char *buffer = intf->altsetting->extra;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700891 int buflen = intf->altsetting->extralen;
892 struct usb_interface *control_interface;
893 struct usb_interface *data_interface;
894 struct usb_endpoint_descriptor *epctrl;
895 struct usb_endpoint_descriptor *epread;
896 struct usb_endpoint_descriptor *epwrite;
897 struct usb_device *usb_dev = interface_to_usbdev(intf);
898 struct acm *acm;
899 int minor;
900 int ctrlsize,readsize;
901 u8 *buf;
902 u8 ac_management_function = 0;
903 u8 call_management_function = 0;
904 int call_interface_num = -1;
905 int data_interface_num;
906 unsigned long quirks;
Oliver Neukum86478942006-05-13 22:50:47 +0200907 int num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +0100908 int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700909
Oliver Neukum86478942006-05-13 22:50:47 +0200910 /* normal quirks */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700911 quirks = (unsigned long)id->driver_info;
Oliver Neukum86478942006-05-13 22:50:47 +0200912 num_rx_buf = (quirks == SINGLE_RX_URB) ? 1 : ACM_NR;
913
914 /* handle quirks deadly to normal probing*/
Linus Torvalds1da177e2005-04-16 15:20:36 -0700915 if (quirks == NO_UNION_NORMAL) {
916 data_interface = usb_ifnum_to_if(usb_dev, 1);
917 control_interface = usb_ifnum_to_if(usb_dev, 0);
918 goto skip_normal_probe;
919 }
920
921 /* normal probing*/
922 if (!buffer) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700923 dev_err(&intf->dev, "Weird descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700924 return -EINVAL;
925 }
926
927 if (!buflen) {
928 if (intf->cur_altsetting->endpoint->extralen && intf->cur_altsetting->endpoint->extra) {
Joe Perches898eb712007-10-18 03:06:30 -0700929 dev_dbg(&intf->dev,"Seeking extra descriptors on endpoint\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700930 buflen = intf->cur_altsetting->endpoint->extralen;
931 buffer = intf->cur_altsetting->endpoint->extra;
932 } else {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700933 dev_err(&intf->dev,
934 "Zero length descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700935 return -EINVAL;
936 }
937 }
938
939 while (buflen > 0) {
940 if (buffer [1] != USB_DT_CS_INTERFACE) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700941 dev_err(&intf->dev, "skipping garbage\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700942 goto next_desc;
943 }
944
945 switch (buffer [2]) {
946 case USB_CDC_UNION_TYPE: /* we've found it */
947 if (union_header) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700948 dev_err(&intf->dev, "More than one "
949 "union descriptor, "
950 "skipping ...\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700951 goto next_desc;
952 }
953 union_header = (struct usb_cdc_union_desc *)
954 buffer;
955 break;
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100956 case USB_CDC_COUNTRY_TYPE: /* export through sysfs*/
957 cfd = (struct usb_cdc_country_functional_desc *)buffer;
958 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700959 case USB_CDC_HEADER_TYPE: /* maybe check version */
960 break; /* for now we ignore it */
961 case USB_CDC_ACM_TYPE:
962 ac_management_function = buffer[3];
963 break;
964 case USB_CDC_CALL_MANAGEMENT_TYPE:
965 call_management_function = buffer[3];
966 call_interface_num = buffer[4];
967 if ((call_management_function & 3) != 3)
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700968 dev_err(&intf->dev, "This device "
969 "cannot do calls on its own. "
970 "It is no modem.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700971 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700972 default:
David Brownellc6dbf552008-04-13 14:00:44 -0700973 /* there are LOTS more CDC descriptors that
974 * could legitimately be found here.
975 */
976 dev_dbg(&intf->dev, "Ignoring descriptor: "
977 "type %02x, length %d\n",
978 buffer[2], buffer[0]);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700979 break;
980 }
981next_desc:
982 buflen -= buffer[0];
983 buffer += buffer[0];
984 }
985
986 if (!union_header) {
987 if (call_interface_num > 0) {
Joe Perches898eb712007-10-18 03:06:30 -0700988 dev_dbg(&intf->dev,"No union descriptor, using call management descriptor\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700989 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = call_interface_num));
990 control_interface = intf;
991 } else {
Joe Perches898eb712007-10-18 03:06:30 -0700992 dev_dbg(&intf->dev,"No union descriptor, giving up\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700993 return -ENODEV;
994 }
995 } else {
996 control_interface = usb_ifnum_to_if(usb_dev, union_header->bMasterInterface0);
997 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = union_header->bSlaveInterface0));
998 if (!control_interface || !data_interface) {
Joe Perches898eb712007-10-18 03:06:30 -0700999 dev_dbg(&intf->dev,"no interfaces\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001000 return -ENODEV;
1001 }
1002 }
1003
1004 if (data_interface_num != call_interface_num)
Joe Perchesdc0d5c12007-12-17 11:40:18 -08001005 dev_dbg(&intf->dev,"Separate call control interface. That is not fully supported.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001006
1007skip_normal_probe:
1008
1009 /*workaround for switched interfaces */
1010 if (data_interface->cur_altsetting->desc.bInterfaceClass != CDC_DATA_INTERFACE_TYPE) {
1011 if (control_interface->cur_altsetting->desc.bInterfaceClass == CDC_DATA_INTERFACE_TYPE) {
1012 struct usb_interface *t;
Joe Perches898eb712007-10-18 03:06:30 -07001013 dev_dbg(&intf->dev,"Your device has switched interfaces.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014
1015 t = control_interface;
1016 control_interface = data_interface;
1017 data_interface = t;
1018 } else {
1019 return -EINVAL;
1020 }
1021 }
Alan Stern74da5d62007-08-02 13:29:10 -04001022
1023 /* Accept probe requests only for the control interface */
1024 if (intf != control_interface)
1025 return -ENODEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001026
1027 if (usb_interface_claimed(data_interface)) { /* valid in this context */
Joe Perches898eb712007-10-18 03:06:30 -07001028 dev_dbg(&intf->dev,"The data interface isn't available\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001029 return -EBUSY;
1030 }
1031
1032
1033 if (data_interface->cur_altsetting->desc.bNumEndpoints < 2)
1034 return -EINVAL;
1035
1036 epctrl = &control_interface->cur_altsetting->endpoint[0].desc;
1037 epread = &data_interface->cur_altsetting->endpoint[0].desc;
1038 epwrite = &data_interface->cur_altsetting->endpoint[1].desc;
1039
1040
1041 /* workaround for switched endpoints */
Luiz Fernando N. Capitulino45aea702006-10-26 13:02:48 -03001042 if (!usb_endpoint_dir_in(epread)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001043 /* descriptors are swapped */
1044 struct usb_endpoint_descriptor *t;
Joe Perches898eb712007-10-18 03:06:30 -07001045 dev_dbg(&intf->dev,"The data interface has switched endpoints\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001046
1047 t = epread;
1048 epread = epwrite;
1049 epwrite = t;
1050 }
1051 dbg("interfaces are valid");
1052 for (minor = 0; minor < ACM_TTY_MINORS && acm_table[minor]; minor++);
1053
1054 if (minor == ACM_TTY_MINORS) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -07001055 dev_err(&intf->dev, "no more free acm devices\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001056 return -ENODEV;
1057 }
1058
Oliver Neukum46f116e2005-10-24 22:42:35 +02001059 if (!(acm = kzalloc(sizeof(struct acm), GFP_KERNEL))) {
Joe Perches898eb712007-10-18 03:06:30 -07001060 dev_dbg(&intf->dev, "out of memory (acm kzalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001061 goto alloc_fail;
1062 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001063
1064 ctrlsize = le16_to_cpu(epctrl->wMaxPacketSize);
Oliver Neukum86478942006-05-13 22:50:47 +02001065 readsize = le16_to_cpu(epread->wMaxPacketSize)* ( quirks == SINGLE_RX_URB ? 1 : 2);
David Engrafe4cf3aa2008-03-20 10:01:34 +01001066 acm->writesize = le16_to_cpu(epwrite->wMaxPacketSize) * 20;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001067 acm->control = control_interface;
1068 acm->data = data_interface;
1069 acm->minor = minor;
1070 acm->dev = usb_dev;
1071 acm->ctrl_caps = ac_management_function;
1072 acm->ctrlsize = ctrlsize;
1073 acm->readsize = readsize;
Oliver Neukum86478942006-05-13 22:50:47 +02001074 acm->rx_buflimit = num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +01001075 acm->urb_task.func = acm_rx_tasklet;
1076 acm->urb_task.data = (unsigned long) acm;
David Howellsc4028952006-11-22 14:57:56 +00001077 INIT_WORK(&acm->work, acm_softint);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001078 INIT_WORK(&acm->waker, acm_waker);
David Brownelle5fbab52008-08-06 18:46:10 -07001079 init_waitqueue_head(&acm->drain_wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001080 spin_lock_init(&acm->throttle_lock);
Oliver Neukum884b6002005-04-21 21:28:02 +02001081 spin_lock_init(&acm->write_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +01001082 spin_lock_init(&acm->read_lock);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001083 mutex_init(&acm->mutex);
David Kubicek61a87ad2005-11-01 18:51:34 +01001084 acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001085
1086 buf = usb_buffer_alloc(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma);
1087 if (!buf) {
Joe Perches898eb712007-10-18 03:06:30 -07001088 dev_dbg(&intf->dev, "out of memory (ctrl buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001089 goto alloc_fail2;
1090 }
1091 acm->ctrl_buffer = buf;
1092
Oliver Neukum884b6002005-04-21 21:28:02 +02001093 if (acm_write_buffers_alloc(acm) < 0) {
Joe Perches898eb712007-10-18 03:06:30 -07001094 dev_dbg(&intf->dev, "out of memory (write buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001095 goto alloc_fail4;
1096 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001097
1098 acm->ctrlurb = usb_alloc_urb(0, GFP_KERNEL);
1099 if (!acm->ctrlurb) {
Joe Perches898eb712007-10-18 03:06:30 -07001100 dev_dbg(&intf->dev, "out of memory (ctrlurb kmalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001101 goto alloc_fail5;
1102 }
Oliver Neukum86478942006-05-13 22:50:47 +02001103 for (i = 0; i < num_rx_buf; i++) {
David Kubicek61a87ad2005-11-01 18:51:34 +01001104 struct acm_ru *rcv = &(acm->ru[i]);
1105
1106 if (!(rcv->urb = usb_alloc_urb(0, GFP_KERNEL))) {
Joe Perches898eb712007-10-18 03:06:30 -07001107 dev_dbg(&intf->dev, "out of memory (read urbs usb_alloc_urb)\n");
David Kubicek61a87ad2005-11-01 18:51:34 +01001108 goto alloc_fail7;
1109 }
1110
1111 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1112 rcv->instance = acm;
1113 }
Oliver Neukum86478942006-05-13 22:50:47 +02001114 for (i = 0; i < num_rx_buf; i++) {
David Brownell672c4e12008-08-06 18:41:12 -07001115 struct acm_rb *rb = &(acm->rb[i]);
David Kubicek61a87ad2005-11-01 18:51:34 +01001116
David Brownell672c4e12008-08-06 18:41:12 -07001117 rb->base = usb_buffer_alloc(acm->dev, readsize,
1118 GFP_KERNEL, &rb->dma);
1119 if (!rb->base) {
Joe Perches898eb712007-10-18 03:06:30 -07001120 dev_dbg(&intf->dev, "out of memory (read bufs usb_buffer_alloc)\n");
David Kubicek61a87ad2005-11-01 18:51:34 +01001121 goto alloc_fail7;
1122 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001123 }
David Engrafe4cf3aa2008-03-20 10:01:34 +01001124 for(i = 0; i < ACM_NW; i++)
1125 {
1126 struct acm_wb *snd = &(acm->wb[i]);
1127
1128 if (!(snd->urb = usb_alloc_urb(0, GFP_KERNEL))) {
1129 dev_dbg(&intf->dev, "out of memory (write urbs usb_alloc_urb)");
1130 goto alloc_fail7;
1131 }
1132
1133 usb_fill_bulk_urb(snd->urb, usb_dev, usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1134 NULL, acm->writesize, acm_write_bulk, snd);
1135 snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1136 snd->instance = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001137 }
1138
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001139 usb_set_intfdata (intf, acm);
1140
1141 i = device_create_file(&intf->dev, &dev_attr_bmCapabilities);
1142 if (i < 0)
1143 goto alloc_fail8;
1144
1145 if (cfd) { /* export the country data */
1146 acm->country_codes = kmalloc(cfd->bLength - 4, GFP_KERNEL);
1147 if (!acm->country_codes)
1148 goto skip_countries;
1149 acm->country_code_size = cfd->bLength - 4;
1150 memcpy(acm->country_codes, (u8 *)&cfd->wCountyCode0, cfd->bLength - 4);
1151 acm->country_rel_date = cfd->iCountryCodeRelDate;
1152
1153 i = device_create_file(&intf->dev, &dev_attr_wCountryCodes);
1154 if (i < 0) {
1155 kfree(acm->country_codes);
1156 goto skip_countries;
1157 }
1158
1159 i = device_create_file(&intf->dev, &dev_attr_iCountryCodeRelDate);
1160 if (i < 0) {
1161 kfree(acm->country_codes);
1162 goto skip_countries;
1163 }
1164 }
1165
1166skip_countries:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001167 usb_fill_int_urb(acm->ctrlurb, usb_dev, usb_rcvintpipe(usb_dev, epctrl->bEndpointAddress),
1168 acm->ctrl_buffer, ctrlsize, acm_ctrl_irq, acm, epctrl->bInterval);
1169 acm->ctrlurb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1170 acm->ctrlurb->transfer_dma = acm->ctrl_dma;
1171
Linus Torvalds1da177e2005-04-16 15:20:36 -07001172 dev_info(&intf->dev, "ttyACM%d: USB ACM device\n", minor);
1173
1174 acm_set_control(acm, acm->ctrlout);
1175
1176 acm->line.dwDTERate = cpu_to_le32(9600);
1177 acm->line.bDataBits = 8;
1178 acm_set_line(acm, &acm->line);
1179
1180 usb_driver_claim_interface(&acm_driver, data_interface, acm);
David Brownell672c4e12008-08-06 18:41:12 -07001181 usb_set_intfdata(data_interface, acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001182
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001183 usb_get_intf(control_interface);
1184 tty_register_device(acm_tty_driver, minor, &control_interface->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001185
1186 acm_table[minor] = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001187
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001188 return 0;
1189alloc_fail8:
David Engrafe4cf3aa2008-03-20 10:01:34 +01001190 for (i = 0; i < ACM_NW; i++)
1191 usb_free_urb(acm->wb[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001192alloc_fail7:
Oliver Neukum830f4022008-06-25 14:17:16 +02001193 acm_read_buffers_free(acm);
Oliver Neukum86478942006-05-13 22:50:47 +02001194 for (i = 0; i < num_rx_buf; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +01001195 usb_free_urb(acm->ru[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001196 usb_free_urb(acm->ctrlurb);
1197alloc_fail5:
Oliver Neukum884b6002005-04-21 21:28:02 +02001198 acm_write_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001199alloc_fail4:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001200 usb_buffer_free(usb_dev, ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
1201alloc_fail2:
1202 kfree(acm);
1203alloc_fail:
1204 return -ENOMEM;
1205}
1206
Oliver Neukum1365baf72007-10-12 17:24:28 +02001207static void stop_data_traffic(struct acm *acm)
1208{
1209 int i;
Oliver Neukum11ea8592008-06-20 11:25:57 +02001210 dbg("Entering stop_data_traffic");
Oliver Neukum1365baf72007-10-12 17:24:28 +02001211
1212 tasklet_disable(&acm->urb_task);
1213
1214 usb_kill_urb(acm->ctrlurb);
David Engrafe4cf3aa2008-03-20 10:01:34 +01001215 for(i = 0; i < ACM_NW; i++)
1216 usb_kill_urb(acm->wb[i].urb);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001217 for (i = 0; i < acm->rx_buflimit; i++)
1218 usb_kill_urb(acm->ru[i].urb);
1219
Oliver Neukum1365baf72007-10-12 17:24:28 +02001220 tasklet_enable(&acm->urb_task);
1221
1222 cancel_work_sync(&acm->work);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001223 cancel_work_sync(&acm->waker);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001224}
1225
Linus Torvalds1da177e2005-04-16 15:20:36 -07001226static void acm_disconnect(struct usb_interface *intf)
1227{
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001228 struct acm *acm = usb_get_intfdata(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001229 struct usb_device *usb_dev = interface_to_usbdev(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001230
David Brownell672c4e12008-08-06 18:41:12 -07001231 /* sibling interface is already cleaning up */
1232 if (!acm)
Oliver Neukum86067eea2006-01-08 12:39:13 +01001233 return;
David Brownell672c4e12008-08-06 18:41:12 -07001234
1235 mutex_lock(&open_mutex);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001236 if (acm->country_codes){
Alan Stern74da5d62007-08-02 13:29:10 -04001237 device_remove_file(&acm->control->dev,
1238 &dev_attr_wCountryCodes);
1239 device_remove_file(&acm->control->dev,
1240 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001241 }
Alan Stern74da5d62007-08-02 13:29:10 -04001242 device_remove_file(&acm->control->dev, &dev_attr_bmCapabilities);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001243 acm->dev = NULL;
Oliver Neukum86067eea2006-01-08 12:39:13 +01001244 usb_set_intfdata(acm->control, NULL);
1245 usb_set_intfdata(acm->data, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001246
Oliver Neukum1365baf72007-10-12 17:24:28 +02001247 stop_data_traffic(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001248
Oliver Neukum884b6002005-04-21 21:28:02 +02001249 acm_write_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001250 usb_buffer_free(usb_dev, acm->ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
Oliver Neukum830f4022008-06-25 14:17:16 +02001251 acm_read_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001252
Oliver Neukum830f4022008-06-25 14:17:16 +02001253 usb_driver_release_interface(&acm_driver, intf == acm->control ?
1254 acm->data : acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001255
1256 if (!acm->used) {
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001257 acm_tty_unregister(acm);
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001258 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001259 return;
1260 }
1261
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001262 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001263
1264 if (acm->tty)
1265 tty_hangup(acm->tty);
1266}
1267
Oliver Neukum35758582008-07-01 19:10:08 +02001268#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001269static int acm_suspend(struct usb_interface *intf, pm_message_t message)
1270{
1271 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001272 int cnt;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001273
Alan Stern65bfd292008-11-25 16:39:18 -05001274 if (message.event & PM_EVENT_AUTO) {
Oliver Neukum11ea8592008-06-20 11:25:57 +02001275 int b;
1276
1277 spin_lock_irq(&acm->read_lock);
1278 spin_lock(&acm->write_lock);
1279 b = acm->processing + acm->transmitting;
1280 spin_unlock(&acm->write_lock);
1281 spin_unlock_irq(&acm->read_lock);
1282 if (b)
1283 return -EBUSY;
1284 }
1285
1286 spin_lock_irq(&acm->read_lock);
1287 spin_lock(&acm->write_lock);
1288 cnt = acm->susp_count++;
1289 spin_unlock(&acm->write_lock);
1290 spin_unlock_irq(&acm->read_lock);
1291
1292 if (cnt)
Oliver Neukum1365baf72007-10-12 17:24:28 +02001293 return 0;
1294 /*
1295 we treat opened interfaces differently,
1296 we must guard against open
1297 */
1298 mutex_lock(&acm->mutex);
1299
1300 if (acm->used)
1301 stop_data_traffic(acm);
1302
1303 mutex_unlock(&acm->mutex);
1304 return 0;
1305}
1306
1307static int acm_resume(struct usb_interface *intf)
1308{
1309 struct acm *acm = usb_get_intfdata(intf);
1310 int rv = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +02001311 int cnt;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001312
Oliver Neukum11ea8592008-06-20 11:25:57 +02001313 spin_lock_irq(&acm->read_lock);
1314 acm->susp_count -= 1;
1315 cnt = acm->susp_count;
1316 spin_unlock_irq(&acm->read_lock);
1317
1318 if (cnt)
Oliver Neukum1365baf72007-10-12 17:24:28 +02001319 return 0;
1320
1321 mutex_lock(&acm->mutex);
1322 if (acm->used) {
1323 rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
1324 if (rv < 0)
Oliver Neukum11ea8592008-06-20 11:25:57 +02001325 goto err_out;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001326
1327 tasklet_schedule(&acm->urb_task);
1328 }
1329
1330err_out:
1331 mutex_unlock(&acm->mutex);
1332 return rv;
1333}
Oliver Neukum35758582008-07-01 19:10:08 +02001334
1335#endif /* CONFIG_PM */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001336/*
1337 * USB driver structure.
1338 */
1339
1340static struct usb_device_id acm_ids[] = {
1341 /* quirky and broken devices */
1342 { USB_DEVICE(0x0870, 0x0001), /* Metricom GS Modem */
1343 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1344 },
Andrey Arapovb0e2a702007-07-04 17:11:42 +02001345 { USB_DEVICE(0x0e8d, 0x0003), /* FIREFLY, MediaTek Inc; andrey.arapov@gmail.com */
1346 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1347 },
Andrew Lunn0f9c7b42008-12-23 17:31:23 +01001348 { USB_DEVICE(0x0e8d, 0x3329), /* MediaTek Inc GPS */
1349 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1350 },
Masahito Omote8753e652005-07-29 12:17:25 -07001351 { USB_DEVICE(0x0482, 0x0203), /* KYOCERA AH-K3001V */
1352 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1353 },
Chris Malley91a9c922006-10-03 10:08:28 +01001354 { USB_DEVICE(0x079b, 0x000f), /* BT On-Air USB MODEM */
1355 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1356 },
Alan Cox7abcf202009-04-06 17:35:01 +01001357 { USB_DEVICE(0x0ace, 0x1602), /* ZyDAS 56K USB MODEM */
1358 .driver_info = SINGLE_RX_URB,
1359 },
Oliver Neukum86478942006-05-13 22:50:47 +02001360 { USB_DEVICE(0x0ace, 0x1608), /* ZyDAS 56K USB MODEM */
1361 .driver_info = SINGLE_RX_URB, /* firmware bug */
1362 },
Oliver Neukum3dd2ae82006-06-23 09:14:17 +02001363 { USB_DEVICE(0x0ace, 0x1611), /* ZyDAS 56K USB MODEM - new version */
1364 .driver_info = SINGLE_RX_URB, /* firmware bug */
1365 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001366 { USB_DEVICE(0x22b8, 0x7000), /* Motorola Q Phone */
1367 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1368 },
Iain McFarlane6149ed52008-05-04 00:13:49 +01001369 { USB_DEVICE(0x0803, 0x3095), /* Zoom Telephonics Model 3095F USB MODEM */
1370 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1371 },
Eric Sandeenc8fd2c32008-08-14 08:25:40 -05001372 { USB_DEVICE(0x0572, 0x1321), /* Conexant USB MODEM CX93010 */
1373 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1374 },
Alan Coxc89c60e2009-01-11 19:53:10 +00001375 { USB_DEVICE(0x0572, 0x1324), /* Conexant USB MODEM RD02-D400 */
1376 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1377 },
Dmitriy Taychenachev155df652009-02-25 12:36:51 +08001378 { USB_DEVICE(0x22b8, 0x6425), /* Motorola MOTOMAGX phones */
1379 },
Adam Richterc332b4e2009-02-18 16:17:15 -08001380 { USB_DEVICE(0x0572, 0x1329), /* Hummingbird huc56s (Conexant) */
1381 .driver_info = NO_UNION_NORMAL, /* union descriptor misplaced on
1382 data interface instead of
1383 communications interface.
1384 Maybe we should define a new
1385 quirk for this. */
1386 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001387
Linus Torvalds1da177e2005-04-16 15:20:36 -07001388 /* control interfaces with various AT-command sets */
1389 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1390 USB_CDC_ACM_PROTO_AT_V25TER) },
1391 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1392 USB_CDC_ACM_PROTO_AT_PCCA101) },
1393 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1394 USB_CDC_ACM_PROTO_AT_PCCA101_WAKE) },
1395 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1396 USB_CDC_ACM_PROTO_AT_GSM) },
1397 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1398 USB_CDC_ACM_PROTO_AT_3G ) },
1399 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1400 USB_CDC_ACM_PROTO_AT_CDMA) },
1401
1402 /* NOTE: COMM/ACM/0xff is likely MSFT RNDIS ... NOT a modem!! */
1403 { }
1404};
1405
1406MODULE_DEVICE_TABLE (usb, acm_ids);
1407
1408static struct usb_driver acm_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001409 .name = "cdc_acm",
1410 .probe = acm_probe,
1411 .disconnect = acm_disconnect,
Oliver Neukum35758582008-07-01 19:10:08 +02001412#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001413 .suspend = acm_suspend,
1414 .resume = acm_resume,
Oliver Neukum35758582008-07-01 19:10:08 +02001415#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001416 .id_table = acm_ids,
Oliver Neukum35758582008-07-01 19:10:08 +02001417#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001418 .supports_autosuspend = 1,
Oliver Neukum35758582008-07-01 19:10:08 +02001419#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001420};
1421
1422/*
1423 * TTY driver structures.
1424 */
1425
Jeff Dikeb68e31d2006-10-02 02:17:18 -07001426static const struct tty_operations acm_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001427 .open = acm_tty_open,
1428 .close = acm_tty_close,
1429 .write = acm_tty_write,
1430 .write_room = acm_tty_write_room,
1431 .ioctl = acm_tty_ioctl,
1432 .throttle = acm_tty_throttle,
1433 .unthrottle = acm_tty_unthrottle,
1434 .chars_in_buffer = acm_tty_chars_in_buffer,
1435 .break_ctl = acm_tty_break_ctl,
1436 .set_termios = acm_tty_set_termios,
1437 .tiocmget = acm_tty_tiocmget,
1438 .tiocmset = acm_tty_tiocmset,
1439};
1440
1441/*
1442 * Init / exit.
1443 */
1444
1445static int __init acm_init(void)
1446{
1447 int retval;
1448 acm_tty_driver = alloc_tty_driver(ACM_TTY_MINORS);
1449 if (!acm_tty_driver)
1450 return -ENOMEM;
1451 acm_tty_driver->owner = THIS_MODULE,
1452 acm_tty_driver->driver_name = "acm",
1453 acm_tty_driver->name = "ttyACM",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001454 acm_tty_driver->major = ACM_TTY_MAJOR,
1455 acm_tty_driver->minor_start = 0,
1456 acm_tty_driver->type = TTY_DRIVER_TYPE_SERIAL,
1457 acm_tty_driver->subtype = SERIAL_TYPE_NORMAL,
Greg Kroah-Hartman331b8312005-06-20 21:15:16 -07001458 acm_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001459 acm_tty_driver->init_termios = tty_std_termios;
1460 acm_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD | HUPCL | CLOCAL;
1461 tty_set_operations(acm_tty_driver, &acm_ops);
1462
1463 retval = tty_register_driver(acm_tty_driver);
1464 if (retval) {
1465 put_tty_driver(acm_tty_driver);
1466 return retval;
1467 }
1468
1469 retval = usb_register(&acm_driver);
1470 if (retval) {
1471 tty_unregister_driver(acm_tty_driver);
1472 put_tty_driver(acm_tty_driver);
1473 return retval;
1474 }
1475
Greg Kroah-Hartman5909f6e2008-08-18 13:21:04 -07001476 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_VERSION ":"
1477 DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001478
1479 return 0;
1480}
1481
1482static void __exit acm_exit(void)
1483{
1484 usb_deregister(&acm_driver);
1485 tty_unregister_driver(acm_tty_driver);
1486 put_tty_driver(acm_tty_driver);
1487}
1488
1489module_init(acm_init);
1490module_exit(acm_exit);
1491
1492MODULE_AUTHOR( DRIVER_AUTHOR );
1493MODULE_DESCRIPTION( DRIVER_DESC );
1494MODULE_LICENSE("GPL");
Scott James Remnante766aeb2009-04-06 17:33:18 +01001495MODULE_ALIAS_CHARDEV_MAJOR(ACM_TTY_MAJOR);