generic.c 16.1 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 */
L
Linus Torvalds 已提交
133 134
	if (serial->num_bulk_in) {
		/* 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 162 163 164 165 166 167 168 169

	if (serial->dev) {
		/* shutdown any bulk reads that might be going on */
		if (serial->num_bulk_out)
			usb_kill_urb(port->write_urb);
		if (serial->num_bulk_in)
			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

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

	/* only do something if we have a bulk out endpoint */
342 343
	if (!serial->num_bulk_out)
		return 0;
L
Linus Torvalds 已提交
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
	spin_lock_irqsave(&port->lock, flags);
	if (serial->type->max_in_flight_urbs) {
		if (port->urbs_in_flight < serial->type->max_in_flight_urbs)
370 371 372
			room = port->bulk_out_size *
				(serial->type->max_in_flight_urbs -
				 port->urbs_in_flight);
373
	} else if (serial->num_bulk_out)
374
		room = kfifo_avail(&port->write_fifo);
375
	spin_unlock_irqrestore(&port->lock, flags);
L
Linus Torvalds 已提交
376

377
	dbg("%s - returns %d", __func__, room);
378
	return room;
L
Linus Torvalds 已提交
379 380
}

A
Alan Cox 已提交
381
int usb_serial_generic_chars_in_buffer(struct tty_struct *tty)
L
Linus Torvalds 已提交
382
{
A
Alan Cox 已提交
383
	struct usb_serial_port *port = tty->driver_data;
L
Linus Torvalds 已提交
384 385
	struct usb_serial *serial = port->serial;
	int chars = 0;
386
	unsigned long flags;
L
Linus Torvalds 已提交
387

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

390 391
	spin_lock_irqsave(&port->lock, flags);
	if (serial->type->max_in_flight_urbs)
392
		chars = port->tx_bytes_flight;
393
	else if (serial->num_bulk_out)
394
		chars = kfifo_len(&port->write_fifo);
395
	spin_unlock_irqrestore(&port->lock, flags);
L
Linus Torvalds 已提交
396

397
	dbg("%s - returns %d", __func__, chars);
A
Alan Cox 已提交
398
	return chars;
L
Linus Torvalds 已提交
399 400
}

401

402 403
void usb_serial_generic_resubmit_read_urb(struct usb_serial_port *port,
			gfp_t mem_flags)
L
Linus Torvalds 已提交
404
{
405
	struct urb *urb = port->read_urb;
406
	struct usb_serial *serial = port->serial;
L
Linus Torvalds 已提交
407 408
	int result;

409
	/* Continue reading from device */
A
Alan Cox 已提交
410 411 412
	usb_fill_bulk_urb(urb, serial->dev,
			   usb_rcvbulkpipe(serial->dev,
					port->bulk_in_endpointAddress),
413 414
			   urb->transfer_buffer,
			   urb->transfer_buffer_length,
A
Alan Cox 已提交
415 416
			   ((serial->type->read_bulk_callback) ?
			     serial->type->read_bulk_callback :
L
Linus Torvalds 已提交
417
			     usb_serial_generic_read_bulk_callback), port);
418
	result = usb_submit_urb(urb, mem_flags);
L
Linus Torvalds 已提交
419
	if (result)
A
Alan Cox 已提交
420 421 422
		dev_err(&port->dev,
			"%s - failed resubmitting read urb, error %d\n",
							__func__, result);
L
Linus Torvalds 已提交
423
}
424
EXPORT_SYMBOL_GPL(usb_serial_generic_resubmit_read_urb);
425

