2019-05-20 09:19:02 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2011-07-08 10:43:57 -07:00
/*
2017-11-28 18:33:06 -08:00
* Hardware monitoring driver for LM25056 / LM25066 / LM5064 / LM5066
2011-07-08 10:43:57 -07:00
*
* Copyright ( c ) 2011 Ericsson AB .
2013-02-06 09:55:37 -08:00
* Copyright ( c ) 2013 Guenter Roeck
2011-07-08 10:43:57 -07:00
*/
2015-08-17 16:17:24 -07:00
# include <linux/bitops.h>
2011-07-08 10:43:57 -07:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/err.h>
# include <linux/slab.h>
# include <linux/i2c.h>
2019-04-14 21:59:19 +00:00
# include <linux/log2.h>
2021-09-28 02:22:40 -07:00
# include <linux/of_device.h>
2011-07-08 10:43:57 -07:00
# include "pmbus.h"
2017-11-28 18:33:06 -08:00
enum chips { lm25056 , lm25066 , lm5064 , lm5066 , lm5066i } ;
2011-07-08 10:43:57 -07:00
# define LM25066_READ_VAUX 0xd0
# define LM25066_MFR_READ_IIN 0xd1
# define LM25066_MFR_READ_PIN 0xd2
# define LM25066_MFR_IIN_OC_WARN_LIMIT 0xd3
# define LM25066_MFR_PIN_OP_WARN_LIMIT 0xd4
# define LM25066_READ_PIN_PEAK 0xd5
# define LM25066_CLEAR_PIN_PEAK 0xd6
# define LM25066_DEVICE_SETUP 0xd9
# define LM25066_READ_AVG_VIN 0xdc
2019-04-14 21:59:19 +00:00
# define LM25066_SAMPLES_FOR_AVG 0xdb
2011-07-08 10:43:57 -07:00
# define LM25066_READ_AVG_VOUT 0xdd
# define LM25066_READ_AVG_IIN 0xde
# define LM25066_READ_AVG_PIN 0xdf
2015-08-17 16:17:24 -07:00
# define LM25066_DEV_SETUP_CL BIT(4) /* Current limit */
2011-07-08 10:43:57 -07:00
2019-04-14 21:59:19 +00:00
# define LM25066_SAMPLES_FOR_AVG_MAX 4096
2013-02-09 15:15:52 -08:00
/* LM25056 only */
# define LM25056_VAUX_OV_WARN_LIMIT 0xe3
# define LM25056_VAUX_UV_WARN_LIMIT 0xe4
2015-08-17 16:17:24 -07:00
# define LM25056_MFR_STS_VAUX_OV_WARN BIT(1)
# define LM25056_MFR_STS_VAUX_UV_WARN BIT(0)
2013-02-09 15:15:52 -08:00
2013-02-09 14:26:51 -08:00
struct __coeff {
short m , b , R ;
} ;
# define PSC_CURRENT_IN_L (PSC_NUM_CLASSES)
# define PSC_POWER_L (PSC_NUM_CLASSES + 1)
2021-09-28 02:22:39 -07:00
static const struct __coeff lm25066_coeff [ ] [ PSC_NUM_CLASSES + 2 ] = {
2013-02-09 15:15:52 -08:00
[ lm25056 ] = {
[ PSC_VOLTAGE_IN ] = {
. m = 16296 ,
2021-09-28 02:22:35 -07:00
. b = 1343 ,
2013-02-09 15:15:52 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN ] = {
. m = 13797 ,
2021-09-28 02:22:35 -07:00
. b = - 1833 ,
2013-02-09 15:15:52 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN_L ] = {
. m = 6726 ,
2021-09-28 02:22:35 -07:00
. b = - 537 ,
2013-02-09 15:15:52 -08:00
. R = - 2 ,
} ,
[ PSC_POWER ] = {
. m = 5501 ,
2021-09-28 02:22:35 -07:00
. b = - 2908 ,
2013-02-09 15:15:52 -08:00
. R = - 3 ,
} ,
[ PSC_POWER_L ] = {
. m = 26882 ,
2021-09-28 02:22:35 -07:00
. b = - 5646 ,
2013-02-09 15:15:52 -08:00
. R = - 4 ,
} ,
[ PSC_TEMPERATURE ] = {
. m = 1580 ,
. b = - 14500 ,
. R = - 2 ,
} ,
} ,
2013-02-09 14:26:51 -08:00
[ lm25066 ] = {
[ PSC_VOLTAGE_IN ] = {
. m = 22070 ,
2021-09-28 02:22:35 -07:00
. b = - 1800 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_VOLTAGE_OUT ] = {
. m = 22070 ,
2021-09-28 02:22:35 -07:00
. b = - 1800 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN ] = {
. m = 13661 ,
2021-09-28 02:22:35 -07:00
. b = - 5200 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN_L ] = {
2021-09-28 02:22:36 -07:00
. m = 6854 ,
2021-09-28 02:22:35 -07:00
. b = - 3100 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_POWER ] = {
. m = 736 ,
2021-09-28 02:22:35 -07:00
. b = - 3300 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_POWER_L ] = {
. m = 369 ,
2021-09-28 02:22:35 -07:00
. b = - 1900 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_TEMPERATURE ] = {
. m = 16 ,
} ,
} ,
[ lm5064 ] = {
[ PSC_VOLTAGE_IN ] = {
. m = 4611 ,
2021-09-28 02:22:35 -07:00
. b = - 642 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_VOLTAGE_OUT ] = {
. m = 4621 ,
2021-09-28 02:22:35 -07:00
. b = 423 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN ] = {
. m = 10742 ,
2021-09-28 02:22:35 -07:00
. b = 1552 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN_L ] = {
. m = 5456 ,
2021-09-28 02:22:35 -07:00
. b = 2118 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_POWER ] = {
. m = 1204 ,
2021-09-28 02:22:35 -07:00
. b = 8524 ,
2013-02-09 14:26:51 -08:00
. R = - 3 ,
} ,
[ PSC_POWER_L ] = {
. m = 612 ,
2021-09-28 02:22:35 -07:00
. b = 11202 ,
2013-02-09 14:26:51 -08:00
. R = - 3 ,
} ,
[ PSC_TEMPERATURE ] = {
. m = 16 ,
} ,
} ,
[ lm5066 ] = {
[ PSC_VOLTAGE_IN ] = {
. m = 4587 ,
2021-09-28 02:22:35 -07:00
. b = - 1200 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_VOLTAGE_OUT ] = {
. m = 4587 ,
2021-09-28 02:22:35 -07:00
. b = - 2400 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN ] = {
. m = 10753 ,
2021-09-28 02:22:35 -07:00
. b = - 1200 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_CURRENT_IN_L ] = {
. m = 5405 ,
2021-09-28 02:22:35 -07:00
. b = - 600 ,
2013-02-09 14:26:51 -08:00
. R = - 2 ,
} ,
[ PSC_POWER ] = {
. m = 1204 ,
2021-09-28 02:22:35 -07:00
. b = - 6000 ,
2013-02-09 14:26:51 -08:00
. R = - 3 ,
} ,
[ PSC_POWER_L ] = {
. m = 605 ,
2021-09-28 02:22:35 -07:00
. b = - 8000 ,
2013-02-09 14:26:51 -08:00
. R = - 3 ,
} ,
[ PSC_TEMPERATURE ] = {
. m = 16 ,
} ,
} ,
2017-08-29 14:21:17 -07:00
[ lm5066i ] = {
[ PSC_VOLTAGE_IN ] = {
. m = 4617 ,
. b = - 140 ,
. R = - 2 ,
} ,
[ PSC_VOLTAGE_OUT ] = {
. m = 4602 ,
. b = 500 ,
. R = - 2 ,
} ,
[ PSC_CURRENT_IN ] = {
. m = 15076 ,
. b = - 504 ,
. R = - 2 ,
} ,
[ PSC_CURRENT_IN_L ] = {
. m = 7645 ,
. b = 100 ,
. R = - 2 ,
} ,
[ PSC_POWER ] = {
. m = 1701 ,
. b = - 4000 ,
. R = - 3 ,
} ,
[ PSC_POWER_L ] = {
. m = 861 ,
. b = - 965 ,
. R = - 3 ,
} ,
[ PSC_TEMPERATURE ] = {
. m = 16 ,
} ,
} ,
2013-02-09 14:26:51 -08:00
} ;
2011-07-08 10:43:57 -07:00
struct lm25066_data {
int id ;
2013-03-16 10:25:04 -07:00
u16 rlimit ; /* Maximum register value */
2011-07-08 10:43:57 -07:00
struct pmbus_driver_info info ;
} ;
# define to_lm25066_data(x) container_of(x, struct lm25066_data, info)
2020-01-14 09:49:27 -08:00
static int lm25066_read_word_data ( struct i2c_client * client , int page ,
int phase , int reg )
2011-07-08 10:43:57 -07:00
{
const struct pmbus_driver_info * info = pmbus_get_driver_info ( client ) ;
const struct lm25066_data * data = to_lm25066_data ( info ) ;
int ret ;
2013-02-06 09:55:37 -08:00
switch ( reg ) {
case PMBUS_VIRT_READ_VMON :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff , LM25066_READ_VAUX ) ;
2013-02-06 09:55:37 -08:00
if ( ret < 0 )
break ;
/* Adjust returned value to match VIN coefficients */
switch ( data - > id ) {
2013-02-09 15:15:52 -08:00
case lm25056 :
/* VIN: 6.14 mV VAUX: 293 uV LSB */
ret = DIV_ROUND_CLOSEST ( ret * 293 , 6140 ) ;
break ;
2013-02-06 09:55:37 -08:00
case lm25066 :
/* VIN: 4.54 mV VAUX: 283.2 uV LSB */
ret = DIV_ROUND_CLOSEST ( ret * 2832 , 45400 ) ;
2011-07-08 10:43:57 -07:00
break ;
2013-02-06 09:55:37 -08:00
case lm5064 :
/* VIN: 4.53 mV VAUX: 700 uV LSB */
ret = DIV_ROUND_CLOSEST ( ret * 70 , 453 ) ;
break ;
case lm5066 :
2017-08-29 14:21:17 -07:00
case lm5066i :
2013-02-06 09:55:37 -08:00
/* VIN: 2.18 mV VAUX: 725 uV LSB */
ret = DIV_ROUND_CLOSEST ( ret * 725 , 2180 ) ;
2011-07-08 10:43:57 -07:00
break ;
}
2013-02-06 09:55:37 -08:00
break ;
2011-07-08 10:43:57 -07:00
case PMBUS_READ_IIN :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_MFR_READ_IIN ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_READ_PIN :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_MFR_READ_PIN ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_IIN_OC_WARN_LIMIT :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
2011-07-08 10:43:57 -07:00
LM25066_MFR_IIN_OC_WARN_LIMIT ) ;
break ;
case PMBUS_PIN_OP_WARN_LIMIT :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
2011-07-08 10:43:57 -07:00
LM25066_MFR_PIN_OP_WARN_LIMIT ) ;
break ;
case PMBUS_VIRT_READ_VIN_AVG :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_READ_AVG_VIN ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_VIRT_READ_VOUT_AVG :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_READ_AVG_VOUT ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_VIRT_READ_IIN_AVG :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_READ_AVG_IIN ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_VIRT_READ_PIN_AVG :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_READ_AVG_PIN ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_VIRT_READ_PIN_MAX :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
LM25066_READ_PIN_PEAK ) ;
2011-07-08 10:43:57 -07:00
break ;
case PMBUS_VIRT_RESET_PIN_HISTORY :
ret = 0 ;
break ;
2019-04-14 21:59:19 +00:00
case PMBUS_VIRT_SAMPLES :
ret = pmbus_read_byte_data ( client , 0 , LM25066_SAMPLES_FOR_AVG ) ;
if ( ret < 0 )
break ;
ret = 1 < < ret ;
break ;
2011-07-08 10:43:57 -07:00
default :
ret = - ENODATA ;
break ;
}
return ret ;
}
2020-01-14 09:49:27 -08:00
static int lm25056_read_word_data ( struct i2c_client * client , int page ,
int phase , int reg )
2013-02-09 15:15:52 -08:00
{
int ret ;
switch ( reg ) {
case PMBUS_VIRT_VMON_UV_WARN_LIMIT :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
2013-02-09 15:15:52 -08:00
LM25056_VAUX_UV_WARN_LIMIT ) ;
if ( ret < 0 )
break ;
/* Adjust returned value to match VIN coefficients */
ret = DIV_ROUND_CLOSEST ( ret * 293 , 6140 ) ;
break ;
case PMBUS_VIRT_VMON_OV_WARN_LIMIT :
2020-01-14 09:49:27 -08:00
ret = pmbus_read_word_data ( client , 0 , 0xff ,
2013-02-09 15:15:52 -08:00
LM25056_VAUX_OV_WARN_LIMIT ) ;
if ( ret < 0 )
break ;
/* Adjust returned value to match VIN coefficients */
ret = DIV_ROUND_CLOSEST ( ret * 293 , 6140 ) ;
break ;
default :
2020-01-14 09:49:27 -08:00
ret = lm25066_read_word_data ( client , page , phase , reg ) ;
2013-02-09 15:15:52 -08:00
break ;
}
return ret ;
}
static int lm25056_read_byte_data ( struct i2c_client * client , int page , int reg )
{
int ret , s ;
switch ( reg ) {
case PMBUS_VIRT_STATUS_VMON :
ret = pmbus_read_byte_data ( client , 0 ,
PMBUS_STATUS_MFR_SPECIFIC ) ;
if ( ret < 0 )
break ;
s = 0 ;
if ( ret & LM25056_MFR_STS_VAUX_UV_WARN )
s | = PB_VOLTAGE_UV_WARNING ;
if ( ret & LM25056_MFR_STS_VAUX_OV_WARN )
s | = PB_VOLTAGE_OV_WARNING ;
ret = s ;
break ;
default :
ret = - ENODATA ;
break ;
}
return ret ;
}
2011-07-08 10:43:57 -07:00
static int lm25066_write_word_data ( struct i2c_client * client , int page , int reg ,
u16 word )
{
2013-03-16 10:25:04 -07:00
const struct pmbus_driver_info * info = pmbus_get_driver_info ( client ) ;
const struct lm25066_data * data = to_lm25066_data ( info ) ;
2011-07-08 10:43:57 -07:00
int ret ;
switch ( reg ) {
2013-03-16 10:25:04 -07:00
case PMBUS_POUT_OP_FAULT_LIMIT :
case PMBUS_POUT_OP_WARN_LIMIT :
2013-02-06 20:49:12 -08:00
case PMBUS_VOUT_UV_WARN_LIMIT :
case PMBUS_OT_FAULT_LIMIT :
case PMBUS_OT_WARN_LIMIT :
2013-03-16 10:25:04 -07:00
case PMBUS_IIN_OC_FAULT_LIMIT :
2013-02-06 20:49:12 -08:00
case PMBUS_VIN_UV_WARN_LIMIT :
2013-03-16 10:25:04 -07:00
case PMBUS_VIN_UV_FAULT_LIMIT :
case PMBUS_VIN_OV_FAULT_LIMIT :
2013-02-06 20:49:12 -08:00
case PMBUS_VIN_OV_WARN_LIMIT :
2013-03-16 10:25:04 -07:00
word = ( ( s16 ) word < 0 ) ? 0 : clamp_val ( word , 0 , data - > rlimit ) ;
2013-02-06 20:49:12 -08:00
ret = pmbus_write_word_data ( client , 0 , reg , word ) ;
break ;
2011-07-08 10:43:57 -07:00
case PMBUS_IIN_OC_WARN_LIMIT :
2013-03-16 10:25:04 -07:00
word = ( ( s16 ) word < 0 ) ? 0 : clamp_val ( word , 0 , data - > rlimit ) ;
2011-07-08 10:43:57 -07:00
ret = pmbus_write_word_data ( client , 0 ,
LM25066_MFR_IIN_OC_WARN_LIMIT ,
word ) ;
break ;
case PMBUS_PIN_OP_WARN_LIMIT :
2013-03-16 10:25:04 -07:00
word = ( ( s16 ) word < 0 ) ? 0 : clamp_val ( word , 0 , data - > rlimit ) ;
2011-07-08 10:43:57 -07:00
ret = pmbus_write_word_data ( client , 0 ,
LM25066_MFR_PIN_OP_WARN_LIMIT ,
word ) ;
break ;
2013-02-09 15:15:52 -08:00
case PMBUS_VIRT_VMON_UV_WARN_LIMIT :
/* Adjust from VIN coefficients (for LM25056) */
word = DIV_ROUND_CLOSEST ( ( int ) word * 6140 , 293 ) ;
2013-03-16 10:25:04 -07:00
word = ( ( s16 ) word < 0 ) ? 0 : clamp_val ( word , 0 , data - > rlimit ) ;
2013-02-09 15:15:52 -08:00
ret = pmbus_write_word_data ( client , 0 ,
LM25056_VAUX_UV_WARN_LIMIT , word ) ;
break ;
case PMBUS_VIRT_VMON_OV_WARN_LIMIT :
/* Adjust from VIN coefficients (for LM25056) */
word = DIV_ROUND_CLOSEST ( ( int ) word * 6140 , 293 ) ;
2013-03-16 10:25:04 -07:00
word = ( ( s16 ) word < 0 ) ? 0 : clamp_val ( word , 0 , data - > rlimit ) ;
2013-02-09 15:15:52 -08:00
ret = pmbus_write_word_data ( client , 0 ,
LM25056_VAUX_OV_WARN_LIMIT , word ) ;
break ;
2011-07-08 10:43:57 -07:00
case PMBUS_VIRT_RESET_PIN_HISTORY :
ret = pmbus_write_byte ( client , 0 , LM25066_CLEAR_PIN_PEAK ) ;
break ;
2019-04-14 21:59:19 +00:00
case PMBUS_VIRT_SAMPLES :
word = clamp_val ( word , 1 , LM25066_SAMPLES_FOR_AVG_MAX ) ;
ret = pmbus_write_byte_data ( client , 0 , LM25066_SAMPLES_FOR_AVG ,
ilog2 ( word ) ) ;
break ;
2011-07-08 10:43:57 -07:00
default :
ret = - ENODATA ;
break ;
}
return ret ;
}
2021-09-28 02:22:37 -07:00
static const struct i2c_device_id lm25066_id [ ] = {
{ " lm25056 " , lm25056 } ,
{ " lm25066 " , lm25066 } ,
{ " lm5064 " , lm5064 } ,
{ " lm5066 " , lm5066 } ,
{ " lm5066i " , lm5066i } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm25066_id ) ;
2021-09-28 02:22:40 -07:00
static const struct of_device_id __maybe_unused lm25066_of_match [ ] = {
{ . compatible = " ti,lm25056 " , . data = ( void * ) lm25056 , } ,
{ . compatible = " ti,lm25066 " , . data = ( void * ) lm25066 , } ,
{ . compatible = " ti,lm5064 " , . data = ( void * ) lm5064 , } ,
{ . compatible = " ti,lm5066 " , . data = ( void * ) lm5066 , } ,
{ . compatible = " ti,lm5066i " , . data = ( void * ) lm5066i , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , lm25066_of_match ) ;
2020-08-08 23:00:04 +02:00
static int lm25066_probe ( struct i2c_client * client )
2011-07-08 10:43:57 -07:00
{
int config ;
2021-09-28 02:22:41 -07:00
u32 shunt ;
2011-07-08 10:43:57 -07:00
struct lm25066_data * data ;
struct pmbus_driver_info * info ;
2021-09-28 02:22:39 -07:00
const struct __coeff * coeff ;
2021-09-28 02:22:40 -07:00
const struct of_device_id * of_id ;
const struct i2c_device_id * i2c_id ;
2011-07-08 10:43:57 -07:00
if ( ! i2c_check_functionality ( client - > adapter ,
I2C_FUNC_SMBUS_READ_BYTE_DATA ) )
return - ENODEV ;
2012-02-22 08:56:43 -08:00
data = devm_kzalloc ( & client - > dev , sizeof ( struct lm25066_data ) ,
GFP_KERNEL ) ;
2011-07-08 10:43:57 -07:00
if ( ! data )
return - ENOMEM ;
config = i2c_smbus_read_byte_data ( client , LM25066_DEVICE_SETUP ) ;
2012-02-22 08:56:43 -08:00
if ( config < 0 )
return config ;
2011-07-08 10:43:57 -07:00
2021-09-28 02:22:40 -07:00
i2c_id = i2c_match_id ( lm25066_id , client ) ;
of_id = of_match_device ( lm25066_of_match , & client - > dev ) ;
if ( of_id & & ( unsigned long ) of_id - > data ! = i2c_id - > driver_data )
dev_notice ( & client - > dev , " Device mismatch: %s in device tree, %s detected \n " ,
of_id - > name , i2c_id - > name ) ;
data - > id = i2c_id - > driver_data ;
2011-07-08 10:43:57 -07:00
info = & data - > info ;
2013-02-06 09:55:37 -08:00
info - > pages = 1 ;
2011-07-08 10:43:57 -07:00
info - > format [ PSC_VOLTAGE_IN ] = direct ;
info - > format [ PSC_VOLTAGE_OUT ] = direct ;
info - > format [ PSC_CURRENT_IN ] = direct ;
info - > format [ PSC_TEMPERATURE ] = direct ;
info - > format [ PSC_POWER ] = direct ;
2013-02-09 15:15:52 -08:00
info - > func [ 0 ] = PMBUS_HAVE_VIN | PMBUS_HAVE_VMON
| PMBUS_HAVE_PIN | PMBUS_HAVE_IIN | PMBUS_HAVE_STATUS_INPUT
2019-04-14 21:59:19 +00:00
| PMBUS_HAVE_TEMP | PMBUS_HAVE_STATUS_TEMP | PMBUS_HAVE_SAMPLES ;
2011-07-08 10:43:57 -07:00
2013-02-09 15:15:52 -08:00
if ( data - > id = = lm25056 ) {
info - > func [ 0 ] | = PMBUS_HAVE_STATUS_VMON ;
info - > read_word_data = lm25056_read_word_data ;
info - > read_byte_data = lm25056_read_byte_data ;
2013-03-16 10:25:04 -07:00
data - > rlimit = 0x0fff ;
2013-02-09 15:15:52 -08:00
} else {
info - > func [ 0 ] | = PMBUS_HAVE_VOUT | PMBUS_HAVE_STATUS_VOUT ;
info - > read_word_data = lm25066_read_word_data ;
2013-03-16 10:25:04 -07:00
data - > rlimit = 0x0fff ;
2013-02-09 15:15:52 -08:00
}
2011-07-08 10:43:57 -07:00
info - > write_word_data = lm25066_write_word_data ;
2013-02-09 14:26:51 -08:00
coeff = & lm25066_coeff [ data - > id ] [ 0 ] ;
info - > m [ PSC_TEMPERATURE ] = coeff [ PSC_TEMPERATURE ] . m ;
info - > b [ PSC_TEMPERATURE ] = coeff [ PSC_TEMPERATURE ] . b ;
info - > R [ PSC_TEMPERATURE ] = coeff [ PSC_TEMPERATURE ] . R ;
info - > m [ PSC_VOLTAGE_IN ] = coeff [ PSC_VOLTAGE_IN ] . m ;
info - > b [ PSC_VOLTAGE_IN ] = coeff [ PSC_VOLTAGE_IN ] . b ;
info - > R [ PSC_VOLTAGE_IN ] = coeff [ PSC_VOLTAGE_IN ] . R ;
info - > m [ PSC_VOLTAGE_OUT ] = coeff [ PSC_VOLTAGE_OUT ] . m ;
info - > b [ PSC_VOLTAGE_OUT ] = coeff [ PSC_VOLTAGE_OUT ] . b ;
info - > R [ PSC_VOLTAGE_OUT ] = coeff [ PSC_VOLTAGE_OUT ] . R ;
info - > R [ PSC_CURRENT_IN ] = coeff [ PSC_CURRENT_IN ] . R ;
info - > R [ PSC_POWER ] = coeff [ PSC_POWER ] . R ;
if ( config & LM25066_DEV_SETUP_CL ) {
info - > m [ PSC_CURRENT_IN ] = coeff [ PSC_CURRENT_IN_L ] . m ;
2017-08-29 14:21:16 -07:00
info - > b [ PSC_CURRENT_IN ] = coeff [ PSC_CURRENT_IN_L ] . b ;
2013-02-09 14:26:51 -08:00
info - > m [ PSC_POWER ] = coeff [ PSC_POWER_L ] . m ;
2017-08-29 14:21:16 -07:00
info - > b [ PSC_POWER ] = coeff [ PSC_POWER_L ] . b ;
2013-02-09 14:26:51 -08:00
} else {
info - > m [ PSC_CURRENT_IN ] = coeff [ PSC_CURRENT_IN ] . m ;
2017-08-29 14:21:16 -07:00
info - > b [ PSC_CURRENT_IN ] = coeff [ PSC_CURRENT_IN ] . b ;
2013-02-09 14:26:51 -08:00
info - > m [ PSC_POWER ] = coeff [ PSC_POWER ] . m ;
2017-08-29 14:21:16 -07:00
info - > b [ PSC_POWER ] = coeff [ PSC_POWER ] . b ;
2011-07-08 10:43:57 -07:00
}
2021-09-28 02:22:41 -07:00
/*
* Values in the TI datasheets are normalized for a 1 mOhm sense
* resistor ; assume that unless DT specifies a value explicitly .
*/
if ( of_property_read_u32 ( client - > dev . of_node , " shunt-resistor-micro-ohms " , & shunt ) )
shunt = 1000 ;
info - > m [ PSC_CURRENT_IN ] = info - > m [ PSC_CURRENT_IN ] * shunt / 1000 ;
info - > m [ PSC_POWER ] = info - > m [ PSC_POWER ] * shunt / 1000 ;
2020-08-08 23:00:04 +02:00
return pmbus_do_probe ( client , info ) ;
2011-07-08 10:43:57 -07:00
}
/* This is the driver that will be inserted */
static struct i2c_driver lm25066_driver = {
. driver = {
. name = " lm25066 " ,
2021-09-28 02:22:40 -07:00
. of_match_table = of_match_ptr ( lm25066_of_match ) ,
} ,
2020-08-08 23:00:04 +02:00
. probe_new = lm25066_probe ,
2011-07-08 10:43:57 -07:00
. id_table = lm25066_id ,
} ;
2012-01-20 15:38:18 +08:00
module_i2c_driver ( lm25066_driver ) ;
2011-07-08 10:43:57 -07:00
MODULE_AUTHOR ( " Guenter Roeck " ) ;
2013-03-16 10:25:04 -07:00
MODULE_DESCRIPTION ( " PMBus driver for LM25066 and compatible chips " ) ;
2011-07-08 10:43:57 -07:00
MODULE_LICENSE ( " GPL " ) ;
2021-04-19 23:07:07 -07:00
MODULE_IMPORT_NS ( PMBUS ) ;