2009-10-08 10:03:57 +04:00
/*
* Regulators driver for Marvell 88 PM8607
*
* Copyright ( C ) 2009 Marvell International Ltd .
* Haojian Zhuang < haojian . zhuang @ marvell . 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 .
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/err.h>
2010-01-08 14:01:24 +03:00
# include <linux/i2c.h>
2009-10-08 10:03:57 +04:00
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
2010-01-08 14:01:24 +03:00
# include <linux/mfd/88pm860x.h>
2011-07-18 00:28:23 +04:00
# include <linux/module.h>
2009-10-08 10:03:57 +04:00
struct pm8607_regulator_info {
struct regulator_desc desc ;
2010-01-08 14:01:24 +03:00
struct pm860x_chip * chip ;
2009-10-08 10:03:57 +04:00
struct regulator_dev * regulator ;
2010-01-08 14:01:24 +03:00
struct i2c_client * i2c ;
2009-10-08 10:03:57 +04:00
2010-05-04 17:54:51 +04:00
unsigned int * vol_table ;
unsigned int * vol_suspend ;
2009-10-08 10:03:57 +04:00
int update_reg ;
int update_bit ;
int slope_double ;
} ;
2010-05-04 17:54:51 +04:00
static const unsigned int BUCK1_table [ ] = {
725000 , 750000 , 775000 , 800000 , 825000 , 850000 , 875000 , 900000 ,
925000 , 950000 , 975000 , 1000000 , 1025000 , 1050000 , 1075000 , 1100000 ,
1125000 , 1150000 , 1175000 , 1200000 , 1225000 , 1250000 , 1275000 , 1300000 ,
1325000 , 1350000 , 1375000 , 1400000 , 1425000 , 1450000 , 1475000 , 1500000 ,
0 , 25000 , 50000 , 75000 , 100000 , 125000 , 150000 , 175000 ,
200000 , 225000 , 250000 , 275000 , 300000 , 325000 , 350000 , 375000 ,
400000 , 425000 , 450000 , 475000 , 500000 , 525000 , 550000 , 575000 ,
600000 , 625000 , 650000 , 675000 , 700000 , 725000 , 750000 , 775000 ,
} ;
2009-10-08 10:03:57 +04:00
2010-05-04 17:54:51 +04:00
static const unsigned int BUCK1_suspend_table [ ] = {
0 , 25000 , 50000 , 75000 , 100000 , 125000 , 150000 , 175000 ,
200000 , 225000 , 250000 , 275000 , 300000 , 325000 , 350000 , 375000 ,
400000 , 425000 , 450000 , 475000 , 500000 , 525000 , 550000 , 575000 ,
600000 , 625000 , 650000 , 675000 , 700000 , 725000 , 750000 , 775000 ,
800000 , 825000 , 850000 , 875000 , 900000 , 925000 , 950000 , 975000 ,
1000000 , 1025000 , 1050000 , 1075000 , 1100000 , 1125000 , 1150000 , 1175000 ,
1200000 , 1225000 , 1250000 , 1275000 , 1300000 , 1325000 , 1350000 , 1375000 ,
1400000 , 1425000 , 1450000 , 1475000 , 1500000 , 1500000 , 1500000 , 1500000 ,
} ;
static const unsigned int BUCK2_table [ ] = {
0 , 50000 , 100000 , 150000 , 200000 , 250000 , 300000 , 350000 ,
400000 , 450000 , 500000 , 550000 , 600000 , 650000 , 700000 , 750000 ,
800000 , 850000 , 900000 , 950000 , 1000000 , 1050000 , 1100000 , 1150000 ,
1200000 , 1250000 , 1300000 , 1350000 , 1400000 , 1450000 , 1500000 , 1550000 ,
1600000 , 1650000 , 1700000 , 1750000 , 1800000 , 1850000 , 1900000 , 1950000 ,
2000000 , 2050000 , 2100000 , 2150000 , 2200000 , 2250000 , 2300000 , 2350000 ,
2400000 , 2450000 , 2500000 , 2550000 , 2600000 , 2650000 , 2700000 , 2750000 ,
2800000 , 2850000 , 2900000 , 2950000 , 3000000 , 3000000 , 3000000 , 3000000 ,
} ;
static const unsigned int BUCK2_suspend_table [ ] = {
0 , 50000 , 100000 , 150000 , 200000 , 250000 , 300000 , 350000 ,
400000 , 450000 , 500000 , 550000 , 600000 , 650000 , 700000 , 750000 ,
800000 , 850000 , 900000 , 950000 , 1000000 , 1050000 , 1100000 , 1150000 ,
1200000 , 1250000 , 1300000 , 1350000 , 1400000 , 1450000 , 1500000 , 1550000 ,
1600000 , 1650000 , 1700000 , 1750000 , 1800000 , 1850000 , 1900000 , 1950000 ,
2000000 , 2050000 , 2100000 , 2150000 , 2200000 , 2250000 , 2300000 , 2350000 ,
2400000 , 2450000 , 2500000 , 2550000 , 2600000 , 2650000 , 2700000 , 2750000 ,
2800000 , 2850000 , 2900000 , 2950000 , 3000000 , 3000000 , 3000000 , 3000000 ,
} ;
static const unsigned int BUCK3_table [ ] = {
0 , 25000 , 50000 , 75000 , 100000 , 125000 , 150000 , 175000 ,
200000 , 225000 , 250000 , 275000 , 300000 , 325000 , 350000 , 375000 ,
400000 , 425000 , 450000 , 475000 , 500000 , 525000 , 550000 , 575000 ,
600000 , 625000 , 650000 , 675000 , 700000 , 725000 , 750000 , 775000 ,
800000 , 825000 , 850000 , 875000 , 900000 , 925000 , 950000 , 975000 ,
1000000 , 1025000 , 1050000 , 1075000 , 1100000 , 1125000 , 1150000 , 1175000 ,
1200000 , 1225000 , 1250000 , 1275000 , 1300000 , 1325000 , 1350000 , 1375000 ,
1400000 , 1425000 , 1450000 , 1475000 , 1500000 , 1500000 , 1500000 , 1500000 ,
} ;
static const unsigned int BUCK3_suspend_table [ ] = {
0 , 25000 , 50000 , 75000 , 100000 , 125000 , 150000 , 175000 ,
200000 , 225000 , 250000 , 275000 , 300000 , 325000 , 350000 , 375000 ,
400000 , 425000 , 450000 , 475000 , 500000 , 525000 , 550000 , 575000 ,
600000 , 625000 , 650000 , 675000 , 700000 , 725000 , 750000 , 775000 ,
800000 , 825000 , 850000 , 875000 , 900000 , 925000 , 950000 , 975000 ,
1000000 , 1025000 , 1050000 , 1075000 , 1100000 , 1125000 , 1150000 , 1175000 ,
1200000 , 1225000 , 1250000 , 1275000 , 1300000 , 1325000 , 1350000 , 1375000 ,
1400000 , 1425000 , 1450000 , 1475000 , 1500000 , 1500000 , 1500000 , 1500000 ,
} ;
static const unsigned int LDO1_table [ ] = {
1800000 , 1200000 , 2800000 , 0 ,
} ;
static const unsigned int LDO1_suspend_table [ ] = {
1800000 , 1200000 , 0 , 0 ,
} ;
static const unsigned int LDO2_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 3300000 ,
} ;
static const unsigned int LDO2_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO3_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 3300000 ,
} ;
static const unsigned int LDO3_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO4_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2900000 , 3300000 ,
} ;
static const unsigned int LDO4_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2900000 , 2900000 ,
} ;
static const unsigned int LDO5_table [ ] = {
2900000 , 3000000 , 3100000 , 3300000 ,
} ;
static const unsigned int LDO5_suspend_table [ ] = {
2900000 , 0 , 0 , 0 ,
} ;
static const unsigned int LDO6_table [ ] = {
1800000 , 1850000 , 2600000 , 2650000 , 2700000 , 2750000 , 2800000 , 3300000 ,
} ;
static const unsigned int LDO6_suspend_table [ ] = {
1800000 , 1850000 , 2600000 , 2650000 , 2700000 , 2750000 , 2800000 , 2900000 ,
} ;
static const unsigned int LDO7_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO7_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO8_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO8_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO9_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 3300000 ,
} ;
static const unsigned int LDO9_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
} ;
static const unsigned int LDO10_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 3300000 ,
1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 ,
} ;
static const unsigned int LDO10_suspend_table [ ] = {
1800000 , 1850000 , 1900000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 ,
1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 ,
} ;
static const unsigned int LDO12_table [ ] = {
1800000 , 1900000 , 2700000 , 2800000 , 2900000 , 3000000 , 3100000 , 3300000 ,
1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 ,
} ;
static const unsigned int LDO12_suspend_table [ ] = {
1800000 , 1900000 , 2700000 , 2800000 , 2900000 , 2900000 , 2900000 , 2900000 ,
1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 , 1200000 ,
} ;
static const unsigned int LDO13_table [ ] = {
2012-02-23 15:52:08 +04:00
1200000 , 1300000 , 1800000 , 2000000 , 2500000 , 2800000 , 3000000 , 0 ,
2010-05-04 17:54:51 +04:00
} ;
static const unsigned int LDO13_suspend_table [ ] = {
0 ,
} ;
static const unsigned int LDO14_table [ ] = {
1800000 , 1850000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 , 3300000 ,
} ;
static const unsigned int LDO14_suspend_table [ ] = {
1800000 , 1850000 , 2700000 , 2750000 , 2800000 , 2850000 , 2900000 , 2900000 ,
} ;
2009-10-08 10:03:57 +04:00
static int pm8607_list_voltage ( struct regulator_dev * rdev , unsigned index )
{
struct pm8607_regulator_info * info = rdev_get_drvdata ( rdev ) ;
int ret = - EINVAL ;
2012-03-28 10:59:23 +04:00
if ( info - > vol_table & & ( index < rdev - > desc - > n_voltages ) ) {
2010-05-04 17:54:51 +04:00
ret = info - > vol_table [ index ] ;
2009-10-08 10:03:57 +04:00
if ( info - > slope_double )
ret < < = 1 ;
}
return ret ;
}
2012-03-28 11:00:36 +04:00
static int pm8607_set_voltage_sel ( struct regulator_dev * rdev , unsigned selector )
2009-10-08 10:03:57 +04:00
{
struct pm8607_regulator_info * info = rdev_get_drvdata ( rdev ) ;
2012-04-24 05:52:32 +04:00
uint8_t val ;
2009-10-08 10:03:57 +04:00
int ret ;
2012-04-24 05:52:32 +04:00
val = ( uint8_t ) ( selector < < ( ffs ( rdev - > desc - > vsel_mask ) - 1 ) ) ;
2009-10-08 10:03:57 +04:00
2012-04-24 05:52:32 +04:00
ret = pm860x_set_bits ( info - > i2c , rdev - > desc - > vsel_reg ,
rdev - > desc - > vsel_mask , val ) ;
2009-10-08 10:03:57 +04:00
if ( ret )
return ret ;
switch ( info - > desc . id ) {
case PM8607_ID_BUCK1 :
case PM8607_ID_BUCK3 :
2010-01-08 14:01:24 +03:00
ret = pm860x_set_bits ( info - > i2c , info - > update_reg ,
2009-10-08 10:03:57 +04:00
1 < < info - > update_bit ,
1 < < info - > update_bit ) ;
break ;
}
return ret ;
}
static struct regulator_ops pm8607_regulator_ops = {
2012-03-28 10:58:29 +04:00
. list_voltage = pm8607_list_voltage ,
2012-03-28 11:00:36 +04:00
. set_voltage_sel = pm8607_set_voltage_sel ,
2012-04-24 05:52:32 +04:00
. get_voltage_sel = regulator_get_voltage_sel_regmap ,
2012-04-17 07:32:57 +04:00
. enable = regulator_enable_regmap ,
. disable = regulator_disable_regmap ,
. is_enabled = regulator_is_enabled_regmap ,
2009-10-08 10:03:57 +04:00
} ;
2012-03-28 10:59:23 +04:00
# define PM8607_DVC(vreg, ureg, ubit, ereg, ebit) \
2009-10-08 10:03:57 +04:00
{ \
. desc = { \
2010-05-04 17:54:51 +04:00
. name = # vreg , \
2009-10-08 10:03:57 +04:00
. ops = & pm8607_regulator_ops , \
. type = REGULATOR_VOLTAGE , \
2010-05-04 17:54:51 +04:00
. id = PM8607_ID_ # # vreg , \
2009-10-08 10:03:57 +04:00
. owner = THIS_MODULE , \
2012-03-28 10:59:23 +04:00
. n_voltages = ARRAY_SIZE ( vreg # # _table ) , \
2012-04-24 05:52:32 +04:00
. vsel_reg = PM8607_ # # vreg , \
. vsel_mask = ARRAY_SIZE ( vreg # # _table ) - 1 , \
2012-04-17 07:32:57 +04:00
. enable_reg = PM8607_ # # ereg , \
. enable_mask = 1 < < ( ebit ) , \
2009-10-08 10:03:57 +04:00
} , \
. update_reg = PM8607_ # # ureg , \
. update_bit = ( ubit ) , \
. slope_double = ( 0 ) , \
2010-05-04 17:54:51 +04:00
. vol_table = ( unsigned int * ) & vreg # # _table , \
. vol_suspend = ( unsigned int * ) & vreg # # _suspend_table , \
2009-10-08 10:03:57 +04:00
}
2012-03-28 10:59:23 +04:00
# define PM8607_LDO(_id, vreg, shift, ereg, ebit) \
2009-10-08 10:03:57 +04:00
{ \
. desc = { \
. name = " LDO " # _id , \
. ops = & pm8607_regulator_ops , \
. type = REGULATOR_VOLTAGE , \
. id = PM8607_ID_LDO # # _id , \
. owner = THIS_MODULE , \
2012-03-28 10:59:23 +04:00
. n_voltages = ARRAY_SIZE ( LDO # # _id # # _table ) , \
2012-04-24 05:52:32 +04:00
. vsel_reg = PM8607_ # # vreg , \
. vsel_mask = ( ARRAY_SIZE ( LDO # # _id # # _table ) - 1 ) < < ( shift ) , \
2012-04-17 07:32:57 +04:00
. enable_reg = PM8607_ # # ereg , \
. enable_mask = 1 < < ( ebit ) , \
2009-10-08 10:03:57 +04:00
} , \
. slope_double = ( 0 ) , \
2010-05-04 17:54:51 +04:00
. vol_table = ( unsigned int * ) & LDO # # _id # # _table , \
. vol_suspend = ( unsigned int * ) & LDO # # _id # # _suspend_table , \
2009-10-08 10:03:57 +04:00
}
static struct pm8607_regulator_info pm8607_regulator_info [ ] = {
2012-03-28 10:59:23 +04:00
PM8607_DVC ( BUCK1 , GO , 0 , SUPPLIES_EN11 , 0 ) ,
PM8607_DVC ( BUCK2 , GO , 1 , SUPPLIES_EN11 , 1 ) ,
PM8607_DVC ( BUCK3 , GO , 2 , SUPPLIES_EN11 , 2 ) ,
PM8607_LDO ( 1 , LDO1 , 0 , SUPPLIES_EN11 , 3 ) ,
PM8607_LDO ( 2 , LDO2 , 0 , SUPPLIES_EN11 , 4 ) ,
PM8607_LDO ( 3 , LDO3 , 0 , SUPPLIES_EN11 , 5 ) ,
PM8607_LDO ( 4 , LDO4 , 0 , SUPPLIES_EN11 , 6 ) ,
PM8607_LDO ( 5 , LDO5 , 0 , SUPPLIES_EN11 , 7 ) ,
PM8607_LDO ( 6 , LDO6 , 0 , SUPPLIES_EN12 , 0 ) ,
PM8607_LDO ( 7 , LDO7 , 0 , SUPPLIES_EN12 , 1 ) ,
PM8607_LDO ( 8 , LDO8 , 0 , SUPPLIES_EN12 , 2 ) ,
PM8607_LDO ( 9 , LDO9 , 0 , SUPPLIES_EN12 , 3 ) ,
PM8607_LDO ( 10 , LDO10 , 0 , SUPPLIES_EN12 , 4 ) ,
PM8607_LDO ( 12 , LDO12 , 0 , SUPPLIES_EN12 , 5 ) ,
PM8607_LDO ( 13 , VIBRATOR_SET , 1 , VIBRATOR_SET , 0 ) ,
PM8607_LDO ( 14 , LDO14 , 0 , SUPPLIES_EN12 , 6 ) ,
2009-10-08 10:03:57 +04:00
} ;
2011-03-07 18:43:11 +03:00
static int __devinit pm8607_regulator_probe ( struct platform_device * pdev )
2009-10-08 10:03:57 +04:00
{
2011-03-07 18:43:11 +03:00
struct pm860x_chip * chip = dev_get_drvdata ( pdev - > dev . parent ) ;
struct pm8607_regulator_info * info = NULL ;
2011-05-06 13:21:23 +04:00
struct regulator_init_data * pdata = pdev - > dev . platform_data ;
2012-04-04 03:50:22 +04:00
struct regulator_config config = { } ;
2011-05-06 13:21:23 +04:00
struct resource * res ;
2009-10-08 10:03:57 +04:00
int i ;
2011-05-06 13:21:23 +04:00
res = platform_get_resource ( pdev , IORESOURCE_IO , 0 ) ;
if ( res = = NULL ) {
dev_err ( & pdev - > dev , " No I/O resource! \n " ) ;
2011-03-07 18:43:11 +03:00
return - EINVAL ;
2011-05-06 13:21:23 +04:00
}
2009-10-08 10:03:57 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( pm8607_regulator_info ) ; i + + ) {
info = & pm8607_regulator_info [ i ] ;
2011-05-06 13:21:23 +04:00
if ( info - > desc . id = = res - > start )
2011-03-07 18:43:11 +03:00
break ;
2009-10-08 10:03:57 +04:00
}
2011-08-04 12:18:11 +04:00
if ( i = = ARRAY_SIZE ( pm8607_regulator_info ) ) {
2011-05-26 21:16:15 +04:00
dev_err ( & pdev - > dev , " Failed to find regulator %llu \n " ,
( unsigned long long ) res - > start ) ;
2009-10-08 10:03:57 +04:00
return - EINVAL ;
}
2010-01-08 14:01:24 +03:00
info - > i2c = ( chip - > id = = CHIP_PM8607 ) ? chip - > client : chip - > companion ;
2009-10-08 10:03:57 +04:00
info - > chip = chip ;
2011-03-07 18:43:11 +03:00
/* check DVC ramp slope double */
2011-05-06 13:21:23 +04:00
if ( ( i = = PM8607_ID_BUCK3 ) & & info - > chip - > buck3_double )
info - > slope_double = 1 ;
2011-03-07 18:43:11 +03:00
2012-04-04 03:50:22 +04:00
config . dev = & pdev - > dev ;
config . init_data = pdata ;
config . driver_data = info ;
2012-04-17 07:32:57 +04:00
if ( chip - > id = = CHIP_PM8607 )
config . regmap = chip - > regmap ;
else
config . regmap = chip - > regmap_companion ;
2011-05-06 13:21:23 +04:00
/* replace driver_data with info */
2012-04-04 03:50:22 +04:00
info - > regulator = regulator_register ( & info - > desc , & config ) ;
2009-10-08 10:03:57 +04:00
if ( IS_ERR ( info - > regulator ) ) {
dev_err ( & pdev - > dev , " failed to register regulator %s \n " ,
info - > desc . name ) ;
return PTR_ERR ( info - > regulator ) ;
}
platform_set_drvdata ( pdev , info ) ;
return 0 ;
}
static int __devexit pm8607_regulator_remove ( struct platform_device * pdev )
{
struct pm8607_regulator_info * info = platform_get_drvdata ( pdev ) ;
2010-04-29 21:33:50 +04:00
platform_set_drvdata ( pdev , NULL ) ;
2009-10-08 10:03:57 +04:00
regulator_unregister ( info - > regulator ) ;
return 0 ;
}
2010-04-29 21:33:50 +04:00
static struct platform_driver pm8607_regulator_driver = {
. driver = {
. name = " 88pm860x-regulator " ,
. owner = THIS_MODULE ,
} ,
. probe = pm8607_regulator_probe ,
. remove = __devexit_p ( pm8607_regulator_remove ) ,
2009-10-08 10:03:57 +04:00
} ;
static int __init pm8607_regulator_init ( void )
{
2010-04-29 21:33:50 +04:00
return platform_driver_register ( & pm8607_regulator_driver ) ;
2009-10-08 10:03:57 +04:00
}
subsys_initcall ( pm8607_regulator_init ) ;
static void __exit pm8607_regulator_exit ( void )
{
2010-04-29 21:33:50 +04:00
platform_driver_unregister ( & pm8607_regulator_driver ) ;
2009-10-08 10:03:57 +04:00
}
module_exit ( pm8607_regulator_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Haojian Zhuang <haojian.zhuang@marvell.com> " ) ;
MODULE_DESCRIPTION ( " Regulator Driver for Marvell 88PM8607 PMIC " ) ;
MODULE_ALIAS ( " platform:88pm8607-regulator " ) ;