2008-11-20 00:58:50 +03:00
/*
* Wacom W8001 penabled serial touchscreen driver
*
* Copyright ( c ) 2008 Jaya Kumar
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of this archive for
* more details .
*
* Layout based on Elo serial touchscreen driver by Vojtech Pavlik
*/
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/input.h>
# include <linux/serio.h>
# include <linux/init.h>
# include <linux/ctype.h>
# define DRIVER_DESC "Wacom W8001 serial touchscreen driver"
MODULE_AUTHOR ( " Jaya Kumar <jayakumar.lkml@gmail.com> " ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_LICENSE ( " GPL " ) ;
# define W8001_MAX_LENGTH 11
2009-09-04 04:22:03 +04:00
# define W8001_LEAD_MASK 0x80
# define W8001_LEAD_BYTE 0x80
# define W8001_TAB_MASK 0x40
# define W8001_TAB_BYTE 0x40
2008-11-20 00:58:50 +03:00
2009-09-04 04:22:03 +04:00
# define W8001_QUERY_PACKET 0x20
# define W8001_CMD_START '1'
# define W8001_CMD_QUERY '*'
2008-11-20 00:58:50 +03:00
struct w8001_coord {
u8 rdy ;
u8 tsw ;
u8 f1 ;
u8 f2 ;
u16 x ;
u16 y ;
u16 pen_pressure ;
u8 tilt_x ;
u8 tilt_y ;
} ;
/*
* Per - touchscreen data .
*/
struct w8001 {
struct input_dev * dev ;
struct serio * serio ;
struct completion cmd_done ;
int id ;
int idx ;
2009-09-04 04:22:03 +04:00
unsigned char response_type ;
unsigned char response [ W8001_MAX_LENGTH ] ;
2008-11-20 00:58:50 +03:00
unsigned char data [ W8001_MAX_LENGTH ] ;
char phys [ 32 ] ;
} ;
2009-09-04 04:22:03 +04:00
static void parse_data ( u8 * data , struct w8001_coord * coord )
2008-11-20 00:58:50 +03:00
{
2009-09-04 04:22:03 +04:00
memset ( coord , 0 , sizeof ( * coord ) ) ;
2008-11-20 00:58:50 +03:00
coord - > rdy = data [ 0 ] & 0x20 ;
coord - > tsw = data [ 0 ] & 0x01 ;
coord - > f1 = data [ 0 ] & 0x02 ;
coord - > f2 = data [ 0 ] & 0x04 ;
coord - > x = ( data [ 1 ] & 0x7F ) < < 9 ;
coord - > x | = ( data [ 2 ] & 0x7F ) < < 2 ;
coord - > x | = ( data [ 6 ] & 0x60 ) > > 5 ;
coord - > y = ( data [ 3 ] & 0x7F ) < < 9 ;
coord - > y | = ( data [ 4 ] & 0x7F ) < < 2 ;
coord - > y | = ( data [ 6 ] & 0x18 ) > > 3 ;
coord - > pen_pressure = data [ 5 ] & 0x7F ;
coord - > pen_pressure | = ( data [ 6 ] & 0x07 ) < < 7 ;
coord - > tilt_x = data [ 7 ] & 0x7F ;
coord - > tilt_y = data [ 8 ] & 0x7F ;
}
2009-09-04 04:22:03 +04:00
static irqreturn_t w8001_interrupt ( struct serio * serio ,
unsigned char data , unsigned int flags )
2008-11-20 00:58:50 +03:00
{
2009-09-04 04:22:03 +04:00
struct w8001 * w8001 = serio_get_drvdata ( serio ) ;
2008-11-20 00:58:50 +03:00
struct input_dev * dev = w8001 - > dev ;
struct w8001_coord coord ;
2009-09-04 04:22:03 +04:00
unsigned char tmp ;
2008-11-20 00:58:50 +03:00
w8001 - > data [ w8001 - > idx ] = data ;
switch ( w8001 - > idx + + ) {
case 0 :
if ( ( data & W8001_LEAD_MASK ) ! = W8001_LEAD_BYTE ) {
pr_debug ( " w8001: unsynchronized data: 0x%02x \n " , data ) ;
w8001 - > idx = 0 ;
}
break ;
2009-09-04 04:22:03 +04:00
2008-11-20 00:58:50 +03:00
case 8 :
tmp = w8001 - > data [ 0 ] & W8001_TAB_MASK ;
if ( unlikely ( tmp = = W8001_TAB_BYTE ) )
break ;
2009-09-04 04:22:03 +04:00
2008-11-20 00:58:50 +03:00
w8001 - > idx = 0 ;
parse_data ( w8001 - > data , & coord ) ;
input_report_abs ( dev , ABS_X , coord . x ) ;
input_report_abs ( dev , ABS_Y , coord . y ) ;
input_report_abs ( dev , ABS_PRESSURE , coord . pen_pressure ) ;
input_report_key ( dev , BTN_TOUCH , coord . tsw ) ;
input_sync ( dev ) ;
break ;
2009-09-04 04:22:03 +04:00
2008-11-20 00:58:50 +03:00
case 10 :
w8001 - > idx = 0 ;
2009-09-04 04:22:03 +04:00
memcpy ( w8001 - > response , w8001 - > data , W8001_MAX_LENGTH ) ;
w8001 - > response_type = W8001_QUERY_PACKET ;
2008-11-20 00:58:50 +03:00
complete ( & w8001 - > cmd_done ) ;
break ;
}
return IRQ_HANDLED ;
}
2009-09-04 04:22:03 +04:00
static int w8001_command ( struct w8001 * w8001 , unsigned char command ,
bool wait_response )
2008-11-20 00:58:50 +03:00
{
2009-09-04 04:22:03 +04:00
int rc ;
2008-11-20 00:58:50 +03:00
2009-09-04 04:22:03 +04:00
w8001 - > response_type = 0 ;
2008-11-20 00:58:50 +03:00
init_completion ( & w8001 - > cmd_done ) ;
2009-09-04 04:22:03 +04:00
rc = serio_write ( w8001 - > serio , command ) ;
if ( rc = = 0 & & wait_response ) {
2008-11-20 00:58:50 +03:00
2009-09-04 04:22:03 +04:00
wait_for_completion_timeout ( & w8001 - > cmd_done , HZ ) ;
if ( w8001 - > response_type ! = W8001_QUERY_PACKET )
rc = - EIO ;
2008-11-20 00:58:50 +03:00
}
return rc ;
}
static int w8001_setup ( struct w8001 * w8001 )
{
struct input_dev * dev = w8001 - > dev ;
2009-09-04 04:22:03 +04:00
struct w8001_coord coord ;
int error ;
2008-11-20 00:58:50 +03:00
2009-09-04 04:22:03 +04:00
error = w8001_command ( w8001 , W8001_CMD_QUERY , true ) ;
if ( error )
return error ;
2008-11-20 00:58:50 +03:00
2009-09-04 04:22:03 +04:00
parse_data ( w8001 - > response , & coord ) ;
2008-11-20 00:58:50 +03:00
input_set_abs_params ( dev , ABS_X , 0 , coord . x , 0 , 0 ) ;
input_set_abs_params ( dev , ABS_Y , 0 , coord . y , 0 , 0 ) ;
input_set_abs_params ( dev , ABS_PRESSURE , 0 , coord . pen_pressure , 0 , 0 ) ;
input_set_abs_params ( dev , ABS_TILT_X , 0 , coord . tilt_x , 0 , 0 ) ;
input_set_abs_params ( dev , ABS_TILT_Y , 0 , coord . tilt_y , 0 , 0 ) ;
2009-09-04 04:22:03 +04:00
return w8001_command ( w8001 , W8001_CMD_START , false ) ;
2008-11-20 00:58:50 +03:00
}
/*
* w8001_disconnect ( ) is the opposite of w8001_connect ( )
*/
static void w8001_disconnect ( struct serio * serio )
{
struct w8001 * w8001 = serio_get_drvdata ( serio ) ;
input_get_device ( w8001 - > dev ) ;
input_unregister_device ( w8001 - > dev ) ;
serio_close ( serio ) ;
serio_set_drvdata ( serio , NULL ) ;
input_put_device ( w8001 - > dev ) ;
kfree ( w8001 ) ;
}
/*
* w8001_connect ( ) is the routine that is called when someone adds a
* new serio device that supports the w8001 protocol and registers it as
* an input device .
*/
static int w8001_connect ( struct serio * serio , struct serio_driver * drv )
{
struct w8001 * w8001 ;
struct input_dev * input_dev ;
int err ;
w8001 = kzalloc ( sizeof ( struct w8001 ) , GFP_KERNEL ) ;
input_dev = input_allocate_device ( ) ;
if ( ! w8001 | | ! input_dev ) {
err = - ENOMEM ;
goto fail1 ;
}
w8001 - > serio = serio ;
w8001 - > id = serio - > id . id ;
w8001 - > dev = input_dev ;
init_completion ( & w8001 - > cmd_done ) ;
snprintf ( w8001 - > phys , sizeof ( w8001 - > phys ) , " %s/input0 " , serio - > phys ) ;
input_dev - > name = " Wacom W8001 Penabled Serial TouchScreen " ;
input_dev - > phys = w8001 - > phys ;
input_dev - > id . bustype = BUS_RS232 ;
input_dev - > id . vendor = SERIO_W8001 ;
input_dev - > id . product = w8001 - > id ;
input_dev - > id . version = 0x0100 ;
input_dev - > dev . parent = & serio - > dev ;
input_dev - > evbit [ 0 ] = BIT_MASK ( EV_KEY ) | BIT_MASK ( EV_ABS ) ;
input_dev - > keybit [ BIT_WORD ( BTN_TOUCH ) ] = BIT_MASK ( BTN_TOUCH ) ;
serio_set_drvdata ( serio , w8001 ) ;
err = serio_open ( serio , drv ) ;
if ( err )
goto fail2 ;
2009-09-04 04:22:03 +04:00
err = w8001_setup ( w8001 ) ;
if ( err )
2008-11-20 00:58:50 +03:00
goto fail3 ;
err = input_register_device ( w8001 - > dev ) ;
if ( err )
goto fail3 ;
return 0 ;
fail3 :
serio_close ( serio ) ;
fail2 :
serio_set_drvdata ( serio , NULL ) ;
fail1 :
input_free_device ( input_dev ) ;
kfree ( w8001 ) ;
return err ;
}
static struct serio_device_id w8001_serio_ids [ ] = {
{
. type = SERIO_RS232 ,
. proto = SERIO_W8001 ,
. id = SERIO_ANY ,
. extra = SERIO_ANY ,
} ,
{ 0 }
} ;
MODULE_DEVICE_TABLE ( serio , w8001_serio_ids ) ;
static struct serio_driver w8001_drv = {
. driver = {
. name = " w8001 " ,
} ,
. description = DRIVER_DESC ,
. id_table = w8001_serio_ids ,
. interrupt = w8001_interrupt ,
. connect = w8001_connect ,
. disconnect = w8001_disconnect ,
} ;
static int __init w8001_init ( void )
{
return serio_register_driver ( & w8001_drv ) ;
}
static void __exit w8001_exit ( void )
{
serio_unregister_driver ( & w8001_drv ) ;
}
module_init ( w8001_init ) ;
module_exit ( w8001_exit ) ;