2018-12-12 11:57:15 +00:00
// SPDX-License-Identifier: GPL-2.0
/*
* SAMA5D2 PIOBU GPIO controller
*
* Copyright ( C ) 2018 Microchip Technology Inc . and its subsidiaries
*
* Author : Andrei Stefanescu < andrei . stefanescu @ microchip . com >
*
*/
# include <linux/bits.h>
# include <linux/gpio/driver.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/mfd/syscon.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/regmap.h>
# define PIOBU_NUM 8
# define PIOBU_REG_SIZE 4
/*
* backup mode protection register for tamper detection
* normal mode protection register for tamper detection
* wakeup signal generation
*/
# define PIOBU_BMPR 0x7C
# define PIOBU_NMPR 0x80
# define PIOBU_WKPR 0x90
# define PIOBU_BASE 0x18 /* PIOBU offset from SECUMOD base register address. */
# define PIOBU_DET_OFFSET 16
/* In the datasheet this bit is called OUTPUT */
# define PIOBU_DIRECTION BIT(8)
# define PIOBU_OUT BIT(8)
# define PIOBU_IN 0
# define PIOBU_SOD BIT(9)
# define PIOBU_PDS BIT(10)
# define PIOBU_HIGH BIT(9)
# define PIOBU_LOW 0
struct sama5d2_piobu {
struct gpio_chip chip ;
struct regmap * regmap ;
} ;
/**
* sama5d2_piobu_setup_pin ( ) - prepares a pin for set_direction call
*
* Do not consider pin for tamper detection ( normal and backup modes )
* Do not consider pin as tamper wakeup interrupt source
*/
static int sama5d2_piobu_setup_pin ( struct gpio_chip * chip , unsigned int pin )
{
int ret ;
struct sama5d2_piobu * piobu = container_of ( chip , struct sama5d2_piobu ,
chip ) ;
unsigned int mask = BIT ( PIOBU_DET_OFFSET + pin ) ;
ret = regmap_update_bits ( piobu - > regmap , PIOBU_BMPR , mask , 0 ) ;
if ( ret )
return ret ;
ret = regmap_update_bits ( piobu - > regmap , PIOBU_NMPR , mask , 0 ) ;
if ( ret )
return ret ;
return regmap_update_bits ( piobu - > regmap , PIOBU_WKPR , mask , 0 ) ;
}
/**
* sama5d2_piobu_write_value ( ) - writes value & mask at the pin ' s PIOBU register
*/
static int sama5d2_piobu_write_value ( struct gpio_chip * chip , unsigned int pin ,
unsigned int mask , unsigned int value )
{
int reg ;
struct sama5d2_piobu * piobu = container_of ( chip , struct sama5d2_piobu ,
chip ) ;
reg = PIOBU_BASE + pin * PIOBU_REG_SIZE ;
return regmap_update_bits ( piobu - > regmap , reg , mask , value ) ;
}
/**
* sama5d2_piobu_read_value ( ) - read the value with masking from the pin ' s PIOBU
* register
*/
static int sama5d2_piobu_read_value ( struct gpio_chip * chip , unsigned int pin ,
unsigned int mask )
{
struct sama5d2_piobu * piobu = container_of ( chip , struct sama5d2_piobu ,
chip ) ;
unsigned int val , reg ;
int ret ;
reg = PIOBU_BASE + pin * PIOBU_REG_SIZE ;
ret = regmap_read ( piobu - > regmap , reg , & val ) ;
if ( ret < 0 )
return ret ;
return val & mask ;
}
/**
* sama5d2_piobu_get_direction ( ) - gpiochip get_direction
*/
static int sama5d2_piobu_get_direction ( struct gpio_chip * chip ,
unsigned int pin )
{
int ret = sama5d2_piobu_read_value ( chip , pin , PIOBU_DIRECTION ) ;
if ( ret < 0 )
return ret ;
return ( ret = = PIOBU_IN ) ? 1 : 0 ;
}
/**
* sama5d2_piobu_direction_input ( ) - gpiochip direction_input
*/
static int sama5d2_piobu_direction_input ( struct gpio_chip * chip ,
unsigned int pin )
{
2018-12-31 17:52:56 +08:00
return sama5d2_piobu_write_value ( chip , pin , PIOBU_DIRECTION , PIOBU_IN ) ;
2018-12-12 11:57:15 +00:00
}
/**
* sama5d2_piobu_direction_output ( ) - gpiochip direction_output
*/
static int sama5d2_piobu_direction_output ( struct gpio_chip * chip ,
unsigned int pin , int value )
{
2018-12-31 17:52:56 +08:00
unsigned int val = PIOBU_OUT ;
if ( value )
val | = PIOBU_HIGH ;
return sama5d2_piobu_write_value ( chip , pin , PIOBU_DIRECTION | PIOBU_SOD ,
val ) ;
2018-12-12 11:57:15 +00:00
}
/**
* sama5d2_piobu_get ( ) - gpiochip get
*/
static int sama5d2_piobu_get ( struct gpio_chip * chip , unsigned int pin )
{
/* if pin is input, read value from PDS else read from SOD */
int ret = sama5d2_piobu_get_direction ( chip , pin ) ;
if ( ret = = 1 )
ret = sama5d2_piobu_read_value ( chip , pin , PIOBU_PDS ) ;
else if ( ! ret )
ret = sama5d2_piobu_read_value ( chip , pin , PIOBU_SOD ) ;
if ( ret < 0 )
return ret ;
return ! ! ret ;
}
/**
* sama5d2_piobu_set ( ) - gpiochip set
*/
static void sama5d2_piobu_set ( struct gpio_chip * chip , unsigned int pin ,
int value )
{
if ( ! value )
value = PIOBU_LOW ;
else
value = PIOBU_HIGH ;
sama5d2_piobu_write_value ( chip , pin , PIOBU_SOD , value ) ;
}
static int sama5d2_piobu_probe ( struct platform_device * pdev )
{
struct sama5d2_piobu * piobu ;
int ret , i ;
piobu = devm_kzalloc ( & pdev - > dev , sizeof ( * piobu ) , GFP_KERNEL ) ;
if ( ! piobu )
return - ENOMEM ;
platform_set_drvdata ( pdev , piobu ) ;
piobu - > chip . label = pdev - > name ;
piobu - > chip . parent = & pdev - > dev ;
piobu - > chip . of_node = pdev - > dev . of_node ;
piobu - > chip . owner = THIS_MODULE ,
piobu - > chip . get_direction = sama5d2_piobu_get_direction ,
piobu - > chip . direction_input = sama5d2_piobu_direction_input ,
piobu - > chip . direction_output = sama5d2_piobu_direction_output ,
piobu - > chip . get = sama5d2_piobu_get ,
piobu - > chip . set = sama5d2_piobu_set ,
piobu - > chip . base = - 1 ,
piobu - > chip . ngpio = PIOBU_NUM ,
piobu - > chip . can_sleep = 0 ,
piobu - > regmap = syscon_node_to_regmap ( pdev - > dev . of_node ) ;
if ( IS_ERR ( piobu - > regmap ) ) {
dev_err ( & pdev - > dev , " Failed to get syscon regmap %ld \n " ,
PTR_ERR ( piobu - > regmap ) ) ;
return PTR_ERR ( piobu - > regmap ) ;
}
ret = devm_gpiochip_add_data ( & pdev - > dev , & piobu - > chip , piobu ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Failed to add gpiochip %d \n " , ret ) ;
return ret ;
}
for ( i = 0 ; i < PIOBU_NUM ; + + i ) {
ret = sama5d2_piobu_setup_pin ( & piobu - > chip , i ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Failed to setup pin: %d %d \n " ,
i , ret ) ;
return ret ;
}
}
return 0 ;
}
static const struct of_device_id sama5d2_piobu_ids [ ] = {
{ . compatible = " atmel,sama5d2-secumod " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , sama5d2_piobu_ids ) ;
static struct platform_driver sama5d2_piobu_driver = {
. driver = {
. name = " sama5d2-piobu " ,
. of_match_table = of_match_ptr ( sama5d2_piobu_ids )
} ,
. probe = sama5d2_piobu_probe ,
} ;
module_platform_driver ( sama5d2_piobu_driver ) ;
MODULE_VERSION ( " 1.0 " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " SAMA5D2 PIOBU controller driver " ) ;
MODULE_AUTHOR ( " Andrei Stefanescu <andrei.stefanescu@microchip.com> " ) ;