generic.c 16.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 * USB Serial Converter Generic functions
 *
 * Copyright (C) 1999 - 2002 Greg Kroah-Hartman (greg@kroah.com)
 *
 *	This program is free software; you can redistribute it and/or
 *	modify it under the terms of the GNU General Public License version
 *	2 as published by the Free Software Foundation.
 *
 */

#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/slab.h>
#include <linux/tty.h>
#include <linux/tty_flip.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/usb.h>
20
#include <linux/usb/serial.h>
A
Alan Cox 已提交
21
#include <linux/uaccess.h>
22
#include <linux/kfifo.h>
23
#include <linux/serial.h>
24

L
Linus Torvalds 已提交
25 26 27
static int debug;

#ifdef CONFIG_USB_SERIAL_GENERIC
28 29 30 31

static int generic_probe(struct usb_interface *interface,
			 const struct usb_device_id *id);

L
Linus Torvalds 已提交
32 33 34 35 36 37 38 39 40 41 42
static __u16 vendor  = 0x05f9;
static __u16 product = 0xffff;

module_param(vendor, ushort, 0);
MODULE_PARM_DESC(vendor, "User specified USB idVendor");

module_param(product, ushort, 0);
MODULE_PARM_DESC(product, "User specified USB idProduct");

static struct usb_device_id generic_device_ids[2]; /* Initially all zeroes. */

43 44
/* we want to look at all devices, as the vendor/product id can change
 * depending on the command line argument */
45
static const struct usb_device_id generic_serial_ids[] = {
46 47 48 49 50 51 52 53 54 55 56 57
	{.driver_info = 42},
	{}
};

static struct usb_driver generic_driver = {
	.name =		"usbserial_generic",
	.probe =	generic_probe,
	.disconnect =	usb_serial_disconnect,
	.id_table =	generic_serial_ids,
	.no_dynamic_id =	1,
};

L
Linus Torvalds 已提交
58
/* All of the device info needed for the Generic Serial Converter */
59
struct usb_serial_driver usb_serial_generic_device = {
60 61
	.driver = {
		.owner =	THIS_MODULE,
62
		.name =		"generic",
63
	},
L
Linus Torvalds 已提交
64
	.id_table =		generic_device_ids,
65
	.usb_driver = 		&generic_driver,
L
Linus Torvalds 已提交
66
	.num_ports =		1,
67 68
	.disconnect =		usb_serial_generic_disconnect,
	.release =		usb_serial_generic_release,
69 70
	.throttle =		usb_serial_generic_throttle,
	.unthrottle =		usb_serial_generic_unthrottle,
71
	.resume =		usb_serial_generic_resume,
L
Linus Torvalds 已提交
72 73 74 75 76 77 78 79 80 81 82 83 84 85
};

static int generic_probe(struct usb_interface *interface,
			       const struct usb_device_id *id)
{
	const struct usb_device_id *id_pattern;

	id_pattern = usb_match_id(interface, generic_device_ids);
	if (id_pattern != NULL)
		return usb_serial_probe(interface, id);
	return -ENODEV;
}
#endif

A
Alan Cox 已提交
86
int usb_serial_generic_register(int _debug)
L
Linus Torvalds 已提交
87 88 89 90 91 92 93
{
	int retval = 0;

	debug = _debug;
#ifdef CONFIG_USB_SERIAL_GENERIC
	generic_device_ids[0].idVendor = vendor;
	generic_device_ids[0].idProduct = product;
A
Alan Cox 已提交
94 95
	generic_device_ids[0].match_flags =
		USB_DEVICE_ID_MATCH_VENDOR | USB_DEVICE_ID_MATCH_PRODUCT;
L
Linus Torvalds 已提交
96 97

	/* register our generic driver with ourselves */
A
Alan Cox 已提交
98
	retval = usb_serial_register(&usb_serial_generic_device);
L
Linus Torvalds 已提交
99 100 101 102 103 104 105 106 107 108
	if (retval)
		goto exit;
	retval = usb_register(&generic_driver);
	if (retval)
		usb_serial_deregister(&usb_serial_generic_device);
exit:
#endif
	return retval;
}

