2018-07-02 13:12:13 -05:00
// SPDX-License-Identifier: GPL-2.0
// TI LM3692x LED chip family driver
2020-07-13 16:51:15 +02:00
// Copyright (C) 2017-18 Texas Instruments Incorporated - https://www.ti.com/
2017-12-12 12:15:50 -06:00
# include <linux/gpio/consumer.h>
# include <linux/i2c.h>
# include <linux/init.h>
# include <linux/leds.h>
2020-01-06 16:48:50 +01:00
# include <linux/log2.h>
2021-05-29 14:19:30 +03:00
# include <linux/mod_devicetable.h>
2017-12-12 12:15:50 -06:00
# include <linux/module.h>
# include <linux/mutex.h>
# include <linux/regmap.h>
# include <linux/regulator/consumer.h>
# include <linux/slab.h>
2018-07-02 13:12:16 -05:00
# define LM36922_MODEL 0
# define LM36923_MODEL 1
2017-12-12 12:15:50 -06:00
# define LM3692X_REV 0x0
# define LM3692X_RESET 0x1
# define LM3692X_EN 0x10
# define LM3692X_BRT_CTRL 0x11
# define LM3692X_PWM_CTRL 0x12
# define LM3692X_BOOST_CTRL 0x13
# define LM3692X_AUTO_FREQ_HI 0x15
# define LM3692X_AUTO_FREQ_LO 0x16
# define LM3692X_BL_ADJ_THRESH 0x17
# define LM3692X_BRT_LSB 0x18
# define LM3692X_BRT_MSB 0x19
# define LM3692X_FAULT_CTRL 0x1e
# define LM3692X_FAULT_FLAGS 0x1f
# define LM3692X_SW_RESET BIT(0)
# define LM3692X_DEVICE_EN BIT(0)
# define LM3692X_LED1_EN BIT(1)
# define LM3692X_LED2_EN BIT(2)
2018-07-02 13:12:16 -05:00
# define LM36923_LED3_EN BIT(3)
# define LM3692X_ENABLE_MASK (LM3692X_DEVICE_EN | LM3692X_LED1_EN | \
LM3692X_LED2_EN | LM36923_LED3_EN )
2017-12-12 12:15:50 -06:00
/* Brightness Control Bits */
# define LM3692X_BL_ADJ_POL BIT(0)
# define LM3692X_RAMP_RATE_125us 0x00
# define LM3692X_RAMP_RATE_250us BIT(1)
# define LM3692X_RAMP_RATE_500us BIT(2)
# define LM3692X_RAMP_RATE_1ms (BIT(1) | BIT(2))
# define LM3692X_RAMP_RATE_2ms BIT(3)
# define LM3692X_RAMP_RATE_4ms (BIT(3) | BIT(1))
# define LM3692X_RAMP_RATE_8ms (BIT(2) | BIT(3))
# define LM3692X_RAMP_RATE_16ms (BIT(1) | BIT(2) | BIT(3))
# define LM3692X_RAMP_EN BIT(4)
# define LM3692X_BRHT_MODE_REG 0x00
# define LM3692X_BRHT_MODE_PWM BIT(5)
# define LM3692X_BRHT_MODE_MULTI_RAMP BIT(6)
# define LM3692X_BRHT_MODE_RAMP_MULTI (BIT(5) | BIT(6))
# define LM3692X_MAP_MODE_EXP BIT(7)
/* PWM Register Bits */
# define LM3692X_PWM_FILTER_100 BIT(0)
# define LM3692X_PWM_FILTER_150 BIT(1)
# define LM3692X_PWM_FILTER_200 (BIT(0) | BIT(1))
# define LM3692X_PWM_HYSTER_1LSB BIT(2)
# define LM3692X_PWM_HYSTER_2LSB BIT(3)
# define LM3692X_PWM_HYSTER_3LSB (BIT(3) | BIT(2))
# define LM3692X_PWM_HYSTER_4LSB BIT(4)
# define LM3692X_PWM_HYSTER_5LSB (BIT(4) | BIT(2))
# define LM3692X_PWM_HYSTER_6LSB (BIT(4) | BIT(3))
# define LM3692X_PWM_POLARITY BIT(5)
# define LM3692X_PWM_SAMP_4MHZ BIT(6)
# define LM3692X_PWM_SAMP_24MHZ BIT(7)
/* Boost Control Bits */
# define LM3692X_OCP_PROT_1A BIT(0)
# define LM3692X_OCP_PROT_1_25A BIT(1)
# define LM3692X_OCP_PROT_1_5A (BIT(0) | BIT(1))
# define LM3692X_OVP_21V BIT(2)
# define LM3692X_OVP_25V BIT(3)
# define LM3692X_OVP_29V (BIT(2) | BIT(3))
# define LM3692X_MIN_IND_22UH BIT(4)
# define LM3692X_BOOST_SW_1MHZ BIT(5)
# define LM3692X_BOOST_SW_NO_SHIFT BIT(6)
/* Fault Control Bits */
# define LM3692X_FAULT_CTRL_OVP BIT(0)
# define LM3692X_FAULT_CTRL_OCP BIT(1)
# define LM3692X_FAULT_CTRL_TSD BIT(2)
# define LM3692X_FAULT_CTRL_OPEN BIT(3)
/* Fault Flag Bits */
# define LM3692X_FAULT_FLAG_OVP BIT(0)
# define LM3692X_FAULT_FLAG_OCP BIT(1)
# define LM3692X_FAULT_FLAG_TSD BIT(2)
# define LM3692X_FAULT_FLAG_SHRT BIT(3)
# define LM3692X_FAULT_FLAG_OPEN BIT(4)
/**
2021-05-28 10:06:24 +01:00
* struct lm3692x_led
* @ lock : Lock for reading / writing the device
* @ client : Pointer to the I2C client
* @ led_dev : LED class device pointer
* @ regmap : Devices register map
* @ enable_gpio : VDDIO / EN gpio to enable communication interface
* @ regulator : LED supply regulator pointer
* @ led_enable : LED sync to be enabled
* @ model_id : Current device model ID enumerated
2017-12-12 12:15:50 -06:00
*/
struct lm3692x_led {
struct mutex lock ;
struct i2c_client * client ;
struct led_classdev led_dev ;
struct regmap * regmap ;
struct gpio_desc * enable_gpio ;
struct regulator * regulator ;
2018-07-02 13:12:16 -05:00
int led_enable ;
int model_id ;
2020-01-04 11:54:18 +01:00
2020-01-06 16:48:53 +01:00
u8 boost_ctrl , brightness_ctrl ;
bool enabled ;
2017-12-12 12:15:50 -06:00
} ;
static const struct reg_default lm3692x_reg_defs [ ] = {
{ LM3692X_EN , 0xf } ,
{ LM3692X_BRT_CTRL , 0x61 } ,
{ LM3692X_PWM_CTRL , 0x73 } ,
{ LM3692X_BOOST_CTRL , 0x6f } ,
{ LM3692X_AUTO_FREQ_HI , 0x0 } ,
{ LM3692X_AUTO_FREQ_LO , 0x0 } ,
{ LM3692X_BL_ADJ_THRESH , 0x0 } ,
{ LM3692X_BRT_LSB , 0x7 } ,
{ LM3692X_BRT_MSB , 0xff } ,
{ LM3692X_FAULT_CTRL , 0x7 } ,
} ;
static const struct regmap_config lm3692x_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = LM3692X_FAULT_FLAGS ,
. reg_defaults = lm3692x_reg_defs ,
. num_reg_defaults = ARRAY_SIZE ( lm3692x_reg_defs ) ,
. cache_type = REGCACHE_RBTREE ,
} ;
static int lm3692x_fault_check ( struct lm3692x_led * led )
{
int ret ;
unsigned int read_buf ;
ret = regmap_read ( led - > regmap , LM3692X_FAULT_FLAGS , & read_buf ) ;
if ( ret )
return ret ;
if ( read_buf )
dev_err ( & led - > client - > dev , " Detected a fault 0x%X \n " , read_buf ) ;
/* The first read may clear the fault. Check again to see if the fault
* still exits and return that value .
*/
regmap_read ( led - > regmap , LM3692X_FAULT_FLAGS , & read_buf ) ;
if ( read_buf )
dev_err ( & led - > client - > dev , " Second read of fault flags 0x%X \n " ,
read_buf ) ;
return read_buf ;
}
2020-01-06 16:48:51 +01:00
static int lm3692x_leds_enable ( struct lm3692x_led * led )
2017-12-12 12:15:50 -06:00
{
2018-07-02 13:12:16 -05:00
int enable_state ;
2019-09-21 14:12:09 -07:00
int ret , reg_ret ;
2017-12-12 12:15:50 -06:00
2020-01-06 16:48:53 +01:00
if ( led - > enabled )
return 0 ;
2017-12-12 12:15:50 -06:00
if ( led - > regulator ) {
ret = regulator_enable ( led - > regulator ) ;
if ( ret ) {
dev_err ( & led - > client - > dev ,
2019-09-21 14:12:08 -07:00
" Failed to enable regulator: %d \n " , ret ) ;
2017-12-12 12:15:50 -06:00
return ret ;
}
}
if ( led - > enable_gpio )
gpiod_direction_output ( led - > enable_gpio , 1 ) ;
ret = lm3692x_fault_check ( led ) ;
if ( ret ) {
2019-09-21 14:12:08 -07:00
dev_err ( & led - > client - > dev , " Cannot read/clear faults: %d \n " ,
ret ) ;
2017-12-12 12:15:50 -06:00
goto out ;
}
ret = regmap_write ( led - > regmap , LM3692X_BRT_CTRL , 0x00 ) ;
if ( ret )
goto out ;
/*
* For glitch free operation , the following data should
2018-07-02 13:12:16 -05:00
* only be written while LEDx enable bits are 0 and the device enable
* bit is set to 1.
2017-12-12 12:15:50 -06:00
* per Section 7.5 .14 of the data sheet
*/
2018-07-02 13:12:16 -05:00
ret = regmap_write ( led - > regmap , LM3692X_EN , LM3692X_DEVICE_EN ) ;
if ( ret )
goto out ;
/* Set the brightness to 0 so when enabled the LEDs do not come
* on with full brightness .
*/
ret = regmap_write ( led - > regmap , LM3692X_BRT_MSB , 0 ) ;
if ( ret )
goto out ;
ret = regmap_write ( led - > regmap , LM3692X_BRT_LSB , 0 ) ;
if ( ret )
goto out ;
2017-12-12 12:15:50 -06:00
ret = regmap_write ( led - > regmap , LM3692X_PWM_CTRL ,
LM3692X_PWM_FILTER_100 | LM3692X_PWM_SAMP_24MHZ ) ;
if ( ret )
goto out ;
2020-01-04 11:54:18 +01:00
ret = regmap_write ( led - > regmap , LM3692X_BOOST_CTRL , led - > boost_ctrl ) ;
2017-12-12 12:15:50 -06:00
if ( ret )
goto out ;
ret = regmap_write ( led - > regmap , LM3692X_AUTO_FREQ_HI , 0x00 ) ;
if ( ret )
goto out ;
ret = regmap_write ( led - > regmap , LM3692X_AUTO_FREQ_LO , 0x00 ) ;
if ( ret )
goto out ;
ret = regmap_write ( led - > regmap , LM3692X_BL_ADJ_THRESH , 0x00 ) ;
if ( ret )
goto out ;
ret = regmap_write ( led - > regmap , LM3692X_BRT_CTRL ,
2019-09-21 14:12:12 -07:00
LM3692X_BL_ADJ_POL | LM3692X_RAMP_EN ) ;
2017-12-12 12:15:50 -06:00
if ( ret )
goto out ;
2018-07-02 13:12:16 -05:00
switch ( led - > led_enable ) {
case 0 :
default :
if ( led - > model_id = = LM36923_MODEL )
enable_state = LM3692X_LED1_EN | LM3692X_LED2_EN |
LM36923_LED3_EN ;
else
enable_state = LM3692X_LED1_EN | LM3692X_LED2_EN ;
break ;
case 1 :
enable_state = LM3692X_LED1_EN ;
break ;
case 2 :
enable_state = LM3692X_LED2_EN ;
break ;
case 3 :
if ( led - > model_id = = LM36923_MODEL ) {
enable_state = LM36923_LED3_EN ;
break ;
}
ret = - EINVAL ;
dev_err ( & led - > client - > dev ,
" LED3 sync not available on this device \n " ) ;
goto out ;
}
ret = regmap_update_bits ( led - > regmap , LM3692X_EN , LM3692X_ENABLE_MASK ,
enable_state | LM3692X_DEVICE_EN ) ;
2020-01-06 16:48:53 +01:00
led - > enabled = true ;
2017-12-12 12:15:50 -06:00
return ret ;
out :
dev_err ( & led - > client - > dev , " Fail writing initialization values \n " ) ;
if ( led - > enable_gpio )
gpiod_direction_output ( led - > enable_gpio , 0 ) ;
if ( led - > regulator ) {
2019-09-21 14:12:09 -07:00
reg_ret = regulator_disable ( led - > regulator ) ;
if ( reg_ret )
2017-12-12 12:15:50 -06:00
dev_err ( & led - > client - > dev ,
2019-09-21 14:12:09 -07:00
" Failed to disable regulator: %d \n " , reg_ret ) ;
2017-12-12 12:15:50 -06:00
}
return ret ;
}
2019-09-21 14:12:09 -07:00
2020-01-06 16:48:52 +01:00
static int lm3692x_leds_disable ( struct lm3692x_led * led )
{
int ret ;
2020-01-06 16:48:53 +01:00
if ( ! led - > enabled )
return 0 ;
2020-01-06 16:48:52 +01:00
ret = regmap_update_bits ( led - > regmap , LM3692X_EN , LM3692X_DEVICE_EN , 0 ) ;
if ( ret ) {
dev_err ( & led - > client - > dev , " Failed to disable regulator: %d \n " ,
ret ) ;
return ret ;
}
if ( led - > enable_gpio )
gpiod_direction_output ( led - > enable_gpio , 0 ) ;
if ( led - > regulator ) {
ret = regulator_disable ( led - > regulator ) ;
if ( ret )
dev_err ( & led - > client - > dev ,
" Failed to disable regulator: %d \n " , ret ) ;
}
2020-01-06 16:48:53 +01:00
led - > enabled = false ;
2020-01-06 16:48:52 +01:00
return ret ;
}
2020-01-06 16:48:51 +01:00
static int lm3692x_brightness_set ( struct led_classdev * led_cdev ,
enum led_brightness brt_val )
{
struct lm3692x_led * led =
container_of ( led_cdev , struct lm3692x_led , led_dev ) ;
int ret ;
int led_brightness_lsb = ( brt_val > > 5 ) ;
mutex_lock ( & led - > lock ) ;
2020-01-06 16:48:53 +01:00
if ( brt_val = = 0 ) {
ret = lm3692x_leds_disable ( led ) ;
goto out ;
} else {
lm3692x_leds_enable ( led ) ;
}
2020-01-06 16:48:51 +01:00
ret = lm3692x_fault_check ( led ) ;
if ( ret ) {
dev_err ( & led - > client - > dev , " Cannot read/clear faults: %d \n " ,
ret ) ;
goto out ;
}
ret = regmap_write ( led - > regmap , LM3692X_BRT_MSB , brt_val ) ;
if ( ret ) {
dev_err ( & led - > client - > dev , " Cannot write MSB: %d \n " , ret ) ;
goto out ;
}
ret = regmap_write ( led - > regmap , LM3692X_BRT_LSB , led_brightness_lsb ) ;
if ( ret ) {
dev_err ( & led - > client - > dev , " Cannot write LSB: %d \n " , ret ) ;
goto out ;
}
out :
mutex_unlock ( & led - > lock ) ;
return ret ;
}
2020-01-06 16:48:50 +01:00
static enum led_brightness lm3692x_max_brightness ( struct lm3692x_led * led ,
u32 max_cur )
{
u32 max_code ;
/* see p.12 of LM36922 data sheet for brightness formula */
max_code = ( ( max_cur * 1000 ) - 37806 ) / 12195 ;
if ( max_code > 0x7FF )
max_code = 0x7FF ;
return max_code > > 3 ;
}
2018-07-02 13:12:16 -05:00
static int lm3692x_probe_dt ( struct lm3692x_led * led )
2017-12-12 12:15:50 -06:00
{
2018-07-02 13:12:14 -05:00
struct fwnode_handle * child = NULL ;
2019-06-09 20:19:04 +02:00
struct led_init_data init_data = { } ;
2020-01-06 16:48:50 +01:00
u32 ovp , max_cur ;
2018-07-02 13:12:14 -05:00
int ret ;
2017-12-12 12:15:50 -06:00
2018-07-02 13:12:16 -05:00
led - > enable_gpio = devm_gpiod_get_optional ( & led - > client - > dev ,
2017-12-12 12:15:50 -06:00
" enable " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( led - > enable_gpio ) ) {
ret = PTR_ERR ( led - > enable_gpio ) ;
2018-07-02 13:12:16 -05:00
dev_err ( & led - > client - > dev , " Failed to get enable gpio: %d \n " ,
ret ) ;
2017-12-12 12:15:50 -06:00
return ret ;
}
2019-09-21 14:12:10 -07:00
led - > regulator = devm_regulator_get_optional ( & led - > client - > dev , " vled " ) ;
if ( IS_ERR ( led - > regulator ) ) {
ret = PTR_ERR ( led - > regulator ) ;
2020-08-26 16:50:10 +02:00
if ( ret ! = - ENODEV )
return dev_err_probe ( & led - > client - > dev , ret ,
" Failed to get vled regulator \n " ) ;
2017-12-12 12:15:50 -06:00
led - > regulator = NULL ;
2019-09-21 14:12:10 -07:00
}
2017-12-12 12:15:50 -06:00
2020-01-04 11:54:18 +01:00
led - > boost_ctrl = LM3692X_BOOST_SW_1MHZ |
LM3692X_BOOST_SW_NO_SHIFT |
LM3692X_OCP_PROT_1_5A ;
ret = device_property_read_u32 ( & led - > client - > dev ,
" ti,ovp-microvolt " , & ovp ) ;
if ( ret ) {
led - > boost_ctrl | = LM3692X_OVP_29V ;
} else {
switch ( ovp ) {
case 17000000 :
break ;
case 21000000 :
led - > boost_ctrl | = LM3692X_OVP_21V ;
break ;
case 25000000 :
led - > boost_ctrl | = LM3692X_OVP_25V ;
break ;
case 29000000 :
led - > boost_ctrl | = LM3692X_OVP_29V ;
break ;
default :
dev_err ( & led - > client - > dev , " Invalid OVP %d \n " , ovp ) ;
return - EINVAL ;
}
}
2018-07-02 13:12:14 -05:00
child = device_get_next_child_node ( & led - > client - > dev , child ) ;
if ( ! child ) {
dev_err ( & led - > client - > dev , " No LED Child node \n " ) ;
2018-07-02 13:12:16 -05:00
return - ENODEV ;
2018-07-02 13:12:14 -05:00
}
2018-07-02 13:12:16 -05:00
ret = fwnode_property_read_u32 ( child , " reg " , & led - > led_enable ) ;
if ( ret ) {
2021-05-10 12:50:35 +03:00
fwnode_handle_put ( child ) ;
2018-07-02 13:12:16 -05:00
dev_err ( & led - > client - > dev , " reg DT property missing \n " ) ;
return ret ;
}
2020-01-06 16:48:50 +01:00
ret = fwnode_property_read_u32 ( child , " led-max-microamp " , & max_cur ) ;
led - > led_dev . max_brightness = ret ? LED_FULL :
lm3692x_max_brightness ( led , max_cur ) ;
2019-06-09 20:19:04 +02:00
init_data . fwnode = child ;
init_data . devicename = led - > client - > name ;
init_data . default_label = " : " ;
2018-07-02 13:12:14 -05:00
2019-06-09 20:19:04 +02:00
ret = devm_led_classdev_register_ext ( & led - > client - > dev , & led - > led_dev ,
& init_data ) ;
2021-05-10 12:50:35 +03:00
if ( ret )
2018-07-02 13:12:16 -05:00
dev_err ( & led - > client - > dev , " led register err: %d \n " , ret ) ;
2021-05-10 12:50:35 +03:00
fwnode_handle_put ( init_data . fwnode ) ;
return ret ;
2018-07-02 13:12:16 -05:00
}
2022-11-18 23:40:09 +01:00
static int lm3692x_probe ( struct i2c_client * client )
2018-07-02 13:12:16 -05:00
{
2022-11-18 23:40:09 +01:00
const struct i2c_device_id * id = i2c_client_get_device_id ( client ) ;
2018-07-02 13:12:16 -05:00
struct lm3692x_led * led ;
int ret ;
led = devm_kzalloc ( & client - > dev , sizeof ( * led ) , GFP_KERNEL ) ;
if ( ! led )
return - ENOMEM ;
mutex_init ( & led - > lock ) ;
led - > client = client ;
led - > led_dev . brightness_set_blocking = lm3692x_brightness_set ;
led - > model_id = id - > driver_data ;
i2c_set_clientdata ( client , led ) ;
led - > regmap = devm_regmap_init_i2c ( client , & lm3692x_regmap_config ) ;
if ( IS_ERR ( led - > regmap ) ) {
ret = PTR_ERR ( led - > regmap ) ;
dev_err ( & client - > dev , " Failed to allocate register map: %d \n " ,
ret ) ;
2017-12-12 12:15:50 -06:00
return ret ;
}
2018-07-02 13:12:16 -05:00
ret = lm3692x_probe_dt ( led ) ;
if ( ret )
return ret ;
2020-01-06 16:48:51 +01:00
ret = lm3692x_leds_enable ( led ) ;
2018-07-02 13:12:16 -05:00
if ( ret )
return ret ;
2017-12-12 12:15:50 -06:00
return 0 ;
}
2022-08-15 10:02:30 +02:00
static void lm3692x_remove ( struct i2c_client * client )
2017-12-12 12:15:50 -06:00
{
struct lm3692x_led * led = i2c_get_clientdata ( client ) ;
2022-02-06 23:08:12 +01:00
lm3692x_leds_disable ( led ) ;
2017-12-12 12:15:50 -06:00
mutex_destroy ( & led - > lock ) ;
}
static const struct i2c_device_id lm3692x_id [ ] = {
2018-07-02 13:12:16 -05:00
{ " lm36922 " , LM36922_MODEL } ,
{ " lm36923 " , LM36923_MODEL } ,
2017-12-12 12:15:50 -06:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm3692x_id ) ;
static const struct of_device_id of_lm3692x_leds_match [ ] = {
{ . compatible = " ti,lm36922 " , } ,
{ . compatible = " ti,lm36923 " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , of_lm3692x_leds_match ) ;
static struct i2c_driver lm3692x_driver = {
. driver = {
. name = " lm3692x " ,
. of_match_table = of_lm3692x_leds_match ,
} ,
2023-05-17 20:05:59 +02:00
. probe = lm3692x_probe ,
2017-12-12 12:15:50 -06:00
. remove = lm3692x_remove ,
. id_table = lm3692x_id ,
} ;
module_i2c_driver ( lm3692x_driver ) ;
MODULE_DESCRIPTION ( " Texas Instruments LM3692X LED driver " ) ;
MODULE_AUTHOR ( " Dan Murphy <dmurphy@ti.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;