2008-04-11 23:06:45 +10:00
/*
* OF helpers for the GPIO API
*
* Copyright ( c ) 2007 - 2008 MontaVista Software , Inc .
*
* Author : Anton Vorontsov < avorontsov @ ru . mvista . com >
*
* 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 .
*/
2010-07-05 16:11:55 -06:00
# include <linux/device.h>
2014-01-27 12:15:08 +05:30
# include <linux/err.h>
2008-04-11 23:06:45 +10:00
# include <linux/errno.h>
2010-07-05 16:11:55 -06:00
# include <linux/module.h>
2008-04-11 23:06:45 +10:00
# include <linux/io.h>
2013-10-17 10:21:37 -07:00
# include <linux/gpio/consumer.h>
2008-04-11 23:06:45 +10:00
# include <linux/of.h>
2010-07-05 16:11:55 -06:00
# include <linux/of_address.h>
2008-04-11 23:06:45 +10:00
# include <linux/of_gpio.h>
2012-10-27 15:21:36 +05:30
# include <linux/pinctrl/pinctrl.h>
2010-07-05 16:11:55 -06:00
# include <linux/slab.h>
2015-02-02 11:44:44 -06:00
# include <linux/gpio/machine.h>
2008-04-11 23:06:45 +10:00
2014-07-22 16:17:41 +09:00
# include "gpiolib.h"
2013-10-17 10:21:37 -07:00
2012-05-25 21:36:15 +08:00
/* Private data structure for of_gpiochip_find_and_xlate */
2012-05-17 13:54:40 -06:00
struct gg_data {
enum of_gpio_flags * flags ;
struct of_phandle_args gpiospec ;
2013-10-17 10:21:37 -07:00
struct gpio_desc * out_gpio ;
2012-05-17 13:54:40 -06:00
} ;
/* Private function for resolving node pointer to gpio_chip */
static int of_gpiochip_find_and_xlate ( struct gpio_chip * gc , void * data )
{
struct gg_data * gg_data = data ;
int ret ;
if ( ( gc - > of_node ! = gg_data - > gpiospec . np ) | |
( gc - > of_gpio_n_cells ! = gg_data - > gpiospec . args_count ) | |
( ! gc - > of_xlate ) )
return false ;
ret = gc - > of_xlate ( gc , & gg_data - > gpiospec , gg_data - > flags ) ;
2015-01-09 09:40:43 +01:00
if ( ret < 0 ) {
2015-02-10 09:48:27 +01:00
/* We've found a gpio chip, but the translation failed.
* Store translation error in out_gpio .
* Return false to keep looking , as more than one gpio chip
* could be registered per of - node .
2015-01-09 09:40:43 +01:00
*/
gg_data - > out_gpio = ERR_PTR ( ret ) ;
2015-02-10 09:48:27 +01:00
return false ;
2015-01-09 09:40:43 +01:00
}
2012-05-17 13:54:40 -06:00
2014-04-22 14:38:21 +02:00
gg_data - > out_gpio = gpiochip_get_desc ( gc , ret ) ;
2012-05-17 13:54:40 -06:00
return true ;
}
2008-04-11 23:06:45 +10:00
/**
2013-10-17 10:21:37 -07:00
* of_get_named_gpiod_flags ( ) - Get a GPIO descriptor and flags for GPIO API
2008-04-11 23:06:45 +10:00
* @ np : device node to get GPIO from
2011-06-27 16:49:57 -07:00
* @ propname : property name containing gpio specifier ( s )
2008-04-11 23:06:45 +10:00
* @ index : index of the GPIO
2008-12-01 06:30:04 +00:00
* @ flags : a flags pointer to fill in
2008-04-11 23:06:45 +10:00
*
2013-10-17 10:21:37 -07:00
* Returns GPIO descriptor to use with Linux GPIO API , or one of the errno
2008-12-01 06:30:04 +00:00
* value on the error condition . If @ flags is not NULL the function also fills
* in flags for the GPIO .
2008-04-11 23:06:45 +10:00
*/
2013-10-17 10:21:37 -07:00
struct gpio_desc * of_get_named_gpiod_flags ( struct device_node * np ,
const char * propname , int index , enum of_gpio_flags * flags )
2008-04-11 23:06:45 +10:00
{
2012-06-28 00:32:14 +02:00
/* Return -EPROBE_DEFER to support probe() functions to be called
* later when the GPIO actually becomes available
*/
2013-10-17 10:21:37 -07:00
struct gg_data gg_data = {
. flags = flags ,
. out_gpio = ERR_PTR ( - EPROBE_DEFER )
} ;
2008-10-10 04:43:17 +00:00
int ret ;
2012-05-17 13:54:40 -06:00
/* .of_xlate might decide to not fill in the flags, so clear it. */
if ( flags )
* flags = 0 ;
2008-04-11 23:06:45 +10:00
2011-12-12 09:25:57 -07:00
ret = of_parse_phandle_with_args ( np , propname , " #gpio-cells " , index ,
2012-05-17 13:54:40 -06:00
& gg_data . gpiospec ) ;
2008-10-10 04:43:17 +00:00
if ( ret ) {
2014-07-04 15:22:09 +05:30
pr_debug ( " %s: can't parse '%s' property of node '%s[%d]' \n " ,
__func__ , propname , np - > full_name , index ) ;
2013-10-17 10:21:37 -07:00
return ERR_PTR ( ret ) ;
2008-04-11 23:06:45 +10:00
}
2012-05-17 13:54:40 -06:00
gpiochip_find ( & gg_data , of_gpiochip_find_and_xlate ) ;
2008-04-11 23:06:45 +10:00
2012-05-17 13:54:40 -06:00
of_node_put ( gg_data . gpiospec . np ) ;
2014-07-04 15:22:09 +05:30
pr_debug ( " %s: parsed '%s' property of node '%s[%d]' - status (%d) \n " ,
__func__ , propname , np - > full_name , index ,
2014-01-27 12:15:08 +05:30
PTR_ERR_OR_ZERO ( gg_data . out_gpio ) ) ;
2012-05-17 13:54:40 -06:00
return gg_data . out_gpio ;
2008-04-11 23:06:45 +10:00
}
2014-05-17 14:54:50 +09:00
int of_get_named_gpio_flags ( struct device_node * np , const char * list_name ,
int index , enum of_gpio_flags * flags )
{
struct gpio_desc * desc ;
desc = of_get_named_gpiod_flags ( np , list_name , index , flags ) ;
if ( IS_ERR ( desc ) )
return PTR_ERR ( desc ) ;
else
return desc_to_gpio ( desc ) ;
}
EXPORT_SYMBOL ( of_get_named_gpio_flags ) ;
2015-02-02 11:44:44 -06:00
/**
* of_get_gpio_hog ( ) - Get a GPIO hog descriptor , names and flags for GPIO API
* @ np : device node to get GPIO from
* @ name : GPIO line name
* @ lflags : gpio_lookup_flags - returned from of_find_gpio ( ) or
* of_get_gpio_hog ( )
* @ dflags : gpiod_flags - optional GPIO initialization flags
*
* Returns GPIO descriptor to use with Linux GPIO API , or one of the errno
* value on the error condition .
*/
static struct gpio_desc * of_get_gpio_hog ( struct device_node * np ,
const char * * name ,
enum gpio_lookup_flags * lflags ,
enum gpiod_flags * dflags )
{
struct device_node * chip_np ;
enum of_gpio_flags xlate_flags ;
struct gpio_desc * desc ;
struct gg_data gg_data = {
. flags = & xlate_flags ,
} ;
u32 tmp ;
int i , ret ;
chip_np = np - > parent ;
if ( ! chip_np )
return ERR_PTR ( - EINVAL ) ;
xlate_flags = 0 ;
* lflags = 0 ;
* dflags = 0 ;
ret = of_property_read_u32 ( chip_np , " #gpio-cells " , & tmp ) ;
if ( ret )
return ERR_PTR ( ret ) ;
if ( tmp > MAX_PHANDLE_ARGS )
return ERR_PTR ( - EINVAL ) ;
gg_data . gpiospec . args_count = tmp ;
gg_data . gpiospec . np = chip_np ;
for ( i = 0 ; i < tmp ; i + + ) {
ret = of_property_read_u32_index ( np , " gpios " , i ,
& gg_data . gpiospec . args [ i ] ) ;
if ( ret )
return ERR_PTR ( ret ) ;
}
gpiochip_find ( & gg_data , of_gpiochip_find_and_xlate ) ;
if ( ! gg_data . out_gpio ) {
if ( np - > parent = = np )
return ERR_PTR ( - ENXIO ) ;
else
return ERR_PTR ( - EINVAL ) ;
}
if ( xlate_flags & OF_GPIO_ACTIVE_LOW )
* lflags | = GPIO_ACTIVE_LOW ;
if ( of_property_read_bool ( np , " input " ) )
* dflags | = GPIOD_IN ;
else if ( of_property_read_bool ( np , " output-low " ) )
* dflags | = GPIOD_OUT_LOW ;
else if ( of_property_read_bool ( np , " output-high " ) )
* dflags | = GPIOD_OUT_HIGH ;
else {
pr_warn ( " GPIO line %d (%s): no hogging state specified, bailing out \n " ,
desc_to_gpio ( gg_data . out_gpio ) , np - > name ) ;
return ERR_PTR ( - EINVAL ) ;
}
if ( name & & of_property_read_string ( np , " line-name " , name ) )
* name = np - > name ;
desc = gg_data . out_gpio ;
return desc ;
}
/**
* of_gpiochip_scan_hogs - Scan gpio - controller and apply GPIO hog as requested
* @ chip : gpio chip to act on
*
* This is only used by of_gpiochip_add to request / set GPIO initial
* configuration .
*/
static void of_gpiochip_scan_hogs ( struct gpio_chip * chip )
{
struct gpio_desc * desc = NULL ;
struct device_node * np ;
const char * name ;
enum gpio_lookup_flags lflags ;
enum gpiod_flags dflags ;
for_each_child_of_node ( chip - > of_node , np ) {
if ( ! of_property_read_bool ( np , " gpio-hog " ) )
continue ;
desc = of_get_gpio_hog ( np , & name , & lflags , & dflags ) ;
if ( IS_ERR ( desc ) )
continue ;
if ( gpiod_hog ( desc , name , lflags , dflags ) )
continue ;
}
}
2008-04-11 23:06:45 +10:00
/**
2008-12-01 06:30:04 +00:00
* of_gpio_simple_xlate - translate gpio_spec to the GPIO number and flags
2010-06-08 07:48:16 -06:00
* @ gc : pointer to the gpio_chip structure
2008-04-11 23:06:45 +10:00
* @ np : device node of the GPIO chip
* @ gpio_spec : gpio specifier as found in the device tree
2008-12-01 06:30:04 +00:00
* @ flags : a flags pointer to fill in
2008-04-11 23:06:45 +10:00
*
* This is simple translation function , suitable for the most 1 : 1 mapped
* gpio chips . This function performs only one sanity check : whether gpio
* is less than ngpios ( that is specified in the gpio_chip ) .
*/
2011-12-12 09:25:57 -07:00
int of_gpio_simple_xlate ( struct gpio_chip * gc ,
const struct of_phandle_args * gpiospec , u32 * flags )
2008-04-11 23:06:45 +10:00
{
2008-12-01 06:30:04 +00:00
/*
* We ' re discouraging gpio_cells < 2 , since that way you ' ll have to
2015-05-18 11:41:43 -07:00
* write your own xlate function ( that will have to retrieve the GPIO
2008-12-01 06:30:04 +00:00
* number and the flags from a single gpio cell - - this is possible ,
* but not recommended ) .
*/
2010-06-08 07:48:16 -06:00
if ( gc - > of_gpio_n_cells < 2 ) {
2008-12-01 06:30:04 +00:00
WARN_ON ( 1 ) ;
return - EINVAL ;
}
2011-12-12 09:25:57 -07:00
if ( WARN_ON ( gpiospec - > args_count < gc - > of_gpio_n_cells ) )
return - EINVAL ;
2012-04-04 02:02:58 +02:00
if ( gpiospec - > args [ 0 ] > = gc - > ngpio )
2008-04-11 23:06:45 +10:00
return - EINVAL ;
2008-12-01 06:30:04 +00:00
if ( flags )
2011-12-12 09:25:57 -07:00
* flags = gpiospec - > args [ 1 ] ;
2008-12-01 06:30:04 +00:00
2011-12-12 09:25:57 -07:00
return gpiospec - > args [ 0 ] ;
2008-04-11 23:06:45 +10:00
}
2011-07-28 16:25:41 +01:00
EXPORT_SYMBOL ( of_gpio_simple_xlate ) ;
2008-04-11 23:06:45 +10:00
/**
* of_mm_gpiochip_add - Add memory mapped GPIO chip ( bank )
* @ np : device node of the GPIO chip
* @ mm_gc : pointer to the of_mm_gpio_chip allocated structure
*
* To use this function you should allocate and fill mm_gc with :
*
* 1 ) In the gpio_chip structure :
* - all the callbacks
2010-06-08 07:48:16 -06:00
* - of_gpio_n_cells
* - of_xlate callback ( optional )
2008-04-11 23:06:45 +10:00
*
* 3 ) In the of_mm_gpio_chip structure :
* - save_regs callback ( optional )
*
* If succeeded , this function will map bank ' s memory and will
* do all necessary work for you . Then you ' ll able to use . regs
* to manage GPIOs from the callbacks .
*/
int of_mm_gpiochip_add ( struct device_node * np ,
struct of_mm_gpio_chip * mm_gc )
{
int ret = - ENOMEM ;
2010-06-08 07:48:16 -06:00
struct gpio_chip * gc = & mm_gc - > gc ;
2008-04-11 23:06:45 +10:00
gc - > label = kstrdup ( np - > full_name , GFP_KERNEL ) ;
if ( ! gc - > label )
goto err0 ;
mm_gc - > regs = of_iomap ( np , 0 ) ;
if ( ! mm_gc - > regs )
goto err1 ;
2008-04-30 00:05:24 +10:00
gc - > base = - 1 ;
2008-04-11 23:06:45 +10:00
if ( mm_gc - > save_regs )
mm_gc - > save_regs ( mm_gc ) ;
2010-06-08 07:48:16 -06:00
mm_gc - > gc . of_node = np ;
2008-04-11 23:06:45 +10:00
ret = gpiochip_add ( gc ) ;
if ( ret )
goto err2 ;
return 0 ;
err2 :
iounmap ( mm_gc - > regs ) ;
err1 :
kfree ( gc - > label ) ;
err0 :
pr_err ( " %s: GPIO chip registration failed with status %d \n " ,
np - > full_name , ret ) ;
return ret ;
}
EXPORT_SYMBOL ( of_mm_gpiochip_add ) ;
2010-06-08 07:48:16 -06:00
2014-12-17 16:51:13 +01:00
/**
* of_mm_gpiochip_remove - Remove memory mapped GPIO chip ( bank )
* @ mm_gc : pointer to the of_mm_gpio_chip allocated structure
*/
void of_mm_gpiochip_remove ( struct of_mm_gpio_chip * mm_gc )
{
struct gpio_chip * gc = & mm_gc - > gc ;
if ( ! mm_gc )
return ;
gpiochip_remove ( gc ) ;
iounmap ( mm_gc - > regs ) ;
kfree ( gc - > label ) ;
}
EXPORT_SYMBOL ( of_mm_gpiochip_remove ) ;
2012-10-27 15:21:36 +05:30
# ifdef CONFIG_PINCTRL
2012-11-06 14:58:55 +01:00
static void of_gpiochip_add_pin_range ( struct gpio_chip * chip )
2012-10-27 15:21:36 +05:30
{
struct device_node * np = chip - > of_node ;
struct of_phandle_args pinspec ;
2012-11-06 16:03:35 +01:00
struct pinctrl_dev * pctldev ;
2012-10-27 15:21:36 +05:30
int index = 0 , ret ;
2013-10-15 15:37:54 +02:00
const char * name ;
static const char group_names_propname [ ] = " gpio-ranges-group-names " ;
struct property * group_names ;
2012-10-27 15:21:36 +05:30
if ( ! np )
return ;
2013-10-15 15:37:54 +02:00
group_names = of_find_property ( np , group_names_propname , NULL ) ;
2013-02-17 19:42:48 +08:00
for ( ; ; index + + ) {
2013-08-14 15:27:12 -06:00
ret = of_parse_phandle_with_fixed_args ( np , " gpio-ranges " , 3 ,
index , & pinspec ) ;
2012-10-27 15:21:36 +05:30
if ( ret )
break ;
2012-11-06 16:03:35 +01:00
pctldev = of_pinctrl_get ( pinspec . np ) ;
if ( ! pctldev )
2012-10-27 15:21:36 +05:30
break ;
2013-10-15 15:37:54 +02:00
if ( pinspec . args [ 2 ] ) {
if ( group_names ) {
ret = of_property_read_string_index ( np ,
group_names_propname ,
index , & name ) ;
if ( strlen ( name ) ) {
pr_err ( " %s: Group name of numeric GPIO ranges must be the empty string. \n " ,
np - > full_name ) ;
break ;
}
}
/* npins != 0: linear range */
ret = gpiochip_add_pin_range ( chip ,
pinctrl_dev_get_devname ( pctldev ) ,
pinspec . args [ 0 ] ,
pinspec . args [ 1 ] ,
pinspec . args [ 2 ] ) ;
if ( ret )
break ;
} else {
/* npins == 0: special range */
if ( pinspec . args [ 1 ] ) {
pr_err ( " %s: Illegal gpio-range format. \n " ,
np - > full_name ) ;
break ;
}
if ( ! group_names ) {
pr_err ( " %s: GPIO group range requested but no %s property. \n " ,
np - > full_name , group_names_propname ) ;
break ;
}
ret = of_property_read_string_index ( np ,
group_names_propname ,
index , & name ) ;
if ( ret )
break ;
if ( ! strlen ( name ) ) {
pr_err ( " %s: Group name of GPIO group range cannot be the empty string. \n " ,
np - > full_name ) ;
break ;
}
ret = gpiochip_add_pingroup_range ( chip , pctldev ,
pinspec . args [ 0 ] , name ) ;
if ( ret )
break ;
}
2013-02-17 19:42:48 +08:00
}
2012-10-27 15:21:36 +05:30
}
# else
2012-11-06 14:58:55 +01:00
static void of_gpiochip_add_pin_range ( struct gpio_chip * chip ) { }
2012-10-27 15:21:36 +05:30
# endif
2010-06-08 07:48:17 -06:00
void of_gpiochip_add ( struct gpio_chip * chip )
{
if ( ( ! chip - > of_node ) & & ( chip - > dev ) )
chip - > of_node = chip - > dev - > of_node ;
if ( ! chip - > of_node )
return ;
if ( ! chip - > of_xlate ) {
chip - > of_gpio_n_cells = 2 ;
chip - > of_xlate = of_gpio_simple_xlate ;
}
2012-10-27 15:21:36 +05:30
of_gpiochip_add_pin_range ( chip ) ;
2010-06-08 07:48:17 -06:00
of_node_get ( chip - > of_node ) ;
2015-02-02 11:44:44 -06:00
of_gpiochip_scan_hogs ( chip ) ;
2010-06-08 07:48:17 -06:00
}
void of_gpiochip_remove ( struct gpio_chip * chip )
{
2012-11-06 15:03:47 +01:00
gpiochip_remove_pin_ranges ( chip ) ;
2014-08-08 12:07:51 +02:00
of_node_put ( chip - > of_node ) ;
2010-06-08 07:48:17 -06:00
}