2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2017-10-14 10:10:53 -07:00
/*
* Driver for I2C connected EETI EXC3000 multiple touch controller
*
* Copyright ( C ) 2017 Ahmet Inan < inan @ distec . de >
*
* minimal implementation based on egalax_ts . c and egalax_i2c . c
*/
# include <linux/bitops.h>
# include <linux/device.h>
# include <linux/i2c.h>
# include <linux/input.h>
# include <linux/input/mt.h>
# include <linux/input/touchscreen.h>
# include <linux/interrupt.h>
# include <linux/module.h>
# include <linux/of.h>
2020-08-06 16:00:43 -07:00
# include <linux/sizes.h>
2017-10-14 10:10:53 -07:00
# include <linux/timer.h>
# include <asm/unaligned.h>
# define EXC3000_NUM_SLOTS 10
# define EXC3000_SLOTS_PER_FRAME 5
# define EXC3000_LEN_FRAME 66
# define EXC3000_LEN_POINT 10
2020-08-06 16:00:43 -07:00
# define EXC3000_MT1_EVENT 0x06
# define EXC3000_MT2_EVENT 0x18
2017-10-14 10:10:53 -07:00
# define EXC3000_TIMEOUT_MS 100
2020-08-06 16:00:43 -07:00
static const struct i2c_device_id exc3000_id [ ] ;
struct eeti_dev_info {
const char * name ;
int max_xy ;
} ;
enum eeti_dev_id {
EETI_EXC3000 ,
EETI_EXC80H60 ,
EETI_EXC80H84 ,
} ;
static struct eeti_dev_info exc3000_info [ ] = {
[ EETI_EXC3000 ] = {
. name = " EETI EXC3000 Touch Screen " ,
. max_xy = SZ_4K - 1 ,
} ,
[ EETI_EXC80H60 ] = {
. name = " EETI EXC80H60 Touch Screen " ,
. max_xy = SZ_16K - 1 ,
} ,
[ EETI_EXC80H84 ] = {
. name = " EETI EXC80H84 Touch Screen " ,
. max_xy = SZ_16K - 1 ,
} ,
} ;
2017-10-14 10:10:53 -07:00
struct exc3000_data {
struct i2c_client * client ;
2020-08-06 16:00:43 -07:00
const struct eeti_dev_info * info ;
2017-10-14 10:10:53 -07:00
struct input_dev * input ;
struct touchscreen_properties prop ;
struct timer_list timer ;
u8 buf [ 2 * EXC3000_LEN_FRAME ] ;
} ;
static void exc3000_report_slots ( struct input_dev * input ,
struct touchscreen_properties * prop ,
const u8 * buf , int num )
{
for ( ; num - - ; buf + = EXC3000_LEN_POINT ) {
if ( buf [ 0 ] & BIT ( 0 ) ) {
input_mt_slot ( input , buf [ 1 ] ) ;
input_mt_report_slot_state ( input , MT_TOOL_FINGER , true ) ;
touchscreen_report_pos ( input , prop ,
get_unaligned_le16 ( buf + 2 ) ,
get_unaligned_le16 ( buf + 4 ) ,
true ) ;
}
}
}
static void exc3000_timer ( struct timer_list * t )
{
struct exc3000_data * data = from_timer ( data , t , timer ) ;
input_mt_sync_frame ( data - > input ) ;
input_sync ( data - > input ) ;
}
2020-08-06 16:00:43 -07:00
static int exc3000_read_frame ( struct exc3000_data * data , u8 * buf )
2017-10-14 10:10:53 -07:00
{
2020-08-06 16:00:43 -07:00
struct i2c_client * client = data - > client ;
u8 expected_event = EXC3000_MT1_EVENT ;
2017-10-14 10:10:53 -07:00
int ret ;
2020-08-06 16:00:43 -07:00
if ( data - > info - > max_xy = = SZ_16K - 1 )
expected_event = EXC3000_MT2_EVENT ;
2017-10-14 10:10:53 -07:00
ret = i2c_master_send ( client , " ' " , 2 ) ;
if ( ret < 0 )
return ret ;
if ( ret ! = 2 )
return - EIO ;
ret = i2c_master_recv ( client , buf , EXC3000_LEN_FRAME ) ;
if ( ret < 0 )
return ret ;
if ( ret ! = EXC3000_LEN_FRAME )
return - EIO ;
2020-08-06 16:00:43 -07:00
if ( get_unaligned_le16 ( buf ) ! = EXC3000_LEN_FRAME )
return - EINVAL ;
if ( buf [ 2 ] ! = expected_event )
2017-10-14 10:10:53 -07:00
return - EINVAL ;
return 0 ;
}
2020-08-06 16:00:43 -07:00
static int exc3000_read_data ( struct exc3000_data * data ,
2017-10-14 10:10:53 -07:00
u8 * buf , int * n_slots )
{
int error ;
2020-08-06 16:00:43 -07:00
error = exc3000_read_frame ( data , buf ) ;
2017-10-14 10:10:53 -07:00
if ( error )
return error ;
* n_slots = buf [ 3 ] ;
if ( ! * n_slots | | * n_slots > EXC3000_NUM_SLOTS )
return - EINVAL ;
if ( * n_slots > EXC3000_SLOTS_PER_FRAME ) {
/* Read 2nd frame to get the rest of the contacts. */
2020-08-06 16:00:43 -07:00
error = exc3000_read_frame ( data , buf + EXC3000_LEN_FRAME ) ;
2017-10-14 10:10:53 -07:00
if ( error )
return error ;
/* 2nd chunk must have number of contacts set to 0. */
if ( buf [ EXC3000_LEN_FRAME + 3 ] ! = 0 )
return - EINVAL ;
}
return 0 ;
}
static irqreturn_t exc3000_interrupt ( int irq , void * dev_id )
{
struct exc3000_data * data = dev_id ;
struct input_dev * input = data - > input ;
u8 * buf = data - > buf ;
int slots , total_slots ;
int error ;
2020-08-06 16:00:43 -07:00
error = exc3000_read_data ( data , buf , & total_slots ) ;
2017-10-14 10:10:53 -07:00
if ( error ) {
/* Schedule a timer to release "stuck" contacts */
mod_timer ( & data - > timer ,
jiffies + msecs_to_jiffies ( EXC3000_TIMEOUT_MS ) ) ;
goto out ;
}
/*
* We read full state successfully , no contacts will be " stuck " .
*/
del_timer_sync ( & data - > timer ) ;
while ( total_slots > 0 ) {
slots = min ( total_slots , EXC3000_SLOTS_PER_FRAME ) ;
exc3000_report_slots ( input , & data - > prop , buf + 4 , slots ) ;
total_slots - = slots ;
buf + = EXC3000_LEN_FRAME ;
}
input_mt_sync_frame ( input ) ;
input_sync ( input ) ;
out :
return IRQ_HANDLED ;
}
2020-05-20 10:30:55 -07:00
static int exc3000_probe ( struct i2c_client * client )
2017-10-14 10:10:53 -07:00
{
struct exc3000_data * data ;
struct input_dev * input ;
2020-08-06 16:00:43 -07:00
int error , max_xy ;
2017-10-14 10:10:53 -07:00
data = devm_kzalloc ( & client - > dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
data - > client = client ;
2020-08-06 16:00:43 -07:00
data - > info = device_get_match_data ( & client - > dev ) ;
if ( ! data - > info ) {
enum eeti_dev_id eeti_dev_id =
i2c_match_id ( exc3000_id , client ) - > driver_data ;
data - > info = & exc3000_info [ eeti_dev_id ] ;
}
2017-10-14 10:10:53 -07:00
timer_setup ( & data - > timer , exc3000_timer , 0 ) ;
input = devm_input_allocate_device ( & client - > dev ) ;
if ( ! input )
return - ENOMEM ;
data - > input = input ;
2020-08-06 16:00:43 -07:00
input - > name = data - > info - > name ;
2017-10-14 10:10:53 -07:00
input - > id . bustype = BUS_I2C ;
2020-08-06 16:00:43 -07:00
max_xy = data - > info - > max_xy ;
input_set_abs_params ( input , ABS_MT_POSITION_X , 0 , max_xy , 0 , 0 ) ;
input_set_abs_params ( input , ABS_MT_POSITION_Y , 0 , max_xy , 0 , 0 ) ;
2017-10-14 10:10:53 -07:00
touchscreen_parse_properties ( input , true , & data - > prop ) ;
error = input_mt_init_slots ( input , EXC3000_NUM_SLOTS ,
INPUT_MT_DIRECT | INPUT_MT_DROP_UNUSED ) ;
if ( error )
return error ;
error = input_register_device ( input ) ;
if ( error )
return error ;
error = devm_request_threaded_irq ( & client - > dev , client - > irq ,
NULL , exc3000_interrupt , IRQF_ONESHOT ,
client - > name , data ) ;
if ( error )
return error ;
return 0 ;
}
static const struct i2c_device_id exc3000_id [ ] = {
2020-08-06 16:00:43 -07:00
{ " exc3000 " , EETI_EXC3000 } ,
{ " exc80h60 " , EETI_EXC80H60 } ,
{ " exc80h84 " , EETI_EXC80H84 } ,
2017-10-14 10:10:53 -07:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , exc3000_id ) ;
# ifdef CONFIG_OF
static const struct of_device_id exc3000_of_match [ ] = {
2020-08-06 16:00:43 -07:00
{ . compatible = " eeti,exc3000 " , . data = & exc3000_info [ EETI_EXC3000 ] } ,
{ . compatible = " eeti,exc80h60 " , . data = & exc3000_info [ EETI_EXC80H60 ] } ,
{ . compatible = " eeti,exc80h84 " , . data = & exc3000_info [ EETI_EXC80H84 ] } ,
2017-10-14 10:10:53 -07:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , exc3000_of_match ) ;
# endif
static struct i2c_driver exc3000_driver = {
. driver = {
. name = " exc3000 " ,
. of_match_table = of_match_ptr ( exc3000_of_match ) ,
} ,
. id_table = exc3000_id ,
2020-05-20 10:30:55 -07:00
. probe_new = exc3000_probe ,
2017-10-14 10:10:53 -07:00
} ;
module_i2c_driver ( exc3000_driver ) ;
MODULE_AUTHOR ( " Ahmet Inan <inan@distec.de> " ) ;
MODULE_DESCRIPTION ( " I2C connected EETI EXC3000 multiple touch controller driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;