2014-09-03 09:48:20 +02:00
/*
* Copyright ( C ) 2014 Marvell Technology Group Ltd .
*
* Antoine Tenart < antoine . tenart @ free - electrons . com >
* Sebastian Hesselbarth < sebastian . hesselbarth @ gmail . com >
*
* This file is licensed under the terms of the GNU General Public
* License version 2. This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/delay.h>
# include <linux/io.h>
2015-05-16 00:41:25 +02:00
# include <linux/mfd/syscon.h>
2014-09-03 09:48:20 +02:00
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_address.h>
# include <linux/platform_device.h>
2015-05-16 00:41:25 +02:00
# include <linux/regmap.h>
2014-09-03 09:48:20 +02:00
# include <linux/reset-controller.h>
# include <linux/slab.h>
# include <linux/types.h>
# define BERLIN_MAX_RESETS 32
# define to_berlin_reset_priv(p) \
container_of ( ( p ) , struct berlin_reset_priv , rcdev )
struct berlin_reset_priv {
2015-05-16 00:41:25 +02:00
struct regmap * regmap ;
2014-09-03 09:48:20 +02:00
struct reset_controller_dev rcdev ;
} ;
static int berlin_reset_reset ( struct reset_controller_dev * rcdev ,
unsigned long id )
{
struct berlin_reset_priv * priv = to_berlin_reset_priv ( rcdev ) ;
int offset = id > > 8 ;
int mask = BIT ( id & 0x1f ) ;
2015-05-16 00:53:44 +02:00
regmap_write ( priv - > regmap , offset , mask ) ;
2014-09-03 09:48:20 +02:00
/* let the reset be effective */
udelay ( 10 ) ;
return 0 ;
}
static struct reset_control_ops berlin_reset_ops = {
. reset = berlin_reset_reset ,
} ;
static int berlin_reset_xlate ( struct reset_controller_dev * rcdev ,
const struct of_phandle_args * reset_spec )
{
unsigned offset , bit ;
if ( WARN_ON ( reset_spec - > args_count ! = rcdev - > of_reset_n_cells ) )
return - EINVAL ;
offset = reset_spec - > args [ 0 ] ;
bit = reset_spec - > args [ 1 ] ;
if ( bit > = BERLIN_MAX_RESETS )
return - EINVAL ;
return ( offset < < 8 ) | bit ;
}
2015-05-16 00:41:25 +02:00
static int berlin2_reset_probe ( struct platform_device * pdev )
{
struct device_node * parent_np = of_get_parent ( pdev - > dev . of_node ) ;
struct berlin_reset_priv * priv ;
priv = devm_kzalloc ( & pdev - > dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
priv - > regmap = syscon_node_to_regmap ( parent_np ) ;
of_node_put ( parent_np ) ;
if ( IS_ERR ( priv - > regmap ) )
return PTR_ERR ( priv - > regmap ) ;
priv - > rcdev . owner = THIS_MODULE ;
priv - > rcdev . ops = & berlin_reset_ops ;
priv - > rcdev . of_node = pdev - > dev . of_node ;
priv - > rcdev . of_reset_n_cells = 2 ;
priv - > rcdev . of_xlate = berlin_reset_xlate ;
reset_controller_register ( & priv - > rcdev ) ;
return 0 ;
}
static const struct of_device_id berlin_reset_dt_match [ ] = {
{ . compatible = " marvell,berlin2-reset " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , berlin_reset_dt_match ) ;
static struct platform_driver berlin_reset_driver = {
. probe = berlin2_reset_probe ,
. driver = {
. name = " berlin2-reset " ,
. of_match_table = berlin_reset_dt_match ,
} ,
} ;
module_platform_driver ( berlin_reset_driver ) ;
MODULE_AUTHOR ( " Antoine Tenart <antoine.tenart@free-electrons.com> " ) ;
MODULE_AUTHOR ( " Sebastian Hesselbarth <sebastian.hesselbarth@gmail.com> " ) ;
MODULE_DESCRIPTION ( " Marvell Berlin reset driver " ) ;
MODULE_LICENSE ( " GPL " ) ;