2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
* lm83 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
2014-01-29 20:40:08 +01:00
* Copyright ( C ) 2003 - 2009 Jean Delvare < jdelvare @ suse . de >
2005-04-16 15:20:36 -07:00
*
* Heavily inspired from the lm78 , lm75 and adm1021 drivers . The LM83 is
* a sensor chip made by National Semiconductor . It reports up to four
* temperatures ( its own plus up to three external ones ) with a 1 deg
* resolution and a 3 - 4 deg accuracy . Complete datasheet can be obtained
* from National ' s website at :
* http : //www.national.com/pf/LM/LM83.html
* Since the datasheet omits to give the chip stepping code , I give it
* here : 0x03 ( at register 0xff ) .
*
2006-03-23 16:19:49 +01:00
* Also supports the LM82 temp sensor , which is basically a stripped down
* model of the LM83 . Datasheet is here :
* http : //www.national.com/pf/LM/LM82.html
2005-04-16 15:20:36 -07:00
*/
2021-12-22 20:24:02 -08:00
# include <linux/bits.h>
2021-12-22 11:42:38 -08:00
# include <linux/err.h>
# include <linux/i2c.h>
2005-04-16 15:20:36 -07:00
# include <linux/init.h>
2005-07-15 21:39:18 -04:00
# include <linux/hwmon.h>
2021-12-22 11:42:38 -08:00
# include <linux/module.h>
2006-01-18 23:19:26 +01:00
# include <linux/mutex.h>
2021-12-22 16:22:00 -08:00
# include <linux/regmap.h>
2021-12-22 11:42:38 -08:00
# include <linux/slab.h>
2006-09-24 21:16:40 +02:00
# include <linux/sysfs.h>
2005-04-16 15:20:36 -07:00
/*
* Addresses to scan
* Address is selected using 2 three - level pins , resulting in 9 possible
* addresses .
*/
2008-02-17 22:28:03 -05:00
static const unsigned short normal_i2c [ ] = {
0x18 , 0x19 , 0x1a , 0x29 , 0x2a , 0x2b , 0x4c , 0x4d , 0x4e , I2C_CLIENT_END } ;
2005-04-16 15:20:36 -07:00
2009-12-14 21:17:27 +01:00
enum chips { lm83 , lm82 } ;
2005-04-16 15:20:36 -07:00
/*
* The LM83 registers
* Manufacturer ID is 0x01 for National Semiconductor .
*/
# define LM83_REG_R_MAN_ID 0xFE
# define LM83_REG_R_CHIP_ID 0xFF
# define LM83_REG_R_CONFIG 0x03
# define LM83_REG_W_CONFIG 0x09
# define LM83_REG_R_STATUS1 0x02
# define LM83_REG_R_STATUS2 0x35
# define LM83_REG_R_LOCAL_TEMP 0x00
# define LM83_REG_R_LOCAL_HIGH 0x05
# define LM83_REG_W_LOCAL_HIGH 0x0B
# define LM83_REG_R_REMOTE1_TEMP 0x30
# define LM83_REG_R_REMOTE1_HIGH 0x38
# define LM83_REG_W_REMOTE1_HIGH 0x50
# define LM83_REG_R_REMOTE2_TEMP 0x01
# define LM83_REG_R_REMOTE2_HIGH 0x07
# define LM83_REG_W_REMOTE2_HIGH 0x0D
# define LM83_REG_R_REMOTE3_TEMP 0x31
# define LM83_REG_R_REMOTE3_HIGH 0x3A
# define LM83_REG_W_REMOTE3_HIGH 0x52
# define LM83_REG_R_TCRIT 0x42
# define LM83_REG_W_TCRIT 0x5A
2021-12-22 16:22:00 -08:00
static const u8 LM83_REG_TEMP [ ] = {
2005-04-16 15:20:36 -07:00
LM83_REG_R_LOCAL_TEMP ,
LM83_REG_R_REMOTE1_TEMP ,
LM83_REG_R_REMOTE2_TEMP ,
2005-06-05 21:16:39 +02:00
LM83_REG_R_REMOTE3_TEMP ,
2021-12-22 20:24:02 -08:00
} ;
static const u8 LM83_REG_MAX [ ] = {
2005-04-16 15:20:36 -07:00
LM83_REG_R_LOCAL_HIGH ,
LM83_REG_R_REMOTE1_HIGH ,
LM83_REG_R_REMOTE2_HIGH ,
2005-06-05 21:16:39 +02:00
LM83_REG_R_REMOTE3_HIGH ,
2021-12-22 20:24:02 -08:00
} ;
/* alarm and fault registers and bits, indexed by channel */
static const u8 LM83_ALARM_REG [ ] = {
LM83_REG_R_STATUS1 , LM83_REG_R_STATUS2 , LM83_REG_R_STATUS1 , LM83_REG_R_STATUS2
} ;
static const u8 LM83_MAX_ALARM_BIT [ ] = {
BIT ( 6 ) , BIT ( 7 ) , BIT ( 4 ) , BIT ( 4 )
} ;
static const u8 LM83_CRIT_ALARM_BIT [ ] = {
BIT ( 0 ) , BIT ( 0 ) , BIT ( 1 ) , BIT ( 1 )
} ;
static const u8 LM83_FAULT_BIT [ ] = {
0 , BIT ( 5 ) , BIT ( 2 ) , BIT ( 2 )
2005-04-16 15:20:36 -07:00
} ;
/*
* Client data ( each client gets its own )
*/
struct lm83_data {
2021-12-22 16:22:00 -08:00
struct regmap * regmap ;
2021-12-22 20:24:02 -08:00
enum chips type ;
2005-04-16 15:20:36 -07:00
} ;
2021-12-22 16:22:00 -08:00
/* regmap code */
static int lm83_regmap_reg_read ( void * context , unsigned int reg , unsigned int * val )
2014-04-12 12:39:41 -07:00
{
2021-12-22 16:22:00 -08:00
struct i2c_client * client = context ;
int ret ;
ret = i2c_smbus_read_byte_data ( client , reg ) ;
if ( ret < 0 )
return ret ;
* val = ret ;
return 0 ;
}
/*
* The regmap write function maps read register addresses to write register
* addresses . This is necessary for regmap register caching to work .
* An alternative would be to clear the regmap cache whenever a register is
* written , but that would be much more expensive .
*/
static int lm83_regmap_reg_write ( void * context , unsigned int reg , unsigned int val )
{
struct i2c_client * client = context ;
switch ( reg ) {
case LM83_REG_R_CONFIG :
case LM83_REG_R_LOCAL_HIGH :
case LM83_REG_R_REMOTE2_HIGH :
reg + = 0x06 ;
break ;
case LM83_REG_R_REMOTE1_HIGH :
case LM83_REG_R_REMOTE3_HIGH :
case LM83_REG_R_TCRIT :
reg + = 0x18 ;
break ;
default :
break ;
2014-04-12 12:39:41 -07:00
}
2021-12-22 16:22:00 -08:00
return i2c_smbus_write_byte_data ( client , reg , val ) ;
}
2014-04-12 12:39:41 -07:00
2021-12-22 16:22:00 -08:00
static bool lm83_regmap_is_volatile ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case LM83_REG_R_LOCAL_TEMP :
case LM83_REG_R_REMOTE1_TEMP :
case LM83_REG_R_REMOTE2_TEMP :
case LM83_REG_R_REMOTE3_TEMP :
case LM83_REG_R_STATUS1 :
case LM83_REG_R_STATUS2 :
return true ;
default :
return false ;
}
2014-04-12 12:39:41 -07:00
}
2021-12-22 16:22:00 -08:00
static const struct regmap_config lm83_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. cache_type = REGCACHE_RBTREE ,
. volatile_reg = lm83_regmap_is_volatile ,
. reg_read = lm83_regmap_reg_read ,
. reg_write = lm83_regmap_reg_write ,
} ;
2021-12-22 20:24:02 -08:00
/* hwmon API */
2005-04-16 15:20:36 -07:00
2021-12-22 20:24:02 -08:00
static int lm83_temp_read ( struct device * dev , u32 attr , int channel , long * val )
2005-06-05 21:16:39 +02:00
{
2021-12-22 16:22:00 -08:00
struct lm83_data * data = dev_get_drvdata ( dev ) ;
unsigned int regval ;
2021-12-22 20:24:02 -08:00
int err ;
2021-12-22 16:22:00 -08:00
2021-12-22 20:24:02 -08:00
switch ( attr ) {
case hwmon_temp_input :
err = regmap_read ( data - > regmap , LM83_REG_TEMP [ channel ] , & regval ) ;
if ( err < 0 )
return err ;
* val = ( s8 ) regval * 1000 ;
break ;
case hwmon_temp_max :
err = regmap_read ( data - > regmap , LM83_REG_MAX [ channel ] , & regval ) ;
if ( err < 0 )
return err ;
* val = ( s8 ) regval * 1000 ;
break ;
case hwmon_temp_crit :
err = regmap_read ( data - > regmap , LM83_REG_R_TCRIT , & regval ) ;
if ( err < 0 )
return err ;
* val = ( s8 ) regval * 1000 ;
break ;
case hwmon_temp_max_alarm :
err = regmap_read ( data - > regmap , LM83_ALARM_REG [ channel ] , & regval ) ;
if ( err < 0 )
return err ;
* val = ! ! ( regval & LM83_MAX_ALARM_BIT [ channel ] ) ;
break ;
case hwmon_temp_crit_alarm :
err = regmap_read ( data - > regmap , LM83_ALARM_REG [ channel ] , & regval ) ;
if ( err < 0 )
return err ;
* val = ! ! ( regval & LM83_CRIT_ALARM_BIT [ channel ] ) ;
break ;
case hwmon_temp_fault :
err = regmap_read ( data - > regmap , LM83_ALARM_REG [ channel ] , & regval ) ;
if ( err < 0 )
return err ;
* val = ! ! ( regval & LM83_FAULT_BIT [ channel ] ) ;
break ;
default :
return - EOPNOTSUPP ;
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
2005-06-05 21:16:39 +02:00
2021-12-22 20:24:02 -08:00
static int lm83_temp_write ( struct device * dev , u32 attr , int channel , long val )
2005-06-05 21:16:39 +02:00
{
2014-04-12 12:46:34 -07:00
struct lm83_data * data = dev_get_drvdata ( dev ) ;
2021-12-22 16:22:00 -08:00
unsigned int regval ;
2012-01-10 23:01:40 +01:00
int err ;
2021-12-22 19:03:04 -08:00
regval = DIV_ROUND_CLOSEST ( clamp_val ( val , - 128000 , 127000 ) , 1000 ) ;
2021-12-22 20:24:02 -08:00
switch ( attr ) {
case hwmon_temp_max :
err = regmap_write ( data - > regmap , LM83_REG_MAX [ channel ] , regval ) ;
if ( err < 0 )
return err ;
break ;
case hwmon_temp_crit :
err = regmap_write ( data - > regmap , LM83_REG_R_TCRIT , regval ) ;
if ( err < 0 )
return err ;
break ;
default :
return - EOPNOTSUPP ;
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
2021-12-22 20:24:02 -08:00
static int lm83_chip_read ( struct device * dev , u32 attr , int channel , long * val )
2005-04-16 15:20:36 -07:00
{
2021-12-22 16:22:00 -08:00
struct lm83_data * data = dev_get_drvdata ( dev ) ;
2021-12-22 20:24:02 -08:00
unsigned int regval ;
2021-12-22 16:22:00 -08:00
int err ;
2021-12-22 20:24:02 -08:00
switch ( attr ) {
case hwmon_chip_alarms :
err = regmap_read ( data - > regmap , LM83_REG_R_STATUS1 , & regval ) ;
if ( err < 0 )
return err ;
* val = regval ;
err = regmap_read ( data - > regmap , LM83_REG_R_STATUS2 , & regval ) ;
if ( err < 0 )
return err ;
* val | = regval < < 8 ;
return 0 ;
default :
return - EOPNOTSUPP ;
}
2021-12-22 16:22:00 -08:00
2021-12-22 20:24:02 -08:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2021-12-22 20:24:02 -08:00
static int lm83_read ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
2006-09-24 20:52:15 +02:00
{
2021-12-22 20:24:02 -08:00
switch ( type ) {
case hwmon_chip :
return lm83_chip_read ( dev , attr , channel , val ) ;
case hwmon_temp :
return lm83_temp_read ( dev , attr , channel , val ) ;
default :
return - EOPNOTSUPP ;
2021-12-22 16:22:00 -08:00
}
2006-09-24 20:52:15 +02:00
}
2021-12-22 20:24:02 -08:00
static int lm83_write ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long val )
{
switch ( type ) {
case hwmon_temp :
return lm83_temp_write ( dev , attr , channel , val ) ;
default :
return - EOPNOTSUPP ;
}
}
2006-09-24 21:16:40 +02:00
2021-12-22 20:24:02 -08:00
static umode_t lm83_is_visible ( const void * _data , enum hwmon_sensor_types type ,
u32 attr , int channel )
{
const struct lm83_data * data = _data ;
2006-09-24 21:16:40 +02:00
2021-12-22 20:24:02 -08:00
/*
* LM82 only supports a single external channel , modeled as channel 2.
*/
if ( data - > type = = lm82 & & ( channel = = 1 | | channel = = 3 ) )
return 0 ;
switch ( type ) {
case hwmon_chip :
if ( attr = = hwmon_chip_alarms )
return 0444 ;
break ;
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_input :
case hwmon_temp_max_alarm :
case hwmon_temp_crit_alarm :
return 0444 ;
case hwmon_temp_fault :
if ( channel )
return 0444 ;
break ;
case hwmon_temp_max :
return 0644 ;
case hwmon_temp_crit :
if ( channel = = 2 )
return 0644 ;
return 0444 ;
default :
break ;
}
break ;
default :
break ;
}
return 0 ;
}
static const struct hwmon_channel_info * lm83_info [ ] = {
HWMON_CHANNEL_INFO ( chip , HWMON_C_ALARMS ) ,
HWMON_CHANNEL_INFO ( temp ,
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_CRIT |
HWMON_T_MAX_ALARM | HWMON_T_CRIT_ALARM ,
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_CRIT |
HWMON_T_MAX_ALARM | HWMON_T_CRIT_ALARM | HWMON_T_FAULT ,
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_CRIT |
HWMON_T_MAX_ALARM | HWMON_T_CRIT_ALARM | HWMON_T_FAULT ,
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_CRIT |
HWMON_T_MAX_ALARM | HWMON_T_CRIT_ALARM | HWMON_T_FAULT
) ,
2006-09-24 21:16:40 +02:00
NULL
} ;
2021-12-22 20:24:02 -08:00
static const struct hwmon_ops lm83_hwmon_ops = {
. is_visible = lm83_is_visible ,
. read = lm83_read ,
. write = lm83_write ,
2006-09-24 21:16:40 +02:00
} ;
2021-12-22 20:24:02 -08:00
static const struct hwmon_chip_info lm83_chip_info = {
. ops = & lm83_hwmon_ops ,
. info = lm83_info ,
} ;
2005-04-16 15:20:36 -07:00
2008-07-16 19:30:14 +02:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2021-12-22 16:23:31 -08:00
static int lm83_detect ( struct i2c_client * client ,
2008-07-16 19:30:14 +02:00
struct i2c_board_info * info )
2005-04-16 15:20:36 -07:00
{
2021-12-22 16:23:31 -08:00
struct i2c_adapter * adapter = client - > adapter ;
2009-12-09 20:35:52 +01:00
const char * name ;
u8 man_id , chip_id ;
2005-04-16 15:20:36 -07:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
2008-07-16 19:30:14 +02:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
2009-12-09 20:35:52 +01:00
/* Detection */
2021-12-22 16:23:31 -08:00
if ( ( i2c_smbus_read_byte_data ( client , LM83_REG_R_STATUS1 ) & 0xA8 ) | |
( i2c_smbus_read_byte_data ( client , LM83_REG_R_STATUS2 ) & 0x48 ) | |
( i2c_smbus_read_byte_data ( client , LM83_REG_R_CONFIG ) & 0x41 ) ) {
2009-12-09 20:35:52 +01:00
dev_dbg ( & adapter - > dev , " LM83 detection failed at 0x%02x \n " ,
2021-12-22 16:23:31 -08:00
client - > addr ) ;
2009-12-09 20:35:52 +01:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2009-12-09 20:35:52 +01:00
/* Identification */
2021-12-22 16:23:31 -08:00
man_id = i2c_smbus_read_byte_data ( client , LM83_REG_R_MAN_ID ) ;
2009-12-09 20:35:52 +01:00
if ( man_id ! = 0x01 ) /* National Semiconductor */
return - ENODEV ;
2005-04-16 15:20:36 -07:00
2021-12-22 16:23:31 -08:00
chip_id = i2c_smbus_read_byte_data ( client , LM83_REG_R_CHIP_ID ) ;
2009-12-09 20:35:52 +01:00
switch ( chip_id ) {
case 0x03 :
2021-12-24 11:31:32 -08:00
/*
* According to the LM82 datasheet dated March 2013 , recent
* revisions of LM82 have a die revision of 0x03 . This was
* confirmed with a real chip . Further details in this revision
* of the LM82 datasheet strongly suggest that LM82 is just a
* repackaged LM83 . It is therefore impossible to distinguish
* those chips from LM83 , and they will be misdetected as LM83 .
*/
2005-04-16 15:20:36 -07:00
name = " lm83 " ;
2009-12-09 20:35:52 +01:00
break ;
case 0x01 :
2006-03-23 16:19:49 +01:00
name = " lm82 " ;
2009-12-09 20:35:52 +01:00
break ;
default :
/* identification failed */
2021-12-22 23:49:59 -08:00
dev_dbg ( & adapter - > dev ,
" Unsupported chip (man_id=0x%02X, chip_id=0x%02X) \n " ,
man_id , chip_id ) ;
2009-12-09 20:35:52 +01:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2008-07-16 19:30:14 +02:00
strlcpy ( info - > type , name , I2C_NAME_SIZE ) ;
return 0 ;
}
2021-12-22 11:49:04 -08:00
static const struct i2c_device_id lm83_id [ ] = {
{ " lm83 " , lm83 } ,
{ " lm82 " , lm82 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm83_id ) ;
2020-08-13 18:02:22 +02:00
2021-12-22 16:23:31 -08:00
static int lm83_probe ( struct i2c_client * client )
2008-07-16 19:30:14 +02:00
{
2021-12-22 16:22:00 -08:00
struct device * dev = & client - > dev ;
2014-04-12 12:46:34 -07:00
struct device * hwmon_dev ;
2008-07-16 19:30:14 +02:00
struct lm83_data * data ;
2021-12-22 16:22:00 -08:00
data = devm_kzalloc ( dev , sizeof ( struct lm83_data ) , GFP_KERNEL ) ;
2012-06-02 09:58:10 -07:00
if ( ! data )
return - ENOMEM ;
2008-07-16 19:30:14 +02:00
2021-12-22 16:22:00 -08:00
data - > regmap = devm_regmap_init ( dev , NULL , client , & lm83_regmap_config ) ;
if ( IS_ERR ( data - > regmap ) )
return PTR_ERR ( data - > regmap ) ;
2005-04-16 15:20:36 -07:00
2021-12-22 20:24:02 -08:00
data - > type = i2c_match_id ( lm83_id , client ) - > driver_data ;
2014-04-12 12:46:34 -07:00
2021-12-22 20:24:02 -08:00
hwmon_dev = devm_hwmon_device_register_with_info ( dev , client - > name ,
data , & lm83_chip_info , NULL ) ;
2014-04-12 12:46:34 -07:00
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2005-04-16 15:20:36 -07:00
}
2014-04-12 12:39:41 -07:00
/*
* Driver data ( common to all clients )
*/
2005-04-16 15:20:36 -07:00
2014-04-12 12:39:41 -07:00
static struct i2c_driver lm83_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " lm83 " ,
} ,
2020-08-13 18:02:22 +02:00
. probe_new = lm83_probe ,
2014-04-12 12:39:41 -07:00
. id_table = lm83_id ,
. detect = lm83_detect ,
. address_list = normal_i2c ,
} ;
2005-04-16 15:20:36 -07:00
2012-01-20 15:38:18 +08:00
module_i2c_driver ( lm83_driver ) ;
2005-04-16 15:20:36 -07:00
2014-01-29 20:40:08 +01:00
MODULE_AUTHOR ( " Jean Delvare <jdelvare@suse.de> " ) ;
2005-04-16 15:20:36 -07:00
MODULE_DESCRIPTION ( " LM83 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;