2011-06-09 14:50:19 -05:00
/*
2016-01-25 09:43:47 -06:00
* GPIO driver for TI TPS65912x PMICs
2011-06-09 14:50:19 -05:00
*
2016-01-25 09:43:47 -06:00
* Copyright ( C ) 2015 Texas Instruments Incorporated - http : //www.ti.com/
* Andrew F . Davis < afd @ ti . com >
2011-06-09 14:50:19 -05:00
*
2016-01-25 09:43:47 -06:00
* 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 .
2011-06-09 14:50:19 -05:00
*
2016-01-25 09:43:47 -06:00
* This program is distributed " as is " WITHOUT ANY WARRANTY of any
* kind , whether expressed or implied ; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License version 2 for more details .
*
* Based on the Arizona GPIO driver and the previous TPS65912 driver by
* Margarita Olaya Cabrera < magi @ slimlogic . co . uk >
2011-06-09 14:50:19 -05:00
*/
# include <linux/gpio.h>
2016-01-25 09:43:47 -06:00
# include <linux/module.h>
2011-06-09 14:50:19 -05:00
# include <linux/platform_device.h>
2016-01-25 09:43:47 -06:00
2011-06-09 14:50:19 -05:00
# include <linux/mfd/tps65912.h>
2016-01-25 09:43:47 -06:00
struct tps65912_gpio {
2011-06-09 14:50:19 -05:00
struct gpio_chip gpio_chip ;
2016-01-25 09:43:47 -06:00
struct tps65912 * tps ;
2011-06-09 14:50:19 -05:00
} ;
2016-01-25 09:43:47 -06:00
static int tps65912_gpio_get_direction ( struct gpio_chip * gc ,
unsigned offset )
2011-06-09 14:50:19 -05:00
{
2016-01-25 09:43:47 -06:00
struct tps65912_gpio * gpio = gpiochip_get_data ( gc ) ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
int ret , val ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
ret = regmap_read ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset , & val ) ;
if ( ret )
return ret ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
if ( val & GPIO_CFG_MASK )
return GPIOF_DIR_OUT ;
else
return GPIOF_DIR_IN ;
2011-06-09 14:50:19 -05:00
}
2016-01-25 09:43:47 -06:00
static int tps65912_gpio_direction_input ( struct gpio_chip * gc , unsigned offset )
2011-06-09 14:50:19 -05:00
{
2016-01-25 09:43:47 -06:00
struct tps65912_gpio * gpio = gpiochip_get_data ( gc ) ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
return regmap_update_bits ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset ,
GPIO_CFG_MASK , 0 ) ;
2011-06-09 14:50:19 -05:00
}
2016-01-25 09:43:47 -06:00
static int tps65912_gpio_direction_output ( struct gpio_chip * gc ,
unsigned offset , int value )
2011-06-09 14:50:19 -05:00
{
2016-01-25 09:43:47 -06:00
struct tps65912_gpio * gpio = gpiochip_get_data ( gc ) ;
2011-06-09 14:50:19 -05:00
/* Set the initial value */
2016-01-25 09:43:47 -06:00
regmap_update_bits ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset ,
GPIO_SET_MASK , value ? GPIO_SET_MASK : 0 ) ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
return regmap_update_bits ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset ,
GPIO_CFG_MASK , GPIO_CFG_MASK ) ;
2011-06-09 14:50:19 -05:00
}
2016-01-25 09:43:47 -06:00
static int tps65912_gpio_get ( struct gpio_chip * gc , unsigned offset )
2011-06-09 14:50:19 -05:00
{
2016-01-25 09:43:47 -06:00
struct tps65912_gpio * gpio = gpiochip_get_data ( gc ) ;
int ret , val ;
ret = regmap_read ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset , & val ) ;
if ( ret )
return ret ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
if ( val & GPIO_STS_MASK )
return 1 ;
return 0 ;
}
static void tps65912_gpio_set ( struct gpio_chip * gc , unsigned offset ,
int value )
{
struct tps65912_gpio * gpio = gpiochip_get_data ( gc ) ;
regmap_update_bits ( gpio - > tps - > regmap , TPS65912_GPIO1 + offset ,
GPIO_SET_MASK , value ? GPIO_SET_MASK : 0 ) ;
2011-06-09 14:50:19 -05:00
}
static struct gpio_chip template_chip = {
2016-01-25 09:43:47 -06:00
. label = " tps65912-gpio " ,
2011-06-09 14:50:19 -05:00
. owner = THIS_MODULE ,
2016-01-25 09:43:47 -06:00
. get_direction = tps65912_gpio_get_direction ,
. direction_input = tps65912_gpio_direction_input ,
. direction_output = tps65912_gpio_direction_output ,
2011-06-09 14:50:19 -05:00
. get = tps65912_gpio_get ,
. set = tps65912_gpio_set ,
. base = - 1 ,
2016-01-25 09:43:47 -06:00
. ngpio = 5 ,
. can_sleep = true ,
2011-06-09 14:50:19 -05:00
} ;
2012-11-19 13:22:34 -05:00
static int tps65912_gpio_probe ( struct platform_device * pdev )
2011-06-09 14:50:19 -05:00
{
2016-01-25 09:43:47 -06:00
struct tps65912 * tps = dev_get_drvdata ( pdev - > dev . parent ) ;
struct tps65912_gpio * gpio ;
2011-06-09 14:50:19 -05:00
int ret ;
2016-01-25 09:43:47 -06:00
gpio = devm_kzalloc ( & pdev - > dev , sizeof ( * gpio ) , GFP_KERNEL ) ;
if ( ! gpio )
2011-06-09 14:50:19 -05:00
return - ENOMEM ;
2016-01-25 09:43:47 -06:00
gpio - > tps = dev_get_drvdata ( pdev - > dev . parent ) ;
gpio - > gpio_chip = template_chip ;
gpio - > gpio_chip . parent = tps - > dev ;
2011-06-09 14:50:19 -05:00
2016-02-22 17:43:28 +05:30
ret = devm_gpiochip_add_data ( & pdev - > dev , & tps65912_gpio - > gpio_chip ,
tps65912_gpio ) ;
2011-06-09 14:50:19 -05:00
if ( ret < 0 ) {
2016-01-25 09:43:47 -06:00
dev_err ( & pdev - > dev , " Could not register gpiochip, %d \n " , ret ) ;
2012-09-01 17:44:27 +08:00
return ret ;
2011-06-09 14:50:19 -05:00
}
2016-01-25 09:43:47 -06:00
platform_set_drvdata ( pdev , gpio ) ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
return 0 ;
2011-06-09 14:50:19 -05:00
}
2016-01-25 09:43:47 -06:00
static const struct platform_device_id tps65912_gpio_id_table [ ] = {
{ " tps65912-gpio " , } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( platform , tps65912_gpio_id_table ) ;
2011-06-09 14:50:19 -05:00
static struct platform_driver tps65912_gpio_driver = {
. driver = {
. name = " tps65912-gpio " ,
} ,
. probe = tps65912_gpio_probe ,
2016-01-25 09:43:47 -06:00
. id_table = tps65912_gpio_id_table ,
2011-06-09 14:50:19 -05:00
} ;
2016-01-25 09:43:47 -06:00
module_platform_driver ( tps65912_gpio_driver ) ;
2011-06-09 14:50:19 -05:00
2016-01-25 09:43:47 -06:00
MODULE_AUTHOR ( " Andrew F. Davis <afd@ti.com> " ) ;
MODULE_DESCRIPTION ( " TPS65912 GPIO driver " ) ;
2011-06-09 14:50:19 -05:00
MODULE_LICENSE ( " GPL v2 " ) ;