blob: f239e3bd6423f89fd262298ab7f6d3642bb08658 [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
David Kubicek61a87ad2005-11-01 18:51:34 +010053#define DRIVER_AUTHOR "Armin Fuerst, Pavel Machek, Johannes Erdfelt, Vojtech Pavlik, David Kubicek"
Linus Torvalds1da177e2005-04-16 15:20:36 -070054#define DRIVER_DESC "USB Abstract Control Model driver for USB modems and ISDN adapters"
55
56static struct usb_driver acm_driver;
57static struct tty_driver *acm_tty_driver;
58static struct acm *acm_table[ACM_TTY_MINORS];
59
Arjan van de Ven4186ecf2006-01-11 15:55:29 +010060static DEFINE_MUTEX(open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -070061
Alan Cox10077d42009-06-11 12:36:09 +010062#define ACM_READY(acm) (acm && acm->dev && acm->port.count)
Linus Torvalds1da177e2005-04-16 15:20:36 -070063
Alan Cox739e0282009-06-11 12:27:50 +010064static const struct tty_port_operations acm_port_ops = {
65};
66
Linus Torvalds1da177e2005-04-16 15:20:36 -070067/*
68 * Functions for ACM control messages.
69 */
70
Alan Cox6e47e062009-06-11 12:37:06 +010071static int acm_ctrl_msg(struct acm *acm, int request, int value,
72 void *buf, int len)
Linus Torvalds1da177e2005-04-16 15:20:36 -070073{
74 int retval = usb_control_msg(acm->dev, usb_sndctrlpipe(acm->dev, 0),
75 request, USB_RT_ACM, value,
76 acm->control->altsetting[0].desc.bInterfaceNumber,
77 buf, len, 5000);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +010078 dev_dbg(&acm->control->dev,
79 "%s - rq 0x%02x, val %#x, len %#x, result %d\n",
80 __func__, request, value, len, retval);
Linus Torvalds1da177e2005-04-16 15:20:36 -070081 return retval < 0 ? retval : 0;
82}
83
84/* devices aren't required to support these requests.
85 * the cdc acm descriptor tells whether they do...
86 */
87#define acm_set_control(acm, control) \
88 acm_ctrl_msg(acm, USB_CDC_REQ_SET_CONTROL_LINE_STATE, control, NULL, 0)
89#define acm_set_line(acm, line) \
90 acm_ctrl_msg(acm, USB_CDC_REQ_SET_LINE_CODING, 0, line, sizeof *(line))
91#define acm_send_break(acm, ms) \
92 acm_ctrl_msg(acm, USB_CDC_REQ_SEND_BREAK, ms, NULL, 0)
93
94/*
Oliver Neukum884b6002005-04-21 21:28:02 +020095 * Write buffer management.
96 * All of these assume proper locks taken by the caller.
97 */
98
99static int acm_wb_alloc(struct acm *acm)
100{
101 int i, wbn;
102 struct acm_wb *wb;
103
David Engrafe4cf3aa2008-03-20 10:01:34 +0100104 wbn = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200105 i = 0;
106 for (;;) {
107 wb = &acm->wb[wbn];
108 if (!wb->use) {
109 wb->use = 1;
110 return wbn;
111 }
Oliver Neukum86478942006-05-13 22:50:47 +0200112 wbn = (wbn + 1) % ACM_NW;
113 if (++i >= ACM_NW)
Oliver Neukum884b6002005-04-21 21:28:02 +0200114 return -1;
115 }
116}
117
Oliver Neukum884b6002005-04-21 21:28:02 +0200118static int acm_wb_is_avail(struct acm *acm)
119{
120 int i, n;
David Brownelle5fbab52008-08-06 18:46:10 -0700121 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200122
Oliver Neukum86478942006-05-13 22:50:47 +0200123 n = ACM_NW;
David Brownelle5fbab52008-08-06 18:46:10 -0700124 spin_lock_irqsave(&acm->write_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100125 for (i = 0; i < ACM_NW; i++)
Oliver Neukum86478942006-05-13 22:50:47 +0200126 n -= acm->wb[i].use;
David Brownelle5fbab52008-08-06 18:46:10 -0700127 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200128 return n;
129}
130
Oliver Neukum884b6002005-04-21 21:28:02 +0200131/*
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800132 * Finish write. Caller must hold acm->write_lock
Oliver Neukum884b6002005-04-21 21:28:02 +0200133 */
David Engrafe4cf3aa2008-03-20 10:01:34 +0100134static void acm_write_done(struct acm *acm, struct acm_wb *wb)
Oliver Neukum884b6002005-04-21 21:28:02 +0200135{
David Engrafe4cf3aa2008-03-20 10:01:34 +0100136 wb->use = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200137 acm->transmitting--;
Oliver Neukum97d35f92009-12-16 17:05:57 +0100138 usb_autopm_put_interface_async(acm->control);
Oliver Neukum884b6002005-04-21 21:28:02 +0200139}
140
141/*
142 * Poke write.
Oliver Neukum11ea8592008-06-20 11:25:57 +0200143 *
144 * the caller is responsible for locking
Oliver Neukum884b6002005-04-21 21:28:02 +0200145 */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200146
147static int acm_start_wb(struct acm *acm, struct acm_wb *wb)
148{
149 int rc;
150
151 acm->transmitting++;
152
153 wb->urb->transfer_buffer = wb->buf;
154 wb->urb->transfer_dma = wb->dmah;
155 wb->urb->transfer_buffer_length = wb->len;
156 wb->urb->dev = acm->dev;
157
Alan Cox6e47e062009-06-11 12:37:06 +0100158 rc = usb_submit_urb(wb->urb, GFP_ATOMIC);
159 if (rc < 0) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100160 dev_err(&acm->data->dev,
161 "%s - usb_submit_urb(write bulk) failed: %d\n",
162 __func__, rc);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200163 acm_write_done(acm, wb);
164 }
165 return rc;
166}
167
David Engrafe4cf3aa2008-03-20 10:01:34 +0100168static int acm_write_start(struct acm *acm, int wbn)
Oliver Neukum884b6002005-04-21 21:28:02 +0200169{
170 unsigned long flags;
David Brownell934da462008-08-06 18:44:12 -0700171 struct acm_wb *wb = &acm->wb[wbn];
Oliver Neukum884b6002005-04-21 21:28:02 +0200172 int rc;
173
174 spin_lock_irqsave(&acm->write_lock, flags);
175 if (!acm->dev) {
David Brownell934da462008-08-06 18:44:12 -0700176 wb->use = 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200177 spin_unlock_irqrestore(&acm->write_lock, flags);
178 return -ENODEV;
179 }
180
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100181 dev_vdbg(&acm->data->dev, "%s - susp_count %d\n", __func__,
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100182 acm->susp_count);
Oliver Neukum97d35f92009-12-16 17:05:57 +0100183 usb_autopm_get_interface_async(acm->control);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200184 if (acm->susp_count) {
Oliver Neukum97d35f92009-12-16 17:05:57 +0100185 if (!acm->delayed_wb)
186 acm->delayed_wb = wb;
187 else
188 usb_autopm_put_interface_async(acm->control);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200189 spin_unlock_irqrestore(&acm->write_lock, flags);
190 return 0; /* A white lie */
191 }
192 usb_mark_last_busy(acm->dev);
193
Oliver Neukum11ea8592008-06-20 11:25:57 +0200194 rc = acm_start_wb(acm, wb);
Oliver Neukum884b6002005-04-21 21:28:02 +0200195 spin_unlock_irqrestore(&acm->write_lock, flags);
196
Oliver Neukum884b6002005-04-21 21:28:02 +0200197 return rc;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200198
Oliver Neukum884b6002005-04-21 21:28:02 +0200199}
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100200/*
201 * attributes exported through sysfs
202 */
203static ssize_t show_caps
204(struct device *dev, struct device_attribute *attr, char *buf)
205{
206 struct usb_interface *intf = to_usb_interface(dev);
207 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum884b6002005-04-21 21:28:02 +0200208
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100209 return sprintf(buf, "%d", acm->ctrl_caps);
210}
211static DEVICE_ATTR(bmCapabilities, S_IRUGO, show_caps, NULL);
212
213static ssize_t show_country_codes
214(struct device *dev, struct device_attribute *attr, char *buf)
215{
216 struct usb_interface *intf = to_usb_interface(dev);
217 struct acm *acm = usb_get_intfdata(intf);
218
219 memcpy(buf, acm->country_codes, acm->country_code_size);
220 return acm->country_code_size;
221}
222
223static DEVICE_ATTR(wCountryCodes, S_IRUGO, show_country_codes, NULL);
224
225static ssize_t show_country_rel_date
226(struct device *dev, struct device_attribute *attr, char *buf)
227{
228 struct usb_interface *intf = to_usb_interface(dev);
229 struct acm *acm = usb_get_intfdata(intf);
230
231 return sprintf(buf, "%d", acm->country_rel_date);
232}
233
234static DEVICE_ATTR(iCountryCodeRelDate, S_IRUGO, show_country_rel_date, NULL);
Oliver Neukum884b6002005-04-21 21:28:02 +0200235/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236 * Interrupt handlers for various ACM device responses
237 */
238
239/* control interface reports status changes with "interrupt" transfers */
David Howells7d12e782006-10-05 14:55:46 +0100240static void acm_ctrl_irq(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241{
242 struct acm *acm = urb->context;
243 struct usb_cdc_notification *dr = urb->transfer_buffer;
Alan Cox10077d42009-06-11 12:36:09 +0100244 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700245 unsigned char *data;
246 int newctrl;
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700247 int retval;
248 int status = urb->status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700250 switch (status) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700251 case 0:
252 /* success */
253 break;
254 case -ECONNRESET:
255 case -ENOENT:
256 case -ESHUTDOWN:
257 /* this urb is terminated, clean up */
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100258 dev_dbg(&acm->control->dev,
259 "%s - urb shutting down with status: %d\n",
260 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700261 return;
262 default:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100263 dev_dbg(&acm->control->dev,
264 "%s - nonzero urb status received: %d\n",
265 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266 goto exit;
267 }
268
269 if (!ACM_READY(acm))
270 goto exit;
271
Johan Hovold7e7797e2011-03-22 11:12:11 +0100272 usb_mark_last_busy(acm->dev);
273
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274 data = (unsigned char *)(dr + 1);
275 switch (dr->bNotificationType) {
Alan Cox6e47e062009-06-11 12:37:06 +0100276 case USB_CDC_NOTIFY_NETWORK_CONNECTION:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100277 dev_dbg(&acm->control->dev, "%s - network connection: %d\n",
278 __func__, dr->wValue);
Alan Cox6e47e062009-06-11 12:37:06 +0100279 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700280
Alan Cox6e47e062009-06-11 12:37:06 +0100281 case USB_CDC_NOTIFY_SERIAL_STATE:
282 tty = tty_port_tty_get(&acm->port);
283 newctrl = get_unaligned_le16(data);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700284
Alan Cox6e47e062009-06-11 12:37:06 +0100285 if (tty) {
286 if (!acm->clocal &&
287 (acm->ctrlin & ~newctrl & ACM_CTRL_DCD)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100288 dev_dbg(&acm->control->dev,
289 "%s - calling hangup\n", __func__);
Alan Cox6e47e062009-06-11 12:37:06 +0100290 tty_hangup(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700291 }
Alan Cox6e47e062009-06-11 12:37:06 +0100292 tty_kref_put(tty);
293 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700294
Alan Cox6e47e062009-06-11 12:37:06 +0100295 acm->ctrlin = newctrl;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700296
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100297 dev_dbg(&acm->control->dev,
298 "%s - input control lines: dcd%c dsr%c break%c "
299 "ring%c framing%c parity%c overrun%c\n",
300 __func__,
Alan Cox6e47e062009-06-11 12:37:06 +0100301 acm->ctrlin & ACM_CTRL_DCD ? '+' : '-',
302 acm->ctrlin & ACM_CTRL_DSR ? '+' : '-',
303 acm->ctrlin & ACM_CTRL_BRK ? '+' : '-',
304 acm->ctrlin & ACM_CTRL_RI ? '+' : '-',
305 acm->ctrlin & ACM_CTRL_FRAMING ? '+' : '-',
306 acm->ctrlin & ACM_CTRL_PARITY ? '+' : '-',
307 acm->ctrlin & ACM_CTRL_OVERRUN ? '+' : '-');
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308 break;
309
Alan Cox6e47e062009-06-11 12:37:06 +0100310 default:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100311 dev_dbg(&acm->control->dev,
312 "%s - unknown notification %d received: index %d "
313 "len %d data0 %d data1 %d\n",
314 __func__,
Alan Cox6e47e062009-06-11 12:37:06 +0100315 dr->bNotificationType, dr->wIndex,
316 dr->wLength, data[0], data[1]);
317 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 }
319exit:
Alan Cox6e47e062009-06-11 12:37:06 +0100320 retval = usb_submit_urb(urb, GFP_ATOMIC);
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700321 if (retval)
Johan Hovold1d9846e2011-03-22 11:12:14 +0100322 dev_err(&acm->control->dev, "%s - usb_submit_urb failed: %d\n",
323 __func__, retval);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700324}
325
326/* data interface returns incoming bytes, or we got unthrottled */
David Howells7d12e782006-10-05 14:55:46 +0100327static void acm_read_bulk(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700328{
David Kubicek61a87ad2005-11-01 18:51:34 +0100329 struct acm_rb *buf;
330 struct acm_ru *rcv = urb->context;
331 struct acm *acm = rcv->instance;
Oliver Neukum86478942006-05-13 22:50:47 +0200332 int status = urb->status;
Greg Kroah-Hartman185d4052007-07-18 10:58:02 -0700333
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100334 dev_vdbg(&acm->data->dev, "%s - status %d\n", __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335
Oliver Neukum11ea8592008-06-20 11:25:57 +0200336 if (!ACM_READY(acm)) {
Johan Hovold1d9846e2011-03-22 11:12:14 +0100337 dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200339 }
340 usb_mark_last_busy(acm->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341
Oliver Neukum86478942006-05-13 22:50:47 +0200342 if (status)
Johan Hovold1d9846e2011-03-22 11:12:14 +0100343 dev_dbg(&acm->data->dev, "%s - non-zero urb status: %d\n",
344 __func__, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345
David Kubicek61a87ad2005-11-01 18:51:34 +0100346 buf = rcv->buffer;
347 buf->size = urb->actual_length;
348
Oliver Neukum86478942006-05-13 22:50:47 +0200349 if (likely(status == 0)) {
350 spin_lock(&acm->read_lock);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200351 acm->processing++;
Oliver Neukum86478942006-05-13 22:50:47 +0200352 list_add_tail(&rcv->list, &acm->spare_read_urbs);
353 list_add_tail(&buf->list, &acm->filled_read_bufs);
354 spin_unlock(&acm->read_lock);
355 } else {
356 /* we drop the buffer due to an error */
357 spin_lock(&acm->read_lock);
358 list_add_tail(&rcv->list, &acm->spare_read_urbs);
359 list_add(&buf->list, &acm->spare_read_bufs);
360 spin_unlock(&acm->read_lock);
361 /* nevertheless the tasklet must be kicked unconditionally
362 so the queue cannot dry up */
363 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200364 if (likely(!acm->susp_count))
365 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700366}
367
368static void acm_rx_tasklet(unsigned long _acm)
369{
370 struct acm *acm = (void *)_acm;
David Kubicek61a87ad2005-11-01 18:51:34 +0100371 struct acm_rb *buf;
Alan Cox10077d42009-06-11 12:36:09 +0100372 struct tty_struct *tty;
David Kubicek61a87ad2005-11-01 18:51:34 +0100373 struct acm_ru *rcv;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200374 unsigned long flags;
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100375 unsigned char throttled;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200376
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100377 dev_vdbg(&acm->data->dev, "%s\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700378
Alan Cox10077d42009-06-11 12:36:09 +0100379 if (!ACM_READY(acm)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100380 dev_dbg(&acm->data->dev, "%s - acm not ready\n", __func__);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100381 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200382 }
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100383
Oliver Neukum834dbca2007-03-06 10:47:04 +0100384 spin_lock_irqsave(&acm->throttle_lock, flags);
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100385 throttled = acm->throttle;
Oliver Neukum834dbca2007-03-06 10:47:04 +0100386 spin_unlock_irqrestore(&acm->throttle_lock, flags);
Alan Cox10077d42009-06-11 12:36:09 +0100387 if (throttled) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100388 dev_dbg(&acm->data->dev, "%s - throttled\n", __func__);
David Kubicek61a87ad2005-11-01 18:51:34 +0100389 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200390 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100391
Alan Cox10077d42009-06-11 12:36:09 +0100392 tty = tty_port_tty_get(&acm->port);
393
David Kubicek61a87ad2005-11-01 18:51:34 +0100394next_buffer:
Jarek Poplawski762f0072006-10-06 07:23:11 +0200395 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100396 if (list_empty(&acm->filled_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200397 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100398 goto urbs;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700399 }
David Kubicek61a87ad2005-11-01 18:51:34 +0100400 buf = list_entry(acm->filled_read_bufs.next,
401 struct acm_rb, list);
402 list_del(&buf->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200403 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100404
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100405 dev_vdbg(&acm->data->dev, "%s - processing buf 0x%p, size = %d\n",
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100406 __func__, buf, buf->size);
Alan Cox10077d42009-06-11 12:36:09 +0100407 if (tty) {
408 spin_lock_irqsave(&acm->throttle_lock, flags);
409 throttled = acm->throttle;
410 spin_unlock_irqrestore(&acm->throttle_lock, flags);
411 if (!throttled) {
Alan Cox10077d42009-06-11 12:36:09 +0100412 tty_insert_flip_string(tty, buf->base, buf->size);
413 tty_flip_buffer_push(tty);
414 } else {
415 tty_kref_put(tty);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100416 dev_dbg(&acm->data->dev, "%s - throttling noticed\n",
417 __func__);
Alan Cox10077d42009-06-11 12:36:09 +0100418 spin_lock_irqsave(&acm->read_lock, flags);
419 list_add(&buf->list, &acm->filled_read_bufs);
420 spin_unlock_irqrestore(&acm->read_lock, flags);
421 return;
422 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700423 }
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:
Alan Cox10077d42009-06-11 12:36:09 +0100431 tty_kref_put(tty);
432
David Kubicek61a87ad2005-11-01 18:51:34 +0100433 while (!list_empty(&acm->spare_read_bufs)) {
Jarek Poplawski762f0072006-10-06 07:23:11 +0200434 spin_lock_irqsave(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100435 if (list_empty(&acm->spare_read_urbs)) {
Oliver Neukum11ea8592008-06-20 11:25:57 +0200436 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200437 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100438 return;
439 }
440 rcv = list_entry(acm->spare_read_urbs.next,
441 struct acm_ru, list);
442 list_del(&rcv->list);
Jarek Poplawski762f0072006-10-06 07:23:11 +0200443 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100444
445 buf = list_entry(acm->spare_read_bufs.next,
446 struct acm_rb, list);
447 list_del(&buf->list);
448
449 rcv->buffer = buf;
450
Oliver Neukumcf7fdd52009-08-04 23:52:09 +0200451 if (acm->is_int_ep)
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +0400452 usb_fill_int_urb(rcv->urb, acm->dev,
453 acm->rx_endpoint,
454 buf->base,
455 acm->readsize,
Oliver Neukumcf7fdd52009-08-04 23:52:09 +0200456 acm_read_bulk, rcv, acm->bInterval);
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +0400457 else
458 usb_fill_bulk_urb(rcv->urb, acm->dev,
459 acm->rx_endpoint,
460 buf->base,
461 acm->readsize,
462 acm_read_bulk, rcv);
David Kubicek61a87ad2005-11-01 18:51:34 +0100463 rcv->urb->transfer_dma = buf->dma;
464 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
465
Alan Cox6e47e062009-06-11 12:37:06 +0100466 /* This shouldn't kill the driver as unsuccessful URBs are
467 returned to the free-urbs-pool and resubmited ASAP */
Oliver Neukum11ea8592008-06-20 11:25:57 +0200468 spin_lock_irqsave(&acm->read_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100469 if (acm->susp_count ||
470 usb_submit_urb(rcv->urb, GFP_ATOMIC) < 0) {
David Kubicek61a87ad2005-11-01 18:51:34 +0100471 list_add(&buf->list, &acm->spare_read_bufs);
David Kubicek61a87ad2005-11-01 18:51:34 +0100472 list_add(&rcv->list, &acm->spare_read_urbs);
Oliver Neukum11ea8592008-06-20 11:25:57 +0200473 acm->processing = 0;
Jarek Poplawski762f0072006-10-06 07:23:11 +0200474 spin_unlock_irqrestore(&acm->read_lock, flags);
David Kubicek61a87ad2005-11-01 18:51:34 +0100475 return;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200476 } else {
477 spin_unlock_irqrestore(&acm->read_lock, flags);
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100478 dev_vdbg(&acm->data->dev,
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100479 "%s - sending urb 0x%p, rcv 0x%p, buf 0x%p\n",
480 __func__, rcv->urb, rcv, buf);
David Kubicek61a87ad2005-11-01 18:51:34 +0100481 }
482 }
Oliver Neukum11ea8592008-06-20 11:25:57 +0200483 spin_lock_irqsave(&acm->read_lock, flags);
484 acm->processing = 0;
485 spin_unlock_irqrestore(&acm->read_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700486}
487
488/* data interface wrote those outgoing bytes */
David Howells7d12e782006-10-05 14:55:46 +0100489static void acm_write_bulk(struct urb *urb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700490{
Ming Leicdc97792008-02-24 18:41:47 +0800491 struct acm_wb *wb = urb->context;
David Brownelle5fbab52008-08-06 18:46:10 -0700492 struct acm *acm = wb->instance;
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800493 unsigned long flags;
Oliver Neukum884b6002005-04-21 21:28:02 +0200494
Johan Hovold4fa46262011-03-22 11:12:16 +0100495 if (urb->status || (urb->actual_length != urb->transfer_buffer_length))
496 dev_vdbg(&acm->data->dev, "%s - len %d/%d, status %d\n",
Johan Hovold1d9846e2011-03-22 11:12:14 +0100497 __func__,
David Brownelle5fbab52008-08-06 18:46:10 -0700498 urb->actual_length,
499 urb->transfer_buffer_length,
500 urb->status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700501
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800502 spin_lock_irqsave(&acm->write_lock, flags);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100503 acm_write_done(acm, wb);
Brandon Philipsad0b65e2008-11-06 11:19:11 -0800504 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200505 if (ACM_READY(acm))
506 schedule_work(&acm->work);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507}
508
David Howellsc4028952006-11-22 14:57:56 +0000509static void acm_softint(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700510{
David Howellsc4028952006-11-22 14:57:56 +0000511 struct acm *acm = container_of(work, struct acm, work);
Alan Cox10077d42009-06-11 12:36:09 +0100512 struct tty_struct *tty;
David Brownelle5fbab52008-08-06 18:46:10 -0700513
Johan Hovold1d9846e2011-03-22 11:12:14 +0100514 dev_vdbg(&acm->data->dev, "%s\n", __func__);
515
Linus Torvalds1da177e2005-04-16 15:20:36 -0700516 if (!ACM_READY(acm))
517 return;
Alan Cox10077d42009-06-11 12:36:09 +0100518 tty = tty_port_tty_get(&acm->port);
Johan Hovold15e5bee2011-03-22 11:12:10 +0100519 if (!tty)
520 return;
Alan Cox10077d42009-06-11 12:36:09 +0100521 tty_wakeup(tty);
522 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700523}
524
525/*
526 * TTY handlers
527 */
528
529static int acm_tty_open(struct tty_struct *tty, struct file *filp)
530{
531 struct acm *acm;
Thadeu Lima de Souza Cascardo42dd2aa2009-06-25 14:41:24 +0100532 int rv = -ENODEV;
David Kubicek61a87ad2005-11-01 18:51:34 +0100533 int i;
Arjan van de Ven4186ecf2006-01-11 15:55:29 +0100534
535 mutex_lock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700536
537 acm = acm_table[tty->index];
538 if (!acm || !acm->dev)
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100539 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700540 else
541 rv = 0;
542
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100543 dev_dbg(&acm->control->dev, "%s\n", __func__);
544
David Engraf28d1dfa2008-03-20 10:53:52 +0100545 set_bit(TTY_NO_WRITE_SPLIT, &tty->flags);
Alan Cox10077d42009-06-11 12:36:09 +0100546
Linus Torvalds1da177e2005-04-16 15:20:36 -0700547 tty->driver_data = acm;
Alan Cox10077d42009-06-11 12:36:09 +0100548 tty_port_tty_set(&acm->port, tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700549
Oliver Neukum94409cc2008-02-11 15:22:29 +0100550 if (usb_autopm_get_interface(acm->control) < 0)
551 goto early_bail;
Oliver Neukum11ea8592008-06-20 11:25:57 +0200552 else
553 acm->control->needs_remote_wakeup = 1;
Oliver Neukum1365baf72007-10-12 17:24:28 +0200554
555 mutex_lock(&acm->mutex);
Alan Cox10077d42009-06-11 12:36:09 +0100556 if (acm->port.count++) {
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100557 mutex_unlock(&acm->mutex);
Oliver Neukum1365baf72007-10-12 17:24:28 +0200558 usb_autopm_put_interface(acm->control);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100559 goto out;
Alan Cox10077d42009-06-11 12:36:09 +0100560 }
Oliver Neukum1365baf72007-10-12 17:24:28 +0200561
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 acm->ctrlurb->dev = acm->dev;
563 if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) {
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100564 dev_err(&acm->control->dev,
565 "%s - usb_submit_urb(ctrl irq) failed\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700566 goto bail_out;
567 }
568
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100569 if (0 > acm_set_control(acm, acm->ctrlout = ACM_CTRL_DTR | ACM_CTRL_RTS) &&
570 (acm->ctrl_caps & USB_CDC_CAP_LINE))
Johan Hovold94d4c892011-03-22 11:12:22 +0100571 goto bail_out;
Alan Cox10077d42009-06-11 12:36:09 +0100572
Oliver Neukum11ea8592008-06-20 11:25:57 +0200573 usb_autopm_put_interface(acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700574
David Kubicek61a87ad2005-11-01 18:51:34 +0100575 INIT_LIST_HEAD(&acm->spare_read_urbs);
576 INIT_LIST_HEAD(&acm->spare_read_bufs);
577 INIT_LIST_HEAD(&acm->filled_read_bufs);
Alan Cox6e47e062009-06-11 12:37:06 +0100578
579 for (i = 0; i < acm->rx_buflimit; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100580 list_add(&(acm->ru[i].list), &acm->spare_read_urbs);
Alan Cox6e47e062009-06-11 12:37:06 +0100581 for (i = 0; i < acm->rx_buflimit; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100582 list_add(&(acm->rb[i].list), &acm->spare_read_bufs);
David Kubicek61a87ad2005-11-01 18:51:34 +0100583
Oliver Neukumca79b7b2007-02-12 08:41:35 +0100584 acm->throttle = 0;
585
Oliver Neukum7af25b42009-09-08 23:51:28 +0200586 set_bit(ASYNCB_INITIALIZED, &acm->port.flags);
Alan Cox10077d42009-06-11 12:36:09 +0100587 rv = tty_port_block_til_ready(&acm->port, tty, filp);
Henry Gebhardt18a77b52009-11-04 11:19:28 +0100588 tasklet_schedule(&acm->urb_task);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100589
Oliver Neukum1365baf72007-10-12 17:24:28 +0200590 mutex_unlock(&acm->mutex);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100591out:
Oliver Neukum94409cc2008-02-11 15:22:29 +0100592 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593 return rv;
594
Linus Torvalds1da177e2005-04-16 15:20:36 -0700595bail_out:
Alan Cox10077d42009-06-11 12:36:09 +0100596 acm->port.count--;
Oliver Neukum1365baf72007-10-12 17:24:28 +0200597 mutex_unlock(&acm->mutex);
Oliver Neukum2b626dc2010-02-03 17:10:22 +0100598 usb_autopm_put_interface(acm->control);
Oliver Neukum94409cc2008-02-11 15:22:29 +0100599early_bail:
600 mutex_unlock(&open_mutex);
Alan Cox10077d42009-06-11 12:36:09 +0100601 tty_port_tty_set(&acm->port, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700602 return -EIO;
603}
604
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700605static void acm_tty_unregister(struct acm *acm)
606{
Johan Hovolddab54c92011-03-22 11:12:21 +0100607 int i;
David Kubicek61a87ad2005-11-01 18:51:34 +0100608
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700609 tty_unregister_device(acm_tty_driver, acm->minor);
610 usb_put_intf(acm->control);
611 acm_table[acm->minor] = NULL;
612 usb_free_urb(acm->ctrlurb);
David Engrafe4cf3aa2008-03-20 10:01:34 +0100613 for (i = 0; i < ACM_NW; i++)
614 usb_free_urb(acm->wb[i].urb);
Johan Hovolddab54c92011-03-22 11:12:21 +0100615 for (i = 0; i < acm->rx_buflimit; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +0100616 usb_free_urb(acm->ru[i].urb);
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100617 kfree(acm->country_codes);
brian@murphy.dk83ef3442005-06-29 16:53:29 -0700618 kfree(acm);
619}
620
Arnd Bergmann4e608672010-06-01 22:53:04 +0200621static void acm_port_down(struct acm *acm)
Alan Cox10077d42009-06-11 12:36:09 +0100622{
Johan Hovolddab54c92011-03-22 11:12:21 +0100623 int i;
624
Alan Cox10077d42009-06-11 12:36:09 +0100625 mutex_lock(&open_mutex);
626 if (acm->dev) {
627 usb_autopm_get_interface(acm->control);
628 acm_set_control(acm, acm->ctrlout = 0);
Alan Cox10077d42009-06-11 12:36:09 +0100629 usb_kill_urb(acm->ctrlurb);
630 for (i = 0; i < ACM_NW; i++)
631 usb_kill_urb(acm->wb[i].urb);
Johan Hovold23b80552011-03-22 11:12:09 +0100632 tasklet_disable(&acm->urb_task);
Johan Hovolddab54c92011-03-22 11:12:21 +0100633 for (i = 0; i < acm->rx_buflimit; i++)
Alan Cox10077d42009-06-11 12:36:09 +0100634 usb_kill_urb(acm->ru[i].urb);
Johan Hovold23b80552011-03-22 11:12:09 +0100635 tasklet_enable(&acm->urb_task);
Alan Cox10077d42009-06-11 12:36:09 +0100636 acm->control->needs_remote_wakeup = 0;
637 usb_autopm_put_interface(acm->control);
638 }
639 mutex_unlock(&open_mutex);
640}
641
642static void acm_tty_hangup(struct tty_struct *tty)
643{
644 struct acm *acm = tty->driver_data;
645 tty_port_hangup(&acm->port);
Arnd Bergmann4e608672010-06-01 22:53:04 +0200646 acm_port_down(acm);
Alan Cox10077d42009-06-11 12:36:09 +0100647}
648
Linus Torvalds1da177e2005-04-16 15:20:36 -0700649static void acm_tty_close(struct tty_struct *tty, struct file *filp)
650{
651 struct acm *acm = tty->driver_data;
652
Alan Cox10077d42009-06-11 12:36:09 +0100653 /* Perform the closing process and see if we need to do the hardware
654 shutdown */
Francesco Lavra051522b2009-11-03 10:53:07 +0000655 if (!acm)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656 return;
Francesco Lavra051522b2009-11-03 10:53:07 +0000657 if (tty_port_close_start(&acm->port, tty, filp) == 0) {
658 mutex_lock(&open_mutex);
659 if (!acm->dev) {
660 tty_port_tty_set(&acm->port, NULL);
661 acm_tty_unregister(acm);
662 tty->driver_data = NULL;
663 }
664 mutex_unlock(&open_mutex);
665 return;
666 }
Arnd Bergmann4e608672010-06-01 22:53:04 +0200667 acm_port_down(acm);
Alan Cox10077d42009-06-11 12:36:09 +0100668 tty_port_close_end(&acm->port, tty);
Alan Cox10077d42009-06-11 12:36:09 +0100669 tty_port_tty_set(&acm->port, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670}
671
Alan Cox6e47e062009-06-11 12:37:06 +0100672static int acm_tty_write(struct tty_struct *tty,
673 const unsigned char *buf, int count)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700674{
675 struct acm *acm = tty->driver_data;
676 int stat;
Oliver Neukum884b6002005-04-21 21:28:02 +0200677 unsigned long flags;
678 int wbn;
679 struct acm_wb *wb;
680
Linus Torvalds1da177e2005-04-16 15:20:36 -0700681 if (!ACM_READY(acm))
682 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700683 if (!count)
684 return 0;
685
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100686 dev_vdbg(&acm->data->dev, "%s - count %d\n", __func__, count);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100687
Oliver Neukum884b6002005-04-21 21:28:02 +0200688 spin_lock_irqsave(&acm->write_lock, flags);
Alan Cox6e47e062009-06-11 12:37:06 +0100689 wbn = acm_wb_alloc(acm);
690 if (wbn < 0) {
Oliver Neukum884b6002005-04-21 21:28:02 +0200691 spin_unlock_irqrestore(&acm->write_lock, flags);
Oliver Neukum884b6002005-04-21 21:28:02 +0200692 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700693 }
Oliver Neukum884b6002005-04-21 21:28:02 +0200694 wb = &acm->wb[wbn];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700695
Oliver Neukum884b6002005-04-21 21:28:02 +0200696 count = (count > acm->writesize) ? acm->writesize : count;
Johan Hovold5e9e75f2011-03-22 11:12:17 +0100697 dev_vdbg(&acm->data->dev, "%s - write %d\n", __func__, count);
Oliver Neukum884b6002005-04-21 21:28:02 +0200698 memcpy(wb->buf, buf, count);
699 wb->len = count;
700 spin_unlock_irqrestore(&acm->write_lock, flags);
701
Alan Cox6e47e062009-06-11 12:37:06 +0100702 stat = acm_write_start(acm, wbn);
703 if (stat < 0)
Oliver Neukum884b6002005-04-21 21:28:02 +0200704 return stat;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700705 return count;
706}
707
708static int acm_tty_write_room(struct tty_struct *tty)
709{
710 struct acm *acm = tty->driver_data;
711 if (!ACM_READY(acm))
712 return -EINVAL;
Oliver Neukum884b6002005-04-21 21:28:02 +0200713 /*
714 * Do not let the line discipline to know that we have a reserve,
715 * or it might get too enthusiastic.
716 */
David Brownell934da462008-08-06 18:44:12 -0700717 return acm_wb_is_avail(acm) ? acm->writesize : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700718}
719
720static int acm_tty_chars_in_buffer(struct tty_struct *tty)
721{
722 struct acm *acm = tty->driver_data;
723 if (!ACM_READY(acm))
Alan Cox23198fd2009-07-20 16:05:27 +0100724 return 0;
Oliver Neukum884b6002005-04-21 21:28:02 +0200725 /*
726 * This is inaccurate (overcounts), but it works.
727 */
Oliver Neukum86478942006-05-13 22:50:47 +0200728 return (ACM_NW - acm_wb_is_avail(acm)) * acm->writesize;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700729}
730
731static void acm_tty_throttle(struct tty_struct *tty)
732{
733 struct acm *acm = tty->driver_data;
734 if (!ACM_READY(acm))
735 return;
736 spin_lock_bh(&acm->throttle_lock);
737 acm->throttle = 1;
738 spin_unlock_bh(&acm->throttle_lock);
739}
740
741static void acm_tty_unthrottle(struct tty_struct *tty)
742{
743 struct acm *acm = tty->driver_data;
744 if (!ACM_READY(acm))
745 return;
746 spin_lock_bh(&acm->throttle_lock);
747 acm->throttle = 0;
748 spin_unlock_bh(&acm->throttle_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +0100749 tasklet_schedule(&acm->urb_task);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700750}
751
Alan Cox9e989662008-07-22 11:18:03 +0100752static int acm_tty_break_ctl(struct tty_struct *tty, int state)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700753{
754 struct acm *acm = tty->driver_data;
Alan Cox9e989662008-07-22 11:18:03 +0100755 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756 if (!ACM_READY(acm))
Alan Cox9e989662008-07-22 11:18:03 +0100757 return -EINVAL;
758 retval = acm_send_break(acm, state ? 0xffff : 0);
759 if (retval < 0)
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100760 dev_dbg(&acm->control->dev, "%s - send break failed\n",
761 __func__);
Alan Cox9e989662008-07-22 11:18:03 +0100762 return retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700763}
764
Alan Cox60b33c12011-02-14 16:26:14 +0000765static int acm_tty_tiocmget(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700766{
767 struct acm *acm = tty->driver_data;
768
769 if (!ACM_READY(acm))
770 return -EINVAL;
771
772 return (acm->ctrlout & ACM_CTRL_DTR ? TIOCM_DTR : 0) |
773 (acm->ctrlout & ACM_CTRL_RTS ? TIOCM_RTS : 0) |
774 (acm->ctrlin & ACM_CTRL_DSR ? TIOCM_DSR : 0) |
775 (acm->ctrlin & ACM_CTRL_RI ? TIOCM_RI : 0) |
776 (acm->ctrlin & ACM_CTRL_DCD ? TIOCM_CD : 0) |
777 TIOCM_CTS;
778}
779
Alan Cox20b9d172011-02-14 16:26:50 +0000780static int acm_tty_tiocmset(struct tty_struct *tty,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700781 unsigned int set, unsigned int clear)
782{
783 struct acm *acm = tty->driver_data;
784 unsigned int newctrl;
785
786 if (!ACM_READY(acm))
787 return -EINVAL;
788
789 newctrl = acm->ctrlout;
Alan Cox6e47e062009-06-11 12:37:06 +0100790 set = (set & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
791 (set & TIOCM_RTS ? ACM_CTRL_RTS : 0);
792 clear = (clear & TIOCM_DTR ? ACM_CTRL_DTR : 0) |
793 (clear & TIOCM_RTS ? ACM_CTRL_RTS : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700794
795 newctrl = (newctrl & ~clear) | set;
796
797 if (acm->ctrlout == newctrl)
798 return 0;
799 return acm_set_control(acm, acm->ctrlout = newctrl);
800}
801
Alan Cox6caa76b2011-02-14 16:27:22 +0000802static int acm_tty_ioctl(struct tty_struct *tty,
Alan Cox6e47e062009-06-11 12:37:06 +0100803 unsigned int cmd, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700804{
805 struct acm *acm = tty->driver_data;
806
807 if (!ACM_READY(acm))
808 return -EINVAL;
809
810 return -ENOIOCTLCMD;
811}
812
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100813static const __u32 acm_tty_speed[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700814 0, 50, 75, 110, 134, 150, 200, 300, 600,
815 1200, 1800, 2400, 4800, 9600, 19200, 38400,
816 57600, 115200, 230400, 460800, 500000, 576000,
817 921600, 1000000, 1152000, 1500000, 2000000,
818 2500000, 3000000, 3500000, 4000000
819};
820
Arjan van de Ven4c4c9432005-11-29 09:43:42 +0100821static const __u8 acm_tty_size[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700822 5, 6, 7, 8
823};
824
Alan Cox6e47e062009-06-11 12:37:06 +0100825static void acm_tty_set_termios(struct tty_struct *tty,
826 struct ktermios *termios_old)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700827{
828 struct acm *acm = tty->driver_data;
Alan Cox606d0992006-12-08 02:38:45 -0800829 struct ktermios *termios = tty->termios;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700830 struct usb_cdc_line_coding newline;
831 int newctrl = acm->ctrlout;
832
833 if (!ACM_READY(acm))
834 return;
835
Alan Cox9b80fee2009-09-19 13:13:23 -0700836 newline.dwDTERate = cpu_to_le32(tty_get_baud_rate(tty));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700837 newline.bCharFormat = termios->c_cflag & CSTOPB ? 2 : 0;
838 newline.bParityType = termios->c_cflag & PARENB ?
Alan Cox6e47e062009-06-11 12:37:06 +0100839 (termios->c_cflag & PARODD ? 1 : 2) +
840 (termios->c_cflag & CMSPAR ? 2 : 0) : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700841 newline.bDataBits = acm_tty_size[(termios->c_cflag & CSIZE) >> 4];
Alan Cox6e47e062009-06-11 12:37:06 +0100842 /* FIXME: Needs to clear unsupported bits in the termios */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700843 acm->clocal = ((termios->c_cflag & CLOCAL) != 0);
844
845 if (!newline.dwDTERate) {
846 newline.dwDTERate = acm->line.dwDTERate;
847 newctrl &= ~ACM_CTRL_DTR;
Alan Cox6e47e062009-06-11 12:37:06 +0100848 } else
849 newctrl |= ACM_CTRL_DTR;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850
851 if (newctrl != acm->ctrlout)
852 acm_set_control(acm, acm->ctrlout = newctrl);
853
854 if (memcmp(&acm->line, &newline, sizeof newline)) {
855 memcpy(&acm->line, &newline, sizeof newline);
Johan Hovolda5cc7ef2011-03-22 11:12:15 +0100856 dev_dbg(&acm->control->dev, "%s - set line: %d %d %d %d\n",
857 __func__,
858 le32_to_cpu(newline.dwDTERate),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700859 newline.bCharFormat, newline.bParityType,
860 newline.bDataBits);
861 acm_set_line(acm, &acm->line);
862 }
863}
864
865/*
866 * USB probe and disconnect routines.
867 */
868
Oliver Neukum830f4022008-06-25 14:17:16 +0200869/* Little helpers: write/read buffers free */
Oliver Neukum884b6002005-04-21 21:28:02 +0200870static void acm_write_buffers_free(struct acm *acm)
871{
872 int i;
873 struct acm_wb *wb;
Oliver Neukuma496c642008-10-21 10:39:04 +0200874 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
Oliver Neukum884b6002005-04-21 21:28:02 +0200875
Alan Cox6e47e062009-06-11 12:37:06 +0100876 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++)
Daniel Mack997ea582010-04-12 13:17:25 +0200877 usb_free_coherent(usb_dev, acm->writesize, wb->buf, wb->dmah);
Oliver Neukum884b6002005-04-21 21:28:02 +0200878}
879
Oliver Neukum830f4022008-06-25 14:17:16 +0200880static void acm_read_buffers_free(struct acm *acm)
881{
882 struct usb_device *usb_dev = interface_to_usbdev(acm->control);
Johan Hovolddab54c92011-03-22 11:12:21 +0100883 int i;
Oliver Neukum830f4022008-06-25 14:17:16 +0200884
Johan Hovolddab54c92011-03-22 11:12:21 +0100885 for (i = 0; i < acm->rx_buflimit; i++)
Daniel Mack997ea582010-04-12 13:17:25 +0200886 usb_free_coherent(usb_dev, acm->readsize,
887 acm->rb[i].base, acm->rb[i].dma);
Oliver Neukum830f4022008-06-25 14:17:16 +0200888}
889
Oliver Neukum884b6002005-04-21 21:28:02 +0200890/* Little helper: write buffers allocate */
891static int acm_write_buffers_alloc(struct acm *acm)
892{
893 int i;
894 struct acm_wb *wb;
895
Oliver Neukum86478942006-05-13 22:50:47 +0200896 for (wb = &acm->wb[0], i = 0; i < ACM_NW; i++, wb++) {
Daniel Mack997ea582010-04-12 13:17:25 +0200897 wb->buf = usb_alloc_coherent(acm->dev, acm->writesize, GFP_KERNEL,
Oliver Neukum884b6002005-04-21 21:28:02 +0200898 &wb->dmah);
899 if (!wb->buf) {
900 while (i != 0) {
901 --i;
902 --wb;
Daniel Mack997ea582010-04-12 13:17:25 +0200903 usb_free_coherent(acm->dev, acm->writesize,
Oliver Neukum884b6002005-04-21 21:28:02 +0200904 wb->buf, wb->dmah);
905 }
906 return -ENOMEM;
907 }
908 }
909 return 0;
910}
911
Alan Cox10077d42009-06-11 12:36:09 +0100912static int acm_probe(struct usb_interface *intf,
913 const struct usb_device_id *id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700914{
915 struct usb_cdc_union_desc *union_header = NULL;
Oliver Neukumc4cabd22007-02-27 15:28:55 +0100916 struct usb_cdc_country_functional_desc *cfd = NULL;
David Brownellc6dbf552008-04-13 14:00:44 -0700917 unsigned char *buffer = intf->altsetting->extra;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700918 int buflen = intf->altsetting->extralen;
919 struct usb_interface *control_interface;
920 struct usb_interface *data_interface;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +0200921 struct usb_endpoint_descriptor *epctrl = NULL;
922 struct usb_endpoint_descriptor *epread = NULL;
923 struct usb_endpoint_descriptor *epwrite = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700924 struct usb_device *usb_dev = interface_to_usbdev(intf);
925 struct acm *acm;
926 int minor;
Alan Cox6e47e062009-06-11 12:37:06 +0100927 int ctrlsize, readsize;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700928 u8 *buf;
929 u8 ac_management_function = 0;
930 u8 call_management_function = 0;
931 int call_interface_num = -1;
932 int data_interface_num;
933 unsigned long quirks;
Oliver Neukum86478942006-05-13 22:50:47 +0200934 int num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +0100935 int i;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +0200936 int combined_interfaces = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700937
Oliver Neukum86478942006-05-13 22:50:47 +0200938 /* normal quirks */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700939 quirks = (unsigned long)id->driver_info;
Oliver Neukum86478942006-05-13 22:50:47 +0200940 num_rx_buf = (quirks == SINGLE_RX_URB) ? 1 : ACM_NR;
941
942 /* handle quirks deadly to normal probing*/
Linus Torvalds1da177e2005-04-16 15:20:36 -0700943 if (quirks == NO_UNION_NORMAL) {
944 data_interface = usb_ifnum_to_if(usb_dev, 1);
945 control_interface = usb_ifnum_to_if(usb_dev, 0);
946 goto skip_normal_probe;
947 }
Alan Cox6e47e062009-06-11 12:37:06 +0100948
Linus Torvalds1da177e2005-04-16 15:20:36 -0700949 /* normal probing*/
950 if (!buffer) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700951 dev_err(&intf->dev, "Weird descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952 return -EINVAL;
953 }
954
955 if (!buflen) {
Toby Gray577045c2010-09-02 10:46:20 +0100956 if (intf->cur_altsetting->endpoint &&
957 intf->cur_altsetting->endpoint->extralen &&
Alan Cox6e47e062009-06-11 12:37:06 +0100958 intf->cur_altsetting->endpoint->extra) {
959 dev_dbg(&intf->dev,
960 "Seeking extra descriptors on endpoint\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700961 buflen = intf->cur_altsetting->endpoint->extralen;
962 buffer = intf->cur_altsetting->endpoint->extra;
963 } else {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700964 dev_err(&intf->dev,
965 "Zero length descriptor references\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700966 return -EINVAL;
967 }
968 }
969
970 while (buflen > 0) {
Alan Cox6e47e062009-06-11 12:37:06 +0100971 if (buffer[1] != USB_DT_CS_INTERFACE) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -0700972 dev_err(&intf->dev, "skipping garbage\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700973 goto next_desc;
974 }
975
Alan Cox6e47e062009-06-11 12:37:06 +0100976 switch (buffer[2]) {
977 case USB_CDC_UNION_TYPE: /* we've found it */
978 if (union_header) {
979 dev_err(&intf->dev, "More than one "
980 "union descriptor, skipping ...\n");
981 goto next_desc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700982 }
Alan Cox6e47e062009-06-11 12:37:06 +0100983 union_header = (struct usb_cdc_union_desc *)buffer;
984 break;
985 case USB_CDC_COUNTRY_TYPE: /* export through sysfs*/
986 cfd = (struct usb_cdc_country_functional_desc *)buffer;
987 break;
988 case USB_CDC_HEADER_TYPE: /* maybe check version */
989 break; /* for now we ignore it */
990 case USB_CDC_ACM_TYPE:
991 ac_management_function = buffer[3];
992 break;
993 case USB_CDC_CALL_MANAGEMENT_TYPE:
994 call_management_function = buffer[3];
995 call_interface_num = buffer[4];
Julian Calabyce126642010-01-05 23:58:20 +1100996 if ( (quirks & NOT_A_MODEM) == 0 && (call_management_function & 3) != 3)
Alan Cox6e47e062009-06-11 12:37:06 +0100997 dev_err(&intf->dev, "This device cannot do calls on its own. It is not a modem.\n");
998 break;
999 default:
1000 /* there are LOTS more CDC descriptors that
1001 * could legitimately be found here.
1002 */
1003 dev_dbg(&intf->dev, "Ignoring descriptor: "
1004 "type %02x, length %d\n",
1005 buffer[2], buffer[0]);
1006 break;
1007 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001008next_desc:
1009 buflen -= buffer[0];
1010 buffer += buffer[0];
1011 }
1012
1013 if (!union_header) {
1014 if (call_interface_num > 0) {
Alan Cox6e47e062009-06-11 12:37:06 +01001015 dev_dbg(&intf->dev, "No union descriptor, using call management descriptor\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001016 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = call_interface_num));
1017 control_interface = intf;
1018 } else {
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001019 if (intf->cur_altsetting->desc.bNumEndpoints != 3) {
1020 dev_dbg(&intf->dev,"No union descriptor, giving up\n");
1021 return -ENODEV;
1022 } else {
1023 dev_warn(&intf->dev,"No union descriptor, testing for castrated device\n");
1024 combined_interfaces = 1;
1025 control_interface = data_interface = intf;
1026 goto look_for_collapsed_interface;
1027 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001028 }
1029 } else {
1030 control_interface = usb_ifnum_to_if(usb_dev, union_header->bMasterInterface0);
1031 data_interface = usb_ifnum_to_if(usb_dev, (data_interface_num = union_header->bSlaveInterface0));
1032 if (!control_interface || !data_interface) {
Alan Cox6e47e062009-06-11 12:37:06 +01001033 dev_dbg(&intf->dev, "no interfaces\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001034 return -ENODEV;
1035 }
1036 }
Alan Cox6e47e062009-06-11 12:37:06 +01001037
Linus Torvalds1da177e2005-04-16 15:20:36 -07001038 if (data_interface_num != call_interface_num)
Alan Cox6e47e062009-06-11 12:37:06 +01001039 dev_dbg(&intf->dev, "Separate call control interface. That is not fully supported.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001040
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001041 if (control_interface == data_interface) {
1042 /* some broken devices designed for windows work this way */
1043 dev_warn(&intf->dev,"Control and data interfaces are not separated!\n");
1044 combined_interfaces = 1;
1045 /* a popular other OS doesn't use it */
1046 quirks |= NO_CAP_LINE;
1047 if (data_interface->cur_altsetting->desc.bNumEndpoints != 3) {
1048 dev_err(&intf->dev, "This needs exactly 3 endpoints\n");
1049 return -EINVAL;
1050 }
1051look_for_collapsed_interface:
1052 for (i = 0; i < 3; i++) {
1053 struct usb_endpoint_descriptor *ep;
1054 ep = &data_interface->cur_altsetting->endpoint[i].desc;
1055
1056 if (usb_endpoint_is_int_in(ep))
1057 epctrl = ep;
1058 else if (usb_endpoint_is_bulk_out(ep))
1059 epwrite = ep;
1060 else if (usb_endpoint_is_bulk_in(ep))
1061 epread = ep;
1062 else
1063 return -EINVAL;
1064 }
1065 if (!epctrl || !epread || !epwrite)
1066 return -ENODEV;
1067 else
1068 goto made_compressed_probe;
1069 }
1070
Linus Torvalds1da177e2005-04-16 15:20:36 -07001071skip_normal_probe:
1072
1073 /*workaround for switched interfaces */
Alan Cox6e47e062009-06-11 12:37:06 +01001074 if (data_interface->cur_altsetting->desc.bInterfaceClass
1075 != CDC_DATA_INTERFACE_TYPE) {
1076 if (control_interface->cur_altsetting->desc.bInterfaceClass
1077 == CDC_DATA_INTERFACE_TYPE) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001078 struct usb_interface *t;
Alan Cox6e47e062009-06-11 12:37:06 +01001079 dev_dbg(&intf->dev,
1080 "Your device has switched interfaces.\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001081 t = control_interface;
1082 control_interface = data_interface;
1083 data_interface = t;
1084 } else {
1085 return -EINVAL;
1086 }
1087 }
Alan Stern74da5d62007-08-02 13:29:10 -04001088
1089 /* Accept probe requests only for the control interface */
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001090 if (!combined_interfaces && intf != control_interface)
Alan Stern74da5d62007-08-02 13:29:10 -04001091 return -ENODEV;
Alan Cox6e47e062009-06-11 12:37:06 +01001092
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001093 if (!combined_interfaces && usb_interface_claimed(data_interface)) {
1094 /* valid in this context */
Alan Cox6e47e062009-06-11 12:37:06 +01001095 dev_dbg(&intf->dev, "The data interface isn't available\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001096 return -EBUSY;
1097 }
1098
1099
1100 if (data_interface->cur_altsetting->desc.bNumEndpoints < 2)
1101 return -EINVAL;
1102
1103 epctrl = &control_interface->cur_altsetting->endpoint[0].desc;
1104 epread = &data_interface->cur_altsetting->endpoint[0].desc;
1105 epwrite = &data_interface->cur_altsetting->endpoint[1].desc;
1106
1107
1108 /* workaround for switched endpoints */
Luiz Fernando N. Capitulino45aea702006-10-26 13:02:48 -03001109 if (!usb_endpoint_dir_in(epread)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001110 /* descriptors are swapped */
1111 struct usb_endpoint_descriptor *t;
Alan Cox6e47e062009-06-11 12:37:06 +01001112 dev_dbg(&intf->dev,
1113 "The data interface has switched endpoints\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001114 t = epread;
1115 epread = epwrite;
1116 epwrite = t;
1117 }
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001118made_compressed_probe:
Johan Hovolda5cc7ef2011-03-22 11:12:15 +01001119 dev_dbg(&intf->dev, "interfaces are valid\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001120 for (minor = 0; minor < ACM_TTY_MINORS && acm_table[minor]; minor++);
1121
1122 if (minor == ACM_TTY_MINORS) {
Greg Kroah-Hartman9908a322008-08-14 09:37:34 -07001123 dev_err(&intf->dev, "no more free acm devices\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001124 return -ENODEV;
1125 }
1126
Alan Cox6e47e062009-06-11 12:37:06 +01001127 acm = kzalloc(sizeof(struct acm), GFP_KERNEL);
1128 if (acm == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001129 dev_err(&intf->dev, "out of memory (acm kzalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001130 goto alloc_fail;
1131 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001132
1133 ctrlsize = le16_to_cpu(epctrl->wMaxPacketSize);
Alan Cox6e47e062009-06-11 12:37:06 +01001134 readsize = le16_to_cpu(epread->wMaxPacketSize) *
1135 (quirks == SINGLE_RX_URB ? 1 : 2);
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001136 acm->combined_interfaces = combined_interfaces;
David Engrafe4cf3aa2008-03-20 10:01:34 +01001137 acm->writesize = le16_to_cpu(epwrite->wMaxPacketSize) * 20;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001138 acm->control = control_interface;
1139 acm->data = data_interface;
1140 acm->minor = minor;
1141 acm->dev = usb_dev;
1142 acm->ctrl_caps = ac_management_function;
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001143 if (quirks & NO_CAP_LINE)
1144 acm->ctrl_caps &= ~USB_CDC_CAP_LINE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001145 acm->ctrlsize = ctrlsize;
1146 acm->readsize = readsize;
Oliver Neukum86478942006-05-13 22:50:47 +02001147 acm->rx_buflimit = num_rx_buf;
David Kubicek61a87ad2005-11-01 18:51:34 +01001148 acm->urb_task.func = acm_rx_tasklet;
1149 acm->urb_task.data = (unsigned long) acm;
David Howellsc4028952006-11-22 14:57:56 +00001150 INIT_WORK(&acm->work, acm_softint);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001151 spin_lock_init(&acm->throttle_lock);
Oliver Neukum884b6002005-04-21 21:28:02 +02001152 spin_lock_init(&acm->write_lock);
David Kubicek61a87ad2005-11-01 18:51:34 +01001153 spin_lock_init(&acm->read_lock);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001154 mutex_init(&acm->mutex);
David Kubicek61a87ad2005-11-01 18:51:34 +01001155 acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress);
Oliver Neukumcf7fdd52009-08-04 23:52:09 +02001156 acm->is_int_ep = usb_endpoint_xfer_int(epread);
1157 if (acm->is_int_ep)
1158 acm->bInterval = epread->bInterval;
Alan Cox739e0282009-06-11 12:27:50 +01001159 tty_port_init(&acm->port);
1160 acm->port.ops = &acm_port_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161
Daniel Mack997ea582010-04-12 13:17:25 +02001162 buf = usb_alloc_coherent(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001163 if (!buf) {
Johan Hovold255ab562011-03-22 11:12:13 +01001164 dev_err(&intf->dev, "out of memory (ctrl buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001165 goto alloc_fail2;
1166 }
1167 acm->ctrl_buffer = buf;
1168
Oliver Neukum884b6002005-04-21 21:28:02 +02001169 if (acm_write_buffers_alloc(acm) < 0) {
Johan Hovold255ab562011-03-22 11:12:13 +01001170 dev_err(&intf->dev, "out of memory (write buffer alloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001171 goto alloc_fail4;
1172 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001173
1174 acm->ctrlurb = usb_alloc_urb(0, GFP_KERNEL);
1175 if (!acm->ctrlurb) {
Johan Hovold255ab562011-03-22 11:12:13 +01001176 dev_err(&intf->dev, "out of memory (ctrlurb kmalloc)\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001177 goto alloc_fail5;
1178 }
Oliver Neukum86478942006-05-13 22:50:47 +02001179 for (i = 0; i < num_rx_buf; i++) {
David Kubicek61a87ad2005-11-01 18:51:34 +01001180 struct acm_ru *rcv = &(acm->ru[i]);
1181
Alan Cox6e47e062009-06-11 12:37:06 +01001182 rcv->urb = usb_alloc_urb(0, GFP_KERNEL);
1183 if (rcv->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001184 dev_err(&intf->dev,
Alan Cox6e47e062009-06-11 12:37:06 +01001185 "out of memory (read urbs usb_alloc_urb)\n");
Axel Linc2572b72010-05-31 08:04:47 +08001186 goto alloc_fail6;
David Kubicek61a87ad2005-11-01 18:51:34 +01001187 }
1188
1189 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1190 rcv->instance = acm;
1191 }
Oliver Neukum86478942006-05-13 22:50:47 +02001192 for (i = 0; i < num_rx_buf; i++) {
David Brownell672c4e12008-08-06 18:41:12 -07001193 struct acm_rb *rb = &(acm->rb[i]);
David Kubicek61a87ad2005-11-01 18:51:34 +01001194
Daniel Mack997ea582010-04-12 13:17:25 +02001195 rb->base = usb_alloc_coherent(acm->dev, readsize,
David Brownell672c4e12008-08-06 18:41:12 -07001196 GFP_KERNEL, &rb->dma);
1197 if (!rb->base) {
Johan Hovold255ab562011-03-22 11:12:13 +01001198 dev_err(&intf->dev,
Daniel Mack997ea582010-04-12 13:17:25 +02001199 "out of memory (read bufs usb_alloc_coherent)\n");
David Kubicek61a87ad2005-11-01 18:51:34 +01001200 goto alloc_fail7;
1201 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001202 }
Alan Cox6e47e062009-06-11 12:37:06 +01001203 for (i = 0; i < ACM_NW; i++) {
David Engrafe4cf3aa2008-03-20 10:01:34 +01001204 struct acm_wb *snd = &(acm->wb[i]);
1205
Alan Cox6e47e062009-06-11 12:37:06 +01001206 snd->urb = usb_alloc_urb(0, GFP_KERNEL);
1207 if (snd->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001208 dev_err(&intf->dev,
Johan Hovold59d7fec2011-03-22 11:12:12 +01001209 "out of memory (write urbs usb_alloc_urb)\n");
Axel Linc2572b72010-05-31 08:04:47 +08001210 goto alloc_fail8;
David Engrafe4cf3aa2008-03-20 10:01:34 +01001211 }
1212
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +04001213 if (usb_endpoint_xfer_int(epwrite))
1214 usb_fill_int_urb(snd->urb, usb_dev,
1215 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1216 NULL, acm->writesize, acm_write_bulk, snd, epwrite->bInterval);
1217 else
1218 usb_fill_bulk_urb(snd->urb, usb_dev,
1219 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1220 NULL, acm->writesize, acm_write_bulk, snd);
David Engrafe4cf3aa2008-03-20 10:01:34 +01001221 snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1222 snd->instance = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001223 }
1224
Alan Cox6e47e062009-06-11 12:37:06 +01001225 usb_set_intfdata(intf, acm);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001226
1227 i = device_create_file(&intf->dev, &dev_attr_bmCapabilities);
1228 if (i < 0)
1229 goto alloc_fail8;
1230
1231 if (cfd) { /* export the country data */
1232 acm->country_codes = kmalloc(cfd->bLength - 4, GFP_KERNEL);
1233 if (!acm->country_codes)
1234 goto skip_countries;
1235 acm->country_code_size = cfd->bLength - 4;
Alan Cox6e47e062009-06-11 12:37:06 +01001236 memcpy(acm->country_codes, (u8 *)&cfd->wCountyCode0,
1237 cfd->bLength - 4);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001238 acm->country_rel_date = cfd->iCountryCodeRelDate;
1239
1240 i = device_create_file(&intf->dev, &dev_attr_wCountryCodes);
1241 if (i < 0) {
1242 kfree(acm->country_codes);
1243 goto skip_countries;
1244 }
1245
Alan Cox6e47e062009-06-11 12:37:06 +01001246 i = device_create_file(&intf->dev,
1247 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001248 if (i < 0) {
Axel Linc2572b72010-05-31 08:04:47 +08001249 device_remove_file(&intf->dev, &dev_attr_wCountryCodes);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001250 kfree(acm->country_codes);
1251 goto skip_countries;
1252 }
1253 }
1254
1255skip_countries:
Alan Cox6e47e062009-06-11 12:37:06 +01001256 usb_fill_int_urb(acm->ctrlurb, usb_dev,
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001257 usb_rcvintpipe(usb_dev, epctrl->bEndpointAddress),
1258 acm->ctrl_buffer, ctrlsize, acm_ctrl_irq, acm,
1259 /* works around buggy devices */
1260 epctrl->bInterval ? epctrl->bInterval : 0xff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001261 acm->ctrlurb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1262 acm->ctrlurb->transfer_dma = acm->ctrl_dma;
1263
Linus Torvalds1da177e2005-04-16 15:20:36 -07001264 dev_info(&intf->dev, "ttyACM%d: USB ACM device\n", minor);
1265
1266 acm_set_control(acm, acm->ctrlout);
1267
1268 acm->line.dwDTERate = cpu_to_le32(9600);
1269 acm->line.bDataBits = 8;
1270 acm_set_line(acm, &acm->line);
1271
1272 usb_driver_claim_interface(&acm_driver, data_interface, acm);
David Brownell672c4e12008-08-06 18:41:12 -07001273 usb_set_intfdata(data_interface, acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001274
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001275 usb_get_intf(control_interface);
1276 tty_register_device(acm_tty_driver, minor, &control_interface->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277
1278 acm_table[minor] = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001279
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001280 return 0;
1281alloc_fail8:
David Engrafe4cf3aa2008-03-20 10:01:34 +01001282 for (i = 0; i < ACM_NW; i++)
1283 usb_free_urb(acm->wb[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001284alloc_fail7:
Oliver Neukum830f4022008-06-25 14:17:16 +02001285 acm_read_buffers_free(acm);
Axel Linc2572b72010-05-31 08:04:47 +08001286alloc_fail6:
Oliver Neukum86478942006-05-13 22:50:47 +02001287 for (i = 0; i < num_rx_buf; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +01001288 usb_free_urb(acm->ru[i].urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001289 usb_free_urb(acm->ctrlurb);
1290alloc_fail5:
Oliver Neukum884b6002005-04-21 21:28:02 +02001291 acm_write_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001292alloc_fail4:
Daniel Mack997ea582010-04-12 13:17:25 +02001293 usb_free_coherent(usb_dev, ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001294alloc_fail2:
1295 kfree(acm);
1296alloc_fail:
1297 return -ENOMEM;
1298}
1299
Oliver Neukum1365baf72007-10-12 17:24:28 +02001300static void stop_data_traffic(struct acm *acm)
1301{
1302 int i;
Johan Hovolda5cc7ef2011-03-22 11:12:15 +01001303
1304 dev_dbg(&acm->control->dev, "%s\n", __func__);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001305
1306 tasklet_disable(&acm->urb_task);
1307
1308 usb_kill_urb(acm->ctrlurb);
Alan Cox6e47e062009-06-11 12:37:06 +01001309 for (i = 0; i < ACM_NW; i++)
David Engrafe4cf3aa2008-03-20 10:01:34 +01001310 usb_kill_urb(acm->wb[i].urb);
Oliver Neukum1365baf72007-10-12 17:24:28 +02001311 for (i = 0; i < acm->rx_buflimit; i++)
1312 usb_kill_urb(acm->ru[i].urb);
1313
Oliver Neukum1365baf72007-10-12 17:24:28 +02001314 tasklet_enable(&acm->urb_task);
1315
1316 cancel_work_sync(&acm->work);
1317}
1318
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319static void acm_disconnect(struct usb_interface *intf)
1320{
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001321 struct acm *acm = usb_get_intfdata(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001322 struct usb_device *usb_dev = interface_to_usbdev(intf);
Alan Cox10077d42009-06-11 12:36:09 +01001323 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001324
David Brownell672c4e12008-08-06 18:41:12 -07001325 /* sibling interface is already cleaning up */
1326 if (!acm)
Oliver Neukum86067eea2006-01-08 12:39:13 +01001327 return;
David Brownell672c4e12008-08-06 18:41:12 -07001328
1329 mutex_lock(&open_mutex);
Alan Cox6e47e062009-06-11 12:37:06 +01001330 if (acm->country_codes) {
Alan Stern74da5d62007-08-02 13:29:10 -04001331 device_remove_file(&acm->control->dev,
1332 &dev_attr_wCountryCodes);
1333 device_remove_file(&acm->control->dev,
1334 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001335 }
Alan Stern74da5d62007-08-02 13:29:10 -04001336 device_remove_file(&acm->control->dev, &dev_attr_bmCapabilities);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001337 acm->dev = NULL;
Oliver Neukum86067eea2006-01-08 12:39:13 +01001338 usb_set_intfdata(acm->control, NULL);
1339 usb_set_intfdata(acm->data, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001340
Oliver Neukum1365baf72007-10-12 17:24:28 +02001341 stop_data_traffic(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001342
Oliver Neukum884b6002005-04-21 21:28:02 +02001343 acm_write_buffers_free(acm);
Daniel Mack997ea582010-04-12 13:17:25 +02001344 usb_free_coherent(usb_dev, acm->ctrlsize, acm->ctrl_buffer,
1345 acm->ctrl_dma);
Oliver Neukum830f4022008-06-25 14:17:16 +02001346 acm_read_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001347
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001348 if (!acm->combined_interfaces)
1349 usb_driver_release_interface(&acm_driver, intf == acm->control ?
Oliver Neukum830f4022008-06-25 14:17:16 +02001350 acm->data : acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001351
Alan Cox10077d42009-06-11 12:36:09 +01001352 if (acm->port.count == 0) {
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001353 acm_tty_unregister(acm);
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001354 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001355 return;
1356 }
1357
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001358 mutex_unlock(&open_mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001359 tty = tty_port_tty_get(&acm->port);
1360 if (tty) {
1361 tty_hangup(tty);
1362 tty_kref_put(tty);
1363 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001364}
1365
Oliver Neukum35758582008-07-01 19:10:08 +02001366#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001367static int acm_suspend(struct usb_interface *intf, pm_message_t message)
1368{
1369 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001370 int cnt;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001371
Alan Stern65bfd292008-11-25 16:39:18 -05001372 if (message.event & PM_EVENT_AUTO) {
Oliver Neukum11ea8592008-06-20 11:25:57 +02001373 int b;
1374
1375 spin_lock_irq(&acm->read_lock);
1376 spin_lock(&acm->write_lock);
1377 b = acm->processing + acm->transmitting;
1378 spin_unlock(&acm->write_lock);
1379 spin_unlock_irq(&acm->read_lock);
1380 if (b)
1381 return -EBUSY;
1382 }
1383
1384 spin_lock_irq(&acm->read_lock);
1385 spin_lock(&acm->write_lock);
1386 cnt = acm->susp_count++;
1387 spin_unlock(&acm->write_lock);
1388 spin_unlock_irq(&acm->read_lock);
1389
1390 if (cnt)
Oliver Neukum1365baf72007-10-12 17:24:28 +02001391 return 0;
1392 /*
1393 we treat opened interfaces differently,
1394 we must guard against open
1395 */
1396 mutex_lock(&acm->mutex);
1397
Alan Cox10077d42009-06-11 12:36:09 +01001398 if (acm->port.count)
Oliver Neukum1365baf72007-10-12 17:24:28 +02001399 stop_data_traffic(acm);
1400
1401 mutex_unlock(&acm->mutex);
1402 return 0;
1403}
1404
1405static int acm_resume(struct usb_interface *intf)
1406{
1407 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001408 struct acm_wb *wb;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001409 int rv = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +02001410 int cnt;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001411
Oliver Neukum11ea8592008-06-20 11:25:57 +02001412 spin_lock_irq(&acm->read_lock);
1413 acm->susp_count -= 1;
1414 cnt = acm->susp_count;
1415 spin_unlock_irq(&acm->read_lock);
1416
1417 if (cnt)
Oliver Neukum1365baf72007-10-12 17:24:28 +02001418 return 0;
1419
1420 mutex_lock(&acm->mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001421 if (acm->port.count) {
Oliver Neukum1365baf72007-10-12 17:24:28 +02001422 rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001423
1424 spin_lock_irq(&acm->write_lock);
1425 if (acm->delayed_wb) {
1426 wb = acm->delayed_wb;
1427 acm->delayed_wb = NULL;
1428 spin_unlock_irq(&acm->write_lock);
Oliver Neukumf0730922010-03-03 00:37:56 +01001429 acm_start_wb(acm, wb);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001430 } else {
1431 spin_unlock_irq(&acm->write_lock);
1432 }
1433
1434 /*
1435 * delayed error checking because we must
1436 * do the write path at all cost
1437 */
Oliver Neukum1365baf72007-10-12 17:24:28 +02001438 if (rv < 0)
Oliver Neukum11ea8592008-06-20 11:25:57 +02001439 goto err_out;
Oliver Neukum1365baf72007-10-12 17:24:28 +02001440
1441 tasklet_schedule(&acm->urb_task);
1442 }
1443
1444err_out:
1445 mutex_unlock(&acm->mutex);
1446 return rv;
1447}
Oliver Neukum35758582008-07-01 19:10:08 +02001448
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001449static int acm_reset_resume(struct usb_interface *intf)
1450{
1451 struct acm *acm = usb_get_intfdata(intf);
1452 struct tty_struct *tty;
1453
1454 mutex_lock(&acm->mutex);
1455 if (acm->port.count) {
1456 tty = tty_port_tty_get(&acm->port);
1457 if (tty) {
1458 tty_hangup(tty);
1459 tty_kref_put(tty);
1460 }
1461 }
1462 mutex_unlock(&acm->mutex);
1463 return acm_resume(intf);
1464}
1465
Oliver Neukum35758582008-07-01 19:10:08 +02001466#endif /* CONFIG_PM */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001467
1468#define NOKIA_PCSUITE_ACM_INFO(x) \
1469 USB_DEVICE_AND_INTERFACE_INFO(0x0421, x, \
1470 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1471 USB_CDC_ACM_PROTO_VENDOR)
1472
Toby Gray4035e452010-09-01 16:01:19 +01001473#define SAMSUNG_PCSUITE_ACM_INFO(x) \
1474 USB_DEVICE_AND_INTERFACE_INFO(0x04e7, x, \
1475 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1476 USB_CDC_ACM_PROTO_VENDOR)
1477
Linus Torvalds1da177e2005-04-16 15:20:36 -07001478/*
1479 * USB driver structure.
1480 */
1481
Németh Márton6ef48522010-01-10 15:33:45 +01001482static const struct usb_device_id acm_ids[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001483 /* quirky and broken devices */
1484 { USB_DEVICE(0x0870, 0x0001), /* Metricom GS Modem */
1485 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1486 },
Andrey Arapovb0e2a702007-07-04 17:11:42 +02001487 { USB_DEVICE(0x0e8d, 0x0003), /* FIREFLY, MediaTek Inc; andrey.arapov@gmail.com */
1488 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1489 },
Andrew Lunn0f9c7b42008-12-23 17:31:23 +01001490 { USB_DEVICE(0x0e8d, 0x3329), /* MediaTek Inc GPS */
1491 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1492 },
Masahito Omote8753e652005-07-29 12:17:25 -07001493 { USB_DEVICE(0x0482, 0x0203), /* KYOCERA AH-K3001V */
1494 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1495 },
Chris Malley91a9c922006-10-03 10:08:28 +01001496 { USB_DEVICE(0x079b, 0x000f), /* BT On-Air USB MODEM */
1497 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1498 },
Alan Cox7abcf202009-04-06 17:35:01 +01001499 { USB_DEVICE(0x0ace, 0x1602), /* ZyDAS 56K USB MODEM */
1500 .driver_info = SINGLE_RX_URB,
1501 },
Oliver Neukum86478942006-05-13 22:50:47 +02001502 { USB_DEVICE(0x0ace, 0x1608), /* ZyDAS 56K USB MODEM */
1503 .driver_info = SINGLE_RX_URB, /* firmware bug */
1504 },
Oliver Neukum3dd2ae82006-06-23 09:14:17 +02001505 { USB_DEVICE(0x0ace, 0x1611), /* ZyDAS 56K USB MODEM - new version */
1506 .driver_info = SINGLE_RX_URB, /* firmware bug */
1507 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001508 { USB_DEVICE(0x22b8, 0x7000), /* Motorola Q Phone */
1509 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1510 },
Iain McFarlane6149ed52008-05-04 00:13:49 +01001511 { USB_DEVICE(0x0803, 0x3095), /* Zoom Telephonics Model 3095F USB MODEM */
1512 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1513 },
Eric Sandeenc8fd2c32008-08-14 08:25:40 -05001514 { USB_DEVICE(0x0572, 0x1321), /* Conexant USB MODEM CX93010 */
1515 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1516 },
Alan Coxc89c60e2009-01-11 19:53:10 +00001517 { USB_DEVICE(0x0572, 0x1324), /* Conexant USB MODEM RD02-D400 */
1518 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1519 },
Xiao Kaijiancab98a02009-05-08 00:48:23 +08001520 { USB_DEVICE(0x0572, 0x1328), /* Shiro / Aztech USB MODEM UM-3100 */
1521 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1522 },
Dmitriy Taychenachev155df652009-02-25 12:36:51 +08001523 { USB_DEVICE(0x22b8, 0x6425), /* Motorola MOTOMAGX phones */
1524 },
Adam Richterc332b4e2009-02-18 16:17:15 -08001525 { USB_DEVICE(0x0572, 0x1329), /* Hummingbird huc56s (Conexant) */
1526 .driver_info = NO_UNION_NORMAL, /* union descriptor misplaced on
1527 data interface instead of
1528 communications interface.
1529 Maybe we should define a new
1530 quirk for this. */
1531 },
Kir Kolyshkin1f17c502009-05-28 20:33:58 +04001532 { USB_DEVICE(0x1bbb, 0x0003), /* Alcatel OT-I650 */
1533 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1534 },
Russ Nelsonc3baa192010-04-21 23:07:03 -04001535 { USB_DEVICE(0x1576, 0x03b1), /* Maretron USB100 */
1536 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1537 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001538
Adrian Taylorc1479a92009-11-19 10:35:33 +00001539 /* Nokia S60 phones expose two ACM channels. The first is
1540 * a modem and is picked up by the standard AT-command
1541 * information below. The second is 'vendor-specific' but
1542 * is treated as a serial device at the S60 end, so we want
1543 * to expose it on Linux too. */
1544 { NOKIA_PCSUITE_ACM_INFO(0x042D), }, /* Nokia 3250 */
1545 { NOKIA_PCSUITE_ACM_INFO(0x04D8), }, /* Nokia 5500 Sport */
1546 { NOKIA_PCSUITE_ACM_INFO(0x04C9), }, /* Nokia E50 */
1547 { NOKIA_PCSUITE_ACM_INFO(0x0419), }, /* Nokia E60 */
1548 { NOKIA_PCSUITE_ACM_INFO(0x044D), }, /* Nokia E61 */
1549 { NOKIA_PCSUITE_ACM_INFO(0x0001), }, /* Nokia E61i */
1550 { NOKIA_PCSUITE_ACM_INFO(0x0475), }, /* Nokia E62 */
1551 { NOKIA_PCSUITE_ACM_INFO(0x0508), }, /* Nokia E65 */
1552 { NOKIA_PCSUITE_ACM_INFO(0x0418), }, /* Nokia E70 */
1553 { NOKIA_PCSUITE_ACM_INFO(0x0425), }, /* Nokia N71 */
1554 { NOKIA_PCSUITE_ACM_INFO(0x0486), }, /* Nokia N73 */
1555 { NOKIA_PCSUITE_ACM_INFO(0x04DF), }, /* Nokia N75 */
1556 { NOKIA_PCSUITE_ACM_INFO(0x000e), }, /* Nokia N77 */
1557 { NOKIA_PCSUITE_ACM_INFO(0x0445), }, /* Nokia N80 */
1558 { NOKIA_PCSUITE_ACM_INFO(0x042F), }, /* Nokia N91 & N91 8GB */
1559 { NOKIA_PCSUITE_ACM_INFO(0x048E), }, /* Nokia N92 */
1560 { NOKIA_PCSUITE_ACM_INFO(0x0420), }, /* Nokia N93 */
1561 { NOKIA_PCSUITE_ACM_INFO(0x04E6), }, /* Nokia N93i */
1562 { NOKIA_PCSUITE_ACM_INFO(0x04B2), }, /* Nokia 5700 XpressMusic */
1563 { NOKIA_PCSUITE_ACM_INFO(0x0134), }, /* Nokia 6110 Navigator (China) */
1564 { NOKIA_PCSUITE_ACM_INFO(0x046E), }, /* Nokia 6110 Navigator */
1565 { NOKIA_PCSUITE_ACM_INFO(0x002f), }, /* Nokia 6120 classic & */
1566 { NOKIA_PCSUITE_ACM_INFO(0x0088), }, /* Nokia 6121 classic */
1567 { NOKIA_PCSUITE_ACM_INFO(0x00fc), }, /* Nokia 6124 classic */
1568 { NOKIA_PCSUITE_ACM_INFO(0x0042), }, /* Nokia E51 */
1569 { NOKIA_PCSUITE_ACM_INFO(0x00b0), }, /* Nokia E66 */
1570 { NOKIA_PCSUITE_ACM_INFO(0x00ab), }, /* Nokia E71 */
1571 { NOKIA_PCSUITE_ACM_INFO(0x0481), }, /* Nokia N76 */
1572 { NOKIA_PCSUITE_ACM_INFO(0x0007), }, /* Nokia N81 & N81 8GB */
1573 { NOKIA_PCSUITE_ACM_INFO(0x0071), }, /* Nokia N82 */
1574 { NOKIA_PCSUITE_ACM_INFO(0x04F0), }, /* Nokia N95 & N95-3 NAM */
1575 { NOKIA_PCSUITE_ACM_INFO(0x0070), }, /* Nokia N95 8GB */
1576 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1577 { NOKIA_PCSUITE_ACM_INFO(0x0099), }, /* Nokia 6210 Navigator, RM-367 */
1578 { NOKIA_PCSUITE_ACM_INFO(0x0128), }, /* Nokia 6210 Navigator, RM-419 */
1579 { NOKIA_PCSUITE_ACM_INFO(0x008f), }, /* Nokia 6220 Classic */
1580 { NOKIA_PCSUITE_ACM_INFO(0x00a0), }, /* Nokia 6650 */
1581 { NOKIA_PCSUITE_ACM_INFO(0x007b), }, /* Nokia N78 */
1582 { NOKIA_PCSUITE_ACM_INFO(0x0094), }, /* Nokia N85 */
1583 { NOKIA_PCSUITE_ACM_INFO(0x003a), }, /* Nokia N96 & N96-3 */
1584 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1585 { NOKIA_PCSUITE_ACM_INFO(0x0108), }, /* Nokia 5320 XpressMusic 2G */
1586 { NOKIA_PCSUITE_ACM_INFO(0x01f5), }, /* Nokia N97, RM-505 */
Przemo Firszt83a4eae2010-06-28 21:29:34 +01001587 { NOKIA_PCSUITE_ACM_INFO(0x02e3), }, /* Nokia 5230, RM-588 */
Toby Gray4035e452010-09-01 16:01:19 +01001588 { NOKIA_PCSUITE_ACM_INFO(0x0178), }, /* Nokia E63 */
1589 { NOKIA_PCSUITE_ACM_INFO(0x010e), }, /* Nokia E75 */
1590 { NOKIA_PCSUITE_ACM_INFO(0x02d9), }, /* Nokia 6760 Slide */
1591 { NOKIA_PCSUITE_ACM_INFO(0x01d0), }, /* Nokia E52 */
1592 { NOKIA_PCSUITE_ACM_INFO(0x0223), }, /* Nokia E72 */
1593 { NOKIA_PCSUITE_ACM_INFO(0x0275), }, /* Nokia X6 */
1594 { NOKIA_PCSUITE_ACM_INFO(0x026c), }, /* Nokia N97 Mini */
1595 { NOKIA_PCSUITE_ACM_INFO(0x0154), }, /* Nokia 5800 XpressMusic */
1596 { NOKIA_PCSUITE_ACM_INFO(0x04ce), }, /* Nokia E90 */
1597 { NOKIA_PCSUITE_ACM_INFO(0x01d4), }, /* Nokia E55 */
Arvid Ephraim Picciani721d92f2011-01-25 15:58:40 +01001598 { NOKIA_PCSUITE_ACM_INFO(0x0302), }, /* Nokia N8 */
Toby Gray4035e452010-09-01 16:01:19 +01001599 { SAMSUNG_PCSUITE_ACM_INFO(0x6651), }, /* Samsung GTi8510 (INNOV8) */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001600
1601 /* NOTE: non-Nokia COMM/ACM/0xff is likely MSFT RNDIS... NOT a modem! */
1602
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001603 /* Support Lego NXT using pbLua firmware */
Julian Calabyce126642010-01-05 23:58:20 +11001604 { USB_DEVICE(0x0694, 0xff00),
1605 .driver_info = NOT_A_MODEM,
Otavio Salvador7893afc2010-09-26 23:35:05 -03001606 },
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001607
Philippe Corbes5b239f02010-08-31 19:31:32 +02001608 /* control interfaces without any protocol set */
1609 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1610 USB_CDC_PROTO_NONE) },
1611
Linus Torvalds1da177e2005-04-16 15:20:36 -07001612 /* control interfaces with various AT-command sets */
1613 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1614 USB_CDC_ACM_PROTO_AT_V25TER) },
1615 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1616 USB_CDC_ACM_PROTO_AT_PCCA101) },
1617 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1618 USB_CDC_ACM_PROTO_AT_PCCA101_WAKE) },
1619 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1620 USB_CDC_ACM_PROTO_AT_GSM) },
1621 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
Alan Cox6e47e062009-06-11 12:37:06 +01001622 USB_CDC_ACM_PROTO_AT_3G) },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1624 USB_CDC_ACM_PROTO_AT_CDMA) },
1625
Linus Torvalds1da177e2005-04-16 15:20:36 -07001626 { }
1627};
1628
Alan Cox6e47e062009-06-11 12:37:06 +01001629MODULE_DEVICE_TABLE(usb, acm_ids);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001630
1631static struct usb_driver acm_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001632 .name = "cdc_acm",
1633 .probe = acm_probe,
1634 .disconnect = acm_disconnect,
Oliver Neukum35758582008-07-01 19:10:08 +02001635#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001636 .suspend = acm_suspend,
1637 .resume = acm_resume,
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001638 .reset_resume = acm_reset_resume,
Oliver Neukum35758582008-07-01 19:10:08 +02001639#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001640 .id_table = acm_ids,
Oliver Neukum35758582008-07-01 19:10:08 +02001641#ifdef CONFIG_PM
Oliver Neukum1365baf72007-10-12 17:24:28 +02001642 .supports_autosuspend = 1,
Oliver Neukum35758582008-07-01 19:10:08 +02001643#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001644};
1645
1646/*
1647 * TTY driver structures.
1648 */
1649
Jeff Dikeb68e31d2006-10-02 02:17:18 -07001650static const struct tty_operations acm_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001651 .open = acm_tty_open,
1652 .close = acm_tty_close,
Alan Cox10077d42009-06-11 12:36:09 +01001653 .hangup = acm_tty_hangup,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001654 .write = acm_tty_write,
1655 .write_room = acm_tty_write_room,
1656 .ioctl = acm_tty_ioctl,
1657 .throttle = acm_tty_throttle,
1658 .unthrottle = acm_tty_unthrottle,
1659 .chars_in_buffer = acm_tty_chars_in_buffer,
1660 .break_ctl = acm_tty_break_ctl,
1661 .set_termios = acm_tty_set_termios,
1662 .tiocmget = acm_tty_tiocmget,
1663 .tiocmset = acm_tty_tiocmset,
1664};
1665
1666/*
1667 * Init / exit.
1668 */
1669
1670static int __init acm_init(void)
1671{
1672 int retval;
1673 acm_tty_driver = alloc_tty_driver(ACM_TTY_MINORS);
1674 if (!acm_tty_driver)
1675 return -ENOMEM;
1676 acm_tty_driver->owner = THIS_MODULE,
1677 acm_tty_driver->driver_name = "acm",
1678 acm_tty_driver->name = "ttyACM",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001679 acm_tty_driver->major = ACM_TTY_MAJOR,
1680 acm_tty_driver->minor_start = 0,
1681 acm_tty_driver->type = TTY_DRIVER_TYPE_SERIAL,
1682 acm_tty_driver->subtype = SERIAL_TYPE_NORMAL,
Greg Kroah-Hartman331b8312005-06-20 21:15:16 -07001683 acm_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001684 acm_tty_driver->init_termios = tty_std_termios;
Alan Cox6e47e062009-06-11 12:37:06 +01001685 acm_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD |
1686 HUPCL | CLOCAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001687 tty_set_operations(acm_tty_driver, &acm_ops);
1688
1689 retval = tty_register_driver(acm_tty_driver);
1690 if (retval) {
1691 put_tty_driver(acm_tty_driver);
1692 return retval;
1693 }
1694
1695 retval = usb_register(&acm_driver);
1696 if (retval) {
1697 tty_unregister_driver(acm_tty_driver);
1698 put_tty_driver(acm_tty_driver);
1699 return retval;
1700 }
1701
Johan Hovolda2c7b932011-03-22 11:12:18 +01001702 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001703
1704 return 0;
1705}
1706
1707static void __exit acm_exit(void)
1708{
1709 usb_deregister(&acm_driver);
1710 tty_unregister_driver(acm_tty_driver);
1711 put_tty_driver(acm_tty_driver);
1712}
1713
1714module_init(acm_init);
1715module_exit(acm_exit);
1716
Alan Cox6e47e062009-06-11 12:37:06 +01001717MODULE_AUTHOR(DRIVER_AUTHOR);
1718MODULE_DESCRIPTION(DRIVER_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001719MODULE_LICENSE("GPL");
Scott James Remnante766aeb2009-04-06 17:33:18 +01001720MODULE_ALIAS_CHARDEV_MAJOR(ACM_TTY_MAJOR);