2005-04-17 02:20:36 +04:00
/*
* USB IR Dongle driver
*
* Copyright ( C ) 2001 - 2002 Greg Kroah - Hartman ( greg @ kroah . com )
* Copyright ( C ) 2002 Gary Brubaker ( xavyer @ ix . netcom . com )
*
* 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 .
*
* This driver allows a USB IrDA device to be used as a " dumb " serial device .
* This can be useful if you do not have access to a full IrDA stack on the
* other side of the connection . If you do have an IrDA stack on both devices ,
* please use the usb - irda driver , as it contains the proper error checking and
* other goodness of a full IrDA stack .
*
* Portions of this driver were taken from drivers / net / irda / irda - usb . c , which
* was written by Roman Weissgaerber < weissg @ vienna . at > , Dag Brattli
* < dag @ brattli . net > , and Jean Tourrilhes < jt @ hpl . hp . com >
*
2008-06-03 15:47:52 +04:00
* See Documentation / usb / usb - serial . txt for more information on using this
* driver
*
* 2008 _Jun_02 Felipe Balbi < me @ felipebalbi . com >
* Introduced common header to be used also in USB Gadget Framework .
* Still needs some other style fixes .
2005-04-17 02:20:36 +04:00
*
2008-11-11 17:48:44 +03:00
* 2007 _Jun_21 Alan Cox < alan @ lxorguk . ukuu . org . uk >
2007-06-22 17:44:54 +04:00
* Minimal cleanups for some of the driver problens and tty layer abuse .
* Still needs fixing to allow multiple dongles .
*
2005-04-17 02:20:36 +04:00
* 2002 _Mar_07 greg kh
* moved some needed structures and # define values from the
* net / irda / irda - usb . h file into our file , as we don ' t want to depend on
* that codebase compiling correctly : )
*
* 2002 _Jan_14 gb
* Added module parameter to force specific number of XBOFs .
* Added ir_xbof_change ( ) .
* Reorganized read_bulk_callback error handling .
* Switched from FILL_BULK_URB ( ) to usb_fill_bulk_urb ( ) .
*
* 2001 _Nov_08 greg kh
* Changed the irda_usb_find_class_desc ( ) function based on comments and
* code from Martin Diehl .
*
* 2001 _Nov_01 greg kh
* Added support for more IrDA USB devices .
* Added support for zero packet . Added buffer override paramater , so
* users can transfer larger packets at once if they wish . Both patches
* came from Dag Brattli < dag @ obexcode . com > .
*
* 2001 _Oct_07 greg kh
* initial version released .
*/
# 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>
# include <linux/spinlock.h>
2008-06-03 15:47:52 +04:00
# include <linux/uaccess.h>
2005-04-17 02:20:36 +04:00
# include <linux/usb.h>
2006-07-12 08:22:58 +04:00
# include <linux/usb/serial.h>
2008-06-03 15:47:52 +04:00
# include <linux/usb/irda.h>
2005-04-17 02:20:36 +04:00
/*
* Version Information
*/
# define DRIVER_VERSION "v0.4"
# define DRIVER_AUTHOR "Greg Kroah-Hartman <greg@kroah.com>"
# define DRIVER_DESC "USB IR Dongle driver"
static int debug ;
/* if overridden by the user, then use their value for the size of the read and
* write urbs */
static int buffer_size ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
/* if overridden by the user, then use the specified number of XBOFs */
static int xbof = - 1 ;
2008-07-22 14:09:07 +04:00
static int ir_startup ( struct usb_serial * serial ) ;
static int ir_open ( struct tty_struct * tty , struct usb_serial_port * port ,
struct file * filep ) ;
2009-06-11 15:26:29 +04:00
static void ir_close ( struct usb_serial_port * port ) ;
2008-07-22 14:09:07 +04:00
static int ir_write ( struct tty_struct * tty , struct usb_serial_port * port ,
const unsigned char * buf , int count ) ;
static void ir_write_bulk_callback ( struct urb * urb ) ;
static void ir_read_bulk_callback ( struct urb * urb ) ;
static void ir_set_termios ( struct tty_struct * tty ,
struct usb_serial_port * port , struct ktermios * old_termios ) ;
2005-04-17 02:20:36 +04:00
2007-06-22 17:44:54 +04:00
/* Not that this lot means you can only have one per system */
2008-06-03 15:47:52 +04:00
static u8 ir_baud ;
static u8 ir_xbof ;
static u8 ir_add_bof ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
static struct usb_device_id ir_id_table [ ] = {
2005-04-17 02:20:36 +04:00
{ USB_DEVICE ( 0x050f , 0x0180 ) } , /* KC Technology, KC-180 */
{ USB_DEVICE ( 0x08e9 , 0x0100 ) } , /* XTNDAccess */
{ USB_DEVICE ( 0x09c4 , 0x0011 ) } , /* ACTiSys ACT-IR2000U */
2008-06-03 15:47:52 +04:00
{ USB_INTERFACE_INFO ( USB_CLASS_APP_SPEC , USB_SUBCLASS_IRDA , 0 ) } ,
2005-04-17 02:20:36 +04:00
{ } /* Terminating entry */
} ;
2008-06-03 15:47:52 +04:00
MODULE_DEVICE_TABLE ( usb , ir_id_table ) ;
2005-04-17 02:20:36 +04:00
static struct usb_driver ir_driver = {
2008-06-03 15:47:52 +04:00
. name = " ir-usb " ,
. probe = usb_serial_probe ,
. disconnect = usb_serial_disconnect ,
. id_table = ir_id_table ,
. no_dynamic_id = 1 ,
2005-04-17 02:20:36 +04:00
} ;
2005-06-21 08:15:16 +04:00
static struct usb_serial_driver ir_device = {
2008-06-03 15:47:52 +04:00
. driver = {
. owner = THIS_MODULE ,
. name = " ir-usb " ,
2005-06-21 08:15:16 +04:00
} ,
2008-06-03 15:47:52 +04:00
. description = " IR Dongle " ,
. usb_driver = & ir_driver ,
. id_table = ir_id_table ,
. num_ports = 1 ,
. set_termios = ir_set_termios ,
. attach = ir_startup ,
. open = ir_open ,
. close = ir_close ,
. write = ir_write ,
. write_bulk_callback = ir_write_bulk_callback ,
. read_bulk_callback = ir_read_bulk_callback ,
2005-04-17 02:20:36 +04:00
} ;
2008-06-03 15:47:52 +04:00
static inline void irda_usb_dump_class_desc ( struct usb_irda_cs_descriptor * desc )
2005-04-17 02:20:36 +04:00
{
dbg ( " bLength=%x " , desc - > bLength ) ;
dbg ( " bDescriptorType=%x " , desc - > bDescriptorType ) ;
2008-06-03 15:47:52 +04:00
dbg ( " bcdSpecRevision=%x " , __le16_to_cpu ( desc - > bcdSpecRevision ) ) ;
2005-04-17 02:20:36 +04:00
dbg ( " bmDataSize=%x " , desc - > bmDataSize ) ;
dbg ( " bmWindowSize=%x " , desc - > bmWindowSize ) ;
dbg ( " bmMinTurnaroundTime=%d " , desc - > bmMinTurnaroundTime ) ;
2008-06-03 15:47:52 +04:00
dbg ( " wBaudRate=%x " , __le16_to_cpu ( desc - > wBaudRate ) ) ;
2005-04-17 02:20:36 +04:00
dbg ( " bmAdditionalBOFs=%x " , desc - > bmAdditionalBOFs ) ;
dbg ( " bIrdaRateSniff=%x " , desc - > bIrdaRateSniff ) ;
dbg ( " bMaxUnicastList=%x " , desc - > bMaxUnicastList ) ;
}
/*------------------------------------------------------------------*/
/*
* Function irda_usb_find_class_desc ( dev , ifnum )
*
* Returns instance of IrDA class descriptor , or NULL if not found
*
* The class descriptor is some extra info that IrDA USB devices will
* offer to us , describing their IrDA characteristics . We will use that in
* irda_usb_init_qos ( )
*
* Based on the same function in drivers / net / irda / irda - usb . c
*/
2008-06-03 15:47:52 +04:00
static struct usb_irda_cs_descriptor *
irda_usb_find_class_desc ( struct usb_device * dev , unsigned int ifnum )
2005-04-17 02:20:36 +04:00
{
2008-06-03 15:47:52 +04:00
struct usb_irda_cs_descriptor * desc ;
2005-04-17 02:20:36 +04:00
int ret ;
2008-06-03 15:47:52 +04:00
desc = kzalloc ( sizeof ( * desc ) , GFP_KERNEL ) ;
if ( ! desc )
2005-04-17 02:20:36 +04:00
return NULL ;
2008-06-03 15:47:52 +04:00
ret = usb_control_msg ( dev , usb_rcvctrlpipe ( dev , 0 ) ,
USB_REQ_CS_IRDA_GET_CLASS_DESC ,
2005-04-17 02:20:36 +04:00
USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE ,
0 , ifnum , desc , sizeof ( * desc ) , 1000 ) ;
2008-06-03 15:47:52 +04:00
2008-03-04 03:08:34 +03:00
dbg ( " %s - ret=%d " , __func__ , ret ) ;
2005-04-17 02:20:36 +04:00
if ( ret < sizeof ( * desc ) ) {
dbg ( " %s - class descriptor read %s (%d) " ,
2008-03-04 03:08:34 +03:00
__func__ ,
2008-06-03 15:47:52 +04:00
( ret < 0 ) ? " failed " : " too short " ,
2005-04-17 02:20:36 +04:00
ret ) ;
goto error ;
}
2008-06-03 15:47:52 +04:00
if ( desc - > bDescriptorType ! = USB_DT_CS_IRDA ) {
2008-03-04 03:08:34 +03:00
dbg ( " %s - bad class descriptor type " , __func__ ) ;
2005-04-17 02:20:36 +04:00
goto error ;
}
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
irda_usb_dump_class_desc ( desc ) ;
return desc ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
error :
kfree ( desc ) ;
return NULL ;
}
static u8 ir_xbof_change ( u8 xbof )
{
u8 result ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
/* reference irda-usb.c */
2008-06-03 15:47:52 +04:00
switch ( xbof ) {
case 48 :
result = 0x10 ;
break ;
case 28 :
case 24 :
result = 0x20 ;
break ;
default :
case 12 :
result = 0x30 ;
break ;
case 5 :
case 6 :
result = 0x40 ;
break ;
case 3 :
result = 0x50 ;
break ;
case 2 :
result = 0x60 ;
break ;
case 1 :
result = 0x70 ;
break ;
case 0 :
result = 0x80 ;
break ;
2005-04-17 02:20:36 +04:00
}
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
return ( result ) ;
}
2008-06-03 15:47:52 +04:00
static int ir_startup ( struct usb_serial * serial )
2005-04-17 02:20:36 +04:00
{
2008-06-03 15:47:52 +04:00
struct usb_irda_cs_descriptor * irda_desc ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
irda_desc = irda_usb_find_class_desc ( serial - > dev , 0 ) ;
if ( ! irda_desc ) {
dev_err ( & serial - > dev - > dev ,
" IRDA class descriptor not found, device not bound \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
2008-06-03 15:47:52 +04:00
dbg ( " %s - Baud rates supported:%s%s%s%s%s%s%s%s%s " ,
2008-03-04 03:08:34 +03:00
__func__ ,
2008-06-03 15:47:52 +04:00
( irda_desc - > wBaudRate & USB_IRDA_BR_2400 ) ? " 2400 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_9600 ) ? " 9600 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_19200 ) ? " 19200 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_38400 ) ? " 38400 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_57600 ) ? " 57600 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_115200 ) ? " 115200 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_576000 ) ? " 576000 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_1152000 ) ? " 1152000 " : " " ,
( irda_desc - > wBaudRate & USB_IRDA_BR_4000000 ) ? " 4000000 " : " " ) ;
switch ( irda_desc - > bmAdditionalBOFs ) {
case USB_IRDA_AB_48 :
ir_add_bof = 48 ;
break ;
case USB_IRDA_AB_24 :
ir_add_bof = 24 ;
break ;
case USB_IRDA_AB_12 :
ir_add_bof = 12 ;
break ;
case USB_IRDA_AB_6 :
ir_add_bof = 6 ;
break ;
case USB_IRDA_AB_3 :
ir_add_bof = 3 ;
break ;
case USB_IRDA_AB_2 :
ir_add_bof = 2 ;
break ;
case USB_IRDA_AB_1 :
ir_add_bof = 1 ;
break ;
case USB_IRDA_AB_0 :
ir_add_bof = 0 ;
break ;
default :
break ;
2005-04-17 02:20:36 +04:00
}
2008-06-03 15:47:52 +04:00
kfree ( irda_desc ) ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2008-07-22 14:09:07 +04:00
static int ir_open ( struct tty_struct * tty ,
struct usb_serial_port * port , struct file * filp )
2005-04-17 02:20:36 +04:00
{
char * buffer ;
int result = 0 ;
2008-03-04 03:08:34 +03:00
dbg ( " %s - port %d " , __func__ , port - > number ) ;
2005-04-17 02:20:36 +04:00
if ( buffer_size ) {
/* override the default buffer sizes */
2008-06-03 15:47:52 +04:00
buffer = kmalloc ( buffer_size , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! buffer ) {
2008-06-03 15:47:52 +04:00
dev_err ( & port - > dev , " %s - out of memory. \n " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
2008-06-03 15:47:52 +04:00
kfree ( port - > read_urb - > transfer_buffer ) ;
2005-04-17 02:20:36 +04:00
port - > read_urb - > transfer_buffer = buffer ;
port - > read_urb - > transfer_buffer_length = buffer_size ;
2008-06-03 15:47:52 +04:00
buffer = kmalloc ( buffer_size , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! buffer ) {
2008-06-03 15:47:52 +04:00
dev_err ( & port - > dev , " %s - out of memory. \n " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
2008-06-03 15:47:52 +04:00
kfree ( port - > write_urb - > transfer_buffer ) ;
2005-04-17 02:20:36 +04:00
port - > write_urb - > transfer_buffer = buffer ;
port - > write_urb - > transfer_buffer_length = buffer_size ;
port - > bulk_out_size = buffer_size ;
}
/* Start reading from the device */
2008-06-03 15:47:52 +04:00
usb_fill_bulk_urb (
2005-04-17 02:20:36 +04:00
port - > read_urb ,
2008-06-03 15:47:52 +04:00
port - > serial - > dev ,
usb_rcvbulkpipe ( port - > serial - > dev ,
port - > bulk_in_endpointAddress ) ,
2005-04-17 02:20:36 +04:00
port - > read_urb - > transfer_buffer ,
port - > read_urb - > transfer_buffer_length ,
ir_read_bulk_callback ,
port ) ;
result = usb_submit_urb ( port - > read_urb , GFP_KERNEL ) ;
if ( result )
2008-06-03 15:47:52 +04:00
dev_err ( & port - > dev ,
" %s - failed submitting read urb, error %d \n " ,
__func__ , result ) ;
2005-04-17 02:20:36 +04:00
return result ;
}
2009-06-11 15:26:29 +04:00
static void ir_close ( struct usb_serial_port * port )
2005-04-17 02:20:36 +04:00
{
2008-03-04 03:08:34 +03:00
dbg ( " %s - port %d " , __func__ , port - > number ) ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
/* shutdown our bulk read */
usb_kill_urb ( port - > read_urb ) ;
}
2008-07-22 14:09:07 +04:00
static int ir_write ( struct tty_struct * tty , struct usb_serial_port * port ,
const unsigned char * buf , int count )
2005-04-17 02:20:36 +04:00
{
unsigned char * transfer_buffer ;
int result ;
int transfer_size ;
2008-03-04 03:08:34 +03:00
dbg ( " %s - port = %d, count = %d " , __func__ , port - > number , count ) ;
2005-04-17 02:20:36 +04:00
if ( count = = 0 )
return 0 ;
2006-09-25 14:51:41 +04:00
spin_lock_bh ( & port - > lock ) ;
2005-04-23 23:49:16 +04:00
if ( port - > write_urb_busy ) {
2006-09-25 14:51:41 +04:00
spin_unlock_bh ( & port - > lock ) ;
2008-03-04 03:08:34 +03:00
dbg ( " %s - already writing " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2005-04-23 23:49:16 +04:00
port - > write_urb_busy = 1 ;
2006-09-25 14:51:41 +04:00
spin_unlock_bh ( & port - > lock ) ;
2005-04-17 02:20:36 +04:00
transfer_buffer = port - > write_urb - > transfer_buffer ;
transfer_size = min ( count , port - > bulk_out_size - 1 ) ;
/*
* The first byte of the packet we send to the device contains an
2008-06-03 15:47:52 +04:00
* inbound header which indicates an additional number of BOFs and
2005-04-17 02:20:36 +04:00
* a baud rate change .
*
* See section 5.4 .2 .2 of the USB IrDA spec .
*/
* transfer_buffer = ir_xbof | ir_baud ;
+ + transfer_buffer ;
2008-06-03 15:47:52 +04:00
memcpy ( transfer_buffer , buf , transfer_size ) ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
usb_fill_bulk_urb (
2005-04-17 02:20:36 +04:00
port - > write_urb ,
port - > serial - > dev ,
usb_sndbulkpipe ( port - > serial - > dev ,
port - > bulk_out_endpointAddress ) ,
port - > write_urb - > transfer_buffer ,
transfer_size + 1 ,
ir_write_bulk_callback ,
port ) ;
port - > write_urb - > transfer_flags = URB_ZERO_PACKET ;
2008-06-03 15:47:52 +04:00
result = usb_submit_urb ( port - > write_urb , GFP_ATOMIC ) ;
2005-04-23 23:49:16 +04:00
if ( result ) {
port - > write_urb_busy = 0 ;
2008-06-03 15:47:52 +04:00
dev_err ( & port - > dev ,
" %s - failed submitting write urb, error %d \n " ,
__func__ , result ) ;
2005-04-23 23:49:16 +04:00
} else
2005-04-17 02:20:36 +04:00
result = transfer_size ;
return result ;
}
2008-06-03 15:47:52 +04:00
static void ir_write_bulk_callback ( struct urb * urb )
2005-04-17 02:20:36 +04:00
{
2008-02-24 13:41:47 +03:00
struct usb_serial_port * port = urb - > context ;
2007-06-16 02:44:13 +04:00
int status = urb - > status ;
2005-04-17 02:20:36 +04:00
2008-03-04 03:08:34 +03:00
dbg ( " %s - port %d " , __func__ , port - > number ) ;
2005-04-23 23:49:16 +04:00
port - > write_urb_busy = 0 ;
2007-06-16 02:44:13 +04:00
if ( status ) {
dbg ( " %s - nonzero write bulk status received: %d " ,
2008-03-04 03:08:34 +03:00
__func__ , status ) ;
2005-04-17 02:20:36 +04:00
return ;
}
2008-06-03 15:47:52 +04:00
usb_serial_debug_data (
2005-04-17 02:20:36 +04:00
debug ,
& port - > dev ,
2008-03-04 03:08:34 +03:00
__func__ ,
2005-04-17 02:20:36 +04:00
urb - > actual_length ,
urb - > transfer_buffer ) ;
2006-05-23 08:58:49 +04:00
usb_serial_port_softint ( port ) ;
2005-04-17 02:20:36 +04:00
}
2008-06-03 15:47:52 +04:00
static void ir_read_bulk_callback ( struct urb * urb )
2005-04-17 02:20:36 +04:00
{
2008-02-24 13:41:47 +03:00
struct usb_serial_port * port = urb - > context ;
2005-04-17 02:20:36 +04:00
struct tty_struct * tty ;
unsigned char * data = urb - > transfer_buffer ;
int result ;
2007-06-16 02:44:13 +04:00
int status = urb - > status ;
2005-04-17 02:20:36 +04:00
2008-03-04 03:08:34 +03:00
dbg ( " %s - port %d " , __func__ , port - > number ) ;
2005-04-17 02:20:36 +04:00
2008-07-22 14:09:07 +04:00
if ( ! port - > port . count ) {
2008-03-04 03:08:34 +03:00
dbg ( " %s - port closed. " , __func__ ) ;
2005-04-17 02:20:36 +04:00
return ;
}
2007-06-16 02:44:13 +04:00
switch ( status ) {
2008-06-03 15:47:52 +04:00
case 0 : /* Successful */
/*
* The first byte of the packet we get from the device
* contains a busy indicator and baud rate change .
* See section 5.4 .1 .2 of the USB IrDA spec .
*/
if ( ( * data & 0x0f ) > 0 )
ir_baud = * data & 0x0f ;
2008-07-22 14:09:07 +04:00
usb_serial_debug_data ( debug , & port - > dev , __func__ ,
urb - > actual_length , data ) ;
2008-10-13 13:39:46 +04:00
tty = tty_port_tty_get ( & port - > port ) ;
2008-06-03 15:47:52 +04:00
if ( tty_buffer_request_room ( tty , urb - > actual_length - 1 ) ) {
2008-07-22 14:09:07 +04:00
tty_insert_flip_string ( tty , data + 1 , urb - > actual_length - 1 ) ;
2008-06-03 15:47:52 +04:00
tty_flip_buffer_push ( tty ) ;
}
2008-10-13 13:39:46 +04:00
tty_kref_put ( tty ) ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
/*
* No break here .
* We want to resubmit the urb so we can read
* again .
*/
case - EPROTO : /* taking inspiration from pl2303.c */
2008-07-22 14:09:07 +04:00
/* Continue trying to always read */
2008-06-03 15:47:52 +04:00
usb_fill_bulk_urb (
port - > read_urb ,
2008-07-22 14:09:07 +04:00
port - > serial - > dev ,
2008-06-03 15:47:52 +04:00
usb_rcvbulkpipe ( port - > serial - > dev ,
port - > bulk_in_endpointAddress ) ,
port - > read_urb - > transfer_buffer ,
port - > read_urb - > transfer_buffer_length ,
ir_read_bulk_callback ,
port ) ;
result = usb_submit_urb ( port - > read_urb , GFP_ATOMIC ) ;
if ( result )
2008-07-22 14:09:07 +04:00
dev_err ( & port - > dev , " %s - failed resubmitting read urb, error %d \n " ,
2008-06-03 15:47:52 +04:00
__func__ , result ) ;
2008-07-22 14:09:07 +04:00
break ;
2008-06-03 15:47:52 +04:00
default :
dbg ( " %s - nonzero read bulk status received: %d " ,
2008-07-22 14:09:07 +04:00
__func__ , status ) ;
break ;
2005-04-17 02:20:36 +04:00
}
return ;
}
2008-07-22 14:09:07 +04:00
static void ir_set_termios ( struct tty_struct * tty ,
struct usb_serial_port * port , struct ktermios * old_termios )
2005-04-17 02:20:36 +04:00
{
unsigned char * transfer_buffer ;
int result ;
2007-06-22 17:44:54 +04:00
speed_t baud ;
int ir_baud ;
2005-04-17 02:20:36 +04:00
2008-03-04 03:08:34 +03:00
dbg ( " %s - port %d " , __func__ , port - > number ) ;
2005-04-17 02:20:36 +04:00
2008-07-22 14:09:07 +04:00
baud = tty_get_baud_rate ( tty ) ;
2007-06-22 17:44:54 +04:00
/*
* FIXME , we should compare the baud request against the
* capability stated in the IR header that we got in the
* startup function .
*/
switch ( baud ) {
2008-06-03 15:47:52 +04:00
case 2400 :
ir_baud = USB_IRDA_BR_2400 ;
break ;
case 9600 :
ir_baud = USB_IRDA_BR_9600 ;
break ;
case 19200 :
ir_baud = USB_IRDA_BR_19200 ;
break ;
case 38400 :
ir_baud = USB_IRDA_BR_38400 ;
break ;
case 57600 :
ir_baud = USB_IRDA_BR_57600 ;
break ;
case 115200 :
ir_baud = USB_IRDA_BR_115200 ;
break ;
case 576000 :
ir_baud = USB_IRDA_BR_576000 ;
break ;
case 1152000 :
ir_baud = USB_IRDA_BR_1152000 ;
break ;
case 4000000 :
ir_baud = USB_IRDA_BR_4000000 ;
break ;
default :
ir_baud = USB_IRDA_BR_9600 ;
baud = 9600 ;
2005-04-17 02:20:36 +04:00
}
2007-06-22 17:44:54 +04:00
if ( xbof = = - 1 )
ir_xbof = ir_xbof_change ( ir_add_bof ) ;
else
ir_xbof = ir_xbof_change ( xbof ) ;
2005-04-17 02:20:36 +04:00
2007-06-22 17:44:54 +04:00
/* FIXME need to check to see if our write urb is busy right
* now , or use a urb pool .
*
* send the baud change out on an " empty " data packet
*/
transfer_buffer = port - > write_urb - > transfer_buffer ;
* transfer_buffer = ir_xbof | ir_baud ;
2005-04-17 02:20:36 +04:00
2008-06-03 15:47:52 +04:00
usb_fill_bulk_urb (
2007-06-22 17:44:54 +04:00
port - > write_urb ,
port - > serial - > dev ,
2008-06-03 15:47:52 +04:00
usb_sndbulkpipe ( port - > serial - > dev ,
port - > bulk_out_endpointAddress ) ,
2007-06-22 17:44:54 +04:00
port - > write_urb - > transfer_buffer ,
1 ,
ir_write_bulk_callback ,
port ) ;
port - > write_urb - > transfer_flags = URB_ZERO_PACKET ;
2008-06-03 15:47:52 +04:00
result = usb_submit_urb ( port - > write_urb , GFP_KERNEL ) ;
2007-06-22 17:44:54 +04:00
if ( result )
2008-06-03 15:47:52 +04:00
dev_err ( & port - > dev ,
" %s - failed submitting write urb, error %d \n " ,
__func__ , result ) ;
2007-10-18 12:24:20 +04:00
/* Only speed changes are supported */
2008-07-22 14:09:07 +04:00
tty_termios_copy_hw ( tty - > termios , old_termios ) ;
tty_encode_baud_rate ( tty , baud , baud ) ;
2005-04-17 02:20:36 +04:00
}
2008-06-03 15:47:52 +04:00
static int __init ir_init ( void )
2005-04-17 02:20:36 +04:00
{
int retval ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
retval = usb_serial_register ( & ir_device ) ;
if ( retval )
goto failed_usb_serial_register ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
retval = usb_register ( & ir_driver ) ;
2008-06-03 15:47:52 +04:00
if ( retval )
2005-04-17 02:20:36 +04:00
goto failed_usb_register ;
2008-06-03 15:47:52 +04:00
2008-08-19 00:21:04 +04:00
printk ( KERN_INFO KBUILD_MODNAME " : " DRIVER_VERSION " : "
DRIVER_DESC " \n " ) ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
return 0 ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
failed_usb_register :
usb_serial_deregister ( & ir_device ) ;
2008-06-03 15:47:52 +04:00
2005-04-17 02:20:36 +04:00
failed_usb_serial_register :
return retval ;
}
2008-06-03 15:47:52 +04:00
static void __exit ir_exit ( void )
2005-04-17 02:20:36 +04:00
{
2008-06-03 15:47:52 +04:00
usb_deregister ( & ir_driver ) ;
usb_serial_deregister ( & ir_device ) ;
2005-04-17 02:20:36 +04:00
}
module_init ( ir_init ) ;
module_exit ( ir_exit ) ;
MODULE_AUTHOR ( DRIVER_AUTHOR ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_LICENSE ( " GPL " ) ;
module_param ( debug , bool , S_IRUGO | S_IWUSR ) ;
MODULE_PARM_DESC ( debug , " Debug enabled or not " ) ;
module_param ( xbof , int , 0 ) ;
MODULE_PARM_DESC ( xbof , " Force specific number of XBOFs " ) ;
module_param ( buffer_size , int , 0 ) ;
MODULE_PARM_DESC ( buffer_size , " Size of the transfer buffers " ) ;