⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 usb-serial.c

📁 优龙2410linux2.6.8内核源代码
💻 C
📖 第 1 页 / 共 3 页
字号:
			dbg ("sub driver rejected device");			kfree (serial);			return retval;		}	}	/* descriptor matches, let's find the endpoints needed */	/* check out the endpoints */	iface_desc = interface->cur_altsetting;	for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {		endpoint = &iface_desc->endpoint[i].desc;				if ((endpoint->bEndpointAddress & 0x80) &&		    ((endpoint->bmAttributes & 3) == 0x02)) {			/* we found a bulk in endpoint */			dbg("found bulk in");			bulk_in_endpoint[num_bulk_in] = endpoint;			++num_bulk_in;		}		if (((endpoint->bEndpointAddress & 0x80) == 0x00) &&		    ((endpoint->bmAttributes & 3) == 0x02)) {			/* we found a bulk out endpoint */			dbg("found bulk out");			bulk_out_endpoint[num_bulk_out] = endpoint;			++num_bulk_out;		}				if ((endpoint->bEndpointAddress & 0x80) &&		    ((endpoint->bmAttributes & 3) == 0x03)) {			/* we found a interrupt in endpoint */			dbg("found interrupt in");			interrupt_in_endpoint[num_interrupt_in] = endpoint;			++num_interrupt_in;		}	}#if defined(CONFIG_USB_SERIAL_PL2303) || defined(CONFIG_USB_SERIAL_PL2303_MODULE)	/* BEGIN HORRIBLE HACK FOR PL2303 */ 	/* this is needed due to the looney way its endpoints are set up */	if (((dev->descriptor.idVendor == PL2303_VENDOR_ID) &&	     (dev->descriptor.idProduct == PL2303_PRODUCT_ID)) ||	    ((dev->descriptor.idVendor == ATEN_VENDOR_ID) &&	     (dev->descriptor.idProduct == ATEN_PRODUCT_ID))) {		if (interface != dev->actconfig->interface[0]) {			/* check out the endpoints of the other interface*/			iface_desc = dev->actconfig->interface[0]->cur_altsetting;			for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {				endpoint = &iface_desc->endpoint[i].desc;				if ((endpoint->bEndpointAddress & 0x80) &&				    ((endpoint->bmAttributes & 3) == 0x03)) {					/* we found a interrupt in endpoint */					dbg("found interrupt in for Prolific device on separate interface");					interrupt_in_endpoint[num_interrupt_in] = endpoint;					++num_interrupt_in;				}			}		}		/* Now make sure the PL-2303 is configured correctly.		 * If not, give up now and hope this hack will work		 * properly during a later invocation of usb_serial_probe		 */		if (num_bulk_in == 0 || num_bulk_out == 0) {			dev_info(&interface->dev, "PL-2303 hack: descriptors matched but endpoints did not\n");			kfree (serial);			return -ENODEV;		}	}	/* END HORRIBLE HACK FOR PL2303 */#endif	/* found all that we need */	dev_info(&interface->dev, "%s converter detected\n", type->name);#ifdef CONFIG_USB_SERIAL_GENERIC	if (type == &usb_serial_generic_device) {		num_ports = num_bulk_out;		if (num_ports == 0) {			dev_err(&interface->dev, "Generic device with no bulk out, not allowed.\n");			kfree (serial);			return -EIO;		}	}#endif	if (!num_ports) {		/* if this device type has a calc_num_ports function, call it */		if (type->calc_num_ports) {			if (!try_module_get(type->owner)) {				dev_err(&interface->dev, "module get failed, exiting\n");				kfree (serial);				return -EIO;			}			num_ports = type->calc_num_ports (serial);			module_put(type->owner);		}		if (!num_ports)			num_ports = type->num_ports;	}	if (get_free_serial (serial, num_ports, &minor) == NULL) {		dev_err(&interface->dev, "No more free serial devices\n");		kfree (serial);		return -ENOMEM;	}	serial->minor = minor;	serial->num_ports = num_ports;	serial->num_bulk_in = num_bulk_in;	serial->num_bulk_out = num_bulk_out;	serial->num_interrupt_in = num_interrupt_in;	/* create our ports, we need as many as the max endpoints */	/* we don't use num_ports here cauz some devices have more endpoint pairs than ports */	max_endpoints = max(num_bulk_in, num_bulk_out);	max_endpoints = max(max_endpoints, num_interrupt_in);	max_endpoints = max(max_endpoints, (int)serial->num_ports);	serial->num_port_pointers = max_endpoints;	dbg("%s - setting up %d port structures for this device", __FUNCTION__, max_endpoints);	for (i = 0; i < max_endpoints; ++i) {		port = kmalloc(sizeof(struct usb_serial_port), GFP_KERNEL);		if (!port)			goto probe_error;		memset(port, 0x00, sizeof(struct usb_serial_port));		port->number = i + serial->minor;		port->serial = serial;		INIT_WORK(&port->work, usb_serial_port_softint, port);		serial->port[i] = port;	}	/* set up the endpoint information */	for (i = 0; i < num_bulk_in; ++i) {		endpoint = bulk_in_endpoint[i];		port = serial->port[i];		port->read_urb = usb_alloc_urb (0, GFP_KERNEL);		if (!port->read_urb) {			dev_err(&interface->dev, "No free urbs available\n");			goto probe_error;		}		buffer_size = endpoint->wMaxPacketSize;		port->bulk_in_endpointAddress = endpoint->bEndpointAddress;		port->bulk_in_buffer = kmalloc (buffer_size, GFP_KERNEL);		if (!port->bulk_in_buffer) {			dev_err(&interface->dev, "Couldn't allocate bulk_in_buffer\n");			goto probe_error;		}		usb_fill_bulk_urb (port->read_urb, dev,				   usb_rcvbulkpipe (dev,					   	    endpoint->bEndpointAddress),				   port->bulk_in_buffer, buffer_size,				   serial->type->read_bulk_callback,				   port);	}	for (i = 0; i < num_bulk_out; ++i) {		endpoint = bulk_out_endpoint[i];		port = serial->port[i];		port->write_urb = usb_alloc_urb(0, GFP_KERNEL);		if (!port->write_urb) {			dev_err(&interface->dev, "No free urbs available\n");			goto probe_error;		}		buffer_size = endpoint->wMaxPacketSize;		port->bulk_out_size = buffer_size;		port->bulk_out_endpointAddress = endpoint->bEndpointAddress;		port->bulk_out_buffer = kmalloc (buffer_size, GFP_KERNEL);		if (!port->bulk_out_buffer) {			dev_err(&interface->dev, "Couldn't allocate bulk_out_buffer\n");			goto probe_error;		}		usb_fill_bulk_urb (port->write_urb, dev,				   usb_sndbulkpipe (dev,						    endpoint->bEndpointAddress),				   port->bulk_out_buffer, buffer_size, 				   serial->type->write_bulk_callback,				   port);	}	for (i = 0; i < num_interrupt_in; ++i) {		endpoint = interrupt_in_endpoint[i];		port = serial->port[i];		port->interrupt_in_urb = usb_alloc_urb(0, GFP_KERNEL);		if (!port->interrupt_in_urb) {			dev_err(&interface->dev, "No free urbs available\n");			goto probe_error;		}		buffer_size = endpoint->wMaxPacketSize;		port->interrupt_in_endpointAddress = endpoint->bEndpointAddress;		port->interrupt_in_buffer = kmalloc (buffer_size, GFP_KERNEL);		if (!port->interrupt_in_buffer) {			dev_err(&interface->dev, "Couldn't allocate interrupt_in_buffer\n");			goto probe_error;		}		usb_fill_int_urb (port->interrupt_in_urb, dev, 				  usb_rcvintpipe (dev,						  endpoint->bEndpointAddress),				  port->interrupt_in_buffer, buffer_size, 				  serial->type->read_int_callback, port, 				  endpoint->bInterval);	}	/* if this device type has an attach function, call it */	if (type->attach) {		if (!try_module_get(type->owner)) {			dev_err(&interface->dev, "module get failed, exiting\n");			goto probe_error;		}		retval = type->attach (serial);		module_put(type->owner);		if (retval < 0)			goto probe_error;		if (retval > 0) {			/* quietly accept this device, but don't bind to a serial port			 * as it's about to disappear */			goto exit;		}	}	/* register all of the individual ports with the driver core */	for (i = 0; i < num_ports; ++i) {		port = serial->port[i];		port->dev.parent = &interface->dev;		port->dev.driver = NULL;		port->dev.bus = &usb_serial_bus_type;		port->dev.release = &port_release;		snprintf (&port->dev.bus_id[0], sizeof(port->dev.bus_id), "ttyUSB%d", port->number);		dbg ("%s - registering %s", __FUNCTION__, port->dev.bus_id);		device_register (&port->dev);	}	usb_serial_console_init (debug, minor);exit:	/* success */	usb_set_intfdata (interface, serial);	return 0;probe_error:	for (i = 0; i < num_bulk_in; ++i) {		port = serial->port[i];		if (!port)			continue;		if (port->read_urb)			usb_free_urb (port->read_urb);		kfree(port->bulk_in_buffer);	}	for (i = 0; i < num_bulk_out; ++i) {		port = serial->port[i];		if (!port)			continue;		if (port->write_urb)			usb_free_urb (port->write_urb);		kfree(port->bulk_out_buffer);	}	for (i = 0; i < num_interrupt_in; ++i) {		port = serial->port[i];		if (!port)			continue;		if (port->interrupt_in_urb)			usb_free_urb (port->interrupt_in_urb);		kfree(port->interrupt_in_buffer);	}	/* return the minor range that this device had */	return_serial (serial);	/* free up any memory that we allocated */	for (i = 0; i < serial->num_port_pointers; ++i)		kfree(serial->port[i]);	kfree (serial);	return -EIO;}void usb_serial_disconnect(struct usb_interface *interface){	struct usb_serial *serial = usb_get_intfdata (interface);	struct device *dev = &interface->dev;	dbg ("%s", __FUNCTION__);	usb_set_intfdata (interface, NULL);	if (serial) {		/* let the last holder of this object 		 * cause it to be cleaned up */		kref_put(&serial->kref);	}	dev_info(dev, "device disconnected\n");}static struct tty_operations serial_ops = {	.open =			serial_open,	.close =		serial_close,	.write =		serial_write,	.write_room =		serial_write_room,	.ioctl =		serial_ioctl,	.set_termios =		serial_set_termios,	.throttle =		serial_throttle,	.unthrottle =		serial_unthrottle,	.break_ctl =		serial_break,	.chars_in_buffer =	serial_chars_in_buffer,	.read_proc =		serial_read_proc,	.tiocmget =		serial_tiocmget,	.tiocmset =		serial_tiocmset,};struct tty_driver *usb_serial_tty_driver;static int __init usb_serial_init(void){	int i;	int result = 0;	usb_serial_tty_driver = alloc_tty_driver(SERIAL_TTY_MINORS);	if (!usb_serial_tty_driver)		return -ENOMEM;	/* Initialize our global data */	for (i = 0; i < SERIAL_TTY_MINORS; ++i) {		serial_table[i] = NULL;	}	bus_register(&usb_serial_bus_type);	/* register the generic driver, if we should */	result = usb_serial_generic_register(debug);	if (result < 0) {		err("%s - registering generic driver failed", __FUNCTION__);		goto exit;	}	usb_serial_tty_driver->owner = THIS_MODULE;	usb_serial_tty_driver->driver_name = "usbserial";	usb_serial_tty_driver->devfs_name = "usb/tts/";	usb_serial_tty_driver->name = 	"ttyUSB";	usb_serial_tty_driver->major = SERIAL_TTY_MAJOR;	usb_serial_tty_driver->minor_start = 0;	usb_serial_tty_driver->type = TTY_DRIVER_TYPE_SERIAL;	usb_serial_tty_driver->subtype = SERIAL_TYPE_NORMAL;	usb_serial_tty_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_NO_DEVFS;	usb_serial_tty_driver->init_termios = tty_std_termios;	usb_serial_tty_driver->init_termios.c_cflag = B9600 | CS8 | CREAD | HUPCL | CLOCAL;	tty_set_operations(usb_serial_tty_driver, &serial_ops);	result = tty_register_driver(usb_serial_tty_driver);	if (result) {		err("%s - tty_register_driver failed", __FUNCTION__);		goto exit_generic;	}	/* register the USB driver */	result = usb_register(&usb_serial_driver);	if (result < 0) {		err("%s - usb_register failed", __FUNCTION__);		goto exit_tty;	}	info(DRIVER_DESC " " DRIVER_VERSION);	return result;exit_tty:	tty_unregister_driver(usb_serial_tty_driver);exit_generic:	usb_serial_generic_deregister();exit:	err ("%s - returning with error %d", __FUNCTION__, result);	put_tty_driver(usb_serial_tty_driver);	return result;}static void __exit usb_serial_exit(void){	usb_serial_console_exit();	usb_serial_generic_deregister();	usb_deregister(&usb_serial_driver);	tty_unregister_driver(usb_serial_tty_driver);	put_tty_driver(usb_serial_tty_driver);	bus_unregister(&usb_serial_bus_type);}module_init(usb_serial_init);module_exit(usb_serial_exit);#define set_to_generic_if_null(type, function)				\	do {								\		if (!type->function) {					\			type->function = usb_serial_generic_##function;	\			dbg("Had to override the " #function		\				 " usb serial operation with the generic one.");\			}						\	} while (0)static void fixup_generic(struct usb_serial_device_type *device){	set_to_generic_if_null(device, open);	set_to_generic_if_null(device, write);	set_to_generic_if_null(device, close);	set_to_generic_if_null(device, write_room);	set_to_generic_if_null(device, chars_in_buffer);	set_to_generic_if_null(device, read_bulk_callback);	set_to_generic_if_null(device, write_bulk_callback);	set_to_generic_if_null(device, shutdown);}int usb_serial_register(struct usb_serial_device_type *new_device){	int retval;	fixup_generic(new_device);	/* Add this device to our list of devices */	list_add(&new_device->driver_list, &usb_serial_driver_list);	retval =  usb_serial_bus_register (new_device);	if (retval)		goto error;	info("USB Serial support registered for %s", new_device->name);	return retval;error:	err("problem %d when registering driver %s", retval, new_device->name);	list_del(&new_device->driver_list);	return retval;}void usb_serial_deregister(struct usb_serial_device_type *device){	info("USB Serial deregistering driver %s", device->name);	list_del(&device->driver_list);	usb_serial_bus_deregister(device);}/* If the usb-serial core is built into the core, the usb-serial drivers   need these symbols to load properly as modules. */EXPORT_SYMBOL(usb_serial_register);EXPORT_SYMBOL(usb_serial_deregister);EXPORT_SYMBOL(usb_serial_probe);EXPORT_SYMBOL(usb_serial_disconnect);EXPORT_SYMBOL(usb_serial_port_softint);/* Module information */MODULE_AUTHOR( DRIVER_AUTHOR );MODULE_DESCRIPTION( DRIVER_DESC );MODULE_LICENSE("GPL");module_param(debug, bool, S_IRUGO | S_IWUSR);MODULE_PARM_DESC(debug, "Debug enabled or not");

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -