2005-04-16 15:20:36 -07:00
/*
* KLSI KL5KUSB105 chip RS232 converter driver
*
2010-05-19 00:01:40 +02:00
* Copyright ( C ) 2010 Johan Hovold < jhovold @ gmail . com >
2005-04-16 15:20:36 -07:00
* Copyright ( C ) 2001 Utz - Uwe Haus < haus @ uuhaus . de >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* All information about the device was acquired using SniffUSB ans snoopUSB
* on Windows98 .
* It was written out of frustration with the PalmConnect USB Serial adapter
* sold by Palm Inc .
* Neither Palm , nor their contractor ( MCCI ) or their supplier ( KLSI ) provided
* information that was not already available .
*
2008-07-22 11:14:00 +01:00
* It seems that KLSI bought some silicon - design information from ScanLogic ,
2005-04-16 15:20:36 -07:00
* whose SL11R processor is at the core of the KL5KUSB chipset from KLSI .
* KLSI has firmware available for their devices ; it is probable that the
* firmware differs from that used by KLSI in their products . If you have an
2008-07-22 11:14:00 +01:00
* original KLSI device and can provide some information on it , I would be
* most interested in adding support for it here . If you have any information
2005-04-16 15:20:36 -07:00
* on the protocol used ( or find errors in my reverse - engineered stuff ) , please
* let me know .
*
* The code was only tested with a PalmConnect USB adapter ; if you
* are adventurous , try it with any KLSI - based device and let me know how it
* breaks so that I can fix it !
*/
/* TODO:
* check modem line signals
* implement handshaking or decide that we do not support it
*/
# 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>
# include <linux/tty_flip.h>
# include <linux/module.h>
2008-07-22 11:14:00 +01:00
# include <linux/uaccess.h>
2008-04-28 07:00:16 +01:00
# include <asm/unaligned.h>
2005-04-16 15:20:36 -07:00
# include <linux/usb.h>
2006-07-11 21:22:58 -07:00
# include <linux/usb/serial.h>
2005-04-16 15:20:36 -07:00
# include "kl5kusb105.h"
static int debug ;
/*
* Version Information
*/
2010-05-19 00:01:40 +02:00
# define DRIVER_VERSION "v0.4"
# define DRIVER_AUTHOR "Utz-Uwe Haus <haus@uuhaus.de>, Johan Hovold <jhovold@gmail.com>"
2005-04-16 15:20:36 -07:00
# define DRIVER_DESC "KLSI KL5KUSB105 chipset USB->Serial Converter driver"
/*
* Function prototypes
*/
2008-07-22 11:14:00 +01:00
static int klsi_105_startup ( struct usb_serial * serial ) ;
2009-06-02 11:53:55 -04:00
static void klsi_105_release ( struct usb_serial * serial ) ;
2009-09-19 13:13:26 -07:00
static int klsi_105_open ( struct tty_struct * tty , struct usb_serial_port * port ) ;
2009-06-11 12:26:29 +01:00
static void klsi_105_close ( struct usb_serial_port * port ) ;
2008-07-22 11:14:00 +01:00
static void klsi_105_set_termios ( struct tty_struct * tty ,
struct usb_serial_port * port , struct ktermios * old ) ;
static int klsi_105_tiocmget ( struct tty_struct * tty , struct file * file ) ;
static int klsi_105_tiocmset ( struct tty_struct * tty , struct file * file ,
unsigned int set , unsigned int clear ) ;
2010-05-19 00:01:40 +02:00
static void klsi_105_process_read_urb ( struct urb * urb ) ;
static int klsi_105_prepare_write_buffer ( struct usb_serial_port * port ,
void * dest , size_t size ) ;
2005-04-16 15:20:36 -07:00
/*
* All of the device info needed for the KLSI converters .
*/
2010-01-10 15:34:24 +01:00
static const struct usb_device_id id_table [ ] = {
2005-04-16 15:20:36 -07:00
{ USB_DEVICE ( PALMCONNECT_VID , PALMCONNECT_PID ) } ,
{ USB_DEVICE ( KLSI_VID , KLSI_KL5KUSB105D_PID ) } ,
{ } /* Terminating entry */
} ;
2008-07-22 11:14:00 +01:00
MODULE_DEVICE_TABLE ( usb , id_table ) ;
2005-04-16 15:20:36 -07:00
static struct usb_driver kl5kusb105d_driver = {
. name = " kl5kusb105d " ,
. probe = usb_serial_probe ,
. disconnect = usb_serial_disconnect ,
. id_table = id_table ,
2010-05-19 00:01:39 +02:00
. no_dynamic_id = 1 ,
2005-04-16 15:20:36 -07:00
} ;
2005-06-20 21:15:16 -07:00
static struct usb_serial_driver kl5kusb105d_device = {
2005-06-20 21:15:16 -07:00
. driver = {
. owner = THIS_MODULE ,
2005-06-20 21:15:16 -07:00
. name = " kl5kusb105d " ,
2005-06-20 21:15:16 -07:00
} ,
2010-05-19 00:01:39 +02:00
. description = " KL5KUSB105D / PalmConnect " ,
. usb_driver = & kl5kusb105d_driver ,
. id_table = id_table ,
. num_ports = 1 ,
2010-05-19 00:01:40 +02:00
. bulk_out_size = 64 ,
2010-05-19 00:01:39 +02:00
. open = klsi_105_open ,
. close = klsi_105_close ,
. set_termios = klsi_105_set_termios ,
/*.break_ctl = klsi_105_break_ctl,*/
. tiocmget = klsi_105_tiocmget ,
. tiocmset = klsi_105_tiocmset ,
. attach = klsi_105_startup ,
. release = klsi_105_release ,
2010-05-19 00:01:40 +02:00
. throttle = usb_serial_generic_throttle ,
. unthrottle = usb_serial_generic_unthrottle ,
. process_read_urb = klsi_105_process_read_urb ,
. prepare_write_buffer = klsi_105_prepare_write_buffer ,
2005-04-16 15:20:36 -07:00
} ;
struct klsi_105_port_settings {
__u8 pktlen ; /* always 5, it seems */
__u8 baudrate ;
__u8 databits ;
__u8 unknown1 ;
__u8 unknown2 ;
} __attribute__ ( ( packed ) ) ;
struct klsi_105_private {
struct klsi_105_port_settings cfg ;
2006-12-08 02:38:45 -08:00
struct ktermios termios ;
2005-04-16 15:20:36 -07:00
unsigned long line_state ; /* modem line settings */
spinlock_t lock ;
} ;
/*
* Handle vendor specific USB requests
*/
# define KLSI_TIMEOUT 5000 /* default urb timeout */
static int klsi_105_chg_port_settings ( struct usb_serial_port * port ,
struct klsi_105_port_settings * settings )
{
int rc ;
2008-07-22 11:14:00 +01:00
rc = usb_control_msg ( port - > serial - > dev ,
usb_sndctrlpipe ( port - > serial - > dev , 0 ) ,
KL5KUSB105A_SIO_SET_DATA ,
USB_TYPE_VENDOR | USB_DIR_OUT | USB_RECIP_INTERFACE ,
0 , /* value */
0 , /* index */
settings ,
sizeof ( struct klsi_105_port_settings ) ,
KLSI_TIMEOUT ) ;
2005-04-16 15:20:36 -07:00
if ( rc < 0 )
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev ,
" Change port settings failed (error = %d) \n " , rc ) ;
2008-08-18 13:21:04 -07:00
dev_info ( & port - > serial - > dev - > dev ,
" %d byte block, baudrate %x, databits %d, u1 %d, u2 %d \n " ,
settings - > pktlen , settings - > baudrate , settings - > databits ,
settings - > unknown1 , settings - > unknown2 ) ;
2008-07-22 11:14:00 +01:00
return rc ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
/* translate a 16-bit status value from the device to linux's TIO bits */
static unsigned long klsi_105_status2linestate ( const __u16 status )
{
unsigned long res = 0 ;
res = ( ( status & KL5KUSB105A_DSR ) ? TIOCM_DSR : 0 )
| ( ( status & KL5KUSB105A_CTS ) ? TIOCM_CTS : 0 )
;
return res ;
}
2010-05-19 00:01:39 +02:00
2008-07-22 11:14:00 +01:00
/*
2005-04-16 15:20:36 -07:00
* Read line control via vendor command and return result through
2008-07-22 11:14:00 +01:00
* * line_state_p
2005-04-16 15:20:36 -07:00
*/
/* It seems that the status buffer has always only 2 bytes length */
# define KLSI_STATUSBUF_LEN 2
static int klsi_105_get_line_state ( struct usb_serial_port * port ,
unsigned long * line_state_p )
{
int rc ;
2009-12-28 23:01:52 +01:00
u8 * status_buf ;
2005-04-16 15:20:36 -07:00
__u16 status ;
2008-08-18 13:21:04 -07:00
dev_info ( & port - > serial - > dev - > dev , " sending SIO Poll request \n " ) ;
2009-12-28 23:01:52 +01:00
status_buf = kmalloc ( KLSI_STATUSBUF_LEN , GFP_KERNEL ) ;
if ( ! status_buf ) {
dev_err ( & port - > dev , " %s - out of memory for status buffer. \n " ,
__func__ ) ;
return - ENOMEM ;
}
status_buf [ 0 ] = 0xff ;
status_buf [ 1 ] = 0xff ;
2008-07-22 11:14:00 +01:00
rc = usb_control_msg ( port - > serial - > dev ,
2005-04-16 15:20:36 -07:00
usb_rcvctrlpipe ( port - > serial - > dev , 0 ) ,
KL5KUSB105A_SIO_POLL ,
2008-07-22 11:14:00 +01:00
USB_TYPE_VENDOR | USB_DIR_IN ,
2005-04-16 15:20:36 -07:00
0 , /* value */
0 , /* index */
status_buf , KLSI_STATUSBUF_LEN ,
10000
) ;
if ( rc < 0 )
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev , " Reading line status failed (error = %d) \n " ,
rc ) ;
2005-04-16 15:20:36 -07:00
else {
2008-05-01 20:52:57 -07:00
status = get_unaligned_le16 ( status_buf ) ;
2005-04-16 15:20:36 -07:00
2008-08-18 13:21:04 -07:00
dev_info ( & port - > serial - > dev - > dev , " read status %x %x " ,
status_buf [ 0 ] , status_buf [ 1 ] ) ;
2005-04-16 15:20:36 -07:00
* line_state_p = klsi_105_status2linestate ( status ) ;
}
2009-12-28 23:01:52 +01:00
kfree ( status_buf ) ;
2008-07-22 11:14:00 +01:00
return rc ;
2005-04-16 15:20:36 -07:00
}
/*
* Driver ' s tty interface functions
*/
2008-07-22 11:14:00 +01:00
static int klsi_105_startup ( struct usb_serial * serial )
2005-04-16 15:20:36 -07:00
{
struct klsi_105_private * priv ;
2010-05-19 00:01:40 +02:00
int i ;
2005-04-16 15:20:36 -07:00
/* check if we support the product id (see keyspan.c)
* FIXME
*/
/* allocate the private data structure */
2008-07-22 11:14:00 +01:00
for ( i = 0 ; i < serial - > num_ports ; i + + ) {
2005-04-16 15:20:36 -07:00
priv = kmalloc ( sizeof ( struct klsi_105_private ) ,
GFP_KERNEL ) ;
if ( ! priv ) {
2008-03-03 16:08:34 -08:00
dbg ( " %skmalloc for klsi_105_private failed. " , __func__ ) ;
2007-03-29 11:23:54 +02:00
i - - ;
goto err_cleanup ;
2005-04-16 15:20:36 -07:00
}
/* set initial values for control structures */
priv - > cfg . pktlen = 5 ;
priv - > cfg . baudrate = kl5kusb105a_sio_b9600 ;
priv - > cfg . databits = kl5kusb105a_dtb_8 ;
priv - > cfg . unknown1 = 0 ;
priv - > cfg . unknown2 = 1 ;
priv - > line_state = 0 ;
usb_set_serial_port_data ( serial - > port [ i ] , priv ) ;
2008-07-22 11:14:00 +01:00
spin_lock_init ( & priv - > lock ) ;
2005-04-16 15:20:36 -07:00
/* priv->termios is left uninitalized until port opening */
init_waitqueue_head ( & serial - > port [ i ] - > write_wait ) ;
}
2008-07-22 11:14:00 +01:00
2007-03-29 11:23:54 +02:00
return 0 ;
err_cleanup :
for ( ; i > = 0 ; i - - ) {
priv = usb_get_serial_port_data ( serial - > port [ i ] ) ;
2010-05-19 00:01:38 +02:00
kfree ( priv ) ;
2007-03-29 11:23:54 +02:00
usb_set_serial_port_data ( serial - > port [ i ] , NULL ) ;
}
return - ENOMEM ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
2009-06-02 11:53:55 -04:00
static void klsi_105_release ( struct usb_serial * serial )
{
int i ;
dbg ( " %s " , __func__ ) ;
2010-05-19 00:01:40 +02:00
for ( i = 0 ; i < serial - > num_ports ; + + i )
kfree ( usb_get_serial_port_data ( serial - > port [ i ] ) ) ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
2009-09-19 13:13:26 -07:00
static int klsi_105_open ( struct tty_struct * tty , struct usb_serial_port * port )
2005-04-16 15:20:36 -07:00
{
struct klsi_105_private * priv = usb_get_serial_port_data ( port ) ;
int retval = 0 ;
int rc ;
int i ;
unsigned long line_state ;
2009-12-28 23:01:52 +01:00
struct klsi_105_port_settings * cfg ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
2008-03-03 16:08:34 -08:00
dbg ( " %s port %d " , __func__ , port - > number ) ;
2005-04-16 15:20:36 -07:00
/* Do a defined restart:
* Set up sane default baud rate and send the ' READ_ON '
2008-07-22 11:14:00 +01:00
* vendor command .
2005-04-16 15:20:36 -07:00
* FIXME : set modem line control ( how ? )
* Then read the modem line control and store values in
* priv - > line_state .
*/
2009-12-28 23:01:52 +01:00
cfg = kmalloc ( sizeof ( * cfg ) , GFP_KERNEL ) ;
if ( ! cfg ) {
dev_err ( & port - > dev , " %s - out of memory for config buffer. \n " ,
__func__ ) ;
return - ENOMEM ;
}
cfg - > pktlen = 5 ;
cfg - > baudrate = kl5kusb105a_sio_b9600 ;
cfg - > databits = kl5kusb105a_dtb_8 ;
cfg - > unknown1 = 0 ;
cfg - > unknown2 = 1 ;
klsi_105_chg_port_settings ( port , cfg ) ;
2008-07-22 11:14:00 +01:00
2005-04-16 15:20:36 -07:00
/* set up termios structure */
2008-07-22 11:14:00 +01:00
spin_lock_irqsave ( & priv - > lock , flags ) ;
2008-07-22 11:09:07 +01:00
priv - > termios . c_iflag = tty - > termios - > c_iflag ;
priv - > termios . c_oflag = tty - > termios - > c_oflag ;
priv - > termios . c_cflag = tty - > termios - > c_cflag ;
priv - > termios . c_lflag = tty - > termios - > c_lflag ;
2008-07-22 11:14:00 +01:00
for ( i = 0 ; i < NCCS ; i + + )
2008-07-22 11:09:07 +01:00
priv - > termios . c_cc [ i ] = tty - > termios - > c_cc [ i ] ;
2009-12-28 23:01:52 +01:00
priv - > cfg . pktlen = cfg - > pktlen ;
priv - > cfg . baudrate = cfg - > baudrate ;
priv - > cfg . databits = cfg - > databits ;
priv - > cfg . unknown1 = cfg - > unknown1 ;
priv - > cfg . unknown2 = cfg - > unknown2 ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
/* READ_ON and urb submission */
2010-05-19 00:01:40 +02:00
rc = usb_serial_generic_open ( tty , port ) ;
2005-04-16 15:20:36 -07:00
if ( rc ) {
retval = rc ;
goto exit ;
}
rc = usb_control_msg ( port - > serial - > dev ,
2008-07-22 11:14:00 +01:00
usb_sndctrlpipe ( port - > serial - > dev , 0 ) ,
2005-04-16 15:20:36 -07:00
KL5KUSB105A_SIO_CONFIGURE ,
USB_TYPE_VENDOR | USB_DIR_OUT | USB_RECIP_INTERFACE ,
KL5KUSB105A_SIO_CONFIGURE_READ_ON ,
0 , /* index */
NULL ,
0 ,
KLSI_TIMEOUT ) ;
if ( rc < 0 ) {
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev , " Enabling read failed (error = %d) \n " , rc ) ;
2005-04-16 15:20:36 -07:00
retval = rc ;
2008-07-22 11:14:00 +01:00
} else
2008-03-03 16:08:34 -08:00
dbg ( " %s - enabled reading " , __func__ ) ;
2005-04-16 15:20:36 -07:00
rc = klsi_105_get_line_state ( port , & line_state ) ;
if ( rc > = 0 ) {
2008-07-22 11:14:00 +01:00
spin_lock_irqsave ( & priv - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
priv - > line_state = line_state ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2008-03-03 16:08:34 -08:00
dbg ( " %s - read line state 0x%lx " , __func__ , line_state ) ;
2005-04-16 15:20:36 -07:00
retval = 0 ;
} else
retval = rc ;
exit :
2009-12-28 23:01:52 +01:00
kfree ( cfg ) ;
2005-04-16 15:20:36 -07:00
return retval ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
2009-06-11 12:26:29 +01:00
static void klsi_105_close ( struct usb_serial_port * port )
2005-04-16 15:20:36 -07:00
{
int rc ;
2008-03-03 16:08:34 -08:00
dbg ( " %s port %d " , __func__ , port - > number ) ;
2005-04-16 15:20:36 -07:00
2008-01-22 12:47:15 +01:00
mutex_lock ( & port - > serial - > disc_mutex ) ;
if ( ! port - > serial - > disconnected ) {
/* send READ_OFF */
2008-07-22 11:14:00 +01:00
rc = usb_control_msg ( port - > serial - > dev ,
usb_sndctrlpipe ( port - > serial - > dev , 0 ) ,
KL5KUSB105A_SIO_CONFIGURE ,
USB_TYPE_VENDOR | USB_DIR_OUT ,
KL5KUSB105A_SIO_CONFIGURE_READ_OFF ,
0 , /* index */
NULL , 0 ,
KLSI_TIMEOUT ) ;
2008-01-22 12:47:15 +01:00
if ( rc < 0 )
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev ,
" Disabling read failed (error = %d) \n " , rc ) ;
2008-01-22 12:47:15 +01:00
}
mutex_unlock ( & port - > serial - > disc_mutex ) ;
2005-04-16 15:20:36 -07:00
/* shutdown our bulk reads and writes */
2010-05-19 00:01:40 +02:00
usb_serial_generic_close ( port ) ;
2005-04-16 15:20:36 -07:00
/* wgg - do I need this? I think so. */
usb_kill_urb ( port - > interrupt_in_urb ) ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
/* We need to write a complete 64-byte data block and encode the
2008-07-22 11:14:00 +01:00
* number actually sent in the first double - byte , LSB - order . That
2005-04-16 15:20:36 -07:00
* leaves at most 62 bytes of payload .
*/
2010-05-19 00:01:40 +02:00
# define KLSI_HDR_LEN 2
static int klsi_105_prepare_write_buffer ( struct usb_serial_port * port ,
void * dest , size_t size )
2005-04-16 15:20:36 -07:00
{
2010-05-19 00:01:40 +02:00
unsigned char * buf = dest ;
int count ;
2007-06-15 15:44:13 -07:00
2010-05-19 00:01:40 +02:00
count = kfifo_out_locked ( & port - > write_fifo , buf + KLSI_HDR_LEN , size ,
& port - > lock ) ;
put_unaligned_le16 ( count , buf ) ;
2005-04-16 15:20:36 -07:00
2010-05-19 00:01:40 +02:00
return count + KLSI_HDR_LEN ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
2010-05-19 00:01:40 +02:00
/* The data received is preceded by a length double-byte in LSB-first order.
*/
static void klsi_105_process_read_urb ( struct urb * urb )
2005-04-16 15:20:36 -07:00
{
2008-02-24 18:41:47 +08:00
struct usb_serial_port * port = urb - > context ;
2005-04-16 15:20:36 -07:00
unsigned char * data = urb - > transfer_buffer ;
2010-05-19 00:01:40 +02:00
struct tty_struct * tty ;
unsigned len ;
2005-04-16 15:20:36 -07:00
2010-05-19 00:01:40 +02:00
/* empty urbs seem to happen, we ignore them */
if ( ! urb - > actual_length )
return ;
2005-04-16 15:20:36 -07:00
2010-05-19 00:01:40 +02:00
if ( urb - > actual_length < = KLSI_HDR_LEN ) {
dbg ( " %s - malformed packet " , __func__ ) ;
2007-06-15 15:44:13 -07:00
return ;
}
2010-05-19 00:01:40 +02:00
tty = tty_port_tty_get ( & port - > port ) ;
if ( ! tty )
return ;
2005-04-16 15:20:36 -07:00
2010-05-19 00:01:40 +02:00
len = get_unaligned_le16 ( data ) ;
if ( len > urb - > actual_length - KLSI_HDR_LEN ) {
dbg ( " %s - packet length mismatch " , __func__ ) ;
len = urb - > actual_length - KLSI_HDR_LEN ;
2005-04-16 15:20:36 -07:00
}
2010-05-19 00:01:40 +02:00
tty_insert_flip_string ( tty , data + KLSI_HDR_LEN , len ) ;
tty_flip_buffer_push ( tty ) ;
tty_kref_put ( tty ) ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
2008-07-22 11:14:00 +01:00
static void klsi_105_set_termios ( struct tty_struct * tty ,
struct usb_serial_port * port ,
struct ktermios * old_termios )
2005-04-16 15:20:36 -07:00
{
struct klsi_105_private * priv = usb_get_serial_port_data ( port ) ;
2008-04-08 17:16:06 +01:00
unsigned int iflag = tty - > termios - > c_iflag ;
2005-04-16 15:20:36 -07:00
unsigned int old_iflag = old_termios - > c_iflag ;
2008-04-08 17:16:06 +01:00
unsigned int cflag = tty - > termios - > c_cflag ;
2005-04-16 15:20:36 -07:00
unsigned int old_cflag = old_termios - > c_cflag ;
2009-12-28 23:01:52 +01:00
struct klsi_105_port_settings * cfg ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
2008-04-08 17:16:06 +01:00
speed_t baud ;
2008-07-22 11:14:00 +01:00
2009-12-28 23:01:52 +01:00
cfg = kmalloc ( sizeof ( * cfg ) , GFP_KERNEL ) ;
if ( ! cfg ) {
dev_err ( & port - > dev , " %s - out of memory for config buffer. \n " ,
__func__ ) ;
return ;
}
2005-04-16 15:20:36 -07:00
/* lock while we are modifying the settings */
2008-07-22 11:14:00 +01:00
spin_lock_irqsave ( & priv - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
/*
* Update baud rate
*/
2008-04-08 17:16:06 +01:00
baud = tty_get_baud_rate ( tty ) ;
2008-07-22 11:14:00 +01:00
if ( ( cflag & CBAUD ) ! = ( old_cflag & CBAUD ) ) {
/* reassert DTR and (maybe) RTS on transition from B0 */
if ( ( old_cflag & CBAUD ) = = B0 ) {
2008-03-03 16:08:34 -08:00
dbg ( " %s: baud was B0 " , __func__ ) ;
2005-04-16 15:20:36 -07:00
#if 0
priv - > control_state | = TIOCM_DTR ;
/* don't set RTS if using hardware flow control */
2008-07-22 11:14:00 +01:00
if ( ! ( old_cflag & CRTSCTS ) )
2005-04-16 15:20:36 -07:00
priv - > control_state | = TIOCM_RTS ;
mct_u232_set_modem_ctrl ( serial , priv - > control_state ) ;
# endif
}
2008-04-08 17:16:06 +01:00
}
2008-07-22 11:14:00 +01:00
switch ( baud ) {
case 0 : /* handled below */
break ;
case 1200 :
priv - > cfg . baudrate = kl5kusb105a_sio_b1200 ;
break ;
case 2400 :
priv - > cfg . baudrate = kl5kusb105a_sio_b2400 ;
break ;
case 4800 :
priv - > cfg . baudrate = kl5kusb105a_sio_b4800 ;
break ;
case 9600 :
priv - > cfg . baudrate = kl5kusb105a_sio_b9600 ;
break ;
case 19200 :
priv - > cfg . baudrate = kl5kusb105a_sio_b19200 ;
break ;
case 38400 :
priv - > cfg . baudrate = kl5kusb105a_sio_b38400 ;
break ;
case 57600 :
priv - > cfg . baudrate = kl5kusb105a_sio_b57600 ;
break ;
case 115200 :
priv - > cfg . baudrate = kl5kusb105a_sio_b115200 ;
break ;
default :
dbg ( " KLSI USB->Serial converter: "
" unsupported baudrate request, using default of 9600 " ) ;
2007-08-10 14:53:34 -07:00
priv - > cfg . baudrate = kl5kusb105a_sio_b9600 ;
2008-07-22 11:14:00 +01:00
baud = 9600 ;
break ;
2008-04-08 17:16:06 +01:00
}
2008-07-22 11:14:00 +01:00
if ( ( cflag & CBAUD ) = = B0 ) {
2008-04-08 17:16:06 +01:00
dbg ( " %s: baud is B0 " , __func__ ) ;
/* Drop RTS and DTR */
/* maybe this should be simulated by sending read
* disable and read enable messages ?
*/
;
2005-04-16 15:20:36 -07:00
#if 0
2008-04-08 17:16:06 +01:00
priv - > control_state & = ~ ( TIOCM_DTR | TIOCM_RTS ) ;
2008-07-22 11:14:00 +01:00
mct_u232_set_modem_ctrl ( serial , priv - > control_state ) ;
2005-04-16 15:20:36 -07:00
# endif
}
2008-04-08 17:16:06 +01:00
tty_encode_baud_rate ( tty , baud , baud ) ;
2005-04-16 15:20:36 -07:00
if ( ( cflag & CSIZE ) ! = ( old_cflag & CSIZE ) ) {
/* set the number of data bits */
switch ( cflag & CSIZE ) {
case CS5 :
2008-03-03 16:08:34 -08:00
dbg ( " %s - 5 bits/byte not supported " , __func__ ) ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2009-12-28 23:01:52 +01:00
goto err ;
2005-04-16 15:20:36 -07:00
case CS6 :
2008-03-03 16:08:34 -08:00
dbg ( " %s - 6 bits/byte not supported " , __func__ ) ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2009-12-28 23:01:52 +01:00
goto err ;
2005-04-16 15:20:36 -07:00
case CS7 :
priv - > cfg . databits = kl5kusb105a_dtb_7 ;
break ;
case CS8 :
priv - > cfg . databits = kl5kusb105a_dtb_8 ;
break ;
default :
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev ,
" CSIZE was not CS5-CS8, using default of 8 \n " ) ;
2005-04-16 15:20:36 -07:00
priv - > cfg . databits = kl5kusb105a_dtb_8 ;
break ;
}
}
/*
* Update line control register ( LCR )
*/
if ( ( cflag & ( PARENB | PARODD ) ) ! = ( old_cflag & ( PARENB | PARODD ) )
2008-07-22 11:14:00 +01:00
| | ( cflag & CSTOPB ) ! = ( old_cflag & CSTOPB ) ) {
2008-04-08 17:16:06 +01:00
/* Not currently supported */
tty - > termios - > c_cflag & = ~ ( PARENB | PARODD | CSTOPB ) ;
2005-04-16 15:20:36 -07:00
#if 0
priv - > last_lcr = 0 ;
/* set the parity */
if ( cflag & PARENB )
priv - > last_lcr | = ( cflag & PARODD ) ?
MCT_U232_PARITY_ODD : MCT_U232_PARITY_EVEN ;
else
priv - > last_lcr | = MCT_U232_PARITY_NONE ;
/* set the number of stop bits */
priv - > last_lcr | = ( cflag & CSTOPB ) ?
MCT_U232_STOP_BITS_2 : MCT_U232_STOP_BITS_1 ;
mct_u232_set_line_ctrl ( serial , priv - > last_lcr ) ;
# endif
;
}
/*
* Set flow control : well , I do not really now how to handle DTR / RTS .
* Just do what we have seen with SniffUSB on Win98 .
*/
2008-07-22 11:14:00 +01:00
if ( ( iflag & IXOFF ) ! = ( old_iflag & IXOFF )
2005-04-16 15:20:36 -07:00
| | ( iflag & IXON ) ! = ( old_iflag & IXON )
2008-07-22 11:14:00 +01:00
| | ( cflag & CRTSCTS ) ! = ( old_cflag & CRTSCTS ) ) {
2008-04-08 17:16:06 +01:00
/* Not currently supported */
tty - > termios - > c_cflag & = ~ CRTSCTS ;
2005-04-16 15:20:36 -07:00
/* Drop DTR/RTS if no flow control otherwise assert */
#if 0
2008-07-22 11:14:00 +01:00
if ( ( iflag & IXOFF ) | | ( iflag & IXON ) | | ( cflag & CRTSCTS ) )
2005-04-16 15:20:36 -07:00
priv - > control_state | = TIOCM_DTR | TIOCM_RTS ;
else
priv - > control_state & = ~ ( TIOCM_DTR | TIOCM_RTS ) ;
mct_u232_set_modem_ctrl ( serial , priv - > control_state ) ;
# endif
;
}
2009-12-28 23:01:52 +01:00
memcpy ( cfg , & priv - > cfg , sizeof ( * cfg ) ) ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
/* now commit changes to device */
2009-12-28 23:01:52 +01:00
klsi_105_chg_port_settings ( port , cfg ) ;
err :
kfree ( cfg ) ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
#if 0
2008-07-22 11:14:00 +01:00
static void mct_u232_break_ctl ( struct tty_struct * tty , int break_state )
2005-04-16 15:20:36 -07:00
{
2008-07-22 11:09:07 +01:00
struct usb_serial_port * port = tty - > driver_data ;
2005-04-16 15:20:36 -07:00
struct usb_serial * serial = port - > serial ;
2008-07-22 11:14:00 +01:00
struct mct_u232_private * priv =
( struct mct_u232_private * ) port - > private ;
2005-04-16 15:20:36 -07:00
unsigned char lcr = priv - > last_lcr ;
2008-03-03 16:08:34 -08:00
dbg ( " %sstate=%d " , __func__ , break_state ) ;
2005-04-16 15:20:36 -07:00
2009-01-02 13:48:56 +00:00
/* LOCKING */
2005-04-16 15:20:36 -07:00
if ( break_state )
lcr | = MCT_U232_SET_BREAK ;
mct_u232_set_line_ctrl ( serial , lcr ) ;
2010-05-19 00:01:39 +02:00
}
2005-04-16 15:20:36 -07:00
# endif
2008-07-22 11:14:00 +01:00
static int klsi_105_tiocmget ( struct tty_struct * tty , struct file * file )
2005-04-16 15:20:36 -07:00
{
2008-07-22 11:09:07 +01:00
struct usb_serial_port * port = tty - > driver_data ;
2005-04-16 15:20:36 -07:00
struct klsi_105_private * priv = usb_get_serial_port_data ( port ) ;
unsigned long flags ;
int rc ;
unsigned long line_state ;
2008-03-03 16:08:34 -08:00
dbg ( " %s - request, just guessing " , __func__ ) ;
2005-04-16 15:20:36 -07:00
rc = klsi_105_get_line_state ( port , & line_state ) ;
if ( rc < 0 ) {
2008-08-20 16:56:34 -07:00
dev_err ( & port - > dev ,
" Reading line control failed (error = %d) \n " , rc ) ;
2005-04-16 15:20:36 -07:00
/* better return value? EAGAIN? */
return rc ;
}
2008-07-22 11:14:00 +01:00
spin_lock_irqsave ( & priv - > lock , flags ) ;
2005-04-16 15:20:36 -07:00
priv - > line_state = line_state ;
2008-07-22 11:14:00 +01:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2008-03-03 16:08:34 -08:00
dbg ( " %s - read line state 0x%lx " , __func__ , line_state ) ;
2005-04-16 15:20:36 -07:00
return ( int ) line_state ;
}
2008-07-22 11:14:00 +01:00
static int klsi_105_tiocmset ( struct tty_struct * tty , struct file * file ,
unsigned int set , unsigned int clear )
2005-04-16 15:20:36 -07:00
{
int retval = - EINVAL ;
2008-07-22 11:14:00 +01:00
2008-03-03 16:08:34 -08:00
dbg ( " %s " , __func__ ) ;
2005-04-16 15:20:36 -07:00
/* if this ever gets implemented, it should be done something like this:
struct usb_serial * serial = port - > serial ;
struct klsi_105_private * priv = usb_get_serial_port_data ( port ) ;
unsigned long flags ;
int control ;
spin_lock_irqsave ( & priv - > lock , flags ) ;
if ( set & TIOCM_RTS )
priv - > control_state | = TIOCM_RTS ;
if ( set & TIOCM_DTR )
priv - > control_state | = TIOCM_DTR ;
if ( clear & TIOCM_RTS )
priv - > control_state & = ~ TIOCM_RTS ;
if ( clear & TIOCM_DTR )
priv - > control_state & = ~ TIOCM_DTR ;
control = priv - > control_state ;
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
retval = mct_u232_set_modem_ctrl ( serial , control ) ;
*/
return retval ;
}
2008-07-22 11:14:00 +01:00
static int __init klsi_105_init ( void )
2005-04-16 15:20:36 -07:00
{
int retval ;
retval = usb_serial_register ( & kl5kusb105d_device ) ;
if ( retval )
goto failed_usb_serial_register ;
retval = usb_register ( & kl5kusb105d_driver ) ;
if ( retval )
goto failed_usb_register ;
2008-08-18 13:21:04 -07:00
printk ( KERN_INFO KBUILD_MODNAME " : " DRIVER_VERSION " : "
DRIVER_DESC " \n " ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
failed_usb_register :
usb_serial_deregister ( & kl5kusb105d_device ) ;
failed_usb_serial_register :
return retval ;
}
2008-07-22 11:14:00 +01:00
static void __exit klsi_105_exit ( void )
2005-04-16 15:20:36 -07:00
{
2008-07-22 11:14:00 +01:00
usb_deregister ( & kl5kusb105d_driver ) ;
usb_serial_deregister ( & kl5kusb105d_device ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-22 11:14:00 +01:00
module_init ( klsi_105_init ) ;
module_exit ( klsi_105_exit ) ;
2005-04-16 15:20:36 -07:00
2008-07-22 11:14:00 +01:00
MODULE_AUTHOR ( DRIVER_AUTHOR ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_LICENSE ( " GPL " ) ;
2005-04-16 15:20:36 -07:00
module_param ( debug , bool , S_IRUGO | S_IWUSR ) ;
MODULE_PARM_DESC ( debug , " enable extensive debugging messages " ) ;