2012-01-09 14:10:25 +04:00
/*
* s5m8767 . c
*
* Copyright ( c ) 2011 Samsung Electronics Co . , Ltd
* http : //www.samsung.com
*
* 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/bug.h>
# include <linux/err.h>
# include <linux/gpio.h>
# include <linux/slab.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
# include <linux/mfd/s5m87xx/s5m-core.h>
# include <linux/mfd/s5m87xx/s5m-pmic.h>
struct s5m8767_info {
struct device * dev ;
struct s5m87xx_dev * iodev ;
int num_regulators ;
struct regulator_dev * * rdev ;
2012-03-09 12:55:34 +04:00
struct s5m_opmode_data * opmode ;
2012-01-09 14:10:25 +04:00
int ramp_delay ;
bool buck2_ramp ;
bool buck3_ramp ;
bool buck4_ramp ;
bool buck2_gpiodvs ;
bool buck3_gpiodvs ;
bool buck4_gpiodvs ;
u8 buck2_vol [ 8 ] ;
u8 buck3_vol [ 8 ] ;
u8 buck4_vol [ 8 ] ;
int buck_gpios [ 3 ] ;
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
int buck_ds [ 3 ] ;
2012-01-09 14:10:25 +04:00
int buck_gpioindex ;
} ;
struct s5m_voltage_desc {
int max ;
int min ;
int step ;
} ;
static const struct s5m_voltage_desc buck_voltage_val1 = {
. max = 2225000 ,
. min = 650000 ,
. step = 6250 ,
} ;
static const struct s5m_voltage_desc buck_voltage_val2 = {
. max = 1600000 ,
. min = 600000 ,
. step = 6250 ,
} ;
static const struct s5m_voltage_desc buck_voltage_val3 = {
. max = 3000000 ,
. min = 750000 ,
. step = 12500 ,
} ;
static const struct s5m_voltage_desc ldo_voltage_val1 = {
. max = 3950000 ,
. min = 800000 ,
. step = 50000 ,
} ;
static const struct s5m_voltage_desc ldo_voltage_val2 = {
. max = 2375000 ,
. min = 800000 ,
. step = 25000 ,
} ;
static const struct s5m_voltage_desc * reg_voltage_map [ ] = {
[ S5M8767_LDO1 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO2 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO3 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO4 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO5 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO6 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO7 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO8 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO9 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO10 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO11 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO12 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO13 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO14 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO15 ] = & ldo_voltage_val2 ,
[ S5M8767_LDO16 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO17 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO18 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO19 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO20 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO21 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO22 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO23 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO24 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO25 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO26 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO27 ] = & ldo_voltage_val1 ,
[ S5M8767_LDO28 ] = & ldo_voltage_val1 ,
[ S5M8767_BUCK1 ] = & buck_voltage_val1 ,
[ S5M8767_BUCK2 ] = & buck_voltage_val2 ,
[ S5M8767_BUCK3 ] = & buck_voltage_val2 ,
[ S5M8767_BUCK4 ] = & buck_voltage_val2 ,
[ S5M8767_BUCK5 ] = & buck_voltage_val1 ,
[ S5M8767_BUCK6 ] = & buck_voltage_val1 ,
[ S5M8767_BUCK7 ] = NULL ,
[ S5M8767_BUCK8 ] = NULL ,
[ S5M8767_BUCK9 ] = & buck_voltage_val3 ,
} ;
2012-05-09 23:37:51 +04:00
static unsigned int s5m8767_opmode_reg [ ] [ 4 ] = {
2012-03-09 12:55:34 +04:00
/* {OFF, ON, LOWPOWER, SUSPEND} */
/* LDO1 ... LDO28 */
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO1 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x0 , 0x0 , 0x0 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO5 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO10 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO15 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x0 , 0x0 , 0x0 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO20 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x0 , 0x0 , 0x0 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO25 */
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* LDO28 */
/* BUCK1 ... BUCK9 */
{ 0x0 , 0x3 , 0x1 , 0x1 } , /* BUCK1 */
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x2 , 0x1 } , /* BUCK5 */
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x1 , 0x1 } ,
{ 0x0 , 0x3 , 0x1 , 0x1 } , /* BUCK9 */
} ;
static int s5m8767_get_register ( struct regulator_dev * rdev , int * reg ,
int * enable_ctrl )
2012-01-09 14:10:25 +04:00
{
2012-01-31 11:13:31 +04:00
int reg_id = rdev_get_id ( rdev ) ;
2012-03-09 12:55:34 +04:00
unsigned int mode ;
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
2012-01-09 14:10:25 +04:00
switch ( reg_id ) {
case S5M8767_LDO1 . . . S5M8767_LDO2 :
* reg = S5M8767_REG_LDO1CTRL + ( reg_id - S5M8767_LDO1 ) ;
break ;
case S5M8767_LDO3 . . . S5M8767_LDO28 :
* reg = S5M8767_REG_LDO3CTRL + ( reg_id - S5M8767_LDO3 ) ;
break ;
case S5M8767_BUCK1 :
* reg = S5M8767_REG_BUCK1CTRL1 ;
break ;
case S5M8767_BUCK2 . . . S5M8767_BUCK4 :
* reg = S5M8767_REG_BUCK2CTRL + ( reg_id - S5M8767_BUCK2 ) * 9 ;
break ;
case S5M8767_BUCK5 :
* reg = S5M8767_REG_BUCK5CTRL1 ;
break ;
case S5M8767_BUCK6 . . . S5M8767_BUCK9 :
* reg = S5M8767_REG_BUCK6CTRL1 + ( reg_id - S5M8767_BUCK6 ) * 2 ;
break ;
default :
return - EINVAL ;
}
2012-03-09 12:55:34 +04:00
mode = s5m8767 - > opmode [ reg_id ] . mode ;
* enable_ctrl = s5m8767_opmode_reg [ reg_id ] [ mode ] < < S5M8767_ENCTRL_SHIFT ;
2012-01-09 14:10:25 +04:00
return 0 ;
}
static int s5m8767_reg_is_enabled ( struct regulator_dev * rdev )
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
int ret , reg ;
2012-03-09 12:55:34 +04:00
int mask = 0xc0 , enable_ctrl ;
2012-01-09 14:10:25 +04:00
u8 val ;
2012-03-09 12:55:34 +04:00
ret = s5m8767_get_register ( rdev , & reg , & enable_ctrl ) ;
2012-01-09 14:10:25 +04:00
if ( ret = = - EINVAL )
return 1 ;
else if ( ret )
return ret ;
ret = s5m_reg_read ( s5m8767 - > iodev , reg , & val ) ;
if ( ret )
return ret ;
2012-03-09 12:55:34 +04:00
return ( val & mask ) = = enable_ctrl ;
2012-01-09 14:10:25 +04:00
}
static int s5m8767_reg_enable ( struct regulator_dev * rdev )
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
int ret , reg ;
2012-03-09 12:55:34 +04:00
int mask = 0xc0 , enable_ctrl ;
2012-01-09 14:10:25 +04:00
2012-03-09 12:55:34 +04:00
ret = s5m8767_get_register ( rdev , & reg , & enable_ctrl ) ;
2012-01-09 14:10:25 +04:00
if ( ret )
return ret ;
2012-03-09 12:55:34 +04:00
return s5m_reg_update ( s5m8767 - > iodev , reg , enable_ctrl , mask ) ;
2012-01-09 14:10:25 +04:00
}
static int s5m8767_reg_disable ( struct regulator_dev * rdev )
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
int ret , reg ;
2012-03-09 12:55:34 +04:00
int mask = 0xc0 , enable_ctrl ;
2012-01-09 14:10:25 +04:00
2012-03-09 12:55:34 +04:00
ret = s5m8767_get_register ( rdev , & reg , & enable_ctrl ) ;
2012-01-09 14:10:25 +04:00
if ( ret )
return ret ;
2012-03-09 12:55:34 +04:00
return s5m_reg_update ( s5m8767 - > iodev , reg , ~ mask , mask ) ;
2012-01-09 14:10:25 +04:00
}
static int s5m8767_get_voltage_register ( struct regulator_dev * rdev , int * _reg )
{
2012-03-10 06:59:43 +04:00
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
2012-01-31 11:13:31 +04:00
int reg_id = rdev_get_id ( rdev ) ;
2012-01-09 14:10:25 +04:00
int reg ;
switch ( reg_id ) {
case S5M8767_LDO1 . . . S5M8767_LDO2 :
reg = S5M8767_REG_LDO1CTRL + ( reg_id - S5M8767_LDO1 ) ;
break ;
case S5M8767_LDO3 . . . S5M8767_LDO28 :
reg = S5M8767_REG_LDO3CTRL + ( reg_id - S5M8767_LDO3 ) ;
break ;
case S5M8767_BUCK1 :
reg = S5M8767_REG_BUCK1CTRL2 ;
break ;
case S5M8767_BUCK2 :
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
reg = S5M8767_REG_BUCK2DVS2 ;
2012-03-10 06:59:43 +04:00
if ( s5m8767 - > buck2_gpiodvs )
reg + = s5m8767 - > buck_gpioindex ;
2012-01-09 14:10:25 +04:00
break ;
case S5M8767_BUCK3 :
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
reg = S5M8767_REG_BUCK3DVS2 ;
2012-03-10 06:59:43 +04:00
if ( s5m8767 - > buck3_gpiodvs )
reg + = s5m8767 - > buck_gpioindex ;
2012-01-09 14:10:25 +04:00
break ;
case S5M8767_BUCK4 :
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
reg = S5M8767_REG_BUCK4DVS2 ;
2012-03-10 06:59:43 +04:00
if ( s5m8767 - > buck4_gpiodvs )
reg + = s5m8767 - > buck_gpioindex ;
2012-01-09 14:10:25 +04:00
break ;
case S5M8767_BUCK5 :
reg = S5M8767_REG_BUCK5CTRL2 ;
break ;
case S5M8767_BUCK6 . . . S5M8767_BUCK9 :
reg = S5M8767_REG_BUCK6CTRL2 + ( reg_id - S5M8767_BUCK6 ) * 2 ;
break ;
default :
return - EINVAL ;
}
* _reg = reg ;
return 0 ;
}
static int s5m8767_get_voltage_sel ( struct regulator_dev * rdev )
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
2012-03-10 06:59:43 +04:00
int reg , mask , ret ;
2012-01-31 11:13:31 +04:00
int reg_id = rdev_get_id ( rdev ) ;
2012-01-09 14:10:25 +04:00
u8 val ;
ret = s5m8767_get_voltage_register ( rdev , & reg ) ;
if ( ret )
return ret ;
2012-03-10 06:59:43 +04:00
mask = ( reg_id < S5M8767_BUCK1 ) ? 0x3f : 0xff ;
2012-01-09 14:10:25 +04:00
ret = s5m_reg_read ( s5m8767 - > iodev , reg , & val ) ;
if ( ret )
return ret ;
val & = mask ;
return val ;
}
2012-03-09 07:31:08 +04:00
static int s5m8767_convert_voltage_to_sel (
2012-01-09 14:10:25 +04:00
const struct s5m_voltage_desc * desc ,
int min_vol , int max_vol )
{
2012-03-09 07:31:08 +04:00
int selector = 0 ;
2012-01-09 14:10:25 +04:00
if ( desc = = NULL )
return - EINVAL ;
if ( max_vol < desc - > min | | min_vol > desc - > max )
return - EINVAL ;
2012-04-13 05:30:05 +04:00
if ( min_vol < desc - > min )
min_vol = desc - > min ;
selector = DIV_ROUND_UP ( min_vol - desc - > min , desc - > step ) ;
2012-01-09 14:10:25 +04:00
2012-03-09 07:31:08 +04:00
if ( desc - > min + desc - > step * selector > max_vol )
2012-01-09 14:10:25 +04:00
return - EINVAL ;
2012-03-09 07:31:08 +04:00
return selector ;
2012-01-09 14:10:25 +04:00
}
2012-06-13 12:28:18 +04:00
static inline int s5m8767_set_high ( struct s5m8767_info * s5m8767 )
2012-04-10 10:05:44 +04:00
{
int temp_index = s5m8767 - > buck_gpioindex ;
gpio_set_value ( s5m8767 - > buck_gpios [ 0 ] , ( temp_index > > 2 ) & 0x1 ) ;
gpio_set_value ( s5m8767 - > buck_gpios [ 1 ] , ( temp_index > > 1 ) & 0x1 ) ;
gpio_set_value ( s5m8767 - > buck_gpios [ 2 ] , temp_index & 0x1 ) ;
2012-06-13 12:28:18 +04:00
return 0 ;
2012-04-10 10:05:44 +04:00
}
2012-06-13 12:28:18 +04:00
static inline int s5m8767_set_low ( struct s5m8767_info * s5m8767 )
2012-04-10 10:05:44 +04:00
{
int temp_index = s5m8767 - > buck_gpioindex ;
gpio_set_value ( s5m8767 - > buck_gpios [ 2 ] , temp_index & 0x1 ) ;
gpio_set_value ( s5m8767 - > buck_gpios [ 1 ] , ( temp_index > > 1 ) & 0x1 ) ;
gpio_set_value ( s5m8767 - > buck_gpios [ 0 ] , ( temp_index > > 2 ) & 0x1 ) ;
2012-06-13 12:28:18 +04:00
return 0 ;
2012-04-10 10:05:44 +04:00
}
2012-06-13 12:28:18 +04:00
static int s5m8767_set_voltage_sel ( struct regulator_dev * rdev ,
unsigned selector )
2012-01-09 14:10:25 +04:00
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
2012-01-31 11:13:31 +04:00
int reg_id = rdev_get_id ( rdev ) ;
2012-06-13 12:28:18 +04:00
int reg , mask , ret = 0 , old_index , index = 0 ;
2012-04-10 10:05:44 +04:00
u8 * buck234_vol = NULL ;
2012-01-09 14:10:25 +04:00
switch ( reg_id ) {
case S5M8767_LDO1 . . . S5M8767_LDO28 :
mask = 0x3f ;
break ;
case S5M8767_BUCK1 . . . S5M8767_BUCK6 :
mask = 0xff ;
2012-04-10 10:05:44 +04:00
if ( reg_id = = S5M8767_BUCK2 & & s5m8767 - > buck2_gpiodvs )
buck234_vol = & s5m8767 - > buck2_vol [ 0 ] ;
else if ( reg_id = = S5M8767_BUCK3 & & s5m8767 - > buck3_gpiodvs )
buck234_vol = & s5m8767 - > buck3_vol [ 0 ] ;
else if ( reg_id = = S5M8767_BUCK4 & & s5m8767 - > buck4_gpiodvs )
buck234_vol = & s5m8767 - > buck4_vol [ 0 ] ;
2012-01-09 14:10:25 +04:00
break ;
case S5M8767_BUCK7 . . . S5M8767_BUCK8 :
return - EINVAL ;
case S5M8767_BUCK9 :
mask = 0xff ;
break ;
default :
return - EINVAL ;
}
2012-04-10 10:05:44 +04:00
/* buck234_vol != NULL means to control buck234 voltage via DVS GPIO */
if ( buck234_vol ) {
2012-06-13 12:28:18 +04:00
while ( * buck234_vol ! = selector ) {
2012-04-10 10:05:44 +04:00
buck234_vol + + ;
index + + ;
}
old_index = s5m8767 - > buck_gpioindex ;
s5m8767 - > buck_gpioindex = index ;
if ( index > old_index )
2012-06-13 12:28:18 +04:00
return s5m8767_set_high ( s5m8767 ) ;
2012-04-10 10:05:44 +04:00
else
2012-06-13 12:28:18 +04:00
return s5m8767_set_low ( s5m8767 ) ;
2012-04-10 10:05:44 +04:00
} else {
ret = s5m8767_get_voltage_register ( rdev , & reg ) ;
if ( ret )
return ret ;
2012-06-13 12:28:18 +04:00
return s5m_reg_update ( s5m8767 - > iodev , reg , selector , mask ) ;
2012-04-10 10:05:44 +04:00
}
2012-01-09 14:10:25 +04:00
}
static int s5m8767_set_voltage_time_sel ( struct regulator_dev * rdev ,
unsigned int old_sel ,
unsigned int new_sel )
{
struct s5m8767_info * s5m8767 = rdev_get_drvdata ( rdev ) ;
const struct s5m_voltage_desc * desc ;
2012-01-31 11:13:31 +04:00
int reg_id = rdev_get_id ( rdev ) ;
2012-01-09 14:10:25 +04:00
desc = reg_voltage_map [ reg_id ] ;
2012-06-13 12:27:16 +04:00
if ( ( old_sel < new_sel ) & & s5m8767 - > ramp_delay )
2012-02-24 10:52:45 +04:00
return DIV_ROUND_UP ( desc - > step * ( new_sel - old_sel ) ,
2012-03-09 11:28:10 +04:00
s5m8767 - > ramp_delay * 1000 ) ;
2012-02-24 10:52:45 +04:00
return 0 ;
2012-01-09 14:10:25 +04:00
}
2012-04-10 10:07:15 +04:00
static struct regulator_ops s5m8767_ops = {
2012-06-18 10:25:27 +04:00
. list_voltage = regulator_list_voltage_linear ,
2012-01-09 14:10:25 +04:00
. is_enabled = s5m8767_reg_is_enabled ,
. enable = s5m8767_reg_enable ,
. disable = s5m8767_reg_disable ,
. get_voltage_sel = s5m8767_get_voltage_sel ,
2012-06-13 12:28:18 +04:00
. set_voltage_sel = s5m8767_set_voltage_sel ,
2012-01-09 14:10:25 +04:00
. set_voltage_time_sel = s5m8767_set_voltage_time_sel ,
} ;
2012-06-18 10:25:27 +04:00
static struct regulator_ops s5m8767_buck78_ops = {
. is_enabled = s5m8767_reg_is_enabled ,
. enable = s5m8767_reg_enable ,
. disable = s5m8767_reg_disable ,
} ;
2012-04-10 10:08:19 +04:00
# define s5m8767_regulator_desc(_name) { \
. name = # _name , \
. id = S5M8767_ # # _name , \
. ops = & s5m8767_ops , \
2012-01-09 14:10:25 +04:00
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
}
2012-06-18 10:25:27 +04:00
# define s5m8767_regulator_buck78_desc(_name) { \
. name = # _name , \
. id = S5M8767_ # # _name , \
. ops = & s5m8767_buck78_ops , \
. type = REGULATOR_VOLTAGE , \
. owner = THIS_MODULE , \
}
2012-01-09 14:10:25 +04:00
static struct regulator_desc regulators [ ] = {
2012-04-10 10:08:19 +04:00
s5m8767_regulator_desc ( LDO1 ) ,
s5m8767_regulator_desc ( LDO2 ) ,
s5m8767_regulator_desc ( LDO3 ) ,
s5m8767_regulator_desc ( LDO4 ) ,
s5m8767_regulator_desc ( LDO5 ) ,
s5m8767_regulator_desc ( LDO6 ) ,
s5m8767_regulator_desc ( LDO7 ) ,
s5m8767_regulator_desc ( LDO8 ) ,
s5m8767_regulator_desc ( LDO9 ) ,
s5m8767_regulator_desc ( LDO10 ) ,
s5m8767_regulator_desc ( LDO11 ) ,
s5m8767_regulator_desc ( LDO12 ) ,
s5m8767_regulator_desc ( LDO13 ) ,
s5m8767_regulator_desc ( LDO14 ) ,
s5m8767_regulator_desc ( LDO15 ) ,
s5m8767_regulator_desc ( LDO16 ) ,
s5m8767_regulator_desc ( LDO17 ) ,
s5m8767_regulator_desc ( LDO18 ) ,
s5m8767_regulator_desc ( LDO19 ) ,
s5m8767_regulator_desc ( LDO20 ) ,
s5m8767_regulator_desc ( LDO21 ) ,
s5m8767_regulator_desc ( LDO22 ) ,
s5m8767_regulator_desc ( LDO23 ) ,
s5m8767_regulator_desc ( LDO24 ) ,
s5m8767_regulator_desc ( LDO25 ) ,
s5m8767_regulator_desc ( LDO26 ) ,
s5m8767_regulator_desc ( LDO27 ) ,
s5m8767_regulator_desc ( LDO28 ) ,
s5m8767_regulator_desc ( BUCK1 ) ,
s5m8767_regulator_desc ( BUCK2 ) ,
s5m8767_regulator_desc ( BUCK3 ) ,
s5m8767_regulator_desc ( BUCK4 ) ,
s5m8767_regulator_desc ( BUCK5 ) ,
s5m8767_regulator_desc ( BUCK6 ) ,
2012-06-18 10:25:27 +04:00
s5m8767_regulator_buck78_desc ( BUCK7 ) ,
s5m8767_regulator_buck78_desc ( BUCK8 ) ,
2012-04-10 10:08:19 +04:00
s5m8767_regulator_desc ( BUCK9 ) ,
2012-01-09 14:10:25 +04:00
} ;
static __devinit int s5m8767_pmic_probe ( struct platform_device * pdev )
{
struct s5m87xx_dev * iodev = dev_get_drvdata ( pdev - > dev . parent ) ;
struct s5m_platform_data * pdata = dev_get_platdata ( iodev - > dev ) ;
2012-04-04 03:50:22 +04:00
struct regulator_config config = { } ;
2012-01-09 14:10:25 +04:00
struct regulator_dev * * rdev ;
struct s5m8767_info * s5m8767 ;
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
int i , ret , size , buck_init ;
2012-01-09 14:10:25 +04:00
if ( ! pdata ) {
dev_err ( pdev - > dev . parent , " Platform data not supplied \n " ) ;
return - ENODEV ;
}
2012-03-10 04:43:02 +04:00
if ( pdata - > buck2_gpiodvs ) {
if ( pdata - > buck3_gpiodvs | | pdata - > buck4_gpiodvs ) {
dev_err ( & pdev - > dev , " S5M8767 GPIO DVS NOT VALID \n " ) ;
return - EINVAL ;
}
}
if ( pdata - > buck3_gpiodvs ) {
if ( pdata - > buck2_gpiodvs | | pdata - > buck4_gpiodvs ) {
dev_err ( & pdev - > dev , " S5M8767 GPIO DVS NOT VALID \n " ) ;
return - EINVAL ;
}
}
if ( pdata - > buck4_gpiodvs ) {
if ( pdata - > buck2_gpiodvs | | pdata - > buck3_gpiodvs ) {
dev_err ( & pdev - > dev , " S5M8767 GPIO DVS NOT VALID \n " ) ;
return - EINVAL ;
}
}
2012-01-09 14:10:25 +04:00
s5m8767 = devm_kzalloc ( & pdev - > dev , sizeof ( struct s5m8767_info ) ,
GFP_KERNEL ) ;
if ( ! s5m8767 )
return - ENOMEM ;
size = sizeof ( struct regulator_dev * ) * ( S5M8767_REG_MAX - 2 ) ;
s5m8767 - > rdev = devm_kzalloc ( & pdev - > dev , size , GFP_KERNEL ) ;
if ( ! s5m8767 - > rdev )
return - ENOMEM ;
rdev = s5m8767 - > rdev ;
s5m8767 - > dev = & pdev - > dev ;
s5m8767 - > iodev = iodev ;
s5m8767 - > num_regulators = S5M8767_REG_MAX - 2 ;
platform_set_drvdata ( pdev , s5m8767 ) ;
s5m8767 - > buck_gpioindex = pdata - > buck_default_idx ;
s5m8767 - > buck2_gpiodvs = pdata - > buck2_gpiodvs ;
s5m8767 - > buck3_gpiodvs = pdata - > buck3_gpiodvs ;
s5m8767 - > buck4_gpiodvs = pdata - > buck4_gpiodvs ;
s5m8767 - > buck_gpios [ 0 ] = pdata - > buck_gpios [ 0 ] ;
s5m8767 - > buck_gpios [ 1 ] = pdata - > buck_gpios [ 1 ] ;
s5m8767 - > buck_gpios [ 2 ] = pdata - > buck_gpios [ 2 ] ;
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
s5m8767 - > buck_ds [ 0 ] = pdata - > buck_ds [ 0 ] ;
s5m8767 - > buck_ds [ 1 ] = pdata - > buck_ds [ 1 ] ;
s5m8767 - > buck_ds [ 2 ] = pdata - > buck_ds [ 2 ] ;
2012-01-09 14:10:25 +04:00
s5m8767 - > ramp_delay = pdata - > buck_ramp_delay ;
s5m8767 - > buck2_ramp = pdata - > buck2_ramp_enable ;
s5m8767 - > buck3_ramp = pdata - > buck3_ramp_enable ;
s5m8767 - > buck4_ramp = pdata - > buck4_ramp_enable ;
2012-03-09 12:55:34 +04:00
s5m8767 - > opmode = pdata - > opmode ;
2012-01-09 14:10:25 +04:00
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
buck_init = s5m8767_convert_voltage_to_sel ( & buck_voltage_val2 ,
pdata - > buck2_init ,
pdata - > buck2_init +
buck_voltage_val2 . step ) ;
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK2DVS2 , buck_init ) ;
buck_init = s5m8767_convert_voltage_to_sel ( & buck_voltage_val2 ,
pdata - > buck3_init ,
pdata - > buck3_init +
buck_voltage_val2 . step ) ;
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK3DVS2 , buck_init ) ;
buck_init = s5m8767_convert_voltage_to_sel ( & buck_voltage_val2 ,
pdata - > buck4_init ,
pdata - > buck4_init +
buck_voltage_val2 . step ) ;
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK4DVS2 , buck_init ) ;
2012-01-09 14:10:25 +04:00
for ( i = 0 ; i < 8 ; i + + ) {
if ( s5m8767 - > buck2_gpiodvs ) {
s5m8767 - > buck2_vol [ i ] =
2012-03-09 07:31:08 +04:00
s5m8767_convert_voltage_to_sel (
2012-01-09 14:10:25 +04:00
& buck_voltage_val2 ,
pdata - > buck2_voltage [ i ] ,
pdata - > buck2_voltage [ i ] +
buck_voltage_val2 . step ) ;
}
if ( s5m8767 - > buck3_gpiodvs ) {
s5m8767 - > buck3_vol [ i ] =
2012-03-09 07:31:08 +04:00
s5m8767_convert_voltage_to_sel (
2012-01-09 14:10:25 +04:00
& buck_voltage_val2 ,
pdata - > buck3_voltage [ i ] ,
pdata - > buck3_voltage [ i ] +
buck_voltage_val2 . step ) ;
}
if ( s5m8767 - > buck4_gpiodvs ) {
s5m8767 - > buck4_vol [ i ] =
2012-03-09 07:31:08 +04:00
s5m8767_convert_voltage_to_sel (
2012-01-09 14:10:25 +04:00
& buck_voltage_val2 ,
pdata - > buck4_voltage [ i ] ,
pdata - > buck4_voltage [ i ] +
buck_voltage_val2 . step ) ;
}
}
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
if ( gpio_is_valid ( pdata - > buck_gpios [ 0 ] ) & &
gpio_is_valid ( pdata - > buck_gpios [ 1 ] ) & &
gpio_is_valid ( pdata - > buck_gpios [ 2 ] ) ) {
ret = gpio_request ( pdata - > buck_gpios [ 0 ] , " S5M8767 SET1 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request "
" for SET1 \n " ) ;
ret = gpio_request ( pdata - > buck_gpios [ 1 ] , " S5M8767 SET2 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request "
" for SET2 \n " ) ;
ret = gpio_request ( pdata - > buck_gpios [ 2 ] , " S5M8767 SET3 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request "
" for SET3 \n " ) ;
/* SET1 GPIO */
gpio_direction_output ( pdata - > buck_gpios [ 0 ] ,
( s5m8767 - > buck_gpioindex > > 2 ) & 0x1 ) ;
/* SET2 GPIO */
gpio_direction_output ( pdata - > buck_gpios [ 1 ] ,
( s5m8767 - > buck_gpioindex > > 1 ) & 0x1 ) ;
/* SET3 GPIO */
gpio_direction_output ( pdata - > buck_gpios [ 2 ] ,
( s5m8767 - > buck_gpioindex > > 0 ) & 0x1 ) ;
ret = 0 ;
} else {
dev_err ( & pdev - > dev , " GPIO NOT VALID \n " ) ;
ret = - EINVAL ;
return ret ;
2012-01-09 14:10:25 +04:00
}
regulator: s5m8767a: Support AP watchdog reset operation
The S5M8767A can't know status of ap reset.
So, After AP watchdog reset, AP can't boot normally.
Problem can be happened like below condition.
- AP Bootable lowest voltage(vdd_arm): 0.9v
- AP DVFS voltage table: 0.8v, 0.9v, 1.0v
- During AP works on lowest voltage(0.8V), watchdog reset is asserted
- AP can't boot, because vdd arm is still 0.8v
Solution
- Basic concept:
After ap watchdog reset, GPIO configuration is changed by default value
- S5M8767A has function of voltage control with gpio (8 levels with 3 gpios)
- Set bootable voltage on level 0 -> can work with default gpio configuration
- In the probing, Change voltage control level from level 0 to level 1
- Execute normal dvfs operation on level 1
- After watchdog reset, ap gpio is set by default value
- PMIC operation mode is changed by ap reset (level1 -> level0)
- Regardless of previous vdd_arm voltage, AP always can be booted.
Signed-off-by: Sangbeom Kim <sbkim73@samsung.com>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-18 04:49:20 +04:00
ret = gpio_request ( pdata - > buck_ds [ 0 ] , " S5M8767 DS2 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request for DS2 \n " ) ;
ret = gpio_request ( pdata - > buck_ds [ 1 ] , " S5M8767 DS3 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request for DS3 \n " ) ;
ret = gpio_request ( pdata - > buck_ds [ 2 ] , " S5M8767 DS4 " ) ;
if ( ret = = - EBUSY )
dev_warn ( & pdev - > dev , " Duplicated gpio request for DS4 \n " ) ;
/* DS2 GPIO */
gpio_direction_output ( pdata - > buck_ds [ 0 ] , 0x0 ) ;
/* DS3 GPIO */
gpio_direction_output ( pdata - > buck_ds [ 1 ] , 0x0 ) ;
/* DS4 GPIO */
gpio_direction_output ( pdata - > buck_ds [ 2 ] , 0x0 ) ;
if ( pdata - > buck2_gpiodvs | | pdata - > buck3_gpiodvs | |
pdata - > buck4_gpiodvs ) {
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_BUCK2CTRL ,
( pdata - > buck2_gpiodvs ) ? ( 1 < < 1 ) : ( 0 < < 1 ) ,
1 < < 1 ) ;
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_BUCK3CTRL ,
( pdata - > buck3_gpiodvs ) ? ( 1 < < 1 ) : ( 0 < < 1 ) ,
1 < < 1 ) ;
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_BUCK4CTRL ,
( pdata - > buck4_gpiodvs ) ? ( 1 < < 1 ) : ( 0 < < 1 ) ,
1 < < 1 ) ;
}
2012-01-09 14:10:25 +04:00
/* Initialize GPIO DVS registers */
for ( i = 0 ; i < 8 ; i + + ) {
if ( s5m8767 - > buck2_gpiodvs ) {
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK2DVS1 + i ,
s5m8767 - > buck2_vol [ i ] ) ;
}
if ( s5m8767 - > buck3_gpiodvs ) {
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK3DVS1 + i ,
s5m8767 - > buck3_vol [ i ] ) ;
}
if ( s5m8767 - > buck4_gpiodvs ) {
s5m_reg_write ( s5m8767 - > iodev , S5M8767_REG_BUCK4DVS1 + i ,
s5m8767 - > buck4_vol [ i ] ) ;
}
}
if ( s5m8767 - > buck2_ramp )
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP , 0x08 , 0x08 ) ;
if ( s5m8767 - > buck3_ramp )
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP , 0x04 , 0x04 ) ;
if ( s5m8767 - > buck4_ramp )
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP , 0x02 , 0x02 ) ;
if ( s5m8767 - > buck2_ramp | | s5m8767 - > buck3_ramp
| | s5m8767 - > buck4_ramp ) {
switch ( s5m8767 - > ramp_delay ) {
2012-06-13 12:27:54 +04:00
case 5 :
2012-01-09 14:10:25 +04:00
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
2012-06-13 12:27:54 +04:00
0x40 , 0xf0 ) ;
break ;
case 10 :
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
0x90 , 0xf0 ) ;
2012-01-12 10:57:09 +04:00
break ;
2012-01-09 14:10:25 +04:00
case 25 :
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
0xd0 , 0xf0 ) ;
2012-01-12 10:57:09 +04:00
break ;
2012-01-09 14:10:25 +04:00
case 50 :
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
0xe0 , 0xf0 ) ;
2012-01-12 10:57:09 +04:00
break ;
2012-01-09 14:10:25 +04:00
case 100 :
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
0xf0 , 0xf0 ) ;
2012-01-12 10:57:09 +04:00
break ;
2012-01-09 14:10:25 +04:00
default :
s5m_reg_update ( s5m8767 - > iodev , S5M8767_REG_DVSRAMP ,
0x90 , 0xf0 ) ;
}
}
for ( i = 0 ; i < pdata - > num_regulators ; i + + ) {
const struct s5m_voltage_desc * desc ;
int id = pdata - > regulators [ i ] . id ;
desc = reg_voltage_map [ id ] ;
2012-06-18 10:25:27 +04:00
if ( desc ) {
2012-01-09 14:10:25 +04:00
regulators [ id ] . n_voltages =
( desc - > max - desc - > min ) / desc - > step + 1 ;
2012-06-18 10:25:27 +04:00
regulators [ id ] . min_uV = desc - > min ;
regulators [ id ] . uV_step = desc - > step ;
}
2012-01-09 14:10:25 +04:00
2012-04-04 03:50:22 +04:00
config . dev = s5m8767 - > dev ;
config . init_data = pdata - > regulators [ i ] . initdata ;
config . driver_data = s5m8767 ;
rdev [ i ] = regulator_register ( & regulators [ id ] , & config ) ;
2012-01-09 14:10:25 +04:00
if ( IS_ERR ( rdev [ i ] ) ) {
ret = PTR_ERR ( rdev [ i ] ) ;
dev_err ( s5m8767 - > dev , " regulator init failed for %d \n " ,
id ) ;
rdev [ i ] = NULL ;
goto err ;
}
}
return 0 ;
err :
for ( i = 0 ; i < s5m8767 - > num_regulators ; i + + )
if ( rdev [ i ] )
regulator_unregister ( rdev [ i ] ) ;
return ret ;
}
static int __devexit s5m8767_pmic_remove ( struct platform_device * pdev )
{
struct s5m8767_info * s5m8767 = platform_get_drvdata ( pdev ) ;
struct regulator_dev * * rdev = s5m8767 - > rdev ;
int i ;
for ( i = 0 ; i < s5m8767 - > num_regulators ; i + + )
if ( rdev [ i ] )
regulator_unregister ( rdev [ i ] ) ;
return 0 ;
}
static const struct platform_device_id s5m8767_pmic_id [ ] = {
{ " s5m8767-pmic " , 0 } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( platform , s5m8767_pmic_id ) ;
static struct platform_driver s5m8767_pmic_driver = {
. driver = {
. name = " s5m8767-pmic " ,
. owner = THIS_MODULE ,
} ,
. probe = s5m8767_pmic_probe ,
. remove = __devexit_p ( s5m8767_pmic_remove ) ,
. id_table = s5m8767_pmic_id ,
} ;
static int __init s5m8767_pmic_init ( void )
{
return platform_driver_register ( & s5m8767_pmic_driver ) ;
}
subsys_initcall ( s5m8767_pmic_init ) ;
static void __exit s5m8767_pmic_exit ( void )
{
platform_driver_unregister ( & s5m8767_pmic_driver ) ;
}
module_exit ( s5m8767_pmic_exit ) ;
/* Module information */
MODULE_AUTHOR ( " Sangbeom Kim <sbkim73@samsung.com> " ) ;
MODULE_DESCRIPTION ( " SAMSUNG S5M8767 Regulator Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;