blob: 519c7b933508b6e79b1b368dcbd3f2343bd613d9 [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 Neukum1365baf2007-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 Neukum1365baf2007-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 Neukum1365baf2007-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 Neukum1365baf2007-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 Neukum1365baf2007-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 Neukum1365baf2007-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++) {
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001180 struct acm_rb *rb = &(acm->rb[i]);
David Kubicek61a87ad2005-11-01 18:51:34 +01001181 struct acm_ru *rcv = &(acm->ru[i]);
1182
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001183 rb->base = usb_alloc_coherent(acm->dev, readsize, GFP_KERNEL,
1184 &rb->dma);
1185 if (!rb->base) {
1186 dev_err(&intf->dev, "out of memory "
1187 "(read bufs usb_alloc_coherent)\n");
1188 goto alloc_fail6;
1189 }
1190
Alan Cox6e47e062009-06-11 12:37:06 +01001191 rcv->urb = usb_alloc_urb(0, GFP_KERNEL);
1192 if (rcv->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001193 dev_err(&intf->dev,
Alan Cox6e47e062009-06-11 12:37:06 +01001194 "out of memory (read urbs usb_alloc_urb)\n");
Axel Linc2572b72010-05-31 08:04:47 +08001195 goto alloc_fail6;
David Kubicek61a87ad2005-11-01 18:51:34 +01001196 }
1197
1198 rcv->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1199 rcv->instance = acm;
1200 }
Alan Cox6e47e062009-06-11 12:37:06 +01001201 for (i = 0; i < ACM_NW; i++) {
David Engrafe4cf3aa2008-03-20 10:01:34 +01001202 struct acm_wb *snd = &(acm->wb[i]);
1203
Alan Cox6e47e062009-06-11 12:37:06 +01001204 snd->urb = usb_alloc_urb(0, GFP_KERNEL);
1205 if (snd->urb == NULL) {
Johan Hovold255ab562011-03-22 11:12:13 +01001206 dev_err(&intf->dev,
Johan Hovold59d7fec2011-03-22 11:12:12 +01001207 "out of memory (write urbs usb_alloc_urb)\n");
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001208 goto alloc_fail7;
David Engrafe4cf3aa2008-03-20 10:01:34 +01001209 }
1210
Arseniy Lartsev5186ffe2009-07-01 16:27:26 +04001211 if (usb_endpoint_xfer_int(epwrite))
1212 usb_fill_int_urb(snd->urb, usb_dev,
1213 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1214 NULL, acm->writesize, acm_write_bulk, snd, epwrite->bInterval);
1215 else
1216 usb_fill_bulk_urb(snd->urb, usb_dev,
1217 usb_sndbulkpipe(usb_dev, epwrite->bEndpointAddress),
1218 NULL, acm->writesize, acm_write_bulk, snd);
David Engrafe4cf3aa2008-03-20 10:01:34 +01001219 snd->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1220 snd->instance = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001221 }
1222
Alan Cox6e47e062009-06-11 12:37:06 +01001223 usb_set_intfdata(intf, acm);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001224
1225 i = device_create_file(&intf->dev, &dev_attr_bmCapabilities);
1226 if (i < 0)
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001227 goto alloc_fail7;
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001228
1229 if (cfd) { /* export the country data */
1230 acm->country_codes = kmalloc(cfd->bLength - 4, GFP_KERNEL);
1231 if (!acm->country_codes)
1232 goto skip_countries;
1233 acm->country_code_size = cfd->bLength - 4;
Alan Cox6e47e062009-06-11 12:37:06 +01001234 memcpy(acm->country_codes, (u8 *)&cfd->wCountyCode0,
1235 cfd->bLength - 4);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001236 acm->country_rel_date = cfd->iCountryCodeRelDate;
1237
1238 i = device_create_file(&intf->dev, &dev_attr_wCountryCodes);
1239 if (i < 0) {
1240 kfree(acm->country_codes);
1241 goto skip_countries;
1242 }
1243
Alan Cox6e47e062009-06-11 12:37:06 +01001244 i = device_create_file(&intf->dev,
1245 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001246 if (i < 0) {
Axel Linc2572b72010-05-31 08:04:47 +08001247 device_remove_file(&intf->dev, &dev_attr_wCountryCodes);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001248 kfree(acm->country_codes);
1249 goto skip_countries;
1250 }
1251 }
1252
1253skip_countries:
Alan Cox6e47e062009-06-11 12:37:06 +01001254 usb_fill_int_urb(acm->ctrlurb, usb_dev,
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001255 usb_rcvintpipe(usb_dev, epctrl->bEndpointAddress),
1256 acm->ctrl_buffer, ctrlsize, acm_ctrl_irq, acm,
1257 /* works around buggy devices */
1258 epctrl->bInterval ? epctrl->bInterval : 0xff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001259 acm->ctrlurb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
1260 acm->ctrlurb->transfer_dma = acm->ctrl_dma;
1261
Linus Torvalds1da177e2005-04-16 15:20:36 -07001262 dev_info(&intf->dev, "ttyACM%d: USB ACM device\n", minor);
1263
1264 acm_set_control(acm, acm->ctrlout);
1265
1266 acm->line.dwDTERate = cpu_to_le32(9600);
1267 acm->line.bDataBits = 8;
1268 acm_set_line(acm, &acm->line);
1269
1270 usb_driver_claim_interface(&acm_driver, data_interface, acm);
David Brownell672c4e12008-08-06 18:41:12 -07001271 usb_set_intfdata(data_interface, acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001272
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001273 usb_get_intf(control_interface);
1274 tty_register_device(acm_tty_driver, minor, &control_interface->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001275
1276 acm_table[minor] = acm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001278 return 0;
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001279alloc_fail7:
David Engrafe4cf3aa2008-03-20 10:01:34 +01001280 for (i = 0; i < ACM_NW; i++)
1281 usb_free_urb(acm->wb[i].urb);
Axel Linc2572b72010-05-31 08:04:47 +08001282alloc_fail6:
Oliver Neukum86478942006-05-13 22:50:47 +02001283 for (i = 0; i < num_rx_buf; i++)
David Kubicek61a87ad2005-11-01 18:51:34 +01001284 usb_free_urb(acm->ru[i].urb);
Johan Hovold74f5e1b2011-03-22 11:12:23 +01001285 acm_read_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001286 usb_free_urb(acm->ctrlurb);
1287alloc_fail5:
Oliver Neukum884b6002005-04-21 21:28:02 +02001288 acm_write_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001289alloc_fail4:
Daniel Mack997ea582010-04-12 13:17:25 +02001290 usb_free_coherent(usb_dev, ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001291alloc_fail2:
1292 kfree(acm);
1293alloc_fail:
1294 return -ENOMEM;
1295}
1296
Oliver Neukum1365baf2007-10-12 17:24:28 +02001297static void stop_data_traffic(struct acm *acm)
1298{
1299 int i;
Johan Hovolda5cc7ef2011-03-22 11:12:15 +01001300
1301 dev_dbg(&acm->control->dev, "%s\n", __func__);
Oliver Neukum1365baf2007-10-12 17:24:28 +02001302
1303 tasklet_disable(&acm->urb_task);
1304
1305 usb_kill_urb(acm->ctrlurb);
Alan Cox6e47e062009-06-11 12:37:06 +01001306 for (i = 0; i < ACM_NW; i++)
David Engrafe4cf3aa2008-03-20 10:01:34 +01001307 usb_kill_urb(acm->wb[i].urb);
Oliver Neukum1365baf2007-10-12 17:24:28 +02001308 for (i = 0; i < acm->rx_buflimit; i++)
1309 usb_kill_urb(acm->ru[i].urb);
1310
Oliver Neukum1365baf2007-10-12 17:24:28 +02001311 tasklet_enable(&acm->urb_task);
1312
1313 cancel_work_sync(&acm->work);
1314}
1315
Linus Torvalds1da177e2005-04-16 15:20:36 -07001316static void acm_disconnect(struct usb_interface *intf)
1317{
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001318 struct acm *acm = usb_get_intfdata(intf);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319 struct usb_device *usb_dev = interface_to_usbdev(intf);
Alan Cox10077d42009-06-11 12:36:09 +01001320 struct tty_struct *tty;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001321
David Brownell672c4e12008-08-06 18:41:12 -07001322 /* sibling interface is already cleaning up */
1323 if (!acm)
Oliver Neukum86067eea2006-01-08 12:39:13 +01001324 return;
David Brownell672c4e12008-08-06 18:41:12 -07001325
1326 mutex_lock(&open_mutex);
Alan Cox6e47e062009-06-11 12:37:06 +01001327 if (acm->country_codes) {
Alan Stern74da5d62007-08-02 13:29:10 -04001328 device_remove_file(&acm->control->dev,
1329 &dev_attr_wCountryCodes);
1330 device_remove_file(&acm->control->dev,
1331 &dev_attr_iCountryCodeRelDate);
Oliver Neukumc4cabd22007-02-27 15:28:55 +01001332 }
Alan Stern74da5d62007-08-02 13:29:10 -04001333 device_remove_file(&acm->control->dev, &dev_attr_bmCapabilities);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001334 acm->dev = NULL;
Oliver Neukum86067eea2006-01-08 12:39:13 +01001335 usb_set_intfdata(acm->control, NULL);
1336 usb_set_intfdata(acm->data, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001337
Oliver Neukum1365baf2007-10-12 17:24:28 +02001338 stop_data_traffic(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001339
Oliver Neukum884b6002005-04-21 21:28:02 +02001340 acm_write_buffers_free(acm);
Daniel Mack997ea582010-04-12 13:17:25 +02001341 usb_free_coherent(usb_dev, acm->ctrlsize, acm->ctrl_buffer,
1342 acm->ctrl_dma);
Oliver Neukum830f4022008-06-25 14:17:16 +02001343 acm_read_buffers_free(acm);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001344
Oliver Neukuma2bfb4a2009-05-16 21:13:19 +02001345 if (!acm->combined_interfaces)
1346 usb_driver_release_interface(&acm_driver, intf == acm->control ?
Oliver Neukum830f4022008-06-25 14:17:16 +02001347 acm->data : acm->control);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001348
Alan Cox10077d42009-06-11 12:36:09 +01001349 if (acm->port.count == 0) {
brian@murphy.dk83ef3442005-06-29 16:53:29 -07001350 acm_tty_unregister(acm);
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001351 mutex_unlock(&open_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001352 return;
1353 }
1354
Arjan van de Ven4186ecf2006-01-11 15:55:29 +01001355 mutex_unlock(&open_mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001356 tty = tty_port_tty_get(&acm->port);
1357 if (tty) {
1358 tty_hangup(tty);
1359 tty_kref_put(tty);
1360 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001361}
1362
Oliver Neukum35758582008-07-01 19:10:08 +02001363#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001364static int acm_suspend(struct usb_interface *intf, pm_message_t message)
1365{
1366 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum11ea8592008-06-20 11:25:57 +02001367 int cnt;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001368
Alan Stern65bfd292008-11-25 16:39:18 -05001369 if (message.event & PM_EVENT_AUTO) {
Oliver Neukum11ea8592008-06-20 11:25:57 +02001370 int b;
1371
1372 spin_lock_irq(&acm->read_lock);
1373 spin_lock(&acm->write_lock);
1374 b = acm->processing + acm->transmitting;
1375 spin_unlock(&acm->write_lock);
1376 spin_unlock_irq(&acm->read_lock);
1377 if (b)
1378 return -EBUSY;
1379 }
1380
1381 spin_lock_irq(&acm->read_lock);
1382 spin_lock(&acm->write_lock);
1383 cnt = acm->susp_count++;
1384 spin_unlock(&acm->write_lock);
1385 spin_unlock_irq(&acm->read_lock);
1386
1387 if (cnt)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001388 return 0;
1389 /*
1390 we treat opened interfaces differently,
1391 we must guard against open
1392 */
1393 mutex_lock(&acm->mutex);
1394
Alan Cox10077d42009-06-11 12:36:09 +01001395 if (acm->port.count)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001396 stop_data_traffic(acm);
1397
1398 mutex_unlock(&acm->mutex);
1399 return 0;
1400}
1401
1402static int acm_resume(struct usb_interface *intf)
1403{
1404 struct acm *acm = usb_get_intfdata(intf);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001405 struct acm_wb *wb;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001406 int rv = 0;
Oliver Neukum11ea8592008-06-20 11:25:57 +02001407 int cnt;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001408
Oliver Neukum11ea8592008-06-20 11:25:57 +02001409 spin_lock_irq(&acm->read_lock);
1410 acm->susp_count -= 1;
1411 cnt = acm->susp_count;
1412 spin_unlock_irq(&acm->read_lock);
1413
1414 if (cnt)
Oliver Neukum1365baf2007-10-12 17:24:28 +02001415 return 0;
1416
1417 mutex_lock(&acm->mutex);
Alan Cox10077d42009-06-11 12:36:09 +01001418 if (acm->port.count) {
Oliver Neukum1365baf2007-10-12 17:24:28 +02001419 rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001420
1421 spin_lock_irq(&acm->write_lock);
1422 if (acm->delayed_wb) {
1423 wb = acm->delayed_wb;
1424 acm->delayed_wb = NULL;
1425 spin_unlock_irq(&acm->write_lock);
Oliver Neukumf0730922010-03-03 00:37:56 +01001426 acm_start_wb(acm, wb);
Oliver Neukum97d35f92009-12-16 17:05:57 +01001427 } else {
1428 spin_unlock_irq(&acm->write_lock);
1429 }
1430
1431 /*
1432 * delayed error checking because we must
1433 * do the write path at all cost
1434 */
Oliver Neukum1365baf2007-10-12 17:24:28 +02001435 if (rv < 0)
Oliver Neukum11ea8592008-06-20 11:25:57 +02001436 goto err_out;
Oliver Neukum1365baf2007-10-12 17:24:28 +02001437
1438 tasklet_schedule(&acm->urb_task);
1439 }
1440
1441err_out:
1442 mutex_unlock(&acm->mutex);
1443 return rv;
1444}
Oliver Neukum35758582008-07-01 19:10:08 +02001445
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001446static int acm_reset_resume(struct usb_interface *intf)
1447{
1448 struct acm *acm = usb_get_intfdata(intf);
1449 struct tty_struct *tty;
1450
1451 mutex_lock(&acm->mutex);
1452 if (acm->port.count) {
1453 tty = tty_port_tty_get(&acm->port);
1454 if (tty) {
1455 tty_hangup(tty);
1456 tty_kref_put(tty);
1457 }
1458 }
1459 mutex_unlock(&acm->mutex);
1460 return acm_resume(intf);
1461}
1462
Oliver Neukum35758582008-07-01 19:10:08 +02001463#endif /* CONFIG_PM */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001464
1465#define NOKIA_PCSUITE_ACM_INFO(x) \
1466 USB_DEVICE_AND_INTERFACE_INFO(0x0421, x, \
1467 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1468 USB_CDC_ACM_PROTO_VENDOR)
1469
Toby Gray4035e452010-09-01 16:01:19 +01001470#define SAMSUNG_PCSUITE_ACM_INFO(x) \
1471 USB_DEVICE_AND_INTERFACE_INFO(0x04e7, x, \
1472 USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM, \
1473 USB_CDC_ACM_PROTO_VENDOR)
1474
Linus Torvalds1da177e2005-04-16 15:20:36 -07001475/*
1476 * USB driver structure.
1477 */
1478
Németh Márton6ef48522010-01-10 15:33:45 +01001479static const struct usb_device_id acm_ids[] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001480 /* quirky and broken devices */
1481 { USB_DEVICE(0x0870, 0x0001), /* Metricom GS Modem */
1482 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1483 },
Andrey Arapovb0e2a702007-07-04 17:11:42 +02001484 { USB_DEVICE(0x0e8d, 0x0003), /* FIREFLY, MediaTek Inc; andrey.arapov@gmail.com */
1485 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1486 },
Andrew Lunn0f9c7b42008-12-23 17:31:23 +01001487 { USB_DEVICE(0x0e8d, 0x3329), /* MediaTek Inc GPS */
1488 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1489 },
Masahito Omote8753e652005-07-29 12:17:25 -07001490 { USB_DEVICE(0x0482, 0x0203), /* KYOCERA AH-K3001V */
1491 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1492 },
Chris Malley91a9c922006-10-03 10:08:28 +01001493 { USB_DEVICE(0x079b, 0x000f), /* BT On-Air USB MODEM */
1494 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1495 },
Alan Cox7abcf202009-04-06 17:35:01 +01001496 { USB_DEVICE(0x0ace, 0x1602), /* ZyDAS 56K USB MODEM */
1497 .driver_info = SINGLE_RX_URB,
1498 },
Oliver Neukum86478942006-05-13 22:50:47 +02001499 { USB_DEVICE(0x0ace, 0x1608), /* ZyDAS 56K USB MODEM */
1500 .driver_info = SINGLE_RX_URB, /* firmware bug */
1501 },
Oliver Neukum3dd2ae82006-06-23 09:14:17 +02001502 { USB_DEVICE(0x0ace, 0x1611), /* ZyDAS 56K USB MODEM - new version */
1503 .driver_info = SINGLE_RX_URB, /* firmware bug */
1504 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001505 { USB_DEVICE(0x22b8, 0x7000), /* Motorola Q Phone */
1506 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1507 },
Iain McFarlane6149ed52008-05-04 00:13:49 +01001508 { USB_DEVICE(0x0803, 0x3095), /* Zoom Telephonics Model 3095F USB MODEM */
1509 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1510 },
Eric Sandeenc8fd2c32008-08-14 08:25:40 -05001511 { USB_DEVICE(0x0572, 0x1321), /* Conexant USB MODEM CX93010 */
1512 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1513 },
Alan Coxc89c60e2009-01-11 19:53:10 +00001514 { USB_DEVICE(0x0572, 0x1324), /* Conexant USB MODEM RD02-D400 */
1515 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1516 },
Xiao Kaijiancab98a02009-05-08 00:48:23 +08001517 { USB_DEVICE(0x0572, 0x1328), /* Shiro / Aztech USB MODEM UM-3100 */
1518 .driver_info = NO_UNION_NORMAL, /* has no union descriptor */
1519 },
Dmitriy Taychenachev155df652009-02-25 12:36:51 +08001520 { USB_DEVICE(0x22b8, 0x6425), /* Motorola MOTOMAGX phones */
1521 },
Adam Richterc332b4e2009-02-18 16:17:15 -08001522 { USB_DEVICE(0x0572, 0x1329), /* Hummingbird huc56s (Conexant) */
1523 .driver_info = NO_UNION_NORMAL, /* union descriptor misplaced on
1524 data interface instead of
1525 communications interface.
1526 Maybe we should define a new
1527 quirk for this. */
1528 },
Kir Kolyshkin1f17c502009-05-28 20:33:58 +04001529 { USB_DEVICE(0x1bbb, 0x0003), /* Alcatel OT-I650 */
1530 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1531 },
Russ Nelsonc3baa192010-04-21 23:07:03 -04001532 { USB_DEVICE(0x1576, 0x03b1), /* Maretron USB100 */
1533 .driver_info = NO_UNION_NORMAL, /* reports zero length descriptor */
1534 },
Oliver Neukum9be84562007-02-12 08:50:03 +01001535
Adrian Taylorc1479a92009-11-19 10:35:33 +00001536 /* Nokia S60 phones expose two ACM channels. The first is
1537 * a modem and is picked up by the standard AT-command
1538 * information below. The second is 'vendor-specific' but
1539 * is treated as a serial device at the S60 end, so we want
1540 * to expose it on Linux too. */
1541 { NOKIA_PCSUITE_ACM_INFO(0x042D), }, /* Nokia 3250 */
1542 { NOKIA_PCSUITE_ACM_INFO(0x04D8), }, /* Nokia 5500 Sport */
1543 { NOKIA_PCSUITE_ACM_INFO(0x04C9), }, /* Nokia E50 */
1544 { NOKIA_PCSUITE_ACM_INFO(0x0419), }, /* Nokia E60 */
1545 { NOKIA_PCSUITE_ACM_INFO(0x044D), }, /* Nokia E61 */
1546 { NOKIA_PCSUITE_ACM_INFO(0x0001), }, /* Nokia E61i */
1547 { NOKIA_PCSUITE_ACM_INFO(0x0475), }, /* Nokia E62 */
1548 { NOKIA_PCSUITE_ACM_INFO(0x0508), }, /* Nokia E65 */
1549 { NOKIA_PCSUITE_ACM_INFO(0x0418), }, /* Nokia E70 */
1550 { NOKIA_PCSUITE_ACM_INFO(0x0425), }, /* Nokia N71 */
1551 { NOKIA_PCSUITE_ACM_INFO(0x0486), }, /* Nokia N73 */
1552 { NOKIA_PCSUITE_ACM_INFO(0x04DF), }, /* Nokia N75 */
1553 { NOKIA_PCSUITE_ACM_INFO(0x000e), }, /* Nokia N77 */
1554 { NOKIA_PCSUITE_ACM_INFO(0x0445), }, /* Nokia N80 */
1555 { NOKIA_PCSUITE_ACM_INFO(0x042F), }, /* Nokia N91 & N91 8GB */
1556 { NOKIA_PCSUITE_ACM_INFO(0x048E), }, /* Nokia N92 */
1557 { NOKIA_PCSUITE_ACM_INFO(0x0420), }, /* Nokia N93 */
1558 { NOKIA_PCSUITE_ACM_INFO(0x04E6), }, /* Nokia N93i */
1559 { NOKIA_PCSUITE_ACM_INFO(0x04B2), }, /* Nokia 5700 XpressMusic */
1560 { NOKIA_PCSUITE_ACM_INFO(0x0134), }, /* Nokia 6110 Navigator (China) */
1561 { NOKIA_PCSUITE_ACM_INFO(0x046E), }, /* Nokia 6110 Navigator */
1562 { NOKIA_PCSUITE_ACM_INFO(0x002f), }, /* Nokia 6120 classic & */
1563 { NOKIA_PCSUITE_ACM_INFO(0x0088), }, /* Nokia 6121 classic */
1564 { NOKIA_PCSUITE_ACM_INFO(0x00fc), }, /* Nokia 6124 classic */
1565 { NOKIA_PCSUITE_ACM_INFO(0x0042), }, /* Nokia E51 */
1566 { NOKIA_PCSUITE_ACM_INFO(0x00b0), }, /* Nokia E66 */
1567 { NOKIA_PCSUITE_ACM_INFO(0x00ab), }, /* Nokia E71 */
1568 { NOKIA_PCSUITE_ACM_INFO(0x0481), }, /* Nokia N76 */
1569 { NOKIA_PCSUITE_ACM_INFO(0x0007), }, /* Nokia N81 & N81 8GB */
1570 { NOKIA_PCSUITE_ACM_INFO(0x0071), }, /* Nokia N82 */
1571 { NOKIA_PCSUITE_ACM_INFO(0x04F0), }, /* Nokia N95 & N95-3 NAM */
1572 { NOKIA_PCSUITE_ACM_INFO(0x0070), }, /* Nokia N95 8GB */
1573 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1574 { NOKIA_PCSUITE_ACM_INFO(0x0099), }, /* Nokia 6210 Navigator, RM-367 */
1575 { NOKIA_PCSUITE_ACM_INFO(0x0128), }, /* Nokia 6210 Navigator, RM-419 */
1576 { NOKIA_PCSUITE_ACM_INFO(0x008f), }, /* Nokia 6220 Classic */
1577 { NOKIA_PCSUITE_ACM_INFO(0x00a0), }, /* Nokia 6650 */
1578 { NOKIA_PCSUITE_ACM_INFO(0x007b), }, /* Nokia N78 */
1579 { NOKIA_PCSUITE_ACM_INFO(0x0094), }, /* Nokia N85 */
1580 { NOKIA_PCSUITE_ACM_INFO(0x003a), }, /* Nokia N96 & N96-3 */
1581 { NOKIA_PCSUITE_ACM_INFO(0x00e9), }, /* Nokia 5320 XpressMusic */
1582 { NOKIA_PCSUITE_ACM_INFO(0x0108), }, /* Nokia 5320 XpressMusic 2G */
1583 { NOKIA_PCSUITE_ACM_INFO(0x01f5), }, /* Nokia N97, RM-505 */
Przemo Firszt83a4eae2010-06-28 21:29:34 +01001584 { NOKIA_PCSUITE_ACM_INFO(0x02e3), }, /* Nokia 5230, RM-588 */
Toby Gray4035e452010-09-01 16:01:19 +01001585 { NOKIA_PCSUITE_ACM_INFO(0x0178), }, /* Nokia E63 */
1586 { NOKIA_PCSUITE_ACM_INFO(0x010e), }, /* Nokia E75 */
1587 { NOKIA_PCSUITE_ACM_INFO(0x02d9), }, /* Nokia 6760 Slide */
1588 { NOKIA_PCSUITE_ACM_INFO(0x01d0), }, /* Nokia E52 */
1589 { NOKIA_PCSUITE_ACM_INFO(0x0223), }, /* Nokia E72 */
1590 { NOKIA_PCSUITE_ACM_INFO(0x0275), }, /* Nokia X6 */
1591 { NOKIA_PCSUITE_ACM_INFO(0x026c), }, /* Nokia N97 Mini */
1592 { NOKIA_PCSUITE_ACM_INFO(0x0154), }, /* Nokia 5800 XpressMusic */
1593 { NOKIA_PCSUITE_ACM_INFO(0x04ce), }, /* Nokia E90 */
1594 { NOKIA_PCSUITE_ACM_INFO(0x01d4), }, /* Nokia E55 */
Arvid Ephraim Picciani721d92f2011-01-25 15:58:40 +01001595 { NOKIA_PCSUITE_ACM_INFO(0x0302), }, /* Nokia N8 */
Toby Gray4035e452010-09-01 16:01:19 +01001596 { SAMSUNG_PCSUITE_ACM_INFO(0x6651), }, /* Samsung GTi8510 (INNOV8) */
Adrian Taylorc1479a92009-11-19 10:35:33 +00001597
1598 /* NOTE: non-Nokia COMM/ACM/0xff is likely MSFT RNDIS... NOT a modem! */
1599
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001600 /* Support Lego NXT using pbLua firmware */
Julian Calabyce126642010-01-05 23:58:20 +11001601 { USB_DEVICE(0x0694, 0xff00),
1602 .driver_info = NOT_A_MODEM,
Otavio Salvador7893afc2010-09-26 23:35:05 -03001603 },
Julian Calaby7c5d8c32010-01-05 23:57:46 +11001604
Philippe Corbes5b239f02010-08-31 19:31:32 +02001605 /* control interfaces without any protocol set */
1606 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1607 USB_CDC_PROTO_NONE) },
1608
Linus Torvalds1da177e2005-04-16 15:20:36 -07001609 /* control interfaces with various AT-command sets */
1610 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1611 USB_CDC_ACM_PROTO_AT_V25TER) },
1612 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1613 USB_CDC_ACM_PROTO_AT_PCCA101) },
1614 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1615 USB_CDC_ACM_PROTO_AT_PCCA101_WAKE) },
1616 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1617 USB_CDC_ACM_PROTO_AT_GSM) },
1618 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
Alan Cox6e47e062009-06-11 12:37:06 +01001619 USB_CDC_ACM_PROTO_AT_3G) },
Linus Torvalds1da177e2005-04-16 15:20:36 -07001620 { USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_ACM,
1621 USB_CDC_ACM_PROTO_AT_CDMA) },
1622
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 { }
1624};
1625
Alan Cox6e47e062009-06-11 12:37:06 +01001626MODULE_DEVICE_TABLE(usb, acm_ids);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001627
1628static struct usb_driver acm_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001629 .name = "cdc_acm",
1630 .probe = acm_probe,
1631 .disconnect = acm_disconnect,
Oliver Neukum35758582008-07-01 19:10:08 +02001632#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001633 .suspend = acm_suspend,
1634 .resume = acm_resume,
Francesco Lavraa91b0c52009-12-08 09:54:11 +01001635 .reset_resume = acm_reset_resume,
Oliver Neukum35758582008-07-01 19:10:08 +02001636#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001637 .id_table = acm_ids,
Oliver Neukum35758582008-07-01 19:10:08 +02001638#ifdef CONFIG_PM
Oliver Neukum1365baf2007-10-12 17:24:28 +02001639 .supports_autosuspend = 1,
Oliver Neukum35758582008-07-01 19:10:08 +02001640#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001641};
1642
1643/*
1644 * TTY driver structures.
1645 */
1646
Jeff Dikeb68e31d2006-10-02 02:17:18 -07001647static const struct tty_operations acm_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001648 .open = acm_tty_open,
1649 .close = acm_tty_close,
Alan Cox10077d42009-06-11 12:36:09 +01001650 .hangup = acm_tty_hangup,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001651 .write = acm_tty_write,
1652 .write_room = acm_tty_write_room,
1653 .ioctl = acm_tty_ioctl,
1654 .throttle = acm_tty_throttle,
1655 .unthrottle = acm_tty_unthrottle,
1656 .chars_in_buffer = acm_tty_chars_in_buffer,
1657 .break_ctl = acm_tty_break_ctl,
1658 .set_termios = acm_tty_set_termios,
1659 .tiocmget = acm_tty_tiocmget,
1660 .tiocmset = acm_tty_tiocmset,
1661};
1662
1663/*
1664 * Init / exit.
1665 */
1666
1667static int __init acm_init(void)
1668{
1669 int retval;
1670 acm_tty_driver = alloc_tty_driver(ACM_TTY_MINORS);
1671 if (!acm_tty_driver)
1672 return -ENOMEM;
1673 acm_tty_driver->owner = THIS_MODULE,
1674 acm_tty_driver->driver_name = "acm",
1675 acm_tty_driver->name = "ttyACM",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001676 acm_tty_driver->major = ACM_TTY_MAJOR,
1677 acm_tty_driver->minor_start = 0,
1678 acm_tty_driver->type = TTY_DRIVER_TYPE_SERIAL,
1679 acm_tty_driver->subtype = SERIAL_TYPE_NORMAL,
Greg Kroah-Hartman331b8312005-06-20 21:15:16 -07001680 acm_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001681 acm_tty_driver->init_termios = tty_std_termios;
Alan Cox6e47e062009-06-11 12:37:06 +01001682 acm_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD |
1683 HUPCL | CLOCAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001684 tty_set_operations(acm_tty_driver, &acm_ops);
1685
1686 retval = tty_register_driver(acm_tty_driver);
1687 if (retval) {
1688 put_tty_driver(acm_tty_driver);
1689 return retval;
1690 }
1691
1692 retval = usb_register(&acm_driver);
1693 if (retval) {
1694 tty_unregister_driver(acm_tty_driver);
1695 put_tty_driver(acm_tty_driver);
1696 return retval;
1697 }
1698
Johan Hovolda2c7b932011-03-22 11:12:18 +01001699 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001700
1701 return 0;
1702}
1703
1704static void __exit acm_exit(void)
1705{
1706 usb_deregister(&acm_driver);
1707 tty_unregister_driver(acm_tty_driver);
1708 put_tty_driver(acm_tty_driver);
1709}
1710
1711module_init(acm_init);
1712module_exit(acm_exit);
1713
Alan Cox6e47e062009-06-11 12:37:06 +01001714MODULE_AUTHOR(DRIVER_AUTHOR);
1715MODULE_DESCRIPTION(DRIVER_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001716MODULE_LICENSE("GPL");
Scott James Remnante766aeb2009-04-06 17:33:18 +01001717MODULE_ALIAS_CHARDEV_MAJOR(ACM_TTY_MAJOR);