A
Alan Cox 已提交
109
void usb_serial_generic_deregister(void)
L
Linus Torvalds 已提交
110 111 112 113
{
#ifdef CONFIG_USB_SERIAL_GENERIC
	/* remove our generic driver */
	usb_deregister(&generic_driver);
A
Alan Cox 已提交
114
	usb_serial_deregister(&usb_serial_generic_device);
L
Linus Torvalds 已提交
115 116 117
#endif
}

118
int usb_serial_generic_open(struct tty_struct *tty, struct usb_serial_port *port)
L
Linus Torvalds 已提交
119 120 121
{
	struct usb_serial *serial = port->serial;
	int result = 0;
122
	unsigned long flags;
L
Linus Torvalds 已提交
123

124
	dbg("%s - port %d", __func__, port->number);
L
Linus Torvalds 已提交
125

126 127 128 129 130 131 132
	/* clear the throttle flags */
	spin_lock_irqsave(&port->lock, flags);
	port->throttled = 0;
	port->throttle_req = 0;
	spin_unlock_irqrestore(&port->lock, flags);

	/* if we have a bulk endpoint, start reading from it */
133
	if (port->bulk_in_size) {
L
Linus Torvalds 已提交
134
		/* Start reading from the device */
A
Alan Cox 已提交
135 136 137
		usb_fill_bulk_urb(port->read_urb, serial->dev,
				   usb_rcvbulkpipe(serial->dev,
						port->bulk_in_endpointAddress),
L
Linus Torvalds 已提交
138 139 140 141 142 143 144 145
				   port->read_urb->transfer_buffer,
				   port->read_urb->transfer_buffer_length,
				   ((serial->type->read_bulk_callback) ?
				     serial->type->read_bulk_callback :
				     usb_serial_generic_read_bulk_callback),
				   port);
		result = usb_submit_urb(port->read_urb, GFP_KERNEL);
		if (result)
A
Alan Cox 已提交
146 147 148
			dev_err(&port->dev,
			    "%s - failed resubmitting read urb, error %d\n",
							__func__, result);
L
Linus Torvalds 已提交
149 150 151 152
	}

	return result;
}
153
EXPORT_SYMBOL_GPL(usb_serial_generic_open);
L
Linus Torvalds 已提交
154

A
Alan Cox 已提交
155
static void generic_cleanup(struct usb_serial_port *port)
L
Linus Torvalds 已提交
156 157 158
{
	struct usb_serial *serial = port->serial;

159
	dbg("%s - port %d", __func__, port->number);
L
Linus Torvalds 已提交
160 161

	if (serial->dev) {
162 163
		/* shutdown any bulk transfers that might be going on */
		if (port->bulk_out_size)
L
Linus Torvalds 已提交
164
			usb_kill_urb(port->write_urb);
165
		if (port->bulk_in_size)
L
Linus Torvalds 已提交
166 167 168 169
			usb_kill_urb(port->read_urb);
	}
}

170
void usb_serial_generic_close(struct usb_serial_port *port)
L
Linus Torvalds 已提交
171
{
172
	dbg("%s - port %d", __func__, port->number);
A
Alan Cox 已提交
173
	generic_cleanup(port);
L
Linus Torvalds 已提交
174 175
}

176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
static int usb_serial_multi_urb_write(struct tty_struct *tty,
	struct usb_serial_port *port, const unsigned char *buf, int count)
{
	unsigned long flags;
	struct urb *urb;
	unsigned char *buffer;
	int status;
	int towrite;
	int bwrite = 0;

	dbg("%s - port %d", __func__, port->number);

	if (count == 0)
		dbg("%s - write request of 0 bytes", __func__);

	while (count > 0) {
		towrite = (count > port->bulk_out_size) ?
			port->bulk_out_size : count;
		spin_lock_irqsave(&port->lock, flags);
		if (port->urbs_in_flight >
		    port->serial->type->max_in_flight_urbs) {
			spin_unlock_irqrestore(&port->lock, flags);
198
			dbg("%s - write limit hit", __func__);
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
			return bwrite;
		}
		port->tx_bytes_flight += towrite;
		port->urbs_in_flight++;
		spin_unlock_irqrestore(&port->lock, flags);

		buffer = kmalloc(towrite, GFP_ATOMIC);
		if (!buffer) {
			dev_err(&port->dev,
			"%s ran out of kernel memory for urb ...\n", __func__);
			goto error_no_buffer;
		}

		urb = usb_alloc_urb(0, GFP_ATOMIC);
		if (!urb) {
			dev_err(&port->dev, "%s - no more free urbs\n",
				__func__);
			goto error_no_urb;
		}

		/* Copy data */
		memcpy(buffer, buf + bwrite, towrite);
		usb_serial_debug_data(debug, &port->dev, __func__,
				      towrite, buffer);
		/* fill the buffer and send it */
		usb_fill_bulk_urb(urb, port->serial->dev,
			usb_sndbulkpipe(port->serial->dev,
					port->bulk_out_endpointAddress),
			buffer, towrite,
			usb_serial_generic_write_bulk_callback, port);

		status = usb_submit_urb(urb, GFP_ATOMIC);
		if (status) {
			dev_err(&port->dev,
				"%s - failed submitting write urb, error %d\n",
				__func__, status);
			goto error;
		}

		/* This urb is the responsibility of the host driver now */
		usb_free_urb(urb);
		dbg("%s write: %d", __func__, towrite);
		count -= towrite;
		bwrite += towrite;
	}
	return bwrite;

error:
	usb_free_urb(urb);
error_no_urb:
	kfree(buffer);
error_no_buffer:
	spin_lock_irqsave(&port->lock, flags);
	port->urbs_in_flight--;
	port->tx_bytes_flight -= towrite;
	spin_unlock_irqrestore(&port->lock, flags);
	return bwrite;
}

258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279
/**
 * usb_serial_generic_write_start - kick off an URB write
 * @port:	Pointer to the &struct usb_serial_port data
 *
 * Returns the number of bytes queued on success. This will be zero if there
 * was nothing to send. Otherwise, it returns a negative errno value
 */
static int usb_serial_generic_write_start(struct usb_serial_port *port)
{
	struct usb_serial *serial = port->serial;
	unsigned char *data;
	int result;
	int count;
	unsigned long flags;
	bool start_io;

	/* Atomically determine whether we can and need to start a USB
	 * operation. */
	spin_lock_irqsave(&port->lock, flags);
	if (port->write_urb_busy)
		start_io = false;
	else {
280
		start_io = (kfifo_len(&port->write_fifo) != 0);
281 282 283 284 285 286 287 288
		port->write_urb_busy = start_io;
	}
	spin_unlock_irqrestore(&port->lock, flags);

	if (!start_io)
		return 0;

	data = port->write_urb->transfer_buffer;
289
	count = kfifo_out_locked(&port->write_fifo, data, port->bulk_out_size, &port->lock);
290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327
	usb_serial_debug_data(debug, &port->dev, __func__, count, data);

	/* set up our urb */
	usb_fill_bulk_urb(port->write_urb, serial->dev,
			   usb_sndbulkpipe(serial->dev,
				port->bulk_out_endpointAddress),
			   port->write_urb->transfer_buffer, count,
			   ((serial->type->write_bulk_callback) ?
			     serial->type->write_bulk_callback :
			     usb_serial_generic_write_bulk_callback),
			   port);

	/* send the data out the bulk port */
	result = usb_submit_urb(port->write_urb, GFP_ATOMIC);
	if (result) {
		dev_err(&port->dev,
			"%s - failed submitting write urb, error %d\n",
						__func__, result);
		/* don't have to grab the lock here, as we will
		   retry if != 0 */
		port->write_urb_busy = 0;
	} else
		result = count;

	return result;
}

/**
 * usb_serial_generic_write - generic write function for serial USB devices
 * @tty:	Pointer to &struct tty_struct for the device
 * @port:	Pointer to the &usb_serial_port structure for the device
 * @buf:	Pointer to the data to write
 * @count:	Number of bytes to write
 *
 * Returns the number of characters actually written, which may be anything
 * from zero to @count. If an error occurs, it returns the negative errno
 * value.
 */
A
Alan Cox 已提交
328 329
int usb_serial_generic_write(struct tty_struct *tty,
	struct usb_serial_port *port, const unsigned char *buf, int count)
