2008-10-08 20:41:43 +09:00
/*
2012-06-20 17:29:04 +09:00
* SuperH Pin Function Controller support .
2008-10-08 20:41:43 +09:00
*
* Copyright ( C ) 2008 Magnus Damm
2012-06-20 17:29:04 +09:00
* Copyright ( C ) 2009 - 2012 Paul Mundt
2008-10-08 20:41:43 +09:00
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*/
2012-12-15 23:50:47 +01:00
# define DRV_NAME "sh-pfc"
2010-10-04 03:54:56 +09:00
2012-12-15 23:50:52 +01:00
# include <linux/bitops.h>
2008-10-08 20:41:43 +09:00
# include <linux/err.h>
2012-12-15 23:50:52 +01:00
# include <linux/errno.h>
2008-10-08 20:41:43 +09:00
# include <linux/io.h>
2011-12-09 12:14:27 +09:00
# include <linux/ioport.h>
2012-12-15 23:50:52 +01:00
# include <linux/kernel.h>
# include <linux/module.h>
2013-06-17 20:50:02 +02:00
# include <linux/of.h>
# include <linux/of_device.h>
2012-07-10 12:08:14 +09:00
# include <linux/pinctrl/machine.h>
2012-12-15 23:50:47 +01:00
# include <linux/platform_device.h>
2012-12-15 23:50:52 +01:00
# include <linux/slab.h>
2011-12-09 12:14:27 +09:00
2012-12-15 23:50:44 +01:00
# include "core.h"
2012-12-15 23:50:55 +01:00
static int sh_pfc_ioremap ( struct sh_pfc * pfc , struct platform_device * pdev )
2011-12-09 12:14:27 +09:00
{
struct resource * res ;
2013-12-11 04:26:21 +01:00
unsigned int k ;
2011-12-09 12:14:27 +09:00
2013-02-16 23:39:07 +01:00
if ( pdev - > num_resources = = 0 )
return - EINVAL ;
2011-12-09 12:14:27 +09:00
2013-12-11 04:26:25 +01:00
pfc - > windows = devm_kzalloc ( pfc - > dev , pdev - > num_resources *
sizeof ( * pfc - > windows ) , GFP_NOWAIT ) ;
if ( ! pfc - > windows )
2012-12-15 23:50:48 +01:00
return - ENOMEM ;
2011-12-09 12:14:27 +09:00
2012-12-15 23:51:18 +01:00
pfc - > num_windows = pdev - > num_resources ;
2012-12-15 23:50:55 +01:00
2012-12-15 23:51:18 +01:00
for ( k = 0 , res = pdev - > resource ; k < pdev - > num_resources ; k + + , res + + ) {
2011-12-09 12:14:27 +09:00
WARN_ON ( resource_type ( res ) ! = IORESOURCE_MEM ) ;
2013-12-11 04:26:25 +01:00
pfc - > windows [ k ] . phys = res - > start ;
pfc - > windows [ k ] . size = resource_size ( res ) ;
pfc - > windows [ k ] . virt =
devm_ioremap_nocache ( pfc - > dev , res - > start ,
resource_size ( res ) ) ;
if ( ! pfc - > windows [ k ] . virt )
2012-12-15 23:50:48 +01:00
return - ENOMEM ;
2011-12-09 12:14:27 +09:00
}
return 0 ;
}
2013-02-17 00:26:33 +01:00
static void __iomem * sh_pfc_phys_to_virt ( struct sh_pfc * pfc ,
unsigned long address )
2011-12-09 12:14:27 +09:00
{
2012-12-15 23:50:53 +01:00
struct sh_pfc_window * window ;
2013-02-16 23:39:07 +01:00
unsigned int i ;
2011-12-09 12:14:27 +09:00
/* scan through physical windows and convert address */
2013-02-16 23:39:07 +01:00
for ( i = 0 ; i < pfc - > num_windows ; i + + ) {
2013-12-11 04:26:25 +01:00
window = pfc - > windows + i ;
2011-12-09 12:14:27 +09:00
if ( address < window - > phys )
continue ;
if ( address > = ( window - > phys + window - > size ) )
continue ;
return window - > virt + ( address - window - > phys ) ;
}
2013-02-16 23:39:07 +01:00
BUG ( ) ;
2013-03-26 01:44:52 +01:00
return NULL ;
2011-12-09 12:14:27 +09:00
}
2008-10-08 20:41:43 +09:00
2013-03-08 17:43:54 +01:00
int sh_pfc_get_pin_index ( struct sh_pfc * pfc , unsigned int pin )
2013-02-14 22:35:09 +01:00
{
2013-02-15 01:33:38 +01:00
unsigned int offset ;
unsigned int i ;
2013-07-15 18:38:30 +02:00
for ( i = 0 , offset = 0 ; i < pfc - > nr_ranges ; + + i ) {
const struct sh_pfc_pin_range * range = & pfc - > ranges [ i ] ;
2013-02-15 01:33:38 +01:00
if ( pin < = range - > end )
2013-07-15 18:38:30 +02:00
return pin > = range - > start
? offset + pin - range - > start : - 1 ;
2013-02-15 01:33:38 +01:00
2013-07-15 18:38:30 +02:00
offset + = range - > end - range - > start + 1 ;
2013-02-15 01:33:38 +01:00
}
2013-03-10 16:38:23 +01:00
return - EINVAL ;
2013-02-14 22:35:09 +01:00
}
2013-07-15 13:03:20 +02:00
static int sh_pfc_enum_in_range ( u16 enum_id , const struct pinmux_range * r )
2008-10-08 20:41:43 +09:00
{
if ( enum_id < r - > begin )
return 0 ;
if ( enum_id > r - > end )
return 0 ;
return 1 ;
}
2013-02-15 02:04:55 +01:00
unsigned long sh_pfc_read_raw_reg ( void __iomem * mapped_reg ,
unsigned long reg_width )
2008-12-25 18:17:26 +09:00
{
switch ( reg_width ) {
case 8 :
2011-12-09 12:14:27 +09:00
return ioread8 ( mapped_reg ) ;
2008-12-25 18:17:26 +09:00
case 16 :
2011-12-09 12:14:27 +09:00
return ioread16 ( mapped_reg ) ;
2008-12-25 18:17:26 +09:00
case 32 :
2011-12-09 12:14:27 +09:00
return ioread32 ( mapped_reg ) ;
2008-12-25 18:17:26 +09:00
}
BUG ( ) ;
return 0 ;
}
2013-02-15 02:04:55 +01:00
void sh_pfc_write_raw_reg ( void __iomem * mapped_reg , unsigned long reg_width ,
unsigned long data )
2008-12-25 18:17:26 +09:00
{
switch ( reg_width ) {
case 8 :
2011-12-09 12:14:27 +09:00
iowrite8 ( data , mapped_reg ) ;
2008-12-25 18:17:26 +09:00
return ;
case 16 :
2011-12-09 12:14:27 +09:00
iowrite16 ( data , mapped_reg ) ;
2008-12-25 18:17:26 +09:00
return ;
case 32 :
2011-12-09 12:14:27 +09:00
iowrite32 ( data , mapped_reg ) ;
2008-12-25 18:17:26 +09:00
return ;
}
BUG ( ) ;
}
2012-12-15 23:50:53 +01:00
static void sh_pfc_config_reg_helper ( struct sh_pfc * pfc ,
2013-02-16 18:47:05 +01:00
const struct pinmux_cfg_reg * crp ,
2012-12-15 23:50:53 +01:00
unsigned long in_pos ,
void __iomem * * mapped_regp ,
unsigned long * maskp ,
unsigned long * posp )
2008-10-08 20:41:43 +09:00
{
2013-12-11 04:26:21 +01:00
unsigned int k ;
2011-12-14 01:01:05 +09:00
2012-12-15 23:50:53 +01:00
* mapped_regp = sh_pfc_phys_to_virt ( pfc , crp - > reg ) ;
2008-10-08 20:41:43 +09:00
2011-12-14 01:01:05 +09:00
if ( crp - > field_width ) {
* maskp = ( 1 < < crp - > field_width ) - 1 ;
* posp = crp - > reg_width - ( ( in_pos + 1 ) * crp - > field_width ) ;
} else {
* maskp = ( 1 < < crp - > var_field_width [ in_pos ] ) - 1 ;
* posp = crp - > reg_width ;
for ( k = 0 ; k < = in_pos ; k + + )
* posp - = crp - > var_field_width [ k ] ;
}
2011-12-14 01:00:55 +09:00
}
2012-12-15 23:50:53 +01:00
static void sh_pfc_write_config_reg ( struct sh_pfc * pfc ,
2013-02-16 18:47:05 +01:00
const struct pinmux_cfg_reg * crp ,
2012-12-15 23:50:53 +01:00
unsigned long field , unsigned long value )
2008-12-25 18:17:18 +09:00
{
2011-12-14 01:00:55 +09:00
void __iomem * mapped_reg ;
2011-12-14 01:01:14 +09:00
unsigned long mask , pos , data ;
2008-12-25 18:17:18 +09:00
2012-12-15 23:50:53 +01:00
sh_pfc_config_reg_helper ( pfc , crp , field , & mapped_reg , & mask , & pos ) ;
2008-10-08 20:41:43 +09:00
2013-03-10 18:00:02 +01:00
dev_dbg ( pfc - > dev , " write_reg addr = %lx, value = %ld, field = %ld, "
" r_width = %ld, f_width = %ld \n " ,
crp - > reg , value , field , crp - > reg_width , crp - > field_width ) ;
2008-12-25 18:17:18 +09:00
mask = ~ ( mask < < pos ) ;
value = value < < pos ;
2008-10-08 20:41:43 +09:00
2012-12-15 23:50:53 +01:00
data = sh_pfc_read_raw_reg ( mapped_reg , crp - > reg_width ) ;
2011-12-14 01:01:14 +09:00
data & = mask ;
data | = value ;
2012-12-15 23:51:20 +01:00
if ( pfc - > info - > unlock_reg )
2012-12-15 23:50:53 +01:00
sh_pfc_write_raw_reg (
2012-12-15 23:51:20 +01:00
sh_pfc_phys_to_virt ( pfc , pfc - > info - > unlock_reg ) , 32 ,
2012-12-15 23:50:53 +01:00
~ data ) ;
2011-12-14 01:01:14 +09:00
2012-12-15 23:50:53 +01:00
sh_pfc_write_raw_reg ( mapped_reg , crp - > reg_width , data ) ;
2008-10-08 20:41:43 +09:00
}
2013-07-15 13:03:20 +02:00
static int sh_pfc_get_config_reg ( struct sh_pfc * pfc , u16 enum_id ,
2013-02-16 18:47:05 +01:00
const struct pinmux_cfg_reg * * crp , int * fieldp ,
2013-03-10 15:29:14 +01:00
int * valuep )
2008-10-08 20:41:43 +09:00
{
2013-02-16 18:47:05 +01:00
const struct pinmux_cfg_reg * config_reg ;
2011-12-14 01:01:05 +09:00
unsigned long r_width , f_width , curr_width , ncomb ;
2013-12-11 04:26:21 +01:00
unsigned int k , m , n , pos , bit_pos ;
2008-10-08 20:41:43 +09:00
k = 0 ;
while ( 1 ) {
2012-12-15 23:51:20 +01:00
config_reg = pfc - > info - > cfg_regs + k ;
2008-10-08 20:41:43 +09:00
r_width = config_reg - > reg_width ;
f_width = config_reg - > field_width ;
if ( ! r_width )
break ;
2011-12-14 01:01:05 +09:00
pos = 0 ;
m = 0 ;
for ( bit_pos = 0 ; bit_pos < r_width ; bit_pos + = curr_width ) {
if ( f_width )
curr_width = f_width ;
else
curr_width = config_reg - > var_field_width [ m ] ;
ncomb = 1 < < curr_width ;
for ( n = 0 ; n < ncomb ; n + + ) {
if ( config_reg - > enum_ids [ pos + n ] = = enum_id ) {
* crp = config_reg ;
* fieldp = m ;
* valuep = n ;
return 0 ;
}
2008-10-08 20:41:43 +09:00
}
2011-12-14 01:01:05 +09:00
pos + = ncomb ;
m + + ;
2008-10-08 20:41:43 +09:00
}
k + + ;
}
2013-03-10 16:38:23 +01:00
return - EINVAL ;
2008-10-08 20:41:43 +09:00
}
2013-07-15 13:03:20 +02:00
static int sh_pfc_mark_to_enum ( struct sh_pfc * pfc , u16 mark , int pos ,
u16 * enum_idp )
2008-10-08 20:41:43 +09:00
{
2013-07-15 13:03:20 +02:00
const u16 * data = pfc - > info - > gpio_data ;
2013-12-11 04:26:21 +01:00
unsigned int k ;
2008-10-08 20:41:43 +09:00
if ( pos ) {
* enum_idp = data [ pos + 1 ] ;
return pos + 1 ;
}
2012-12-15 23:51:20 +01:00
for ( k = 0 ; k < pfc - > info - > gpio_data_size ; k + + ) {
2013-02-14 17:36:56 +01:00
if ( data [ k ] = = mark ) {
2008-10-08 20:41:43 +09:00
* enum_idp = data [ k + 1 ] ;
return k + 1 ;
}
}
2013-03-10 18:00:02 +01:00
dev_err ( pfc - > dev , " cannot locate data/mark enum_id for mark %d \n " ,
mark ) ;
2013-03-10 16:38:23 +01:00
return - EINVAL ;
2008-10-08 20:41:43 +09:00
}
2013-03-10 15:29:14 +01:00
int sh_pfc_config_mux ( struct sh_pfc * pfc , unsigned mark , int pinmux_type )
2008-10-08 20:41:43 +09:00
{
2013-02-16 18:47:05 +01:00
const struct pinmux_cfg_reg * cr = NULL ;
2013-07-15 13:03:20 +02:00
u16 enum_id ;
2013-02-16 18:47:05 +01:00
const struct pinmux_range * range ;
2011-12-14 01:00:46 +09:00
int in_range , pos , field , value ;
2013-03-10 16:38:23 +01:00
int ret ;
2008-10-08 20:41:43 +09:00
switch ( pinmux_type ) {
2013-03-10 17:30:25 +01:00
case PINMUX_TYPE_GPIO :
2008-10-08 20:41:43 +09:00
case PINMUX_TYPE_FUNCTION :
range = NULL ;
break ;
case PINMUX_TYPE_OUTPUT :
2012-12-15 23:51:20 +01:00
range = & pfc - > info - > output ;
2008-10-08 20:41:43 +09:00
break ;
case PINMUX_TYPE_INPUT :
2012-12-15 23:51:20 +01:00
range = & pfc - > info - > input ;
2008-10-08 20:41:43 +09:00
break ;
default :
2013-03-10 16:38:23 +01:00
return - EINVAL ;
2008-10-08 20:41:43 +09:00
}
pos = 0 ;
enum_id = 0 ;
2011-12-14 01:00:46 +09:00
field = 0 ;
value = 0 ;
2013-03-10 17:30:25 +01:00
/* Iterate over all the configuration fields we need to update. */
2008-10-08 20:41:43 +09:00
while ( 1 ) {
2013-02-14 17:36:56 +01:00
pos = sh_pfc_mark_to_enum ( pfc , mark , pos , & enum_id ) ;
2013-03-10 16:38:23 +01:00
if ( pos < 0 )
return pos ;
2008-10-08 20:41:43 +09:00
if ( ! enum_id )
break ;
2013-03-10 17:30:25 +01:00
/* Check if the configuration field selects a function. If it
* doesn ' t , skip the field if it ' s not applicable to the
* requested pinmux type .
*/
2012-12-15 23:51:20 +01:00
in_range = sh_pfc_enum_in_range ( enum_id , & pfc - > info - > function ) ;
2010-01-19 13:52:28 +00:00
if ( ! in_range ) {
2013-03-10 17:30:25 +01:00
if ( pinmux_type = = PINMUX_TYPE_FUNCTION ) {
/* Functions are allowed to modify all
* fields .
*/
in_range = 1 ;
} else if ( pinmux_type ! = PINMUX_TYPE_GPIO ) {
/* Input/output types can only modify fields
* that correspond to their respective ranges .
2010-01-19 13:52:28 +00:00
*/
2012-12-15 23:50:53 +01:00
in_range = sh_pfc_enum_in_range ( enum_id , range ) ;
2010-01-19 13:52:28 +00:00
/*
* special case pass through for fixed
* input - only or output - only pins without
* function enum register association .
*/
if ( in_range & & enum_id = = range - > force )
continue ;
}
2013-03-10 17:30:25 +01:00
/* GPIOs are only allowed to modify function fields. */
2008-10-22 18:29:17 +09:00
}
2008-10-08 20:41:43 +09:00
if ( ! in_range )
continue ;
2013-03-10 16:38:23 +01:00
ret = sh_pfc_get_config_reg ( pfc , enum_id , & cr , & field , & value ) ;
if ( ret < 0 )
return ret ;
2008-10-08 20:41:43 +09:00
2013-03-10 15:29:14 +01:00
sh_pfc_write_config_reg ( pfc , cr , field , value ) ;
2008-10-08 20:41:43 +09:00
}
return 0 ;
}
2013-07-15 18:38:30 +02:00
static int sh_pfc_init_ranges ( struct sh_pfc * pfc )
{
struct sh_pfc_pin_range * range ;
unsigned int nr_ranges ;
unsigned int i ;
if ( pfc - > info - > pins [ 0 ] . pin = = ( u16 ) - 1 ) {
/* Pin number -1 denotes that the SoC doesn't report pin numbers
* in its pin arrays yet . Consider the pin numbers range as
* continuous and allocate a single range .
*/
pfc - > nr_ranges = 1 ;
pfc - > ranges = devm_kzalloc ( pfc - > dev , sizeof ( * pfc - > ranges ) ,
GFP_KERNEL ) ;
if ( pfc - > ranges = = NULL )
return - ENOMEM ;
pfc - > ranges - > start = 0 ;
pfc - > ranges - > end = pfc - > info - > nr_pins - 1 ;
pfc - > nr_gpio_pins = pfc - > info - > nr_pins ;
return 0 ;
}
2013-07-15 21:10:54 +02:00
/* Count, allocate and fill the ranges. The PFC SoC data pins array must
* be sorted by pin numbers , and pins without a GPIO port must come
* last .
*/
2013-07-15 18:38:30 +02:00
for ( i = 1 , nr_ranges = 1 ; i < pfc - > info - > nr_pins ; + + i ) {
if ( pfc - > info - > pins [ i - 1 ] . pin ! = pfc - > info - > pins [ i ] . pin - 1 )
nr_ranges + + ;
}
pfc - > nr_ranges = nr_ranges ;
pfc - > ranges = devm_kzalloc ( pfc - > dev , sizeof ( * pfc - > ranges ) * nr_ranges ,
GFP_KERNEL ) ;
if ( pfc - > ranges = = NULL )
return - ENOMEM ;
range = pfc - > ranges ;
range - > start = pfc - > info - > pins [ 0 ] . pin ;
for ( i = 1 ; i < pfc - > info - > nr_pins ; + + i ) {
2013-07-15 21:10:54 +02:00
if ( pfc - > info - > pins [ i - 1 ] . pin = = pfc - > info - > pins [ i ] . pin - 1 )
continue ;
range - > end = pfc - > info - > pins [ i - 1 ] . pin ;
if ( ! ( pfc - > info - > pins [ i - 1 ] . configs & SH_PFC_PIN_CFG_NO_GPIO ) )
pfc - > nr_gpio_pins = range - > end + 1 ;
range + + ;
range - > start = pfc - > info - > pins [ i ] . pin ;
2013-07-15 18:38:30 +02:00
}
range - > end = pfc - > info - > pins [ i - 1 ] . pin ;
2013-07-15 21:10:54 +02:00
if ( ! ( pfc - > info - > pins [ i - 1 ] . configs & SH_PFC_PIN_CFG_NO_GPIO ) )
pfc - > nr_gpio_pins = range - > end + 1 ;
2013-07-15 18:38:30 +02:00
return 0 ;
}
2013-06-17 20:50:02 +02:00
# ifdef CONFIG_OF
static const struct of_device_id sh_pfc_of_table [ ] = {
# ifdef CONFIG_PINCTRL_PFC_R8A73A4
{
. compatible = " renesas,pfc-r8a73a4 " ,
. data = & r8a73a4_pinmux_info ,
} ,
# endif
# ifdef CONFIG_PINCTRL_PFC_R8A7740
{
. compatible = " renesas,pfc-r8a7740 " ,
. data = & r8a7740_pinmux_info ,
} ,
# endif
# ifdef CONFIG_PINCTRL_PFC_R8A7778
{
. compatible = " renesas,pfc-r8a7778 " ,
. data = & r8a7778_pinmux_info ,
} ,
# endif
# ifdef CONFIG_PINCTRL_PFC_R8A7779
{
. compatible = " renesas,pfc-r8a7779 " ,
. data = & r8a7779_pinmux_info ,
} ,
# endif
# ifdef CONFIG_PINCTRL_PFC_R8A7790
{
. compatible = " renesas,pfc-r8a7790 " ,
. data = & r8a7790_pinmux_info ,
} ,
# endif
2013-10-17 06:46:05 +09:00
# ifdef CONFIG_PINCTRL_PFC_R8A7791
{
. compatible = " renesas,pfc-r8a7791 " ,
. data = & r8a7791_pinmux_info ,
} ,
# endif
2013-06-17 20:50:02 +02:00
# ifdef CONFIG_PINCTRL_PFC_SH7372
{
. compatible = " renesas,pfc-sh7372 " ,
. data = & sh7372_pinmux_info ,
} ,
# endif
# ifdef CONFIG_PINCTRL_PFC_SH73A0
{
. compatible = " renesas,pfc-sh73a0 " ,
. data = & sh73a0_pinmux_info ,
} ,
# endif
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , sh_pfc_of_table ) ;
# endif
2012-12-15 23:50:47 +01:00
static int sh_pfc_probe ( struct platform_device * pdev )
2008-10-08 20:41:43 +09:00
{
2013-06-17 20:50:02 +02:00
const struct platform_device_id * platid = platform_get_device_id ( pdev ) ;
# ifdef CONFIG_OF
struct device_node * np = pdev - > dev . of_node ;
# endif
2013-02-16 18:47:05 +01:00
const struct sh_pfc_soc_info * info ;
2012-12-15 23:50:47 +01:00
struct sh_pfc * pfc ;
2008-12-25 18:17:18 +09:00
int ret ;
2008-10-08 20:41:43 +09:00
2013-06-17 20:50:02 +02:00
# ifdef CONFIG_OF
if ( np )
info = of_match_device ( sh_pfc_of_table , & pdev - > dev ) - > data ;
else
# endif
info = platid ? ( const void * ) platid - > driver_data : NULL ;
2012-12-15 23:51:20 +01:00
if ( info = = NULL )
2012-12-15 23:50:47 +01:00
return - ENODEV ;
2008-10-08 20:41:43 +09:00
2013-02-14 21:23:47 +09:00
pfc = devm_kzalloc ( & pdev - > dev , sizeof ( * pfc ) , GFP_KERNEL ) ;
2012-12-15 23:50:47 +01:00
if ( pfc = = NULL )
return - ENOMEM ;
2012-12-15 23:50:43 +01:00
2012-12-15 23:51:20 +01:00
pfc - > info = info ;
2012-12-15 23:50:47 +01:00
pfc - > dev = & pdev - > dev ;
2012-12-15 23:50:55 +01:00
ret = sh_pfc_ioremap ( pfc , pdev ) ;
2012-12-15 23:50:47 +01:00
if ( unlikely ( ret < 0 ) )
2011-12-09 12:14:27 +09:00
return ret ;
2012-12-15 23:50:47 +01:00
spin_lock_init ( & pfc - > lock ) ;
2008-12-25 18:17:34 +09:00
2013-04-21 20:21:57 +02:00
if ( info - > ops & & info - > ops - > init ) {
ret = info - > ops - > init ( pfc ) ;
if ( ret < 0 )
return ret ;
}
2012-07-10 12:08:14 +09:00
pinctrl_provide_dummies ( ) ;
2011-12-09 12:14:27 +09:00
2013-07-15 18:38:30 +02:00
ret = sh_pfc_init_ranges ( pfc ) ;
if ( ret < 0 )
return ret ;
2012-07-10 12:08:14 +09:00
/*
* Initialize pinctrl bindings first
*/
2012-12-15 23:50:47 +01:00
ret = sh_pfc_register_pinctrl ( pfc ) ;
2012-12-15 23:50:45 +01:00
if ( unlikely ( ret ! = 0 ) )
2013-04-21 20:21:57 +02:00
goto error ;
2012-07-10 12:08:14 +09:00
2012-12-15 23:50:46 +01:00
# ifdef CONFIG_GPIO_SH_PFC
2012-07-10 12:08:14 +09:00
/*
* Then the GPIO chip
*/
2012-12-15 23:50:47 +01:00
ret = sh_pfc_register_gpiochip ( pfc ) ;
2012-12-15 23:50:46 +01:00
if ( unlikely ( ret ! = 0 ) ) {
2012-07-10 12:08:14 +09:00
/*
* If the GPIO chip fails to come up we still leave the
* PFC state as it is , given that there are already
* extant users of it that have succeeded by this point .
*/
2013-03-10 18:00:02 +01:00
dev_notice ( pfc - > dev , " failed to init GPIO chip, ignoring... \n " ) ;
2012-06-20 17:29:04 +09:00
}
2012-12-15 23:50:46 +01:00
# endif
2010-10-04 03:54:56 +09:00
2012-12-15 23:50:47 +01:00
platform_set_drvdata ( pdev , pfc ) ;
2013-03-10 18:00:02 +01:00
dev_info ( pfc - > dev , " %s support registered \n " , info - > name ) ;
2012-07-10 12:08:14 +09:00
2012-06-20 17:29:04 +09:00
return 0 ;
2013-04-21 20:21:57 +02:00
error :
if ( info - > ops & & info - > ops - > exit )
info - > ops - > exit ( pfc ) ;
return ret ;
2010-10-04 03:54:56 +09:00
}
2012-12-15 23:50:46 +01:00
2012-12-15 23:50:47 +01:00
static int sh_pfc_remove ( struct platform_device * pdev )
{
struct sh_pfc * pfc = platform_get_drvdata ( pdev ) ;
# ifdef CONFIG_GPIO_SH_PFC
sh_pfc_unregister_gpiochip ( pfc ) ;
# endif
sh_pfc_unregister_pinctrl ( pfc ) ;
2013-04-21 20:21:57 +02:00
if ( pfc - > info - > ops & & pfc - > info - > ops - > exit )
pfc - > info - > ops - > exit ( pfc ) ;
2012-12-15 23:50:47 +01:00
return 0 ;
}
static const struct platform_device_id sh_pfc_id_table [ ] = {
2013-03-26 22:49:49 +09:00
# ifdef CONFIG_PINCTRL_PFC_R8A73A4
{ " pfc-r8a73a4 " , ( kernel_ulong_t ) & r8a73a4_pinmux_info } ,
# endif
2012-12-15 23:51:21 +01:00
# ifdef CONFIG_PINCTRL_PFC_R8A7740
{ " pfc-r8a7740 " , ( kernel_ulong_t ) & r8a7740_pinmux_info } ,
2012-12-15 23:51:22 +01:00
# endif
2013-04-12 05:37:20 +00:00
# ifdef CONFIG_PINCTRL_PFC_R8A7778
{ " pfc-r8a7778 " , ( kernel_ulong_t ) & r8a7778_pinmux_info } ,
# endif
2012-12-15 23:51:22 +01:00
# ifdef CONFIG_PINCTRL_PFC_R8A7779
{ " pfc-r8a7779 " , ( kernel_ulong_t ) & r8a7779_pinmux_info } ,
2012-12-15 23:51:23 +01:00
# endif
2013-04-08 11:08:53 +09:00
# ifdef CONFIG_PINCTRL_PFC_R8A7790
{ " pfc-r8a7790 " , ( kernel_ulong_t ) & r8a7790_pinmux_info } ,
# endif
2013-10-17 06:46:05 +09:00
# ifdef CONFIG_PINCTRL_PFC_R8A7791
{ " pfc-r8a7791 " , ( kernel_ulong_t ) & r8a7791_pinmux_info } ,
# endif
2012-12-15 23:51:29 +01:00
# ifdef CONFIG_PINCTRL_PFC_SH7203
{ " pfc-sh7203 " , ( kernel_ulong_t ) & sh7203_pinmux_info } ,
# endif
2012-12-15 23:51:30 +01:00
# ifdef CONFIG_PINCTRL_PFC_SH7264
{ " pfc-sh7264 " , ( kernel_ulong_t ) & sh7264_pinmux_info } ,
# endif
2012-12-15 23:51:31 +01:00
# ifdef CONFIG_PINCTRL_PFC_SH7269
{ " pfc-sh7269 " , ( kernel_ulong_t ) & sh7269_pinmux_info } ,
# endif
2012-12-15 23:51:23 +01:00
# ifdef CONFIG_PINCTRL_PFC_SH7372
{ " pfc-sh7372 " , ( kernel_ulong_t ) & sh7372_pinmux_info } ,
2012-12-15 23:51:24 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH73A0
{ " pfc-sh73a0 " , ( kernel_ulong_t ) & sh73a0_pinmux_info } ,
2012-12-15 23:51:32 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7720
{ " pfc-sh7720 " , ( kernel_ulong_t ) & sh7720_pinmux_info } ,
2012-12-15 23:51:33 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7722
{ " pfc-sh7722 " , ( kernel_ulong_t ) & sh7722_pinmux_info } ,
2012-12-15 23:51:34 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7723
{ " pfc-sh7723 " , ( kernel_ulong_t ) & sh7723_pinmux_info } ,
2012-12-15 23:51:35 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7724
{ " pfc-sh7724 " , ( kernel_ulong_t ) & sh7724_pinmux_info } ,
2012-12-15 23:51:36 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7734
{ " pfc-sh7734 " , ( kernel_ulong_t ) & sh7734_pinmux_info } ,
2012-12-15 23:51:37 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7757
{ " pfc-sh7757 " , ( kernel_ulong_t ) & sh7757_pinmux_info } ,
2012-12-15 23:51:38 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7785
{ " pfc-sh7785 " , ( kernel_ulong_t ) & sh7785_pinmux_info } ,
2012-12-15 23:51:39 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SH7786
{ " pfc-sh7786 " , ( kernel_ulong_t ) & sh7786_pinmux_info } ,
2012-12-15 23:51:40 +01:00
# endif
# ifdef CONFIG_PINCTRL_PFC_SHX3
{ " pfc-shx3 " , ( kernel_ulong_t ) & shx3_pinmux_info } ,
2012-12-15 23:51:21 +01:00
# endif
2012-12-15 23:50:47 +01:00
{ " sh-pfc " , 0 } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( platform , sh_pfc_id_table ) ;
static struct platform_driver sh_pfc_driver = {
. probe = sh_pfc_probe ,
. remove = sh_pfc_remove ,
. id_table = sh_pfc_id_table ,
. driver = {
. name = DRV_NAME ,
. owner = THIS_MODULE ,
2013-06-17 20:50:02 +02:00
. of_match_table = of_match_ptr ( sh_pfc_of_table ) ,
2012-12-15 23:50:47 +01:00
} ,
} ;
2012-12-15 23:50:54 +01:00
static int __init sh_pfc_init ( void )
{
return platform_driver_register ( & sh_pfc_driver ) ;
2012-12-15 23:50:47 +01:00
}
2012-12-15 23:50:54 +01:00
postcore_initcall ( sh_pfc_init ) ;
2012-12-15 23:50:47 +01:00
static void __exit sh_pfc_exit ( void )
{
platform_driver_unregister ( & sh_pfc_driver ) ;
}
module_exit ( sh_pfc_exit ) ;
2012-12-15 23:50:46 +01:00
MODULE_AUTHOR ( " Magnus Damm, Paul Mundt, Laurent Pinchart " ) ;
MODULE_DESCRIPTION ( " Pin Control and GPIO driver for SuperH pin function controller " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;