2011-06-09 14:50:19 -05:00
/*
* Copyright 2011 Texas Instruments Inc .
*
* Author : Margarita Olaya < magi @ slimlogic . co . uk >
*
* 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 driver is based on wm8350 implementation .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/errno.h>
# include <linux/gpio.h>
# include <linux/mfd/core.h>
# include <linux/platform_device.h>
# include <linux/seq_file.h>
# include <linux/slab.h>
# include <linux/mfd/tps65912.h>
struct tps65912_gpio_data {
struct tps65912 * tps65912 ;
struct gpio_chip gpio_chip ;
} ;
static int tps65912_gpio_get ( struct gpio_chip * gc , unsigned offset )
{
2015-12-07 14:50:42 +01:00
struct tps65912_gpio_data * tps65912_gpio = gpiochip_get_data ( gc ) ;
2015-02-19 01:52:25 +00:00
struct tps65912 * tps65912 = tps65912_gpio - > tps65912 ;
2011-06-09 14:50:19 -05:00
int val ;
val = tps65912_reg_read ( tps65912 , TPS65912_GPIO1 + offset ) ;
if ( val & GPIO_STS_MASK )
return 1 ;
return 0 ;
}
static void tps65912_gpio_set ( struct gpio_chip * gc , unsigned offset ,
int value )
{
2015-12-07 14:50:42 +01:00
struct tps65912_gpio_data * tps65912_gpio = gpiochip_get_data ( gc ) ;
2015-02-19 01:52:25 +00:00
struct tps65912 * tps65912 = tps65912_gpio - > tps65912 ;
2011-06-09 14:50:19 -05:00
if ( value )
tps65912_set_bits ( tps65912 , TPS65912_GPIO1 + offset ,
GPIO_SET_MASK ) ;
else
tps65912_clear_bits ( tps65912 , TPS65912_GPIO1 + offset ,
GPIO_SET_MASK ) ;
}
static int tps65912_gpio_output ( struct gpio_chip * gc , unsigned offset ,
int value )
{
2015-12-07 14:50:42 +01:00
struct tps65912_gpio_data * tps65912_gpio = gpiochip_get_data ( gc ) ;
2015-02-19 01:52:25 +00:00
struct tps65912 * tps65912 = tps65912_gpio - > tps65912 ;
2011-06-09 14:50:19 -05:00
/* Set the initial value */
tps65912_gpio_set ( gc , offset , value ) ;
return tps65912_set_bits ( tps65912 , TPS65912_GPIO1 + offset ,
GPIO_CFG_MASK ) ;
}
static int tps65912_gpio_input ( struct gpio_chip * gc , unsigned offset )
{
2015-12-07 14:50:42 +01:00
struct tps65912_gpio_data * tps65912_gpio = gpiochip_get_data ( gc ) ;
2015-02-19 01:52:25 +00:00
struct tps65912 * tps65912 = tps65912_gpio - > tps65912 ;
2011-06-09 14:50:19 -05:00
return tps65912_clear_bits ( tps65912 , TPS65912_GPIO1 + offset ,
GPIO_CFG_MASK ) ;
}
static struct gpio_chip template_chip = {
. label = " tps65912 " ,
. owner = THIS_MODULE ,
. direction_input = tps65912_gpio_input ,
. direction_output = tps65912_gpio_output ,
. get = tps65912_gpio_get ,
. set = tps65912_gpio_set ,
2013-12-04 14:42:46 +01:00
. can_sleep = true ,
2011-06-09 14:50:19 -05:00
. ngpio = 5 ,
. base = - 1 ,
} ;
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
{
struct tps65912 * tps65912 = dev_get_drvdata ( pdev - > dev . parent ) ;
2013-07-30 17:08:05 +09:00
struct tps65912_board * pdata = dev_get_platdata ( tps65912 - > dev ) ;
2011-06-09 14:50:19 -05:00
struct tps65912_gpio_data * tps65912_gpio ;
int ret ;
2012-09-01 17:44:27 +08:00
tps65912_gpio = devm_kzalloc ( & pdev - > dev , sizeof ( * tps65912_gpio ) ,
GFP_KERNEL ) ;
2011-06-09 14:50:19 -05:00
if ( tps65912_gpio = = NULL )
return - ENOMEM ;
tps65912_gpio - > tps65912 = tps65912 ;
tps65912_gpio - > gpio_chip = template_chip ;
2015-11-04 09:56:26 +01:00
tps65912_gpio - > gpio_chip . parent = & pdev - > dev ;
2011-06-09 14:50:19 -05:00
if ( pdata & & pdata - > gpio_base )
tps65912_gpio - > gpio_chip . base = pdata - > gpio_base ;
2015-12-07 14:50:42 +01:00
ret = gpiochip_add_data ( & tps65912_gpio - > gpio_chip , tps65912_gpio ) ;
2011-06-09 14:50:19 -05:00
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " Failed to register gpiochip, %d \n " , ret ) ;
2012-09-01 17:44:27 +08:00
return ret ;
2011-06-09 14:50:19 -05:00
}
platform_set_drvdata ( pdev , tps65912_gpio ) ;
return ret ;
}
2012-11-19 13:25:50 -05:00
static int tps65912_gpio_remove ( struct platform_device * pdev )
2011-06-09 14:50:19 -05:00
{
struct tps65912_gpio_data * tps65912_gpio = platform_get_drvdata ( pdev ) ;
2014-07-12 22:30:12 +02:00
gpiochip_remove ( & tps65912_gpio - > gpio_chip ) ;
return 0 ;
2011-06-09 14:50:19 -05:00
}
static struct platform_driver tps65912_gpio_driver = {
. driver = {
. name = " tps65912-gpio " ,
} ,
. probe = tps65912_gpio_probe ,
2012-11-19 13:20:08 -05:00
. remove = tps65912_gpio_remove ,
2011-06-09 14:50:19 -05:00
} ;
static int __init tps65912_gpio_init ( void )
{
return platform_driver_register ( & tps65912_gpio_driver ) ;
}
subsys_initcall ( tps65912_gpio_init ) ;
static void __exit tps65912_gpio_exit ( void )
{
platform_driver_unregister ( & tps65912_gpio_driver ) ;
}
module_exit ( tps65912_gpio_exit ) ;
MODULE_AUTHOR ( " Margarita Olaya Cabrera <magi@slimlogic.co.uk> " ) ;
MODULE_DESCRIPTION ( " GPIO interface for TPS65912 PMICs " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform:tps65912-gpio " ) ;