2016-08-31 11:45:46 +02:00
/*
* GPIO Testing Device Driver
*
* Copyright ( C ) 2014 Kamlakant Patel < kamlakant . patel @ broadcom . com >
* Copyright ( C ) 2015 - 2016 Bamvor Jian Zhang < bamvor . zhangjian @ linaro . org >
2017-06-09 13:41:31 +02:00
* Copyright ( C ) 2017 Bartosz Golaszewski < brgl @ bgdev . pl >
2016-08-31 11:45:46 +02:00
*
* 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 .
*
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/gpio/driver.h>
2017-02-06 15:11:08 +01:00
# include <linux/gpio/consumer.h>
2016-08-31 11:45:46 +02:00
# include <linux/platform_device.h>
2017-02-06 13:10:37 +01:00
# include <linux/slab.h>
2017-02-06 15:11:07 +01:00
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/irq_work.h>
2017-02-06 15:11:08 +01:00
# include <linux/debugfs.h>
# include <linux/uaccess.h>
# include "gpiolib.h"
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
# define GPIO_MOCKUP_NAME "gpio-mockup"
# define GPIO_MOCKUP_MAX_GC 10
2017-06-09 13:41:28 +02:00
/*
* We ' re storing two values per chip : the GPIO base and the number
* of GPIO lines .
*/
# define GPIO_MOCKUP_MAX_RANGES (GPIO_MOCKUP_MAX_GC * 2)
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
enum {
2017-05-25 10:33:39 +02:00
GPIO_MOCKUP_DIR_OUT = 0 ,
GPIO_MOCKUP_DIR_IN = 1 ,
2016-08-31 11:45:46 +02:00
} ;
/*
* struct gpio_pin_status - structure describing a GPIO status
* @ dir : Configures direction of gpio as " in " or " out " , 0 = in , 1 = out
* @ value : Configures status of the gpio as 0 ( low ) or 1 ( high )
*/
2017-02-06 13:10:35 +01:00
struct gpio_mockup_line_status {
int dir ;
2016-08-31 11:45:46 +02:00
bool value ;
2017-05-25 10:33:41 +02:00
bool irq_enabled ;
2016-08-31 11:45:46 +02:00
} ;
2017-02-06 15:11:07 +01:00
struct gpio_mockup_irq_context {
struct irq_work work ;
int irq ;
} ;
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip {
2016-08-31 11:45:46 +02:00
struct gpio_chip gc ;
2017-02-06 13:10:35 +01:00
struct gpio_mockup_line_status * lines ;
2017-02-06 15:11:07 +01:00
struct gpio_mockup_irq_context irq_ctx ;
2017-02-06 15:11:08 +01:00
struct dentry * dbg_dir ;
} ;
struct gpio_mockup_dbgfs_private {
struct gpio_mockup_chip * chip ;
struct gpio_desc * desc ;
int offset ;
2016-08-31 11:45:46 +02:00
} ;
2017-06-09 13:41:28 +02:00
static int gpio_mockup_ranges [ GPIO_MOCKUP_MAX_RANGES ] ;
2016-08-31 11:45:46 +02:00
static int gpio_mockup_params_nr ;
module_param_array ( gpio_mockup_ranges , int , & gpio_mockup_params_nr , 0400 ) ;
2017-02-06 13:10:37 +01:00
static bool gpio_mockup_named_lines ;
module_param_named ( gpio_mockup_named_lines ,
gpio_mockup_named_lines , bool , 0400 ) ;
2017-02-06 13:10:35 +01:00
static const char gpio_mockup_name_start = ' A ' ;
2017-02-06 15:11:08 +01:00
static struct dentry * gpio_mockup_dbg_dir ;
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
static int gpio_mockup_get ( struct gpio_chip * gc , unsigned int offset )
2016-08-31 11:45:46 +02:00
{
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
return chip - > lines [ offset ] . value ;
2016-08-31 11:45:46 +02:00
}
2017-02-06 13:10:35 +01:00
static void gpio_mockup_set ( struct gpio_chip * gc , unsigned int offset ,
2016-08-31 11:45:46 +02:00
int value )
{
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
chip - > lines [ offset ] . value = ! ! value ;
2016-08-31 11:45:46 +02:00
}
2017-02-06 13:10:35 +01:00
static int gpio_mockup_dirout ( struct gpio_chip * gc , unsigned int offset ,
2016-08-31 11:45:46 +02:00
int value )
{
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
gpio_mockup_set ( gc , offset , value ) ;
2017-05-25 10:33:39 +02:00
chip - > lines [ offset ] . dir = GPIO_MOCKUP_DIR_OUT ;
2016-08-31 11:45:46 +02:00
return 0 ;
}
2017-02-06 13:10:35 +01:00
static int gpio_mockup_dirin ( struct gpio_chip * gc , unsigned int offset )
2016-08-31 11:45:46 +02:00
{
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
2017-05-25 10:33:39 +02:00
chip - > lines [ offset ] . dir = GPIO_MOCKUP_DIR_IN ;
2016-08-31 11:45:46 +02:00
return 0 ;
}
2017-02-06 13:10:35 +01:00
static int gpio_mockup_get_direction ( struct gpio_chip * gc , unsigned int offset )
2016-08-31 11:45:46 +02:00
{
2017-02-06 13:10:35 +01:00
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
return chip - > lines [ offset ] . dir ;
2016-08-31 11:45:46 +02:00
}
2017-02-06 13:10:37 +01:00
static int gpio_mockup_name_lines ( struct device * dev ,
struct gpio_mockup_chip * chip )
{
struct gpio_chip * gc = & chip - > gc ;
char * * names ;
int i ;
2017-06-09 13:41:32 +02:00
names = devm_kcalloc ( dev , gc - > ngpio , sizeof ( char * ) , GFP_KERNEL ) ;
2017-02-06 13:10:37 +01:00
if ( ! names )
return - ENOMEM ;
for ( i = 0 ; i < gc - > ngpio ; i + + ) {
names [ i ] = devm_kasprintf ( dev , GFP_KERNEL ,
" %s-%d " , gc - > label , i ) ;
if ( ! names [ i ] )
return - ENOMEM ;
}
gc - > names = ( const char * const * ) names ;
return 0 ;
}
2017-02-06 15:11:07 +01:00
static int gpio_mockup_to_irq ( struct gpio_chip * chip , unsigned int offset )
{
return chip - > irq_base + offset ;
}
2017-05-25 10:33:41 +02:00
static void gpio_mockup_irqmask ( struct irq_data * data )
{
struct gpio_chip * gc = irq_data_get_irq_chip_data ( data ) ;
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
chip - > lines [ data - > irq - gc - > irq_base ] . irq_enabled = false ;
}
static void gpio_mockup_irqunmask ( struct irq_data * data )
{
struct gpio_chip * gc = irq_data_get_irq_chip_data ( data ) ;
struct gpio_mockup_chip * chip = gpiochip_get_data ( gc ) ;
chip - > lines [ data - > irq - gc - > irq_base ] . irq_enabled = true ;
}
2017-02-06 15:11:07 +01:00
static struct irq_chip gpio_mockup_irqchip = {
. name = GPIO_MOCKUP_NAME ,
. irq_mask = gpio_mockup_irqmask ,
. irq_unmask = gpio_mockup_irqunmask ,
} ;
static void gpio_mockup_handle_irq ( struct irq_work * work )
{
struct gpio_mockup_irq_context * irq_ctx ;
irq_ctx = container_of ( work , struct gpio_mockup_irq_context , work ) ;
handle_simple_irq ( irq_to_desc ( irq_ctx - > irq ) ) ;
}
static int gpio_mockup_irqchip_setup ( struct device * dev ,
struct gpio_mockup_chip * chip )
{
struct gpio_chip * gc = & chip - > gc ;
int irq_base , i ;
2017-03-04 17:23:29 +01:00
irq_base = devm_irq_alloc_descs ( dev , - 1 , 0 , gc - > ngpio , 0 ) ;
2017-02-06 15:11:07 +01:00
if ( irq_base < 0 )
return irq_base ;
gc - > irq_base = irq_base ;
gc - > irqchip = & gpio_mockup_irqchip ;
for ( i = 0 ; i < gc - > ngpio ; i + + ) {
irq_set_chip ( irq_base + i , gc - > irqchip ) ;
2017-05-25 10:33:41 +02:00
irq_set_chip_data ( irq_base + i , gc ) ;
2017-02-06 15:11:07 +01:00
irq_set_handler ( irq_base + i , & handle_simple_irq ) ;
irq_modify_status ( irq_base + i ,
IRQ_NOREQUEST | IRQ_NOAUTOEN , IRQ_NOPROBE ) ;
}
init_irq_work ( & chip - > irq_ctx . work , gpio_mockup_handle_irq ) ;
return 0 ;
}
2017-02-06 15:11:08 +01:00
static ssize_t gpio_mockup_event_write ( struct file * file ,
const char __user * usr_buf ,
size_t size , loff_t * ppos )
{
struct gpio_mockup_dbgfs_private * priv ;
struct gpio_mockup_chip * chip ;
struct seq_file * sfile ;
struct gpio_desc * desc ;
struct gpio_chip * gc ;
2017-06-09 13:41:25 +02:00
int rv , val ;
2017-02-06 15:11:08 +01:00
2017-06-09 13:41:25 +02:00
rv = kstrtoint_from_user ( usr_buf , size , 0 , & val ) ;
if ( rv )
return rv ;
if ( val ! = 0 & & val ! = 1 )
return - EINVAL ;
2017-06-09 13:41:26 +02:00
sfile = file - > private_data ;
priv = sfile - > private ;
desc = priv - > desc ;
chip = priv - > chip ;
gc = & chip - > gc ;
2017-05-25 10:33:41 +02:00
2017-06-09 13:41:26 +02:00
if ( chip - > lines [ priv - > offset ] . irq_enabled ) {
gpiod_set_value_cansleep ( desc , val ) ;
priv - > chip - > irq_ctx . irq = gc - > irq_base + priv - > offset ;
irq_work_queue ( & priv - > chip - > irq_ctx . work ) ;
}
2017-02-06 15:11:08 +01:00
return size ;
}
static int gpio_mockup_event_open ( struct inode * inode , struct file * file )
{
return single_open ( file , NULL , inode - > i_private ) ;
}
static const struct file_operations gpio_mockup_event_ops = {
. owner = THIS_MODULE ,
. open = gpio_mockup_event_open ,
. write = gpio_mockup_event_write ,
. llseek = no_llseek ,
} ;
static void gpio_mockup_debugfs_setup ( struct device * dev ,
struct gpio_mockup_chip * chip )
{
struct gpio_mockup_dbgfs_private * priv ;
struct dentry * evfile ;
struct gpio_chip * gc ;
char * name ;
int i ;
gc = & chip - > gc ;
chip - > dbg_dir = debugfs_create_dir ( gc - > label , gpio_mockup_dbg_dir ) ;
if ( ! chip - > dbg_dir )
goto err ;
for ( i = 0 ; i < gc - > ngpio ; i + + ) {
name = devm_kasprintf ( dev , GFP_KERNEL , " %d " , i ) ;
if ( ! name )
goto err ;
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
goto err ;
priv - > chip = chip ;
priv - > offset = i ;
priv - > desc = & gc - > gpiodev - > descs [ i ] ;
evfile = debugfs_create_file ( name , 0200 , chip - > dbg_dir , priv ,
& gpio_mockup_event_ops ) ;
if ( ! evfile )
goto err ;
}
return ;
err :
dev_err ( dev , " error creating debugfs directory \n " ) ;
}
2017-02-06 13:10:35 +01:00
static int gpio_mockup_add ( struct device * dev ,
struct gpio_mockup_chip * chip ,
2016-08-31 11:45:46 +02:00
const char * name , int base , int ngpio )
{
2017-02-06 13:10:35 +01:00
struct gpio_chip * gc = & chip - > gc ;
2017-02-06 13:10:37 +01:00
int ret ;
2016-08-31 11:45:46 +02:00
2017-02-06 13:10:35 +01:00
gc - > base = base ;
gc - > ngpio = ngpio ;
gc - > label = name ;
gc - > owner = THIS_MODULE ;
gc - > parent = dev ;
gc - > get = gpio_mockup_get ;
gc - > set = gpio_mockup_set ;
gc - > direction_output = gpio_mockup_dirout ;
gc - > direction_input = gpio_mockup_dirin ;
gc - > get_direction = gpio_mockup_get_direction ;
2017-02-06 15:11:07 +01:00
gc - > to_irq = gpio_mockup_to_irq ;
2017-02-06 13:10:35 +01:00
2017-06-09 13:41:32 +02:00
chip - > lines = devm_kcalloc ( dev , gc - > ngpio ,
sizeof ( * chip - > lines ) , GFP_KERNEL ) ;
2017-02-06 13:10:36 +01:00
if ( ! chip - > lines )
return - ENOMEM ;
2017-02-06 13:10:35 +01:00
2017-02-06 13:10:37 +01:00
if ( gpio_mockup_named_lines ) {
ret = gpio_mockup_name_lines ( dev , chip ) ;
if ( ret )
return ret ;
}
2017-02-06 15:11:07 +01:00
ret = gpio_mockup_irqchip_setup ( dev , chip ) ;
if ( ret )
return ret ;
2017-02-06 15:11:08 +01:00
ret = devm_gpiochip_add_data ( dev , & chip - > gc , chip ) ;
if ( ret )
return ret ;
if ( gpio_mockup_dbg_dir )
gpio_mockup_debugfs_setup ( dev , chip ) ;
return 0 ;
2016-08-31 11:45:46 +02:00
}
2017-02-06 13:10:35 +01:00
static int gpio_mockup_probe ( struct platform_device * pdev )
2016-08-31 11:45:46 +02:00
{
2017-06-09 13:41:28 +02:00
int ret , i , base , ngpio , num_chips ;
2016-12-20 12:28:20 +01:00
struct device * dev = & pdev - > dev ;
2017-06-09 13:41:28 +02:00
struct gpio_mockup_chip * chips ;
2016-12-20 12:28:19 +01:00
char * chip_name ;
2016-08-31 11:45:46 +02:00
2017-06-09 13:41:27 +02:00
if ( gpio_mockup_params_nr < 2 | | ( gpio_mockup_params_nr % 2 ) )
2016-08-31 11:45:46 +02:00
return - EINVAL ;
2017-06-09 13:41:28 +02:00
/* Each chip is described by two values. */
num_chips = gpio_mockup_params_nr / 2 ;
2017-06-09 13:41:32 +02:00
chips = devm_kcalloc ( dev , num_chips , sizeof ( * chips ) , GFP_KERNEL ) ;
2017-02-06 13:10:35 +01:00
if ( ! chips )
2016-08-31 11:45:46 +02:00
return - ENOMEM ;
2017-02-06 13:10:35 +01:00
platform_set_drvdata ( pdev , chips ) ;
2016-08-31 11:45:46 +02:00
2017-06-09 13:41:28 +02:00
for ( i = 0 ; i < num_chips ; i + + ) {
2016-08-31 11:45:46 +02:00
base = gpio_mockup_ranges [ i * 2 ] ;
2017-02-06 13:10:35 +01:00
2016-08-31 11:45:46 +02:00
if ( base = = - 1 )
ngpio = gpio_mockup_ranges [ i * 2 + 1 ] ;
else
ngpio = gpio_mockup_ranges [ i * 2 + 1 ] - base ;
if ( ngpio > = 0 ) {
2016-12-20 12:28:19 +01:00
chip_name = devm_kasprintf ( dev , GFP_KERNEL ,
2017-02-06 13:10:35 +01:00
" %s-%c " , GPIO_MOCKUP_NAME ,
gpio_mockup_name_start + i ) ;
2016-12-20 12:28:19 +01:00
if ( ! chip_name )
return - ENOMEM ;
2017-02-06 13:10:35 +01:00
ret = gpio_mockup_add ( dev , & chips [ i ] ,
2016-08-31 11:45:46 +02:00
chip_name , base , ngpio ) ;
} else {
2017-06-09 13:41:29 +02:00
ret = - EINVAL ;
2016-08-31 11:45:46 +02:00
}
2017-02-06 13:10:35 +01:00
2016-08-31 11:45:46 +02:00
if ( ret ) {
2017-06-09 13:41:30 +02:00
dev_err ( dev ,
" adding gpiochip failed: %d (base: %d, ngpio: %d) \n " ,
ret , base , base < 0 ? ngpio : base + ngpio ) ;
2016-08-31 11:45:46 +02:00
return ret ;
}
}
return 0 ;
}
2017-02-06 13:10:35 +01:00
static struct platform_driver gpio_mockup_driver = {
2016-08-31 11:45:46 +02:00
. driver = {
2017-02-06 13:10:35 +01:00
. name = GPIO_MOCKUP_NAME ,
2016-12-20 12:28:20 +01:00
} ,
2017-02-06 13:10:35 +01:00
. probe = gpio_mockup_probe ,
2016-08-31 11:45:46 +02:00
} ;
static struct platform_device * pdev ;
static int __init mock_device_init ( void )
{
int err ;
2017-02-06 15:11:08 +01:00
gpio_mockup_dbg_dir = debugfs_create_dir ( " gpio-mockup-event " , NULL ) ;
if ( ! gpio_mockup_dbg_dir )
pr_err ( " %s: error creating debugfs directory \n " ,
GPIO_MOCKUP_NAME ) ;
2017-02-06 13:10:35 +01:00
pdev = platform_device_alloc ( GPIO_MOCKUP_NAME , - 1 ) ;
2016-08-31 11:45:46 +02:00
if ( ! pdev )
return - ENOMEM ;
err = platform_device_add ( pdev ) ;
if ( err ) {
platform_device_put ( pdev ) ;
return err ;
}
2017-02-06 13:10:35 +01:00
err = platform_driver_register ( & gpio_mockup_driver ) ;
2016-08-31 11:45:46 +02:00
if ( err ) {
platform_device_unregister ( pdev ) ;
return err ;
}
return 0 ;
}
static void __exit mock_device_exit ( void )
{
2017-02-06 15:11:08 +01:00
debugfs_remove_recursive ( gpio_mockup_dbg_dir ) ;
2017-02-06 13:10:35 +01:00
platform_driver_unregister ( & gpio_mockup_driver ) ;
2016-08-31 11:45:46 +02:00
platform_device_unregister ( pdev ) ;
}
module_init ( mock_device_init ) ;
module_exit ( mock_device_exit ) ;
MODULE_AUTHOR ( " Kamlakant Patel <kamlakant.patel@broadcom.com> " ) ;
MODULE_AUTHOR ( " Bamvor Jian Zhang <bamvor.zhangjian@linaro.org> " ) ;
2017-06-09 13:41:31 +02:00
MODULE_AUTHOR ( " Bartosz Golaszewski <brgl@bgdev.pl> " ) ;
2016-08-31 11:45:46 +02:00
MODULE_DESCRIPTION ( " GPIO Testing driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;