2012-04-27 20:26:16 +08:00
/*
* Core driver for the imx pin controller
*
* Copyright ( C ) 2012 Freescale Semiconductor , Inc .
* Copyright ( C ) 2012 Linaro Ltd .
*
* Author : Dong Aisheng < dong . aisheng @ linaro . org >
*
* 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/err.h>
# include <linux/init.h>
# include <linux/io.h>
2016-02-25 18:23:07 +01:00
# include <linux/mfd/syscon.h>
2012-04-27 20:26:16 +08:00
# include <linux/of.h>
# include <linux/of_device.h>
2015-09-25 16:06:00 -05:00
# include <linux/of_address.h>
2012-04-27 20:26:16 +08:00
# include <linux/pinctrl/machine.h>
# include <linux/pinctrl/pinconf.h>
# include <linux/pinctrl/pinctrl.h>
# include <linux/pinctrl/pinmux.h>
# include <linux/slab.h>
2016-02-25 18:23:07 +01:00
# include <linux/regmap.h>
2012-04-27 20:26:16 +08:00
2014-09-03 13:37:38 +02:00
# include "../core.h"
2012-04-27 20:26:16 +08:00
# include "pinctrl-imx.h"
/* The bits in CONFIG cell defined in binding doc*/
# define IMX_NO_PAD_CTL 0x80000000 /* no pin config need */
# define IMX_PAD_SION 0x40000000 /* set SION */
/**
* @ dev : a pointer back to containing device
* @ base : the offset to the controller in virtual memory
*/
struct imx_pinctrl {
struct device * dev ;
struct pinctrl_dev * pctl ;
void __iomem * base ;
2015-09-25 16:06:00 -05:00
void __iomem * input_sel_base ;
2012-04-27 20:26:16 +08:00
const struct imx_pinctrl_soc_info * info ;
} ;
2016-06-13 17:18:34 +02:00
static inline const struct imx_pin_group * imx_pinctrl_find_group_by_name (
2012-04-27 20:26:16 +08:00
const struct imx_pinctrl_soc_info * info ,
const char * name )
{
const struct imx_pin_group * grp = NULL ;
int i ;
for ( i = 0 ; i < info - > ngroups ; i + + ) {
if ( ! strcmp ( info - > groups [ i ] . name , name ) ) {
grp = & info - > groups [ i ] ;
break ;
}
}
return grp ;
}
static int imx_get_groups_count ( struct pinctrl_dev * pctldev )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
return info - > ngroups ;
}
static const char * imx_get_group_name ( struct pinctrl_dev * pctldev ,
unsigned selector )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
return info - > groups [ selector ] . name ;
}
static int imx_get_group_pins ( struct pinctrl_dev * pctldev , unsigned selector ,
const unsigned * * pins ,
unsigned * npins )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
if ( selector > = info - > ngroups )
return - EINVAL ;
2013-07-28 16:29:22 +02:00
* pins = info - > groups [ selector ] . pin_ids ;
2012-04-27 20:26:16 +08:00
* npins = info - > groups [ selector ] . npins ;
return 0 ;
}
static void imx_pin_dbg_show ( struct pinctrl_dev * pctldev , struct seq_file * s ,
unsigned offset )
{
seq_printf ( s , " %s " , dev_name ( pctldev - > dev ) ) ;
}
static int imx_dt_node_to_map ( struct pinctrl_dev * pctldev ,
struct device_node * np ,
struct pinctrl_map * * map , unsigned * num_maps )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
const struct imx_pin_group * grp ;
struct pinctrl_map * new_map ;
struct device_node * parent ;
int map_num = 1 ;
2012-06-20 18:13:47 +08:00
int i , j ;
2012-04-27 20:26:16 +08:00
/*
* first find the group of this node and check if we need create
* config maps for pins
*/
grp = imx_pinctrl_find_group_by_name ( info , np - > name ) ;
if ( ! grp ) {
dev_err ( info - > dev , " unable to find group for node %s \n " ,
np - > name ) ;
return - EINVAL ;
}
for ( i = 0 ; i < grp - > npins ; i + + ) {
2013-07-28 16:29:22 +02:00
if ( ! ( grp - > pins [ i ] . config & IMX_NO_PAD_CTL ) )
2012-04-27 20:26:16 +08:00
map_num + + ;
}
new_map = kmalloc ( sizeof ( struct pinctrl_map ) * map_num , GFP_KERNEL ) ;
if ( ! new_map )
return - ENOMEM ;
* map = new_map ;
* num_maps = map_num ;
/* create mux map */
parent = of_get_parent ( np ) ;
2012-06-07 22:19:26 +05:30
if ( ! parent ) {
kfree ( new_map ) ;
2012-04-27 20:26:16 +08:00
return - EINVAL ;
2012-06-07 22:19:26 +05:30
}
2012-04-27 20:26:16 +08:00
new_map [ 0 ] . type = PIN_MAP_TYPE_MUX_GROUP ;
new_map [ 0 ] . data . mux . function = parent - > name ;
new_map [ 0 ] . data . mux . group = np - > name ;
of_node_put ( parent ) ;
/* create config map */
new_map + + ;
2012-06-20 18:13:47 +08:00
for ( i = j = 0 ; i < grp - > npins ; i + + ) {
2013-07-28 16:29:22 +02:00
if ( ! ( grp - > pins [ i ] . config & IMX_NO_PAD_CTL ) ) {
2012-06-20 18:13:47 +08:00
new_map [ j ] . type = PIN_MAP_TYPE_CONFIGS_PIN ;
new_map [ j ] . data . configs . group_or_pin =
2013-07-28 16:29:22 +02:00
pin_get_name ( pctldev , grp - > pins [ i ] . pin ) ;
new_map [ j ] . data . configs . configs = & grp - > pins [ i ] . config ;
2012-06-20 18:13:47 +08:00
new_map [ j ] . data . configs . num_configs = 1 ;
j + + ;
2012-04-27 20:26:16 +08:00
}
}
dev_dbg ( pctldev - > dev , " maps: function %s group %s num %d \n " ,
2012-06-08 21:33:12 +08:00
( * map ) - > data . mux . function , ( * map ) - > data . mux . group , map_num ) ;
2012-04-27 20:26:16 +08:00
return 0 ;
}
static void imx_dt_free_map ( struct pinctrl_dev * pctldev ,
struct pinctrl_map * map , unsigned num_maps )
{
2012-06-09 00:52:11 +05:30
kfree ( map ) ;
2012-04-27 20:26:16 +08:00
}
2013-02-16 10:25:07 +01:00
static const struct pinctrl_ops imx_pctrl_ops = {
2012-04-27 20:26:16 +08:00
. get_groups_count = imx_get_groups_count ,
. get_group_name = imx_get_group_name ,
. get_group_pins = imx_get_group_pins ,
. pin_dbg_show = imx_pin_dbg_show ,
. dt_node_to_map = imx_dt_node_to_map ,
. dt_free_map = imx_dt_free_map ,
} ;
2014-09-03 13:02:56 +02:00
static int imx_pmx_set ( struct pinctrl_dev * pctldev , unsigned selector ,
unsigned group )
2012-04-27 20:26:16 +08:00
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
const struct imx_pin_reg * pin_reg ;
unsigned int npins , pin_id ;
int i ;
2013-07-28 16:29:22 +02:00
struct imx_pin_group * grp ;
2012-04-27 20:26:16 +08:00
/*
* Configure the mux mode for each pin in the group for a specific
* function .
*/
2013-07-28 16:29:22 +02:00
grp = & info - > groups [ group ] ;
npins = grp - > npins ;
2012-04-27 20:26:16 +08:00
dev_dbg ( ipctl - > dev , " enable function %s group %s \n " ,
2013-07-28 16:29:22 +02:00
info - > functions [ selector ] . name , grp - > name ) ;
2012-04-27 20:26:16 +08:00
for ( i = 0 ; i < npins ; i + + ) {
2013-07-28 16:29:22 +02:00
struct imx_pin * pin = & grp - > pins [ i ] ;
pin_id = pin - > pin ;
2013-02-20 10:32:52 +08:00
pin_reg = & info - > pin_regs [ pin_id ] ;
2012-04-27 20:26:16 +08:00
2014-09-06 18:25:04 +02:00
if ( pin_reg - > mux_reg = = - 1 ) {
2016-06-01 22:21:53 +03:00
dev_dbg ( ipctl - > dev , " Pin(%s) does not support mux function \n " ,
2012-04-27 20:26:16 +08:00
info - > pins [ pin_id ] . name ) ;
2016-06-01 22:21:53 +03:00
continue ;
2012-04-27 20:26:16 +08:00
}
2013-05-28 17:32:07 +08:00
if ( info - > flags & SHARE_MUX_CONF_REG ) {
u32 reg ;
reg = readl ( ipctl - > base + pin_reg - > mux_reg ) ;
reg & = ~ ( 0x7 < < 20 ) ;
2013-07-28 16:29:22 +02:00
reg | = ( pin - > mux_mode < < 20 ) ;
2013-05-28 17:32:07 +08:00
writel ( reg , ipctl - > base + pin_reg - > mux_reg ) ;
} else {
2013-07-28 16:29:22 +02:00
writel ( pin - > mux_mode , ipctl - > base + pin_reg - > mux_reg ) ;
2013-05-28 17:32:07 +08:00
}
2012-04-27 20:26:16 +08:00
dev_dbg ( ipctl - > dev , " write: offset 0x%x val 0x%x \n " ,
2013-07-28 16:29:22 +02:00
pin_reg - > mux_reg , pin - > mux_mode ) ;
2012-04-27 20:26:16 +08:00
2013-08-04 21:39:23 +08:00
/*
* If the select input value begins with 0xff , it ' s a quirky
* select input and the value should be interpreted as below .
* 31 23 15 7 0
* | 0xff | shift | width | select |
* It ' s used to work around the problem that the select
* input for some pin is not implemented in the select
* input register but in some general purpose register .
* We encode the select input value , width and shift of
* the bit field into input_val cell of pin function ID
* in device tree , and then decode them here for setting
* up the select input bits in general purpose register .
*/
2013-07-28 16:29:22 +02:00
if ( pin - > input_val > > 24 = = 0xff ) {
u32 val = pin - > input_val ;
2013-08-04 21:39:23 +08:00
u8 select = val & 0xff ;
u8 width = ( val > > 8 ) & 0xff ;
u8 shift = ( val > > 16 ) & 0xff ;
u32 mask = ( ( 1 < < width ) - 1 ) < < shift ;
/*
* The input_reg [ i ] here is actually some IOMUXC general
* purpose register , not regular select input register .
*/
2013-10-28 14:01:16 +08:00
val = readl ( ipctl - > base + pin - > input_reg ) ;
2013-08-04 21:39:23 +08:00
val & = ~ mask ;
val | = select < < shift ;
2013-10-28 14:01:16 +08:00
writel ( val , ipctl - > base + pin - > input_reg ) ;
} else if ( pin - > input_reg ) {
2013-08-04 21:39:23 +08:00
/*
* Regular select input register can never be at offset
* 0 , and we only print register value for regular case .
*/
2015-09-25 16:06:00 -05:00
if ( ipctl - > input_sel_base )
writel ( pin - > input_val , ipctl - > input_sel_base +
pin - > input_reg ) ;
else
writel ( pin - > input_val , ipctl - > base +
pin - > input_reg ) ;
2012-04-27 20:26:16 +08:00
dev_dbg ( ipctl - > dev ,
" ==>select_input: offset 0x%x val 0x%x \n " ,
2013-07-28 16:29:22 +02:00
pin - > input_reg , pin - > input_val ) ;
2012-04-27 20:26:16 +08:00
}
}
return 0 ;
}
static int imx_pmx_get_funcs_count ( struct pinctrl_dev * pctldev )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
return info - > nfunctions ;
}
static const char * imx_pmx_get_func_name ( struct pinctrl_dev * pctldev ,
unsigned selector )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
return info - > functions [ selector ] . name ;
}
static int imx_pmx_get_groups ( struct pinctrl_dev * pctldev , unsigned selector ,
const char * const * * groups ,
unsigned * const num_groups )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
* groups = info - > functions [ selector ] . groups ;
* num_groups = info - > functions [ selector ] . num_groups ;
return 0 ;
}
2014-10-16 21:47:57 +02:00
static int imx_pmx_gpio_request_enable ( struct pinctrl_dev * pctldev ,
struct pinctrl_gpio_range * range , unsigned offset )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
const struct imx_pin_reg * pin_reg ;
struct imx_pin_group * grp ;
struct imx_pin * imx_pin ;
unsigned int pin , group ;
u32 reg ;
/* Currently implementation only for shared mux/conf register */
if ( ! ( info - > flags & SHARE_MUX_CONF_REG ) )
2016-09-08 04:48:14 +03:00
return 0 ;
2014-10-16 21:47:57 +02:00
pin_reg = & info - > pin_regs [ offset ] ;
if ( pin_reg - > mux_reg = = - 1 )
return - EINVAL ;
/* Find the pinctrl config with GPIO mux mode for the requested pin */
for ( group = 0 ; group < info - > ngroups ; group + + ) {
grp = & info - > groups [ group ] ;
for ( pin = 0 ; pin < grp - > npins ; pin + + ) {
imx_pin = & grp - > pins [ pin ] ;
if ( imx_pin - > pin = = offset & & ! imx_pin - > mux_mode )
goto mux_pin ;
}
}
return - EINVAL ;
mux_pin :
reg = readl ( ipctl - > base + pin_reg - > mux_reg ) ;
reg & = ~ ( 0x7 < < 20 ) ;
reg | = imx_pin - > config ;
writel ( reg , ipctl - > base + pin_reg - > mux_reg ) ;
return 0 ;
}
2016-01-08 10:50:30 -08:00
static void imx_pmx_gpio_disable_free ( struct pinctrl_dev * pctldev ,
struct pinctrl_gpio_range * range , unsigned offset )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
const struct imx_pin_reg * pin_reg ;
u32 reg ;
/*
* Only Vybrid has the input / output buffer enable flags ( IBE / OBE )
* They are part of the shared mux / conf register .
*/
if ( ! ( info - > flags & SHARE_MUX_CONF_REG ) )
return ;
pin_reg = & info - > pin_regs [ offset ] ;
if ( pin_reg - > mux_reg = = - 1 )
return ;
/* Clear IBE/OBE/PUE to disable the pin (Hi-Z) */
reg = readl ( ipctl - > base + pin_reg - > mux_reg ) ;
reg & = ~ 0x7 ;
writel ( reg , ipctl - > base + pin_reg - > mux_reg ) ;
}
2014-10-16 21:47:57 +02:00
static int imx_pmx_gpio_set_direction ( struct pinctrl_dev * pctldev ,
struct pinctrl_gpio_range * range , unsigned offset , bool input )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
const struct imx_pin_reg * pin_reg ;
u32 reg ;
/*
* Only Vybrid has the input / output buffer enable flags ( IBE / OBE )
* They are part of the shared mux / conf register .
*/
if ( ! ( info - > flags & SHARE_MUX_CONF_REG ) )
2016-09-08 04:48:14 +03:00
return 0 ;
2014-10-16 21:47:57 +02:00
pin_reg = & info - > pin_regs [ offset ] ;
if ( pin_reg - > mux_reg = = - 1 )
return - EINVAL ;
/* IBE always enabled allows us to read the value "on the wire" */
reg = readl ( ipctl - > base + pin_reg - > mux_reg ) ;
if ( input )
reg & = ~ 0x2 ;
else
reg | = 0x2 ;
writel ( reg , ipctl - > base + pin_reg - > mux_reg ) ;
return 0 ;
}
2013-02-16 10:25:07 +01:00
static const struct pinmux_ops imx_pmx_ops = {
2012-04-27 20:26:16 +08:00
. get_functions_count = imx_pmx_get_funcs_count ,
. get_function_name = imx_pmx_get_func_name ,
. get_function_groups = imx_pmx_get_groups ,
2014-09-03 13:02:56 +02:00
. set_mux = imx_pmx_set ,
2014-10-16 21:47:57 +02:00
. gpio_request_enable = imx_pmx_gpio_request_enable ,
2016-01-08 10:50:30 -08:00
. gpio_disable_free = imx_pmx_gpio_disable_free ,
2014-10-16 21:47:57 +02:00
. gpio_set_direction = imx_pmx_gpio_set_direction ,
2012-04-27 20:26:16 +08:00
} ;
static int imx_pinconf_get ( struct pinctrl_dev * pctldev ,
unsigned pin_id , unsigned long * config )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
2013-02-20 10:32:52 +08:00
const struct imx_pin_reg * pin_reg = & info - > pin_regs [ pin_id ] ;
2012-04-27 20:26:16 +08:00
2014-09-06 18:25:04 +02:00
if ( pin_reg - > conf_reg = = - 1 ) {
2012-04-27 20:26:16 +08:00
dev_err ( info - > dev , " Pin(%s) does not support config function \n " ,
info - > pins [ pin_id ] . name ) ;
return - EINVAL ;
}
* config = readl ( ipctl - > base + pin_reg - > conf_reg ) ;
2013-05-28 17:32:07 +08:00
if ( info - > flags & SHARE_MUX_CONF_REG )
* config & = 0xffff ;
2012-04-27 20:26:16 +08:00
return 0 ;
}
static int imx_pinconf_set ( struct pinctrl_dev * pctldev ,
2013-08-27 11:32:12 -07:00
unsigned pin_id , unsigned long * configs ,
unsigned num_configs )
2012-04-27 20:26:16 +08:00
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
2013-02-20 10:32:52 +08:00
const struct imx_pin_reg * pin_reg = & info - > pin_regs [ pin_id ] ;
2013-08-27 11:32:12 -07:00
int i ;
2012-04-27 20:26:16 +08:00
2014-09-06 18:25:04 +02:00
if ( pin_reg - > conf_reg = = - 1 ) {
2012-04-27 20:26:16 +08:00
dev_err ( info - > dev , " Pin(%s) does not support config function \n " ,
info - > pins [ pin_id ] . name ) ;
return - EINVAL ;
}
dev_dbg ( ipctl - > dev , " pinconf set pin %s \n " ,
info - > pins [ pin_id ] . name ) ;
2013-08-27 11:32:12 -07:00
for ( i = 0 ; i < num_configs ; i + + ) {
if ( info - > flags & SHARE_MUX_CONF_REG ) {
u32 reg ;
reg = readl ( ipctl - > base + pin_reg - > conf_reg ) ;
reg & = ~ 0xffff ;
reg | = configs [ i ] ;
writel ( reg , ipctl - > base + pin_reg - > conf_reg ) ;
} else {
writel ( configs [ i ] , ipctl - > base + pin_reg - > conf_reg ) ;
}
dev_dbg ( ipctl - > dev , " write: offset 0x%x val 0x%lx \n " ,
pin_reg - > conf_reg , configs [ i ] ) ;
} /* for each config */
2012-04-27 20:26:16 +08:00
return 0 ;
}
static void imx_pinconf_dbg_show ( struct pinctrl_dev * pctldev ,
struct seq_file * s , unsigned pin_id )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
2013-02-20 10:32:52 +08:00
const struct imx_pin_reg * pin_reg = & info - > pin_regs [ pin_id ] ;
2012-04-27 20:26:16 +08:00
unsigned long config ;
2015-01-27 23:50:25 +01:00
if ( ! pin_reg | | pin_reg - > conf_reg = = - 1 ) {
2012-04-27 20:26:16 +08:00
seq_printf ( s , " N/A " ) ;
return ;
}
config = readl ( ipctl - > base + pin_reg - > conf_reg ) ;
seq_printf ( s , " 0x%lx " , config ) ;
}
static void imx_pinconf_group_dbg_show ( struct pinctrl_dev * pctldev ,
struct seq_file * s , unsigned group )
{
struct imx_pinctrl * ipctl = pinctrl_dev_get_drvdata ( pctldev ) ;
const struct imx_pinctrl_soc_info * info = ipctl - > info ;
struct imx_pin_group * grp ;
unsigned long config ;
const char * name ;
int i , ret ;
if ( group > info - > ngroups )
return ;
seq_printf ( s , " \n " ) ;
grp = & info - > groups [ group ] ;
for ( i = 0 ; i < grp - > npins ; i + + ) {
2013-07-28 16:29:22 +02:00
struct imx_pin * pin = & grp - > pins [ i ] ;
name = pin_get_name ( pctldev , pin - > pin ) ;
ret = imx_pinconf_get ( pctldev , pin - > pin , & config ) ;
2012-04-27 20:26:16 +08:00
if ( ret )
return ;
2016-08-20 01:05:37 +03:00
seq_printf ( s , " %s: 0x%lx \n " , name , config ) ;
2012-04-27 20:26:16 +08:00
}
}
2013-02-16 10:25:07 +01:00
static const struct pinconf_ops imx_pinconf_ops = {
2012-04-27 20:26:16 +08:00
. pin_config_get = imx_pinconf_get ,
. pin_config_set = imx_pinconf_set ,
. pin_config_dbg_show = imx_pinconf_dbg_show ,
. pin_config_group_dbg_show = imx_pinconf_group_dbg_show ,
} ;
2013-02-20 10:32:52 +08:00
/*
* Each pin represented in fsl , pins consists of 5 u32 PIN_FUNC_ID and
* 1 u32 CONFIG , so 24 types in total for each pin .
*/
# define FSL_PIN_SIZE 24
2013-05-28 17:32:07 +08:00
# define SHARE_FSL_PIN_SIZE 20
2012-04-27 20:26:16 +08:00
2012-12-21 13:10:23 -08:00
static int imx_pinctrl_parse_groups ( struct device_node * np ,
struct imx_pin_group * grp ,
struct imx_pinctrl_soc_info * info ,
u32 index )
2012-04-27 20:26:16 +08:00
{
2013-05-28 17:32:07 +08:00
int size , pin_size ;
2012-09-18 14:54:00 +08:00
const __be32 * list ;
2013-02-20 10:32:52 +08:00
int i ;
2012-04-27 20:26:16 +08:00
u32 config ;
dev_dbg ( info - > dev , " group(%d): %s \n " , index , np - > name ) ;
2013-05-28 17:32:07 +08:00
if ( info - > flags & SHARE_MUX_CONF_REG )
pin_size = SHARE_FSL_PIN_SIZE ;
else
pin_size = FSL_PIN_SIZE ;
2012-04-27 20:26:16 +08:00
/* Initialise group */
grp - > name = np - > name ;
/*
* the binding format is fsl , pins = < PIN_FUNC_ID CONFIG . . . > ,
* do sanity check and calculate pins number
*/
list = of_get_property ( np , " fsl,pins " , & size ) ;
2013-08-09 14:20:51 +02:00
if ( ! list ) {
dev_err ( info - > dev , " no fsl,pins property in node %s \n " , np - > full_name ) ;
return - EINVAL ;
}
2012-04-27 20:26:16 +08:00
/* we do not check return since it's safe node passed down */
2013-05-28 17:32:07 +08:00
if ( ! size | | size % pin_size ) {
2013-08-09 14:20:50 +02:00
dev_err ( info - > dev , " Invalid fsl,pins property in node %s \n " , np - > full_name ) ;
2012-04-27 20:26:16 +08:00
return - EINVAL ;
}
2013-05-28 17:32:07 +08:00
grp - > npins = size / pin_size ;
2013-07-28 16:29:22 +02:00
grp - > pins = devm_kzalloc ( info - > dev , grp - > npins * sizeof ( struct imx_pin ) ,
2012-04-27 20:26:16 +08:00
GFP_KERNEL ) ;
2013-07-28 16:29:22 +02:00
grp - > pin_ids = devm_kzalloc ( info - > dev , grp - > npins * sizeof ( unsigned int ) ,
2012-04-27 20:26:16 +08:00
GFP_KERNEL ) ;
2013-07-28 16:29:22 +02:00
if ( ! grp - > pins | | ! grp - > pin_ids )
return - ENOMEM ;
2013-02-20 10:32:52 +08:00
for ( i = 0 ; i < grp - > npins ; i + + ) {
u32 mux_reg = be32_to_cpu ( * list + + ) ;
2013-05-28 17:32:07 +08:00
u32 conf_reg ;
unsigned int pin_id ;
struct imx_pin_reg * pin_reg ;
2013-07-28 16:29:22 +02:00
struct imx_pin * pin = & grp - > pins [ i ] ;
2013-02-20 10:32:52 +08:00
2015-09-25 16:05:59 -05:00
if ( ! ( info - > flags & ZERO_OFFSET_VALID ) & & ! mux_reg )
mux_reg = - 1 ;
2015-03-24 16:26:18 +01:00
if ( info - > flags & SHARE_MUX_CONF_REG ) {
2013-05-28 17:32:07 +08:00
conf_reg = mux_reg ;
2015-03-24 16:26:18 +01:00
} else {
2013-05-28 17:32:07 +08:00
conf_reg = be32_to_cpu ( * list + + ) ;
2015-03-24 16:26:18 +01:00
if ( ! conf_reg )
conf_reg = - 1 ;
}
2013-05-28 17:32:07 +08:00
2015-09-25 16:05:59 -05:00
pin_id = ( mux_reg ! = - 1 ) ? mux_reg / 4 : conf_reg / 4 ;
2013-05-28 17:32:07 +08:00
pin_reg = & info - > pin_regs [ pin_id ] ;
2013-07-28 16:29:22 +02:00
pin - > pin = pin_id ;
grp - > pin_ids [ i ] = pin_id ;
2013-02-20 10:32:52 +08:00
pin_reg - > mux_reg = mux_reg ;
pin_reg - > conf_reg = conf_reg ;
2013-07-28 16:29:22 +02:00
pin - > input_reg = be32_to_cpu ( * list + + ) ;
pin - > mux_mode = be32_to_cpu ( * list + + ) ;
pin - > input_val = be32_to_cpu ( * list + + ) ;
2013-02-20 10:32:52 +08:00
2012-04-27 20:26:16 +08:00
/* SION bit is in mux register */
config = be32_to_cpu ( * list + + ) ;
if ( config & IMX_PAD_SION )
2013-07-28 16:29:22 +02:00
pin - > mux_mode | = IOMUXC_CONFIG_SION ;
pin - > config = config & ~ IMX_PAD_SION ;
2012-04-27 20:26:16 +08:00
2014-04-13 12:09:05 -03:00
dev_dbg ( info - > dev , " %s: 0x%x 0x%08lx " , info - > pins [ pin_id ] . name ,
2013-08-23 10:38:57 +02:00
pin - > mux_mode , pin - > config ) ;
}
2012-06-09 00:52:11 +05:30
2012-04-27 20:26:16 +08:00
return 0 ;
}
2012-12-21 13:10:23 -08:00
static int imx_pinctrl_parse_functions ( struct device_node * np ,
struct imx_pinctrl_soc_info * info ,
u32 index )
2012-04-27 20:26:16 +08:00
{
struct device_node * child ;
struct imx_pmx_func * func ;
struct imx_pin_group * grp ;
u32 i = 0 ;
dev_dbg ( info - > dev , " parse function(%d): %s \n " , index , np - > name ) ;
func = & info - > functions [ index ] ;
/* Initialise function */
func - > name = np - > name ;
func - > num_groups = of_get_child_count ( np ) ;
2014-06-26 13:28:16 +02:00
if ( func - > num_groups = = 0 ) {
2013-08-09 14:20:50 +02:00
dev_err ( info - > dev , " no groups defined in %s \n " , np - > full_name ) ;
2012-04-27 20:26:16 +08:00
return - EINVAL ;
}
func - > groups = devm_kzalloc ( info - > dev ,
func - > num_groups * sizeof ( char * ) , GFP_KERNEL ) ;
for_each_child_of_node ( np , child ) {
func - > groups [ i ] = child - > name ;
2015-09-24 15:53:57 -05:00
grp = & info - > groups [ info - > group_index + + ] ;
2013-08-09 14:20:52 +02:00
imx_pinctrl_parse_groups ( child , grp , info , i + + ) ;
2012-04-27 20:26:16 +08:00
}
return 0 ;
}
2015-04-10 16:22:38 +02:00
/*
* Check if the DT contains pins in the direct child nodes . This indicates the
* newer DT format to store pins . This function returns true if the first found
* fsl , pins property is in a child of np . Otherwise false is returned .
*/
static bool imx_pinctrl_dt_is_flat_functions ( struct device_node * np )
{
struct device_node * function_np ;
struct device_node * pinctrl_np ;
for_each_child_of_node ( np , function_np ) {
if ( of_property_read_bool ( function_np , " fsl,pins " ) )
return true ;
for_each_child_of_node ( function_np , pinctrl_np ) {
if ( of_property_read_bool ( pinctrl_np , " fsl,pins " ) )
return false ;
}
}
return true ;
}
2012-12-21 13:10:23 -08:00
static int imx_pinctrl_probe_dt ( struct platform_device * pdev ,
2012-04-27 20:26:16 +08:00
struct imx_pinctrl_soc_info * info )
{
struct device_node * np = pdev - > dev . of_node ;
struct device_node * child ;
u32 nfuncs = 0 ;
u32 i = 0 ;
2015-04-10 16:22:38 +02:00
bool flat_funcs ;
2012-04-27 20:26:16 +08:00
if ( ! np )
return - ENODEV ;
2015-04-10 16:22:38 +02:00
flat_funcs = imx_pinctrl_dt_is_flat_functions ( np ) ;
if ( flat_funcs ) {
nfuncs = 1 ;
} else {
nfuncs = of_get_child_count ( np ) ;
if ( nfuncs < = 0 ) {
dev_err ( & pdev - > dev , " no functions defined \n " ) ;
return - EINVAL ;
}
2012-04-27 20:26:16 +08:00
}
info - > nfunctions = nfuncs ;
info - > functions = devm_kzalloc ( & pdev - > dev , nfuncs * sizeof ( struct imx_pmx_func ) ,
GFP_KERNEL ) ;
if ( ! info - > functions )
return - ENOMEM ;
2016-10-18 14:09:15 -07:00
info - > group_index = 0 ;
2015-04-10 16:22:38 +02:00
if ( flat_funcs ) {
info - > ngroups = of_get_child_count ( np ) ;
} else {
info - > ngroups = 0 ;
for_each_child_of_node ( np , child )
info - > ngroups + = of_get_child_count ( child ) ;
}
2012-04-27 20:26:16 +08:00
info - > groups = devm_kzalloc ( & pdev - > dev , info - > ngroups * sizeof ( struct imx_pin_group ) ,
GFP_KERNEL ) ;
if ( ! info - > groups )
return - ENOMEM ;
2015-04-10 16:22:38 +02:00
if ( flat_funcs ) {
imx_pinctrl_parse_functions ( np , info , 0 ) ;
} else {
for_each_child_of_node ( np , child )
imx_pinctrl_parse_functions ( child , info , i + + ) ;
}
2012-04-27 20:26:16 +08:00
return 0 ;
}
2012-12-21 13:10:23 -08:00
int imx_pinctrl_probe ( struct platform_device * pdev ,
struct imx_pinctrl_soc_info * info )
2012-04-27 20:26:16 +08:00
{
2016-02-25 18:23:07 +01:00
struct regmap_config config = { . name = " gpr " } ;
2015-09-25 16:06:00 -05:00
struct device_node * dev_np = pdev - > dev . of_node ;
2016-05-18 17:31:59 +08:00
struct pinctrl_desc * imx_pinctrl_desc ;
2015-09-25 16:06:00 -05:00
struct device_node * np ;
2012-04-27 20:26:16 +08:00
struct imx_pinctrl * ipctl ;
struct resource * res ;
2016-02-25 18:23:07 +01:00
struct regmap * gpr ;
2015-02-06 17:30:56 +01:00
int ret , i ;
2012-04-27 20:26:16 +08:00
2013-02-20 10:32:52 +08:00
if ( ! info | | ! info - > pins | | ! info - > npins ) {
2012-04-27 20:26:16 +08:00
dev_err ( & pdev - > dev , " wrong pinctrl info \n " ) ;
return - EINVAL ;
}
info - > dev = & pdev - > dev ;
2016-02-25 18:23:07 +01:00
if ( info - > gpr_compatible ) {
gpr = syscon_regmap_lookup_by_compatible ( info - > gpr_compatible ) ;
if ( ! IS_ERR ( gpr ) )
regmap_attach_dev ( & pdev - > dev , gpr , & config ) ;
}
2012-04-27 20:26:16 +08:00
/* Create state holders etc for this driver */
ipctl = devm_kzalloc ( & pdev - > dev , sizeof ( * ipctl ) , GFP_KERNEL ) ;
if ( ! ipctl )
return - ENOMEM ;
2014-09-06 18:25:04 +02:00
info - > pin_regs = devm_kmalloc ( & pdev - > dev , sizeof ( * info - > pin_regs ) *
2013-02-20 10:32:52 +08:00
info - > npins , GFP_KERNEL ) ;
if ( ! info - > pin_regs )
return - ENOMEM ;
2015-02-06 17:30:56 +01:00
for ( i = 0 ; i < info - > npins ; i + + ) {
info - > pin_regs [ i ] . mux_reg = - 1 ;
info - > pin_regs [ i ] . conf_reg = - 1 ;
}
2013-02-20 10:32:52 +08:00
2012-04-27 20:26:16 +08:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-01-21 11:09:14 +01:00
ipctl - > base = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( ipctl - > base ) )
return PTR_ERR ( ipctl - > base ) ;
2012-04-27 20:26:16 +08:00
2015-09-25 16:06:00 -05:00
if ( of_property_read_bool ( dev_np , " fsl,input-sel " ) ) {
np = of_parse_phandle ( dev_np , " fsl,input-sel " , 0 ) ;
2016-03-09 02:45:36 +02:00
if ( ! np ) {
2015-09-25 16:06:00 -05:00
dev_err ( & pdev - > dev , " iomuxc fsl,input-sel property not found \n " ) ;
return - EINVAL ;
}
2016-03-09 02:45:36 +02:00
ipctl - > input_sel_base = of_iomap ( np , 0 ) ;
2015-09-25 16:06:00 -05:00
of_node_put ( np ) ;
2016-03-09 02:45:36 +02:00
if ( ! ipctl - > input_sel_base ) {
dev_err ( & pdev - > dev ,
" iomuxc input select base address not found \n " ) ;
return - ENOMEM ;
}
2015-09-25 16:06:00 -05:00
}
2016-05-18 17:31:59 +08:00
imx_pinctrl_desc = devm_kzalloc ( & pdev - > dev , sizeof ( * imx_pinctrl_desc ) ,
GFP_KERNEL ) ;
if ( ! imx_pinctrl_desc )
return - ENOMEM ;
imx_pinctrl_desc - > name = dev_name ( & pdev - > dev ) ;
imx_pinctrl_desc - > pins = info - > pins ;
imx_pinctrl_desc - > npins = info - > npins ;
2016-11-24 19:11:51 +01:00
imx_pinctrl_desc - > pctlops = & imx_pctrl_ops ;
imx_pinctrl_desc - > pmxops = & imx_pmx_ops ;
imx_pinctrl_desc - > confops = & imx_pinconf_ops ;
imx_pinctrl_desc - > owner = THIS_MODULE ;
2012-04-27 20:26:16 +08:00
ret = imx_pinctrl_probe_dt ( pdev , info ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " fail to probe dt properties \n " ) ;
return ret ;
}
ipctl - > info = info ;
ipctl - > dev = info - > dev ;
platform_set_drvdata ( pdev , ipctl ) ;
2016-05-18 17:31:59 +08:00
ipctl - > pctl = devm_pinctrl_register ( & pdev - > dev ,
imx_pinctrl_desc , ipctl ) ;
2015-06-09 13:01:16 +09:00
if ( IS_ERR ( ipctl - > pctl ) ) {
2012-04-27 20:26:16 +08:00
dev_err ( & pdev - > dev , " could not register IMX pinctrl driver \n " ) ;
2015-06-09 13:01:16 +09:00
return PTR_ERR ( ipctl - > pctl ) ;
2012-04-27 20:26:16 +08:00
}
dev_info ( & pdev - > dev , " initialized IMX pinctrl driver \n " ) ;
return 0 ;
}