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

📄 ftdi_sio.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 5 页
字号:
		.driver_info = (kernel_ulong_t)&ftdi_USB_UIRT_quirk },	{ USB_DEVICE(FTDI_VID, PROTEGO_SPECIAL_1) },	{ USB_DEVICE(FTDI_VID, PROTEGO_R2X0) },	{ USB_DEVICE(FTDI_VID, PROTEGO_SPECIAL_3) },	{ USB_DEVICE(FTDI_VID, PROTEGO_SPECIAL_4) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E808_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E809_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80A_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80B_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80C_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80D_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80E_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E80F_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E888_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E889_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88A_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88B_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88C_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88D_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88E_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_GUDEADS_E88F_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ELV_UO100_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ELV_UM100_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ELV_UR100_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ELV_ALC8500_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_PYRAMID_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ELV_FHZ1000PC_PID) },	/*	 * These will probably use user-space drivers.  Uncomment them if	 * you need them or use the user-specified vendor/product module	 * parameters (see ftdi_sio.h for the numbers).  Make a fuss if	 * you think the driver should recognize any of them by default.	 */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_CLI7000_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_PPS7330_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_TFM100_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_UDF77_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_UIO88_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_UAD8_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_UDA7_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_USI2_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_T1100_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_PCD200_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_ULA200_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_CSI8_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_EM1000DL_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_PCK100_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_RFP500_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_FS20SIG_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_WS300PC_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_FHZ1300PC_PID) }, */	/* { USB_DEVICE(FTDI_VID, FTDI_ELV_WS500_PID) }, */ 	{ USB_DEVICE(FTDI_VID, LINX_SDMUSBQSS_PID) }, 	{ USB_DEVICE(FTDI_VID, LINX_MASTERDEVEL2_PID) }, 	{ USB_DEVICE(FTDI_VID, LINX_FUTURE_0_PID) }, 	{ USB_DEVICE(FTDI_VID, LINX_FUTURE_1_PID) }, 	{ USB_DEVICE(FTDI_VID, LINX_FUTURE_2_PID) }, 	{ USB_DEVICE(FTDI_VID, FTDI_CCSICDU20_0_PID) }, 	{ USB_DEVICE(FTDI_VID, FTDI_CCSICDU40_1_PID) },	{ USB_DEVICE(FTDI_VID, INSIDE_ACCESSO) },	{ USB_DEVICE(INTREPID_VID, INTREPID_VALUECAN_PID) },	{ USB_DEVICE(INTREPID_VID, INTREPID_NEOVI_PID) },	{ USB_DEVICE(FALCOM_VID, FALCOM_TWIST_PID) },	{ USB_DEVICE(FALCOM_VID, FALCOM_SAMBA_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_SUUNTO_SPORTS_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_RM_CANVIEW_PID) },	{ USB_DEVICE(BANDB_VID, BANDB_USOTL4_PID) },	{ USB_DEVICE(BANDB_VID, BANDB_USTL4_PID) },	{ USB_DEVICE(BANDB_VID, BANDB_USO9ML2_PID) },	{ USB_DEVICE(FTDI_VID, EVER_ECO_PRO_CDS) },	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_0_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_1_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_2_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_0_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_1_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_2_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_3_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_4_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_5_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_6_PID) },	{ USB_DEVICE(FTDI_VID, XSENS_CONVERTER_7_PID) },	{ USB_DEVICE(MOBILITY_VID, MOBILITY_USB_SERIAL_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ACTIVE_ROBOTS_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_MHAM_Y6_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_MHAM_Y8_PID) },	{ USB_DEVICE(EVOLUTION_VID, EVOLUTION_ER1_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ARTEMIS_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ATIK_ATK16_PID) },	{ USB_DEVICE(FTDI_VID, FTDI_ATIK_ATK16HR_PID) },	{ USB_DEVICE(KOBIL_VID, KOBIL_CONV_B1_PID) },	{ USB_DEVICE(KOBIL_VID, KOBIL_CONV_KAAN_PID) },	{ },					/* Optional parameter entry */	{ }					/* Terminating entry */};MODULE_DEVICE_TABLE (usb, id_table_combined);static struct usb_driver ftdi_driver = {	.name =		"ftdi_sio",	.probe =	usb_serial_probe,	.disconnect =	usb_serial_disconnect,	.id_table =	id_table_combined,};static char *ftdi_chip_name[] = {	[SIO] = "SIO",	/* the serial part of FT8U100AX */	[FT8U232AM] = "FT8U232AM",	[FT232BM] = "FT232BM",	[FT2232C] = "FT2232C",};/* Constants for read urb and write urb */#define BUFSZ 512#define PKTSZ 64/* rx_flags */#define THROTTLED		0x01#define ACTUALLY_THROTTLED	0x02struct ftdi_private {	ftdi_chip_type_t chip_type;				/* type of the device, either SIO or FT8U232AM */	int baud_base;		/* baud base clock for divisor setting */	int custom_divisor;	/* custom_divisor kludge, this is for baud_base (different from what goes to the chip!) */	__u16 last_set_data_urb_value ;				/* the last data state set - needed for doing a break */        int write_offset;       /* This is the offset in the usb data block to write the serial data - 				 * it is different between devices				 */	int flags;		/* some ASYNC_xxxx flags are supported */	unsigned long last_dtr_rts;	/* saved modem control outputs */        wait_queue_head_t delta_msr_wait; /* Used for TIOCMIWAIT */ 	char prev_status, diff_status;        /* Used for TIOCMIWAIT */	__u8 rx_flags;		/* receive state flags (throttling) */	spinlock_t rx_lock;	/* spinlock for receive state */	struct work_struct rx_work;	int rx_processed;	__u16 interface;	/* FT2232C port interface (0 for FT232/245) */	int force_baud;		/* if non-zero, force the baud rate to this value */	int force_rtscts;	/* if non-zero, force RTS-CTS to always be enabled */};/* Used for TIOCMIWAIT */#define FTDI_STATUS_B0_MASK	(FTDI_RS0_CTS | FTDI_RS0_DSR | FTDI_RS0_RI | FTDI_RS0_RLSD)#define FTDI_STATUS_B1_MASK	(FTDI_RS_BI)/* End TIOCMIWAIT */#define FTDI_IMPL_ASYNC_FLAGS = ( ASYNC_SPD_HI | ASYNC_SPD_VHI \ ASYNC_SPD_CUST | ASYNC_SPD_SHI | ASYNC_SPD_WARP )/* function prototypes for a FTDI serial converter */static int  ftdi_sio_probe	(struct usb_serial *serial, const struct usb_device_id *id);static int  ftdi_sio_attach		(struct usb_serial *serial);static void ftdi_shutdown		(struct usb_serial *serial);static int  ftdi_open			(struct usb_serial_port *port, struct file *filp);static void ftdi_close			(struct usb_serial_port *port, struct file *filp);static int  ftdi_write			(struct usb_serial_port *port, const unsigned char *buf, int count);static int  ftdi_write_room		(struct usb_serial_port *port);static int  ftdi_chars_in_buffer	(struct usb_serial_port *port);static void ftdi_write_bulk_callback	(struct urb *urb, struct pt_regs *regs);static void ftdi_read_bulk_callback	(struct urb *urb, struct pt_regs *regs);static void ftdi_process_read		(void *param);static void ftdi_set_termios		(struct usb_serial_port *port, struct termios * old);static int  ftdi_tiocmget               (struct usb_serial_port *port, struct file *file);static int  ftdi_tiocmset		(struct usb_serial_port *port, struct file * file, unsigned int set, unsigned int clear);static int  ftdi_ioctl			(struct usb_serial_port *port, struct file * file, unsigned int cmd, unsigned long arg);static void ftdi_break_ctl		(struct usb_serial_port *port, int break_state );static void ftdi_throttle		(struct usb_serial_port *port);static void ftdi_unthrottle		(struct usb_serial_port *port);static unsigned short int ftdi_232am_baud_base_to_divisor (int baud, int base);static unsigned short int ftdi_232am_baud_to_divisor (int baud);static __u32 ftdi_232bm_baud_base_to_divisor (int baud, int base);static __u32 ftdi_232bm_baud_to_divisor (int baud);static struct usb_serial_driver ftdi_sio_device = {	.driver = {		.owner =	THIS_MODULE,		.name =		"ftdi_sio",	},	.description =		"FTDI USB Serial Device",	.id_table =		id_table_combined,	.num_interrupt_in =	0,	.num_bulk_in =		1,	.num_bulk_out =		1,	.num_ports =		1,	.probe =		ftdi_sio_probe,	.open =			ftdi_open,	.close =		ftdi_close,	.throttle =		ftdi_throttle,	.unthrottle =		ftdi_unthrottle,	.write =		ftdi_write,	.write_room =		ftdi_write_room,	.chars_in_buffer =	ftdi_chars_in_buffer,	.read_bulk_callback =	ftdi_read_bulk_callback,	.write_bulk_callback =	ftdi_write_bulk_callback,	.tiocmget =             ftdi_tiocmget,	.tiocmset =             ftdi_tiocmset,	.ioctl =		ftdi_ioctl,	.set_termios =		ftdi_set_termios,	.break_ctl =		ftdi_break_ctl,	.attach =		ftdi_sio_attach,	.shutdown =		ftdi_shutdown,};#define WDR_TIMEOUT 5000 /* default urb timeout */#define WDR_SHORT_TIMEOUT 1000	/* shorter urb timeout *//* High and low are for DTR, RTS etc etc */#define HIGH 1#define LOW 0/* * *************************************************************************** * Utlity functions * *************************************************************************** */static unsigned short int ftdi_232am_baud_base_to_divisor(int baud, int base){	unsigned short int divisor;	int divisor3 = base / 2 / baud; // divisor shifted 3 bits to the left	if ((divisor3 & 0x7) == 7) divisor3 ++; // round x.7/8 up to x+1	divisor = divisor3 >> 3;	divisor3 &= 0x7;	if (divisor3 == 1) divisor |= 0xc000; else // 0.125	if (divisor3 >= 4) divisor |= 0x4000; else // 0.5	if (divisor3 != 0) divisor |= 0x8000;      // 0.25	if (divisor == 1) divisor = 0;	/* special case for maximum baud rate */	return divisor;}static unsigned short int ftdi_232am_baud_to_divisor(int baud){	 return(ftdi_232am_baud_base_to_divisor(baud, 48000000));}static __u32 ftdi_232bm_baud_base_to_divisor(int baud, int base){	static const unsigned char divfrac[8] = { 0, 3, 2, 4, 1, 5, 6, 7 };	__u32 divisor;	int divisor3 = base / 2 / baud; // divisor shifted 3 bits to the left	divisor = divisor3 >> 3;	divisor |= (__u32)divfrac[divisor3 & 0x7] << 14;	/* Deal with special cases for highest baud rates. */	if (divisor == 1) divisor = 0; else	// 1.0	if (divisor == 0x4001) divisor = 1;	// 1.5	return divisor;}static __u32 ftdi_232bm_baud_to_divisor(int baud){	 return(ftdi_232bm_baud_base_to_divisor(baud, 48000000));}#define set_mctrl(port, set)		update_mctrl((port), (set), 0)#define clear_mctrl(port, clear)	update_mctrl((port), 0, (clear))static int update_mctrl(struct usb_serial_port *port, unsigned int set, unsigned int clear){	struct ftdi_private *priv = usb_get_serial_port_data(port);	char *buf;	unsigned urb_value;	int rv;	if (((set | clear) & (TIOCM_DTR | TIOCM_RTS)) == 0) {		dbg("%s - DTR|RTS not being set|cleared", __FUNCTION__);		return 0;	/* no change */	}	buf = kmalloc(1, GFP_NOIO);	if (!buf) {		return -ENOMEM;	}	clear &= ~set;	/* 'set' takes precedence over 'clear' */	urb_value = 0;	if (clear & TIOCM_DTR)		urb_value |= FTDI_SIO_SET_DTR_LOW;	if (clear & TIOCM_RTS)		urb_value |= FTDI_SIO_SET_RTS_LOW;	if (set & TIOCM_DTR)		urb_value |= FTDI_SIO_SET_DTR_HIGH;	if (set & TIOCM_RTS)		urb_value |= FTDI_SIO_SET_RTS_HIGH;	rv = usb_control_msg(port->serial->dev,			       usb_sndctrlpipe(port->serial->dev, 0),			       FTDI_SIO_SET_MODEM_CTRL_REQUEST, 			       FTDI_SIO_SET_MODEM_CTRL_REQUEST_TYPE,			       urb_value, priv->interface,			       buf, 0, WDR_TIMEOUT);	kfree(buf);	if (rv < 0) {		err("%s Error from MODEM_CTRL urb: DTR %s, RTS %s",				__FUNCTION__,				(set & TIOCM_DTR) ? "HIGH" :				(clear & TIOCM_DTR) ? "LOW" : "unchanged",				(set & TIOCM_RTS) ? "HIGH" :				(clear & TIOCM_RTS) ? "LOW" : "unchanged");	} else {		dbg("%s - DTR %s, RTS %s", __FUNCTION__,				(set & TIOCM_DTR) ? "HIGH" :				(clear & TIOCM_DTR) ? "LOW" : "unchanged",				(set & TIOCM_RTS) ? "HIGH" :				(clear & TIOCM_RTS) ? "LOW" : "unchanged");		priv->last_dtr_rts = (priv->last_dtr_rts & ~clear) | set;	}	return rv;}static __u32 get_ftdi_divisor(struct usb_serial_port * port);static int change_speed(struct usb_serial_port *port){	struct ftdi_private *priv = usb_get_serial_port_data(port);	char *buf;        __u16 urb_value;	__u16 urb_index;	__u32 urb_index_value;	int rv;	buf = kmalloc(1, GFP_NOIO);	if (!buf)		return -ENOMEM;	urb_index_value = get_ftdi_divisor(port);	urb_value = (__u16)urb_index_value;	urb_index = (__u16)(urb_index_value >> 16);	if (priv->interface) {	/* FT2232C */		urb_index = (__u16)((urb_index << 8) | priv->interface);	}		rv = usb_control_msg(port->serial->dev,			    usb_sndctrlpipe(port->serial->dev, 0),			    FTDI_SIO_SET_BAUDRATE_REQUEST,			    FTDI_SIO_SET_BAUDRATE_REQUEST_TYPE,			    urb_value, urb_index,			    buf, 0, WDR_SHORT_TIMEOUT);	kfree(buf);	return rv;}static __u32 get_ftdi_divisor(struct usb_serial_port * port){ /* get_ftdi_divisor */	struct ftdi_private *priv = usb_get_serial_port_data(port);	__u32 div_value = 0;	int div_okay = 1;	int baud;	/*	 * The logic involved in setting the baudrate can be cleanly split in 3 steps.	 * Obtaining the actual baud rate is a little tricky since unix traditionally	 * somehow ignored the possibility to set non-standard baud rates.	 * 1. Standard baud rates are set in tty->termios->c_cflag	 * 2. If these are not enough, you can set any speed using alt_speed as follows:	 *    - set tty->termios->c_cflag speed to B38400	 *    - set your real speed in tty->alt_speed; it gets ignored when	 *      alt_speed==0, (or)	 *    - call TIOCSSERIAL ioctl with (struct serial_struct) set as follows:	 *      flags & ASYNC_SPD_MASK == ASYNC_SPD_[HI, VHI, SHI, WARP], this just	 *      sets alt_speed to (HI: 57600, VHI: 115200, SHI: 230400, WARP: 460800)	 * ** Steps 1, 2 are done courtesy of tty_get_baud_rate	 * 3. You can also set baud rate by setting custom divisor as follows	 *    - set tty->termios->c_cflag speed to B38400	 *    - call TIOCSSERIAL ioctl with (struct serial_struct) set as follows:	 *      o flags & ASYNC_SPD_MASK == ASYNC_SPD_CUST	 *      o custom_divisor set to baud_base / your_new_baudrate	 * ** Step 3 is done courtesy of code borrowed from serial.c - I should really	 *    spend some time and separate+move this common code to serial.c, it is	 *    replicated in nearly every serial driver you see.	 */	/* 1. Get the baud rate from the tty settings, this observes alt_speed hack */	baud = tty_get_baud_rate(port->tty);	dbg("%s - tty_get_baud_rate reports speed %d", __FUNCTION__, baud);	/* 2. Observe async-compatible custom_divisor hack, update baudrate if needed */

⌨️ 快捷键说明

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