2020-06-12 12:05:20 +08:00
// SPDX-License-Identifier: GPL-2.0-only
//
// Copyright 2020 Google LLC.
# include <linux/module.h>
# include <linux/of.h>
# include <linux/platform_data/cros_ec_proto.h>
# include <linux/platform_device.h>
# include <linux/regulator/driver.h>
# include <linux/regulator/machine.h>
# include <linux/regulator/of_regulator.h>
# include <linux/slab.h>
struct cros_ec_regulator_data {
struct regulator_desc desc ;
struct regulator_dev * dev ;
struct cros_ec_device * ec_dev ;
u32 index ;
u16 * voltages_mV ;
u16 num_voltages ;
} ;
static int cros_ec_cmd ( struct cros_ec_device * ec , u32 version , u32 command ,
void * outdata , u32 outsize , void * indata , u32 insize )
{
struct cros_ec_command * msg ;
int ret ;
msg = kzalloc ( sizeof ( * msg ) + max ( outsize , insize ) , GFP_KERNEL ) ;
if ( ! msg )
return - ENOMEM ;
msg - > version = version ;
msg - > command = command ;
msg - > outsize = outsize ;
msg - > insize = insize ;
if ( outdata & & outsize > 0 )
memcpy ( msg - > data , outdata , outsize ) ;
ret = cros_ec_cmd_xfer_status ( ec , msg ) ;
if ( ret < 0 )
goto cleanup ;
if ( insize )
memcpy ( indata , msg - > data , insize ) ;
cleanup :
kfree ( msg ) ;
return ret ;
}
static int cros_ec_regulator_enable ( struct regulator_dev * dev )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
struct ec_params_regulator_enable cmd = {
. index = data - > index ,
. enable = 1 ,
} ;
return cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_ENABLE , & cmd ,
sizeof ( cmd ) , NULL , 0 ) ;
}
static int cros_ec_regulator_disable ( struct regulator_dev * dev )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
struct ec_params_regulator_enable cmd = {
. index = data - > index ,
. enable = 0 ,
} ;
return cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_ENABLE , & cmd ,
sizeof ( cmd ) , NULL , 0 ) ;
}
static int cros_ec_regulator_is_enabled ( struct regulator_dev * dev )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
struct ec_params_regulator_is_enabled cmd = {
. index = data - > index ,
} ;
struct ec_response_regulator_is_enabled resp ;
int ret ;
ret = cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_IS_ENABLED , & cmd ,
sizeof ( cmd ) , & resp , sizeof ( resp ) ) ;
if ( ret < 0 )
return ret ;
return resp . enabled ;
}
static int cros_ec_regulator_list_voltage ( struct regulator_dev * dev ,
unsigned int selector )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
if ( selector > = data - > num_voltages )
return - EINVAL ;
return data - > voltages_mV [ selector ] * 1000 ;
}
static int cros_ec_regulator_get_voltage ( struct regulator_dev * dev )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
struct ec_params_regulator_get_voltage cmd = {
. index = data - > index ,
} ;
struct ec_response_regulator_get_voltage resp ;
int ret ;
ret = cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_GET_VOLTAGE , & cmd ,
sizeof ( cmd ) , & resp , sizeof ( resp ) ) ;
if ( ret < 0 )
return ret ;
return resp . voltage_mv * 1000 ;
}
static int cros_ec_regulator_set_voltage ( struct regulator_dev * dev , int min_uV ,
int max_uV , unsigned int * selector )
{
struct cros_ec_regulator_data * data = rdev_get_drvdata ( dev ) ;
int min_mV = DIV_ROUND_UP ( min_uV , 1000 ) ;
int max_mV = max_uV / 1000 ;
struct ec_params_regulator_set_voltage cmd = {
. index = data - > index ,
. min_mv = min_mV ,
. max_mv = max_mV ,
} ;
/*
* This can happen when the given range [ min_uV , max_uV ] doesn ' t
* contain any voltage that can be represented exactly in mV .
*/
if ( min_mV > max_mV )
return - EINVAL ;
return cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_SET_VOLTAGE , & cmd ,
sizeof ( cmd ) , NULL , 0 ) ;
}
2020-07-11 13:44:09 +02:00
static const struct regulator_ops cros_ec_regulator_voltage_ops = {
2020-06-12 12:05:20 +08:00
. enable = cros_ec_regulator_enable ,
. disable = cros_ec_regulator_disable ,
. is_enabled = cros_ec_regulator_is_enabled ,
. list_voltage = cros_ec_regulator_list_voltage ,
. get_voltage = cros_ec_regulator_get_voltage ,
. set_voltage = cros_ec_regulator_set_voltage ,
} ;
static int cros_ec_regulator_init_info ( struct device * dev ,
struct cros_ec_regulator_data * data )
{
struct ec_params_regulator_get_info cmd = {
. index = data - > index ,
} ;
struct ec_response_regulator_get_info resp ;
int ret ;
ret = cros_ec_cmd ( data - > ec_dev , 0 , EC_CMD_REGULATOR_GET_INFO , & cmd ,
sizeof ( cmd ) , & resp , sizeof ( resp ) ) ;
if ( ret < 0 )
return ret ;
data - > num_voltages =
min_t ( u16 , ARRAY_SIZE ( resp . voltages_mv ) , resp . num_voltages ) ;
data - > voltages_mV =
devm_kmemdup ( dev , resp . voltages_mv ,
sizeof ( u16 ) * data - > num_voltages , GFP_KERNEL ) ;
2020-08-02 11:25:09 +08:00
if ( ! data - > voltages_mV )
return - ENOMEM ;
2020-06-12 12:05:20 +08:00
data - > desc . n_voltages = data - > num_voltages ;
/* Make sure the returned name is always a valid string */
resp . name [ ARRAY_SIZE ( resp . name ) - 1 ] = ' \0 ' ;
data - > desc . name = devm_kstrdup ( dev , resp . name , GFP_KERNEL ) ;
if ( ! data - > desc . name )
return - ENOMEM ;
return 0 ;
}
static int cros_ec_regulator_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct device_node * np = dev - > of_node ;
struct cros_ec_regulator_data * drvdata ;
struct regulator_init_data * init_data ;
struct regulator_config cfg = { } ;
struct regulator_desc * desc ;
int ret ;
drvdata = devm_kzalloc (
& pdev - > dev , sizeof ( struct cros_ec_regulator_data ) , GFP_KERNEL ) ;
if ( ! drvdata )
return - ENOMEM ;
drvdata - > ec_dev = dev_get_drvdata ( dev - > parent ) ;
desc = & drvdata - > desc ;
init_data = of_get_regulator_init_data ( dev , np , desc ) ;
if ( ! init_data )
return - EINVAL ;
ret = of_property_read_u32 ( np , " reg " , & drvdata - > index ) ;
if ( ret < 0 )
return ret ;
desc - > owner = THIS_MODULE ;
desc - > type = REGULATOR_VOLTAGE ;
desc - > ops = & cros_ec_regulator_voltage_ops ;
ret = cros_ec_regulator_init_info ( dev , drvdata ) ;
if ( ret < 0 )
return ret ;
cfg . dev = & pdev - > dev ;
cfg . init_data = init_data ;
cfg . driver_data = drvdata ;
cfg . of_node = np ;
drvdata - > dev = devm_regulator_register ( dev , & drvdata - > desc , & cfg ) ;
if ( IS_ERR ( drvdata - > dev ) ) {
2021-05-12 15:58:24 +08:00
ret = PTR_ERR ( drvdata - > dev ) ;
2020-06-12 12:05:20 +08:00
dev_err ( & pdev - > dev , " Failed to register regulator: %d \n " , ret ) ;
2021-05-12 15:58:24 +08:00
return ret ;
2020-06-12 12:05:20 +08:00
}
platform_set_drvdata ( pdev , drvdata ) ;
return 0 ;
}
static const struct of_device_id regulator_cros_ec_of_match [ ] = {
{ . compatible = " google,cros-ec-regulator " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , regulator_cros_ec_of_match ) ;
static struct platform_driver cros_ec_regulator_driver = {
. probe = cros_ec_regulator_probe ,
. driver = {
. name = " cros-ec-regulator " ,
. of_match_table = regulator_cros_ec_of_match ,
} ,
} ;
module_platform_driver ( cros_ec_regulator_driver ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " ChromeOS EC controlled regulator " ) ;
MODULE_AUTHOR ( " Pi-Hsun Shih <pihsun@chromium.org> " ) ;