2012-09-05 10:57:13 +08:00
/*
* System Control Driver
*
* 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/io.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/of_platform.h>
# include <linux/platform_device.h>
# include <linux/regmap.h>
2013-02-11 18:48:00 -02:00
# include <linux/mfd/syscon.h>
2012-09-05 10:57:13 +08:00
static struct platform_driver syscon_driver ;
struct syscon {
struct regmap * regmap ;
} ;
2013-03-13 21:34:20 +04:00
static int syscon_match_node ( struct device * dev , void * data )
2012-09-05 10:57:13 +08:00
{
struct device_node * dn = data ;
2013-03-13 21:34:19 +04:00
return ( dev - > of_node = = dn ) ? 1 : 0 ;
2012-09-05 10:57:13 +08:00
}
struct regmap * syscon_node_to_regmap ( struct device_node * np )
{
struct syscon * syscon ;
struct device * dev ;
dev = driver_find_device ( & syscon_driver . driver , NULL , np ,
2013-03-13 21:34:20 +04:00
syscon_match_node ) ;
2012-09-05 10:57:13 +08:00
if ( ! dev )
return ERR_PTR ( - EPROBE_DEFER ) ;
syscon = dev_get_drvdata ( dev ) ;
return syscon - > regmap ;
}
EXPORT_SYMBOL_GPL ( syscon_node_to_regmap ) ;
struct regmap * syscon_regmap_lookup_by_compatible ( const char * s )
{
struct device_node * syscon_np ;
struct regmap * regmap ;
syscon_np = of_find_compatible_node ( NULL , NULL , s ) ;
if ( ! syscon_np )
return ERR_PTR ( - ENODEV ) ;
regmap = syscon_node_to_regmap ( syscon_np ) ;
of_node_put ( syscon_np ) ;
return regmap ;
}
EXPORT_SYMBOL_GPL ( syscon_regmap_lookup_by_compatible ) ;
2013-03-13 21:34:20 +04:00
static int syscon_match_pdevname ( struct device * dev , void * data )
{
struct platform_device * pdev = to_platform_device ( dev ) ;
const struct platform_device_id * id = platform_get_device_id ( pdev ) ;
if ( id )
if ( ! strcmp ( id - > name , ( const char * ) data ) )
return 1 ;
return ! strcmp ( dev_name ( dev ) , ( const char * ) data ) ;
}
struct regmap * syscon_regmap_lookup_by_pdevname ( const char * s )
{
struct device * dev ;
struct syscon * syscon ;
dev = driver_find_device ( & syscon_driver . driver , NULL , ( void * ) s ,
syscon_match_pdevname ) ;
if ( ! dev )
return ERR_PTR ( - EPROBE_DEFER ) ;
syscon = dev_get_drvdata ( dev ) ;
return syscon - > regmap ;
}
EXPORT_SYMBOL_GPL ( syscon_regmap_lookup_by_pdevname ) ;
2012-09-05 10:57:13 +08:00
struct regmap * syscon_regmap_lookup_by_phandle ( struct device_node * np ,
const char * property )
{
struct device_node * syscon_np ;
struct regmap * regmap ;
syscon_np = of_parse_phandle ( np , property , 0 ) ;
if ( ! syscon_np )
return ERR_PTR ( - ENODEV ) ;
regmap = syscon_node_to_regmap ( syscon_np ) ;
of_node_put ( syscon_np ) ;
return regmap ;
}
EXPORT_SYMBOL_GPL ( syscon_regmap_lookup_by_phandle ) ;
static const struct of_device_id of_syscon_match [ ] = {
{ . compatible = " syscon " , } ,
{ } ,
} ;
static struct regmap_config syscon_regmap_config = {
. reg_bits = 32 ,
. val_bits = 32 ,
. reg_stride = 4 ,
} ;
2012-11-19 13:23:04 -05:00
static int syscon_probe ( struct platform_device * pdev )
2012-09-05 10:57:13 +08:00
{
struct device * dev = & pdev - > dev ;
struct syscon * syscon ;
2013-03-13 21:34:20 +04:00
struct resource * res ;
2013-07-14 10:39:42 +04:00
void __iomem * base ;
2012-09-05 10:57:13 +08:00
2013-03-13 21:34:20 +04:00
syscon = devm_kzalloc ( dev , sizeof ( * syscon ) , GFP_KERNEL ) ;
2012-09-05 10:57:13 +08:00
if ( ! syscon )
return - ENOMEM ;
2013-03-13 21:34:20 +04:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - ENOENT ;
2012-09-05 10:57:13 +08:00
2013-07-14 10:39:42 +04:00
base = devm_ioremap ( dev , res - > start , resource_size ( res ) ) ;
if ( ! base )
2013-03-13 21:34:20 +04:00
return - ENOMEM ;
2012-09-05 10:57:13 +08:00
2013-03-13 21:34:20 +04:00
syscon_regmap_config . max_register = res - > end - res - > start - 3 ;
2013-07-14 10:39:42 +04:00
syscon - > regmap = devm_regmap_init_mmio ( dev , base ,
2012-09-05 10:57:13 +08:00
& syscon_regmap_config ) ;
if ( IS_ERR ( syscon - > regmap ) ) {
dev_err ( dev , " regmap init failed \n " ) ;
return PTR_ERR ( syscon - > regmap ) ;
}
platform_set_drvdata ( pdev , syscon ) ;
2013-04-09 21:50:02 +04:00
dev_info ( dev , " regmap %pR registered \n " , res ) ;
2012-09-05 10:57:13 +08:00
return 0 ;
}
2013-03-13 21:34:20 +04:00
static const struct platform_device_id syscon_ids [ ] = {
{ " syscon " , } ,
2013-05-13 21:07:36 +04:00
# ifdef CONFIG_ARCH_CLPS711X
{ " clps711x-syscon " , } ,
# endif
2013-03-13 21:34:20 +04:00
{ }
} ;
2012-09-05 10:57:13 +08:00
static struct platform_driver syscon_driver = {
. driver = {
. name = " syscon " ,
. owner = THIS_MODULE ,
. of_match_table = of_syscon_match ,
} ,
. probe = syscon_probe ,
2013-03-13 21:34:20 +04:00
. id_table = syscon_ids ,
2012-09-05 10:57:13 +08:00
} ;
static int __init syscon_init ( void )
{
return platform_driver_register ( & syscon_driver ) ;
}
postcore_initcall ( syscon_init ) ;
static void __exit syscon_exit ( void )
{
platform_driver_unregister ( & syscon_driver ) ;
}
module_exit ( syscon_exit ) ;
MODULE_AUTHOR ( " Dong Aisheng <dong.aisheng@linaro.org> " ) ;
MODULE_DESCRIPTION ( " System Control driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;