2012-06-23 16:07:57 +04:00
/*
* arizona - ldo1 . c - - LDO1 supply for Arizona devices
*
* Copyright 2012 Wolfson Microelectronics PLC .
*
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . 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/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/bitops.h>
# include <linux/err.h>
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
# include <linux/gpio.h>
# include <linux/slab.h>
# include <linux/mfd/arizona/core.h>
# include <linux/mfd/arizona/pdata.h>
# include <linux/mfd/arizona/registers.h>
struct arizona_ldo1 {
struct regulator_dev * regulator ;
struct arizona * arizona ;
struct regulator_consumer_supply supply ;
struct regulator_init_data init_data ;
} ;
2012-11-27 22:48:33 +04:00
static int arizona_ldo1_hc_list_voltage ( struct regulator_dev * rdev ,
unsigned int selector )
{
if ( selector > = rdev - > desc - > n_voltages )
return - EINVAL ;
if ( selector = = rdev - > desc - > n_voltages - 1 )
return 1800000 ;
else
return rdev - > desc - > min_uV + ( rdev - > desc - > uV_step * selector ) ;
}
static int arizona_ldo1_hc_map_voltage ( struct regulator_dev * rdev ,
int min_uV , int max_uV )
{
int sel ;
sel = DIV_ROUND_UP ( min_uV - rdev - > desc - > min_uV , rdev - > desc - > uV_step ) ;
if ( sel > = rdev - > desc - > n_voltages )
sel = rdev - > desc - > n_voltages - 1 ;
return sel ;
}
static int arizona_ldo1_hc_set_voltage_sel ( struct regulator_dev * rdev ,
unsigned sel )
{
struct arizona_ldo1 * ldo = rdev_get_drvdata ( rdev ) ;
struct regmap * regmap = ldo - > arizona - > regmap ;
unsigned int val ;
int ret ;
if ( sel = = rdev - > desc - > n_voltages - 1 )
val = ARIZONA_LDO1_HI_PWR ;
else
val = 0 ;
ret = regmap_update_bits ( regmap , ARIZONA_LDO1_CONTROL_2 ,
ARIZONA_LDO1_HI_PWR , val ) ;
if ( ret ! = 0 )
return ret ;
ret = regmap_update_bits ( regmap , ARIZONA_DYNAMIC_FREQUENCY_SCALING_1 ,
ARIZONA_SUBSYS_MAX_FREQ , val ) ;
if ( ret ! = 0 )
return ret ;
if ( val )
return 0 ;
val = sel < < ARIZONA_LDO1_VSEL_SHIFT ;
return regmap_update_bits ( regmap , ARIZONA_LDO1_CONTROL_1 ,
ARIZONA_LDO1_VSEL_MASK , val ) ;
}
static int arizona_ldo1_hc_get_voltage_sel ( struct regulator_dev * rdev )
{
struct arizona_ldo1 * ldo = rdev_get_drvdata ( rdev ) ;
struct regmap * regmap = ldo - > arizona - > regmap ;
unsigned int val ;
int ret ;
ret = regmap_read ( regmap , ARIZONA_LDO1_CONTROL_2 , & val ) ;
if ( ret ! = 0 )
return ret ;
if ( val & ARIZONA_LDO1_HI_PWR )
return rdev - > desc - > n_voltages - 1 ;
ret = regmap_read ( regmap , ARIZONA_LDO1_CONTROL_1 , & val ) ;
if ( ret ! = 0 )
return ret ;
return ( val & ARIZONA_LDO1_VSEL_MASK ) > > ARIZONA_LDO1_VSEL_SHIFT ;
}
static struct regulator_ops arizona_ldo1_hc_ops = {
. list_voltage = arizona_ldo1_hc_list_voltage ,
. map_voltage = arizona_ldo1_hc_map_voltage ,
. get_voltage_sel = arizona_ldo1_hc_get_voltage_sel ,
. set_voltage_sel = arizona_ldo1_hc_set_voltage_sel ,
. get_bypass = regulator_get_bypass_regmap ,
. set_bypass = regulator_set_bypass_regmap ,
} ;
static const struct regulator_desc arizona_ldo1_hc = {
. name = " LDO1 " ,
. supply_name = " LDOVDD " ,
. type = REGULATOR_VOLTAGE ,
. ops = & arizona_ldo1_hc_ops ,
. bypass_reg = ARIZONA_LDO1_CONTROL_1 ,
. bypass_mask = ARIZONA_LDO1_BYPASS ,
. min_uV = 900000 ,
. uV_step = 50000 ,
. n_voltages = 8 ,
. enable_time = 500 ,
. owner = THIS_MODULE ,
} ;
2012-06-23 16:07:57 +04:00
static struct regulator_ops arizona_ldo1_ops = {
. list_voltage = regulator_list_voltage_linear ,
. map_voltage = regulator_map_voltage_linear ,
. get_voltage_sel = regulator_get_voltage_sel_regmap ,
. set_voltage_sel = regulator_set_voltage_sel_regmap ,
2012-08-28 03:04:35 +04:00
. get_bypass = regulator_get_bypass_regmap ,
. set_bypass = regulator_set_bypass_regmap ,
2012-06-23 16:07:57 +04:00
} ;
static const struct regulator_desc arizona_ldo1 = {
. name = " LDO1 " ,
. supply_name = " LDOVDD " ,
. type = REGULATOR_VOLTAGE ,
. ops = & arizona_ldo1_ops ,
. vsel_reg = ARIZONA_LDO1_CONTROL_1 ,
. vsel_mask = ARIZONA_LDO1_VSEL_MASK ,
2012-08-28 03:04:35 +04:00
. bypass_reg = ARIZONA_LDO1_CONTROL_1 ,
. bypass_mask = ARIZONA_LDO1_BYPASS ,
2012-06-23 16:07:57 +04:00
. min_uV = 900000 ,
. uV_step = 50000 ,
2012-11-27 22:25:13 +04:00
. n_voltages = 7 ,
2012-11-27 21:02:29 +04:00
. enable_time = 500 ,
2012-06-23 16:07:57 +04:00
. owner = THIS_MODULE ,
} ;
2012-11-27 21:51:46 +04:00
static const struct regulator_init_data arizona_ldo1_dvfs = {
. constraints = {
. min_uV = 1200000 ,
. max_uV = 1800000 ,
. valid_ops_mask = REGULATOR_CHANGE_STATUS |
REGULATOR_CHANGE_VOLTAGE ,
} ,
. num_consumer_supplies = 1 ,
} ;
2012-06-23 16:07:57 +04:00
static const struct regulator_init_data arizona_ldo1_default = {
2012-06-27 17:32:06 +04:00
. constraints = {
. valid_ops_mask = REGULATOR_CHANGE_STATUS ,
} ,
2012-06-23 16:07:57 +04:00
. num_consumer_supplies = 1 ,
} ;
2012-11-19 22:22:22 +04:00
static int arizona_ldo1_probe ( struct platform_device * pdev )
2012-06-23 16:07:57 +04:00
{
struct arizona * arizona = dev_get_drvdata ( pdev - > dev . parent ) ;
2012-11-27 22:48:33 +04:00
const struct regulator_desc * desc ;
2012-06-23 16:07:57 +04:00
struct regulator_config config = { } ;
struct arizona_ldo1 * ldo1 ;
int ret ;
ldo1 = devm_kzalloc ( & pdev - > dev , sizeof ( * ldo1 ) , GFP_KERNEL ) ;
if ( ldo1 = = NULL ) {
dev_err ( & pdev - > dev , " Unable to allocate private data \n " ) ;
return - ENOMEM ;
}
ldo1 - > arizona = arizona ;
/*
* Since the chip usually supplies itself we provide some
* default init_data for it . This will be overridden with
* platform data if provided .
*/
2012-11-27 21:51:46 +04:00
switch ( arizona - > type ) {
case WM5102 :
2012-11-27 22:48:33 +04:00
desc = & arizona_ldo1_hc ;
2012-11-27 21:51:46 +04:00
ldo1 - > init_data = arizona_ldo1_dvfs ;
break ;
default :
2012-11-27 22:48:33 +04:00
desc = & arizona_ldo1 ;
2012-11-27 21:51:46 +04:00
ldo1 - > init_data = arizona_ldo1_default ;
break ;
}
2012-06-23 16:07:57 +04:00
ldo1 - > init_data . consumer_supplies = & ldo1 - > supply ;
ldo1 - > supply . supply = " DCVDD " ;
ldo1 - > supply . dev_name = dev_name ( arizona - > dev ) ;
config . dev = arizona - > dev ;
config . driver_data = ldo1 ;
config . regmap = arizona - > regmap ;
2012-06-27 17:32:06 +04:00
config . ena_gpio = arizona - > pdata . ldoena ;
2012-06-23 16:07:57 +04:00
if ( arizona - > pdata . ldo1 )
config . init_data = arizona - > pdata . ldo1 ;
else
config . init_data = & ldo1 - > init_data ;
2012-11-27 22:48:33 +04:00
ldo1 - > regulator = regulator_register ( desc , & config ) ;
2012-06-23 16:07:57 +04:00
if ( IS_ERR ( ldo1 - > regulator ) ) {
ret = PTR_ERR ( ldo1 - > regulator ) ;
dev_err ( arizona - > dev , " Failed to register LDO1 supply: %d \n " ,
ret ) ;
return ret ;
}
platform_set_drvdata ( pdev , ldo1 ) ;
return 0 ;
}
2012-11-19 22:26:10 +04:00
static int arizona_ldo1_remove ( struct platform_device * pdev )
2012-06-23 16:07:57 +04:00
{
struct arizona_ldo1 * ldo1 = platform_get_drvdata ( pdev ) ;
regulator_unregister ( ldo1 - > regulator ) ;
return 0 ;
}
static struct platform_driver arizona_ldo1_driver = {
. probe = arizona_ldo1_probe ,
2012-11-19 22:20:42 +04:00
. remove = arizona_ldo1_remove ,
2012-06-23 16:07:57 +04:00
. driver = {
. name = " arizona-ldo1 " ,
. owner = THIS_MODULE ,
} ,
} ;
module_platform_driver ( arizona_ldo1_driver ) ;
/* Module information */
MODULE_AUTHOR ( " Mark Brown <broonie@opensource.wolfsonmicro.com> " ) ;
MODULE_DESCRIPTION ( " Arizona LDO1 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform:arizona-ldo1 " ) ;