visor.c

来自「linux 内核源代码」· C语言 代码 · 共 1,016 行 · 第 1/2 页

C
1,016
字号
	struct visor_private *priv = usb_get_serial_port_data(port);	unsigned char *data = urb->transfer_buffer;	int status = urb->status;	struct tty_struct *tty;	int result;	int available_room;	dbg("%s - port %d", __FUNCTION__, port->number);	if (status) {		dbg("%s - nonzero read bulk status received: %d",		    __FUNCTION__, status);		return;	}	usb_serial_debug_data(debug, &port->dev, __FUNCTION__, urb->actual_length, data);	tty = port->tty;	if (tty && urb->actual_length) {		available_room = tty_buffer_request_room(tty, urb->actual_length);		if (available_room) {			tty_insert_flip_string(tty, data, available_room);			tty_flip_buffer_push(tty);		}		spin_lock(&priv->lock);		priv->bytes_in += available_room;	} else {		spin_lock(&priv->lock);	}	/* Continue trying to always read if we should */	if (!priv->throttled) {		usb_fill_bulk_urb (port->read_urb, port->serial->dev,				   usb_rcvbulkpipe(port->serial->dev,						   port->bulk_in_endpointAddress),				   port->read_urb->transfer_buffer,				   port->read_urb->transfer_buffer_length,				   visor_read_bulk_callback, port);		result = usb_submit_urb(port->read_urb, GFP_ATOMIC);		if (result)			dev_err(&port->dev, "%s - failed resubmitting read urb, error %d\n", __FUNCTION__, result);	} else {		priv->actually_throttled = 1;	}	spin_unlock(&priv->lock);}static void visor_read_int_callback (struct urb *urb){	struct usb_serial_port *port = (struct usb_serial_port *)urb->context;	int status = urb->status;	int result;	switch (status) {	case 0:		/* success */		break;	case -ECONNRESET:	case -ENOENT:	case -ESHUTDOWN:		/* this urb is terminated, clean up */		dbg("%s - urb shutting down with status: %d",		    __FUNCTION__, status);		return;	default:		dbg("%s - nonzero urb status received: %d",		    __FUNCTION__, status);		goto exit;	}	/*	 * This information is still unknown what it can be used for.	 * If anyone has an idea, please let the author know...	 *	 * Rumor has it this endpoint is used to notify when data	 * is ready to be read from the bulk ones.	 */	usb_serial_debug_data(debug, &port->dev, __FUNCTION__,			      urb->actual_length, urb->transfer_buffer);exit:	result = usb_submit_urb (urb, GFP_ATOMIC);	if (result)		dev_err(&urb->dev->dev, "%s - Error %d submitting interrupt urb\n",			__FUNCTION__, result);}static void visor_throttle (struct usb_serial_port *port){	struct visor_private *priv = usb_get_serial_port_data(port);	unsigned long flags;	dbg("%s - port %d", __FUNCTION__, port->number);	spin_lock_irqsave(&priv->lock, flags);	priv->throttled = 1;	spin_unlock_irqrestore(&priv->lock, flags);}static void visor_unthrottle (struct usb_serial_port *port){	struct visor_private *priv = usb_get_serial_port_data(port);	unsigned long flags;	int result;	dbg("%s - port %d", __FUNCTION__, port->number);	spin_lock_irqsave(&priv->lock, flags);	priv->throttled = 0;	priv->actually_throttled = 0;	spin_unlock_irqrestore(&priv->lock, flags);	port->read_urb->dev = port->serial->dev;	result = usb_submit_urb(port->read_urb, GFP_ATOMIC);	if (result)		dev_err(&port->dev, "%s - failed submitting read urb, error %d\n", __FUNCTION__, result);}static int palm_os_3_probe (struct usb_serial *serial, const struct usb_device_id *id){	struct device *dev = &serial->dev->dev;	struct visor_connection_info *connection_info;	unsigned char *transfer_buffer;	char *string;	int retval = 0;	int i;	int num_ports = 0;	dbg("%s", __FUNCTION__);	transfer_buffer = kmalloc (sizeof (*connection_info), GFP_KERNEL);	if (!transfer_buffer) {		dev_err(dev, "%s - kmalloc(%Zd) failed.\n", __FUNCTION__,			sizeof(*connection_info));		return -ENOMEM;	}	/* send a get connection info request */	retval = usb_control_msg (serial->dev,				  usb_rcvctrlpipe(serial->dev, 0),				  VISOR_GET_CONNECTION_INFORMATION,				  0xc2, 0x0000, 0x0000, transfer_buffer,				  sizeof(*connection_info), 300);	if (retval < 0) {		dev_err(dev, "%s - error %d getting connection information\n",			__FUNCTION__, retval);		goto exit;	}	if (retval == sizeof(*connection_info)) {	        connection_info = (struct visor_connection_info *)transfer_buffer;		num_ports = le16_to_cpu(connection_info->num_ports);		for (i = 0; i < num_ports; ++i) {			switch (connection_info->connections[i].port_function_id) {				case VISOR_FUNCTION_GENERIC:					string = "Generic";					break;				case VISOR_FUNCTION_DEBUGGER:					string = "Debugger";					break;				case VISOR_FUNCTION_HOTSYNC:					string = "HotSync";					break;				case VISOR_FUNCTION_CONSOLE:					string = "Console";					break;				case VISOR_FUNCTION_REMOTE_FILE_SYS:					string = "Remote File System";					break;				default:					string = "unknown";					break;			}			dev_info(dev, "%s: port %d, is for %s use\n",				serial->type->description,				connection_info->connections[i].port, string);		}	}	/*	* Handle devices that report invalid stuff here.	*/	if (num_ports == 0 || num_ports > 2) {		dev_warn (dev, "%s: No valid connect info available\n",			serial->type->description);		num_ports = 2;	}  	dev_info(dev, "%s: Number of ports: %d\n", serial->type->description,		num_ports);	/*	 * save off our num_ports info so that we can use it in the	 * calc_num_ports callback	 */	usb_set_serial_data(serial, (void *)(long)num_ports);	/* ask for the number of bytes available, but ignore the response as it is broken */	retval = usb_control_msg (serial->dev,				  usb_rcvctrlpipe(serial->dev, 0),				  VISOR_REQUEST_BYTES_AVAILABLE,				  0xc2, 0x0000, 0x0005, transfer_buffer,				  0x02, 300);	if (retval < 0)		dev_err(dev, "%s - error %d getting bytes available request\n",			__FUNCTION__, retval);	retval = 0;exit:	kfree (transfer_buffer);	return retval;}static int palm_os_4_probe (struct usb_serial *serial, const struct usb_device_id *id){	struct device *dev = &serial->dev->dev;	struct palm_ext_connection_info *connection_info;	unsigned char *transfer_buffer;	int retval;	dbg("%s", __FUNCTION__);	transfer_buffer =  kmalloc (sizeof (*connection_info), GFP_KERNEL);	if (!transfer_buffer) {		dev_err(dev, "%s - kmalloc(%Zd) failed.\n", __FUNCTION__,			sizeof(*connection_info));		return -ENOMEM;	}	retval = usb_control_msg (serial->dev,				  usb_rcvctrlpipe(serial->dev, 0), 				  PALM_GET_EXT_CONNECTION_INFORMATION,				  0xc2, 0x0000, 0x0000, transfer_buffer,				  sizeof (*connection_info), 300);	if (retval < 0)		dev_err(dev, "%s - error %d getting connection info\n",			__FUNCTION__, retval);	else		usb_serial_debug_data(debug, &serial->dev->dev, __FUNCTION__,				      retval, transfer_buffer);	kfree (transfer_buffer);	return 0;}static int visor_probe (struct usb_serial *serial, const struct usb_device_id *id){	int retval = 0;	int (*startup) (struct usb_serial *serial, const struct usb_device_id *id);	dbg("%s", __FUNCTION__);	if (serial->dev->actconfig->desc.bConfigurationValue != 1) {		err("active config #%d != 1 ??",			serial->dev->actconfig->desc.bConfigurationValue);		return -ENODEV;	}	if (id->driver_info) {		startup = (void *)id->driver_info;		retval = startup(serial, id);	}	return retval;}static int visor_calc_num_ports (struct usb_serial *serial){	int num_ports = (int)(long)(usb_get_serial_data(serial));	if (num_ports)		usb_set_serial_data(serial, NULL);	return num_ports;}static int generic_startup(struct usb_serial *serial){	struct usb_serial_port **ports = serial->port;	struct visor_private *priv;	int i;	for (i = 0; i < serial->num_ports; ++i) {		priv = kzalloc (sizeof(*priv), GFP_KERNEL);		if (!priv) {			while (i-- != 0) {				priv = usb_get_serial_port_data(ports[i]);				usb_set_serial_port_data(ports[i], NULL);				kfree(priv);			}			return -ENOMEM;		}		spin_lock_init(&priv->lock);		usb_set_serial_port_data(ports[i], priv);	}	return 0;}static int clie_3_5_startup (struct usb_serial *serial){	struct device *dev = &serial->dev->dev;	int result;	u8 data;	dbg("%s", __FUNCTION__);	/*	 * Note that PEG-300 series devices expect the following two calls.	 */	/* get the config number */	result = usb_control_msg (serial->dev, usb_rcvctrlpipe(serial->dev, 0),				  USB_REQ_GET_CONFIGURATION, USB_DIR_IN,				  0, 0, &data, 1, 3000);	if (result < 0) {		dev_err(dev, "%s: get config number failed: %d\n", __FUNCTION__, result);		return result;	}	if (result != 1) {		dev_err(dev, "%s: get config number bad return length: %d\n", __FUNCTION__, result);		return -EIO;	}	/* get the interface number */	result = usb_control_msg (serial->dev, usb_rcvctrlpipe(serial->dev, 0),				  USB_REQ_GET_INTERFACE, 				  USB_DIR_IN | USB_RECIP_INTERFACE,				  0, 0, &data, 1, 3000);	if (result < 0) {		dev_err(dev, "%s: get interface number failed: %d\n", __FUNCTION__, result);		return result;	}	if (result != 1) {		dev_err(dev, "%s: get interface number bad return length: %d\n", __FUNCTION__, result);		return -EIO;	}	return generic_startup(serial);} static int treo_attach (struct usb_serial *serial){	struct usb_serial_port *swap_port;	/* Only do this endpoint hack for the Handspring devices with	 * interrupt in endpoints, which for now are the Treo devices. */	if (!((le16_to_cpu(serial->dev->descriptor.idVendor) == HANDSPRING_VENDOR_ID) ||	      (le16_to_cpu(serial->dev->descriptor.idVendor) == KYOCERA_VENDOR_ID)) ||	    (serial->num_interrupt_in == 0))		goto generic_startup;	dbg("%s", __FUNCTION__);	/*	* It appears that Treos and Kyoceras want to use the 	* 1st bulk in endpoint to communicate with the 2nd bulk out endpoint, 	* so let's swap the 1st and 2nd bulk in and interrupt endpoints.  	* Note that swapping the bulk out endpoints would break lots of 	* apps that want to communicate on the second port.	*/#define COPY_PORT(dest, src)						\	dest->read_urb = src->read_urb;					\	dest->bulk_in_endpointAddress = src->bulk_in_endpointAddress;	\	dest->bulk_in_buffer = src->bulk_in_buffer;			\	dest->interrupt_in_urb = src->interrupt_in_urb;			\	dest->interrupt_in_endpointAddress = src->interrupt_in_endpointAddress;	\	dest->interrupt_in_buffer = src->interrupt_in_buffer;	swap_port = kmalloc(sizeof(*swap_port), GFP_KERNEL);	if (!swap_port)		return -ENOMEM;	COPY_PORT(swap_port, serial->port[0]);	COPY_PORT(serial->port[0], serial->port[1]);	COPY_PORT(serial->port[1], swap_port);	kfree(swap_port);generic_startup:	return generic_startup(serial);}static int clie_5_attach (struct usb_serial *serial){	dbg("%s", __FUNCTION__);	/* TH55 registers 2 ports. 	   Communication in from the UX50/TH55 uses bulk_in_endpointAddress from port 0 	   Communication out to the UX50/TH55 uses bulk_out_endpointAddress from port 1 	   	   Lets do a quick and dirty mapping	 */		/* some sanity check */	if (serial->num_ports < 2)		return -1;			/* port 0 now uses the modified endpoint Address */	serial->port[0]->bulk_out_endpointAddress = serial->port[1]->bulk_out_endpointAddress;	return generic_startup(serial);}static void visor_shutdown (struct usb_serial *serial){	struct visor_private *priv;	int i;	dbg("%s", __FUNCTION__);	for (i = 0; i < serial->num_ports; i++) {		priv = usb_get_serial_port_data(serial->port[i]);		if (priv) {			usb_set_serial_port_data(serial->port[i], NULL);			kfree(priv);		}	}}static int visor_ioctl (struct usb_serial_port *port, struct file * file, unsigned int cmd, unsigned long arg){	dbg("%s - port %d, cmd 0x%.4x", __FUNCTION__, port->number, cmd);	return -ENOIOCTLCMD;}static int __init visor_init (void){	int i, retval;	/* Only if parameters were passed to us */	if ((vendor>0) && (product>0)) {		struct usb_device_id usb_dev_temp[]=			{{USB_DEVICE(vendor, product),			.driver_info = (kernel_ulong_t)&palm_os_4_probe }};		/* Find the last entry in id_table */		for (i=0; ; i++) {			if (id_table[i].idVendor==0) {				id_table[i] = usb_dev_temp[0];				break;			}		}		/* Find the last entry in id_table_combined */		for (i=0; ; i++) {			if (id_table_combined[i].idVendor==0) {				id_table_combined[i] = usb_dev_temp[0];				break;			}		}		info("Untested USB device specified at time of module insertion");		info("Warning: This is not guaranteed to work");		info("Using a newer kernel is preferred to this method");		info("Adding Palm OS protocol 4.x support for unknown device: 0x%x/0x%x",			vendor, product);	}	retval = usb_serial_register(&handspring_device);	if (retval)		goto failed_handspring_register;	retval = usb_serial_register(&clie_3_5_device);	if (retval)		goto failed_clie_3_5_register;	retval = usb_serial_register(&clie_5_device);	if (retval)		goto failed_clie_5_register;	retval = usb_register(&visor_driver);	if (retval) 		goto failed_usb_register;	info(DRIVER_DESC);	return 0;failed_usb_register:	usb_serial_deregister(&clie_5_device);failed_clie_5_register:	usb_serial_deregister(&clie_3_5_device);failed_clie_3_5_register:	usb_serial_deregister(&handspring_device);failed_handspring_register:	return retval;}static void __exit visor_exit (void){	usb_deregister (&visor_driver);	usb_serial_deregister (&handspring_device);	usb_serial_deregister (&clie_3_5_device);	usb_serial_deregister (&clie_5_device);}module_init(visor_init);module_exit(visor_exit);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");module_param(stats, bool, S_IRUGO | S_IWUSR);MODULE_PARM_DESC(stats, "Enables statistics or not");module_param(vendor, ushort, 0);MODULE_PARM_DESC(vendor, "User specified vendor ID");module_param(product, ushort, 0);MODULE_PARM_DESC(product, "User specified product ID");

⌨️ 快捷键说明

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