426
/* Push data to tty layer and resubmit the bulk read URB */
A
Alan Cox 已提交
427
static void flush_and_resubmit_read_urb(struct usb_serial_port *port)
428 429
{
	struct urb *urb = port->read_urb;
A
Alan Cox 已提交
430
	struct tty_struct *tty = tty_port_tty_get(&port->port);
431 432 433 434 435
	char *ch = (char *)urb->transfer_buffer;
	int i;

	if (!tty)
		goto done;
436

437 438 439 440 441 442 443 444
	/* 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 已提交
445
			if (!usb_serial_handle_sysrq_char(tty, port, *ch))
446 447
				tty_insert_flip_char(tty, *ch, TTY_NORMAL);
		}
448
	}
449
	tty_flip_buffer_push(tty);
A
Alan Cox 已提交
450
	tty_kref_put(tty);
451 452
done:
	usb_serial_generic_resubmit_read_urb(port, GFP_ATOMIC);
453 454
}

A
Alan Cox 已提交
455
void usb_serial_generic_read_bulk_callback(struct urb *urb)
456
{
457
	struct usb_serial_port *port = urb->context;
458
	unsigned char *data = urb->transfer_buffer;
459
	int status = urb->status;
460
	unsigned long flags;
461

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

464 465
	if (unlikely(status != 0)) {
		dbg("%s - nonzero read bulk status received: %d",
466
		    __func__, status);
467 468 469
		return;
	}

A
Alan Cox 已提交
470 471
	usb_serial_debug_data(debug, &port->dev, __func__,
						urb->actual_length, data);
472 473

	/* Throttle the device if requested by tty */
474
	spin_lock_irqsave(&port->lock, flags);
A
Alan Cox 已提交
475 476
	port->throttled = port->throttle_req;
	if (!port->throttled) {
477
		spin_unlock_irqrestore(&port->lock, flags);
478
		flush_and_resubmit_read_urb(port);
A
Alan Cox 已提交
479
	} else
480
		spin_unlock_irqrestore(&port->lock, flags);
481
}
482
EXPORT_SYMBOL_GPL(usb_serial_generic_read_bulk_callback);
L
Linus Torvalds 已提交
483

A
Alan Cox 已提交
484
void usb_serial_generic_write_bulk_callback(struct urb *urb)
L
Linus Torvalds 已提交
485
{
486
	unsigned long flags;
487
	struct usb_serial_port *port = urb->context;
488
	int status = urb->status;
L
Linus Torvalds 已提交
489

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

492
	if (port->serial->type->max_in_flight_urbs) {
493 494
		kfree(urb->transfer_buffer);

495 496 497 498 499 500
		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);
501 502 503 504 505 506

		if (status) {
			dbg("%s - nonzero multi-urb write bulk status "
				"received: %d", __func__, status);
			return;
		}
507 508 509
	} else {
		port->write_urb_busy = 0;

510 511 512
		if (status) {
			dbg("%s - nonzero multi-urb write bulk status "
				"received: %d", __func__, status);
513
			kfifo_reset_out(&port->write_fifo);
514 515
		} else
			usb_serial_generic_write_start(port);
L
Linus Torvalds 已提交
516
	}
517

518
	usb_serial_port_softint(port);
L
Linus Torvalds 已提交
519
}
520
EXPORT_SYMBOL_GPL(usb_serial_generic_write_bulk_callback);
L
Linus Torvalds 已提交
521

A
Alan Cox 已提交
522
void usb_serial_generic_throttle(struct tty_struct *tty)
523
{
A
Alan Cox 已提交
524
	struct usb_serial_port *port = tty->driver_data;
525 526
	unsigned long flags;

527
	dbg("%s - port %d", __func__, port->number);
528 529 530 531 532 533 534 535

	/* 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 已提交
536
void usb_serial_generic_unthrottle(struct tty_struct *tty)
537
{
A
Alan Cox 已提交
538
	struct usb_serial_port *port = tty->driver_data;
539 540 541
	int was_throttled;
	unsigned long flags;

542
	dbg("%s - port %d", __func__, port->number);
543 544 545 546 547 548 549 550

	/* 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) {
551
		/* Resume reading from device */
552
		flush_and_resubmit_read_urb(port);
553 554 555
	}
}

A
Alan Cox 已提交
556 557
int usb_serial_handle_sysrq_char(struct tty_struct *tty,
			struct usb_serial_port *port, unsigned int ch)
558
{
559
	if (port->sysrq && port->console) {
560
		if (ch && time_before(jiffies, port->sysrq)) {
A
Alan Cox 已提交
561
			handle_sysrq(ch, tty);
562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581
			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);

582 583 584 585 586 587 588
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];
589
		if (!test_bit(ASYNCB_INITIALIZED, &port->port.flags))
590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608
			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);

609
void usb_serial_generic_disconnect(struct usb_serial *serial)
L
Linus Torvalds 已提交
610 611 612
{
	int i;

613
	dbg("%s", __func__);
L
Linus Torvalds 已提交
614 615

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

620 621 622 623
void usb_serial_generic_release(struct usb_serial *serial)
{
	dbg("%s", __func__);
}