2008-10-20 23:51:46 +02:00
/*
2011-06-04 18:38:28 -06:00
* Access to GPIOs on TWL4030 / TPS659x0 chips
2008-10-20 23:51:46 +02:00
*
* Copyright ( C ) 2006 - 2007 Texas Instruments , Inc .
* Copyright ( C ) 2006 MontaVista Software , Inc .
*
* Code re - arranged and cleaned up by :
* Syed Mohammed Khasim < x0khasim @ ti . com >
*
* Initial Code :
* Andy Lowe / Nishanth Menon
*
* 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 .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/kthread.h>
# include <linux/irq.h>
# include <linux/gpio.h>
# include <linux/platform_device.h>
2012-02-29 22:51:32 +01:00
# include <linux/of.h>
# include <linux/irqdomain.h>
2008-10-20 23:51:46 +02:00
2009-12-13 20:05:51 +01:00
# include <linux/i2c/twl.h>
2008-10-20 23:51:46 +02:00
/*
* The GPIO " subchip " supports 18 GPIOs which can be configured as
* inputs or outputs , with pullups or pulldowns on each pin . Each
* GPIO can trigger interrupts on either or both edges .
*
* GPIO interrupts can be fed to either of two IRQ lines ; this is
* intended to support multiple hosts .
*
* There are also two LED pins used sometimes as output - only GPIOs .
*/
/* genirq interfaces are not available to modules */
# ifdef MODULE
# define is_module() true
# else
# define is_module() false
# endif
/* GPIO_CTRL Fields */
# define MASK_GPIO_CTRL_GPIO0CD1 BIT(0)
# define MASK_GPIO_CTRL_GPIO1CD2 BIT(1)
# define MASK_GPIO_CTRL_GPIO_ON BIT(2)
/* Mask for GPIO registers when aggregated into a 32-bit integer */
# define GPIO_32_MASK 0x0003ffff
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv {
struct gpio_chip gpio_chip ;
2012-12-20 10:44:11 +01:00
struct mutex mutex ;
2012-12-06 10:52:05 +00:00
int irq_base ;
2012-12-20 10:44:11 +01:00
/* Bitfields for state caching */
2012-12-06 10:52:05 +00:00
unsigned int usage_count ;
2012-12-20 10:44:11 +01:00
unsigned int direction ;
unsigned int out_state ;
2012-12-06 10:52:05 +00:00
} ;
2008-10-20 23:51:46 +02:00
/*----------------------------------------------------------------------*/
2012-12-06 10:52:05 +00:00
static inline struct gpio_twl4030_priv * to_gpio_twl4030 ( struct gpio_chip * chip )
{
return container_of ( chip , struct gpio_twl4030_priv , gpio_chip ) ;
}
2008-10-20 23:51:46 +02:00
/*
* To configure TWL4030 GPIO module registers
*/
static inline int gpio_twl4030_write ( u8 address , u8 data )
{
2009-12-13 21:23:33 +01:00
return twl_i2c_write_u8 ( TWL4030_MODULE_GPIO , data , address ) ;
2008-10-20 23:51:46 +02:00
}
/*----------------------------------------------------------------------*/
/*
2012-11-13 10:35:13 +01:00
* LED register offsets from TWL_MODULE_LED base
2008-10-20 23:51:46 +02:00
* PWMs A and B are dedicated to LEDs A and B , respectively .
*/
2012-11-13 10:35:13 +01:00
# define TWL4030_LED_LEDEN_REG 0x00
# define TWL4030_PWMAON_REG 0x01
# define TWL4030_PWMAOFF_REG 0x02
# define TWL4030_PWMBON_REG 0x03
# define TWL4030_PWMBOFF_REG 0x04
2008-10-20 23:51:46 +02:00
/* LEDEN bits */
# define LEDEN_LEDAON BIT(0)
# define LEDEN_LEDBON BIT(1)
# define LEDEN_LEDAEXT BIT(2)
# define LEDEN_LEDBEXT BIT(3)
# define LEDEN_LEDAPWM BIT(4)
# define LEDEN_LEDBPWM BIT(5)
# define LEDEN_PWM_LENGTHA BIT(6)
# define LEDEN_PWM_LENGTHB BIT(7)
# define PWMxON_LENGTH BIT(7)
/*----------------------------------------------------------------------*/
/*
* To read a TWL4030 GPIO module register
*/
static inline int gpio_twl4030_read ( u8 address )
{
u8 data ;
int ret = 0 ;
2009-12-13 21:23:33 +01:00
ret = twl_i2c_read_u8 ( TWL4030_MODULE_GPIO , & data , address ) ;
2008-10-20 23:51:46 +02:00
return ( ret < 0 ) ? ret : data ;
}
/*----------------------------------------------------------------------*/
2012-12-20 10:44:11 +01:00
static u8 cached_leden ;
2008-10-20 23:51:46 +02:00
/* The LED lines are open drain outputs ... a FET pulls to GND, so an
* external pullup is needed . We could also expose the integrated PWM
* as a LED brightness control ; we initialize it as " always on " .
*/
static void twl4030_led_set_value ( int led , int value )
{
u8 mask = LEDEN_LEDAON | LEDEN_LEDAPWM ;
int status ;
if ( led )
mask < < = 1 ;
if ( value )
cached_leden & = ~ mask ;
else
cached_leden | = mask ;
2009-12-13 21:23:33 +01:00
status = twl_i2c_write_u8 ( TWL4030_MODULE_LED , cached_leden ,
2012-11-13 10:35:13 +01:00
TWL4030_LED_LEDEN_REG ) ;
2008-10-20 23:51:46 +02:00
}
static int twl4030_set_gpio_direction ( int gpio , int is_input )
{
u8 d_bnk = gpio > > 3 ;
u8 d_msk = BIT ( gpio & 0x7 ) ;
u8 reg = 0 ;
u8 base = REG_GPIODATADIR1 + d_bnk ;
int ret = 0 ;
ret = gpio_twl4030_read ( base ) ;
if ( ret > = 0 ) {
if ( is_input )
reg = ret & ~ d_msk ;
else
reg = ret | d_msk ;
ret = gpio_twl4030_write ( base , reg ) ;
}
return ret ;
}
static int twl4030_set_gpio_dataout ( int gpio , int enable )
{
u8 d_bnk = gpio > > 3 ;
u8 d_msk = BIT ( gpio & 0x7 ) ;
u8 base = 0 ;
if ( enable )
base = REG_SETGPIODATAOUT1 + d_bnk ;
else
base = REG_CLEARGPIODATAOUT1 + d_bnk ;
return gpio_twl4030_write ( base , d_msk ) ;
}
static int twl4030_get_gpio_datain ( int gpio )
{
u8 d_bnk = gpio > > 3 ;
u8 d_off = gpio & 0x7 ;
u8 base = 0 ;
int ret = 0 ;
base = REG_GPIODATAIN1 + d_bnk ;
ret = gpio_twl4030_read ( base ) ;
if ( ret > 0 )
ret = ( ret > > d_off ) & 0x1 ;
return ret ;
}
/*----------------------------------------------------------------------*/
static int twl_request ( struct gpio_chip * chip , unsigned offset )
{
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
2008-10-20 23:51:46 +02:00
int status = 0 ;
2012-12-20 10:44:11 +01:00
mutex_lock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
/* Support the two LED outputs as output-only GPIOs. */
if ( offset > = TWL4030_GPIO_MAX ) {
u8 ledclr_mask = LEDEN_LEDAON | LEDEN_LEDAEXT
| LEDEN_LEDAPWM | LEDEN_PWM_LENGTHA ;
2012-11-13 10:35:13 +01:00
u8 reg = TWL4030_PWMAON_REG ;
2008-10-20 23:51:46 +02:00
offset - = TWL4030_GPIO_MAX ;
if ( offset ) {
ledclr_mask < < = 1 ;
2012-11-13 10:35:13 +01:00
reg = TWL4030_PWMBON_REG ;
2008-10-20 23:51:46 +02:00
}
/* initialize PWM to always-drive */
2012-11-13 10:35:13 +01:00
/* Configure PWM OFF register first */
status = twl_i2c_write_u8 ( TWL4030_MODULE_LED , 0x7f , reg + 1 ) ;
2008-10-20 23:51:46 +02:00
if ( status < 0 )
goto done ;
2012-11-13 10:35:13 +01:00
/* Followed by PWM ON register */
status = twl_i2c_write_u8 ( TWL4030_MODULE_LED , 0x7f , reg ) ;
2008-10-20 23:51:46 +02:00
if ( status < 0 )
goto done ;
/* init LED to not-driven (high) */
2012-11-13 10:35:13 +01:00
status = twl_i2c_read_u8 ( TWL4030_MODULE_LED , & cached_leden ,
TWL4030_LED_LEDEN_REG ) ;
2008-10-20 23:51:46 +02:00
if ( status < 0 )
goto done ;
cached_leden & = ~ ledclr_mask ;
2012-11-13 10:35:13 +01:00
status = twl_i2c_write_u8 ( TWL4030_MODULE_LED , cached_leden ,
TWL4030_LED_LEDEN_REG ) ;
2008-10-20 23:51:46 +02:00
if ( status < 0 )
goto done ;
status = 0 ;
goto done ;
}
/* on first use, turn GPIO module "on" */
2012-12-06 10:52:05 +00:00
if ( ! priv - > usage_count ) {
2008-10-20 23:51:46 +02:00
struct twl4030_gpio_platform_data * pdata ;
u8 value = MASK_GPIO_CTRL_GPIO_ON ;
/* optionally have the first two GPIOs switch vMMC1
* and vMMC2 power supplies based on card presence .
*/
2013-07-30 17:08:05 +09:00
pdata = dev_get_platdata ( chip - > dev ) ;
2012-02-29 22:51:32 +01:00
if ( pdata )
value | = pdata - > mmc_cd & 0x03 ;
2008-10-20 23:51:46 +02:00
status = gpio_twl4030_write ( REG_GPIO_CTRL , value ) ;
}
2012-12-06 10:52:05 +00:00
done :
2008-10-20 23:51:46 +02:00
if ( ! status )
2012-12-06 10:52:05 +00:00
priv - > usage_count | = BIT ( offset ) ;
2008-10-20 23:51:46 +02:00
2012-12-20 10:44:11 +01:00
mutex_unlock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
return status ;
}
static void twl_free ( struct gpio_chip * chip , unsigned offset )
{
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
2012-12-20 10:44:11 +01:00
mutex_lock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
if ( offset > = TWL4030_GPIO_MAX ) {
twl4030_led_set_value ( offset - TWL4030_GPIO_MAX , 1 ) ;
2012-12-20 10:44:11 +01:00
goto out ;
2008-10-20 23:51:46 +02:00
}
2012-12-06 10:52:05 +00:00
priv - > usage_count & = ~ BIT ( offset ) ;
2008-10-20 23:51:46 +02:00
/* on last use, switch off GPIO module */
2012-12-06 10:52:05 +00:00
if ( ! priv - > usage_count )
2008-10-20 23:51:46 +02:00
gpio_twl4030_write ( REG_GPIO_CTRL , 0x0 ) ;
2012-12-20 10:44:11 +01:00
out :
mutex_unlock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
}
static int twl_direction_in ( struct gpio_chip * chip , unsigned offset )
{
2012-12-20 10:44:11 +01:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
int ret ;
mutex_lock ( & priv - > mutex ) ;
if ( offset < TWL4030_GPIO_MAX )
ret = twl4030_set_gpio_direction ( offset , 1 ) ;
else
ret = - EINVAL ;
if ( ! ret )
priv - > direction & = ~ BIT ( offset ) ;
mutex_unlock ( & priv - > mutex ) ;
return ret ;
2008-10-20 23:51:46 +02:00
}
static int twl_get ( struct gpio_chip * chip , unsigned offset )
{
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
2012-12-20 10:44:11 +01:00
int ret ;
2008-10-20 23:51:46 +02:00
int status = 0 ;
2012-12-20 10:44:11 +01:00
mutex_lock ( & priv - > mutex ) ;
if ( ! ( priv - > usage_count & BIT ( offset ) ) ) {
ret = - EPERM ;
goto out ;
}
2012-12-06 10:52:05 +00:00
2012-12-20 10:44:11 +01:00
if ( priv - > direction & BIT ( offset ) )
status = priv - > out_state & BIT ( offset ) ;
2008-10-20 23:51:46 +02:00
else
2012-12-20 10:44:11 +01:00
status = twl4030_get_gpio_datain ( offset ) ;
2012-12-06 10:52:05 +00:00
2012-12-20 10:44:11 +01:00
ret = ( status < = 0 ) ? 0 : 1 ;
out :
mutex_unlock ( & priv - > mutex ) ;
return ret ;
2008-10-20 23:51:46 +02:00
}
2012-12-20 10:44:11 +01:00
static void twl_set ( struct gpio_chip * chip , unsigned offset , int value )
2008-10-20 23:51:46 +02:00
{
2012-12-20 10:44:11 +01:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
mutex_lock ( & priv - > mutex ) ;
if ( offset < TWL4030_GPIO_MAX )
2008-10-20 23:51:46 +02:00
twl4030_set_gpio_dataout ( offset , value ) ;
2012-12-20 10:44:11 +01:00
else
2008-10-20 23:51:46 +02:00
twl4030_led_set_value ( offset - TWL4030_GPIO_MAX , value ) ;
2012-12-20 10:44:11 +01:00
if ( value )
priv - > out_state | = BIT ( offset ) ;
else
priv - > out_state & = ~ BIT ( offset ) ;
mutex_unlock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
}
2012-12-20 10:44:11 +01:00
static int twl_direction_out ( struct gpio_chip * chip , unsigned offset , int value )
2008-10-20 23:51:46 +02:00
{
2012-12-20 10:44:11 +01:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
mutex_lock ( & priv - > mutex ) ;
2008-10-20 23:51:46 +02:00
if ( offset < TWL4030_GPIO_MAX )
twl4030_set_gpio_dataout ( offset , value ) ;
2012-12-20 10:44:11 +01:00
priv - > direction | = BIT ( offset ) ;
mutex_unlock ( & priv - > mutex ) ;
twl_set ( chip , offset , value ) ;
return 0 ;
2008-10-20 23:51:46 +02:00
}
static int twl_to_irq ( struct gpio_chip * chip , unsigned offset )
{
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv = to_gpio_twl4030 ( chip ) ;
return ( priv - > irq_base & & ( offset < TWL4030_GPIO_MAX ) )
? ( priv - > irq_base + offset )
2008-10-20 23:51:46 +02:00
: - EINVAL ;
}
2012-12-06 10:52:05 +00:00
static struct gpio_chip template_chip = {
2008-10-20 23:51:46 +02:00
. label = " twl4030 " ,
. owner = THIS_MODULE ,
. request = twl_request ,
. free = twl_free ,
. direction_input = twl_direction_in ,
. get = twl_get ,
. direction_output = twl_direction_out ,
. set = twl_set ,
. to_irq = twl_to_irq ,
. can_sleep = 1 ,
} ;
/*----------------------------------------------------------------------*/
2012-11-19 13:22:34 -05:00
static int gpio_twl4030_pulls ( u32 ups , u32 downs )
2008-10-20 23:51:46 +02:00
{
2012-11-13 09:28:45 +01:00
u8 message [ 5 ] ;
2008-10-20 23:51:46 +02:00
unsigned i , gpio_bit ;
/* For most pins, a pulldown was enabled by default.
* We should have data that ' s specific to this board .
*/
2012-11-13 09:28:45 +01:00
for ( gpio_bit = 1 , i = 0 ; i < 5 ; i + + ) {
2008-10-20 23:51:46 +02:00
u8 bit_mask ;
unsigned j ;
for ( bit_mask = 0 , j = 0 ; j < 8 ; j + = 2 , gpio_bit < < = 1 ) {
if ( ups & gpio_bit )
bit_mask | = 1 < < ( j + 1 ) ;
else if ( downs & gpio_bit )
bit_mask | = 1 < < ( j + 0 ) ;
}
message [ i ] = bit_mask ;
}
2009-12-13 21:23:33 +01:00
return twl_i2c_write ( TWL4030_MODULE_GPIO , message ,
2008-10-20 23:51:46 +02:00
REG_GPIOPUPDCTR1 , 5 ) ;
}
2012-11-19 13:22:34 -05:00
static int gpio_twl4030_debounce ( u32 debounce , u8 mmc_cd )
2009-01-06 14:42:26 -08:00
{
2012-11-13 09:28:45 +01:00
u8 message [ 3 ] ;
2009-01-06 14:42:26 -08:00
/* 30 msec of debouncing is always used for MMC card detect,
* and is optional for everything else .
*/
2012-11-13 09:28:45 +01:00
message [ 0 ] = ( debounce & 0xff ) | ( mmc_cd & 0x03 ) ;
2009-01-06 14:42:26 -08:00
debounce > > = 8 ;
2012-11-13 09:28:45 +01:00
message [ 1 ] = ( debounce & 0xff ) ;
2009-01-06 14:42:26 -08:00
debounce > > = 8 ;
2012-11-13 09:28:45 +01:00
message [ 2 ] = ( debounce & 0x03 ) ;
2009-01-06 14:42:26 -08:00
2009-12-13 21:23:33 +01:00
return twl_i2c_write ( TWL4030_MODULE_GPIO , message ,
2009-01-06 14:42:26 -08:00
REG_GPIO_DEBEN1 , 3 ) ;
}
2008-10-20 23:51:46 +02:00
static int gpio_twl4030_remove ( struct platform_device * pdev ) ;
2012-09-05 09:46:25 +02:00
static struct twl4030_gpio_platform_data * of_gpio_twl4030 ( struct device * dev )
{
struct twl4030_gpio_platform_data * omap_twl_info ;
omap_twl_info = devm_kzalloc ( dev , sizeof ( * omap_twl_info ) , GFP_KERNEL ) ;
if ( ! omap_twl_info )
return NULL ;
omap_twl_info - > use_leds = of_property_read_bool ( dev - > of_node ,
" ti,use-leds " ) ;
of_property_read_u32 ( dev - > of_node , " ti,debounce " ,
& omap_twl_info - > debounce ) ;
of_property_read_u32 ( dev - > of_node , " ti,mmc-cd " ,
( u32 * ) & omap_twl_info - > mmc_cd ) ;
of_property_read_u32 ( dev - > of_node , " ti,pullups " ,
& omap_twl_info - > pullups ) ;
of_property_read_u32 ( dev - > of_node , " ti,pulldowns " ,
& omap_twl_info - > pulldowns ) ;
return omap_twl_info ;
}
2012-11-19 13:22:34 -05:00
static int gpio_twl4030_probe ( struct platform_device * pdev )
2008-10-20 23:51:46 +02:00
{
2013-07-30 17:08:05 +09:00
struct twl4030_gpio_platform_data * pdata = dev_get_platdata ( & pdev - > dev ) ;
2012-02-29 22:51:32 +01:00
struct device_node * node = pdev - > dev . of_node ;
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv ;
2012-02-29 22:48:32 +01:00
int ret , irq_base ;
2008-10-20 23:51:46 +02:00
2012-12-06 10:52:05 +00:00
priv = devm_kzalloc ( & pdev - > dev , sizeof ( struct gpio_twl4030_priv ) ,
GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
2008-10-20 23:51:46 +02:00
/* maybe setup IRQs */
2012-02-29 22:48:32 +01:00
if ( is_module ( ) ) {
dev_err ( & pdev - > dev , " can't dispatch IRQs from modules \n " ) ;
goto no_irqs ;
}
irq_base = irq_alloc_descs ( - 1 , 0 , TWL4030_GPIO_MAX , 0 ) ;
if ( irq_base < 0 ) {
dev_err ( & pdev - > dev , " Failed to alloc irq_descs \n " ) ;
return irq_base ;
2008-10-20 23:51:46 +02:00
}
2012-02-29 22:51:32 +01:00
irq_domain_add_legacy ( node , TWL4030_GPIO_MAX , irq_base , 0 ,
& irq_domain_simple_ops , NULL ) ;
2012-02-29 22:48:32 +01:00
ret = twl4030_sih_setup ( & pdev - > dev , TWL4030_MODULE_GPIO , irq_base ) ;
if ( ret < 0 )
return ret ;
2012-12-06 10:52:05 +00:00
priv - > irq_base = irq_base ;
2012-02-29 22:48:32 +01:00
2008-10-20 23:51:46 +02:00
no_irqs :
2012-12-06 10:52:05 +00:00
priv - > gpio_chip = template_chip ;
priv - > gpio_chip . base = - 1 ;
priv - > gpio_chip . ngpio = TWL4030_GPIO_MAX ;
priv - > gpio_chip . dev = & pdev - > dev ;
2008-10-20 23:51:46 +02:00
2012-12-20 10:44:11 +01:00
mutex_init ( & priv - > mutex ) ;
2012-09-05 09:46:25 +02:00
if ( node )
pdata = of_gpio_twl4030 ( & pdev - > dev ) ;
2012-02-29 22:51:32 +01:00
2012-09-05 09:46:25 +02:00
if ( pdata = = NULL ) {
dev_err ( & pdev - > dev , " Platform data is missing \n " ) ;
return - ENXIO ;
2012-02-29 22:51:32 +01:00
}
2008-10-20 23:51:46 +02:00
2012-09-05 09:46:25 +02:00
/*
* NOTE : boards may waste power if they don ' t set pullups
* and pulldowns correctly . . . default for non - ULPI pins is
* pulldown , and some other pins may have external pullups
* or pulldowns . Careful !
*/
ret = gpio_twl4030_pulls ( pdata - > pullups , pdata - > pulldowns ) ;
if ( ret )
dev_dbg ( & pdev - > dev , " pullups %.05x %.05x --> %d \n " ,
pdata - > pullups , pdata - > pulldowns , ret ) ;
ret = gpio_twl4030_debounce ( pdata - > debounce , pdata - > mmc_cd ) ;
if ( ret )
dev_dbg ( & pdev - > dev , " debounce %.03x %.01x --> %d \n " ,
pdata - > debounce , pdata - > mmc_cd , ret ) ;
/*
* NOTE : we assume VIBRA_CTL . VIBRA_EN , in MODULE_AUDIO_VOICE ,
* is ( still ) clear if use_leds is set .
*/
if ( pdata - > use_leds )
2012-12-06 10:52:05 +00:00
priv - > gpio_chip . ngpio + = 2 ;
2012-09-05 09:46:25 +02:00
2012-12-06 10:52:05 +00:00
ret = gpiochip_add ( & priv - > gpio_chip ) ;
2008-10-20 23:51:46 +02:00
if ( ret < 0 ) {
2012-02-29 22:48:32 +01:00
dev_err ( & pdev - > dev , " could not register gpiochip, %d \n " , ret ) ;
2012-12-06 10:52:05 +00:00
priv - > gpio_chip . ngpio = 0 ;
2008-10-20 23:51:46 +02:00
gpio_twl4030_remove ( pdev ) ;
2012-09-04 17:43:30 -07:00
goto out ;
}
2012-12-06 10:52:05 +00:00
platform_set_drvdata ( pdev , priv ) ;
2012-09-04 17:43:30 -07:00
if ( pdata & & pdata - > setup ) {
2008-10-20 23:51:46 +02:00
int status ;
2012-12-06 10:52:05 +00:00
status = pdata - > setup ( & pdev - > dev , priv - > gpio_chip . base ,
TWL4030_GPIO_MAX ) ;
2008-10-20 23:51:46 +02:00
if ( status )
dev_dbg ( & pdev - > dev , " setup --> %d \n " , status ) ;
}
2012-09-04 17:43:30 -07:00
out :
2008-10-20 23:51:46 +02:00
return ret ;
}
2012-11-19 13:25:50 -05:00
/* Cannot use as gpio_twl4030_probe() calls us */
2009-10-26 16:50:06 -07:00
static int gpio_twl4030_remove ( struct platform_device * pdev )
2008-10-20 23:51:46 +02:00
{
2013-07-30 17:08:05 +09:00
struct twl4030_gpio_platform_data * pdata = dev_get_platdata ( & pdev - > dev ) ;
2012-12-06 10:52:05 +00:00
struct gpio_twl4030_priv * priv = platform_get_drvdata ( pdev ) ;
2008-10-20 23:51:46 +02:00
int status ;
2012-02-29 22:51:32 +01:00
if ( pdata & & pdata - > teardown ) {
2012-12-06 10:52:05 +00:00
status = pdata - > teardown ( & pdev - > dev , priv - > gpio_chip . base ,
TWL4030_GPIO_MAX ) ;
2008-10-20 23:51:46 +02:00
if ( status ) {
dev_dbg ( & pdev - > dev , " teardown --> %d \n " , status ) ;
return status ;
}
}
2012-12-06 10:52:05 +00:00
status = gpiochip_remove ( & priv - > gpio_chip ) ;
2008-10-20 23:51:46 +02:00
if ( status < 0 )
return status ;
if ( is_module ( ) )
return 0 ;
/* REVISIT no support yet for deregistering all the IRQs */
WARN_ON ( 1 ) ;
return - EIO ;
}
2012-02-29 22:51:32 +01:00
static const struct of_device_id twl_gpio_match [ ] = {
{ . compatible = " ti,twl4030-gpio " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , twl_gpio_match ) ;
2008-10-20 23:51:46 +02:00
/* Note: this hardware lives inside an I2C-based multi-function device. */
MODULE_ALIAS ( " platform:twl4030_gpio " ) ;
static struct platform_driver gpio_twl4030_driver = {
2012-02-29 22:51:32 +01:00
. driver = {
. name = " twl4030_gpio " ,
. owner = THIS_MODULE ,
. of_match_table = of_match_ptr ( twl_gpio_match ) ,
} ,
2008-10-20 23:51:46 +02:00
. probe = gpio_twl4030_probe ,
2009-10-26 16:50:06 -07:00
. remove = gpio_twl4030_remove ,
2008-10-20 23:51:46 +02:00
} ;
static int __init gpio_twl4030_init ( void )
{
return platform_driver_register ( & gpio_twl4030_driver ) ;
}
subsys_initcall ( gpio_twl4030_init ) ;
static void __exit gpio_twl4030_exit ( void )
{
platform_driver_unregister ( & gpio_twl4030_driver ) ;
}
module_exit ( gpio_twl4030_exit ) ;
MODULE_AUTHOR ( " Texas Instruments, Inc. " ) ;
MODULE_DESCRIPTION ( " GPIO interface for TWL4030 " ) ;
MODULE_LICENSE ( " GPL " ) ;