diff options
Diffstat (limited to 'drivers/usb/serial/ftdi_sio.c')
| -rw-r--r-- | drivers/usb/serial/ftdi_sio.c | 830 | 
1 files changed, 499 insertions, 331 deletions
diff --git a/drivers/usb/serial/ftdi_sio.c b/drivers/usb/serial/ftdi_sio.c index 76f8b355667..8a3813be1b2 100644 --- a/drivers/usb/serial/ftdi_sio.c +++ b/drivers/usb/serial/ftdi_sio.c @@ -1,7 +1,7 @@  /*   * USB FTDI SIO driver   * - *	Copyright (C) 2009 - 2010 + *	Copyright (C) 2009 - 2013   *	    Johan Hovold (jhovold@gmail.com)   *	Copyright (C) 1999 - 2001   *	    Greg Kroah-Hartman (greg@kroah.com) @@ -17,7 +17,7 @@   * See Documentation/usb/usb-serial.txt for more information on using this   * driver   * - * See http://ftdi-usb-sio.sourceforge.net for upto date testing info + * See http://ftdi-usb-sio.sourceforge.net for up to date testing info   *	and extra documentation   *   * Change entries from 2004 and earlier can be found in versions of this @@ -33,7 +33,6 @@  #include <linux/kernel.h>  #include <linux/errno.h> -#include <linux/init.h>  #include <linux/slab.h>  #include <linux/tty.h>  #include <linux/tty_driver.h> @@ -48,19 +47,11 @@  #include "ftdi_sio.h"  #include "ftdi_sio_ids.h" -/* - * Version Information - */ -#define DRIVER_VERSION "v1.6.0"  #define DRIVER_AUTHOR "Greg Kroah-Hartman <greg@kroah.com>, Bill Ryder <bryder@sgi.com>, Kuba Ober <kuba@mareimbrium.org>, Andreas Mohr, Johan Hovold <jhovold@gmail.com>"  #define DRIVER_DESC "USB FTDI Serial Converters Driver" -static int debug; -static __u16 vendor = FTDI_VID; -static __u16 product;  struct ftdi_private { -	struct kref kref;  	enum ftdi_chip_type chip_type;  				/* type of device, either SIO or FT8U232AM */  	int baud_base;		/* baud base clock for divisor setting */ @@ -73,9 +64,8 @@ struct ftdi_private {  				 */  	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 */ -	struct usb_serial_port *port; +	char prev_status;        /* Used for TIOCMIWAIT */ +	char transmit_empty;	/* If transmitter is empty or not */  	__u16 interface;	/* FT2232C, FT2232H or FT4232H port interface  				   (0 for FT232/245) */ @@ -99,6 +89,8 @@ struct ftdi_sio_quirk {  static int   ftdi_jtag_probe(struct usb_serial *serial);  static int   ftdi_mtxorb_hack_setup(struct usb_serial *serial);  static int   ftdi_NDI_device_setup(struct usb_serial *serial); +static int   ftdi_stmclite_probe(struct usb_serial *serial); +static int   ftdi_8u2232c_probe(struct usb_serial *serial);  static void  ftdi_USB_UIRT_setup(struct ftdi_private *priv);  static void  ftdi_HE_TIRA1_setup(struct ftdi_private *priv); @@ -122,6 +114,14 @@ static struct ftdi_sio_quirk ftdi_HE_TIRA1_quirk = {  	.port_probe = ftdi_HE_TIRA1_setup,  }; +static struct ftdi_sio_quirk ftdi_stmclite_quirk = { +	.probe	= ftdi_stmclite_probe, +}; + +static struct ftdi_sio_quirk ftdi_8u2232c_quirk = { +	.probe	= ftdi_8u2232c_probe, +}; +  /*   * The 8U232AM has the same API as the sio except for:   * - it can support MUCH higher baudrates; up to: @@ -141,14 +141,18 @@ static struct ftdi_sio_quirk ftdi_HE_TIRA1_quirk = {  /* - * Device ID not listed? Test via module params product/vendor or - * /sys/bus/usb/ftdi_sio/new_id, then send patch/report! + * Device ID not listed? Test it using + * /sys/bus/usb-serial/drivers/ftdi_sio/new_id and send a patch or report.   */ -static struct usb_device_id id_table_combined [] = { +static const struct usb_device_id id_table_combined[] = { +	{ USB_DEVICE(FTDI_VID, FTDI_ZEITCONTROL_TAGTRACE_MIFARE_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_CTI_MINI_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_CTI_NANO_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_AMC232_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_CANUSB_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_CANDAPTER_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_NXTCAM_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_EV3CON_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SCS_DEVICE_0_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SCS_DEVICE_1_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SCS_DEVICE_2_PID) }, @@ -169,8 +173,11 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_8U232AM_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_8U232AM_ALT_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_232RL_PID) }, -	{ USB_DEVICE(FTDI_VID, FTDI_8U2232C_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_8U2232C_PID) , +		.driver_info = (kernel_ulong_t)&ftdi_8u2232c_quirk },  	{ USB_DEVICE(FTDI_VID, FTDI_4232H_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_232H_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_FTX_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_MICRO_CHAMELEON_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_RELAIS_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_OPENDCC_PID) }, @@ -178,9 +185,15 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_OPENDCC_THROTTLE_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_OPENDCC_GATEWAY_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_OPENDCC_GBM_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_OPENDCC_GBM_BOOST_PID) }, +	{ USB_DEVICE(NEWPORT_VID, NEWPORT_AGILIS_PID) }, +	{ USB_DEVICE(NEWPORT_VID, NEWPORT_CONEX_CC_PID) }, +	{ USB_DEVICE(NEWPORT_VID, NEWPORT_CONEX_AGP_PID) },  	{ USB_DEVICE(INTERBIOMETRICS_VID, INTERBIOMETRICS_IOBOARD_PID) },  	{ USB_DEVICE(INTERBIOMETRICS_VID, INTERBIOMETRICS_MINI_IOBOARD_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SPROG_II) }, +	{ USB_DEVICE(FTDI_VID, FTDI_TAGSYS_LP101_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_TAGSYS_P200X_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_LENZ_LIUSB_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_XF_632_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_XF_634_PID) }, @@ -191,6 +204,8 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_XF_640_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_XF_642_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_DSS20_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_URBAN_0_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_URBAN_1_PID) },  	{ USB_DEVICE(FTDI_NF_RIC_VID, FTDI_NF_RIC_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_VNHCPCUSB_D_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_MTXORB_0_PID) }, @@ -201,6 +216,7 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_MTXORB_5_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_MTXORB_6_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_R2000KU_TRUE_RNG) }, +	{ USB_DEVICE(FTDI_VID, FTDI_VARDAAN_PID) },  	{ USB_DEVICE(MTXORB_VID, MTXORB_FTDI_RANGE_0100_PID) },  	{ USB_DEVICE(MTXORB_VID, MTXORB_FTDI_RANGE_0101_PID) },  	{ USB_DEVICE(MTXORB_VID, MTXORB_FTDI_RANGE_0102_PID) }, @@ -516,8 +532,13 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803_6_PID) },  	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803_7_PID) },  	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803_8_PID) }, +	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803R_1_PID) }, +	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803R_2_PID) }, +	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803R_3_PID) }, +	{ USB_DEVICE(SEALEVEL_VID, SEALEVEL_2803R_4_PID) },  	{ USB_DEVICE(IDTECH_VID, IDTECH_IDT1221U_PID) },  	{ USB_DEVICE(OCT_VID, OCT_US101_PID) }, +	{ USB_DEVICE(OCT_VID, OCT_DK201_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_HE_TIRA1_PID),  		.driver_info = (kernel_ulong_t)&ftdi_HE_TIRA1_quirk },  	{ USB_DEVICE(FTDI_VID, FTDI_USB_UIRT_PID), @@ -556,9 +577,15 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_IBS_APP70_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_IBS_PEDO_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_IBS_PROD_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_TAVIR_STK500_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_TIAO_UMPA_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(FTDI_VID, FTDI_NT_ORIONLXM_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	/*  	 * ELV devices:  	 */ +	{ USB_DEVICE(FTDI_ELV_VID, FTDI_ELV_WS300_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ELV_USR_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ELV_MSM1_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ELV_KL100_PID) }, @@ -614,7 +641,9 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_OCEANIC_PID) },  	{ USB_DEVICE(TTI_VID, TTI_QL355P_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_RM_CANVIEW_PID) }, +	{ USB_DEVICE(ACTON_VID, ACTON_SPECTRAPRO_PID) },  	{ USB_DEVICE(CONTEC_VID, CONTEC_COM1USBH_PID) }, +	{ USB_DEVICE(MITSUBISHI_VID, MITSUBISHI_FXUSB_PID) },  	{ USB_DEVICE(BANDB_VID, BANDB_USOTL4_PID) },  	{ USB_DEVICE(BANDB_VID, BANDB_USTL4_PID) },  	{ USB_DEVICE(BANDB_VID, BANDB_USO9ML2_PID) }, @@ -635,6 +664,7 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, EVER_ECO_PRO_CDS) },  	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_1_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_2_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_4N_GALAXY_DE_3_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) }, @@ -643,6 +673,7 @@ static struct usb_device_id id_table_combined [] = {  	{ 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(FTDI_VID, FTDI_OMNI1509) },  	{ USB_DEVICE(MOBILITY_VID, MOBILITY_USB_SERIAL_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ACTIVE_ROBOTS_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_MHAM_KW_PID) }, @@ -674,11 +705,23 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_PCDJ_DAC2_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_RRCIRKITS_LOCOBUFFER_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ASK_RDR400_PID) }, -	{ USB_DEVICE(ICOM_ID1_VID, ICOM_ID1_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_NZR_SEM_USB_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_1_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_OPC_U_UC_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2C1_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2C2_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2D_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2VT_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2VR_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP4KVT_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP4KVR_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2KVT_PID) }, +	{ USB_DEVICE(ICOM_VID, ICOM_ID_RP2KVR_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ACG_HFDUAL_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_YEI_SERVOCENTER31_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_THORLABS_PID) }, -	{ USB_DEVICE(TESTO_VID, TESTO_USB_INTERFACE_PID) }, +	{ USB_DEVICE(TESTO_VID, TESTO_1_PID) }, +	{ USB_DEVICE(TESTO_VID, TESTO_3_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_GAMMA_SCOUT_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_TACTRIX_OPENPORT_13M_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_TACTRIX_OPENPORT_13S_PID) }, @@ -695,14 +738,44 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_NDI_AURORA_SCU_PID),  		.driver_info = (kernel_ulong_t)&ftdi_NDI_device_quirk },  	{ USB_DEVICE(TELLDUS_VID, TELLDUS_TELLSTICK_PID) }, -	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_SERIAL_VX7_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_S03_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_59_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_57A_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_57B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_29A_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_29B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_29F_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_62B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_S01_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_63_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_29C_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_81B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_82B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_K5D_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_K4Y_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_K5G_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_S05_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_60_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_61_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_62_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_63B_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_64_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_65_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_92_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_92D_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_W5R_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_A5R_PID) }, +	{ USB_DEVICE(RTSYSTEMS_VID, RTSYSTEMS_USB_PW1_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_MAXSTREAM_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_PHI_FISCO_PID) },  	{ USB_DEVICE(TML_VID, TML_USB_SERIAL_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_ELSTER_UNICOM_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_PROPOX_JTAGCABLEII_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_PROPOX_ISPCABLEIII_PID) },  	{ USB_DEVICE(OLIMEX_VID, OLIMEX_ARM_USB_OCD_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(OLIMEX_VID, OLIMEX_ARM_USB_OCD_H_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(FIC_VID, FIC_NEO1973_DEBUG_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(FTDI_VID, FTDI_OOCDLINK_PID), @@ -711,6 +784,8 @@ static struct usb_device_id id_table_combined [] = {  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(FTDI_VID, LMI_LM3S_EVAL_BOARD_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(FTDI_VID, LMI_LM3S_ICDI_BOARD_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(FTDI_VID, FTDI_TURTELIZER_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(RATOC_VENDOR_ID, RATOC_PRODUCT_ID_USB60F) }, @@ -757,15 +832,41 @@ static struct usb_device_id id_table_combined [] = {  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(ADI_VID, ADI_GNICEPLUS_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE_AND_INTERFACE_INFO(MICROCHIP_VID, MICROCHIP_USB_BOARD_PID, +					USB_CLASS_VENDOR_SPEC, +					USB_SUBCLASS_VENDOR_SPEC, 0x00) },  	{ USB_DEVICE(JETI_VID, JETI_SPC1201_PID) },  	{ USB_DEVICE(MARVELL_VID, MARVELL_SHEEVAPLUG_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(LARSENBRUSGAARD_VID, LB_ALTITRACK_PID) },  	{ USB_DEVICE(GN_OTOMETRICS_VID, AURICAL_USB_PID) }, +	{ USB_DEVICE(FTDI_VID, PI_C865_PID) }, +	{ USB_DEVICE(FTDI_VID, PI_C857_PID) }, +	{ USB_DEVICE(PI_VID, PI_C866_PID) }, +	{ USB_DEVICE(PI_VID, PI_C663_PID) }, +	{ USB_DEVICE(PI_VID, PI_C725_PID) }, +	{ USB_DEVICE(PI_VID, PI_E517_PID) }, +	{ USB_DEVICE(PI_VID, PI_C863_PID) }, +	{ USB_DEVICE(PI_VID, PI_E861_PID) }, +	{ USB_DEVICE(PI_VID, PI_C867_PID) }, +	{ USB_DEVICE(PI_VID, PI_E609_PID) }, +	{ USB_DEVICE(PI_VID, PI_E709_PID) }, +	{ USB_DEVICE(PI_VID, PI_100F_PID) }, +	{ USB_DEVICE(PI_VID, PI_1011_PID) }, +	{ USB_DEVICE(PI_VID, PI_1012_PID) }, +	{ USB_DEVICE(PI_VID, PI_1013_PID) }, +	{ USB_DEVICE(PI_VID, PI_1014_PID) }, +	{ USB_DEVICE(PI_VID, PI_1015_PID) }, +	{ USB_DEVICE(PI_VID, PI_1016_PID) }, +	{ USB_DEVICE(KONDO_VID, KONDO_USB_SERIAL_PID) },  	{ USB_DEVICE(BAYER_VID, BAYER_CONTOUR_CABLE_PID) },  	{ USB_DEVICE(FTDI_VID, MARVELL_OPENRD_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(FTDI_VID, TI_XDS100V2_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk },  	{ USB_DEVICE(FTDI_VID, HAMEG_HO820_PID) }, +	{ USB_DEVICE(FTDI_VID, HAMEG_HO720_PID) }, +	{ USB_DEVICE(FTDI_VID, HAMEG_HO730_PID) },  	{ USB_DEVICE(FTDI_VID, HAMEG_HO870_PID) },  	{ USB_DEVICE(FTDI_VID, MJSG_GENERIC_PID) },  	{ USB_DEVICE(FTDI_VID, MJSG_SR_RADIO_PID) }, @@ -794,22 +895,63 @@ static struct usb_device_id id_table_combined [] = {  	{ USB_DEVICE(FTDI_VID, FTDI_SCIENCESCOPE_LOGBOOKML_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SCIENCESCOPE_LS_LOGBOOK_PID) },  	{ USB_DEVICE(FTDI_VID, FTDI_SCIENCESCOPE_HS_LOGBOOK_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_CINTERION_MC55I_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_DOTEC_PID) },  	{ USB_DEVICE(QIHARDWARE_VID, MILKYMISTONE_JTAGSERIAL_PID),  		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, -	{ },					/* Optional parameter entry */ +	{ USB_DEVICE(ST_VID, ST_STMCLT_2232_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(ST_VID, ST_STMCLT_4232_PID), +		.driver_info = (kernel_ulong_t)&ftdi_stmclite_quirk }, +	{ USB_DEVICE(FTDI_VID, FTDI_RF_R106) }, +	{ USB_DEVICE(FTDI_VID, FTDI_DISTORTEC_JTAG_LOCK_PICK_PID), +		.driver_info = (kernel_ulong_t)&ftdi_jtag_quirk }, +	{ USB_DEVICE(FTDI_VID, FTDI_LUMEL_PD12_PID) }, +	/* Crucible Devices */ +	{ USB_DEVICE(FTDI_VID, FTDI_CT_COMET_PID) }, +	{ USB_DEVICE(FTDI_VID, FTDI_Z3X_PID) }, +	/* Cressi Devices */ +	{ USB_DEVICE(FTDI_VID, FTDI_CRESSI_PID) }, +	/* Brainboxes Devices */ +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_VX_001_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_VX_012_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_VX_023_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_VX_034_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_101_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_3_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_4_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_5_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_6_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_7_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_160_8_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_257_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_279_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_279_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_279_3_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_279_4_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_313_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_324_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_346_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_346_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_357_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_606_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_606_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_606_3_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_701_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_701_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_842_1_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_842_2_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_842_3_PID) }, +	{ USB_DEVICE(BRAINBOXES_VID, BRAINBOXES_US_842_4_PID) }, +	/* Infineon Devices */ +	{ USB_DEVICE_INTERFACE_NUMBER(INFINEON_VID, INFINEON_TRIBOARD_PID, 1) },  	{ }					/* 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, -	.no_dynamic_id =	1, -}; -  static const char *ftdi_chip_name[] = {  	[SIO] = "SIO",	/* the serial part of FT8U100AX */  	[FT8U232AM] = "FT8U232AM", @@ -817,7 +959,9 @@ static const char *ftdi_chip_name[] = {  	[FT2232C] = "FT2232C",  	[FT232RL] = "FT232RL",  	[FT2232H] = "FT2232H", -	[FT4232H] = "FT4232H" +	[FT4232H] = "FT4232H", +	[FT232H]  = "FT232H", +	[FTX]     = "FT-X"  }; @@ -826,28 +970,27 @@ static const char *ftdi_chip_name[] = {  #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_port_probe(struct usb_serial_port *port);  static int  ftdi_sio_port_remove(struct usb_serial_port *port);  static int  ftdi_open(struct tty_struct *tty, struct usb_serial_port *port); -static void ftdi_close(struct usb_serial_port *port);  static void ftdi_dtr_rts(struct usb_serial_port *port, int on);  static void ftdi_process_read_urb(struct urb *urb);  static int ftdi_prepare_write_buffer(struct usb_serial_port *port,  						void *dest, size_t size);  static void ftdi_set_termios(struct tty_struct *tty,  			struct usb_serial_port *port, struct ktermios *old); -static int  ftdi_tiocmget(struct tty_struct *tty, struct file *file); -static int  ftdi_tiocmset(struct tty_struct *tty, struct file *file, +static int  ftdi_tiocmget(struct tty_struct *tty); +static int  ftdi_tiocmset(struct tty_struct *tty,  			unsigned int set, unsigned int clear); -static int  ftdi_ioctl(struct tty_struct *tty, struct file *file, +static int  ftdi_ioctl(struct tty_struct *tty,  			unsigned int cmd, unsigned long arg);  static void ftdi_break_ctl(struct tty_struct *tty, int break_state); +static bool ftdi_tx_empty(struct usb_serial_port *port); +static int ftdi_get_modem_status(struct usb_serial_port *port, +						unsigned char status[2]);  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); @@ -862,7 +1005,6 @@ static struct usb_serial_driver ftdi_sio_device = {  		.name =		"ftdi_sio",  	},  	.description =		"FTDI USB Serial Device", -	.usb_driver = 		&ftdi_driver,  	.id_table =		id_table_combined,  	.num_ports =		1,  	.bulk_in_size =		512, @@ -871,7 +1013,6 @@ static struct usb_serial_driver ftdi_sio_device = {  	.port_probe =		ftdi_sio_port_probe,  	.port_remove =		ftdi_sio_port_remove,  	.open =			ftdi_open, -	.close =		ftdi_close,  	.dtr_rts =		ftdi_dtr_rts,  	.throttle =		usb_serial_generic_throttle,  	.unthrottle =		usb_serial_generic_unthrottle, @@ -879,19 +1020,22 @@ static struct usb_serial_driver ftdi_sio_device = {  	.prepare_write_buffer =	ftdi_prepare_write_buffer,  	.tiocmget =		ftdi_tiocmget,  	.tiocmset =		ftdi_tiocmset, +	.tiocmiwait =		usb_serial_generic_tiocmiwait, +	.get_icount =           usb_serial_generic_get_icount,  	.ioctl =		ftdi_ioctl,  	.set_termios =		ftdi_set_termios,  	.break_ctl =		ftdi_break_ctl, +	.tx_empty =		ftdi_tx_empty, +}; + +static struct usb_serial_driver * const serial_drivers[] = { +	&ftdi_sio_device, NULL  };  #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 -  /*   * ***************************************************************************   * Utility functions @@ -951,7 +1095,7 @@ static __u32 ftdi_2232h_baud_base_to_divisor(int baud, int base)  	int divisor3;  	/* hi-speed baud rate is 10-bit sampling instead of 16-bit */ -	divisor3 = (base / 10 / baud) * 8; +	divisor3 = base * 8 / (baud * 10);  	divisor = divisor3 >> 3;  	divisor |= (__u32)divfrac[divisor3 & 0x7] << 14; @@ -981,11 +1125,12 @@ 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); +	struct device *dev = &port->dev;  	unsigned urb_value;  	int rv;  	if (((set | clear) & (TIOCM_DTR | TIOCM_RTS)) == 0) { -		dbg("%s - DTR|RTS not being set|cleared", __func__); +		dev_dbg(dev, "%s - DTR|RTS not being set|cleared\n", __func__);  		return 0;	/* no change */  	} @@ -1006,18 +1151,15 @@ static int update_mctrl(struct usb_serial_port *port, unsigned int set,  			       urb_value, priv->interface,  			       NULL, 0, WDR_TIMEOUT);  	if (rv < 0) { -		dbg("%s Error from MODEM_CTRL urb: DTR %s, RTS %s", -				__func__, -				(set & TIOCM_DTR) ? "HIGH" : -				(clear & TIOCM_DTR) ? "LOW" : "unchanged", -				(set & TIOCM_RTS) ? "HIGH" : -				(clear & TIOCM_RTS) ? "LOW" : "unchanged"); +		dev_dbg(dev, "%s Error from MODEM_CTRL urb: DTR %s, RTS %s\n", +			__func__, +			(set & TIOCM_DTR) ? "HIGH" : (clear & TIOCM_DTR) ? "LOW" : "unchanged", +			(set & TIOCM_RTS) ? "HIGH" : (clear & TIOCM_RTS) ? "LOW" : "unchanged"); +		rv = usb_translate_errors(rv);  	} else { -		dbg("%s - DTR %s, RTS %s", __func__, -				(set & TIOCM_DTR) ? "HIGH" : -				(clear & TIOCM_DTR) ? "LOW" : "unchanged", -				(set & TIOCM_RTS) ? "HIGH" : -				(clear & TIOCM_RTS) ? "LOW" : "unchanged"); +		dev_dbg(dev, "%s - DTR %s, RTS %s\n", __func__, +			(set & TIOCM_DTR) ? "HIGH" : (clear & TIOCM_DTR) ? "LOW" : "unchanged", +			(set & TIOCM_RTS) ? "HIGH" : (clear & TIOCM_RTS) ? "LOW" : "unchanged");  		/* FIXME: locking on last_dtr_rts */  		priv->last_dtr_rts = (priv->last_dtr_rts & ~clear) | set;  	} @@ -1029,6 +1171,7 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  						struct usb_serial_port *port)  {  	struct ftdi_private *priv = usb_get_serial_port_data(port); +	struct device *dev = &port->dev;  	__u32 div_value = 0;  	int div_okay = 1;  	int baud; @@ -1064,7 +1207,7 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  	      alt_speed hack */  	baud = tty_get_baud_rate(tty); -	dbg("%s - tty_get_baud_rate reports speed %d", __func__, baud); +	dev_dbg(dev, "%s - tty_get_baud_rate reports speed %d\n", __func__, baud);  	/* 2. Observe async-compatible custom_divisor hack, update baudrate  	   if needed */ @@ -1073,8 +1216,8 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  	    ((priv->flags & ASYNC_SPD_MASK) == ASYNC_SPD_CUST) &&  	     (priv->custom_divisor)) {  		baud = priv->baud_base / priv->custom_divisor; -		dbg("%s - custom divisor %d sets baud rate to %d", -				__func__, priv->custom_divisor, baud); +		dev_dbg(dev, "%s - custom divisor %d sets baud rate to %d\n", +			__func__, priv->custom_divisor, baud);  	}  	/* 3. Convert baudrate to device-specific divisor */ @@ -1096,8 +1239,8 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  		case 115200: div_value = ftdi_sio_b115200; break;  		} /* baud */  		if (div_value == 0) { -			dbg("%s - Baudrate (%d) requested is not supported", -							__func__,  baud); +			dev_dbg(dev, "%s - Baudrate (%d) requested is not supported\n", +				__func__,  baud);  			div_value = ftdi_sio_b9600;  			baud = 9600;  			div_okay = 0; @@ -1107,7 +1250,7 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  		if (baud <= 3000000) {  			div_value = ftdi_232am_baud_to_divisor(baud);  		} else { -			dbg("%s - Baud rate too high!", __func__); +			dev_dbg(dev, "%s - Baud rate too high!\n", __func__);  			baud = 9600;  			div_value = ftdi_232am_baud_to_divisor(9600);  			div_okay = 0; @@ -1115,7 +1258,8 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  		break;  	case FT232BM: /* FT232BM chip */  	case FT2232C: /* FT2232C chip */ -	case FT232RL: +	case FT232RL: /* FT232RL chip */ +	case FTX:     /* FT-X series */  		if (baud <= 3000000) {  			__u16 product_id = le16_to_cpu(  				port->serial->dev->descriptor.idProduct); @@ -1129,7 +1273,7 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  			}  			div_value = ftdi_232bm_baud_to_divisor(baud);  		} else { -			dbg("%s - Baud rate too high!", __func__); +			dev_dbg(dev, "%s - Baud rate too high!\n", __func__);  			div_value = ftdi_232bm_baud_to_divisor(9600);  			div_okay = 0;  			baud = 9600; @@ -1137,12 +1281,13 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  		break;  	case FT2232H: /* FT2232H chip */  	case FT4232H: /* FT4232H chip */ -		if ((baud <= 12000000) & (baud >= 1200)) { +	case FT232H:  /* FT232H chip */ +		if ((baud <= 12000000) && (baud >= 1200)) {  			div_value = ftdi_2232h_baud_to_divisor(baud);  		} else if (baud < 1200) {  			div_value = ftdi_232bm_baud_to_divisor(baud);  		} else { -			dbg("%s - Baud rate too high!", __func__); +			dev_dbg(dev, "%s - Baud rate too high!\n", __func__);  			div_value = ftdi_232bm_baud_to_divisor(9600);  			div_okay = 0;  			baud = 9600; @@ -1151,7 +1296,7 @@ static __u32 get_ftdi_divisor(struct tty_struct *tty,  	} /* priv->chip_type */  	if (div_okay) { -		dbg("%s - Baud rate set to %d (divisor 0x%lX) on chip %s", +		dev_dbg(dev, "%s - Baud rate set to %d (divisor 0x%lX) on chip %s\n",  			__func__, baud, (unsigned long)div_value,  			ftdi_chip_name[priv->chip_type]);  	} @@ -1171,7 +1316,10 @@ static int change_speed(struct tty_struct *tty, struct usb_serial_port *port)  	urb_index_value = get_ftdi_divisor(tty, port);  	urb_value = (__u16)urb_index_value;  	urb_index = (__u16)(urb_index_value >> 16); -	if (priv->interface) {	/* FT2232C */ +	if ((priv->chip_type == FT2232C) || (priv->chip_type == FT2232H) || +		(priv->chip_type == FT4232H) || (priv->chip_type == FT232H)) { +		/* Probably the BM type needs the MSB of the encoded fractional +		 * divider also moved like for the chips above. Any infos? */  		urb_index = (__u16)((urb_index << 8) | priv->interface);  	} @@ -1194,7 +1342,7 @@ static int write_latency_timer(struct usb_serial_port *port)  	if (priv->flags & ASYNC_LOW_LATENCY)  		l = 1; -	dbg("%s: setting latency timer = %i", __func__, l); +	dev_dbg(&port->dev, "%s: setting latency timer = %i\n", __func__, l);  	rv = usb_control_msg(udev,  			     usb_sndctrlpipe(udev, 0), @@ -1214,8 +1362,6 @@ static int read_latency_timer(struct usb_serial_port *port)  	unsigned char *buf;  	int rv; -	dbg("%s", __func__); -  	buf = kmalloc(1, GFP_KERNEL);  	if (!buf)  		return -ENOMEM; @@ -1280,8 +1426,7 @@ static int set_serial_info(struct tty_struct *tty,  		goto check_and_exit;  	} -	if ((new_serial.baud_base != priv->baud_base) && -	    (new_serial.baud_base < 9600)) { +	if (new_serial.baud_base != priv->baud_base) {  		mutex_unlock(&priv->cfg_lock);  		return -EINVAL;  	} @@ -1320,6 +1465,23 @@ check_and_exit:  	return 0;  } +static int get_lsr_info(struct usb_serial_port *port, +			struct serial_struct __user *retinfo) +{ +	struct ftdi_private *priv = usb_get_serial_port_data(port); +	unsigned int result = 0; + +	if (!retinfo) +		return -EFAULT; + +	if (priv->transmit_empty) +		result = TIOCSER_TEMT; + +	if (copy_to_user(retinfo, &result, sizeof(unsigned int))) +		return -EFAULT; +	return 0; +} +  /* Determine type of FTDI chip based on USB config and descriptor. */  static void ftdi_determine_type(struct usb_serial_port *port) @@ -1335,8 +1497,8 @@ static void ftdi_determine_type(struct usb_serial_port *port)  	version = le16_to_cpu(udev->descriptor.bcdDevice);  	interfaces = udev->actconfig->desc.bNumInterfaces; -	dbg("%s: bcdDevice = 0x%x, bNumInterfaces = %u", __func__, -			version, interfaces); +	dev_dbg(&port->dev, "%s: bcdDevice = 0x%x, bNumInterfaces = %u\n", __func__, +		version, interfaces);  	if (interfaces > 1) {  		int inter; @@ -1366,8 +1528,9 @@ static void ftdi_determine_type(struct usb_serial_port *port)  		/* BM-type devices have a bug where bcdDevice gets set  		 * to 0x200 when iSerialNumber is 0.  */  		if (version < 0x500) { -			dbg("%s: something fishy - bcdDevice too low for multi-interface device", -					__func__); +			dev_dbg(&port->dev, +				"%s: something fishy - bcdDevice too low for multi-interface device\n", +				__func__);  		}  	} else if (version < 0x200) {  		/* Old device.  Assume it's the original SIO. */ @@ -1381,10 +1544,17 @@ static void ftdi_determine_type(struct usb_serial_port *port)  	} else if (version < 0x600) {  		/* Assume it's an FT232BM (or FT245BM) */  		priv->chip_type = FT232BM; -	} else { -		/* Assume it's an FT232R */ +	} else if (version < 0x900) { +		/* Assume it's an FT232RL */  		priv->chip_type = FT232RL; +	} else if (version < 0x1000) { +		/* Assume it's an FT232H */ +		priv->chip_type = FT232H; +	} else { +		/* Assume it's an FT-X series device */ +		priv->chip_type = FTX;  	} +  	dev_info(&udev->dev, "Detected %s\n", ftdi_chip_name[priv->chip_type]);  } @@ -1399,14 +1569,17 @@ static void ftdi_set_max_packet_size(struct usb_serial_port *port)  	struct usb_device *udev = serial->dev;  	struct usb_interface *interface = serial->interface; -	struct usb_endpoint_descriptor *ep_desc = &interface->cur_altsetting->endpoint[1].desc; +	struct usb_endpoint_descriptor *ep_desc;  	unsigned num_endpoints; -	int i; +	unsigned i;  	num_endpoints = interface->cur_altsetting->desc.bNumEndpoints;  	dev_info(&udev->dev, "Number of endpoints %d\n", num_endpoints); +	if (!num_endpoints) +		return; +  	/* NOTE: some customers have programmed FT232R/FT245R devices  	 * with an endpoint size of 0 - not good.  In this case, we  	 * want to override the endpoint descriptor setting and use a @@ -1422,7 +1595,7 @@ static void ftdi_set_max_packet_size(struct usb_serial_port *port)  	}  	/* set max packet size based on descriptor */ -	priv->max_packet_size = le16_to_cpu(ep_desc->wMaxPacketSize); +	priv->max_packet_size = usb_endpoint_maxp(ep_desc);  	dev_info(&udev->dev, "Setting MaxPacketSize %d\n", priv->max_packet_size);  } @@ -1434,8 +1607,8 @@ static void ftdi_set_max_packet_size(struct usb_serial_port *port)   * ***************************************************************************   */ -static ssize_t show_latency_timer(struct device *dev, -				struct device_attribute *attr, char *buf) +static ssize_t latency_timer_show(struct device *dev, +				  struct device_attribute *attr, char *buf)  {  	struct usb_serial_port *port = to_usb_serial_port(dev);  	struct ftdi_private *priv = usb_get_serial_port_data(port); @@ -1445,11 +1618,10 @@ static ssize_t show_latency_timer(struct device *dev,  		return sprintf(buf, "%i\n", priv->latency);  } -  /* Write a new value of the latency timer, in units of milliseconds. */ -static ssize_t store_latency_timer(struct device *dev, -			struct device_attribute *attr, const char *valbuf, -			size_t count) +static ssize_t latency_timer_store(struct device *dev, +				   struct device_attribute *attr, +				   const char *valbuf, size_t count)  {  	struct usb_serial_port *port = to_usb_serial_port(dev);  	struct ftdi_private *priv = usb_get_serial_port_data(port); @@ -1462,6 +1634,7 @@ static ssize_t store_latency_timer(struct device *dev,  		return -EIO;  	return count;  } +static DEVICE_ATTR_RW(latency_timer);  /* Write an event character directly to the FTDI register.  The ASCII     value is in the low 8 bits, with the enable bit in the 9th bit. */ @@ -1474,7 +1647,7 @@ static ssize_t store_event_char(struct device *dev,  	int v = simple_strtoul(valbuf, NULL, 10);  	int rv; -	dbg("%s: setting event char = %i", __func__, v); +	dev_dbg(&port->dev, "%s: setting event char = %i\n", __func__, v);  	rv = usb_control_msg(udev,  			     usb_sndctrlpipe(udev, 0), @@ -1483,15 +1656,12 @@ static ssize_t store_event_char(struct device *dev,  			     v, priv->interface,  			     NULL, 0, WDR_TIMEOUT);  	if (rv < 0) { -		dbg("Unable to write event character: %i", rv); +		dev_dbg(&port->dev, "Unable to write event character: %i\n", rv);  		return -EIO;  	}  	return count;  } - -static DEVICE_ATTR(latency_timer, S_IWUSR | S_IRUGO, show_latency_timer, -							store_latency_timer);  static DEVICE_ATTR(event_char, S_IWUSR, NULL, store_event_char);  static int create_sysfs_attrs(struct usb_serial_port *port) @@ -1499,19 +1669,19 @@ static int create_sysfs_attrs(struct usb_serial_port *port)  	struct ftdi_private *priv = usb_get_serial_port_data(port);  	int retval = 0; -	dbg("%s", __func__); -  	/* XXX I've no idea if the original SIO supports the event_char  	 * sysfs parameter, so I'm playing it safe.  */  	if (priv->chip_type != SIO) { -		dbg("sysfs attributes for %s", ftdi_chip_name[priv->chip_type]); +		dev_dbg(&port->dev, "sysfs attributes for %s\n", ftdi_chip_name[priv->chip_type]);  		retval = device_create_file(&port->dev, &dev_attr_event_char);  		if ((!retval) &&  		    (priv->chip_type == FT232BM ||  		     priv->chip_type == FT2232C ||  		     priv->chip_type == FT232RL ||  		     priv->chip_type == FT2232H || -		     priv->chip_type == FT4232H)) { +		     priv->chip_type == FT4232H || +		     priv->chip_type == FT232H || +		     priv->chip_type == FTX)) {  			retval = device_create_file(&port->dev,  						    &dev_attr_latency_timer);  		} @@ -1523,8 +1693,6 @@ static void remove_sysfs_attrs(struct usb_serial_port *port)  {  	struct ftdi_private *priv = usb_get_serial_port_data(port); -	dbg("%s", __func__); -  	/* XXX see create_sysfs_attrs */  	if (priv->chip_type != SIO) {  		device_remove_file(&port->dev, &dev_attr_event_char); @@ -1532,7 +1700,9 @@ static void remove_sysfs_attrs(struct usb_serial_port *port)  		    priv->chip_type == FT2232C ||  		    priv->chip_type == FT232RL ||  		    priv->chip_type == FT2232H || -		    priv->chip_type == FT4232H) { +		    priv->chip_type == FT4232H || +		    priv->chip_type == FT232H || +		    priv->chip_type == FTX) {  			device_remove_file(&port->dev, &dev_attr_latency_timer);  		}  	} @@ -1569,25 +1739,17 @@ static int ftdi_sio_port_probe(struct usb_serial_port *port)  	struct ftdi_sio_quirk *quirk = usb_get_serial_data(port->serial); -	dbg("%s", __func__); -  	priv = kzalloc(sizeof(struct ftdi_private), GFP_KERNEL); -	if (!priv) { -		dev_err(&port->dev, "%s- kmalloc(%Zd) failed.\n", __func__, -					sizeof(struct ftdi_private)); +	if (!priv)  		return -ENOMEM; -	} -	kref_init(&priv->kref);  	mutex_init(&priv->cfg_lock); -	init_waitqueue_head(&priv->delta_msr_wait);  	priv->flags = ASYNC_LOW_LATENCY;  	if (quirk && quirk->port_probe)  		quirk->port_probe(priv); -	priv->port = port;  	usb_set_serial_port_data(port, priv);  	ftdi_determine_type(port); @@ -1604,8 +1766,6 @@ static int ftdi_sio_port_probe(struct usb_serial_port *port)  /* Called from usbserial:serial_probe */  static void ftdi_USB_UIRT_setup(struct ftdi_private *priv)  { -	dbg("%s", __func__); -  	priv->flags |= ASYNC_SPD_CUST;  	priv->custom_divisor = 77;  	priv->force_baud = 38400; @@ -1616,8 +1776,6 @@ static void ftdi_USB_UIRT_setup(struct ftdi_private *priv)  static void ftdi_HE_TIRA1_setup(struct ftdi_private *priv)  { -	dbg("%s", __func__); -  	priv->flags |= ASYNC_SPD_CUST;  	priv->custom_divisor = 240;  	priv->force_baud = 38400; @@ -1626,7 +1784,8 @@ static void ftdi_HE_TIRA1_setup(struct ftdi_private *priv)  /*   * Module parameter to control latency timer for NDI FTDI-based USB devices. - * If this value is not set in modprobe.conf.local its value will be set to 1ms. + * If this value is not set in /etc/modprobe.d/ its value will be set + * to 1ms.   */  static int ndi_latency_timer = 1; @@ -1645,8 +1804,8 @@ static int ftdi_NDI_device_setup(struct usb_serial *serial)  	if (latency > 99)  		latency = 99; -	dbg("%s setting NDI device latency to %d", __func__, latency); -	dev_info(&udev->dev, "NDI device with a latency value of %d", latency); +	dev_dbg(&udev->dev, "%s setting NDI device latency to %d\n", __func__, latency); +	dev_info(&udev->dev, "NDI device with a latency value of %d\n", latency);  	/* FIXME: errors are not returned */  	usb_control_msg(udev, usb_sndctrlpipe(udev, 0), @@ -1666,8 +1825,6 @@ static int ftdi_jtag_probe(struct usb_serial *serial)  	struct usb_device *udev = serial->dev;  	struct usb_interface *interface = serial->interface; -	dbg("%s", __func__); -  	if (interface == udev->actconfig->interface[0]) {  		dev_info(&udev->dev,  			 "Ignoring serial port reserved for JTAG\n"); @@ -1677,6 +1834,38 @@ static int ftdi_jtag_probe(struct usb_serial *serial)  	return 0;  } +static int ftdi_8u2232c_probe(struct usb_serial *serial) +{ +	struct usb_device *udev = serial->dev; + +	if ((udev->manufacturer && !strcmp(udev->manufacturer, "CALAO Systems")) || +	    (udev->product && !strcmp(udev->product, "BeagleBone/XDS100V2"))) +		return ftdi_jtag_probe(serial); + +	return 0; +} + +/* + * First two ports on JTAG adaptors using an FT4232 such as STMicroelectronics's + * ST Micro Connect Lite are reserved for JTAG or other non-UART interfaces and + * can be accessed from userspace. + * The next two ports are enabled as UARTs by default, where port 2 is + * a conventional RS-232 UART. + */ +static int ftdi_stmclite_probe(struct usb_serial *serial) +{ +	struct usb_device *udev = serial->dev; +	struct usb_interface *interface = serial->interface; + +	if (interface == udev->actconfig->interface[0] || +	    interface == udev->actconfig->interface[1]) { +		dev_info(&udev->dev, "Ignoring serial port reserved for JTAG\n"); +		return -ENODEV; +	} + +	return 0; +} +  /*   * The Matrix Orbital VK204-25-USB has an invalid IN endpoint.   * We have to correct it if we want to read from it. @@ -1695,22 +1884,13 @@ static int ftdi_mtxorb_hack_setup(struct usb_serial *serial)  	return 0;  } -static void ftdi_sio_priv_release(struct kref *k) -{ -	struct ftdi_private *priv = container_of(k, struct ftdi_private, kref); - -	kfree(priv); -} -  static int ftdi_sio_port_remove(struct usb_serial_port *port)  {  	struct ftdi_private *priv = usb_get_serial_port_data(port); -	dbg("%s", __func__); -  	remove_sysfs_attrs(port); -	kref_put(&priv->kref, ftdi_sio_priv_release); +	kfree(priv);  	return 0;  } @@ -1719,9 +1899,6 @@ static int ftdi_open(struct tty_struct *tty, struct usb_serial_port *port)  {  	struct usb_device *dev = port->serial->dev;  	struct ftdi_private *priv = usb_get_serial_port_data(port); -	int result; - -	dbg("%s", __func__);  	/* No error checking for this (will get errors later anyway) */  	/* See ftdi_sio.h for description of what is reset */ @@ -1736,53 +1913,31 @@ static int ftdi_open(struct tty_struct *tty, struct usb_serial_port *port)  	/* ftdi_set_termios  will send usb control messages */  	if (tty) -		ftdi_set_termios(tty, port, tty->termios); +		ftdi_set_termios(tty, port, NULL); -	/* Start reading from the device */ -	result = usb_serial_generic_open(tty, port); -	if (!result) -		kref_get(&priv->kref); - -	return result; +	return usb_serial_generic_open(tty, port);  }  static void ftdi_dtr_rts(struct usb_serial_port *port, int on)  {  	struct ftdi_private *priv = usb_get_serial_port_data(port); -	mutex_lock(&port->serial->disc_mutex); -	if (!port->serial->disconnected) { -		/* Disable flow control */ -		if (!on && usb_control_msg(port->serial->dev, +	/* Disable flow control */ +	if (!on) { +		if (usb_control_msg(port->serial->dev,  			    usb_sndctrlpipe(port->serial->dev, 0),  			    FTDI_SIO_SET_FLOW_CTRL_REQUEST,  			    FTDI_SIO_SET_FLOW_CTRL_REQUEST_TYPE,  			    0, priv->interface, NULL, 0,  			    WDR_TIMEOUT) < 0) { -			    dev_err(&port->dev, "error from flowcontrol urb\n"); +			dev_err(&port->dev, "error from flowcontrol urb\n");  		} -		/* drop RTS and DTR */ -		if (on) -			set_mctrl(port, TIOCM_DTR | TIOCM_RTS); -		else -			clear_mctrl(port, TIOCM_DTR | TIOCM_RTS);  	} -	mutex_unlock(&port->serial->disc_mutex); -} - -/* - * usbserial:__serial_close  only calls ftdi_close if the point is open - * - *   This only gets called when it is the last close - */ -static void ftdi_close(struct usb_serial_port *port) -{ -	struct ftdi_private *priv = usb_get_serial_port_data(port); - -	dbg("%s", __func__); - -	usb_serial_generic_close(port); -	kref_put(&priv->kref, ftdi_sio_priv_release); +	/* drop RTS and DTR */ +	if (on) +		set_mctrl(port, TIOCM_DTR | TIOCM_RTS); +	else +		clear_mctrl(port, TIOCM_DTR | TIOCM_RTS);  }  /* The SIO requires the first byte to have: @@ -1812,6 +1967,7 @@ static int ftdi_prepare_write_buffer(struct usb_serial_port *port,  			c = kfifo_out(&port->write_fifo, &buffer[i + 1], len);  			if (!c)  				break; +			port->icount.tx += c;  			buffer[i] = (c << 2) + 1;  			count += c + 1;  		} @@ -1819,6 +1975,7 @@ static int ftdi_prepare_write_buffer(struct usb_serial_port *port,  	} else {  		count = kfifo_out_locked(&port->write_fifo, dest, size,  								&port->lock); +		port->icount.tx += count;  	}  	return count; @@ -1826,19 +1983,16 @@ static int ftdi_prepare_write_buffer(struct usb_serial_port *port,  #define FTDI_RS_ERR_MASK (FTDI_RS_BI | FTDI_RS_PE | FTDI_RS_FE | FTDI_RS_OE) -static int ftdi_process_packet(struct tty_struct *tty, -		struct usb_serial_port *port, struct ftdi_private *priv, -		char *packet, int len) +static int ftdi_process_packet(struct usb_serial_port *port, +		struct ftdi_private *priv, char *packet, int len)  {  	int i;  	char status;  	char flag;  	char *ch; -	dbg("%s - port %d", __func__, port->number); -  	if (len < 2) { -		dbg("malformed packet"); +		dev_dbg(&port->dev, "malformed packet\n");  		return 0;  	} @@ -1847,8 +2001,26 @@ static int ftdi_process_packet(struct tty_struct *tty,  	   are only processed once.  */  	status = packet[0] & FTDI_STATUS_B0_MASK;  	if (status != priv->prev_status) { -		priv->diff_status |= status ^ priv->prev_status; -		wake_up_interruptible(&priv->delta_msr_wait); +		char diff_status = status ^ priv->prev_status; + +		if (diff_status & FTDI_RS0_CTS) +			port->icount.cts++; +		if (diff_status & FTDI_RS0_DSR) +			port->icount.dsr++; +		if (diff_status & FTDI_RS0_RI) +			port->icount.rng++; +		if (diff_status & FTDI_RS0_RLSD) { +			struct tty_struct *tty; + +			port->icount.dcd++; +			tty = tty_port_tty_get(&port->port); +			if (tty) +				usb_serial_handle_dcd_change(port, tty, +						status & FTDI_RS0_RLSD); +			tty_kref_put(tty); +		} + +		wake_up_interruptible(&port->port.delta_msr_wait);  		priv->prev_status = status;  	} @@ -1858,29 +2030,41 @@ static int ftdi_process_packet(struct tty_struct *tty,  		 * over framing errors */  		if (packet[1] & FTDI_RS_BI) {  			flag = TTY_BREAK; +			port->icount.brk++;  			usb_serial_handle_break(port);  		} else if (packet[1] & FTDI_RS_PE) {  			flag = TTY_PARITY; +			port->icount.parity++;  		} else if (packet[1] & FTDI_RS_FE) {  			flag = TTY_FRAME; +			port->icount.frame++;  		}  		/* Overrun is special, not associated with a char */ -		if (packet[1] & FTDI_RS_OE) -			tty_insert_flip_char(tty, 0, TTY_OVERRUN); +		if (packet[1] & FTDI_RS_OE) { +			port->icount.overrun++; +			tty_insert_flip_char(&port->port, 0, TTY_OVERRUN); +		}  	} +	/* save if the transmitter is empty or not */ +	if (packet[1] & FTDI_RS_TEMT) +		priv->transmit_empty = 1; +	else +		priv->transmit_empty = 0; +  	len -= 2;  	if (!len)  		return 0;	/* status only */ +	port->icount.rx += len;  	ch = packet + 2;  	if (port->port.console && port->sysrq) {  		for (i = 0; i < len; i++, ch++) {  			if (!usb_serial_handle_sysrq_char(port, *ch)) -				tty_insert_flip_char(tty, *ch, flag); +				tty_insert_flip_char(&port->port, *ch, flag);  		}  	} else { -		tty_insert_flip_string_fixed_flag(tty, ch, flag, len); +		tty_insert_flip_string_fixed_flag(&port->port, ch, flag, len);  	}  	return len; @@ -1889,25 +2073,19 @@ static int ftdi_process_packet(struct tty_struct *tty,  static void ftdi_process_read_urb(struct urb *urb)  {  	struct usb_serial_port *port = urb->context; -	struct tty_struct *tty;  	struct ftdi_private *priv = usb_get_serial_port_data(port);  	char *data = (char *)urb->transfer_buffer;  	int i;  	int len;  	int count = 0; -	tty = tty_port_tty_get(&port->port); -	if (!tty) -		return; -  	for (i = 0; i < urb->actual_length; i += priv->max_packet_size) {  		len = min_t(int, urb->actual_length - i, priv->max_packet_size); -		count += ftdi_process_packet(tty, port, priv, &data[i], len); +		count += ftdi_process_packet(port, priv, &data[i], len);  	}  	if (count) -		tty_flip_buffer_push(tty); -	tty_kref_put(tty); +		tty_flip_buffer_push(&port->port);  }  static void ftdi_break_ctl(struct tty_struct *tty, int break_state) @@ -1931,15 +2109,29 @@ static void ftdi_break_ctl(struct tty_struct *tty, int break_state)  			FTDI_SIO_SET_DATA_REQUEST_TYPE,  			urb_value , priv->interface,  			NULL, 0, WDR_TIMEOUT) < 0) { -		dev_err(&port->dev, "%s FAILED to enable/disable break state " -			"(state was %d)\n", __func__, break_state); +		dev_err(&port->dev, "%s FAILED to enable/disable break state (state was %d)\n", +			__func__, break_state);  	} -	dbg("%s break state is %d - urb is %d", __func__, -						break_state, urb_value); +	dev_dbg(&port->dev, "%s break state is %d - urb is %d\n", __func__, +		break_state, urb_value);  } +static bool ftdi_tx_empty(struct usb_serial_port *port) +{ +	unsigned char buf[2]; +	int ret; + +	ret = ftdi_get_modem_status(port, buf); +	if (ret == 2) { +		if (!(buf[1] & FTDI_RS_TEMT)) +			return false; +	} + +	return true; +} +  /* old_termios contains the original termios settings and tty->termios contains   * the new setting to be used   * WARNING: set_termios calls this with old_termios in kernel space @@ -1948,8 +2140,9 @@ static void ftdi_set_termios(struct tty_struct *tty,  		struct usb_serial_port *port, struct ktermios *old_termios)  {  	struct usb_device *dev = port->serial->dev; +	struct device *ddev = &port->dev;  	struct ftdi_private *priv = usb_get_serial_port_data(port); -	struct ktermios *termios = tty->termios; +	struct ktermios *termios = &tty->termios;  	unsigned int cflag = termios->c_cflag;  	__u16 urb_value; /* will hold the new flags */ @@ -1958,31 +2151,63 @@ static void ftdi_set_termios(struct tty_struct *tty,  	unsigned char vstop;  	unsigned char vstart; -	dbg("%s", __func__); -  	/* Force baud rate if this device requires it, unless it is set to  	   B0. */  	if (priv->force_baud && ((termios->c_cflag & CBAUD) != B0)) { -		dbg("%s: forcing baud rate for this device", __func__); +		dev_dbg(ddev, "%s: forcing baud rate for this device\n", __func__);  		tty_encode_baud_rate(tty, priv->force_baud,  					priv->force_baud);  	}  	/* Force RTS-CTS if this device requires it. */  	if (priv->force_rtscts) { -		dbg("%s: forcing rtscts for this device", __func__); +		dev_dbg(ddev, "%s: forcing rtscts for this device\n", __func__);  		termios->c_cflag |= CRTSCTS;  	} +	/* +	 * All FTDI UART chips are limited to CS7/8. We shouldn't pretend to +	 * support CS5/6 and revert the CSIZE setting instead. +	 * +	 * CS5 however is used to control some smartcard readers which abuse +	 * this limitation to switch modes. Original FTDI chips fall back to +	 * eight data bits. +	 * +	 * TODO: Implement a quirk to only allow this with mentioned +	 *       readers. One I know of (Argolis Smartreader V1) +	 *       returns "USB smartcard server" as iInterface string. +	 *       The vendor didn't bother with a custom VID/PID of +	 *       course. +	 */ +	if (C_CSIZE(tty) == CS6) { +		dev_warn(ddev, "requested CSIZE setting not supported\n"); + +		termios->c_cflag &= ~CSIZE; +		if (old_termios) +			termios->c_cflag |= old_termios->c_cflag & CSIZE; +		else +			termios->c_cflag |= CS8; +	} +  	cflag = termios->c_cflag; -	/* FIXME -For this cut I don't care if the line is really changing or -	   not  - so just do the change regardless  - should be able to -	   compare old_termios and tty->termios */ +	if (!old_termios) +		goto no_skip; + +	if (old_termios->c_cflag == termios->c_cflag +	    && old_termios->c_ispeed == termios->c_ispeed +	    && old_termios->c_ospeed == termios->c_ospeed) +		goto no_c_cflag_changes; +  	/* NOTE These routines can get interrupted by  	   ftdi_sio_read_bulk_callback  - need to examine what this means -  	   don't see any problems yet */ +	if ((old_termios->c_cflag & (CSIZE|PARODD|PARENB|CMSPAR|CSTOPB)) == +	    (termios->c_cflag & (CSIZE|PARODD|PARENB|CMSPAR|CSTOPB))) +		goto no_data_parity_stop_changes; + +no_skip:  	/* Set number of data bits, parity, stop bits */  	urb_value = 0; @@ -2000,13 +2225,19 @@ static void ftdi_set_termios(struct tty_struct *tty,  	} else {  		urb_value |= FTDI_SIO_SET_DATA_PARITY_NONE;  	} -	if (cflag & CSIZE) { -		switch (cflag & CSIZE) { -		case CS7: urb_value |= 7; dbg("Setting CS7"); break; -		case CS8: urb_value |= 8; dbg("Setting CS8"); break; -		default: -			dev_err(&port->dev, "CSIZE was set but not CS7-CS8\n"); -		} +	switch (cflag & CSIZE) { +	case CS5: +		dev_dbg(ddev, "Setting CS5 quirk\n"); +		break; +	case CS7: +		urb_value |= 7; +		dev_dbg(ddev, "Setting CS7\n"); +		break; +	default: +	case CS8: +		urb_value |= 8; +		dev_dbg(ddev, "Setting CS8\n"); +		break;  	}  	/* This is needed by the break command since it uses the same command @@ -2018,11 +2249,12 @@ static void ftdi_set_termios(struct tty_struct *tty,  			    FTDI_SIO_SET_DATA_REQUEST_TYPE,  			    urb_value , priv->interface,  			    NULL, 0, WDR_SHORT_TIMEOUT) < 0) { -		dev_err(&port->dev, "%s FAILED to set " -			"databits/stopbits/parity\n", __func__); +		dev_err(ddev, "%s FAILED to set databits/stopbits/parity\n", +			__func__);  	}  	/* Now do the baudrate */ +no_data_parity_stop_changes:  	if ((cflag & CBAUD) == B0) {  		/* Disable flow control */  		if (usb_control_msg(dev, usb_sndctrlpipe(dev, 0), @@ -2030,8 +2262,7 @@ static void ftdi_set_termios(struct tty_struct *tty,  				    FTDI_SIO_SET_FLOW_CTRL_REQUEST_TYPE,  				    0, priv->interface,  				    NULL, 0, WDR_TIMEOUT) < 0) { -			dev_err(&port->dev, -				"%s error from disable flowcontrol urb\n", +			dev_err(ddev, "%s error from disable flowcontrol urb\n",  				__func__);  		}  		/* Drop RTS and DTR */ @@ -2040,28 +2271,26 @@ static void ftdi_set_termios(struct tty_struct *tty,  		/* set the baudrate determined before */  		mutex_lock(&priv->cfg_lock);  		if (change_speed(tty, port)) -			dev_err(&port->dev, "%s urb failed to set baudrate\n", -				__func__); +			dev_err(ddev, "%s urb failed to set baudrate\n", __func__);  		mutex_unlock(&priv->cfg_lock);  		/* Ensure RTS and DTR are raised when baudrate changed from 0 */ -		if (!old_termios || (old_termios->c_cflag & CBAUD) == B0) +		if (old_termios && (old_termios->c_cflag & CBAUD) == B0)  			set_mctrl(port, TIOCM_DTR | TIOCM_RTS);  	}  	/* Set flow control */  	/* Note device also supports DTR/CD (ugh) and Xon/Xoff in hardware */ +no_c_cflag_changes:  	if (cflag & CRTSCTS) { -		dbg("%s Setting to CRTSCTS flow control", __func__); +		dev_dbg(ddev, "%s Setting to CRTSCTS flow control\n", __func__);  		if (usb_control_msg(dev,  				    usb_sndctrlpipe(dev, 0),  				    FTDI_SIO_SET_FLOW_CTRL_REQUEST,  				    FTDI_SIO_SET_FLOW_CTRL_REQUEST_TYPE,  				    0 , (FTDI_SIO_RTS_CTS_HS | priv->interface),  				    NULL, 0, WDR_TIMEOUT) < 0) { -			dev_err(&port->dev, -				"urb failed to set to rts/cts flow control\n"); +			dev_err(ddev, "urb failed to set to rts/cts flow control\n");  		} -  	} else {  		/*  		 * Xon/Xoff code @@ -2071,8 +2300,8 @@ static void ftdi_set_termios(struct tty_struct *tty,  		 * code is executed.  		 */  		if (iflag & IXOFF) { -			dbg("%s  request to enable xonxoff iflag=%04x", -							__func__, iflag); +			dev_dbg(ddev, "%s  request to enable xonxoff iflag=%04x\n", +				__func__, iflag);  			/* Try to enable the XON/XOFF on the ftdi_sio  			 * Set the vstart and vstop -- could have been done up  			 * above where a lot of other dereferencing is done but @@ -2097,31 +2326,33 @@ static void ftdi_set_termios(struct tty_struct *tty,  			/* else clause to only run if cflag ! CRTSCTS and iflag  			 * ! XOFF. CHECKME Assuming XON/XOFF handled by tty  			 * stack - not by device */ -			dbg("%s Turning off hardware flow control", __func__); +			dev_dbg(ddev, "%s Turning off hardware flow control\n", __func__);  			if (usb_control_msg(dev,  					    usb_sndctrlpipe(dev, 0),  					    FTDI_SIO_SET_FLOW_CTRL_REQUEST,  					    FTDI_SIO_SET_FLOW_CTRL_REQUEST_TYPE,  					    0, priv->interface,  					    NULL, 0, WDR_TIMEOUT) < 0) { -				dev_err(&port->dev, -					"urb failed to clear flow control\n"); +				dev_err(ddev, "urb failed to clear flow control\n");  			}  		} -  	}  } -static int ftdi_tiocmget(struct tty_struct *tty, struct file *file) +/* + * Get modem-control status. + * + * Returns the number of status bytes retrieved (device dependant), or + * negative error code. + */ +static int ftdi_get_modem_status(struct usb_serial_port *port, +						unsigned char status[2])  { -	struct usb_serial_port *port = tty->driver_data;  	struct ftdi_private *priv = usb_get_serial_port_data(port);  	unsigned char *buf;  	int len;  	int ret; -	dbg("%s TIOCMGET", __func__); -  	buf = kmalloc(2, GFP_KERNEL);  	if (!buf)  		return -ENOMEM; @@ -2139,6 +2370,8 @@ static int ftdi_tiocmget(struct tty_struct *tty, struct file *file)  	case FT232RL:  	case FT2232H:  	case FT4232H: +	case FT232H: +	case FTX:  		len = 2;  		break;  	default: @@ -2152,34 +2385,58 @@ static int ftdi_tiocmget(struct tty_struct *tty, struct file *file)  			FTDI_SIO_GET_MODEM_STATUS_REQUEST_TYPE,  			0, priv->interface,  			buf, len, WDR_TIMEOUT); -	if (ret < 0) +	if (ret < 0) { +		dev_err(&port->dev, "failed to get modem status: %d\n", ret); +		ret = usb_translate_errors(ret);  		goto out; +	} -	ret = (buf[0] & FTDI_SIO_DSR_MASK ? TIOCM_DSR : 0) | -		(buf[0] & FTDI_SIO_CTS_MASK ? TIOCM_CTS : 0) | -		(buf[0]  & FTDI_SIO_RI_MASK  ? TIOCM_RI  : 0) | -		(buf[0]  & FTDI_SIO_RLSD_MASK ? TIOCM_CD  : 0) | -		priv->last_dtr_rts; +	status[0] = buf[0]; +	if (ret > 1) +		status[1] = buf[1]; +	else +		status[1] = 0; + +	dev_dbg(&port->dev, "%s - 0x%02x%02x\n", __func__, status[0], +								status[1]);  out:  	kfree(buf); + +	return ret; +} + +static int ftdi_tiocmget(struct tty_struct *tty) +{ +	struct usb_serial_port *port = tty->driver_data; +	struct ftdi_private *priv = usb_get_serial_port_data(port); +	unsigned char buf[2]; +	int ret; + +	ret = ftdi_get_modem_status(port, buf); +	if (ret < 0) +		return ret; + +	ret =	(buf[0] & FTDI_SIO_DSR_MASK  ? TIOCM_DSR : 0) | +		(buf[0] & FTDI_SIO_CTS_MASK  ? TIOCM_CTS : 0) | +		(buf[0] & FTDI_SIO_RI_MASK   ? TIOCM_RI  : 0) | +		(buf[0] & FTDI_SIO_RLSD_MASK ? TIOCM_CD  : 0) | +		priv->last_dtr_rts; +  	return ret;  } -static int ftdi_tiocmset(struct tty_struct *tty, struct file *file, +static int ftdi_tiocmset(struct tty_struct *tty,  			unsigned int set, unsigned int clear)  {  	struct usb_serial_port *port = tty->driver_data; -	dbg("%s TIOCMSET", __func__); +  	return update_mctrl(port, set, clear);  } -static int ftdi_ioctl(struct tty_struct *tty, struct file *file, +static int ftdi_ioctl(struct tty_struct *tty,  					unsigned int cmd, unsigned long arg)  {  	struct usb_serial_port *port = tty->driver_data; -	struct ftdi_private *priv = usb_get_serial_port_data(port); - -	dbg("%s cmd 0x%04x", __func__, cmd);  	/* Based on code from acm.c and others */  	switch (cmd) { @@ -2191,110 +2448,21 @@ static int ftdi_ioctl(struct tty_struct *tty, struct file *file,  	case TIOCSSERIAL: /* sets serial port data */  		return set_serial_info(tty, port,  					(struct serial_struct __user *) arg); - -	/* -	 * Wait for any of the 4 modem inputs (DCD,RI,DSR,CTS) to change -	 * - mask passed in arg for lines of interest -	 *   (use |'ed TIOCM_RNG/DSR/CD/CTS for masking) -	 * Caller should use TIOCGICOUNT to see which one it was. -	 * (except that the driver doesn't support it !) -	 * -	 * This code is borrowed from linux/drivers/char/serial.c -	 */ -	case TIOCMIWAIT: -		while (priv != NULL) { -			interruptible_sleep_on(&priv->delta_msr_wait); -			/* see if a signal did it */ -			if (signal_pending(current)) -				return -ERESTARTSYS; -			else { -				char diff = priv->diff_status; - -				if (diff == 0) -					return -EIO; /* no change => error */ - -				/* Consume all events */ -				priv->diff_status = 0; - -				/* Return 0 if caller wanted to know about -				   these bits */ -				if (((arg & TIOCM_RNG) && (diff & FTDI_RS0_RI)) || -				    ((arg & TIOCM_DSR) && (diff & FTDI_RS0_DSR)) || -				    ((arg & TIOCM_CD)  && (diff & FTDI_RS0_RLSD)) || -				    ((arg & TIOCM_CTS) && (diff & FTDI_RS0_CTS))) { -					return 0; -				} -				/* -				 * Otherwise caller can't care less about what -				 * happened,and so we continue to wait for more -				 * events. -				 */ -			} -		} -		return 0; +	case TIOCSERGETLSR: +		return get_lsr_info(port, (struct serial_struct __user *)arg); +		break;  	default:  		break;  	} -	/* This is not necessarily an error - turns out the higher layers -	 * will do some ioctls themselves (see comment above) -	 */ -	dbg("%s arg not supported - it was 0x%04x - check /usr/include/asm/ioctls.h", __func__, cmd); -	return -ENOIOCTLCMD; -} -static int __init ftdi_init(void) -{ -	int retval; - -	dbg("%s", __func__); -	if (vendor > 0 && product > 0) { -		/* Add user specified VID/PID to reserved element of table. */ -		int i; -		for (i = 0; id_table_combined[i].idVendor; i++) -			; -		id_table_combined[i].match_flags = USB_DEVICE_ID_MATCH_DEVICE; -		id_table_combined[i].idVendor = vendor; -		id_table_combined[i].idProduct = product; -	} -	retval = usb_serial_register(&ftdi_sio_device); -	if (retval) -		goto failed_sio_register; -	retval = usb_register(&ftdi_driver); -	if (retval) -		goto failed_usb_register; - -	printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_VERSION ":" -	       DRIVER_DESC "\n"); -	return 0; -failed_usb_register: -	usb_serial_deregister(&ftdi_sio_device); -failed_sio_register: -	return retval; -} - -static void __exit ftdi_exit(void) -{ -	dbg("%s", __func__); - -	usb_deregister(&ftdi_driver); -	usb_serial_deregister(&ftdi_sio_device); +	return -ENOIOCTLCMD;  } - -module_init(ftdi_init); -module_exit(ftdi_exit); +module_usb_serial_driver(serial_drivers, id_table_combined);  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(vendor, ushort, 0); -MODULE_PARM_DESC(vendor, "User specified vendor ID (default=" -		__MODULE_STRING(FTDI_VID)")"); -module_param(product, ushort, 0); -MODULE_PARM_DESC(product, "User specified product ID"); -  module_param(ndi_latency_timer, int, S_IRUGO | S_IWUSR);  MODULE_PARM_DESC(ndi_latency_timer, "NDI device latency timer override");  | 
