2012-07-11 21:08:17 +09:00
/*
* s2mps11 . c
*
2014-03-07 11:10:50 +01:00
* Copyright ( c ) 2012 - 2014 Samsung Electronics Co . , Ltd
2012-07-11 21:08:17 +09:00
* http : //www.samsung.com
*
2014-03-07 11:10:50 +01:00
* 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 .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2012-07-11 21:08:17 +09:00
*
*/
# include <linux/bug.h>
# include <linux/err.h>
# include <linux/gpio.h>
# include <linux/slab.h>
# include <linux/module.h>
2013-06-29 18:21:17 +05:30
# include <linux/of.h>
2013-06-29 18:21:16 +05:30
# include <linux/regmap.h>
2012-07-11 21:08:17 +09:00
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
2013-06-29 18:21:17 +05:30
# include <linux/regulator/of_regulator.h>
2014-04-14 10:09:07 +02:00
# include <linux/of_gpio.h>
2012-07-11 21:08:17 +09:00
# include <linux/mfd/samsung/core.h>
# include <linux/mfd/samsung/s2mps11.h>
2014-03-07 11:10:50 +01:00
# include <linux/mfd/samsung/s2mps14.h>
2013-06-29 18:21:17 +05:30
2012-07-11 21:08:17 +09:00
struct s2mps11_info {
2014-02-28 11:01:50 +01:00
unsigned int rdev_num ;
2012-07-11 21:08:17 +09:00
int ramp_delay2 ;
int ramp_delay34 ;
int ramp_delay5 ;
int ramp_delay16 ;
int ramp_delay7810 ;
int ramp_delay9 ;
2014-03-07 11:10:51 +01:00
/*
* One bit for each S2MPS14 regulator whether the suspend mode
* was enabled .
*/
unsigned int s2mps14_suspend_state : 30 ;
2014-04-14 10:09:07 +02:00
/* Array of size rdev_num with GPIO-s for external sleep control */
int * ext_control_gpio ;
2012-07-11 21:08:17 +09:00
} ;
static int get_ramp_delay ( int ramp_delay )
{
unsigned char cnt = 0 ;
2013-06-24 16:50:55 +05:30
ramp_delay / = 6250 ;
2012-07-11 21:08:17 +09:00
while ( true ) {
ramp_delay = ramp_delay > > 1 ;
if ( ramp_delay = = 0 )
break ;
cnt + + ;
}
2013-08-05 14:08:37 +08:00
if ( cnt > 3 )
cnt = 3 ;
2012-07-11 21:08:17 +09:00
return cnt ;
}
2013-06-24 16:50:56 +05:30
static int s2mps11_regulator_set_voltage_time_sel ( struct regulator_dev * rdev ,
unsigned int old_selector ,
unsigned int new_selector )
{
struct s2mps11_info * s2mps11 = rdev_get_drvdata ( rdev ) ;
unsigned int ramp_delay = 0 ;
int old_volt , new_volt ;
2014-01-26 21:57:12 -02:00
switch ( rdev_get_id ( rdev ) ) {
2013-06-24 16:50:56 +05:30
case S2MPS11_BUCK2 :
ramp_delay = s2mps11 - > ramp_delay2 ;
break ;
case S2MPS11_BUCK3 :
case S2MPS11_BUCK4 :
ramp_delay = s2mps11 - > ramp_delay34 ;
break ;
case S2MPS11_BUCK5 :
ramp_delay = s2mps11 - > ramp_delay5 ;
break ;
case S2MPS11_BUCK6 :
case S2MPS11_BUCK1 :
ramp_delay = s2mps11 - > ramp_delay16 ;
break ;
case S2MPS11_BUCK7 :
case S2MPS11_BUCK8 :
case S2MPS11_BUCK10 :
ramp_delay = s2mps11 - > ramp_delay7810 ;
break ;
case S2MPS11_BUCK9 :
ramp_delay = s2mps11 - > ramp_delay9 ;
}
if ( ramp_delay = = 0 )
ramp_delay = rdev - > desc - > ramp_delay ;
old_volt = rdev - > desc - > min_uV + ( rdev - > desc - > uV_step * old_selector ) ;
new_volt = rdev - > desc - > min_uV + ( rdev - > desc - > uV_step * new_selector ) ;
return DIV_ROUND_UP ( abs ( new_volt - old_volt ) , ramp_delay ) ;
}
2013-06-29 18:21:16 +05:30
static int s2mps11_set_ramp_delay ( struct regulator_dev * rdev , int ramp_delay )
{
struct s2mps11_info * s2mps11 = rdev_get_drvdata ( rdev ) ;
unsigned int ramp_val , ramp_shift , ramp_reg = S2MPS11_REG_RAMP_BUCK ;
unsigned int ramp_enable = 1 , enable_shift = 0 ;
int ret ;
2014-01-26 21:57:12 -02:00
switch ( rdev_get_id ( rdev ) ) {
2013-06-29 18:21:16 +05:30
case S2MPS11_BUCK1 :
if ( ramp_delay > s2mps11 - > ramp_delay16 )
s2mps11 - > ramp_delay16 = ramp_delay ;
else
ramp_delay = s2mps11 - > ramp_delay16 ;
ramp_shift = S2MPS11_BUCK16_RAMP_SHIFT ;
break ;
case S2MPS11_BUCK2 :
enable_shift = S2MPS11_BUCK2_RAMP_EN_SHIFT ;
if ( ! ramp_delay ) {
ramp_enable = 0 ;
break ;
}
s2mps11 - > ramp_delay2 = ramp_delay ;
ramp_shift = S2MPS11_BUCK2_RAMP_SHIFT ;
ramp_reg = S2MPS11_REG_RAMP ;
break ;
case S2MPS11_BUCK3 :
enable_shift = S2MPS11_BUCK3_RAMP_EN_SHIFT ;
if ( ! ramp_delay ) {
ramp_enable = 0 ;
break ;
}
if ( ramp_delay > s2mps11 - > ramp_delay34 )
s2mps11 - > ramp_delay34 = ramp_delay ;
else
ramp_delay = s2mps11 - > ramp_delay34 ;
ramp_shift = S2MPS11_BUCK34_RAMP_SHIFT ;
ramp_reg = S2MPS11_REG_RAMP ;
break ;
case S2MPS11_BUCK4 :
enable_shift = S2MPS11_BUCK4_RAMP_EN_SHIFT ;
if ( ! ramp_delay ) {
ramp_enable = 0 ;
break ;
}
if ( ramp_delay > s2mps11 - > ramp_delay34 )
s2mps11 - > ramp_delay34 = ramp_delay ;
else
ramp_delay = s2mps11 - > ramp_delay34 ;
ramp_shift = S2MPS11_BUCK34_RAMP_SHIFT ;
ramp_reg = S2MPS11_REG_RAMP ;
break ;
case S2MPS11_BUCK5 :
s2mps11 - > ramp_delay5 = ramp_delay ;
ramp_shift = S2MPS11_BUCK5_RAMP_SHIFT ;
break ;
case S2MPS11_BUCK6 :
enable_shift = S2MPS11_BUCK6_RAMP_EN_SHIFT ;
if ( ! ramp_delay ) {
ramp_enable = 0 ;
break ;
}
if ( ramp_delay > s2mps11 - > ramp_delay16 )
s2mps11 - > ramp_delay16 = ramp_delay ;
else
ramp_delay = s2mps11 - > ramp_delay16 ;
ramp_shift = S2MPS11_BUCK16_RAMP_SHIFT ;
break ;
case S2MPS11_BUCK7 :
case S2MPS11_BUCK8 :
case S2MPS11_BUCK10 :
if ( ramp_delay > s2mps11 - > ramp_delay7810 )
s2mps11 - > ramp_delay7810 = ramp_delay ;
else
ramp_delay = s2mps11 - > ramp_delay7810 ;
ramp_shift = S2MPS11_BUCK7810_RAMP_SHIFT ;
break ;
case S2MPS11_BUCK9 :
s2mps11 - > ramp_delay9 = ramp_delay ;
ramp_shift = S2MPS11_BUCK9_RAMP_SHIFT ;
break ;
default :
return 0 ;
}
if ( ! ramp_enable )
goto ramp_disable ;
2014-05-06 08:37:36 +02:00
/* Ramp delay can be enabled/disabled only for buck[2346] */
if ( ( rdev_get_id ( rdev ) > = S2MPS11_BUCK2 & &
rdev_get_id ( rdev ) < = S2MPS11_BUCK4 ) | |
rdev_get_id ( rdev ) = = S2MPS11_BUCK6 ) {
ret = regmap_update_bits ( rdev - > regmap , S2MPS11_REG_RAMP ,
1 < < enable_shift , 1 < < enable_shift ) ;
if ( ret ) {
dev_err ( & rdev - > dev , " failed to enable ramp rate \n " ) ;
return ret ;
}
2013-06-29 18:21:16 +05:30
}
ramp_val = get_ramp_delay ( ramp_delay ) ;
2013-08-05 14:08:37 +08:00
return regmap_update_bits ( rdev - > regmap , ramp_reg , 0x3 < < ramp_shift ,
ramp_val < < ramp_shift ) ;
2013-06-29 18:21:16 +05:30
ramp_disable :
2013-08-03 17:10:42 +08:00
return regmap_update_bits ( rdev - > regmap , S2MPS11_REG_RAMP ,
1 < < enable_shift , 0 ) ;
2013-06-29 18:21:16 +05:30
}
2012-07-11 21:08:17 +09:00
static struct regulator_ops s2mps11_ldo_ops = {
. list_voltage = regulator_list_voltage_linear ,
. map_voltage = regulator_map_voltage_linear ,
. is_enabled = regulator_is_enabled_regmap ,
. enable = regulator_enable_regmap ,
. disable = regulator_disable_regmap ,
. get_voltage_sel = regulator_get_voltage_sel_regmap ,
. set_voltage_sel = regulator_set_voltage_sel_regmap ,
. set_voltage_time_sel = regulator_set_voltage_time_sel ,
} ;
static struct regulator_ops s2mps11_buck_ops = {
. list_voltage = regulator_list_voltage_linear ,
. map_voltage = regulator_map_voltage_linear ,
. is_enabled = regulator_is_enabled_regmap ,
. enable = regulator_enable_regmap ,
. disable = regulator_disable_regmap ,
. get_voltage_sel = regulator_get_voltage_sel_regmap ,
. set_voltage_sel = regulator_set_voltage_sel_regmap ,
2013-06-24 16:50:56 +05:30
. set_voltage_time_sel = s2mps11_regulator_set_voltage_time_sel ,
2013-06-29 18:21:16 +05:30
. set_ramp_delay = s2mps11_set_ramp_delay ,
2012-07-11 21:08:17 +09:00
} ;
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_ldo1(num) { \
2012-07-11 21:08:17 +09:00
. name = " LDO " # num , \
. id = S2MPS11_LDO # # num , \
. ops = & s2mps11_ldo_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_LDO_MIN , \
. uV_step = S2MPS11_LDO_STEP1 , \
. n_voltages = S2MPS11_LDO_N_VOLTAGES , \
. vsel_reg = S2MPS11_REG_L1CTRL + num - 1 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_LDO_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_L1CTRL + num - 1 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_ldo2(num) { \
2012-07-11 21:08:17 +09:00
. name = " LDO " # num , \
. id = S2MPS11_LDO # # num , \
. ops = & s2mps11_ldo_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_LDO_MIN , \
. uV_step = S2MPS11_LDO_STEP2 , \
. n_voltages = S2MPS11_LDO_N_VOLTAGES , \
. vsel_reg = S2MPS11_REG_L1CTRL + num - 1 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_LDO_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_L1CTRL + num - 1 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_buck1_4(num) { \
2012-07-11 21:08:17 +09:00
. name = " BUCK " # num , \
. id = S2MPS11_BUCK # # num , \
. ops = & s2mps11_buck_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_BUCK_MIN1 , \
. uV_step = S2MPS11_BUCK_STEP1 , \
. n_voltages = S2MPS11_BUCK_N_VOLTAGES , \
2013-06-24 16:50:55 +05:30
. ramp_delay = S2MPS11_RAMP_DELAY , \
2012-07-11 21:08:17 +09:00
. vsel_reg = S2MPS11_REG_B1CTRL2 + ( num - 1 ) * 2 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_BUCK_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_B1CTRL1 + ( num - 1 ) * 2 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_buck5 { \
2012-07-11 21:08:17 +09:00
. name = " BUCK5 " , \
. id = S2MPS11_BUCK5 , \
. ops = & s2mps11_buck_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_BUCK_MIN1 , \
. uV_step = S2MPS11_BUCK_STEP1 , \
. n_voltages = S2MPS11_BUCK_N_VOLTAGES , \
2013-06-24 16:50:55 +05:30
. ramp_delay = S2MPS11_RAMP_DELAY , \
2012-07-11 21:08:17 +09:00
. vsel_reg = S2MPS11_REG_B5CTRL2 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_BUCK_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_B5CTRL1 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_buck6_8(num) { \
2012-07-11 21:08:17 +09:00
. name = " BUCK " # num , \
. id = S2MPS11_BUCK # # num , \
. ops = & s2mps11_buck_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_BUCK_MIN1 , \
. uV_step = S2MPS11_BUCK_STEP1 , \
. n_voltages = S2MPS11_BUCK_N_VOLTAGES , \
2013-06-24 16:50:55 +05:30
. ramp_delay = S2MPS11_RAMP_DELAY , \
2012-07-11 21:08:17 +09:00
. vsel_reg = S2MPS11_REG_B6CTRL2 + ( num - 6 ) * 2 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_BUCK_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_B6CTRL1 + ( num - 6 ) * 2 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_buck9 { \
2012-07-11 21:08:17 +09:00
. name = " BUCK9 " , \
. id = S2MPS11_BUCK9 , \
. ops = & s2mps11_buck_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_BUCK_MIN3 , \
. uV_step = S2MPS11_BUCK_STEP3 , \
. n_voltages = S2MPS11_BUCK_N_VOLTAGES , \
2013-06-24 16:50:55 +05:30
. ramp_delay = S2MPS11_RAMP_DELAY , \
2012-07-11 21:08:17 +09:00
. vsel_reg = S2MPS11_REG_B9CTRL2 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_BUCK_VSEL_MASK , \
2012-07-11 21:08:17 +09:00
. enable_reg = S2MPS11_REG_B9CTRL1 , \
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-07 11:10:50 +01:00
# define regulator_desc_s2mps11_buck10 { \
2012-07-11 21:08:17 +09:00
. name = " BUCK10 " , \
. id = S2MPS11_BUCK10 , \
. ops = & s2mps11_buck_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS11_BUCK_MIN2 , \
. uV_step = S2MPS11_BUCK_STEP2 , \
. n_voltages = S2MPS11_BUCK_N_VOLTAGES , \
2013-06-24 16:50:55 +05:30
. ramp_delay = S2MPS11_RAMP_DELAY , \
2013-01-30 08:02:27 -05:00
. vsel_reg = S2MPS11_REG_B10CTRL2 , \
2012-07-12 09:35:50 +08:00
. vsel_mask = S2MPS11_BUCK_VSEL_MASK , \
2013-01-30 08:02:27 -05:00
. enable_reg = S2MPS11_REG_B10CTRL1 , \
2012-07-11 21:08:17 +09:00
. enable_mask = S2MPS11_ENABLE_MASK \
}
2014-03-03 16:53:51 +01:00
static const struct regulator_desc s2mps11_regulators [ ] = {
2014-03-07 11:10:50 +01:00
regulator_desc_s2mps11_ldo2 ( 1 ) ,
regulator_desc_s2mps11_ldo1 ( 2 ) ,
regulator_desc_s2mps11_ldo1 ( 3 ) ,
regulator_desc_s2mps11_ldo1 ( 4 ) ,
regulator_desc_s2mps11_ldo1 ( 5 ) ,
regulator_desc_s2mps11_ldo2 ( 6 ) ,
regulator_desc_s2mps11_ldo1 ( 7 ) ,
regulator_desc_s2mps11_ldo1 ( 8 ) ,
regulator_desc_s2mps11_ldo1 ( 9 ) ,
regulator_desc_s2mps11_ldo1 ( 10 ) ,
regulator_desc_s2mps11_ldo2 ( 11 ) ,
regulator_desc_s2mps11_ldo1 ( 12 ) ,
regulator_desc_s2mps11_ldo1 ( 13 ) ,
regulator_desc_s2mps11_ldo1 ( 14 ) ,
regulator_desc_s2mps11_ldo1 ( 15 ) ,
regulator_desc_s2mps11_ldo1 ( 16 ) ,
regulator_desc_s2mps11_ldo1 ( 17 ) ,
regulator_desc_s2mps11_ldo1 ( 18 ) ,
regulator_desc_s2mps11_ldo1 ( 19 ) ,
regulator_desc_s2mps11_ldo1 ( 20 ) ,
regulator_desc_s2mps11_ldo1 ( 21 ) ,
regulator_desc_s2mps11_ldo2 ( 22 ) ,
regulator_desc_s2mps11_ldo2 ( 23 ) ,
regulator_desc_s2mps11_ldo1 ( 24 ) ,
regulator_desc_s2mps11_ldo1 ( 25 ) ,
regulator_desc_s2mps11_ldo1 ( 26 ) ,
regulator_desc_s2mps11_ldo2 ( 27 ) ,
regulator_desc_s2mps11_ldo1 ( 28 ) ,
regulator_desc_s2mps11_ldo1 ( 29 ) ,
regulator_desc_s2mps11_ldo1 ( 30 ) ,
regulator_desc_s2mps11_ldo1 ( 31 ) ,
regulator_desc_s2mps11_ldo1 ( 32 ) ,
regulator_desc_s2mps11_ldo1 ( 33 ) ,
regulator_desc_s2mps11_ldo1 ( 34 ) ,
regulator_desc_s2mps11_ldo1 ( 35 ) ,
regulator_desc_s2mps11_ldo1 ( 36 ) ,
regulator_desc_s2mps11_ldo1 ( 37 ) ,
regulator_desc_s2mps11_ldo1 ( 38 ) ,
regulator_desc_s2mps11_buck1_4 ( 1 ) ,
regulator_desc_s2mps11_buck1_4 ( 2 ) ,
regulator_desc_s2mps11_buck1_4 ( 3 ) ,
regulator_desc_s2mps11_buck1_4 ( 4 ) ,
regulator_desc_s2mps11_buck5 ,
regulator_desc_s2mps11_buck6_8 ( 6 ) ,
regulator_desc_s2mps11_buck6_8 ( 7 ) ,
regulator_desc_s2mps11_buck6_8 ( 8 ) ,
regulator_desc_s2mps11_buck9 ,
regulator_desc_s2mps11_buck10 ,
} ;
2014-03-07 11:10:51 +01:00
static int s2mps14_regulator_enable ( struct regulator_dev * rdev )
{
struct s2mps11_info * s2mps11 = rdev_get_drvdata ( rdev ) ;
unsigned int val ;
if ( s2mps11 - > s2mps14_suspend_state & ( 1 < < rdev_get_id ( rdev ) ) )
val = S2MPS14_ENABLE_SUSPEND ;
2014-04-30 10:40:42 +02:00
else if ( gpio_is_valid ( s2mps11 - > ext_control_gpio [ rdev_get_id ( rdev ) ] ) )
2014-04-14 10:09:07 +02:00
val = S2MPS14_ENABLE_EXT_CONTROL ;
2014-03-07 11:10:51 +01:00
else
val = rdev - > desc - > enable_mask ;
return regmap_update_bits ( rdev - > regmap , rdev - > desc - > enable_reg ,
rdev - > desc - > enable_mask , val ) ;
}
static int s2mps14_regulator_set_suspend_disable ( struct regulator_dev * rdev )
{
int ret ;
unsigned int val ;
struct s2mps11_info * s2mps11 = rdev_get_drvdata ( rdev ) ;
/* LDO3 should be always on and does not support suspend mode */
if ( rdev_get_id ( rdev ) = = S2MPS14_LDO3 )
return 0 ;
ret = regmap_read ( rdev - > regmap , rdev - > desc - > enable_reg , & val ) ;
if ( ret < 0 )
return ret ;
s2mps11 - > s2mps14_suspend_state | = ( 1 < < rdev_get_id ( rdev ) ) ;
/*
* Don ' t enable suspend mode if regulator is already disabled because
* this would effectively for a short time turn on the regulator after
* resuming .
* However we still want to toggle the suspend_state bit for regulator
* in case if it got enabled before suspending the system .
*/
if ( ! ( val & rdev - > desc - > enable_mask ) )
return 0 ;
return regmap_update_bits ( rdev - > regmap , rdev - > desc - > enable_reg ,
rdev - > desc - > enable_mask , S2MPS14_ENABLE_SUSPEND ) ;
}
2014-03-07 11:10:50 +01:00
static struct regulator_ops s2mps14_reg_ops = {
. list_voltage = regulator_list_voltage_linear ,
. map_voltage = regulator_map_voltage_linear ,
. is_enabled = regulator_is_enabled_regmap ,
2014-03-07 11:10:51 +01:00
. enable = s2mps14_regulator_enable ,
2014-03-07 11:10:50 +01:00
. disable = regulator_disable_regmap ,
. get_voltage_sel = regulator_get_voltage_sel_regmap ,
. set_voltage_sel = regulator_set_voltage_sel_regmap ,
. set_voltage_time_sel = regulator_set_voltage_time_sel ,
2014-03-07 11:10:51 +01:00
. set_suspend_disable = s2mps14_regulator_set_suspend_disable ,
2014-03-07 11:10:50 +01:00
} ;
# define regulator_desc_s2mps14_ldo1(num) { \
. name = " LDO " # num , \
. id = S2MPS14_LDO # # num , \
. ops = & s2mps14_reg_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS14_LDO_MIN_800MV , \
. uV_step = S2MPS14_LDO_STEP_25MV , \
. n_voltages = S2MPS14_LDO_N_VOLTAGES , \
. vsel_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. vsel_mask = S2MPS14_LDO_VSEL_MASK , \
. enable_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. enable_mask = S2MPS14_ENABLE_MASK \
}
# define regulator_desc_s2mps14_ldo2(num) { \
. name = " LDO " # num , \
. id = S2MPS14_LDO # # num , \
. ops = & s2mps14_reg_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS14_LDO_MIN_1800MV , \
. uV_step = S2MPS14_LDO_STEP_25MV , \
. n_voltages = S2MPS14_LDO_N_VOLTAGES , \
. vsel_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. vsel_mask = S2MPS14_LDO_VSEL_MASK , \
. enable_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. enable_mask = S2MPS14_ENABLE_MASK \
}
# define regulator_desc_s2mps14_ldo3(num) { \
. name = " LDO " # num , \
. id = S2MPS14_LDO # # num , \
. ops = & s2mps14_reg_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS14_LDO_MIN_800MV , \
. uV_step = S2MPS14_LDO_STEP_12_5MV , \
. n_voltages = S2MPS14_LDO_N_VOLTAGES , \
. vsel_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. vsel_mask = S2MPS14_LDO_VSEL_MASK , \
. enable_reg = S2MPS14_REG_L1CTRL + num - 1 , \
. enable_mask = S2MPS14_ENABLE_MASK \
}
# define regulator_desc_s2mps14_buck1235(num) { \
. name = " BUCK " # num , \
. id = S2MPS14_BUCK # # num , \
. ops = & s2mps14_reg_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS14_BUCK1235_MIN_600MV , \
. uV_step = S2MPS14_BUCK1235_STEP_6_25MV , \
. n_voltages = S2MPS14_BUCK_N_VOLTAGES , \
. linear_min_sel = S2MPS14_BUCK1235_START_SEL , \
. ramp_delay = S2MPS14_BUCK_RAMP_DELAY , \
. vsel_reg = S2MPS14_REG_B1CTRL2 + ( num - 1 ) * 2 , \
. vsel_mask = S2MPS14_BUCK_VSEL_MASK , \
. enable_reg = S2MPS14_REG_B1CTRL1 + ( num - 1 ) * 2 , \
. enable_mask = S2MPS14_ENABLE_MASK \
}
# define regulator_desc_s2mps14_buck4(num) { \
. name = " BUCK " # num , \
. id = S2MPS14_BUCK # # num , \
. ops = & s2mps14_reg_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
. min_uV = S2MPS14_BUCK4_MIN_1400MV , \
. uV_step = S2MPS14_BUCK4_STEP_12_5MV , \
. n_voltages = S2MPS14_BUCK_N_VOLTAGES , \
. linear_min_sel = S2MPS14_BUCK4_START_SEL , \
. ramp_delay = S2MPS14_BUCK_RAMP_DELAY , \
. vsel_reg = S2MPS14_REG_B1CTRL2 + ( num - 1 ) * 2 , \
. vsel_mask = S2MPS14_BUCK_VSEL_MASK , \
. enable_reg = S2MPS14_REG_B1CTRL1 + ( num - 1 ) * 2 , \
. enable_mask = S2MPS14_ENABLE_MASK \
}
static const struct regulator_desc s2mps14_regulators [ ] = {
regulator_desc_s2mps14_ldo3 ( 1 ) ,
regulator_desc_s2mps14_ldo3 ( 2 ) ,
regulator_desc_s2mps14_ldo1 ( 3 ) ,
regulator_desc_s2mps14_ldo1 ( 4 ) ,
regulator_desc_s2mps14_ldo3 ( 5 ) ,
regulator_desc_s2mps14_ldo3 ( 6 ) ,
regulator_desc_s2mps14_ldo1 ( 7 ) ,
regulator_desc_s2mps14_ldo2 ( 8 ) ,
regulator_desc_s2mps14_ldo3 ( 9 ) ,
regulator_desc_s2mps14_ldo3 ( 10 ) ,
regulator_desc_s2mps14_ldo1 ( 11 ) ,
regulator_desc_s2mps14_ldo2 ( 12 ) ,
regulator_desc_s2mps14_ldo2 ( 13 ) ,
regulator_desc_s2mps14_ldo2 ( 14 ) ,
regulator_desc_s2mps14_ldo2 ( 15 ) ,
regulator_desc_s2mps14_ldo2 ( 16 ) ,
regulator_desc_s2mps14_ldo2 ( 17 ) ,
regulator_desc_s2mps14_ldo2 ( 18 ) ,
regulator_desc_s2mps14_ldo1 ( 19 ) ,
regulator_desc_s2mps14_ldo1 ( 20 ) ,
regulator_desc_s2mps14_ldo1 ( 21 ) ,
regulator_desc_s2mps14_ldo3 ( 22 ) ,
regulator_desc_s2mps14_ldo1 ( 23 ) ,
regulator_desc_s2mps14_ldo2 ( 24 ) ,
regulator_desc_s2mps14_ldo2 ( 25 ) ,
regulator_desc_s2mps14_buck1235 ( 1 ) ,
regulator_desc_s2mps14_buck1235 ( 2 ) ,
regulator_desc_s2mps14_buck1235 ( 3 ) ,
regulator_desc_s2mps14_buck4 ( 4 ) ,
regulator_desc_s2mps14_buck1235 ( 5 ) ,
2012-07-11 21:08:17 +09:00
} ;
2014-04-14 10:09:07 +02:00
static int s2mps14_pmic_enable_ext_control ( struct s2mps11_info * s2mps11 ,
struct regulator_dev * rdev )
{
return regmap_update_bits ( rdev - > regmap , rdev - > desc - > enable_reg ,
rdev - > desc - > enable_mask , S2MPS14_ENABLE_EXT_CONTROL ) ;
}
static void s2mps14_pmic_dt_parse_ext_control_gpio ( struct platform_device * pdev ,
2014-04-14 10:09:06 +02:00
struct of_regulator_match * rdata , struct s2mps11_info * s2mps11 )
2014-04-14 10:09:07 +02:00
{
int * gpio = s2mps11 - > ext_control_gpio ;
unsigned int i ;
unsigned int valid_regulators [ 3 ] = { S2MPS14_LDO10 , S2MPS14_LDO11 ,
S2MPS14_LDO12 } ;
for ( i = 0 ; i < ARRAY_SIZE ( valid_regulators ) ; i + + ) {
unsigned int reg = valid_regulators [ i ] ;
if ( ! rdata [ reg ] . init_data | | ! rdata [ reg ] . of_node )
continue ;
gpio [ reg ] = of_get_named_gpio ( rdata [ reg ] . of_node ,
" samsung,ext-control-gpios " , 0 ) ;
2014-04-30 10:40:42 +02:00
if ( gpio_is_valid ( gpio [ reg ] ) )
2014-04-14 10:09:07 +02:00
dev_dbg ( & pdev - > dev , " Using GPIO %d for ext-control over %d/%s \n " ,
gpio [ reg ] , reg , rdata [ reg ] . name ) ;
}
}
static int s2mps11_pmic_dt_parse ( struct platform_device * pdev ,
struct of_regulator_match * rdata , struct s2mps11_info * s2mps11 ,
enum sec_device_type dev_type )
2014-04-14 10:09:06 +02:00
{
struct device_node * reg_np ;
reg_np = of_get_child_by_name ( pdev - > dev . parent - > of_node , " regulators " ) ;
if ( ! reg_np ) {
dev_err ( & pdev - > dev , " could not find regulators sub-node \n " ) ;
return - EINVAL ;
}
of_regulator_match ( & pdev - > dev , reg_np , rdata , s2mps11 - > rdev_num ) ;
2014-04-14 10:09:07 +02:00
if ( dev_type = = S2MPS14X )
s2mps14_pmic_dt_parse_ext_control_gpio ( pdev , rdata , s2mps11 ) ;
2014-04-14 10:09:06 +02:00
of_node_put ( reg_np ) ;
return 0 ;
}
2012-11-19 13:22:22 -05:00
static int s2mps11_pmic_probe ( struct platform_device * pdev )
2012-07-11 21:08:17 +09:00
{
struct sec_pmic_dev * iodev = dev_get_drvdata ( pdev - > dev . parent ) ;
2014-04-14 10:09:06 +02:00
struct sec_platform_data * pdata = NULL ;
2014-02-28 11:01:50 +01:00
struct of_regulator_match * rdata = NULL ;
2012-07-11 21:08:17 +09:00
struct regulator_config config = { } ;
struct s2mps11_info * s2mps11 ;
2014-02-28 11:01:50 +01:00
int i , ret = 0 ;
2014-03-03 16:53:51 +01:00
const struct regulator_desc * regulators ;
enum sec_device_type dev_type ;
2012-07-11 21:08:17 +09:00
s2mps11 = devm_kzalloc ( & pdev - > dev , sizeof ( struct s2mps11_info ) ,
GFP_KERNEL ) ;
if ( ! s2mps11 )
return - ENOMEM ;
2014-03-03 16:53:51 +01:00
dev_type = platform_get_device_id ( pdev ) - > driver_data ;
switch ( dev_type ) {
case S2MPS11X :
s2mps11 - > rdev_num = ARRAY_SIZE ( s2mps11_regulators ) ;
regulators = s2mps11_regulators ;
break ;
2014-03-07 11:10:50 +01:00
case S2MPS14X :
s2mps11 - > rdev_num = ARRAY_SIZE ( s2mps14_regulators ) ;
regulators = s2mps14_regulators ;
break ;
2014-03-03 16:53:51 +01:00
default :
dev_err ( & pdev - > dev , " Invalid device type: %u \n " , dev_type ) ;
return - EINVAL ;
} ;
2014-02-28 11:01:50 +01:00
2014-04-14 10:09:07 +02:00
s2mps11 - > ext_control_gpio = devm_kzalloc ( & pdev - > dev ,
sizeof ( * s2mps11 - > ext_control_gpio ) * s2mps11 - > rdev_num ,
GFP_KERNEL ) ;
if ( ! s2mps11 - > ext_control_gpio )
return - ENOMEM ;
2014-04-30 10:40:42 +02:00
/*
* 0 is a valid GPIO so initialize all GPIO - s to negative value
* to indicate that external control won ' t be used for this regulator .
*/
for ( i = 0 ; i < s2mps11 - > rdev_num ; i + + )
s2mps11 - > ext_control_gpio [ i ] = - EINVAL ;
2014-04-14 10:09:07 +02:00
2013-06-29 18:21:18 +05:30
if ( ! iodev - > dev - > of_node ) {
2014-04-14 10:09:06 +02:00
if ( iodev - > pdata ) {
pdata = iodev - > pdata ;
2013-06-29 18:21:18 +05:30
goto common_reg ;
} else {
dev_err ( pdev - > dev . parent ,
" Platform data or DT node not supplied \n " ) ;
return - ENODEV ;
}
}
2013-06-29 18:21:17 +05:30
2014-03-03 16:53:51 +01:00
rdata = kzalloc ( sizeof ( * rdata ) * s2mps11 - > rdev_num , GFP_KERNEL ) ;
2014-02-28 11:01:50 +01:00
if ( ! rdata )
return - ENOMEM ;
2014-03-03 16:53:51 +01:00
for ( i = 0 ; i < s2mps11 - > rdev_num ; i + + )
2013-06-29 18:21:17 +05:30
rdata [ i ] . name = regulators [ i ] . name ;
2014-04-14 10:09:07 +02:00
ret = s2mps11_pmic_dt_parse ( pdev , rdata , s2mps11 , dev_type ) ;
2014-04-14 10:09:06 +02:00
if ( ret )
2014-02-28 11:01:50 +01:00
goto out ;
2013-06-29 18:21:17 +05:30
common_reg :
platform_set_drvdata ( pdev , s2mps11 ) ;
2012-07-11 21:08:17 +09:00
2013-06-29 18:21:17 +05:30
config . dev = & pdev - > dev ;
2013-12-11 15:07:43 +01:00
config . regmap = iodev - > regmap_pmic ;
2013-06-29 18:21:17 +05:30
config . driver_data = s2mps11 ;
2014-04-30 10:40:42 +02:00
config . ena_gpio_flags = GPIOF_OUT_INIT_HIGH ;
2014-03-03 16:53:51 +01:00
for ( i = 0 ; i < s2mps11 - > rdev_num ; i + + ) {
2014-02-28 11:01:48 +01:00
struct regulator_dev * regulator ;
2014-04-14 10:09:06 +02:00
if ( pdata ) {
2013-06-29 18:21:17 +05:30
config . init_data = pdata - > regulators [ i ] . initdata ;
2014-01-30 14:51:19 +01:00
config . of_node = pdata - > regulators [ i ] . reg_node ;
2013-06-29 18:21:17 +05:30
} else {
config . init_data = rdata [ i ] . init_data ;
config . of_node = rdata [ i ] . of_node ;
}
2014-04-30 10:40:42 +02:00
config . ena_gpio = s2mps11 - > ext_control_gpio [ i ] ;
2014-04-14 10:09:07 +02:00
2014-02-28 11:01:48 +01:00
regulator = devm_regulator_register ( & pdev - > dev ,
2013-09-04 12:12:15 +05:30
& regulators [ i ] , & config ) ;
2014-02-28 11:01:48 +01:00
if ( IS_ERR ( regulator ) ) {
ret = PTR_ERR ( regulator ) ;
2012-07-12 09:37:37 +08:00
dev_err ( & pdev - > dev , " regulator init failed for %d \n " ,
i ) ;
2014-02-28 11:01:50 +01:00
goto out ;
2012-07-11 21:08:17 +09:00
}
2014-04-14 10:09:07 +02:00
2014-04-30 10:40:42 +02:00
if ( gpio_is_valid ( s2mps11 - > ext_control_gpio [ i ] ) ) {
2014-04-14 10:09:07 +02:00
ret = s2mps14_pmic_enable_ext_control ( s2mps11 ,
regulator ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev ,
" failed to enable GPIO control over %s: %d \n " ,
regulator - > desc - > name , ret ) ;
goto out ;
}
}
2012-07-11 21:08:17 +09:00
}
2014-02-28 11:01:50 +01:00
out :
kfree ( rdata ) ;
return ret ;
2012-07-11 21:08:17 +09:00
}
static const struct platform_device_id s2mps11_pmic_id [ ] = {
2014-02-28 11:01:50 +01:00
{ " s2mps11-pmic " , S2MPS11X } ,
2014-03-07 11:10:50 +01:00
{ " s2mps14-pmic " , S2MPS14X } ,
2012-07-11 21:08:17 +09:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( platform , s2mps11_pmic_id ) ;
static struct platform_driver s2mps11_pmic_driver = {
. driver = {
. name = " s2mps11-pmic " ,
. owner = THIS_MODULE ,
} ,
. probe = s2mps11_pmic_probe ,
. id_table = s2mps11_pmic_id ,
} ;
static int __init s2mps11_pmic_init ( void )
{
return platform_driver_register ( & s2mps11_pmic_driver ) ;
}
subsys_initcall ( s2mps11_pmic_init ) ;
static void __exit s2mps11_pmic_exit ( void )
{
platform_driver_unregister ( & s2mps11_pmic_driver ) ;
}
module_exit ( s2mps11_pmic_exit ) ;
/* Module information */
MODULE_AUTHOR ( " Sangbeom Kim <sbkim73@samsung.com> " ) ;
2014-03-07 11:10:50 +01:00
MODULE_DESCRIPTION ( " SAMSUNG S2MPS11/S2MPS14 Regulator Driver " ) ;
2012-07-11 21:08:17 +09:00
MODULE_LICENSE ( " GPL " ) ;