2010-12-14 14:00:54 +08:00
/*
* Regulator Driver for Freescale MC13xxx PMIC
*
* Copyright 2010 Yong Shen < yong . shen @ linaro . org >
*
* Based on mc13783 regulator driver :
* Copyright ( C ) 2008 Sascha Hauer , Pengutronix < s . hauer @ pengutronix . de >
* Copyright 2009 Alberto Panizzo < maramaopercheseimorto @ gmail . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* Regs infos taken from mc13xxx drivers from freescale and mc13xxx . pdf file
* from freescale
*/
# include <linux/mfd/mc13xxx.h>
# include <linux/regulator/machine.h>
# include <linux/regulator/driver.h>
2011-12-21 23:00:46 +08:00
# include <linux/regulator/of_regulator.h>
2010-12-14 14:00:54 +08:00
# include <linux/platform_device.h>
# include <linux/kernel.h>
# include <linux/slab.h>
# include <linux/init.h>
# include <linux/err.h>
2011-07-17 16:28:23 -04:00
# include <linux/module.h>
2011-12-21 23:00:46 +08:00
# include <linux/of.h>
2010-12-14 14:00:54 +08:00
# include "mc13xxx.h"
static int mc13xxx_regulator_enable ( struct regulator_dev * rdev )
{
struct mc13xxx_regulator_priv * priv = rdev_get_drvdata ( rdev ) ;
struct mc13xxx_regulator * mc13xxx_regulators = priv - > mc13xxx_regulators ;
int id = rdev_get_id ( rdev ) ;
dev_dbg ( rdev_get_dev ( rdev ) , " %s id: %d \n " , __func__ , id ) ;
2014-06-08 15:26:29 +04:00
return mc13xxx_reg_rmw ( priv - > mc13xxx , mc13xxx_regulators [ id ] . reg ,
mc13xxx_regulators [ id ] . enable_bit ,
mc13xxx_regulators [ id ] . enable_bit ) ;
2010-12-14 14:00:54 +08:00
}
static int mc13xxx_regulator_disable ( struct regulator_dev * rdev )
{
struct mc13xxx_regulator_priv * priv = rdev_get_drvdata ( rdev ) ;
struct mc13xxx_regulator * mc13xxx_regulators = priv - > mc13xxx_regulators ;
int id = rdev_get_id ( rdev ) ;
dev_dbg ( rdev_get_dev ( rdev ) , " %s id: %d \n " , __func__ , id ) ;
2014-06-08 15:26:29 +04:00
return mc13xxx_reg_rmw ( priv - > mc13xxx , mc13xxx_regulators [ id ] . reg ,
mc13xxx_regulators [ id ] . enable_bit , 0 ) ;
2010-12-14 14:00:54 +08:00
}
static int mc13xxx_regulator_is_enabled ( struct regulator_dev * rdev )
{
struct mc13xxx_regulator_priv * priv = rdev_get_drvdata ( rdev ) ;
struct mc13xxx_regulator * mc13xxx_regulators = priv - > mc13xxx_regulators ;
int ret , id = rdev_get_id ( rdev ) ;
unsigned int val ;
ret = mc13xxx_reg_read ( priv - > mc13xxx , mc13xxx_regulators [ id ] . reg , & val ) ;
if ( ret )
return ret ;
return ( val & mc13xxx_regulators [ id ] . enable_bit ) ! = 0 ;
}
2012-03-20 10:46:35 +08:00
static int mc13xxx_regulator_set_voltage_sel ( struct regulator_dev * rdev ,
unsigned selector )
2010-12-14 14:00:54 +08:00
{
struct mc13xxx_regulator_priv * priv = rdev_get_drvdata ( rdev ) ;
struct mc13xxx_regulator * mc13xxx_regulators = priv - > mc13xxx_regulators ;
2012-03-20 10:46:35 +08:00
int id = rdev_get_id ( rdev ) ;
2010-12-14 14:00:54 +08:00
2014-06-08 15:26:29 +04:00
return mc13xxx_reg_rmw ( priv - > mc13xxx , mc13xxx_regulators [ id ] . vsel_reg ,
mc13xxx_regulators [ id ] . vsel_mask ,
selector < < mc13xxx_regulators [ id ] . vsel_shift ) ;
2010-12-14 14:00:54 +08:00
}
static int mc13xxx_regulator_get_voltage ( struct regulator_dev * rdev )
{
struct mc13xxx_regulator_priv * priv = rdev_get_drvdata ( rdev ) ;
struct mc13xxx_regulator * mc13xxx_regulators = priv - > mc13xxx_regulators ;
int ret , id = rdev_get_id ( rdev ) ;
unsigned int val ;
dev_dbg ( rdev_get_dev ( rdev ) , " %s id: %d \n " , __func__ , id ) ;
ret = mc13xxx_reg_read ( priv - > mc13xxx ,
mc13xxx_regulators [ id ] . vsel_reg , & val ) ;
if ( ret )
return ret ;
val = ( val & mc13xxx_regulators [ id ] . vsel_mask )
> > mc13xxx_regulators [ id ] . vsel_shift ;
dev_dbg ( rdev_get_dev ( rdev ) , " %s id: %d val: %d \n " , __func__ , id , val ) ;
2011-05-18 20:56:45 +08:00
BUG_ON ( val > = mc13xxx_regulators [ id ] . desc . n_voltages ) ;
2010-12-14 14:00:54 +08:00
2012-06-08 15:41:48 +08:00
return rdev - > desc - > volt_table [ val ] ;
2010-12-14 14:00:54 +08:00
}
struct regulator_ops mc13xxx_regulator_ops = {
. enable = mc13xxx_regulator_enable ,
. disable = mc13xxx_regulator_disable ,
. is_enabled = mc13xxx_regulator_is_enabled ,
2012-06-08 15:41:48 +08:00
. list_voltage = regulator_list_voltage_table ,
2012-03-20 10:46:35 +08:00
. set_voltage_sel = mc13xxx_regulator_set_voltage_sel ,
2010-12-14 14:00:54 +08:00
. get_voltage = mc13xxx_regulator_get_voltage ,
} ;
2010-12-15 14:10:25 +00:00
EXPORT_SYMBOL_GPL ( mc13xxx_regulator_ops ) ;
2010-12-14 14:00:54 +08:00
int mc13xxx_fixed_regulator_set_voltage ( struct regulator_dev * rdev , int min_uV ,
int max_uV , unsigned * selector )
{
int id = rdev_get_id ( rdev ) ;
dev_dbg ( rdev_get_dev ( rdev ) , " %s id: %d min_uV: %d max_uV: %d \n " ,
__func__ , id , min_uV , max_uV ) ;
2012-06-08 15:41:48 +08:00
if ( min_uV < = rdev - > desc - > volt_table [ 0 ] & &
2012-07-13 23:01:14 +08:00
rdev - > desc - > volt_table [ 0 ] < = max_uV ) {
* selector = 0 ;
2010-12-14 14:00:54 +08:00
return 0 ;
2012-07-13 23:01:14 +08:00
} else {
2010-12-14 14:00:54 +08:00
return - EINVAL ;
2012-07-13 23:01:14 +08:00
}
2010-12-14 14:00:54 +08:00
}
2010-12-15 14:10:25 +00:00
EXPORT_SYMBOL_GPL ( mc13xxx_fixed_regulator_set_voltage ) ;
2010-12-14 14:00:54 +08:00
struct regulator_ops mc13xxx_fixed_regulator_ops = {
. enable = mc13xxx_regulator_enable ,
. disable = mc13xxx_regulator_disable ,
. is_enabled = mc13xxx_regulator_is_enabled ,
2012-06-08 15:41:48 +08:00
. list_voltage = regulator_list_voltage_table ,
2010-12-14 14:00:54 +08:00
. set_voltage = mc13xxx_fixed_regulator_set_voltage ,
} ;
2010-12-15 14:10:25 +00:00
EXPORT_SYMBOL_GPL ( mc13xxx_fixed_regulator_ops ) ;
2010-12-14 14:00:54 +08:00
2011-12-21 23:00:46 +08:00
# ifdef CONFIG_OF
2012-11-19 13:22:22 -05:00
int mc13xxx_get_num_regulators_dt ( struct platform_device * pdev )
2011-12-21 23:00:46 +08:00
{
2013-01-30 20:54:49 +08:00
struct device_node * parent ;
int num ;
2011-12-21 23:00:46 +08:00
2014-03-02 11:44:35 +04:00
if ( ! pdev - > dev . parent - > of_node )
2014-02-25 16:49:27 +05:30
return - ENODEV ;
2014-03-02 11:44:35 +04:00
parent = of_get_child_by_name ( pdev - > dev . parent - > of_node , " regulators " ) ;
2011-12-21 23:00:46 +08:00
if ( ! parent )
return - ENODEV ;
2013-01-30 20:54:49 +08:00
num = of_get_child_count ( parent ) ;
2013-01-27 21:16:56 +08:00
of_node_put ( parent ) ;
2011-12-21 23:00:46 +08:00
return num ;
}
2012-02-09 16:43:01 -05:00
EXPORT_SYMBOL_GPL ( mc13xxx_get_num_regulators_dt ) ;
2011-12-21 23:00:46 +08:00
2012-11-19 13:22:22 -05:00
struct mc13xxx_regulator_init_data * mc13xxx_parse_regulators_dt (
2011-12-21 23:00:46 +08:00
struct platform_device * pdev , struct mc13xxx_regulator * regulators ,
2013-04-27 10:29:24 +04:00
int num_regulators )
2011-12-21 23:00:46 +08:00
{
struct mc13xxx_regulator_priv * priv = platform_get_drvdata ( pdev ) ;
struct mc13xxx_regulator_init_data * data , * p ;
struct device_node * parent , * child ;
2013-01-21 12:25:45 -06:00
int i , parsed = 0 ;
2014-03-02 11:44:35 +04:00
if ( ! pdev - > dev . parent - > of_node )
2014-02-25 16:49:27 +05:30
return NULL ;
2014-03-02 11:44:35 +04:00
parent = of_get_child_by_name ( pdev - > dev . parent - > of_node , " regulators " ) ;
2011-12-21 23:00:46 +08:00
if ( ! parent )
return NULL ;
data = devm_kzalloc ( & pdev - > dev , sizeof ( * data ) * priv - > num_regulators ,
GFP_KERNEL ) ;
2013-01-27 21:16:56 +08:00
if ( ! data ) {
of_node_put ( parent ) ;
2011-12-21 23:00:46 +08:00
return NULL ;
2013-01-27 21:16:56 +08:00
}
2011-12-21 23:00:46 +08:00
p = data ;
for_each_child_of_node ( parent , child ) {
2013-04-27 10:29:24 +04:00
int found = 0 ;
2011-12-21 23:00:46 +08:00
for ( i = 0 ; i < num_regulators ; i + + ) {
2013-04-27 10:29:24 +04:00
if ( ! regulators [ i ] . desc . name )
continue ;
2011-12-21 23:00:46 +08:00
if ( ! of_node_cmp ( child - > name ,
regulators [ i ] . desc . name ) ) {
p - > id = i ;
p - > init_data = of_get_regulator_init_data (
& pdev - > dev , child ) ;
p - > node = child ;
p + + ;
2013-01-21 12:25:45 -06:00
parsed + + ;
2013-04-27 10:29:24 +04:00
found = 1 ;
2011-12-21 23:00:46 +08:00
break ;
}
}
2013-04-27 10:29:24 +04:00
if ( ! found )
dev_warn ( & pdev - > dev ,
" Unknown regulator: %s \n " , child - > name ) ;
2011-12-21 23:00:46 +08:00
}
2013-01-27 21:16:56 +08:00
of_node_put ( parent ) ;
2011-12-21 23:00:46 +08:00
2013-04-27 10:29:24 +04:00
priv - > num_regulators = parsed ;
2011-12-21 23:00:46 +08:00
return data ;
}
2012-02-09 16:43:01 -05:00
EXPORT_SYMBOL_GPL ( mc13xxx_parse_regulators_dt ) ;
2011-12-21 23:00:46 +08:00
# endif
2010-12-14 14:00:54 +08:00
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_AUTHOR ( " Yong Shen <yong.shen@linaro.org> " ) ;
MODULE_DESCRIPTION ( " Regulator Driver for Freescale MC13xxx PMIC " ) ;
MODULE_ALIAS ( " mc13xxx-regulator-core " ) ;