L
Linus Torvalds 已提交
330 331 332 333
{
	struct usb_serial *serial = port->serial;
	int result;

334
	dbg("%s - port %d", __func__, port->number);
L
Linus Torvalds 已提交
335

336 337 338 339
	/* only do something if we have a bulk out endpoint */
	if (!port->bulk_out_size)
		return -ENODEV;

L
Linus Torvalds 已提交
340
	if (count == 0) {
341
		dbg("%s - write request of 0 bytes", __func__);
A
Alan Cox 已提交
342
		return 0;
L
Linus Torvalds 已提交
343 344
	}

345 346 347
	if (serial->type->max_in_flight_urbs)
		return usb_serial_multi_urb_write(tty, port,
						  buf, count);
L
Linus Torvalds 已提交
348

349
	count = kfifo_in_locked(&port->write_fifo, buf, count, &port->lock);
350
	result = usb_serial_generic_write_start(port);
L
Linus Torvalds 已提交
351

352 353
	if (result >= 0)
		result = count;
L
Linus Torvalds 已提交
354

355
	return result;
L
Linus Torvalds 已提交
356
}
357
EXPORT_SYMBOL_GPL(usb_serial_generic_write);
L
Linus Torvalds 已提交
358

A
Alan Cox 已提交
359
int usb_serial_generic_write_room(struct tty_struct *tty)
L
Linus Torvalds 已提交
360
{
A
Alan Cox 已提交
361
	struct usb_serial_port *port = tty->driver_data;
L
Linus Torvalds 已提交
362
	struct usb_serial *serial = port->serial;
363
	unsigned long flags;
L
Linus Torvalds 已提交
364 365
	int room = 0;

366
	dbg("%s - port %d", __func__, port->number);
367 368 369 370

	if (!port->bulk_out_size)
		return 0;

371 372 373
	spin_lock_irqsave(&port->lock, flags);
	if (serial->type->max_in_flight_urbs) {
		if (port->urbs_in_flight < serial->type->max_in_flight_urbs)
374 375 376
			room = port->bulk_out_size *
				(serial->type->max_in_flight_urbs -
				 port->urbs_in_flight);
377
	} else {
378
		room = kfifo_avail(&port->write_fifo);
379
	}
380
	spin_unlock_irqrestore(&port->lock, flags);
L
Linus Torvalds 已提交
381

382
	dbg("%s - returns %d", __func__, room);
383
	return room;
L
Linus Torvalds 已提交
384 385
}

A
Alan Cox 已提交
386
int usb_serial_generic_chars_in_buffer(struct tty_struct *tty)
L
Linus Torvalds 已提交
387
{
A
Alan Cox 已提交
388
	struct usb_serial_port *port = tty->driver_data;
L
Linus Torvalds 已提交
389
	struct usb_serial *serial = port->serial;
390
	unsigned long flags;
391
	int chars;
L
Linus Torvalds 已提交
392

393
	dbg("%s - port %d", __func__, port->number);
L
Linus Torvalds 已提交
394

395 396 397
	if (!port->bulk_out_size)
		return 0;

398 399
	spin_lock_irqsave(&port->lock, flags);
	if (serial->type->max_in_flight_urbs)
400
		chars = port->tx_bytes_flight;
401
	else
402
		chars = kfifo_len(&port->write_fifo);
403
	spin_unlock_irqrestore(&port->lock, flags);
L
Linus Torvalds 已提交
404

405
	dbg("%s - returns %d", __func__, chars);
A
Alan Cox 已提交
406
	return chars;
L
Linus Torvalds 已提交
407 408
}

409

410 411
void usb_serial_generic_resubmit_read_urb(struct usb_serial_port *port,
			gfp_t mem_flags)
