2010-10-27 15:33:18 -07:00
/*
* 74 Hx164 - Generic serial - in / parallel - out 8 - bits shift register GPIO driver
*
* Copyright ( C ) 2010 Gabor Juhos < juhosg @ openwrt . org >
* Copyright ( C ) 2010 Miguel Gaio < miguel . gaio @ efixo . com >
*
* 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 .
*/
# include <linux/init.h>
# include <linux/mutex.h>
# include <linux/spi/spi.h>
# include <linux/gpio.h>
2012-09-10 22:35:39 +02:00
# include <linux/of_gpio.h>
2010-10-27 15:33:18 -07:00
# include <linux/slab.h>
2011-07-03 13:38:09 -04:00
# include <linux/module.h>
2010-10-27 15:33:18 -07:00
2012-09-10 22:35:39 +02:00
# define GEN_74X164_NUMBER_GPIOS 8
2010-10-27 15:33:18 -07:00
struct gen_74x164_chip {
struct gpio_chip gpio_chip ;
struct mutex lock ;
2012-09-10 22:35:39 +02:00
u32 registers ;
2015-11-30 15:35:26 +01:00
/*
* Since the registers are chained , every byte sent will make
* the previous byte shift to the next register in the
* chain . Thus , the first byte sent will end up in the last
* register at the end of the transfer . So , to have a logical
* numbering , store the bytes in reverse order .
*/
2015-11-30 15:35:25 +01:00
u8 buffer [ 0 ] ;
2010-10-27 15:33:18 -07:00
} ;
static int __gen_74x164_write_config ( struct gen_74x164_chip * chip )
{
2016-06-17 18:39:28 +02:00
return spi_write ( to_spi_device ( chip - > gpio_chip . parent ) , chip - > buffer ,
chip - > registers ) ;
2010-10-27 15:33:18 -07:00
}
static int gen_74x164_get_value ( struct gpio_chip * gc , unsigned offset )
{
2015-12-03 18:20:29 +01:00
struct gen_74x164_chip * chip = gpiochip_get_data ( gc ) ;
2015-11-30 15:35:26 +01:00
u8 bank = chip - > registers - 1 - offset / 8 ;
2012-09-10 22:35:39 +02:00
u8 pin = offset % 8 ;
2010-10-27 15:33:18 -07:00
int ret ;
mutex_lock ( & chip - > lock ) ;
2012-09-10 22:35:39 +02:00
ret = ( chip - > buffer [ bank ] > > pin ) & 0x1 ;
2010-10-27 15:33:18 -07:00
mutex_unlock ( & chip - > lock ) ;
return ret ;
}
static void gen_74x164_set_value ( struct gpio_chip * gc ,
unsigned offset , int val )
{
2015-12-03 18:20:29 +01:00
struct gen_74x164_chip * chip = gpiochip_get_data ( gc ) ;
2015-11-30 15:35:26 +01:00
u8 bank = chip - > registers - 1 - offset / 8 ;
2012-09-10 22:35:39 +02:00
u8 pin = offset % 8 ;
2010-10-27 15:33:18 -07:00
mutex_lock ( & chip - > lock ) ;
if ( val )
2012-09-10 22:35:39 +02:00
chip - > buffer [ bank ] | = ( 1 < < pin ) ;
2010-10-27 15:33:18 -07:00
else
2012-09-10 22:35:39 +02:00
chip - > buffer [ bank ] & = ~ ( 1 < < pin ) ;
2010-10-27 15:33:18 -07:00
__gen_74x164_write_config ( chip ) ;
mutex_unlock ( & chip - > lock ) ;
}
2016-03-14 16:19:18 +01:00
static void gen_74x164_set_multiple ( struct gpio_chip * gc , unsigned long * mask ,
unsigned long * bits )
{
struct gen_74x164_chip * chip = gpiochip_get_data ( gc ) ;
unsigned int i , idx , shift ;
u8 bank , bankmask ;
mutex_lock ( & chip - > lock ) ;
for ( i = 0 , bank = chip - > registers - 1 ; i < chip - > registers ;
i + + , bank - - ) {
idx = i / sizeof ( * mask ) ;
shift = i % sizeof ( * mask ) * BITS_PER_BYTE ;
bankmask = mask [ idx ] > > shift ;
if ( ! bankmask )
continue ;
chip - > buffer [ bank ] & = ~ bankmask ;
chip - > buffer [ bank ] | = bankmask & ( bits [ idx ] > > shift ) ;
}
__gen_74x164_write_config ( chip ) ;
mutex_unlock ( & chip - > lock ) ;
}
2011-05-27 16:35:59 -07:00
static int gen_74x164_direction_output ( struct gpio_chip * gc ,
unsigned offset , int val )
{
gen_74x164_set_value ( gc , offset , val ) ;
return 0 ;
}
2012-11-19 13:22:34 -05:00
static int gen_74x164_probe ( struct spi_device * spi )
2010-10-27 15:33:18 -07:00
{
struct gen_74x164_chip * chip ;
2015-11-30 15:35:25 +01:00
u32 nregs ;
2010-10-27 15:33:18 -07:00
int ret ;
/*
* bits_per_word cannot be configured in platform data
*/
spi - > bits_per_word = 8 ;
ret = spi_setup ( spi ) ;
if ( ret < 0 )
return ret ;
2015-11-30 15:35:25 +01:00
if ( of_property_read_u32 ( spi - > dev . of_node , " registers-number " ,
& nregs ) ) {
dev_err ( & spi - > dev ,
" Missing registers-number property in the DT. \n " ) ;
return - EINVAL ;
}
chip = devm_kzalloc ( & spi - > dev , sizeof ( * chip ) + nregs , GFP_KERNEL ) ;
2010-10-27 15:33:18 -07:00
if ( ! chip )
return - ENOMEM ;
2013-03-15 18:17:18 +09:00
spi_set_drvdata ( spi , chip ) ;
2010-10-27 15:33:18 -07:00
2011-05-27 16:35:59 -07:00
chip - > gpio_chip . label = spi - > modalias ;
chip - > gpio_chip . direction_output = gen_74x164_direction_output ;
2010-10-27 15:33:18 -07:00
chip - > gpio_chip . get = gen_74x164_get_value ;
chip - > gpio_chip . set = gen_74x164_set_value ;
2016-03-14 16:19:18 +01:00
chip - > gpio_chip . set_multiple = gen_74x164_set_multiple ;
2013-12-07 14:08:22 +04:00
chip - > gpio_chip . base = - 1 ;
2012-09-10 22:35:39 +02:00
2015-11-30 15:35:25 +01:00
chip - > registers = nregs ;
2012-09-10 22:35:39 +02:00
chip - > gpio_chip . ngpio = GEN_74X164_NUMBER_GPIOS * chip - > registers ;
2013-12-04 14:42:46 +01:00
chip - > gpio_chip . can_sleep = true ;
2015-11-04 09:56:26 +01:00
chip - > gpio_chip . parent = & spi - > dev ;
2010-10-27 15:33:18 -07:00
chip - > gpio_chip . owner = THIS_MODULE ;
2013-12-07 14:08:23 +04:00
mutex_init ( & chip - > lock ) ;
2010-10-27 15:33:18 -07:00
ret = __gen_74x164_write_config ( chip ) ;
if ( ret ) {
dev_err ( & spi - > dev , " Failed writing: %d \n " , ret ) ;
goto exit_destroy ;
}
2015-12-03 18:20:29 +01:00
ret = gpiochip_add_data ( & chip - > gpio_chip , chip ) ;
2013-12-07 14:08:23 +04:00
if ( ! ret )
return 0 ;
2010-10-27 15:33:18 -07:00
exit_destroy :
mutex_destroy ( & chip - > lock ) ;
2013-12-07 14:08:23 +04:00
2010-10-27 15:33:18 -07:00
return ret ;
}
2012-11-19 13:25:50 -05:00
static int gen_74x164_remove ( struct spi_device * spi )
2010-10-27 15:33:18 -07:00
{
2013-12-07 14:08:23 +04:00
struct gen_74x164_chip * chip = spi_get_drvdata ( spi ) ;
2010-10-27 15:33:18 -07:00
2014-07-12 22:30:12 +02:00
gpiochip_remove ( & chip - > gpio_chip ) ;
mutex_destroy ( & chip - > lock ) ;
2010-10-27 15:33:18 -07:00
2014-07-12 22:30:12 +02:00
return 0 ;
2010-10-27 15:33:18 -07:00
}
2012-09-07 14:18:13 +02:00
static const struct of_device_id gen_74x164_dt_ids [ ] = {
{ . compatible = " fairchild,74hc595 " } ,
2016-03-14 23:32:10 +00:00
{ . compatible = " nxp,74lvc594 " } ,
2012-09-07 14:18:13 +02:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , gen_74x164_dt_ids ) ;
2010-10-27 15:33:18 -07:00
static struct spi_driver gen_74x164_driver = {
. driver = {
2011-05-27 16:35:59 -07:00
. name = " 74x164 " ,
2013-09-19 17:28:08 +05:30
. of_match_table = gen_74x164_dt_ids ,
2010-10-27 15:33:18 -07:00
} ,
. probe = gen_74x164_probe ,
2012-11-19 13:20:08 -05:00
. remove = gen_74x164_remove ,
2010-10-27 15:33:18 -07:00
} ;
2012-09-05 10:40:50 +02:00
module_spi_driver ( gen_74x164_driver ) ;
2010-10-27 15:33:18 -07:00
MODULE_AUTHOR ( " Gabor Juhos <juhosg@openwrt.org> " ) ;
MODULE_AUTHOR ( " Miguel Gaio <miguel.gaio@efixo.com> " ) ;
MODULE_DESCRIPTION ( " GPIO expander driver for 74X164 8-bits shift register " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;