2019-04-17 22:16:29 +08:00
// SPDX-License-Identifier: GPL-2.0+
//
// arizona-ldo1.c -- LDO1 supply for Arizona devices
//
// Copyright 2012 Wolfson Microelectronics PLC.
//
// Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
2012-06-23 13:07:57 +01:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/bitops.h>
# include <linux/err.h>
2014-05-27 17:27:34 +01:00
# include <linux/of.h>
2018-05-14 10:06:24 +02:00
# include <linux/gpio/consumer.h>
2012-06-23 13:07:57 +01:00
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
2014-04-16 10:01:39 +01:00
# include <linux/regulator/of_regulator.h>
2012-06-23 13:07:57 +01:00
# include <linux/slab.h>
2017-04-18 11:43:52 +01:00
# include <linux/regulator/arizona-ldo1.h>
2012-06-23 13:07:57 +01:00
# include <linux/mfd/arizona/core.h>
# include <linux/mfd/arizona/pdata.h>
# include <linux/mfd/arizona/registers.h>
2019-05-21 11:04:38 +01:00
# include <linux/mfd/madera/core.h>
# include <linux/mfd/madera/pdata.h>
# include <linux/mfd/madera/registers.h>
2012-06-23 13:07:57 +01:00
struct arizona_ldo1 {
struct regulator_dev * regulator ;
2017-04-18 11:43:53 +01:00
struct regmap * regmap ;
2012-06-23 13:07:57 +01:00
struct regulator_consumer_supply supply ;
struct regulator_init_data init_data ;
2018-06-19 16:10:00 +01:00
struct gpio_desc * ena_gpiod ;
2012-06-23 13:07:57 +01:00
} ;
2012-11-27 18:48:33 +00:00
static int arizona_ldo1_hc_set_voltage_sel ( struct regulator_dev * rdev ,
unsigned sel )
{
2019-02-25 17:13:49 +08:00
struct regmap * regmap = rdev_get_regmap ( rdev ) ;
2012-11-27 18:48:33 +00:00
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 ;
if ( val )
return 0 ;
2019-02-25 17:13:49 +08:00
return regulator_set_voltage_sel_regmap ( rdev , sel ) ;
2012-11-27 18:48:33 +00:00
}
static int arizona_ldo1_hc_get_voltage_sel ( struct regulator_dev * rdev )
{
2019-02-25 17:13:49 +08:00
struct regmap * regmap = rdev_get_regmap ( rdev ) ;
2012-11-27 18:48:33 +00:00
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 ;
2019-02-25 17:13:49 +08:00
return regulator_get_voltage_sel_regmap ( rdev ) ;
2012-11-27 18:48:33 +00:00
}
2017-01-28 19:08:41 +05:30
static const struct regulator_ops arizona_ldo1_hc_ops = {
2019-01-19 20:36:36 +08:00
. list_voltage = regulator_list_voltage_linear_range ,
. map_voltage = regulator_map_voltage_linear_range ,
2012-11-27 18:48:33 +00:00
. 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 ,
} ;
2020-05-08 18:43:36 +03:00
static const struct linear_range arizona_ldo1_hc_ranges [ ] = {
2019-01-19 20:36:36 +08:00
REGULATOR_LINEAR_RANGE ( 900000 , 0 , 0x6 , 50000 ) ,
REGULATOR_LINEAR_RANGE ( 1800000 , 0x7 , 0x7 , 0 ) ,
} ;
2012-11-27 18:48:33 +00:00
static const struct regulator_desc arizona_ldo1_hc = {
. name = " LDO1 " ,
. supply_name = " LDOVDD " ,
. type = REGULATOR_VOLTAGE ,
. ops = & arizona_ldo1_hc_ops ,
2019-02-25 17:13:49 +08:00
. vsel_reg = ARIZONA_LDO1_CONTROL_1 ,
. vsel_mask = ARIZONA_LDO1_VSEL_MASK ,
2012-11-27 18:48:33 +00:00
. bypass_reg = ARIZONA_LDO1_CONTROL_1 ,
. bypass_mask = ARIZONA_LDO1_BYPASS ,
2019-01-19 20:36:36 +08:00
. linear_ranges = arizona_ldo1_hc_ranges ,
. n_linear_ranges = ARRAY_SIZE ( arizona_ldo1_hc_ranges ) ,
2012-11-27 18:48:33 +00:00
. n_voltages = 8 ,
2013-03-26 12:02:32 +00:00
. enable_time = 1500 ,
2016-10-21 14:47:42 +01:00
. ramp_delay = 24000 ,
2012-11-27 18:48:33 +00:00
. owner = THIS_MODULE ,
} ;
2017-01-28 19:08:41 +05:30
static const struct regulator_ops arizona_ldo1_ops = {
2012-06-23 13:07:57 +01:00
. 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 ,
} ;
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 ,
. min_uV = 900000 ,
2014-03-18 10:49:17 +00:00
. uV_step = 25000 ,
. n_voltages = 13 ,
2012-11-27 17:02:29 +00:00
. enable_time = 500 ,
2016-10-21 14:47:42 +01:00
. ramp_delay = 24000 ,
2012-06-23 13:07:57 +01:00
. owner = THIS_MODULE ,
} ;
2012-11-27 17:51:46 +00: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 13:07:57 +01:00
static const struct regulator_init_data arizona_ldo1_default = {
2012-06-27 14:32:06 +01:00
. constraints = {
. valid_ops_mask = REGULATOR_CHANGE_STATUS ,
} ,
2012-06-23 13:07:57 +01:00
. num_consumer_supplies = 1 ,
} ;
2015-05-11 13:58:07 +01:00
static const struct regulator_init_data arizona_ldo1_wm5110 = {
. constraints = {
. min_uV = 1175000 ,
. max_uV = 1200000 ,
. valid_ops_mask = REGULATOR_CHANGE_STATUS |
REGULATOR_CHANGE_VOLTAGE ,
} ,
. num_consumer_supplies = 1 ,
} ;
2019-05-21 11:04:38 +01:00
static const struct regulator_desc madera_ldo1 = {
. name = " LDO1 " ,
. supply_name = " LDOVDD " ,
. type = REGULATOR_VOLTAGE ,
. ops = & arizona_ldo1_ops ,
. vsel_reg = MADERA_LDO1_CONTROL_1 ,
. vsel_mask = MADERA_LDO1_VSEL_MASK ,
. min_uV = 900000 ,
. uV_step = 25000 ,
. n_voltages = 13 ,
. enable_time = 3000 ,
. owner = THIS_MODULE ,
} ;
static const struct regulator_init_data madera_ldo1_default = {
. constraints = {
. min_uV = 1200000 ,
. max_uV = 1200000 ,
. valid_ops_mask = REGULATOR_CHANGE_STATUS ,
} ,
. num_consumer_supplies = 1 ,
} ;
2017-04-18 11:43:52 +01:00
static int arizona_ldo1_of_get_pdata ( struct arizona_ldo1_pdata * pdata ,
2014-11-10 14:43:53 +01:00
struct regulator_config * config ,
2017-04-18 11:43:52 +01:00
const struct regulator_desc * desc ,
bool * external_dcvdd )
2014-04-16 10:01:38 +01:00
{
2014-04-16 10:01:39 +01:00
struct arizona_ldo1 * ldo1 = config - > driver_data ;
2017-04-18 11:43:52 +01:00
struct device_node * np = config - > dev - > of_node ;
2014-04-16 10:01:39 +01:00
struct device_node * init_node , * dcvdd_node ;
struct regulator_init_data * init_data ;
2014-04-16 10:01:38 +01:00
2015-10-15 10:37:12 +01:00
init_node = of_get_child_by_name ( np , " ldo1 " ) ;
dcvdd_node = of_parse_phandle ( np , " DCVDD-supply " , 0 ) ;
2014-04-16 10:01:39 +01:00
if ( init_node ) {
config - > of_node = init_node ;
2017-04-18 11:43:52 +01:00
init_data = of_get_regulator_init_data ( config - > dev , init_node ,
desc ) ;
2014-04-16 10:01:39 +01:00
if ( init_data ) {
init_data - > consumer_supplies = & ldo1 - > supply ;
init_data - > num_consumer_supplies = 1 ;
if ( dcvdd_node & & dcvdd_node ! = init_node )
2017-04-18 11:43:52 +01:00
* external_dcvdd = true ;
2014-04-16 10:01:39 +01:00
2017-04-18 11:43:52 +01:00
pdata - > init_data = init_data ;
2014-04-16 10:01:39 +01:00
}
} else if ( dcvdd_node ) {
2017-04-18 11:43:52 +01:00
* external_dcvdd = true ;
2014-04-16 10:01:39 +01:00
}
of_node_put ( dcvdd_node ) ;
2014-04-16 10:01:38 +01:00
return 0 ;
}
2017-04-18 11:43:54 +01:00
static int arizona_ldo1_common_init ( struct platform_device * pdev ,
struct arizona_ldo1 * ldo1 ,
const struct regulator_desc * desc ,
struct arizona_ldo1_pdata * pdata ,
bool * external_dcvdd )
2012-06-23 13:07:57 +01:00
{
2017-04-18 11:43:54 +01:00
struct device * parent_dev = pdev - > dev . parent ;
2012-06-23 13:07:57 +01:00
struct regulator_config config = { } ;
int ret ;
2017-04-18 11:43:54 +01:00
* external_dcvdd = false ;
2012-11-27 17:51:46 +00:00
2012-06-23 13:07:57 +01:00
ldo1 - > supply . supply = " DCVDD " ;
2017-04-18 11:43:54 +01:00
ldo1 - > init_data . consumer_supplies = & ldo1 - > supply ;
ldo1 - > supply . dev_name = dev_name ( parent_dev ) ;
2012-06-23 13:07:57 +01:00
2017-04-18 11:43:54 +01:00
config . dev = parent_dev ;
2012-06-23 13:07:57 +01:00
config . driver_data = ldo1 ;
2017-04-18 11:43:54 +01:00
config . regmap = ldo1 - > regmap ;
2014-04-16 10:01:38 +01:00
if ( IS_ENABLED ( CONFIG_OF ) ) {
2017-04-18 11:43:54 +01:00
if ( ! dev_get_platdata ( parent_dev ) ) {
ret = arizona_ldo1_of_get_pdata ( pdata ,
2017-04-18 11:43:52 +01:00
& config , desc ,
2017-04-18 11:43:54 +01:00
external_dcvdd ) ;
2014-04-16 10:01:38 +01:00
if ( ret < 0 )
return ret ;
}
}
2018-06-19 16:10:00 +01:00
/* We assume that high output = regulator off
* Don ' t use devm , since we need to get against the parent device
* so clean up would happen at the wrong time
*/
config . ena_gpiod = gpiod_get_optional ( parent_dev , " wlf,ldoena " ,
2018-10-15 11:02:40 +02:00
GPIOD_OUT_LOW | GPIOD_FLAGS_BIT_NONEXCLUSIVE ) ;
2018-05-14 10:06:24 +02:00
if ( IS_ERR ( config . ena_gpiod ) )
return PTR_ERR ( config . ena_gpiod ) ;
2012-06-23 13:07:57 +01:00
2018-06-19 16:10:00 +01:00
ldo1 - > ena_gpiod = config . ena_gpiod ;
2017-04-18 11:43:54 +01:00
if ( pdata - > init_data )
config . init_data = pdata - > init_data ;
2012-06-23 13:07:57 +01:00
else
config . init_data = & ldo1 - > init_data ;
2014-04-16 10:01:38 +01:00
/*
* LDO1 can only be used to supply DCVDD so if it has no
* consumers then DCVDD is supplied externally .
*/
if ( config . init_data - > num_consumer_supplies = = 0 )
2017-04-18 11:43:54 +01:00
* external_dcvdd = true ;
2014-04-16 10:01:38 +01:00
2013-08-31 11:56:47 +01:00
ldo1 - > regulator = devm_regulator_register ( & pdev - > dev , desc , & config ) ;
2015-02-20 16:08:43 +00:00
of_node_put ( config . of_node ) ;
2012-06-23 13:07:57 +01:00
if ( IS_ERR ( ldo1 - > regulator ) ) {
ret = PTR_ERR ( ldo1 - > regulator ) ;
2017-04-18 11:43:54 +01:00
dev_err ( & pdev - > dev , " Failed to register LDO1 supply: %d \n " ,
2012-06-23 13:07:57 +01:00
ret ) ;
return ret ;
}
platform_set_drvdata ( pdev , ldo1 ) ;
return 0 ;
}
2017-04-18 11:43:54 +01:00
static int arizona_ldo1_probe ( struct platform_device * pdev )
{
struct arizona * arizona = dev_get_drvdata ( pdev - > dev . parent ) ;
struct arizona_ldo1 * ldo1 ;
const struct regulator_desc * desc ;
bool external_dcvdd ;
int ret ;
ldo1 = devm_kzalloc ( & pdev - > dev , sizeof ( * ldo1 ) , GFP_KERNEL ) ;
if ( ! ldo1 )
return - ENOMEM ;
ldo1 - > regmap = arizona - > regmap ;
/*
* Since the chip usually supplies itself we provide some
* default init_data for it . This will be overridden with
* platform data if provided .
*/
switch ( arizona - > type ) {
case WM5102 :
case WM8997 :
case WM8998 :
case WM1814 :
desc = & arizona_ldo1_hc ;
ldo1 - > init_data = arizona_ldo1_dvfs ;
break ;
case WM5110 :
case WM8280 :
desc = & arizona_ldo1 ;
ldo1 - > init_data = arizona_ldo1_wm5110 ;
break ;
default :
desc = & arizona_ldo1 ;
ldo1 - > init_data = arizona_ldo1_default ;
break ;
}
ret = arizona_ldo1_common_init ( pdev , ldo1 , desc ,
& arizona - > pdata . ldo1 ,
& external_dcvdd ) ;
if ( ret = = 0 )
arizona - > external_dcvdd = external_dcvdd ;
return ret ;
}
2023-12-05 13:26:16 +01:00
static void arizona_ldo1_remove ( struct platform_device * pdev )
2018-06-19 16:10:00 +01:00
{
struct arizona_ldo1 * ldo1 = platform_get_drvdata ( pdev ) ;
if ( ldo1 - > ena_gpiod )
gpiod_put ( ldo1 - > ena_gpiod ) ;
}
2019-05-21 11:04:38 +01:00
static int madera_ldo1_probe ( struct platform_device * pdev )
{
struct madera * madera = dev_get_drvdata ( pdev - > dev . parent ) ;
struct arizona_ldo1 * ldo1 ;
bool external_dcvdd ;
int ret ;
ldo1 = devm_kzalloc ( & pdev - > dev , sizeof ( * ldo1 ) , GFP_KERNEL ) ;
if ( ! ldo1 )
return - ENOMEM ;
ldo1 - > regmap = madera - > regmap ;
ldo1 - > init_data = madera_ldo1_default ;
ret = arizona_ldo1_common_init ( pdev , ldo1 , & madera_ldo1 ,
& madera - > pdata . ldo1 ,
& external_dcvdd ) ;
if ( ret )
return ret ;
madera - > internal_dcvdd = ! external_dcvdd ;
return 0 ;
}
2012-06-23 13:07:57 +01:00
static struct platform_driver arizona_ldo1_driver = {
. probe = arizona_ldo1_probe ,
2023-12-05 13:26:16 +01:00
. remove_new = arizona_ldo1_remove ,
2012-06-23 13:07:57 +01:00
. driver = {
. name = " arizona-ldo1 " ,
2023-03-23 13:20:46 +00:00
. probe_type = PROBE_FORCE_SYNCHRONOUS ,
2012-06-23 13:07:57 +01:00
} ,
} ;
2019-05-21 11:04:38 +01:00
static struct platform_driver madera_ldo1_driver = {
. probe = madera_ldo1_probe ,
2023-12-05 13:26:16 +01:00
. remove_new = arizona_ldo1_remove ,
2019-05-21 11:04:38 +01:00
. driver = {
. name = " madera-ldo1 " ,
2023-03-23 13:20:46 +00:00
. probe_type = PROBE_FORCE_SYNCHRONOUS ,
2019-05-21 11:04:38 +01:00
} ,
} ;
static struct platform_driver * const madera_ldo1_drivers [ ] = {
& arizona_ldo1_driver ,
& madera_ldo1_driver ,
} ;
static int __init arizona_ldo1_init ( void )
{
return platform_register_drivers ( madera_ldo1_drivers ,
ARRAY_SIZE ( madera_ldo1_drivers ) ) ;
}
module_init ( arizona_ldo1_init ) ;
static void __exit madera_ldo1_exit ( void )
{
platform_unregister_drivers ( madera_ldo1_drivers ,
ARRAY_SIZE ( madera_ldo1_drivers ) ) ;
}
module_exit ( madera_ldo1_exit ) ;
2012-06-23 13:07:57 +01:00
/* Module information */
MODULE_AUTHOR ( " Mark Brown <broonie@opensource.wolfsonmicro.com> " ) ;
MODULE_DESCRIPTION ( " Arizona LDO1 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform:arizona-ldo1 " ) ;
2019-05-21 11:04:38 +01:00
MODULE_ALIAS ( " platform:madera-ldo1 " ) ;