L
Linus Torvalds 已提交
412
{
413
	struct urb *urb = port->read_urb;
414
	struct usb_serial *serial = port->serial;
L
Linus Torvalds 已提交
415 416
	int result;

417
	/* Continue reading from device */
A
Alan Cox 已提交
418 419 420
	usb_fill_bulk_urb(urb, serial->dev,
			   usb_rcvbulkpipe(serial->dev,
					port->bulk_in_endpointAddress),
421 422
			   urb->transfer_buffer,
			   urb->transfer_buffer_length,
A
Alan Cox 已提交
423 424
			   ((serial->type->read_bulk_callback) ?
			     serial->type->read_bulk_callback :
L
Linus Torvalds 已提交
425
			     usb_serial_generic_read_bulk_callback), port);
426

427
	result = usb_submit_urb(urb, mem_flags);
428
	if (result && result != -EPERM) {
A
Alan Cox 已提交
429 430 431
		dev_err(&port->dev,
			"%s - failed resubmitting read urb, error %d\n",
							__func__, result);
432
	}
L
Linus Torvalds 已提交
433
}
434
EXPORT_SYMBOL_GPL(usb_serial_generic_resubmit_read_urb);
435

436
/* Push data to tty layer and resubmit the bulk read URB */
A
Alan Cox 已提交
437
static void flush_and_resubmit_read_urb(struct usb_serial_port *port)
438 439
{
	struct urb *urb = port->read_urb;
A
Alan Cox 已提交
440
	struct tty_struct *tty = tty_port_tty_get(&port->port);
441 442 443 444 445
	char *ch = (char *)urb->transfer_buffer;
	int i;

	if (!tty)
		goto done;
446

447 448 449 450 451 452 453 454
	/* The per character mucking around with sysrq path it too slow for
	   stuff like 3G modems, so shortcircuit it in the 99.9999999% of cases
	   where the USB serial is not a console anyway */
	if (!port->console || !port->sysrq)
		tty_insert_flip_string(tty, ch, urb->actual_length);
	else {
		/* Push data to tty */
		for (i = 0; i < urb->actual_length; i++, ch++) {
A
Alan Cox 已提交
455
			if (!usb_serial_handle_sysrq_char(tty, port, *ch))
456 457
				tty_insert_flip_char(tty, *ch, TTY_NORMAL);
		}
458
	}
459
	tty_flip_buffer_push(tty);
A
Alan Cox 已提交
460
	tty_kref_put(tty);
461 462
done:
	usb_serial_generic_resubmit_read_urb(port, GFP_ATOMIC);
463 464
}

A
Alan Cox 已提交
465
void usb_serial_generic_read_bulk_callback(struct urb *urb)
466
{
467
	struct usb_serial_port *port = urb->context;
468
	unsigned char *data = urb->transfer_buffer;
469
	int status = urb->status;
470
	unsigned long flags;
471

472
	dbg("%s - port %d", __func__, port->number);
473

474 475
	if (unlikely(status != 0)) {
		dbg("%s - nonzero read bulk status received: %d",
476
		    __func__, status);
477 478 479
		return;
	}

A
Alan Cox 已提交
480 481
	usb_serial_debug_data(debug, &port->dev, __func__,
						urb->actual_length, data);
482 483

	/* Throttle the device if requested by tty */
484
	spin_lock_irqsave(&port->lock, flags);
A
Alan Cox 已提交
485 486
	port->throttled = port->throttle_req;
	if (!port->throttled) {
487
		spin_unlock_irqrestore(&port->lock, flags);
488
		flush_and_resubmit_read_urb(port);
A
Alan Cox 已提交
489
	} else
490
		spin_unlock_irqrestore(&port->lock, flags);
491
}
492
EXPORT_SYMBOL_GPL(usb_serial_generic_read_bulk_callback);
L
Linus Torvalds 已提交
493

A
Alan Cox 已提交
494
void usb_serial_generic_write_bulk_callback(struct urb *urb)
L
Linus Torvalds 已提交
495
{
496
	unsigned long flags;
497
	struct usb_serial_port *port = urb->context;
498
	int status = urb->status;
L
Linus Torvalds 已提交
499

500
	dbg("%s - port %d", __func__, port->number);
L
Linus Torvalds 已提交
501

502
	if (port->serial->type->max_in_flight_urbs) {
503 504
		kfree(urb->transfer_buffer);

505 506 507 508 509 510 511 512 513
		spin_lock_irqsave(&port->lock, flags);
		--port->urbs_in_flight;
		port->tx_bytes_flight -= urb->transfer_buffer_length;
		if (port->urbs_in_flight < 0)
			port->urbs_in_flight = 0;
		spin_unlock_irqrestore(&port->lock, flags);
	} else {
		port->write_urb_busy = 0;

514
		if (status)
515
			kfifo_reset_out(&port->write_fifo);
516
		else
517
			usb_serial_generic_write_start(port);
L
Linus Torvalds 已提交
518
	}
519

520 521 522
	if (status)
		dbg("%s - non-zero urb status: %d", __func__, status);

523
	usb_serial_port_softint(port);
L
Linus Torvalds 已提交
524
}
525
EXPORT_SYMBOL_GPL(usb_serial_generic_write_bulk_callback);
L
Linus Torvalds 已提交
526

