2012-11-09 23:56:59 -08:00
/*
* STMicroelectronics STMPE811 Touchscreen Driver
2010-07-02 14:10:29 +02:00
*
* ( C ) 2010 Luotao Fu < l . fu @ pengutronix . de >
* All rights reserved .
*
* 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 .
*
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/interrupt.h>
# include <linux/device.h>
2012-11-22 23:42:12 -08:00
# include <linux/of.h>
2010-07-02 14:10:29 +02:00
# include <linux/platform_device.h>
# include <linux/input.h>
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/i2c.h>
# include <linux/workqueue.h>
# include <linux/mfd/stmpe.h>
/* Register layouts and functionalities are identical on all stmpexxx variants
* with touchscreen controller
*/
# define STMPE_REG_INT_STA 0x0B
# define STMPE_REG_ADC_CTRL1 0x20
# define STMPE_REG_ADC_CTRL2 0x21
# define STMPE_REG_TSC_CTRL 0x40
# define STMPE_REG_TSC_CFG 0x41
# define STMPE_REG_FIFO_TH 0x4A
# define STMPE_REG_FIFO_STA 0x4B
# define STMPE_REG_FIFO_SIZE 0x4C
# define STMPE_REG_TSC_DATA_XYZ 0x52
# define STMPE_REG_TSC_FRACTION_Z 0x56
# define STMPE_REG_TSC_I_DRIVE 0x58
# define OP_MOD_XYZ 0
# define STMPE_TSC_CTRL_TSC_EN (1<<0)
# define STMPE_FIFO_STA_RESET (1<<0)
# define STMPE_IRQ_TOUCH_DET 0
# define SAMPLE_TIME(x) ((x & 0xf) << 4)
# define MOD_12B(x) ((x & 0x1) << 3)
# define REF_SEL(x) ((x & 0x1) << 1)
# define ADC_FREQ(x) (x & 0x3)
# define AVE_CTRL(x) ((x & 0x3) << 6)
# define DET_DELAY(x) ((x & 0x7) << 3)
# define SETTLING(x) (x & 0x7)
# define FRACTION_Z(x) (x & 0x7)
# define I_DRIVE(x) (x & 0x1)
# define OP_MODE(x) ((x & 0x7) << 1)
# define STMPE_TS_NAME "stmpe-ts"
# define XY_MASK 0xfff
2016-02-22 11:37:49 -08:00
/**
* struct stmpe_touch - stmpe811 touch screen controller state
* @ stmpe : pointer back to STMPE MFD container
* @ idev : registered input device
* @ work : a work item used to scan the device
* @ dev : a pointer back to the MFD cell struct device *
* @ sample_time : ADC converstion time in number of clock .
* ( 0 - > 36 clocks , 1 - > 44 clocks , 2 - > 56 clocks , 3 - > 64 clocks ,
* 4 - > 80 clocks , 5 - > 96 clocks , 6 - > 144 clocks ) ,
* recommended is 4.
* @ mod_12b : ADC Bit mode ( 0 - > 10 bit ADC , 1 - > 12 bit ADC )
* @ ref_sel : ADC reference source
* ( 0 - > internal reference , 1 - > external reference )
* @ adc_freq : ADC Clock speed
* ( 0 - > 1.625 MHz , 1 - > 3.25 MHz , 2 | | 3 - > 6.5 MHz )
* @ ave_ctrl : Sample average control
* ( 0 - > 1 sample , 1 - > 2 samples , 2 - > 4 samples , 3 - > 8 samples )
* @ touch_det_delay : Touch detect interrupt delay
* ( 0 - > 10 us , 1 - > 50 us , 2 - > 100 us , 3 - > 500 us ,
* 4 - > 1 ms , 5 - > 5 ms , 6 - > 10 ms , 7 - > 50 ms )
* recommended is 3
* @ settling : Panel driver settling time
* ( 0 - > 10 us , 1 - > 100 us , 2 - > 500 us , 3 - > 1 ms ,
* 4 - > 5 ms , 5 - > 10 ms , 6 for 50 ms , 7 - > 100 ms )
* recommended is 2
* @ fraction_z : Length of the fractional part in z
* ( fraction_z ( [ 0. .7 ] ) = Count of the fractional part )
* recommended is 7
* @ i_drive : current limit value of the touchscreen drivers
* ( 0 - > 20 mA typical 35 mA max , 1 - > 50 mA typical 80 mA max )
*/
2010-07-02 14:10:29 +02:00
struct stmpe_touch {
struct stmpe * stmpe ;
struct input_dev * idev ;
struct delayed_work work ;
struct device * dev ;
u8 sample_time ;
u8 mod_12b ;
u8 ref_sel ;
u8 adc_freq ;
u8 ave_ctrl ;
u8 touch_det_delay ;
u8 settling ;
u8 fraction_z ;
u8 i_drive ;
} ;
static int __stmpe_reset_fifo ( struct stmpe * stmpe )
{
int ret ;
ret = stmpe_set_bits ( stmpe , STMPE_REG_FIFO_STA ,
STMPE_FIFO_STA_RESET , STMPE_FIFO_STA_RESET ) ;
if ( ret )
return ret ;
return stmpe_set_bits ( stmpe , STMPE_REG_FIFO_STA ,
STMPE_FIFO_STA_RESET , 0 ) ;
}
static void stmpe_work ( struct work_struct * work )
{
int int_sta ;
u32 timeout = 40 ;
struct stmpe_touch * ts =
container_of ( work , struct stmpe_touch , work . work ) ;
int_sta = stmpe_reg_read ( ts - > stmpe , STMPE_REG_INT_STA ) ;
/*
* touch_det sometimes get desasserted or just get stuck . This appears
* to be a silicon bug , We still have to clearify this with the
* manufacture . As a workaround We release the key anyway if the
* touch_det keeps coming in after 4 ms , while the FIFO contains no value
* during the whole time .
*/
while ( ( int_sta & ( 1 < < STMPE_IRQ_TOUCH_DET ) ) & & ( timeout > 0 ) ) {
timeout - - ;
int_sta = stmpe_reg_read ( ts - > stmpe , STMPE_REG_INT_STA ) ;
udelay ( 100 ) ;
}
/* reset the FIFO before we report release event */
__stmpe_reset_fifo ( ts - > stmpe ) ;
input_report_abs ( ts - > idev , ABS_PRESSURE , 0 ) ;
2012-12-16 23:10:56 -08:00
input_report_key ( ts - > idev , BTN_TOUCH , 0 ) ;
2010-07-02 14:10:29 +02:00
input_sync ( ts - > idev ) ;
}
static irqreturn_t stmpe_ts_handler ( int irq , void * data )
{
u8 data_set [ 4 ] ;
int x , y , z ;
struct stmpe_touch * ts = data ;
/*
* Cancel scheduled polling for release if we have new value
* available . Wait if the polling is already running .
*/
cancel_delayed_work_sync ( & ts - > work ) ;
/*
* The FIFO sometimes just crashes and stops generating interrupts . This
* appears to be a silicon bug . We still have to clearify this with
* the manufacture . As a workaround we disable the TSC while we are
* collecting data and flush the FIFO after reading
*/
stmpe_set_bits ( ts - > stmpe , STMPE_REG_TSC_CTRL ,
STMPE_TSC_CTRL_TSC_EN , 0 ) ;
stmpe_block_read ( ts - > stmpe , STMPE_REG_TSC_DATA_XYZ , 4 , data_set ) ;
x = ( data_set [ 0 ] < < 4 ) | ( data_set [ 1 ] > > 4 ) ;
y = ( ( data_set [ 1 ] & 0xf ) < < 8 ) | data_set [ 2 ] ;
z = data_set [ 3 ] ;
input_report_abs ( ts - > idev , ABS_X , x ) ;
input_report_abs ( ts - > idev , ABS_Y , y ) ;
input_report_abs ( ts - > idev , ABS_PRESSURE , z ) ;
2012-12-16 23:10:56 -08:00
input_report_key ( ts - > idev , BTN_TOUCH , 1 ) ;
2010-07-02 14:10:29 +02:00
input_sync ( ts - > idev ) ;
/* flush the FIFO after we have read out our values. */
__stmpe_reset_fifo ( ts - > stmpe ) ;
/* reenable the tsc */
stmpe_set_bits ( ts - > stmpe , STMPE_REG_TSC_CTRL ,
STMPE_TSC_CTRL_TSC_EN , STMPE_TSC_CTRL_TSC_EN ) ;
/* start polling for touch_det to detect release */
Input: smtpe-ts - wait 50mS until polling for pen-up
Wait a little bit longer, 50mS instead of 20mS, until the driver starts
polling for pen-up. The problematic behavior before this patch is applied
is as follows. The behavior was observed on the STMPE610QTR controller.
Upon a physical pen-down event, the touchscreen reports one set of x-y-p
coordinates and a pen-down event. After that, the pen-up polling is
triggered and since the controller is not ready yet, the polling mistakenly
detects a pen-up event while the physical state is still such that the pen
is down on the touch surface.
The pen-up handling flushes the controller FIFO, so after that, all the
samples in the controller are discarded. The controller becomes ready
shortly after this bogus pen-up handling and does generate again a pen-down
interrupt. This time, the controller contains x-y-p samples which all read
as zero. Since pressure value is zero, this set of samples is effectively
ignored by userland.
In the end, the driver just bounces between pen-down and bogus pen-up
handling, generating no useful results. Fix this by giving the controller a
bit more time before polling it for pen-up.
Signed-off-by: Marek Vasut <marex@denx.de>
Reviewed-by: Viresh Kumar <viresh.kumar@linaro.org>
Signed-off-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
2015-05-07 22:25:49 -07:00
schedule_delayed_work ( & ts - > work , msecs_to_jiffies ( 50 ) ) ;
2010-07-02 14:10:29 +02:00
return IRQ_HANDLED ;
}
2012-11-23 21:38:25 -08:00
static int stmpe_init_hw ( struct stmpe_touch * ts )
2010-07-02 14:10:29 +02:00
{
int ret ;
u8 adc_ctrl1 , adc_ctrl1_mask , tsc_cfg , tsc_cfg_mask ;
struct stmpe * stmpe = ts - > stmpe ;
struct device * dev = ts - > dev ;
ret = stmpe_enable ( stmpe , STMPE_BLOCK_TOUCHSCREEN | STMPE_BLOCK_ADC ) ;
if ( ret ) {
dev_err ( dev , " Could not enable clock for ADC and TS \n " ) ;
return ret ;
}
adc_ctrl1 = SAMPLE_TIME ( ts - > sample_time ) | MOD_12B ( ts - > mod_12b ) |
REF_SEL ( ts - > ref_sel ) ;
adc_ctrl1_mask = SAMPLE_TIME ( 0xff ) | MOD_12B ( 0xff ) | REF_SEL ( 0xff ) ;
ret = stmpe_set_bits ( stmpe , STMPE_REG_ADC_CTRL1 ,
adc_ctrl1_mask , adc_ctrl1 ) ;
if ( ret ) {
dev_err ( dev , " Could not setup ADC \n " ) ;
return ret ;
}
ret = stmpe_set_bits ( stmpe , STMPE_REG_ADC_CTRL2 ,
ADC_FREQ ( 0xff ) , ADC_FREQ ( ts - > adc_freq ) ) ;
if ( ret ) {
dev_err ( dev , " Could not setup ADC \n " ) ;
return ret ;
}
tsc_cfg = AVE_CTRL ( ts - > ave_ctrl ) | DET_DELAY ( ts - > touch_det_delay ) |
SETTLING ( ts - > settling ) ;
tsc_cfg_mask = AVE_CTRL ( 0xff ) | DET_DELAY ( 0xff ) | SETTLING ( 0xff ) ;
ret = stmpe_set_bits ( stmpe , STMPE_REG_TSC_CFG , tsc_cfg_mask , tsc_cfg ) ;
if ( ret ) {
dev_err ( dev , " Could not config touch \n " ) ;
return ret ;
}
ret = stmpe_set_bits ( stmpe , STMPE_REG_TSC_FRACTION_Z ,
FRACTION_Z ( 0xff ) , FRACTION_Z ( ts - > fraction_z ) ) ;
if ( ret ) {
dev_err ( dev , " Could not config touch \n " ) ;
return ret ;
}
ret = stmpe_set_bits ( stmpe , STMPE_REG_TSC_I_DRIVE ,
I_DRIVE ( 0xff ) , I_DRIVE ( ts - > i_drive ) ) ;
if ( ret ) {
dev_err ( dev , " Could not config touch \n " ) ;
return ret ;
}
/* set FIFO to 1 for single point reading */
ret = stmpe_reg_write ( stmpe , STMPE_REG_FIFO_TH , 1 ) ;
if ( ret ) {
dev_err ( dev , " Could not set FIFO \n " ) ;
return ret ;
}
ret = stmpe_set_bits ( stmpe , STMPE_REG_TSC_CTRL ,
OP_MODE ( 0xff ) , OP_MODE ( OP_MOD_XYZ ) ) ;
if ( ret ) {
dev_err ( dev , " Could not set mode \n " ) ;
return ret ;
}
return 0 ;
}
static int stmpe_ts_open ( struct input_dev * dev )
{
struct stmpe_touch * ts = input_get_drvdata ( dev ) ;
int ret = 0 ;
ret = __stmpe_reset_fifo ( ts - > stmpe ) ;
if ( ret )
return ret ;
return stmpe_set_bits ( ts - > stmpe , STMPE_REG_TSC_CTRL ,
STMPE_TSC_CTRL_TSC_EN , STMPE_TSC_CTRL_TSC_EN ) ;
}
static void stmpe_ts_close ( struct input_dev * dev )
{
struct stmpe_touch * ts = input_get_drvdata ( dev ) ;
cancel_delayed_work_sync ( & ts - > work ) ;
stmpe_set_bits ( ts - > stmpe , STMPE_REG_TSC_CTRL ,
STMPE_TSC_CTRL_TSC_EN , 0 ) ;
}
2012-11-22 23:42:12 -08:00
static void stmpe_ts_get_platform_info ( struct platform_device * pdev ,
struct stmpe_touch * ts )
2010-07-02 14:10:29 +02:00
{
2012-11-22 23:42:12 -08:00
struct device_node * np = pdev - > dev . of_node ;
2015-05-22 13:44:33 -07:00
u32 val ;
2012-11-22 23:42:12 -08:00
2015-05-22 13:44:33 -07:00
if ( np ) {
2012-11-22 23:42:12 -08:00
if ( ! of_property_read_u32 ( np , " st,sample-time " , & val ) )
ts - > sample_time = val ;
if ( ! of_property_read_u32 ( np , " st,mod-12b " , & val ) )
ts - > mod_12b = val ;
if ( ! of_property_read_u32 ( np , " st,ref-sel " , & val ) )
ts - > ref_sel = val ;
if ( ! of_property_read_u32 ( np , " st,adc-freq " , & val ) )
ts - > adc_freq = val ;
if ( ! of_property_read_u32 ( np , " st,ave-ctrl " , & val ) )
ts - > ave_ctrl = val ;
if ( ! of_property_read_u32 ( np , " st,touch-det-delay " , & val ) )
ts - > touch_det_delay = val ;
if ( ! of_property_read_u32 ( np , " st,settling " , & val ) )
ts - > settling = val ;
if ( ! of_property_read_u32 ( np , " st,fraction-z " , & val ) )
ts - > fraction_z = val ;
if ( ! of_property_read_u32 ( np , " st,i-drive " , & val ) )
ts - > i_drive = val ;
}
}
2012-11-23 21:38:25 -08:00
static int stmpe_input_probe ( struct platform_device * pdev )
2012-11-22 23:42:12 -08:00
{
2015-05-22 13:44:33 -07:00
struct stmpe * stmpe = dev_get_drvdata ( pdev - > dev . parent ) ;
2010-07-02 14:10:29 +02:00
struct stmpe_touch * ts ;
struct input_dev * idev ;
2012-11-09 23:56:59 -08:00
int error ;
2010-07-15 22:43:54 +04:00
int ts_irq ;
2010-07-02 14:10:29 +02:00
ts_irq = platform_get_irq_byname ( pdev , " FIFO_TH " ) ;
if ( ts_irq < 0 )
return ts_irq ;
2012-11-09 23:56:59 -08:00
ts = devm_kzalloc ( & pdev - > dev , sizeof ( * ts ) , GFP_KERNEL ) ;
if ( ! ts )
return - ENOMEM ;
2010-07-02 14:10:29 +02:00
2012-11-09 23:56:59 -08:00
idev = devm_input_allocate_device ( & pdev - > dev ) ;
if ( ! idev )
return - ENOMEM ;
2010-07-02 14:10:29 +02:00
platform_set_drvdata ( pdev , ts ) ;
2015-05-22 13:44:33 -07:00
ts - > stmpe = stmpe ;
2010-07-02 14:10:29 +02:00
ts - > idev = idev ;
ts - > dev = & pdev - > dev ;
2012-11-22 23:42:12 -08:00
stmpe_ts_get_platform_info ( pdev , ts ) ;
2010-07-02 14:10:29 +02:00
INIT_DELAYED_WORK ( & ts - > work , stmpe_work ) ;
2012-11-09 23:56:59 -08:00
error = devm_request_threaded_irq ( & pdev - > dev , ts_irq ,
NULL , stmpe_ts_handler ,
IRQF_ONESHOT , STMPE_TS_NAME , ts ) ;
if ( error ) {
2010-07-02 14:10:29 +02:00
dev_err ( & pdev - > dev , " Failed to request IRQ %d \n " , ts_irq ) ;
2012-11-09 23:56:59 -08:00
return error ;
2010-07-02 14:10:29 +02:00
}
2012-11-09 23:56:59 -08:00
error = stmpe_init_hw ( ts ) ;
if ( error )
return error ;
2010-07-02 14:10:29 +02:00
idev - > name = STMPE_TS_NAME ;
2012-11-10 00:08:20 -08:00
idev - > phys = STMPE_TS_NAME " /input0 " ;
2010-07-02 14:10:29 +02:00
idev - > id . bustype = BUS_I2C ;
idev - > open = stmpe_ts_open ;
idev - > close = stmpe_ts_close ;
input_set_drvdata ( idev , ts ) ;
2015-05-22 13:44:33 -07:00
input_set_capability ( idev , EV_KEY , BTN_TOUCH ) ;
2010-07-02 14:10:29 +02:00
input_set_abs_params ( idev , ABS_X , 0 , XY_MASK , 0 , 0 ) ;
input_set_abs_params ( idev , ABS_Y , 0 , XY_MASK , 0 , 0 ) ;
input_set_abs_params ( idev , ABS_PRESSURE , 0x0 , 0xff , 0 , 0 ) ;
2012-11-09 23:56:59 -08:00
error = input_register_device ( idev ) ;
if ( error ) {
2010-07-02 14:10:29 +02:00
dev_err ( & pdev - > dev , " Could not register input device \n " ) ;
2012-11-09 23:56:59 -08:00
return error ;
2010-07-02 14:10:29 +02:00
}
2012-11-09 23:56:59 -08:00
return 0 ;
2010-07-02 14:10:29 +02:00
}
2012-11-23 21:50:47 -08:00
static int stmpe_ts_remove ( struct platform_device * pdev )
2010-07-02 14:10:29 +02:00
{
struct stmpe_touch * ts = platform_get_drvdata ( pdev ) ;
stmpe_disable ( ts - > stmpe , STMPE_BLOCK_TOUCHSCREEN ) ;
return 0 ;
}
static struct platform_driver stmpe_ts_driver = {
. driver = {
2015-05-22 13:09:11 -07:00
. name = STMPE_TS_NAME ,
} ,
2010-07-02 14:10:29 +02:00
. probe = stmpe_input_probe ,
2012-11-23 21:27:39 -08:00
. remove = stmpe_ts_remove ,
2010-07-02 14:10:29 +02:00
} ;
2011-11-29 11:14:13 -08:00
module_platform_driver ( stmpe_ts_driver ) ;
2010-07-02 14:10:29 +02:00
2015-05-22 13:09:11 -07:00
static const struct of_device_id stmpe_ts_ids [ ] = {
{ . compatible = " st,stmpe-ts " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , stmpe_ts_ids ) ;
2010-07-02 14:10:29 +02:00
MODULE_AUTHOR ( " Luotao Fu <l.fu@pengutronix.de> " ) ;
MODULE_DESCRIPTION ( " STMPEXXX touchscreen driver " ) ;
MODULE_LICENSE ( " GPL " ) ;