2021-01-12 18:01:01 +09:00
// SPDX-License-Identifier: GPL-2.0-only
2017-04-25 20:32:09 +02:00
/*
2021-01-12 18:01:03 +09:00
* ROHM BD9571MWV - M and BD9574MWF - M GPIO driver
2017-04-25 20:32:09 +02:00
*
* Copyright ( C ) 2017 Marek Vasut < marek . vasut + renesas @ gmail . com >
*
* Based on the TPS65086 driver
*
* NOTE : Interrupts are not supported yet .
*/
# include <linux/gpio/driver.h>
2021-01-12 18:01:03 +09:00
# include <linux/mfd/rohm-generic.h>
2017-04-25 20:32:09 +02:00
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/mfd/bd9571mwv.h>
struct bd9571mwv_gpio {
2021-01-12 18:01:02 +09:00
struct regmap * regmap ;
2017-04-25 20:32:09 +02:00
struct gpio_chip chip ;
} ;
static int bd9571mwv_gpio_get_direction ( struct gpio_chip * chip ,
unsigned int offset )
{
struct bd9571mwv_gpio * gpio = gpiochip_get_data ( chip ) ;
int ret , val ;
2021-01-12 18:01:02 +09:00
ret = regmap_read ( gpio - > regmap , BD9571MWV_GPIO_DIR , & val ) ;
2017-04-25 20:32:09 +02:00
if ( ret < 0 )
return ret ;
2019-11-06 10:54:12 +02:00
if ( val & BIT ( offset ) )
return GPIO_LINE_DIRECTION_IN ;
2017-04-25 20:32:09 +02:00
2019-11-06 10:54:12 +02:00
return GPIO_LINE_DIRECTION_OUT ;
2017-04-25 20:32:09 +02:00
}
static int bd9571mwv_gpio_direction_input ( struct gpio_chip * chip ,
unsigned int offset )
{
struct bd9571mwv_gpio * gpio = gpiochip_get_data ( chip ) ;
2021-01-12 18:01:02 +09:00
regmap_update_bits ( gpio - > regmap , BD9571MWV_GPIO_DIR , BIT ( offset ) , 0 ) ;
2017-04-25 20:32:09 +02:00
return 0 ;
}
static int bd9571mwv_gpio_direction_output ( struct gpio_chip * chip ,
unsigned int offset , int value )
{
struct bd9571mwv_gpio * gpio = gpiochip_get_data ( chip ) ;
/* Set the initial value */
2021-01-12 18:01:02 +09:00
regmap_update_bits ( gpio - > regmap , BD9571MWV_GPIO_OUT ,
2017-04-25 20:32:09 +02:00
BIT ( offset ) , value ? BIT ( offset ) : 0 ) ;
2021-01-12 18:01:02 +09:00
regmap_update_bits ( gpio - > regmap , BD9571MWV_GPIO_DIR ,
2017-04-25 20:32:09 +02:00
BIT ( offset ) , BIT ( offset ) ) ;
return 0 ;
}
static int bd9571mwv_gpio_get ( struct gpio_chip * chip , unsigned int offset )
{
struct bd9571mwv_gpio * gpio = gpiochip_get_data ( chip ) ;
int ret , val ;
2021-01-12 18:01:02 +09:00
ret = regmap_read ( gpio - > regmap , BD9571MWV_GPIO_IN , & val ) ;
2017-04-25 20:32:09 +02:00
if ( ret < 0 )
return ret ;
return val & BIT ( offset ) ;
}
static void bd9571mwv_gpio_set ( struct gpio_chip * chip , unsigned int offset ,
int value )
{
struct bd9571mwv_gpio * gpio = gpiochip_get_data ( chip ) ;
2021-01-12 18:01:02 +09:00
regmap_update_bits ( gpio - > regmap , BD9571MWV_GPIO_OUT ,
2017-04-25 20:32:09 +02:00
BIT ( offset ) , value ? BIT ( offset ) : 0 ) ;
}
static const struct gpio_chip template_chip = {
. label = " bd9571mwv-gpio " ,
. owner = THIS_MODULE ,
. get_direction = bd9571mwv_gpio_get_direction ,
. direction_input = bd9571mwv_gpio_direction_input ,
. direction_output = bd9571mwv_gpio_direction_output ,
. get = bd9571mwv_gpio_get ,
. set = bd9571mwv_gpio_set ,
. base = - 1 ,
. ngpio = 2 ,
. can_sleep = true ,
} ;
static int bd9571mwv_gpio_probe ( struct platform_device * pdev )
{
struct bd9571mwv_gpio * gpio ;
gpio = devm_kzalloc ( & pdev - > dev , sizeof ( * gpio ) , GFP_KERNEL ) ;
if ( ! gpio )
return - ENOMEM ;
2021-01-12 18:01:02 +09:00
gpio - > regmap = dev_get_regmap ( pdev - > dev . parent , NULL ) ;
2017-04-25 20:32:09 +02:00
gpio - > chip = template_chip ;
2021-01-12 18:01:02 +09:00
gpio - > chip . parent = pdev - > dev . parent ;
2017-04-25 20:32:09 +02:00
2021-05-14 12:26:14 +03:00
return devm_gpiochip_add_data ( & pdev - > dev , & gpio - > chip , gpio ) ;
2017-04-25 20:32:09 +02:00
}
static const struct platform_device_id bd9571mwv_gpio_id_table [ ] = {
2021-01-12 18:01:03 +09:00
{ " bd9571mwv-gpio " , ROHM_CHIP_TYPE_BD9571 } ,
{ " bd9574mwf-gpio " , ROHM_CHIP_TYPE_BD9574 } ,
2017-04-25 20:32:09 +02:00
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( platform , bd9571mwv_gpio_id_table ) ;
static struct platform_driver bd9571mwv_gpio_driver = {
. driver = {
. name = " bd9571mwv-gpio " ,
} ,
. probe = bd9571mwv_gpio_probe ,
. id_table = bd9571mwv_gpio_id_table ,
} ;
module_platform_driver ( bd9571mwv_gpio_driver ) ;
MODULE_AUTHOR ( " Marek Vasut <marek.vasut+renesas@gmail.com> " ) ;
MODULE_DESCRIPTION ( " BD9571MWV GPIO driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;