A
Alan Cox 已提交
527
void usb_serial_generic_throttle(struct tty_struct *tty)
528
{
A
Alan Cox 已提交
529
	struct usb_serial_port *port = tty->driver_data;
530 531
	unsigned long flags;

532
	dbg("%s - port %d", __func__, port->number);
533 534 535 536 537 538 539 540

	/* Set the throttle request flag. It will be picked up
	 * by usb_serial_generic_read_bulk_callback(). */
	spin_lock_irqsave(&port->lock, flags);
	port->throttle_req = 1;
	spin_unlock_irqrestore(&port->lock, flags);
}

A
Alan Cox 已提交
541
void usb_serial_generic_unthrottle(struct tty_struct *tty)
542
{
A
Alan Cox 已提交
543
	struct usb_serial_port *port = tty->driver_data;
544 545 546
	int was_throttled;
	unsigned long flags;

547
	dbg("%s - port %d", __func__, port->number);
548 549 550 551 552 553 554 555

	/* Clear the throttle flags */
	spin_lock_irqsave(&port->lock, flags);
	was_throttled = port->throttled;
	port->throttled = port->throttle_req = 0;
	spin_unlock_irqrestore(&port->lock, flags);

	if (was_throttled) {
556
		/* Resume reading from device */
557
		flush_and_resubmit_read_urb(port);
558 559 560
	}
}

A
Alan Cox 已提交
561 562
int usb_serial_handle_sysrq_char(struct tty_struct *tty,
			struct usb_serial_port *port, unsigned int ch)
563
{
564
	if (port->sysrq && port->console) {
565
		if (ch && time_before(jiffies, port->sysrq)) {
A
Alan Cox 已提交
566
			handle_sysrq(ch, tty);
567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586
			port->sysrq = 0;
			return 1;
		}
		port->sysrq = 0;
	}
	return 0;
}
EXPORT_SYMBOL_GPL(usb_serial_handle_sysrq_char);

int usb_serial_handle_break(struct usb_serial_port *port)
{
	if (!port->sysrq) {
		port->sysrq = jiffies + HZ*5;
		return 1;
	}
	port->sysrq = 0;
	return 0;
}
EXPORT_SYMBOL_GPL(usb_serial_handle_break);

587 588 589 590 591 592 593
int usb_serial_generic_resume(struct usb_serial *serial)
{
	struct usb_serial_port *port;
	int i, c = 0, r;

	for (i = 0; i < serial->num_ports; i++) {
		port = serial->port[i];
594
		if (!test_bit(ASYNCB_INITIALIZED, &port->port.flags))
595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613
			continue;

		if (port->read_urb) {
			r = usb_submit_urb(port->read_urb, GFP_NOIO);
			if (r < 0)
				c++;
		}

		if (port->write_urb) {
			r = usb_serial_generic_write_start(port);
			if (r < 0)
				c++;
		}
	}

	return c ? -EIO : 0;
}
EXPORT_SYMBOL_GPL(usb_serial_generic_resume);

614
void usb_serial_generic_disconnect(struct usb_serial *serial)
L
Linus Torvalds 已提交
615 616 617
{
	int i;

618
	dbg("%s", __func__);
L
Linus Torvalds 已提交
619 620

	/* stop reads and writes on all ports */
A
Alan Cox 已提交
621
	for (i = 0; i < serial->num_ports; ++i)
L
Linus Torvalds 已提交
622 623 624
		generic_cleanup(serial->port[i]);
}

625 626 627 628
void usb_serial_generic_release(struct usb_serial *serial)
{
	dbg("%s", __func__);
}