2012-11-16 09:15:25 +04:00
/*
* SPEAr platform SPI chipselect abstraction over gpiolib
*
* Copyright ( C ) 2012 ST Microelectronics
2014-04-19 02:07:16 +04:00
* Shiraz Hashim < shiraz . linux . kernel @ gmail . com >
2012-11-16 09:15:25 +04:00
*
* This file is licensed under the terms of the GNU General Public
* License version 2. This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/err.h>
2018-06-27 12:07:02 +03:00
# include <linux/gpio/driver.h>
2012-11-16 09:15:25 +04:00
# include <linux/io.h>
2016-08-22 19:48:32 +03:00
# include <linux/init.h>
2012-11-16 09:15:25 +04:00
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/types.h>
/* maximum chipselects */
# define NUM_OF_GPIO 4
/*
* Provision is available on some SPEAr SoCs to control ARM PL022 spi cs
* through system registers . This register lies outside spi ( pl022 )
* address space into system registers .
*
* It provides control for spi chip select lines so that any chipselect
* ( out of 4 possible chipselects in pl022 ) can be made low to select
* the particular slave .
*/
/**
* struct spear_spics - represents spi chip select control
* @ base : base address
* @ perip_cfg : configuration register
* @ sw_enable_bit : bit to enable s / w control over chipselects
* @ cs_value_bit : bit to program high or low chipselect
* @ cs_enable_mask : mask to select bits required to select chipselect
* @ cs_enable_shift : bit pos of cs_enable_mask
* @ use_count : use count of a spi controller cs lines
* @ last_off : stores last offset caller of set_value ( )
* @ chip : gpio_chip abstraction
*/
struct spear_spics {
void __iomem * base ;
u32 perip_cfg ;
u32 sw_enable_bit ;
u32 cs_value_bit ;
u32 cs_enable_mask ;
u32 cs_enable_shift ;
unsigned long use_count ;
int last_off ;
struct gpio_chip chip ;
} ;
/* gpio framework specific routines */
static int spics_get_value ( struct gpio_chip * chip , unsigned offset )
{
return - ENXIO ;
}
static void spics_set_value ( struct gpio_chip * chip , unsigned offset , int value )
{
2015-12-07 16:26:58 +03:00
struct spear_spics * spics = gpiochip_get_data ( chip ) ;
2012-11-16 09:15:25 +04:00
u32 tmp ;
/* select chip select from register */
tmp = readl_relaxed ( spics - > base + spics - > perip_cfg ) ;
if ( spics - > last_off ! = offset ) {
spics - > last_off = offset ;
tmp & = ~ ( spics - > cs_enable_mask < < spics - > cs_enable_shift ) ;
tmp | = offset < < spics - > cs_enable_shift ;
}
/* toggle chip select line */
tmp & = ~ ( 0x1 < < spics - > cs_value_bit ) ;
tmp | = value < < spics - > cs_value_bit ;
writel_relaxed ( tmp , spics - > base + spics - > perip_cfg ) ;
}
static int spics_direction_input ( struct gpio_chip * chip , unsigned offset )
{
return - ENXIO ;
}
static int spics_direction_output ( struct gpio_chip * chip , unsigned offset ,
int value )
{
spics_set_value ( chip , offset , value ) ;
return 0 ;
}
static int spics_request ( struct gpio_chip * chip , unsigned offset )
{
2015-12-07 16:26:58 +03:00
struct spear_spics * spics = gpiochip_get_data ( chip ) ;
2012-11-16 09:15:25 +04:00
u32 tmp ;
if ( ! spics - > use_count + + ) {
tmp = readl_relaxed ( spics - > base + spics - > perip_cfg ) ;
tmp | = 0x1 < < spics - > sw_enable_bit ;
tmp | = 0x1 < < spics - > cs_value_bit ;
writel_relaxed ( tmp , spics - > base + spics - > perip_cfg ) ;
}
return 0 ;
}
static void spics_free ( struct gpio_chip * chip , unsigned offset )
{
2015-12-07 16:26:58 +03:00
struct spear_spics * spics = gpiochip_get_data ( chip ) ;
2012-11-16 09:15:25 +04:00
u32 tmp ;
if ( ! - - spics - > use_count ) {
tmp = readl_relaxed ( spics - > base + spics - > perip_cfg ) ;
tmp & = ~ ( 0x1 < < spics - > sw_enable_bit ) ;
writel_relaxed ( tmp , spics - > base + spics - > perip_cfg ) ;
}
}
static int spics_gpio_probe ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
struct spear_spics * spics ;
spics = devm_kzalloc ( & pdev - > dev , sizeof ( * spics ) , GFP_KERNEL ) ;
2014-04-29 12:42:56 +04:00
if ( ! spics )
2012-11-16 09:15:25 +04:00
return - ENOMEM ;
2019-03-11 21:55:07 +03:00
spics - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2013-01-21 14:09:01 +04:00
if ( IS_ERR ( spics - > base ) )
return PTR_ERR ( spics - > base ) ;
2012-11-16 09:15:25 +04:00
if ( of_property_read_u32 ( np , " st-spics,peripcfg-reg " ,
& spics - > perip_cfg ) )
goto err_dt_data ;
if ( of_property_read_u32 ( np , " st-spics,sw-enable-bit " ,
& spics - > sw_enable_bit ) )
goto err_dt_data ;
if ( of_property_read_u32 ( np , " st-spics,cs-value-bit " ,
& spics - > cs_value_bit ) )
goto err_dt_data ;
if ( of_property_read_u32 ( np , " st-spics,cs-enable-mask " ,
& spics - > cs_enable_mask ) )
goto err_dt_data ;
if ( of_property_read_u32 ( np , " st-spics,cs-enable-shift " ,
& spics - > cs_enable_shift ) )
goto err_dt_data ;
spics - > chip . ngpio = NUM_OF_GPIO ;
spics - > chip . base = - 1 ;
spics - > chip . request = spics_request ;
spics - > chip . free = spics_free ;
spics - > chip . direction_input = spics_direction_input ;
spics - > chip . direction_output = spics_direction_output ;
spics - > chip . get = spics_get_value ;
spics - > chip . set = spics_set_value ;
spics - > chip . label = dev_name ( & pdev - > dev ) ;
2015-11-04 11:56:26 +03:00
spics - > chip . parent = & pdev - > dev ;
2012-11-16 09:15:25 +04:00
spics - > chip . owner = THIS_MODULE ;
spics - > last_off = - 1 ;
2021-05-16 10:43:11 +03:00
return devm_gpiochip_add_data ( & pdev - > dev , & spics - > chip , spics ) ;
2012-11-16 09:15:25 +04:00
err_dt_data :
dev_err ( & pdev - > dev , " DT probe failed \n " ) ;
return - EINVAL ;
}
static const struct of_device_id spics_gpio_of_match [ ] = {
{ . compatible = " st,spear-spics-gpio " } ,
{ }
} ;
static struct platform_driver spics_gpio_driver = {
. probe = spics_gpio_probe ,
. driver = {
. name = " spear-spics-gpio " ,
. of_match_table = spics_gpio_of_match ,
} ,
} ;
static int __init spics_gpio_init ( void )
{
return platform_driver_register ( & spics_gpio_driver ) ;
}
subsys_initcall ( spics_gpio_init ) ;