blob: 45975b4984eabff36729c02d8ef877da1c6508ed [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * USB Serial Converter driver
3 *
Greg Kroah-Hartman502b95c2005-06-20 21:15:16 -07004 * Copyright (C) 1999 - 2005 Greg Kroah-Hartman (greg@kroah.com)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 * Copyright (C) 2000 Peter Berger (pberger@brimson.com)
6 * Copyright (C) 2000 Al Borchers (borchers@steinerpoint.com)
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License version
10 * 2 as published by the Free Software Foundation.
11 *
Greg Kroah-Hartman502b95c2005-06-20 21:15:16 -070012 * This driver was originally based on the ACM driver by Armin Fuerst (which was
Linus Torvalds1da177e2005-04-16 15:20:36 -070013 * based on a driver by Brad Keryan)
14 *
Alan Coxa8d6f0a2008-07-22 11:12:24 +010015 * See Documentation/usb/usb-serial.txt for more information on using this
16 * driver
Linus Torvalds1da177e2005-04-16 15:20:36 -070017 *
Linus Torvalds1da177e2005-04-16 15:20:36 -070018 */
19
Linus Torvalds1da177e2005-04-16 15:20:36 -070020#include <linux/kernel.h>
21#include <linux/errno.h>
22#include <linux/init.h>
23#include <linux/slab.h>
Alexey Dobriyan405f5572009-07-11 22:08:37 +040024#include <linux/smp_lock.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070025#include <linux/tty.h>
26#include <linux/tty_driver.h>
27#include <linux/tty_flip.h>
28#include <linux/module.h>
29#include <linux/moduleparam.h>
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -070030#include <linux/seq_file.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070031#include <linux/spinlock.h>
Luiz Fernando Capitulino1ce7dd22006-03-24 17:12:31 -030032#include <linux/mutex.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070033#include <linux/list.h>
Alan Coxa8d6f0a2008-07-22 11:12:24 +010034#include <linux/uaccess.h>
Alan Coxc56d3002009-07-28 00:34:58 +010035#include <linux/serial.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070036#include <linux/usb.h>
Greg Kroah-Hartmana9698882006-07-11 21:22:58 -070037#include <linux/usb/serial.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070038#include "pl2303.h"
39
40/*
41 * Version Information
42 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070043#define DRIVER_AUTHOR "Greg Kroah-Hartman, greg@kroah.com, http://www.kroah.com/linux/"
44#define DRIVER_DESC "USB Serial Driver core"
45
46/* Driver structure we register with the USB core */
47static struct usb_driver usb_serial_driver = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 .name = "usbserial",
49 .probe = usb_serial_probe,
50 .disconnect = usb_serial_disconnect,
Oliver Neukumec225592007-04-27 20:54:57 +020051 .suspend = usb_serial_suspend,
52 .resume = usb_serial_resume,
Greg Kroah-Hartmanba9dc652005-11-16 13:41:28 -080053 .no_dynamic_id = 1,
Linus Torvalds1da177e2005-04-16 15:20:36 -070054};
55
56/* There is no MODULE_DEVICE_TABLE for usbserial.c. Instead
57 the MODULE_DEVICE_TABLE declarations in each serial driver
58 cause the "hotplug" program to pull in whatever module is necessary
59 via modprobe, and modprobe will load usbserial because the serial
60 drivers depend on it.
61*/
62
63static int debug;
Alan Coxa8d6f0a2008-07-22 11:12:24 +010064/* initially all NULL */
65static struct usb_serial *serial_table[SERIAL_TTY_MINORS];
Oliver Neukum3ddad822007-07-24 15:13:42 +020066static DEFINE_MUTEX(table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070067static LIST_HEAD(usb_serial_driver_list);
68
Alan Stern8bc2c1b2009-09-01 11:38:59 -040069/*
70 * Look up the serial structure. If it is found and it hasn't been
71 * disconnected, return with its disc_mutex held and its refcount
72 * incremented. Otherwise return NULL.
73 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070074struct usb_serial *usb_serial_get_by_index(unsigned index)
75{
Oliver Neukum34ef50e2007-01-13 07:29:26 +010076 struct usb_serial *serial;
77
Oliver Neukum3ddad822007-07-24 15:13:42 +020078 mutex_lock(&table_lock);
Oliver Neukum34ef50e2007-01-13 07:29:26 +010079 serial = serial_table[index];
Linus Torvalds1da177e2005-04-16 15:20:36 -070080
Alan Stern8bc2c1b2009-09-01 11:38:59 -040081 if (serial) {
82 mutex_lock(&serial->disc_mutex);
83 if (serial->disconnected) {
84 mutex_unlock(&serial->disc_mutex);
85 serial = NULL;
86 } else {
87 kref_get(&serial->kref);
88 }
89 }
Oliver Neukum3ddad822007-07-24 15:13:42 +020090 mutex_unlock(&table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070091 return serial;
92}
93
Alan Coxa8d6f0a2008-07-22 11:12:24 +010094static struct usb_serial *get_free_serial(struct usb_serial *serial,
95 int num_ports, unsigned int *minor)
Linus Torvalds1da177e2005-04-16 15:20:36 -070096{
97 unsigned int i, j;
98 int good_spot;
99
Harvey Harrison441b62c2008-03-03 16:08:34 -0800100 dbg("%s %d", __func__, num_ports);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700101
102 *minor = 0;
Oliver Neukum3ddad822007-07-24 15:13:42 +0200103 mutex_lock(&table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700104 for (i = 0; i < SERIAL_TTY_MINORS; ++i) {
105 if (serial_table[i])
106 continue;
107
108 good_spot = 1;
109 for (j = 1; j <= num_ports-1; ++j)
110 if ((i+j >= SERIAL_TTY_MINORS) || (serial_table[i+j])) {
111 good_spot = 0;
112 i += j;
113 break;
114 }
115 if (good_spot == 0)
116 continue;
117
118 *minor = i;
Oliver Neukuma1f721c2007-03-05 15:23:51 +0100119 j = 0;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800120 dbg("%s - minor base = %d", __func__, *minor);
Oliver Neukuma1f721c2007-03-05 15:23:51 +0100121 for (i = *minor; (i < (*minor + num_ports)) && (i < SERIAL_TTY_MINORS); ++i) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122 serial_table[i] = serial;
Oliver Neukuma1f721c2007-03-05 15:23:51 +0100123 serial->port[j++]->number = i;
124 }
Oliver Neukum3ddad822007-07-24 15:13:42 +0200125 mutex_unlock(&table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126 return serial;
127 }
Oliver Neukum3ddad822007-07-24 15:13:42 +0200128 mutex_unlock(&table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700129 return NULL;
130}
131
132static void return_serial(struct usb_serial *serial)
133{
134 int i;
135
Harvey Harrison441b62c2008-03-03 16:08:34 -0800136 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700137
Alan Stern8bc2c1b2009-09-01 11:38:59 -0400138 mutex_lock(&table_lock);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100139 for (i = 0; i < serial->num_ports; ++i)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700140 serial_table[serial->minor + i] = NULL;
Alan Stern8bc2c1b2009-09-01 11:38:59 -0400141 mutex_unlock(&table_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700142}
143
144static void destroy_serial(struct kref *kref)
145{
146 struct usb_serial *serial;
147 struct usb_serial_port *port;
148 int i;
149
150 serial = to_usb_serial(kref);
151
Harvey Harrison441b62c2008-03-03 16:08:34 -0800152 dbg("%s - %s", __func__, serial->type->description);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700153
Jim Radford521b85a2007-03-13 08:30:50 -0700154 /* return the minor range that this device had */
Alan Stern0282b7f2008-07-29 12:01:04 -0400155 if (serial->minor != SERIAL_TTY_NO_MINOR)
156 return_serial(serial);
Jim Radford521b85a2007-03-13 08:30:50 -0700157
Alan Sternf9c99bb2009-06-02 11:53:55 -0400158 serial->type->release(serial);
159
Alan Stern41bd34d2009-09-01 11:38:34 -0400160 /* Now that nothing is using the ports, they can be freed */
161 for (i = 0; i < serial->num_port_pointers; ++i) {
Alan Sternf9c99bb2009-06-02 11:53:55 -0400162 port = serial->port[i];
Alan Stern41bd34d2009-09-01 11:38:34 -0400163 if (port) {
164 port->serial = NULL;
Alan Sternf9c99bb2009-06-02 11:53:55 -0400165 put_device(&port->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700166 }
167 }
168
169 usb_put_dev(serial->dev);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100170 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700171}
172
Guennadi Liakhovetski73e487f2006-04-25 07:46:17 +0200173void usb_serial_put(struct usb_serial *serial)
174{
175 kref_put(&serial->kref, destroy_serial);
176}
177
Linus Torvalds1da177e2005-04-16 15:20:36 -0700178/*****************************************************************************
179 * Driver tty interface functions
180 *****************************************************************************/
Alan Sternf5b09532009-09-01 11:38:44 -0400181
182/**
183 * serial_install - install tty
184 * @driver: the driver (USB in our case)
185 * @tty: the tty being created
186 *
187 * Create the termios objects for this tty. We use the default
188 * USB serial settings but permit them to be overridden by
189 * serial->type->init_termios.
Alan Sterncc56cd02009-09-01 11:39:13 -0400190 *
191 * This is the first place a new tty gets used. Hence this is where we
192 * acquire references to the usb_serial structure and the driver module,
193 * where we store a pointer to the port, and where we do an autoresume.
Alan Stern74556122009-09-01 11:39:40 -0400194 * All these actions are reversed in serial_release().
Alan Sternf5b09532009-09-01 11:38:44 -0400195 */
196static int serial_install(struct tty_driver *driver, struct tty_struct *tty)
197{
198 int idx = tty->index;
199 struct usb_serial *serial;
Alan Sterncc56cd02009-09-01 11:39:13 -0400200 struct usb_serial_port *port;
201 int retval = -ENODEV;
202
Alan Sternff8324d2009-09-01 11:39:51 -0400203 dbg("%s", __func__);
204
Alan Sterncc56cd02009-09-01 11:39:13 -0400205 serial = usb_serial_get_by_index(idx);
206 if (!serial)
207 return retval;
208
209 port = serial->port[idx - serial->minor];
210 if (!port)
211 goto error_no_port;
212 if (!try_module_get(serial->type->driver.owner))
213 goto error_module_get;
214
Alan Stern7e29bb42009-09-01 11:39:22 -0400215 /* perform the standard setup */
216 retval = tty_init_termios(tty);
217 if (retval)
218 goto error_init_termios;
219
Alan Sterncc56cd02009-09-01 11:39:13 -0400220 retval = usb_autopm_get_interface(serial->interface);
221 if (retval)
222 goto error_get_interface;
Alan Sternf5b09532009-09-01 11:38:44 -0400223
Alan Sterncc56cd02009-09-01 11:39:13 -0400224 mutex_unlock(&serial->disc_mutex);
225
Alan Stern7e29bb42009-09-01 11:39:22 -0400226 /* allow the driver to update the settings */
227 if (serial->type->init_termios)
228 serial->type->init_termios(tty);
229
Alan Sterncc56cd02009-09-01 11:39:13 -0400230 tty->driver_data = port;
231
Alan Sternf5b09532009-09-01 11:38:44 -0400232 /* Final install (we use the default method) */
233 tty_driver_kref_get(driver);
234 tty->count++;
235 driver->ttys[idx] = tty;
Alan Sterncc56cd02009-09-01 11:39:13 -0400236 return retval;
237
Alan Sterncc56cd02009-09-01 11:39:13 -0400238 error_get_interface:
Alan Stern7e29bb42009-09-01 11:39:22 -0400239 error_init_termios:
Alan Sterncc56cd02009-09-01 11:39:13 -0400240 module_put(serial->type->driver.owner);
241 error_module_get:
242 error_no_port:
243 usb_serial_put(serial);
244 mutex_unlock(&serial->disc_mutex);
245 return retval;
Alan Sternf5b09532009-09-01 11:38:44 -0400246}
247
Alan Stern320348c2009-09-01 11:39:59 -0400248static int serial_open(struct tty_struct *tty, struct file *filp)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249{
Alan Stern320348c2009-09-01 11:39:59 -0400250 struct usb_serial_port *port = tty->driver_data;
251 struct usb_serial *serial = port->serial;
252 int retval;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700253
Alan Sternff8324d2009-09-01 11:39:51 -0400254 dbg("%s - port %d", __func__, port->number);
255
Alan Stern320348c2009-09-01 11:39:59 -0400256 spin_lock_irq(&port->port.lock);
257 if (!tty_hung_up_p(filp))
258 ++port->port.count;
259 spin_unlock_irq(&port->port.lock);
Alan Cox4a90f092008-10-13 10:39:46 +0100260 tty_port_tty_set(&port->port, tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700261
Alan Stern320348c2009-09-01 11:39:59 -0400262 /* Do the device-specific open only if the hardware isn't
263 * already initialized.
264 */
265 if (!test_bit(ASYNCB_INITIALIZED, &port->port.flags)) {
266 if (mutex_lock_interruptible(&port->mutex))
267 return -ERESTARTSYS;
Alan Stern2d931482009-04-14 11:31:02 -0400268 mutex_lock(&serial->disc_mutex);
Alan Stern320348c2009-09-01 11:39:59 -0400269 if (serial->disconnected)
270 retval = -ENODEV;
271 else
272 retval = port->serial->type->open(tty, port);
Alan Stern2d931482009-04-14 11:31:02 -0400273 mutex_unlock(&serial->disc_mutex);
Alan Stern320348c2009-09-01 11:39:59 -0400274 mutex_unlock(&port->mutex);
275 if (retval)
276 return retval;
Alan Coxc56d3002009-07-28 00:34:58 +0100277 set_bit(ASYNCB_INITIALIZED, &port->port.flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700278 }
Alan Stern320348c2009-09-01 11:39:59 -0400279
Alan Cox335f8512009-06-11 12:26:29 +0100280 /* Now do the correct tty layer semantics */
281 retval = tty_port_block_til_ready(&port->port, tty, filp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 return retval;
283}
284
Alan Cox335f8512009-06-11 12:26:29 +0100285/**
Alan Stern74556122009-09-01 11:39:40 -0400286 * serial_down - shut down hardware
Alan Sternf5b09532009-09-01 11:38:44 -0400287 * @port: port to shut down
Alan Cox335f8512009-06-11 12:26:29 +0100288 *
Alan Sternf5b09532009-09-01 11:38:44 -0400289 * Shut down a USB serial port unless it is the console. We never
290 * shut down the console hardware as it will always be in use.
Alan Cox335f8512009-06-11 12:26:29 +0100291 */
Alan Stern74556122009-09-01 11:39:40 -0400292static void serial_down(struct usb_serial_port *port)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700293{
Alan Cox335f8512009-06-11 12:26:29 +0100294 struct usb_serial_driver *drv = port->serial->type;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700295
Alan Sternf5b09532009-09-01 11:38:44 -0400296 /*
297 * The console is magical. Do not hang up the console hardware
298 * or there will be tears.
299 */
Alan Cox335f8512009-06-11 12:26:29 +0100300 if (port->console)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700301 return;
302
Alan Sternff8324d2009-09-01 11:39:51 -0400303 /* Don't call the close method if the hardware hasn't been
304 * initialized.
305 */
306 if (!test_and_clear_bit(ASYNCB_INITIALIZED, &port->port.flags))
307 return;
308
Luiz Fernando Capitulino1ce7dd22006-03-24 17:12:31 -0300309 mutex_lock(&port->mutex);
Alan Cox335f8512009-06-11 12:26:29 +0100310 if (drv->close)
311 drv->close(port);
Luiz Fernando Capitulino1ce7dd22006-03-24 17:12:31 -0300312 mutex_unlock(&port->mutex);
Alan Cox335f8512009-06-11 12:26:29 +0100313}
314
Alan Sternf5b09532009-09-01 11:38:44 -0400315static void serial_hangup(struct tty_struct *tty)
Alan Cox335f8512009-06-11 12:26:29 +0100316{
Alan Cox4455e342009-09-19 13:13:24 -0700317 struct usb_serial_port *port = tty->driver_data;
Alan Sternff8324d2009-09-01 11:39:51 -0400318
319 dbg("%s - port %d", __func__, port->number);
320
Alan Stern74556122009-09-01 11:39:40 -0400321 serial_down(port);
Alan Sternf5b09532009-09-01 11:38:44 -0400322 tty_port_hangup(&port->port);
Alan Cox335f8512009-06-11 12:26:29 +0100323}
324
325static void serial_close(struct tty_struct *tty, struct file *filp)
326{
327 struct usb_serial_port *port = tty->driver_data;
328
329 dbg("%s - port %d", __func__, port->number);
330
Alan Sternff8324d2009-09-01 11:39:51 -0400331 if (tty_hung_up_p(filp))
332 return;
Alan Cox335f8512009-06-11 12:26:29 +0100333 if (tty_port_close_start(&port->port, tty, filp) == 0)
334 return;
Alan Stern74556122009-09-01 11:39:40 -0400335 serial_down(port);
Alan Cox335f8512009-06-11 12:26:29 +0100336 tty_port_close_end(&port->port, tty);
337 tty_port_tty_set(&port->port, NULL);
Alan Cox335f8512009-06-11 12:26:29 +0100338}
339
Alan Sternf5b09532009-09-01 11:38:44 -0400340/**
Alan Stern74556122009-09-01 11:39:40 -0400341 * serial_release - free resources post close/hangup
Alan Sternf5b09532009-09-01 11:38:44 -0400342 * @port: port to free up
343 *
344 * Do the resource freeing and refcount dropping for the port.
345 * Avoid freeing the console.
346 *
347 * Called when the last tty kref is dropped.
348 */
Alan Stern74556122009-09-01 11:39:40 -0400349static void serial_release(struct tty_struct *tty)
Alan Cox335f8512009-06-11 12:26:29 +0100350{
351 struct usb_serial_port *port = tty->driver_data;
Alan Sternf5b09532009-09-01 11:38:44 -0400352 struct usb_serial *serial;
353 struct module *owner;
354
355 /* The console is magical. Do not hang up the console hardware
356 * or there will be tears.
357 */
Alan Sterncc56cd02009-09-01 11:39:13 -0400358 if (port->console)
Alan Sternf5b09532009-09-01 11:38:44 -0400359 return;
360
Alan Sternff8324d2009-09-01 11:39:51 -0400361 dbg("%s - port %d", __func__, port->number);
362
Alan Stern74556122009-09-01 11:39:40 -0400363 /* Standard shutdown processing */
364 tty_shutdown(tty);
365
Alan Sterncc56cd02009-09-01 11:39:13 -0400366 tty->driver_data = NULL;
367
Alan Sternf5b09532009-09-01 11:38:44 -0400368 serial = port->serial;
369 owner = serial->type->driver.owner;
370
371 mutex_lock(&serial->disc_mutex);
372 if (!serial->disconnected)
373 usb_autopm_put_interface(serial->interface);
374 mutex_unlock(&serial->disc_mutex);
375
376 usb_serial_put(serial);
377 module_put(owner);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700378}
379
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100380static int serial_write(struct tty_struct *tty, const unsigned char *buf,
381 int count)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200383 struct usb_serial_port *port = tty->driver_data;
Oliver Neukum3ff4fd92007-01-13 07:32:27 +0100384 int retval = -ENODEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700385
Alan Coxf34d7a52008-04-30 00:54:13 -0700386 if (port->serial->dev->state == USB_STATE_NOTATTACHED)
Luiz Fernando Capitulino487f9c62005-11-28 19:16:05 -0200387 goto exit;
388
Harvey Harrison441b62c2008-03-03 16:08:34 -0800389 dbg("%s - port %d, %d byte(s)", __func__, port->number, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700390
Alan Cox95da3102008-07-22 11:09:07 +0100391 /* count is managed under the mutex lock for the tty so cannot
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100392 drop to zero until after the last close completes */
Alan Cox95da3102008-07-22 11:09:07 +0100393 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700394
395 /* pass on to the driver specific version of this function */
Alan Cox95da3102008-07-22 11:09:07 +0100396 retval = port->serial->type->write(tty, port, buf, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397
398exit:
399 return retval;
400}
401
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100402static int serial_write_room(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700403{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200404 struct usb_serial_port *port = tty->driver_data;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800405 dbg("%s - port %d", __func__, port->number);
Alan Cox95da3102008-07-22 11:09:07 +0100406 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407 /* pass on to the driver specific version of this function */
Alan Cox95da3102008-07-22 11:09:07 +0100408 return port->serial->type->write_room(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700409}
410
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100411static int serial_chars_in_buffer(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700412{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200413 struct usb_serial_port *port = tty->driver_data;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800414 dbg("%s = port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700415
Alan Coxeff69372009-01-02 13:47:06 +0000416 /* if the device was unplugged then any remaining characters
417 fell out of the connector ;) */
418 if (port->serial->disconnected)
419 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700420 /* pass on to the driver specific version of this function */
Alan Cox95da3102008-07-22 11:09:07 +0100421 return port->serial->type->chars_in_buffer(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422}
423
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100424static void serial_throttle(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200426 struct usb_serial_port *port = tty->driver_data;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800427 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428
Alan Cox95da3102008-07-22 11:09:07 +0100429 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700430 /* pass on to the driver specific version of this function */
431 if (port->serial->type->throttle)
Alan Cox95da3102008-07-22 11:09:07 +0100432 port->serial->type->throttle(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700433}
434
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100435static void serial_unthrottle(struct tty_struct *tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200437 struct usb_serial_port *port = tty->driver_data;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800438 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700439
Alan Cox95da3102008-07-22 11:09:07 +0100440 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700441 /* pass on to the driver specific version of this function */
442 if (port->serial->type->unthrottle)
Alan Cox95da3102008-07-22 11:09:07 +0100443 port->serial->type->unthrottle(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700444}
445
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100446static int serial_ioctl(struct tty_struct *tty, struct file *file,
447 unsigned int cmd, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700448{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200449 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700450 int retval = -ENODEV;
451
Harvey Harrison441b62c2008-03-03 16:08:34 -0800452 dbg("%s - port %d, cmd 0x%.4x", __func__, port->number, cmd);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700453
Alan Cox95da3102008-07-22 11:09:07 +0100454 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700455
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100456 /* pass on to the driver specific version of this function
457 if it is available */
Alan Coxf34d7a52008-04-30 00:54:13 -0700458 if (port->serial->type->ioctl) {
Alan Cox95da3102008-07-22 11:09:07 +0100459 retval = port->serial->type->ioctl(tty, file, cmd, arg);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100460 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700461 retval = -ENOIOCTLCMD;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700462 return retval;
463}
464
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100465static void serial_set_termios(struct tty_struct *tty, struct ktermios *old)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200467 struct usb_serial_port *port = tty->driver_data;
Harvey Harrison441b62c2008-03-03 16:08:34 -0800468 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700469
Alan Cox95da3102008-07-22 11:09:07 +0100470 WARN_ON(!port->port.count);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100471 /* pass on to the driver specific version of this function
472 if it is available */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700473 if (port->serial->type->set_termios)
Alan Cox95da3102008-07-22 11:09:07 +0100474 port->serial->type->set_termios(tty, port, old);
Alan Cox33785092007-10-18 01:24:22 -0700475 else
476 tty_termios_copy_hw(tty->termios, old);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700477}
478
Alan Cox9e989662008-07-22 11:18:03 +0100479static int serial_break(struct tty_struct *tty, int break_state)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200481 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700482
Harvey Harrison441b62c2008-03-03 16:08:34 -0800483 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700484
Alan Cox95da3102008-07-22 11:09:07 +0100485 WARN_ON(!port->port.count);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100486 /* pass on to the driver specific version of this function
487 if it is available */
Alan Cox6b447f042009-01-02 13:48:56 +0000488 if (port->serial->type->break_ctl)
Alan Cox95da3102008-07-22 11:09:07 +0100489 port->serial->type->break_ctl(tty, break_state);
Alan Cox9e989662008-07-22 11:18:03 +0100490 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700491}
492
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700493static int serial_proc_show(struct seq_file *m, void *v)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700494{
495 struct usb_serial *serial;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496 int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497 char tmp[40];
498
Harvey Harrison441b62c2008-03-03 16:08:34 -0800499 dbg("%s", __func__);
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700500 seq_puts(m, "usbserinfo:1.0 driver:2.0\n");
501 for (i = 0; i < SERIAL_TTY_MINORS; ++i) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700502 serial = usb_serial_get_by_index(i);
503 if (serial == NULL)
504 continue;
505
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700506 seq_printf(m, "%d:", i);
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -0700507 if (serial->type->driver.owner)
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700508 seq_printf(m, " module:%s",
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100509 module_name(serial->type->driver.owner));
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700510 seq_printf(m, " name:\"%s\"",
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100511 serial->type->description);
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700512 seq_printf(m, " vendor:%04x product:%04x",
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100513 le16_to_cpu(serial->dev->descriptor.idVendor),
514 le16_to_cpu(serial->dev->descriptor.idProduct));
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700515 seq_printf(m, " num_ports:%d", serial->num_ports);
516 seq_printf(m, " port:%d", i - serial->minor + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700517 usb_make_path(serial->dev, tmp, sizeof(tmp));
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700518 seq_printf(m, " path:%s", tmp);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100519
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700520 seq_putc(m, '\n');
Guennadi Liakhovetski73e487f2006-04-25 07:46:17 +0200521 usb_serial_put(serial);
Alan Stern8bc2c1b2009-09-01 11:38:59 -0400522 mutex_unlock(&serial->disc_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700523 }
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700524 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700525}
526
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -0700527static int serial_proc_open(struct inode *inode, struct file *file)
528{
529 return single_open(file, serial_proc_show, NULL);
530}
531
532static const struct file_operations serial_proc_fops = {
533 .owner = THIS_MODULE,
534 .open = serial_proc_open,
535 .read = seq_read,
536 .llseek = seq_lseek,
537 .release = single_release,
538};
539
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100540static int serial_tiocmget(struct tty_struct *tty, struct file *file)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200542 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700543
Harvey Harrison441b62c2008-03-03 16:08:34 -0800544 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545
Alan Cox95da3102008-07-22 11:09:07 +0100546 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700547 if (port->serial->type->tiocmget)
Alan Cox95da3102008-07-22 11:09:07 +0100548 return port->serial->type->tiocmget(tty, file);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700549 return -EINVAL;
550}
551
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100552static int serial_tiocmset(struct tty_struct *tty, struct file *file,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700553 unsigned int set, unsigned int clear)
554{
Tobias Klauser81671dd2005-07-04 19:32:51 +0200555 struct usb_serial_port *port = tty->driver_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700556
Harvey Harrison441b62c2008-03-03 16:08:34 -0800557 dbg("%s - port %d", __func__, port->number);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558
Alan Cox95da3102008-07-22 11:09:07 +0100559 WARN_ON(!port->port.count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700560 if (port->serial->type->tiocmset)
Alan Cox95da3102008-07-22 11:09:07 +0100561 return port->serial->type->tiocmset(tty, file, set, clear);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 return -EINVAL;
563}
564
Pete Zaitcevcf2c7482006-05-22 21:58:49 -0700565/*
566 * We would be calling tty_wakeup here, but unfortunately some line
567 * disciplines have an annoying habit of calling tty->write from
568 * the write wakeup callback (e.g. n_hdlc.c).
569 */
570void usb_serial_port_softint(struct usb_serial_port *port)
571{
572 schedule_work(&port->work);
573}
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100574EXPORT_SYMBOL_GPL(usb_serial_port_softint);
Pete Zaitcevcf2c7482006-05-22 21:58:49 -0700575
David Howellsc4028952006-11-22 14:57:56 +0000576static void usb_serial_port_work(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700577{
David Howellsc4028952006-11-22 14:57:56 +0000578 struct usb_serial_port *port =
579 container_of(work, struct usb_serial_port, work);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700580 struct tty_struct *tty;
581
Harvey Harrison441b62c2008-03-03 16:08:34 -0800582 dbg("%s - port %d", __func__, port->number);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100583
Alan Cox4a90f092008-10-13 10:39:46 +0100584 tty = tty_port_tty_get(&port->port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700585 if (!tty)
586 return;
587
588 tty_wakeup(tty);
Alan Cox4a90f092008-10-13 10:39:46 +0100589 tty_kref_put(tty);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700590}
591
Oliver Neukum34ef50e2007-01-13 07:29:26 +0100592static void kill_traffic(struct usb_serial_port *port)
Pete Zaitcev34f8e762006-06-21 15:00:45 -0700593{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700594 usb_kill_urb(port->read_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700595 usb_kill_urb(port->write_urb);
Oliver Neukum5adceac2007-08-17 14:01:38 +0200596 /*
597 * This is tricky.
598 * Some drivers submit the read_urb in the
599 * handler for the write_urb or vice versa
600 * this order determines the order in which
601 * usb_kill_urb() must be used to reliably
602 * kill the URBs. As it is unknown here,
603 * both orders must be used in turn.
604 * The call below is not redundant.
605 */
606 usb_kill_urb(port->read_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700607 usb_kill_urb(port->interrupt_in_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700608 usb_kill_urb(port->interrupt_out_urb);
Oliver Neukum34ef50e2007-01-13 07:29:26 +0100609}
610
Alan Stern41bd34d2009-09-01 11:38:34 -0400611static void port_release(struct device *dev)
Oliver Neukum34ef50e2007-01-13 07:29:26 +0100612{
Alan Stern41bd34d2009-09-01 11:38:34 -0400613 struct usb_serial_port *port = to_usb_serial_port(dev);
614
615 dbg ("%s - %s", __func__, dev_name(dev));
616
Alan Stern2d931482009-04-14 11:31:02 -0400617 /*
618 * Stop all the traffic before cancelling the work, so that
619 * nobody will restart it by calling usb_serial_port_softint.
620 */
Oliver Neukum34ef50e2007-01-13 07:29:26 +0100621 kill_traffic(port);
Alan Stern2d931482009-04-14 11:31:02 -0400622 cancel_work_sync(&port->work);
623
Oliver Neukum34ef50e2007-01-13 07:29:26 +0100624 usb_free_urb(port->read_urb);
625 usb_free_urb(port->write_urb);
626 usb_free_urb(port->interrupt_in_urb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700627 usb_free_urb(port->interrupt_out_urb);
628 kfree(port->bulk_in_buffer);
629 kfree(port->bulk_out_buffer);
630 kfree(port->interrupt_in_buffer);
631 kfree(port->interrupt_out_buffer);
632 kfree(port);
633}
634
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100635static struct usb_serial *create_serial(struct usb_device *dev,
636 struct usb_interface *interface,
637 struct usb_serial_driver *driver)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700638{
639 struct usb_serial *serial;
640
Eric Sesterhenn80b6ca42006-02-27 21:29:43 +0100641 serial = kzalloc(sizeof(*serial), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700642 if (!serial) {
Harvey Harrison441b62c2008-03-03 16:08:34 -0800643 dev_err(&dev->dev, "%s - out of memory\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700644 return NULL;
645 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700646 serial->dev = usb_get_dev(dev);
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -0700647 serial->type = driver;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700648 serial->interface = interface;
649 kref_init(&serial->kref);
Oliver Neukuma1cd7e92008-01-16 17:18:52 +0100650 mutex_init(&serial->disc_mutex);
Alan Stern0282b7f2008-07-29 12:01:04 -0400651 serial->minor = SERIAL_TTY_NO_MINOR;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700652
653 return serial;
654}
655
Greg Kroah-Hartman93bacef2006-12-17 21:50:23 +0100656static const struct usb_device_id *match_dynamic_id(struct usb_interface *intf,
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100657 struct usb_serial_driver *drv)
Greg Kroah-Hartman93bacef2006-12-17 21:50:23 +0100658{
659 struct usb_dynid *dynid;
660
661 spin_lock(&drv->dynids.lock);
662 list_for_each_entry(dynid, &drv->dynids.list, node) {
663 if (usb_match_one_id(intf, &dynid->id)) {
664 spin_unlock(&drv->dynids.lock);
665 return &dynid->id;
666 }
667 }
668 spin_unlock(&drv->dynids.lock);
669 return NULL;
670}
671
672static const struct usb_device_id *get_iface_id(struct usb_serial_driver *drv,
673 struct usb_interface *intf)
674{
675 const struct usb_device_id *id;
676
677 id = usb_match_id(intf, drv->id_table);
678 if (id) {
679 dbg("static descriptor matches");
680 goto exit;
681 }
682 id = match_dynamic_id(intf, drv);
683 if (id)
684 dbg("dynamic descriptor matches");
685exit:
686 return id;
687}
688
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100689static struct usb_serial_driver *search_serial_device(
690 struct usb_interface *iface)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700691{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700692 const struct usb_device_id *id;
Alan Stern063a2da2007-10-10 16:24:06 -0400693 struct usb_serial_driver *drv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700694
Adrian Bunk93b1fae2006-01-10 00:13:33 +0100695 /* Check if the usb id matches a known device */
Alan Stern063a2da2007-10-10 16:24:06 -0400696 list_for_each_entry(drv, &usb_serial_driver_list, driver_list) {
697 id = get_iface_id(drv, iface);
Greg Kroah-Hartman93bacef2006-12-17 21:50:23 +0100698 if (id)
Alan Stern063a2da2007-10-10 16:24:06 -0400699 return drv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700700 }
701
702 return NULL;
703}
704
Alan Cox335f8512009-06-11 12:26:29 +0100705static int serial_carrier_raised(struct tty_port *port)
706{
707 struct usb_serial_port *p = container_of(port, struct usb_serial_port, port);
708 struct usb_serial_driver *drv = p->serial->type;
709 if (drv->carrier_raised)
710 return drv->carrier_raised(p);
711 /* No carrier control - don't block */
712 return 1;
713}
714
715static void serial_dtr_rts(struct tty_port *port, int on)
716{
717 struct usb_serial_port *p = container_of(port, struct usb_serial_port, port);
718 struct usb_serial_driver *drv = p->serial->type;
719 if (drv->dtr_rts)
720 drv->dtr_rts(p, on);
721}
722
723static const struct tty_port_operations serial_port_ops = {
724 .carrier_raised = serial_carrier_raised,
725 .dtr_rts = serial_dtr_rts,
726};
727
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728int usb_serial_probe(struct usb_interface *interface,
729 const struct usb_device_id *id)
730{
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100731 struct usb_device *dev = interface_to_usbdev(interface);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700732 struct usb_serial *serial = NULL;
733 struct usb_serial_port *port;
734 struct usb_host_interface *iface_desc;
735 struct usb_endpoint_descriptor *endpoint;
736 struct usb_endpoint_descriptor *interrupt_in_endpoint[MAX_NUM_PORTS];
737 struct usb_endpoint_descriptor *interrupt_out_endpoint[MAX_NUM_PORTS];
738 struct usb_endpoint_descriptor *bulk_in_endpoint[MAX_NUM_PORTS];
739 struct usb_endpoint_descriptor *bulk_out_endpoint[MAX_NUM_PORTS];
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -0700740 struct usb_serial_driver *type = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700741 int retval;
Andre Hauptdd9ca5d2008-06-18 15:56:00 +0200742 unsigned int minor;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700743 int buffer_size;
744 int i;
745 int num_interrupt_in = 0;
746 int num_interrupt_out = 0;
747 int num_bulk_in = 0;
748 int num_bulk_out = 0;
749 int num_ports = 0;
750 int max_endpoints;
751
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100752 lock_kernel(); /* guard against unloading a serial driver module */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700753 type = search_serial_device(interface);
754 if (!type) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100755 unlock_kernel();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756 dbg("none matched");
757 return -ENODEV;
758 }
759
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100760 serial = create_serial(dev, interface, type);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700761 if (!serial) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100762 unlock_kernel();
Harvey Harrison441b62c2008-03-03 16:08:34 -0800763 dev_err(&interface->dev, "%s - out of memory\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700764 return -ENOMEM;
765 }
766
767 /* if this device type has a probe function, call it */
768 if (type->probe) {
769 const struct usb_device_id *id;
770
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -0700771 if (!try_module_get(type->driver.owner)) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100772 unlock_kernel();
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100773 dev_err(&interface->dev,
774 "module get failed, exiting\n");
775 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776 return -EIO;
777 }
778
Greg Kroah-Hartman93bacef2006-12-17 21:50:23 +0100779 id = get_iface_id(type, interface);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700780 retval = type->probe(serial, id);
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -0700781 module_put(type->driver.owner);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700782
783 if (retval) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100784 unlock_kernel();
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100785 dbg("sub driver rejected device");
786 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700787 return retval;
788 }
789 }
790
791 /* descriptor matches, let's find the endpoints needed */
792 /* check out the endpoints */
793 iface_desc = interface->cur_altsetting;
794 for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {
795 endpoint = &iface_desc->endpoint[i].desc;
Luiz Fernando N. Capitulino4fa1bbf2006-09-27 11:58:53 -0700796
797 if (usb_endpoint_is_bulk_in(endpoint)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700798 /* we found a bulk in endpoint */
799 dbg("found bulk in on endpoint %d", i);
800 bulk_in_endpoint[num_bulk_in] = endpoint;
801 ++num_bulk_in;
802 }
803
Luiz Fernando N. Capitulino4fa1bbf2006-09-27 11:58:53 -0700804 if (usb_endpoint_is_bulk_out(endpoint)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700805 /* we found a bulk out endpoint */
806 dbg("found bulk out on endpoint %d", i);
807 bulk_out_endpoint[num_bulk_out] = endpoint;
808 ++num_bulk_out;
809 }
Luiz Fernando N. Capitulino4fa1bbf2006-09-27 11:58:53 -0700810
811 if (usb_endpoint_is_int_in(endpoint)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700812 /* we found a interrupt in endpoint */
813 dbg("found interrupt in on endpoint %d", i);
814 interrupt_in_endpoint[num_interrupt_in] = endpoint;
815 ++num_interrupt_in;
816 }
817
Luiz Fernando N. Capitulino4fa1bbf2006-09-27 11:58:53 -0700818 if (usb_endpoint_is_int_out(endpoint)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819 /* we found an interrupt out endpoint */
820 dbg("found interrupt out on endpoint %d", i);
821 interrupt_out_endpoint[num_interrupt_out] = endpoint;
822 ++num_interrupt_out;
823 }
824 }
825
826#if defined(CONFIG_USB_SERIAL_PL2303) || defined(CONFIG_USB_SERIAL_PL2303_MODULE)
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100827 /* BEGIN HORRIBLE HACK FOR PL2303 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700828 /* this is needed due to the looney way its endpoints are set up */
829 if (((le16_to_cpu(dev->descriptor.idVendor) == PL2303_VENDOR_ID) &&
830 (le16_to_cpu(dev->descriptor.idProduct) == PL2303_PRODUCT_ID)) ||
831 ((le16_to_cpu(dev->descriptor.idVendor) == ATEN_VENDOR_ID) &&
Johannes Steingraeber8fd80132006-09-16 16:17:34 +0200832 (le16_to_cpu(dev->descriptor.idProduct) == ATEN_PRODUCT_ID)) ||
833 ((le16_to_cpu(dev->descriptor.idVendor) == ALCOR_VENDOR_ID) &&
Andreas Bombece816cf2008-09-14 01:58:55 +0200834 (le16_to_cpu(dev->descriptor.idProduct) == ALCOR_PRODUCT_ID)) ||
835 ((le16_to_cpu(dev->descriptor.idVendor) == SIEMENS_VENDOR_ID) &&
836 (le16_to_cpu(dev->descriptor.idProduct) == SIEMENS_PRODUCT_ID_EF81))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700837 if (interface != dev->actconfig->interface[0]) {
838 /* check out the endpoints of the other interface*/
839 iface_desc = dev->actconfig->interface[0]->cur_altsetting;
840 for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {
841 endpoint = &iface_desc->endpoint[i].desc;
Luiz Fernando N. Capitulino4fa1bbf2006-09-27 11:58:53 -0700842 if (usb_endpoint_is_int_in(endpoint)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700843 /* we found a interrupt in endpoint */
844 dbg("found interrupt in for Prolific device on separate interface");
845 interrupt_in_endpoint[num_interrupt_in] = endpoint;
846 ++num_interrupt_in;
847 }
848 }
849 }
850
851 /* Now make sure the PL-2303 is configured correctly.
852 * If not, give up now and hope this hack will work
853 * properly during a later invocation of usb_serial_probe
854 */
855 if (num_bulk_in == 0 || num_bulk_out == 0) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100856 unlock_kernel();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700857 dev_info(&interface->dev, "PL-2303 hack: descriptors matched but endpoints did not\n");
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100858 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700859 return -ENODEV;
860 }
861 }
862 /* END HORRIBLE HACK FOR PL2303 */
863#endif
864
Linus Torvalds1da177e2005-04-16 15:20:36 -0700865#ifdef CONFIG_USB_SERIAL_GENERIC
866 if (type == &usb_serial_generic_device) {
867 num_ports = num_bulk_out;
868 if (num_ports == 0) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100869 unlock_kernel();
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100870 dev_err(&interface->dev,
871 "Generic device with no bulk out, not allowed.\n");
872 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700873 return -EIO;
874 }
875 }
876#endif
877 if (!num_ports) {
878 /* if this device type has a calc_num_ports function, call it */
879 if (type->calc_num_ports) {
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -0700880 if (!try_module_get(type->driver.owner)) {
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100881 unlock_kernel();
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100882 dev_err(&interface->dev,
883 "module get failed, exiting\n");
884 kfree(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885 return -EIO;
886 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100887 num_ports = type->calc_num_ports(serial);
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -0700888 module_put(type->driver.owner);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700889 }
890 if (!num_ports)
891 num_ports = type->num_ports;
892 }
893
Linus Torvalds1da177e2005-04-16 15:20:36 -0700894 serial->num_ports = num_ports;
895 serial->num_bulk_in = num_bulk_in;
896 serial->num_bulk_out = num_bulk_out;
897 serial->num_interrupt_in = num_interrupt_in;
898 serial->num_interrupt_out = num_interrupt_out;
899
Alan Stern063a2da2007-10-10 16:24:06 -0400900 /* found all that we need */
901 dev_info(&interface->dev, "%s converter detected\n",
902 type->description);
903
Linus Torvalds1da177e2005-04-16 15:20:36 -0700904 /* create our ports, we need as many as the max endpoints */
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100905 /* we don't use num_ports here because some devices have more
906 endpoint pairs than ports */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700907 max_endpoints = max(num_bulk_in, num_bulk_out);
908 max_endpoints = max(max_endpoints, num_interrupt_in);
909 max_endpoints = max(max_endpoints, num_interrupt_out);
910 max_endpoints = max(max_endpoints, (int)serial->num_ports);
911 serial->num_port_pointers = max_endpoints;
Oliver Neukum4b10f0f2007-01-13 07:31:27 +0100912 unlock_kernel();
913
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100914 dbg("%s - setting up %d port structures for this device",
915 __func__, max_endpoints);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700916 for (i = 0; i < max_endpoints; ++i) {
Eric Sesterhenn80b6ca42006-02-27 21:29:43 +0100917 port = kzalloc(sizeof(struct usb_serial_port), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700918 if (!port)
919 goto probe_error;
Alan Cox4a90f092008-10-13 10:39:46 +0100920 tty_port_init(&port->port);
Alan Cox335f8512009-06-11 12:26:29 +0100921 port->port.ops = &serial_port_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700922 port->serial = serial;
Greg Kroah-Hartman507ca9b2005-04-23 12:49:16 -0700923 spin_lock_init(&port->lock);
Luiz Fernando Capitulino1ce7dd22006-03-24 17:12:31 -0300924 mutex_init(&port->mutex);
David Howellsc4028952006-11-22 14:57:56 +0000925 INIT_WORK(&port->work, usb_serial_port_work);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700926 serial->port[i] = port;
Alan Stern41bd34d2009-09-01 11:38:34 -0400927 port->dev.parent = &interface->dev;
928 port->dev.driver = NULL;
929 port->dev.bus = &usb_serial_bus_type;
930 port->dev.release = &port_release;
931 device_initialize(&port->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700932 }
933
934 /* set up the endpoint information */
935 for (i = 0; i < num_bulk_in; ++i) {
936 endpoint = bulk_in_endpoint[i];
937 port = serial->port[i];
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100938 port->read_urb = usb_alloc_urb(0, GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700939 if (!port->read_urb) {
940 dev_err(&interface->dev, "No free urbs available\n");
941 goto probe_error;
942 }
943 buffer_size = le16_to_cpu(endpoint->wMaxPacketSize);
944 port->bulk_in_size = buffer_size;
945 port->bulk_in_endpointAddress = endpoint->bEndpointAddress;
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100946 port->bulk_in_buffer = kmalloc(buffer_size, GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700947 if (!port->bulk_in_buffer) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100948 dev_err(&interface->dev,
949 "Couldn't allocate bulk_in_buffer\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950 goto probe_error;
951 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100952 usb_fill_bulk_urb(port->read_urb, dev,
953 usb_rcvbulkpipe(dev,
954 endpoint->bEndpointAddress),
955 port->bulk_in_buffer, buffer_size,
956 serial->type->read_bulk_callback, port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700957 }
958
959 for (i = 0; i < num_bulk_out; ++i) {
960 endpoint = bulk_out_endpoint[i];
961 port = serial->port[i];
962 port->write_urb = usb_alloc_urb(0, GFP_KERNEL);
963 if (!port->write_urb) {
964 dev_err(&interface->dev, "No free urbs available\n");
965 goto probe_error;
966 }
967 buffer_size = le16_to_cpu(endpoint->wMaxPacketSize);
968 port->bulk_out_size = buffer_size;
969 port->bulk_out_endpointAddress = endpoint->bEndpointAddress;
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100970 port->bulk_out_buffer = kmalloc(buffer_size, GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700971 if (!port->bulk_out_buffer) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100972 dev_err(&interface->dev,
973 "Couldn't allocate bulk_out_buffer\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700974 goto probe_error;
975 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100976 usb_fill_bulk_urb(port->write_urb, dev,
977 usb_sndbulkpipe(dev,
978 endpoint->bEndpointAddress),
979 port->bulk_out_buffer, buffer_size,
980 serial->type->write_bulk_callback, port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700981 }
982
983 if (serial->type->read_int_callback) {
984 for (i = 0; i < num_interrupt_in; ++i) {
985 endpoint = interrupt_in_endpoint[i];
986 port = serial->port[i];
987 port->interrupt_in_urb = usb_alloc_urb(0, GFP_KERNEL);
988 if (!port->interrupt_in_urb) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100989 dev_err(&interface->dev,
990 "No free urbs available\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700991 goto probe_error;
992 }
993 buffer_size = le16_to_cpu(endpoint->wMaxPacketSize);
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100994 port->interrupt_in_endpointAddress =
995 endpoint->bEndpointAddress;
996 port->interrupt_in_buffer = kmalloc(buffer_size,
997 GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700998 if (!port->interrupt_in_buffer) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +0100999 dev_err(&interface->dev,
1000 "Couldn't allocate interrupt_in_buffer\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001001 goto probe_error;
1002 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001003 usb_fill_int_urb(port->interrupt_in_urb, dev,
1004 usb_rcvintpipe(dev,
1005 endpoint->bEndpointAddress),
1006 port->interrupt_in_buffer, buffer_size,
1007 serial->type->read_int_callback, port,
1008 endpoint->bInterval);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001009 }
1010 } else if (num_interrupt_in) {
1011 dbg("the device claims to support interrupt in transfers, but read_int_callback is not defined");
1012 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001013
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014 if (serial->type->write_int_callback) {
1015 for (i = 0; i < num_interrupt_out; ++i) {
1016 endpoint = interrupt_out_endpoint[i];
1017 port = serial->port[i];
1018 port->interrupt_out_urb = usb_alloc_urb(0, GFP_KERNEL);
1019 if (!port->interrupt_out_urb) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001020 dev_err(&interface->dev,
1021 "No free urbs available\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001022 goto probe_error;
1023 }
1024 buffer_size = le16_to_cpu(endpoint->wMaxPacketSize);
1025 port->interrupt_out_size = buffer_size;
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001026 port->interrupt_out_endpointAddress =
1027 endpoint->bEndpointAddress;
1028 port->interrupt_out_buffer = kmalloc(buffer_size,
1029 GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001030 if (!port->interrupt_out_buffer) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001031 dev_err(&interface->dev,
1032 "Couldn't allocate interrupt_out_buffer\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001033 goto probe_error;
1034 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001035 usb_fill_int_urb(port->interrupt_out_urb, dev,
1036 usb_sndintpipe(dev,
1037 endpoint->bEndpointAddress),
1038 port->interrupt_out_buffer, buffer_size,
1039 serial->type->write_int_callback, port,
1040 endpoint->bInterval);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001041 }
1042 } else if (num_interrupt_out) {
1043 dbg("the device claims to support interrupt out transfers, but write_int_callback is not defined");
1044 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001045
Linus Torvalds1da177e2005-04-16 15:20:36 -07001046 /* if this device type has an attach function, call it */
1047 if (type->attach) {
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -07001048 if (!try_module_get(type->driver.owner)) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001049 dev_err(&interface->dev,
1050 "module get failed, exiting\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001051 goto probe_error;
1052 }
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001053 retval = type->attach(serial);
Greg Kroah-Hartman18fcac32005-06-20 21:15:16 -07001054 module_put(type->driver.owner);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001055 if (retval < 0)
1056 goto probe_error;
1057 if (retval > 0) {
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001058 /* quietly accept this device, but don't bind to a
1059 serial port as it's about to disappear */
Alan Stern0a3c8542009-05-27 11:25:52 -04001060 serial->num_ports = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001061 goto exit;
1062 }
1063 }
1064
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001065 if (get_free_serial(serial, num_ports, &minor) == NULL) {
Oliver Neukum34ef50e2007-01-13 07:29:26 +01001066 dev_err(&interface->dev, "No more free serial devices\n");
1067 goto probe_error;
1068 }
Oliver Neukumc744f992007-02-26 15:43:00 +01001069 serial->minor = minor;
Oliver Neukum34ef50e2007-01-13 07:29:26 +01001070
Linus Torvalds1da177e2005-04-16 15:20:36 -07001071 /* register all of the individual ports with the driver core */
1072 for (i = 0; i < num_ports; ++i) {
1073 port = serial->port[i];
Kay Sievers0031a062008-05-02 06:02:41 +02001074 dev_set_name(&port->dev, "ttyUSB%d", port->number);
Kay Sievers7071a3c2008-05-02 06:02:41 +02001075 dbg ("%s - registering %s", __func__, dev_name(&port->dev));
Alan Sternc706ebd2009-06-02 11:54:11 -04001076 port->dev_state = PORT_REGISTERING;
Alan Stern41bd34d2009-09-01 11:38:34 -04001077 retval = device_add(&port->dev);
Alan Sternc706ebd2009-06-02 11:54:11 -04001078 if (retval) {
Greg Kroah-Hartman13f4db92006-08-28 11:43:25 -07001079 dev_err(&port->dev, "Error registering port device, "
1080 "continuing\n");
Alan Sternc706ebd2009-06-02 11:54:11 -04001081 port->dev_state = PORT_UNREGISTERED;
1082 } else {
1083 port->dev_state = PORT_REGISTERED;
1084 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001085 }
1086
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001087 usb_serial_console_init(debug, minor);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001088
1089exit:
1090 /* success */
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001091 usb_set_intfdata(interface, serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001092 return 0;
1093
1094probe_error:
Alan Stern41bd34d2009-09-01 11:38:34 -04001095 usb_serial_put(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001096 return -EIO;
1097}
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001098EXPORT_SYMBOL_GPL(usb_serial_probe);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001099
1100void usb_serial_disconnect(struct usb_interface *interface)
1101{
1102 int i;
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001103 struct usb_serial *serial = usb_get_intfdata(interface);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001104 struct device *dev = &interface->dev;
1105 struct usb_serial_port *port;
1106
Guennadi Liakhovetski73e487f2006-04-25 07:46:17 +02001107 usb_serial_console_disconnect(serial);
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001108 dbg("%s", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001109
Oliver Neukuma1cd7e92008-01-16 17:18:52 +01001110 mutex_lock(&serial->disc_mutex);
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001111 usb_set_intfdata(interface, NULL);
Oliver Neukuma1cd7e92008-01-16 17:18:52 +01001112 /* must set a flag, to signal subdrivers */
1113 serial->disconnected = 1;
Alan Stern2d931482009-04-14 11:31:02 -04001114 mutex_unlock(&serial->disc_mutex);
1115
Oliver Neukuma1cd7e92008-01-16 17:18:52 +01001116 for (i = 0; i < serial->num_ports; ++i) {
1117 port = serial->port[i];
1118 if (port) {
Alan Cox4a90f092008-10-13 10:39:46 +01001119 struct tty_struct *tty = tty_port_tty_get(&port->port);
1120 if (tty) {
Alan Coxd2b39182009-09-19 13:13:23 -07001121 tty_vhangup(tty);
Alan Cox4a90f092008-10-13 10:39:46 +01001122 tty_kref_put(tty);
1123 }
Oliver Neukuma1cd7e92008-01-16 17:18:52 +01001124 kill_traffic(port);
Alan Stern2d931482009-04-14 11:31:02 -04001125 cancel_work_sync(&port->work);
Alan Sternc706ebd2009-06-02 11:54:11 -04001126 if (port->dev_state == PORT_REGISTERED) {
1127
1128 /* Make sure the port is bound so that the
1129 * driver's port_remove method is called.
1130 */
1131 if (!port->dev.driver) {
1132 int rc;
1133
1134 port->dev.driver =
1135 &serial->type->driver;
1136 rc = device_bind_driver(&port->dev);
1137 }
1138 port->dev_state = PORT_UNREGISTERING;
1139 device_del(&port->dev);
1140 port->dev_state = PORT_UNREGISTERED;
1141 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001142 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001143 }
Alan Sternf9c99bb2009-06-02 11:53:55 -04001144 serial->type->disconnect(serial);
Alan Stern2d931482009-04-14 11:31:02 -04001145
Alan Stern41bd34d2009-09-01 11:38:34 -04001146 /* let the last holder of this object cause it to be cleaned up */
Oliver Neukuma1cd7e92008-01-16 17:18:52 +01001147 usb_serial_put(serial);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148 dev_info(dev, "device disconnected\n");
1149}
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001150EXPORT_SYMBOL_GPL(usb_serial_disconnect);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001151
Oliver Neukumec225592007-04-27 20:54:57 +02001152int usb_serial_suspend(struct usb_interface *intf, pm_message_t message)
1153{
1154 struct usb_serial *serial = usb_get_intfdata(intf);
1155 struct usb_serial_port *port;
1156 int i, r = 0;
1157
Oliver Neukumf8bece82009-02-05 16:54:25 +01001158 serial->suspending = 1;
1159
Oliver Neukum81e5b232009-07-21 08:47:34 +02001160 if (serial->type->suspend) {
1161 r = serial->type->suspend(serial, message);
1162 if (r < 0)
1163 goto err_out;
1164 }
1165
Oliver Neukume31c1882007-07-23 08:58:39 +02001166 for (i = 0; i < serial->num_ports; ++i) {
1167 port = serial->port[i];
1168 if (port)
1169 kill_traffic(port);
Oliver Neukumec225592007-04-27 20:54:57 +02001170 }
1171
Oliver Neukum81e5b232009-07-21 08:47:34 +02001172err_out:
Oliver Neukumec225592007-04-27 20:54:57 +02001173 return r;
1174}
1175EXPORT_SYMBOL(usb_serial_suspend);
1176
1177int usb_serial_resume(struct usb_interface *intf)
1178{
1179 struct usb_serial *serial = usb_get_intfdata(intf);
Oliver Neukumc49cfa912009-02-06 18:06:43 +01001180 int rv;
Oliver Neukumec225592007-04-27 20:54:57 +02001181
Oliver Neukumf8bece82009-02-05 16:54:25 +01001182 serial->suspending = 0;
Sarah Sharp8abaee22007-10-25 10:58:43 -07001183 if (serial->type->resume)
Oliver Neukumc49cfa912009-02-06 18:06:43 +01001184 rv = serial->type->resume(serial);
1185 else
1186 rv = usb_serial_generic_resume(serial);
Oliver Neukumf8bece82009-02-05 16:54:25 +01001187
Oliver Neukumc49cfa912009-02-06 18:06:43 +01001188 return rv;
Oliver Neukumec225592007-04-27 20:54:57 +02001189}
1190EXPORT_SYMBOL(usb_serial_resume);
1191
Jeff Dikeb68e31d2006-10-02 02:17:18 -07001192static const struct tty_operations serial_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001193 .open = serial_open,
1194 .close = serial_close,
1195 .write = serial_write,
Alan Cox335f8512009-06-11 12:26:29 +01001196 .hangup = serial_hangup,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197 .write_room = serial_write_room,
1198 .ioctl = serial_ioctl,
1199 .set_termios = serial_set_termios,
1200 .throttle = serial_throttle,
1201 .unthrottle = serial_unthrottle,
1202 .break_ctl = serial_break,
1203 .chars_in_buffer = serial_chars_in_buffer,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001204 .tiocmget = serial_tiocmget,
1205 .tiocmset = serial_tiocmset,
Alan Stern74556122009-09-01 11:39:40 -04001206 .shutdown = serial_release,
Alan Coxfe1ae7f2009-09-19 13:13:33 -07001207 .install = serial_install,
Alexey Dobriyan6fd69d32009-03-31 15:19:21 -07001208 .proc_fops = &serial_proc_fops,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001209};
1210
Alan Cox335f8512009-06-11 12:26:29 +01001211
Linus Torvalds1da177e2005-04-16 15:20:36 -07001212struct tty_driver *usb_serial_tty_driver;
1213
1214static int __init usb_serial_init(void)
1215{
1216 int i;
1217 int result;
1218
1219 usb_serial_tty_driver = alloc_tty_driver(SERIAL_TTY_MINORS);
1220 if (!usb_serial_tty_driver)
1221 return -ENOMEM;
1222
1223 /* Initialize our global data */
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001224 for (i = 0; i < SERIAL_TTY_MINORS; ++i)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001225 serial_table[i] = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001226
1227 result = bus_register(&usb_serial_bus_type);
1228 if (result) {
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001229 printk(KERN_ERR "usb-serial: %s - registering bus driver "
1230 "failed\n", __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001231 goto exit_bus;
1232 }
1233
Linus Torvalds1da177e2005-04-16 15:20:36 -07001234 usb_serial_tty_driver->owner = THIS_MODULE;
1235 usb_serial_tty_driver->driver_name = "usbserial";
Linus Torvalds1da177e2005-04-16 15:20:36 -07001236 usb_serial_tty_driver->name = "ttyUSB";
1237 usb_serial_tty_driver->major = SERIAL_TTY_MAJOR;
1238 usb_serial_tty_driver->minor_start = 0;
1239 usb_serial_tty_driver->type = TTY_DRIVER_TYPE_SERIAL;
1240 usb_serial_tty_driver->subtype = SERIAL_TYPE_NORMAL;
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001241 usb_serial_tty_driver->flags = TTY_DRIVER_REAL_RAW |
1242 TTY_DRIVER_DYNAMIC_DEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001243 usb_serial_tty_driver->init_termios = tty_std_termios;
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001244 usb_serial_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD
1245 | HUPCL | CLOCAL;
Alan Coxa5b6f602008-04-08 17:16:06 +01001246 usb_serial_tty_driver->init_termios.c_ispeed = 9600;
1247 usb_serial_tty_driver->init_termios.c_ospeed = 9600;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001248 tty_set_operations(usb_serial_tty_driver, &serial_ops);
1249 result = tty_register_driver(usb_serial_tty_driver);
1250 if (result) {
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001251 printk(KERN_ERR "usb-serial: %s - tty_register_driver failed\n",
1252 __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001253 goto exit_reg_driver;
1254 }
1255
1256 /* register the USB driver */
1257 result = usb_register(&usb_serial_driver);
1258 if (result < 0) {
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001259 printk(KERN_ERR "usb-serial: %s - usb_register failed\n",
1260 __func__);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001261 goto exit_tty;
1262 }
1263
Greg Kroah-Hartman06299db2005-05-26 05:55:55 -07001264 /* register the generic driver, if we should */
1265 result = usb_serial_generic_register(debug);
1266 if (result < 0) {
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001267 printk(KERN_ERR "usb-serial: %s - registering generic "
1268 "driver failed\n", __func__);
Greg Kroah-Hartman06299db2005-05-26 05:55:55 -07001269 goto exit_generic;
1270 }
1271
Greg Kroah-Hartmanc197a8d2008-08-18 13:21:04 -07001272 printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_DESC "\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001273
1274 return result;
1275
Greg Kroah-Hartman06299db2005-05-26 05:55:55 -07001276exit_generic:
1277 usb_deregister(&usb_serial_driver);
1278
Linus Torvalds1da177e2005-04-16 15:20:36 -07001279exit_tty:
1280 tty_unregister_driver(usb_serial_tty_driver);
1281
1282exit_reg_driver:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001283 bus_unregister(&usb_serial_bus_type);
1284
1285exit_bus:
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001286 printk(KERN_ERR "usb-serial: %s - returning with error %d\n",
1287 __func__, result);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001288 put_tty_driver(usb_serial_tty_driver);
1289 return result;
1290}
1291
1292
1293static void __exit usb_serial_exit(void)
1294{
1295 usb_serial_console_exit();
1296
1297 usb_serial_generic_deregister();
1298
1299 usb_deregister(&usb_serial_driver);
1300 tty_unregister_driver(usb_serial_tty_driver);
1301 put_tty_driver(usb_serial_tty_driver);
1302 bus_unregister(&usb_serial_bus_type);
1303}
1304
1305
1306module_init(usb_serial_init);
1307module_exit(usb_serial_exit);
1308
1309#define set_to_generic_if_null(type, function) \
1310 do { \
1311 if (!type->function) { \
1312 type->function = usb_serial_generic_##function; \
1313 dbg("Had to override the " #function \
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001314 " usb serial operation with the generic one.");\
Linus Torvalds1da177e2005-04-16 15:20:36 -07001315 } \
1316 } while (0)
1317
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -07001318static void fixup_generic(struct usb_serial_driver *device)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319{
1320 set_to_generic_if_null(device, open);
1321 set_to_generic_if_null(device, write);
1322 set_to_generic_if_null(device, close);
1323 set_to_generic_if_null(device, write_room);
1324 set_to_generic_if_null(device, chars_in_buffer);
1325 set_to_generic_if_null(device, read_bulk_callback);
1326 set_to_generic_if_null(device, write_bulk_callback);
Alan Sternf9c99bb2009-06-02 11:53:55 -04001327 set_to_generic_if_null(device, disconnect);
1328 set_to_generic_if_null(device, release);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001329}
1330
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001331int usb_serial_register(struct usb_serial_driver *driver)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001332{
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001333 /* must be called with BKL held */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001334 int retval;
1335
Dave Younge4abe662009-02-14 21:21:13 +08001336 if (usb_disabled())
1337 return -ENODEV;
1338
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -07001339 fixup_generic(driver);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001340
Greg Kroah-Hartman269bda12005-06-20 21:15:16 -07001341 if (!driver->description)
1342 driver->description = driver->driver.name;
1343
Linus Torvalds1da177e2005-04-16 15:20:36 -07001344 /* Add this device to our list of devices */
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -07001345 list_add(&driver->driver_list, &usb_serial_driver_list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001346
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -07001347 retval = usb_serial_bus_register(driver);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001348 if (retval) {
Greg Kroah-Hartman194343d2008-08-20 16:56:34 -07001349 printk(KERN_ERR "usb-serial: problem %d when registering "
1350 "driver %s\n", retval, driver->description);
Greg Kroah-Hartmanea653702005-06-20 21:15:16 -07001351 list_del(&driver->driver_list);
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001352 } else
Greg Kroah-Hartmanc197a8d2008-08-18 13:21:04 -07001353 printk(KERN_INFO "USB Serial support registered for %s\n",
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001354 driver->description);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001355
1356 return retval;
1357}
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001358EXPORT_SYMBOL_GPL(usb_serial_register);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001359
1360
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001361void usb_serial_deregister(struct usb_serial_driver *device)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001362{
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001363 /* must be called with BKL held */
Greg Kroah-Hartmanc197a8d2008-08-18 13:21:04 -07001364 printk(KERN_INFO "USB Serial deregistering driver %s\n",
1365 device->description);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001366 list_del(&device->driver_list);
1367 usb_serial_bus_deregister(device);
1368}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001369EXPORT_SYMBOL_GPL(usb_serial_deregister);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001370
1371/* Module information */
Alan Coxa8d6f0a2008-07-22 11:12:24 +01001372MODULE_AUTHOR(DRIVER_AUTHOR);
1373MODULE_DESCRIPTION(DRIVER_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001374MODULE_LICENSE("GPL");
1375
1376module_param(debug, bool, S_IRUGO | S_IWUSR);
1377MODULE_PARM_DESC(debug, "Debug enabled or not");