2019-05-06 22:16:14 +03:00
// SPDX-License-Identifier: GPL-2.0
// TI LM3697 LED chip family driver
2020-07-13 17:51:15 +03:00
// Copyright (C) 2018 Texas Instruments Incorporated - https://www.ti.com/
2019-05-06 22:16:14 +03:00
2021-05-29 14:19:31 +03:00
# include <linux/bits.h>
2019-05-06 22:16:14 +03:00
# include <linux/gpio/consumer.h>
# include <linux/i2c.h>
2021-05-29 14:19:31 +03:00
# include <linux/mod_devicetable.h>
# include <linux/module.h>
# include <linux/property.h>
# include <linux/regmap.h>
2019-05-06 22:16:14 +03:00
# include <linux/regulator/consumer.h>
2021-05-29 14:19:31 +03:00
# include <linux/types.h>
2019-05-06 22:16:14 +03:00
# include <linux/leds-ti-lmu-common.h>
# define LM3697_REV 0x0
# define LM3697_RESET 0x1
# define LM3697_OUTPUT_CONFIG 0x10
# define LM3697_CTRL_A_RAMP 0x11
# define LM3697_CTRL_B_RAMP 0x12
# define LM3697_CTRL_A_B_RT_RAMP 0x13
# define LM3697_CTRL_A_B_RAMP_CFG 0x14
# define LM3697_CTRL_A_B_BRT_CFG 0x16
# define LM3697_CTRL_A_FS_CURR_CFG 0x17
# define LM3697_CTRL_B_FS_CURR_CFG 0x18
# define LM3697_PWM_CFG 0x1c
# define LM3697_CTRL_A_BRT_LSB 0x20
# define LM3697_CTRL_A_BRT_MSB 0x21
# define LM3697_CTRL_B_BRT_LSB 0x22
# define LM3697_CTRL_B_BRT_MSB 0x23
# define LM3697_CTRL_ENABLE 0x24
# define LM3697_SW_RESET BIT(0)
# define LM3697_CTRL_A_EN BIT(0)
# define LM3697_CTRL_B_EN BIT(1)
# define LM3697_CTRL_A_B_EN (LM3697_CTRL_A_EN | LM3697_CTRL_B_EN)
# define LM3697_MAX_LED_STRINGS 3
# define LM3697_CONTROL_A 0
# define LM3697_CONTROL_B 1
# define LM3697_MAX_CONTROL_BANKS 2
/**
* struct lm3697_led -
* @ hvled_strings : Array of LED strings associated with a control bank
* @ label : LED label
* @ led_dev : LED class device
* @ priv : Pointer to the device struct
* @ lmu_data : Register and setting values for common code
* @ control_bank : Control bank the LED is associated to . 0 is control bank A
* 1 is control bank B
2021-05-28 12:06:25 +03:00
* @ enabled : LED brightness level ( or LED_OFF )
* @ num_leds : Number of LEDs available
2019-05-06 22:16:14 +03:00
*/
struct lm3697_led {
u32 hvled_strings [ LM3697_MAX_LED_STRINGS ] ;
char label [ LED_MAX_NAME_SIZE ] ;
struct led_classdev led_dev ;
struct lm3697 * priv ;
struct ti_lmu_bank lmu_data ;
int control_bank ;
int enabled ;
int num_leds ;
} ;
/**
* struct lm3697 -
* @ enable_gpio : Hardware enable gpio
* @ regulator : LED supply regulator pointer
* @ client : Pointer to the I2C client
* @ regmap : Devices register map
* @ dev : Pointer to the devices device struct
* @ lock : Lock for reading / writing the device
* @ leds : Array of LED strings
2021-05-28 12:06:25 +03:00
* @ bank_cfg : OUTPUT_CONFIG register values
* @ num_banks : Number of control banks
2019-05-06 22:16:14 +03:00
*/
struct lm3697 {
struct gpio_desc * enable_gpio ;
struct regulator * regulator ;
struct i2c_client * client ;
struct regmap * regmap ;
struct device * dev ;
struct mutex lock ;
int bank_cfg ;
2020-10-03 01:27:00 +03:00
int num_banks ;
2019-05-06 22:16:14 +03:00
2023-09-15 23:10:10 +03:00
struct lm3697_led leds [ ] __counted_by ( num_banks ) ;
2019-05-06 22:16:14 +03:00
} ;
static const struct reg_default lm3697_reg_defs [ ] = {
{ LM3697_OUTPUT_CONFIG , 0x6 } ,
{ LM3697_CTRL_A_RAMP , 0x0 } ,
{ LM3697_CTRL_B_RAMP , 0x0 } ,
{ LM3697_CTRL_A_B_RT_RAMP , 0x0 } ,
{ LM3697_CTRL_A_B_RAMP_CFG , 0x0 } ,
{ LM3697_CTRL_A_B_BRT_CFG , 0x0 } ,
{ LM3697_CTRL_A_FS_CURR_CFG , 0x13 } ,
{ LM3697_CTRL_B_FS_CURR_CFG , 0x13 } ,
{ LM3697_PWM_CFG , 0xc } ,
{ LM3697_CTRL_A_BRT_LSB , 0x0 } ,
{ LM3697_CTRL_A_BRT_MSB , 0x0 } ,
{ LM3697_CTRL_B_BRT_LSB , 0x0 } ,
{ LM3697_CTRL_B_BRT_MSB , 0x0 } ,
{ LM3697_CTRL_ENABLE , 0x0 } ,
} ;
static const struct regmap_config lm3697_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = LM3697_CTRL_ENABLE ,
. reg_defaults = lm3697_reg_defs ,
. num_reg_defaults = ARRAY_SIZE ( lm3697_reg_defs ) ,
. cache_type = REGCACHE_FLAT ,
} ;
static int lm3697_brightness_set ( struct led_classdev * led_cdev ,
enum led_brightness brt_val )
{
struct lm3697_led * led = container_of ( led_cdev , struct lm3697_led ,
led_dev ) ;
int ctrl_en_val = ( 1 < < led - > control_bank ) ;
2020-09-18 01:33:03 +03:00
struct device * dev = led - > priv - > dev ;
2019-05-06 22:16:14 +03:00
int ret ;
mutex_lock ( & led - > priv - > lock ) ;
if ( brt_val = = LED_OFF ) {
ret = regmap_update_bits ( led - > priv - > regmap , LM3697_CTRL_ENABLE ,
ctrl_en_val , ~ ctrl_en_val ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot write ctrl register \n " ) ;
2019-05-06 22:16:14 +03:00
goto brightness_out ;
}
led - > enabled = LED_OFF ;
} else {
ret = ti_lmu_common_set_brightness ( & led - > lmu_data , brt_val ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot write brightness \n " ) ;
2019-05-06 22:16:14 +03:00
goto brightness_out ;
}
if ( ! led - > enabled ) {
ret = regmap_update_bits ( led - > priv - > regmap ,
LM3697_CTRL_ENABLE ,
ctrl_en_val , ctrl_en_val ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot enable the device \n " ) ;
2019-05-06 22:16:14 +03:00
goto brightness_out ;
}
led - > enabled = brt_val ;
}
}
brightness_out :
mutex_unlock ( & led - > priv - > lock ) ;
return ret ;
}
static int lm3697_init ( struct lm3697 * priv )
{
2020-09-18 01:33:03 +03:00
struct device * dev = priv - > dev ;
2019-05-06 22:16:14 +03:00
struct lm3697_led * led ;
int i , ret ;
if ( priv - > enable_gpio ) {
gpiod_direction_output ( priv - > enable_gpio , 1 ) ;
} else {
ret = regmap_write ( priv - > regmap , LM3697_RESET , LM3697_SW_RESET ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot reset the device \n " ) ;
2019-05-06 22:16:14 +03:00
goto out ;
}
}
ret = regmap_write ( priv - > regmap , LM3697_CTRL_ENABLE , 0x0 ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot write ctrl enable \n " ) ;
2019-05-06 22:16:14 +03:00
goto out ;
}
ret = regmap_write ( priv - > regmap , LM3697_OUTPUT_CONFIG , priv - > bank_cfg ) ;
if ( ret )
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Cannot write OUTPUT config \n " ) ;
2019-05-06 22:16:14 +03:00
2020-10-03 01:27:00 +03:00
for ( i = 0 ; i < priv - > num_banks ; i + + ) {
2019-05-06 22:16:14 +03:00
led = & priv - > leds [ i ] ;
ret = ti_lmu_common_set_ramp ( & led - > lmu_data ) ;
if ( ret )
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Setting the ramp rate failed \n " ) ;
2019-05-06 22:16:14 +03:00
}
out :
return ret ;
}
static int lm3697_probe_dt ( struct lm3697 * priv )
{
struct fwnode_handle * child = NULL ;
2020-09-18 01:33:03 +03:00
struct device * dev = priv - > dev ;
2019-05-06 22:16:14 +03:00
struct lm3697_led * led ;
2020-09-18 01:33:03 +03:00
int ret = - EINVAL ;
2019-05-06 22:16:14 +03:00
int control_bank ;
size_t i = 0 ;
int j ;
2020-09-18 01:33:03 +03:00
priv - > enable_gpio = devm_gpiod_get_optional ( dev , " enable " ,
GPIOD_OUT_LOW ) ;
2021-05-10 12:50:39 +03:00
if ( IS_ERR ( priv - > enable_gpio ) )
return dev_err_probe ( dev , PTR_ERR ( priv - > enable_gpio ) ,
" Failed to get enable GPIO \n " ) ;
2019-05-06 22:16:14 +03:00
2020-09-18 01:33:03 +03:00
priv - > regulator = devm_regulator_get ( dev , " vled " ) ;
2019-05-06 22:16:14 +03:00
if ( IS_ERR ( priv - > regulator ) )
priv - > regulator = NULL ;
2020-09-18 01:33:03 +03:00
device_for_each_child_node ( dev , child ) {
2020-09-18 01:33:02 +03:00
struct led_init_data init_data = { } ;
2019-05-06 22:16:14 +03:00
ret = fwnode_property_read_u32 ( child , " reg " , & control_bank ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " reg property missing \n " ) ;
2019-05-06 22:16:14 +03:00
goto child_out ;
}
if ( control_bank > LM3697_CONTROL_B ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " reg property is invalid \n " ) ;
2019-05-06 22:16:14 +03:00
ret = - EINVAL ;
goto child_out ;
}
led = & priv - > leds [ i ] ;
2020-09-18 01:33:03 +03:00
ret = ti_lmu_common_get_brt_res ( dev , child , & led - > lmu_data ) ;
2019-05-06 22:16:14 +03:00
if ( ret )
2020-09-18 01:33:03 +03:00
dev_warn ( dev ,
" brightness resolution property missing \n " ) ;
2019-05-06 22:16:14 +03:00
led - > control_bank = control_bank ;
led - > lmu_data . regmap = priv - > regmap ;
led - > lmu_data . runtime_ramp_reg = LM3697_CTRL_A_RAMP +
control_bank ;
led - > lmu_data . msb_brightness_reg = LM3697_CTRL_A_BRT_MSB +
led - > control_bank * 2 ;
led - > lmu_data . lsb_brightness_reg = LM3697_CTRL_A_BRT_LSB +
led - > control_bank * 2 ;
2019-07-23 23:14:59 +03:00
led - > num_leds = fwnode_property_count_u32 ( child , " led-sources " ) ;
2019-05-06 22:16:14 +03:00
if ( led - > num_leds > LM3697_MAX_LED_STRINGS ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Too many LED strings defined \n " ) ;
2019-05-06 22:16:14 +03:00
continue ;
}
ret = fwnode_property_read_u32_array ( child , " led-sources " ,
led - > hvled_strings ,
led - > num_leds ) ;
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " led-sources property missing \n " ) ;
2019-05-06 22:16:14 +03:00
goto child_out ;
}
for ( j = 0 ; j < led - > num_leds ; j + + )
priv - > bank_cfg | =
( led - > control_bank < < led - > hvled_strings [ j ] ) ;
2020-09-18 01:33:03 +03:00
ret = ti_lmu_common_get_ramp_params ( dev , child , & led - > lmu_data ) ;
2019-05-06 22:16:14 +03:00
if ( ret )
2020-09-18 01:33:03 +03:00
dev_warn ( dev , " runtime-ramp properties missing \n " ) ;
2019-05-06 22:16:14 +03:00
2020-09-18 01:33:02 +03:00
init_data . fwnode = child ;
init_data . devicename = priv - > client - > name ;
/* for backwards compatibility if `label` is not present */
init_data . default_label = " : " ;
2019-05-06 22:16:14 +03:00
led - > priv = priv ;
led - > led_dev . max_brightness = led - > lmu_data . max_brightness ;
led - > led_dev . brightness_set_blocking = lm3697_brightness_set ;
2020-09-18 01:33:03 +03:00
ret = devm_led_classdev_register_ext ( dev , & led - > led_dev ,
2020-09-18 01:33:02 +03:00
& init_data ) ;
2019-05-06 22:16:14 +03:00
if ( ret ) {
2020-09-18 01:33:03 +03:00
dev_err ( dev , " led register err: %d \n " , ret ) ;
2019-05-06 22:16:14 +03:00
goto child_out ;
}
i + + ;
}
2021-05-29 14:19:32 +03:00
return ret ;
2019-05-06 22:16:14 +03:00
child_out :
2021-05-29 14:19:32 +03:00
fwnode_handle_put ( child ) ;
2019-05-06 22:16:14 +03:00
return ret ;
}
2022-11-19 01:40:10 +03:00
static int lm3697_probe ( struct i2c_client * client )
2019-05-06 22:16:14 +03:00
{
2020-09-18 01:33:03 +03:00
struct device * dev = & client - > dev ;
2019-05-06 22:16:14 +03:00
struct lm3697 * led ;
int count ;
int ret ;
2020-09-18 01:33:03 +03:00
count = device_get_child_node_count ( dev ) ;
2020-10-03 01:27:00 +03:00
if ( ! count | | count > LM3697_MAX_CONTROL_BANKS ) {
dev_err ( dev , " Strange device tree! " ) ;
2019-05-06 22:16:14 +03:00
return - ENODEV ;
}
2020-09-18 01:33:03 +03:00
led = devm_kzalloc ( dev , struct_size ( led , leds , count ) , GFP_KERNEL ) ;
2019-05-06 22:16:14 +03:00
if ( ! led )
return - ENOMEM ;
mutex_init ( & led - > lock ) ;
i2c_set_clientdata ( client , led ) ;
led - > client = client ;
2020-09-18 01:33:03 +03:00
led - > dev = dev ;
2020-10-03 01:27:00 +03:00
led - > num_banks = count ;
2019-05-06 22:16:14 +03:00
led - > regmap = devm_regmap_init_i2c ( client , & lm3697_regmap_config ) ;
if ( IS_ERR ( led - > regmap ) ) {
ret = PTR_ERR ( led - > regmap ) ;
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Failed to allocate register map: %d \n " , ret ) ;
2019-05-06 22:16:14 +03:00
return ret ;
}
ret = lm3697_probe_dt ( led ) ;
if ( ret )
return ret ;
return lm3697_init ( led ) ;
}
2022-08-15 11:02:30 +03:00
static void lm3697_remove ( struct i2c_client * client )
2019-05-06 22:16:14 +03:00
{
struct lm3697 * led = i2c_get_clientdata ( client ) ;
2020-09-18 01:33:03 +03:00
struct device * dev = & led - > client - > dev ;
2019-05-06 22:16:14 +03:00
int ret ;
ret = regmap_update_bits ( led - > regmap , LM3697_CTRL_ENABLE ,
LM3697_CTRL_A_B_EN , 0 ) ;
2022-08-15 11:02:26 +03:00
if ( ret )
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Failed to disable the device \n " ) ;
2019-05-06 22:16:14 +03:00
if ( led - > enable_gpio )
gpiod_direction_output ( led - > enable_gpio , 0 ) ;
if ( led - > regulator ) {
ret = regulator_disable ( led - > regulator ) ;
if ( ret )
2020-09-18 01:33:03 +03:00
dev_err ( dev , " Failed to disable regulator \n " ) ;
2019-05-06 22:16:14 +03:00
}
mutex_destroy ( & led - > lock ) ;
}
static const struct i2c_device_id lm3697_id [ ] = {
{ " lm3697 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm3697_id ) ;
static const struct of_device_id of_lm3697_leds_match [ ] = {
{ . compatible = " ti,lm3697 " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , of_lm3697_leds_match ) ;
static struct i2c_driver lm3697_driver = {
. driver = {
. name = " lm3697 " ,
. of_match_table = of_lm3697_leds_match ,
} ,
2023-05-17 21:05:59 +03:00
. probe = lm3697_probe ,
2019-05-06 22:16:14 +03:00
. remove = lm3697_remove ,
. id_table = lm3697_id ,
} ;
module_i2c_driver ( lm3697_driver ) ;
MODULE_DESCRIPTION ( " Texas Instruments LM3697 LED driver " ) ;
MODULE_AUTHOR ( " Dan Murphy <dmurphy@ti.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;