2019-06-04 10:10:57 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2010-07-02 16:52:09 +05:30
/*
* Copyright ( C ) ST - Ericsson SA 2010
*
* Author : Rabin Vincent < rabin . vincent @ stericsson . com > for ST - Ericsson
*/
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/slab.h>
2018-06-27 11:30:48 +02:00
# include <linux/gpio/driver.h>
2010-07-02 16:52:09 +05:30
# include <linux/interrupt.h>
2012-11-26 17:06:51 +05:30
# include <linux/of.h>
2010-07-02 16:52:09 +05:30
# include <linux/mfd/stmpe.h>
2014-10-02 07:55:41 +02:00
# include <linux/seq_file.h>
2016-09-27 15:59:12 -07:00
# include <linux/bitops.h>
2010-07-02 16:52:09 +05:30
/*
* These registers are modified under the irq bus lock and cached to avoid
* unnecessary writes in bus_sync_unlock .
*/
enum { REG_RE , REG_FE , REG_IE } ;
2016-08-10 09:39:12 +02:00
enum { LSB , CSB , MSB } ;
2010-07-02 16:52:09 +05:30
# define CACHE_NR_REGS 3
2014-05-08 23:16:34 +02:00
/* No variant has more than 24 GPIOs */
# define CACHE_NR_BANKS (24 / 8)
2010-07-02 16:52:09 +05:30
struct stmpe_gpio {
struct gpio_chip chip ;
struct stmpe * stmpe ;
struct device * dev ;
struct mutex irq_lock ;
2015-01-13 08:00:29 +01:00
u32 norequest_mask ;
2010-07-02 16:52:09 +05:30
/* Caches of interrupt control registers for bus_lock */
u8 regs [ CACHE_NR_REGS ] [ CACHE_NR_BANKS ] ;
u8 oldregs [ CACHE_NR_REGS ] [ CACHE_NR_BANKS ] ;
} ;
static int stmpe_gpio_get ( struct gpio_chip * chip , unsigned offset )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
2016-08-10 09:39:12 +02:00
u8 reg = stmpe - > regs [ STMPE_IDX_GPMR_LSB + ( offset / 8 ) ] ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
int ret ;
ret = stmpe_reg_read ( stmpe , reg ) ;
if ( ret < 0 )
return ret ;
2012-02-27 11:19:43 +05:30
return ! ! ( ret & mask ) ;
2010-07-02 16:52:09 +05:30
}
static void stmpe_gpio_set ( struct gpio_chip * chip , unsigned offset , int val )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
int which = val ? STMPE_IDX_GPSR_LSB : STMPE_IDX_GPCR_LSB ;
2016-08-10 09:39:12 +02:00
u8 reg = stmpe - > regs [ which + ( offset / 8 ) ] ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
2011-12-14 09:28:27 +05:30
/*
* Some variants have single register for gpio set / clear functionality .
* For them we need to write 0 to clear and 1 to set .
*/
if ( stmpe - > regs [ STMPE_IDX_GPSR_LSB ] = = stmpe - > regs [ STMPE_IDX_GPCR_LSB ] )
stmpe_set_bits ( stmpe , reg , mask , val ? mask : 0 ) ;
else
stmpe_reg_write ( stmpe , reg , mask ) ;
2010-07-02 16:52:09 +05:30
}
2016-04-28 15:00:18 +02:00
static int stmpe_gpio_get_direction ( struct gpio_chip * chip ,
unsigned offset )
{
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
u8 reg = stmpe - > regs [ STMPE_IDX_GPDR_LSB ] - ( offset / 8 ) ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2016-04-28 15:00:18 +02:00
int ret ;
ret = stmpe_reg_read ( stmpe , reg ) ;
if ( ret < 0 )
return ret ;
2019-11-06 10:54:12 +02:00
if ( ret & mask )
return GPIO_LINE_DIRECTION_OUT ;
return GPIO_LINE_DIRECTION_IN ;
2016-04-28 15:00:18 +02:00
}
2010-07-02 16:52:09 +05:30
static int stmpe_gpio_direction_output ( struct gpio_chip * chip ,
unsigned offset , int val )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
2016-08-10 09:39:12 +02:00
u8 reg = stmpe - > regs [ STMPE_IDX_GPDR_LSB + ( offset / 8 ) ] ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
stmpe_gpio_set ( chip , offset , val ) ;
return stmpe_set_bits ( stmpe , reg , mask , mask ) ;
}
static int stmpe_gpio_direction_input ( struct gpio_chip * chip ,
unsigned offset )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
2016-08-10 09:39:12 +02:00
u8 reg = stmpe - > regs [ STMPE_IDX_GPDR_LSB + ( offset / 8 ) ] ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
return stmpe_set_bits ( stmpe , reg , mask , 0 ) ;
}
static int stmpe_gpio_request ( struct gpio_chip * chip , unsigned offset )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( chip ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
2016-09-27 16:11:10 -07:00
if ( stmpe_gpio - > norequest_mask & BIT ( offset ) )
2010-08-16 17:14:44 +02:00
return - EINVAL ;
2016-09-27 16:11:10 -07:00
return stmpe_set_altfunc ( stmpe , BIT ( offset ) , STMPE_BLOCK_GPIO ) ;
2010-07-02 16:52:09 +05:30
}
2016-09-11 14:14:37 +02:00
static const struct gpio_chip template_chip = {
2010-07-02 16:52:09 +05:30
. label = " stmpe " ,
. owner = THIS_MODULE ,
2016-04-28 15:00:18 +02:00
. get_direction = stmpe_gpio_get_direction ,
2010-07-02 16:52:09 +05:30
. direction_input = stmpe_gpio_direction_input ,
. get = stmpe_gpio_get ,
. direction_output = stmpe_gpio_direction_output ,
. set = stmpe_gpio_set ,
. request = stmpe_gpio_request ,
2013-12-04 14:42:46 +01:00
. can_sleep = true ,
2010-07-02 16:52:09 +05:30
} ;
2011-01-12 17:00:17 -08:00
static int stmpe_gpio_irq_set_type ( struct irq_data * d , unsigned int type )
2010-07-02 16:52:09 +05:30
{
2014-04-15 23:38:56 +02:00
struct gpio_chip * gc = irq_data_get_irq_chip_data ( d ) ;
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2012-12-10 10:07:54 +00:00
int offset = d - > hwirq ;
2010-07-02 16:52:09 +05:30
int regoffset = offset / 8 ;
2016-09-27 16:11:10 -07:00
int mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
2014-10-02 07:55:27 +02:00
if ( type & IRQ_TYPE_LEVEL_LOW | | type & IRQ_TYPE_LEVEL_HIGH )
2010-07-02 16:52:09 +05:30
return - EINVAL ;
2016-08-10 09:39:15 +02:00
/* STMPE801 and STMPE 1600 don't have RE and FE registers */
if ( stmpe_gpio - > stmpe - > partnum = = STMPE801 | |
stmpe_gpio - > stmpe - > partnum = = STMPE1600 )
2011-12-14 09:28:27 +05:30
return 0 ;
2014-10-02 07:55:27 +02:00
if ( type & IRQ_TYPE_EDGE_RISING )
2010-07-02 16:52:09 +05:30
stmpe_gpio - > regs [ REG_RE ] [ regoffset ] | = mask ;
else
stmpe_gpio - > regs [ REG_RE ] [ regoffset ] & = ~ mask ;
2014-10-02 07:55:27 +02:00
if ( type & IRQ_TYPE_EDGE_FALLING )
2010-07-02 16:52:09 +05:30
stmpe_gpio - > regs [ REG_FE ] [ regoffset ] | = mask ;
else
stmpe_gpio - > regs [ REG_FE ] [ regoffset ] & = ~ mask ;
return 0 ;
}
2011-01-12 17:00:17 -08:00
static void stmpe_gpio_irq_lock ( struct irq_data * d )
2010-07-02 16:52:09 +05:30
{
2014-04-15 23:38:56 +02:00
struct gpio_chip * gc = irq_data_get_irq_chip_data ( d ) ;
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2010-07-02 16:52:09 +05:30
mutex_lock ( & stmpe_gpio - > irq_lock ) ;
}
2011-01-12 17:00:17 -08:00
static void stmpe_gpio_irq_sync_unlock ( struct irq_data * d )
2010-07-02 16:52:09 +05:30
{
2014-04-15 23:38:56 +02:00
struct gpio_chip * gc = irq_data_get_irq_chip_data ( d ) ;
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2010-07-02 16:52:09 +05:30
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
int num_banks = DIV_ROUND_UP ( stmpe - > num_gpios , 8 ) ;
2016-08-10 09:39:12 +02:00
static const u8 regmap [ CACHE_NR_REGS ] [ CACHE_NR_BANKS ] = {
[ REG_RE ] [ LSB ] = STMPE_IDX_GPRER_LSB ,
[ REG_RE ] [ CSB ] = STMPE_IDX_GPRER_CSB ,
[ REG_RE ] [ MSB ] = STMPE_IDX_GPRER_MSB ,
[ REG_FE ] [ LSB ] = STMPE_IDX_GPFER_LSB ,
[ REG_FE ] [ CSB ] = STMPE_IDX_GPFER_CSB ,
[ REG_FE ] [ MSB ] = STMPE_IDX_GPFER_MSB ,
[ REG_IE ] [ LSB ] = STMPE_IDX_IEGPIOR_LSB ,
[ REG_IE ] [ CSB ] = STMPE_IDX_IEGPIOR_CSB ,
[ REG_IE ] [ MSB ] = STMPE_IDX_IEGPIOR_MSB ,
2010-07-02 16:52:09 +05:30
} ;
int i , j ;
2018-01-12 13:16:08 +01:00
/*
* STMPE1600 : to be able to get IRQ from pins ,
* a read must be done on GPMR register , or a write in
* GPSR or GPCR registers
*/
if ( stmpe - > partnum = = STMPE1600 ) {
stmpe_reg_read ( stmpe , stmpe - > regs [ STMPE_IDX_GPMR_LSB ] ) ;
stmpe_reg_read ( stmpe , stmpe - > regs [ STMPE_IDX_GPMR_CSB ] ) ;
}
2010-07-02 16:52:09 +05:30
for ( i = 0 ; i < CACHE_NR_REGS ; i + + ) {
2016-08-10 09:39:15 +02:00
/* STMPE801 and STMPE1600 don't have RE and FE registers */
if ( ( stmpe - > partnum = = STMPE801 | |
stmpe - > partnum = = STMPE1600 ) & &
( i ! = REG_IE ) )
2011-12-14 09:28:27 +05:30
continue ;
2010-07-02 16:52:09 +05:30
for ( j = 0 ; j < num_banks ; j + + ) {
u8 old = stmpe_gpio - > oldregs [ i ] [ j ] ;
u8 new = stmpe_gpio - > regs [ i ] [ j ] ;
if ( new = = old )
continue ;
stmpe_gpio - > oldregs [ i ] [ j ] = new ;
2016-08-10 09:39:12 +02:00
stmpe_reg_write ( stmpe , stmpe - > regs [ regmap [ i ] [ j ] ] , new ) ;
2010-07-02 16:52:09 +05:30
}
}
mutex_unlock ( & stmpe_gpio - > irq_lock ) ;
}
2011-01-12 17:00:17 -08:00
static void stmpe_gpio_irq_mask ( struct irq_data * d )
2010-07-02 16:52:09 +05:30
{
2014-04-15 23:38:56 +02:00
struct gpio_chip * gc = irq_data_get_irq_chip_data ( d ) ;
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2012-12-10 10:07:54 +00:00
int offset = d - > hwirq ;
2010-07-02 16:52:09 +05:30
int regoffset = offset / 8 ;
2016-09-27 16:11:10 -07:00
int mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
stmpe_gpio - > regs [ REG_IE ] [ regoffset ] & = ~ mask ;
}
2011-01-12 17:00:17 -08:00
static void stmpe_gpio_irq_unmask ( struct irq_data * d )
2010-07-02 16:52:09 +05:30
{
2014-04-15 23:38:56 +02:00
struct gpio_chip * gc = irq_data_get_irq_chip_data ( d ) ;
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2012-12-10 10:07:54 +00:00
int offset = d - > hwirq ;
2010-07-02 16:52:09 +05:30
int regoffset = offset / 8 ;
2016-09-27 16:11:10 -07:00
int mask = BIT ( offset % 8 ) ;
2010-07-02 16:52:09 +05:30
stmpe_gpio - > regs [ REG_IE ] [ regoffset ] | = mask ;
}
2014-10-02 07:55:41 +02:00
static void stmpe_dbg_show_one ( struct seq_file * s ,
struct gpio_chip * gc ,
unsigned offset , unsigned gpio )
{
2015-12-07 14:32:13 +01:00
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
2014-10-02 07:55:41 +02:00
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
const char * label = gpiochip_is_requested ( gc , offset ) ;
bool val = ! ! stmpe_gpio_get ( gc , offset ) ;
2016-08-10 09:39:12 +02:00
u8 bank = offset / 8 ;
u8 dir_reg = stmpe - > regs [ STMPE_IDX_GPDR_LSB + bank ] ;
2016-09-27 16:11:10 -07:00
u8 mask = BIT ( offset % 8 ) ;
2014-10-02 07:55:41 +02:00
int ret ;
u8 dir ;
ret = stmpe_reg_read ( stmpe , dir_reg ) ;
if ( ret < 0 )
return ;
dir = ! ! ( ret & mask ) ;
if ( dir ) {
seq_printf ( s , " gpio-%-3d (%-20.20s) out %s " ,
gpio , label ? : " (none) " ,
val ? " hi " : " lo " ) ;
} else {
2016-08-10 09:39:08 +02:00
u8 edge_det_reg ;
u8 rise_reg ;
u8 fall_reg ;
u8 irqen_reg ;
2017-11-28 18:23:39 +00:00
static const char * const edge_det_values [ ] = {
" edge-inactive " ,
" edge-asserted " ,
" not-supported "
} ;
static const char * const rise_values [ ] = {
" no-rising-edge-detection " ,
" rising-edge-detection " ,
" not-supported "
} ;
static const char * const fall_values [ ] = {
" no-falling-edge-detection " ,
" falling-edge-detection " ,
" not-supported "
} ;
2016-08-10 09:39:08 +02:00
# define NOT_SUPPORTED_IDX 2
u8 edge_det = NOT_SUPPORTED_IDX ;
u8 rise = NOT_SUPPORTED_IDX ;
u8 fall = NOT_SUPPORTED_IDX ;
2014-10-02 07:55:41 +02:00
bool irqen ;
2016-08-10 09:39:08 +02:00
switch ( stmpe - > partnum ) {
case STMPE610 :
case STMPE811 :
case STMPE1601 :
case STMPE2401 :
case STMPE2403 :
2016-08-10 09:39:12 +02:00
edge_det_reg = stmpe - > regs [ STMPE_IDX_GPEDR_LSB + bank ] ;
2016-08-10 09:39:08 +02:00
ret = stmpe_reg_read ( stmpe , edge_det_reg ) ;
if ( ret < 0 )
return ;
edge_det = ! ! ( ret & mask ) ;
2020-08-23 17:36:59 -05:00
fallthrough ;
2016-08-10 09:39:08 +02:00
case STMPE1801 :
2016-08-10 09:39:12 +02:00
rise_reg = stmpe - > regs [ STMPE_IDX_GPRER_LSB + bank ] ;
fall_reg = stmpe - > regs [ STMPE_IDX_GPFER_LSB + bank ] ;
2016-08-10 09:39:08 +02:00
ret = stmpe_reg_read ( stmpe , rise_reg ) ;
if ( ret < 0 )
return ;
rise = ! ! ( ret & mask ) ;
ret = stmpe_reg_read ( stmpe , fall_reg ) ;
if ( ret < 0 )
return ;
fall = ! ! ( ret & mask ) ;
2020-08-23 17:36:59 -05:00
fallthrough ;
2016-08-10 09:39:08 +02:00
case STMPE801 :
2016-08-10 09:39:15 +02:00
case STMPE1600 :
2016-08-10 09:39:12 +02:00
irqen_reg = stmpe - > regs [ STMPE_IDX_IEGPIOR_LSB + bank ] ;
2016-08-10 09:39:08 +02:00
break ;
default :
2014-10-02 07:55:41 +02:00
return ;
2016-08-10 09:39:08 +02:00
}
2014-10-02 07:55:41 +02:00
ret = stmpe_reg_read ( stmpe , irqen_reg ) ;
if ( ret < 0 )
return ;
irqen = ! ! ( ret & mask ) ;
2016-08-10 09:39:08 +02:00
seq_printf ( s , " gpio-%-3d (%-20.20s) in %s %13s %13s %25s %25s " ,
2014-10-02 07:55:41 +02:00
gpio , label ? : " (none) " ,
val ? " hi " : " lo " ,
2016-08-10 09:39:08 +02:00
edge_det_values [ edge_det ] ,
irqen ? " IRQ-enabled " : " IRQ-disabled " ,
rise_values [ rise ] ,
fall_values [ fall ] ) ;
2014-10-02 07:55:41 +02:00
}
}
static void stmpe_dbg_show ( struct seq_file * s , struct gpio_chip * gc )
{
unsigned i ;
unsigned gpio = gc - > base ;
for ( i = 0 ; i < gc - > ngpio ; i + + , gpio + + ) {
stmpe_dbg_show_one ( s , gc , i , gpio ) ;
2018-01-12 19:30:50 +01:00
seq_putc ( s , ' \n ' ) ;
2014-10-02 07:55:41 +02:00
}
}
2010-07-02 16:52:09 +05:30
static struct irq_chip stmpe_gpio_irq_chip = {
. name = " stmpe-gpio " ,
2011-01-12 17:00:17 -08:00
. irq_bus_lock = stmpe_gpio_irq_lock ,
. irq_bus_sync_unlock = stmpe_gpio_irq_sync_unlock ,
. irq_mask = stmpe_gpio_irq_mask ,
. irq_unmask = stmpe_gpio_irq_unmask ,
. irq_set_type = stmpe_gpio_irq_set_type ,
2010-07-02 16:52:09 +05:30
} ;
2018-03-28 10:59:57 -07:00
# define MAX_GPIOS 24
2010-07-02 16:52:09 +05:30
static irqreturn_t stmpe_gpio_irq ( int irq , void * dev )
{
struct stmpe_gpio * stmpe_gpio = dev ;
struct stmpe * stmpe = stmpe_gpio - > stmpe ;
2016-08-10 09:39:15 +02:00
u8 statmsbreg ;
2010-07-02 16:52:09 +05:30
int num_banks = DIV_ROUND_UP ( stmpe - > num_gpios , 8 ) ;
2018-03-28 10:59:57 -07:00
u8 status [ DIV_ROUND_UP ( MAX_GPIOS , 8 ) ] ;
2010-07-02 16:52:09 +05:30
int ret ;
int i ;
2016-08-10 09:39:15 +02:00
/*
* the stmpe_block_read ( ) call below , imposes to set statmsbreg
* with the register located at the lowest address . As STMPE1600
* variant is the only one which respect registers address ' s order
* ( LSB regs located at lowest address than MSB ones ) whereas all
* the others have a registers layout with MSB located before the
* LSB regs .
*/
if ( stmpe - > partnum = = STMPE1600 )
statmsbreg = stmpe - > regs [ STMPE_IDX_ISGPIOR_LSB ] ;
else
statmsbreg = stmpe - > regs [ STMPE_IDX_ISGPIOR_MSB ] ;
2010-07-02 16:52:09 +05:30
ret = stmpe_block_read ( stmpe , statmsbreg , num_banks , status ) ;
if ( ret < 0 )
return IRQ_NONE ;
for ( i = 0 ; i < num_banks ; i + + ) {
2016-08-10 09:39:15 +02:00
int bank = ( stmpe_gpio - > stmpe - > partnum = = STMPE1600 ) ? i :
num_banks - i - 1 ;
2010-07-02 16:52:09 +05:30
unsigned int enabled = stmpe_gpio - > regs [ REG_IE ] [ bank ] ;
unsigned int stat = status [ i ] ;
stat & = enabled ;
if ( ! stat )
continue ;
while ( stat ) {
int bit = __ffs ( stat ) ;
int line = bank * 8 + bit ;
2017-11-07 19:15:47 +01:00
int child_irq = irq_find_mapping ( stmpe_gpio - > chip . irq . domain ,
2013-10-11 19:51:38 +02:00
line ) ;
2010-07-02 16:52:09 +05:30
2013-10-11 19:51:38 +02:00
handle_nested_irq ( child_irq ) ;
2016-09-27 16:11:10 -07:00
stat & = ~ BIT ( bit ) ;
2010-07-02 16:52:09 +05:30
}
2016-08-10 09:39:09 +02:00
/*
* interrupt status register write has no effect on
2016-08-10 09:39:15 +02:00
* 801 / 1801 / 1600 , bits are cleared when read .
* Edge detect register is not present on 801 / 1600 / 1801
2016-08-10 09:39:09 +02:00
*/
2016-10-12 09:25:20 +03:00
if ( stmpe - > partnum ! = STMPE801 & & stmpe - > partnum ! = STMPE1600 & &
2016-08-10 09:39:15 +02:00
stmpe - > partnum ! = STMPE1801 ) {
2016-08-10 09:39:09 +02:00
stmpe_reg_write ( stmpe , statmsbreg + i , status [ i ] ) ;
2016-08-10 09:39:12 +02:00
stmpe_reg_write ( stmpe ,
2016-11-23 23:21:17 +01:00
stmpe - > regs [ STMPE_IDX_GPEDR_MSB ] + i ,
2016-08-10 09:39:12 +02:00
status [ i ] ) ;
2016-08-10 09:39:09 +02:00
}
2010-07-02 16:52:09 +05:30
}
return IRQ_HANDLED ;
}
2019-09-04 16:01:04 +02:00
static void stmpe_init_irq_valid_mask ( struct gpio_chip * gc ,
unsigned long * valid_mask ,
unsigned int ngpios )
{
struct stmpe_gpio * stmpe_gpio = gpiochip_get_data ( gc ) ;
int i ;
if ( ! stmpe_gpio - > norequest_mask )
return ;
/* Forbid unused lines to be mapped as IRQs */
for ( i = 0 ; i < sizeof ( u32 ) ; i + + ) {
if ( stmpe_gpio - > norequest_mask & BIT ( i ) )
clear_bit ( i , valid_mask ) ;
}
}
2021-05-16 09:14:25 +03:00
static void stmpe_gpio_disable ( void * stmpe )
{
stmpe_disable ( stmpe , STMPE_BLOCK_GPIO ) ;
}
2012-11-19 13:22:34 -05:00
static int stmpe_gpio_probe ( struct platform_device * pdev )
2010-07-02 16:52:09 +05:30
{
struct stmpe * stmpe = dev_get_drvdata ( pdev - > dev . parent ) ;
2012-11-26 17:06:51 +05:30
struct device_node * np = pdev - > dev . of_node ;
2010-07-02 16:52:09 +05:30
struct stmpe_gpio * stmpe_gpio ;
2018-01-12 20:48:40 +01:00
int ret , irq ;
2010-07-02 16:52:09 +05:30
2018-03-28 10:59:57 -07:00
if ( stmpe - > num_gpios > MAX_GPIOS ) {
dev_err ( & pdev - > dev , " Need to increase maximum GPIO number \n " ) ;
return - EINVAL ;
}
2021-05-16 09:14:25 +03:00
stmpe_gpio = devm_kzalloc ( & pdev - > dev , sizeof ( * stmpe_gpio ) , GFP_KERNEL ) ;
2010-07-02 16:52:09 +05:30
if ( ! stmpe_gpio )
return - ENOMEM ;
mutex_init ( & stmpe_gpio - > irq_lock ) ;
stmpe_gpio - > dev = & pdev - > dev ;
stmpe_gpio - > stmpe = stmpe ;
stmpe_gpio - > chip = template_chip ;
stmpe_gpio - > chip . ngpio = stmpe - > num_gpios ;
2015-11-04 09:56:26 +01:00
stmpe_gpio - > chip . parent = & pdev - > dev ;
2014-05-08 23:16:34 +02:00
stmpe_gpio - > chip . base = - 1 ;
2010-07-02 16:52:09 +05:30
2014-10-02 07:55:41 +02:00
if ( IS_ENABLED ( CONFIG_DEBUG_FS ) )
stmpe_gpio - > chip . dbg_show = stmpe_dbg_show ;
2015-01-13 08:00:29 +01:00
of_property_read_u32 ( np , " st,norequest-mask " ,
& stmpe_gpio - > norequest_mask ) ;
2012-11-26 17:06:51 +05:30
2018-01-12 20:44:15 +01:00
irq = platform_get_irq ( pdev , 0 ) ;
2014-05-08 23:16:34 +02:00
if ( irq < 0 )
2012-01-26 22:17:15 +01:00
dev_info ( & pdev - > dev ,
2014-04-15 23:38:56 +02:00
" device configured in no-irq mode: "
2012-01-26 22:17:15 +01:00
" irqs are not available \n " ) ;
2010-07-02 16:52:09 +05:30
ret = stmpe_enable ( stmpe , STMPE_BLOCK_GPIO ) ;
if ( ret )
2021-05-16 09:14:25 +03:00
return ret ;
ret = devm_add_action_or_reset ( & pdev - > dev , stmpe_gpio_disable , stmpe ) ;
if ( ret )
return ret ;
2010-07-02 16:52:09 +05:30
2014-04-15 23:38:56 +02:00
if ( irq > 0 ) {
2020-07-16 12:06:38 +02:00
struct gpio_irq_chip * girq ;
2014-04-15 23:38:56 +02:00
ret = devm_request_threaded_irq ( & pdev - > dev , irq , NULL ,
stmpe_gpio_irq , IRQF_ONESHOT ,
" stmpe-gpio " , stmpe_gpio ) ;
2012-01-26 22:17:15 +01:00
if ( ret ) {
dev_err ( & pdev - > dev , " unable to get irq: %d \n " , ret ) ;
2021-05-16 09:14:25 +03:00
return ret ;
2012-01-26 22:17:15 +01:00
}
2010-07-02 16:52:09 +05:30
2020-07-16 12:06:38 +02:00
girq = & stmpe_gpio - > chip . irq ;
girq - > chip = & stmpe_gpio_irq_chip ;
/* This will let us handle the parent IRQ in the driver */
girq - > parent_handler = NULL ;
girq - > num_parents = 0 ;
girq - > parents = NULL ;
girq - > default_type = IRQ_TYPE_NONE ;
girq - > handler = handle_simple_irq ;
girq - > threaded = true ;
2020-10-19 15:44:29 +02:00
girq - > init_valid_mask = stmpe_init_irq_valid_mask ;
2010-07-02 16:52:09 +05:30
}
2021-05-16 09:14:25 +03:00
return devm_gpiochip_add_data ( & pdev - > dev , & stmpe_gpio - > chip , stmpe_gpio ) ;
2010-07-02 16:52:09 +05:30
}
static struct platform_driver stmpe_gpio_driver = {
2016-05-09 19:59:56 -04:00
. driver = {
. suppress_bind_attrs = true ,
. name = " stmpe-gpio " ,
} ,
2010-07-02 16:52:09 +05:30
. probe = stmpe_gpio_probe ,
} ;
static int __init stmpe_gpio_init ( void )
{
return platform_driver_register ( & stmpe_gpio_driver ) ;
}
subsys_initcall ( stmpe_gpio_init ) ;