2015-09-05 11:31:21 -07:00
/*
* Freescale i . MX6UL touchscreen controller driver
*
* Copyright ( C ) 2015 Freescale Semiconductor , Inc .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/gpio/consumer.h>
# include <linux/input.h>
# include <linux/slab.h>
# include <linux/completion.h>
# include <linux/delay.h>
# include <linux/of.h>
# include <linux/interrupt.h>
# include <linux/platform_device.h>
# include <linux/clk.h>
# include <linux/io.h>
/* ADC configuration registers field define */
# define ADC_AIEN (0x1 << 7)
# define ADC_CONV_DISABLE 0x1F
# define ADC_CAL (0x1 << 7)
# define ADC_CALF 0x2
# define ADC_12BIT_MODE (0x2 << 2)
# define ADC_IPG_CLK 0x00
# define ADC_CLK_DIV_8 (0x03 << 5)
# define ADC_SHORT_SAMPLE_MODE (0x0 << 4)
# define ADC_HARDWARE_TRIGGER (0x1 << 13)
# define SELECT_CHANNEL_4 0x04
# define SELECT_CHANNEL_1 0x01
# define DISABLE_CONVERSION_INT (0x0 << 7)
/* ADC registers */
# define REG_ADC_HC0 0x00
# define REG_ADC_HC1 0x04
# define REG_ADC_HC2 0x08
# define REG_ADC_HC3 0x0C
# define REG_ADC_HC4 0x10
# define REG_ADC_HS 0x14
# define REG_ADC_R0 0x18
# define REG_ADC_CFG 0x2C
# define REG_ADC_GC 0x30
# define REG_ADC_GS 0x34
# define ADC_TIMEOUT msecs_to_jiffies(100)
/* TSC registers */
# define REG_TSC_BASIC_SETING 0x00
# define REG_TSC_PRE_CHARGE_TIME 0x10
# define REG_TSC_FLOW_CONTROL 0x20
# define REG_TSC_MEASURE_VALUE 0x30
# define REG_TSC_INT_EN 0x40
# define REG_TSC_INT_SIG_EN 0x50
# define REG_TSC_INT_STATUS 0x60
# define REG_TSC_DEBUG_MODE 0x70
# define REG_TSC_DEBUG_MODE2 0x80
/* TSC configuration registers field define */
# define DETECT_4_WIRE_MODE (0x0 << 4)
# define AUTO_MEASURE 0x1
# define MEASURE_SIGNAL 0x1
# define DETECT_SIGNAL (0x1 << 4)
# define VALID_SIGNAL (0x1 << 8)
# define MEASURE_INT_EN 0x1
# define MEASURE_SIG_EN 0x1
# define VALID_SIG_EN (0x1 << 8)
# define DE_GLITCH_2 (0x2 << 29)
# define START_SENSE (0x1 << 12)
# define TSC_DISABLE (0x1 << 16)
# define DETECT_MODE 0x2
struct imx6ul_tsc {
struct device * dev ;
struct input_dev * input ;
void __iomem * tsc_regs ;
void __iomem * adc_regs ;
struct clk * tsc_clk ;
struct clk * adc_clk ;
struct gpio_desc * xnur_gpio ;
int measure_delay_time ;
int pre_charge_time ;
struct completion completion ;
} ;
/*
* TSC module need ADC to get the measure value . So
* before config TSC , we should initialize ADC module .
*/
2015-09-14 10:36:35 -07:00
static int imx6ul_adc_init ( struct imx6ul_tsc * tsc )
2015-09-05 11:31:21 -07:00
{
int adc_hc = 0 ;
int adc_gc ;
int adc_gs ;
int adc_cfg ;
int timeout ;
reinit_completion ( & tsc - > completion ) ;
adc_cfg = readl ( tsc - > adc_regs + REG_ADC_CFG ) ;
adc_cfg | = ADC_12BIT_MODE | ADC_IPG_CLK ;
adc_cfg | = ADC_CLK_DIV_8 | ADC_SHORT_SAMPLE_MODE ;
adc_cfg & = ~ ADC_HARDWARE_TRIGGER ;
writel ( adc_cfg , tsc - > adc_regs + REG_ADC_CFG ) ;
/* enable calibration interrupt */
adc_hc | = ADC_AIEN ;
adc_hc | = ADC_CONV_DISABLE ;
writel ( adc_hc , tsc - > adc_regs + REG_ADC_HC0 ) ;
/* start ADC calibration */
adc_gc = readl ( tsc - > adc_regs + REG_ADC_GC ) ;
adc_gc | = ADC_CAL ;
writel ( adc_gc , tsc - > adc_regs + REG_ADC_GC ) ;
timeout = wait_for_completion_timeout
( & tsc - > completion , ADC_TIMEOUT ) ;
2015-09-14 10:36:35 -07:00
if ( timeout = = 0 ) {
2015-09-05 11:31:21 -07:00
dev_err ( tsc - > dev , " Timeout for adc calibration \n " ) ;
2015-09-14 10:36:35 -07:00
return - ETIMEDOUT ;
}
2015-09-05 11:31:21 -07:00
adc_gs = readl ( tsc - > adc_regs + REG_ADC_GS ) ;
2015-09-14 10:36:35 -07:00
if ( adc_gs & ADC_CALF ) {
2015-09-05 11:31:21 -07:00
dev_err ( tsc - > dev , " ADC calibration failed \n " ) ;
2015-09-14 10:36:35 -07:00
return - EINVAL ;
}
2015-09-05 11:31:21 -07:00
/* TSC need the ADC work in hardware trigger */
adc_cfg = readl ( tsc - > adc_regs + REG_ADC_CFG ) ;
adc_cfg | = ADC_HARDWARE_TRIGGER ;
writel ( adc_cfg , tsc - > adc_regs + REG_ADC_CFG ) ;
2015-09-14 10:36:35 -07:00
return 0 ;
2015-09-05 11:31:21 -07:00
}
/*
* This is a TSC workaround . Currently TSC misconnect two
* ADC channels , this function remap channel configure for
* hardware trigger .
*/
static void imx6ul_tsc_channel_config ( struct imx6ul_tsc * tsc )
{
int adc_hc0 , adc_hc1 , adc_hc2 , adc_hc3 , adc_hc4 ;
adc_hc0 = DISABLE_CONVERSION_INT ;
writel ( adc_hc0 , tsc - > adc_regs + REG_ADC_HC0 ) ;
adc_hc1 = DISABLE_CONVERSION_INT | SELECT_CHANNEL_4 ;
writel ( adc_hc1 , tsc - > adc_regs + REG_ADC_HC1 ) ;
adc_hc2 = DISABLE_CONVERSION_INT ;
writel ( adc_hc2 , tsc - > adc_regs + REG_ADC_HC2 ) ;
adc_hc3 = DISABLE_CONVERSION_INT | SELECT_CHANNEL_1 ;
writel ( adc_hc3 , tsc - > adc_regs + REG_ADC_HC3 ) ;
adc_hc4 = DISABLE_CONVERSION_INT ;
writel ( adc_hc4 , tsc - > adc_regs + REG_ADC_HC4 ) ;
}
/*
* TSC setting , confige the pre - charge time and measure delay time .
* different touch screen may need different pre - charge time and
* measure delay time .
*/
static void imx6ul_tsc_set ( struct imx6ul_tsc * tsc )
{
int basic_setting = 0 ;
int start ;
basic_setting | = tsc - > measure_delay_time < < 8 ;
basic_setting | = DETECT_4_WIRE_MODE | AUTO_MEASURE ;
writel ( basic_setting , tsc - > tsc_regs + REG_TSC_BASIC_SETING ) ;
writel ( DE_GLITCH_2 , tsc - > tsc_regs + REG_TSC_DEBUG_MODE2 ) ;
writel ( tsc - > pre_charge_time , tsc - > tsc_regs + REG_TSC_PRE_CHARGE_TIME ) ;
writel ( MEASURE_INT_EN , tsc - > tsc_regs + REG_TSC_INT_EN ) ;
writel ( MEASURE_SIG_EN | VALID_SIG_EN ,
tsc - > tsc_regs + REG_TSC_INT_SIG_EN ) ;
/* start sense detection */
start = readl ( tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
start | = START_SENSE ;
start & = ~ TSC_DISABLE ;
writel ( start , tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
}
2015-09-14 10:36:35 -07:00
static int imx6ul_tsc_init ( struct imx6ul_tsc * tsc )
2015-09-05 11:31:21 -07:00
{
2015-09-14 10:36:35 -07:00
int err ;
err = imx6ul_adc_init ( tsc ) ;
if ( err )
return err ;
2015-09-05 11:31:21 -07:00
imx6ul_tsc_channel_config ( tsc ) ;
imx6ul_tsc_set ( tsc ) ;
2015-09-14 10:36:35 -07:00
return 0 ;
2015-09-05 11:31:21 -07:00
}
static void imx6ul_tsc_disable ( struct imx6ul_tsc * tsc )
{
int tsc_flow ;
int adc_cfg ;
/* TSC controller enters to idle status */
tsc_flow = readl ( tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
tsc_flow | = TSC_DISABLE ;
writel ( tsc_flow , tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
/* ADC controller enters to stop mode */
adc_cfg = readl ( tsc - > adc_regs + REG_ADC_HC0 ) ;
adc_cfg | = ADC_CONV_DISABLE ;
writel ( adc_cfg , tsc - > adc_regs + REG_ADC_HC0 ) ;
}
/* Delay some time (max 2ms), wait the pre-charge done. */
static bool tsc_wait_detect_mode ( struct imx6ul_tsc * tsc )
{
unsigned long timeout = jiffies + msecs_to_jiffies ( 2 ) ;
int state_machine ;
int debug_mode2 ;
do {
if ( time_after ( jiffies , timeout ) )
return false ;
usleep_range ( 200 , 400 ) ;
debug_mode2 = readl ( tsc - > tsc_regs + REG_TSC_DEBUG_MODE2 ) ;
state_machine = ( debug_mode2 > > 20 ) & 0x7 ;
} while ( state_machine ! = DETECT_MODE ) ;
usleep_range ( 200 , 400 ) ;
return true ;
}
static irqreturn_t tsc_irq_fn ( int irq , void * dev_id )
{
struct imx6ul_tsc * tsc = dev_id ;
int status ;
int value ;
int x , y ;
int start ;
status = readl ( tsc - > tsc_regs + REG_TSC_INT_STATUS ) ;
/* write 1 to clear the bit measure-signal */
writel ( MEASURE_SIGNAL | DETECT_SIGNAL ,
tsc - > tsc_regs + REG_TSC_INT_STATUS ) ;
/* It's a HW self-clean bit. Set this bit and start sense detection */
start = readl ( tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
start | = START_SENSE ;
writel ( start , tsc - > tsc_regs + REG_TSC_FLOW_CONTROL ) ;
if ( status & MEASURE_SIGNAL ) {
value = readl ( tsc - > tsc_regs + REG_TSC_MEASURE_VALUE ) ;
x = ( value > > 16 ) & 0x0fff ;
y = value & 0x0fff ;
/*
* In detect mode , we can get the xnur gpio value ,
* otherwise assume contact is stiull active .
*/
if ( ! tsc_wait_detect_mode ( tsc ) | |
gpiod_get_value_cansleep ( tsc - > xnur_gpio ) ) {
input_report_key ( tsc - > input , BTN_TOUCH , 1 ) ;
input_report_abs ( tsc - > input , ABS_X , x ) ;
input_report_abs ( tsc - > input , ABS_Y , y ) ;
} else {
input_report_key ( tsc - > input , BTN_TOUCH , 0 ) ;
}
input_sync ( tsc - > input ) ;
}
return IRQ_HANDLED ;
}
static irqreturn_t adc_irq_fn ( int irq , void * dev_id )
{
struct imx6ul_tsc * tsc = dev_id ;
int coco ;
int value ;
coco = readl ( tsc - > adc_regs + REG_ADC_HS ) ;
if ( coco & 0x01 ) {
value = readl ( tsc - > adc_regs + REG_ADC_R0 ) ;
complete ( & tsc - > completion ) ;
}
return IRQ_HANDLED ;
}
static int imx6ul_tsc_open ( struct input_dev * input_dev )
{
struct imx6ul_tsc * tsc = input_get_drvdata ( input_dev ) ;
int err ;
err = clk_prepare_enable ( tsc - > adc_clk ) ;
if ( err ) {
dev_err ( tsc - > dev ,
" Could not prepare or enable the adc clock: %d \n " ,
err ) ;
return err ;
}
err = clk_prepare_enable ( tsc - > tsc_clk ) ;
if ( err ) {
dev_err ( tsc - > dev ,
" Could not prepare or enable the tsc clock: %d \n " ,
err ) ;
clk_disable_unprepare ( tsc - > adc_clk ) ;
return err ;
}
2015-09-14 10:36:35 -07:00
return imx6ul_tsc_init ( tsc ) ;
2015-09-05 11:31:21 -07:00
}
static void imx6ul_tsc_close ( struct input_dev * input_dev )
{
struct imx6ul_tsc * tsc = input_get_drvdata ( input_dev ) ;
imx6ul_tsc_disable ( tsc ) ;
clk_disable_unprepare ( tsc - > tsc_clk ) ;
clk_disable_unprepare ( tsc - > adc_clk ) ;
}
static int imx6ul_tsc_probe ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
struct imx6ul_tsc * tsc ;
struct input_dev * input_dev ;
struct resource * tsc_mem ;
struct resource * adc_mem ;
int err ;
int tsc_irq ;
int adc_irq ;
2015-09-14 10:37:31 -07:00
tsc = devm_kzalloc ( & pdev - > dev , sizeof ( * tsc ) , GFP_KERNEL ) ;
2015-09-05 11:31:21 -07:00
if ( ! tsc )
return - ENOMEM ;
input_dev = devm_input_allocate_device ( & pdev - > dev ) ;
if ( ! input_dev )
return - ENOMEM ;
2015-09-14 10:37:55 -07:00
input_dev - > name = " iMX6UL Touchscreen Controller " ;
2015-09-05 11:31:21 -07:00
input_dev - > id . bustype = BUS_HOST ;
input_dev - > open = imx6ul_tsc_open ;
input_dev - > close = imx6ul_tsc_close ;
input_set_capability ( input_dev , EV_KEY , BTN_TOUCH ) ;
input_set_abs_params ( input_dev , ABS_X , 0 , 0xFFF , 0 , 0 ) ;
input_set_abs_params ( input_dev , ABS_Y , 0 , 0xFFF , 0 , 0 ) ;
input_set_drvdata ( input_dev , tsc ) ;
tsc - > dev = & pdev - > dev ;
tsc - > input = input_dev ;
init_completion ( & tsc - > completion ) ;
tsc - > xnur_gpio = devm_gpiod_get ( & pdev - > dev , " xnur " , GPIOD_IN ) ;
if ( IS_ERR ( tsc - > xnur_gpio ) ) {
err = PTR_ERR ( tsc - > xnur_gpio ) ;
dev_err ( & pdev - > dev ,
" failed to request GPIO tsc_X- (xnur): %d \n " , err ) ;
return err ;
}
tsc_mem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
tsc - > tsc_regs = devm_ioremap_resource ( & pdev - > dev , tsc_mem ) ;
if ( IS_ERR ( tsc - > tsc_regs ) ) {
err = PTR_ERR ( tsc - > tsc_regs ) ;
dev_err ( & pdev - > dev , " failed to remap tsc memory: %d \n " , err ) ;
return err ;
}
adc_mem = platform_get_resource ( pdev , IORESOURCE_MEM , 1 ) ;
tsc - > adc_regs = devm_ioremap_resource ( & pdev - > dev , adc_mem ) ;
if ( IS_ERR ( tsc - > adc_regs ) ) {
err = PTR_ERR ( tsc - > adc_regs ) ;
dev_err ( & pdev - > dev , " failed to remap adc memory: %d \n " , err ) ;
return err ;
}
tsc - > tsc_clk = devm_clk_get ( & pdev - > dev , " tsc " ) ;
if ( IS_ERR ( tsc - > tsc_clk ) ) {
err = PTR_ERR ( tsc - > tsc_clk ) ;
dev_err ( & pdev - > dev , " failed getting tsc clock: %d \n " , err ) ;
return err ;
}
tsc - > adc_clk = devm_clk_get ( & pdev - > dev , " adc " ) ;
if ( IS_ERR ( tsc - > adc_clk ) ) {
err = PTR_ERR ( tsc - > adc_clk ) ;
dev_err ( & pdev - > dev , " failed getting adc clock: %d \n " , err ) ;
return err ;
}
tsc_irq = platform_get_irq ( pdev , 0 ) ;
if ( tsc_irq < 0 ) {
dev_err ( & pdev - > dev , " no tsc irq resource? \n " ) ;
return tsc_irq ;
}
adc_irq = platform_get_irq ( pdev , 1 ) ;
2015-09-14 10:37:08 -07:00
if ( adc_irq < 0 ) {
2015-09-05 11:31:21 -07:00
dev_err ( & pdev - > dev , " no adc irq resource? \n " ) ;
return adc_irq ;
}
err = devm_request_threaded_irq ( tsc - > dev , tsc_irq ,
NULL , tsc_irq_fn , IRQF_ONESHOT ,
dev_name ( & pdev - > dev ) , tsc ) ;
if ( err ) {
dev_err ( & pdev - > dev ,
" failed requesting tsc irq %d: %d \n " ,
tsc_irq , err ) ;
return err ;
}
err = devm_request_irq ( tsc - > dev , adc_irq , adc_irq_fn , 0 ,
dev_name ( & pdev - > dev ) , tsc ) ;
if ( err ) {
dev_err ( & pdev - > dev ,
" failed requesting adc irq %d: %d \n " ,
adc_irq , err ) ;
return err ;
}
err = of_property_read_u32 ( np , " measure-delay-time " ,
& tsc - > measure_delay_time ) ;
if ( err )
tsc - > measure_delay_time = 0xffff ;
err = of_property_read_u32 ( np , " pre-charge-time " ,
& tsc - > pre_charge_time ) ;
if ( err )
tsc - > pre_charge_time = 0xfff ;
err = input_register_device ( tsc - > input ) ;
if ( err ) {
dev_err ( & pdev - > dev ,
" failed to register input device: %d \n " , err ) ;
return err ;
}
platform_set_drvdata ( pdev , tsc ) ;
return 0 ;
}
static int __maybe_unused imx6ul_tsc_suspend ( struct device * dev )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct imx6ul_tsc * tsc = platform_get_drvdata ( pdev ) ;
struct input_dev * input_dev = tsc - > input ;
mutex_lock ( & input_dev - > mutex ) ;
if ( input_dev - > users ) {
imx6ul_tsc_disable ( tsc ) ;
clk_disable_unprepare ( tsc - > tsc_clk ) ;
clk_disable_unprepare ( tsc - > adc_clk ) ;
}
mutex_unlock ( & input_dev - > mutex ) ;
return 0 ;
}
static int __maybe_unused imx6ul_tsc_resume ( struct device * dev )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
struct imx6ul_tsc * tsc = platform_get_drvdata ( pdev ) ;
struct input_dev * input_dev = tsc - > input ;
int retval = 0 ;
mutex_lock ( & input_dev - > mutex ) ;
if ( input_dev - > users ) {
retval = clk_prepare_enable ( tsc - > adc_clk ) ;
if ( retval )
goto out ;
retval = clk_prepare_enable ( tsc - > tsc_clk ) ;
if ( retval ) {
clk_disable_unprepare ( tsc - > adc_clk ) ;
goto out ;
}
2015-09-14 10:36:35 -07:00
retval = imx6ul_tsc_init ( tsc ) ;
2015-09-05 11:31:21 -07:00
}
out :
mutex_unlock ( & input_dev - > mutex ) ;
return retval ;
}
static SIMPLE_DEV_PM_OPS ( imx6ul_tsc_pm_ops ,
imx6ul_tsc_suspend , imx6ul_tsc_resume ) ;
static const struct of_device_id imx6ul_tsc_match [ ] = {
{ . compatible = " fsl,imx6ul-tsc " , } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , imx6ul_tsc_match ) ;
static struct platform_driver imx6ul_tsc_driver = {
. driver = {
. name = " imx6ul-tsc " ,
. of_match_table = imx6ul_tsc_match ,
. pm = & imx6ul_tsc_pm_ops ,
} ,
. probe = imx6ul_tsc_probe ,
} ;
module_platform_driver ( imx6ul_tsc_driver ) ;
MODULE_AUTHOR ( " Haibo Chen <haibo.chen@freescale.com> " ) ;
MODULE_DESCRIPTION ( " Freescale i.MX6UL Touchscreen controller driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;