2019-05-27 09:55:06 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
2012-01-15 01:42:20 +04:00
* lm77 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
*
* Copyright ( c ) 2004 Andras BALI < drewie @ freemail . hu >
*
* Heavily based on lm75 . c by Frodo Looijaard < frodol @ dds . nl > . The LM77
* is a temperature sensor and thermal window comparator with 0.5 deg
* resolution made by National Semiconductor . Complete datasheet can be
* obtained at their site :
* http : //www.national.com/pf/LM/LM77.html
*/
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
2005-07-16 05:39:18 +04:00
# include <linux/hwmon.h>
2008-01-04 01:35:33 +03:00
# include <linux/hwmon-sysfs.h>
2005-07-16 05:39:18 +04:00
# include <linux/err.h>
2006-01-19 01:19:26 +03:00
# include <linux/mutex.h>
2005-04-17 02:20:36 +04:00
/* Addresses to scan */
2008-02-18 06:28:03 +03:00
static const unsigned short normal_i2c [ ] = { 0x48 , 0x49 , 0x4a , 0x4b ,
I2C_CLIENT_END } ;
2005-04-17 02:20:36 +04:00
/* The LM77 registers */
# define LM77_REG_TEMP 0x00
# define LM77_REG_CONF 0x01
# define LM77_REG_TEMP_HYST 0x02
# define LM77_REG_TEMP_CRIT 0x03
# define LM77_REG_TEMP_MIN 0x04
# define LM77_REG_TEMP_MAX 0x05
2014-04-12 20:17:37 +04:00
enum temp_index {
t_input = 0 ,
t_crit ,
t_min ,
t_max ,
t_hyst ,
t_num_temp
} ;
static const u8 temp_regs [ t_num_temp ] = {
[ t_input ] = LM77_REG_TEMP ,
[ t_min ] = LM77_REG_TEMP_MIN ,
[ t_max ] = LM77_REG_TEMP_MAX ,
[ t_crit ] = LM77_REG_TEMP_CRIT ,
[ t_hyst ] = LM77_REG_TEMP_HYST ,
} ;
2005-04-17 02:20:36 +04:00
/* Each client has this additional data */
struct lm77_data {
2014-04-12 20:25:25 +04:00
struct i2c_client * client ;
2006-01-19 01:19:26 +03:00
struct mutex update_lock ;
2005-04-17 02:20:36 +04:00
char valid ;
unsigned long last_updated ; /* In jiffies */
2014-04-12 20:17:37 +04:00
int temp [ t_num_temp ] ; /* index using temp_index */
2005-04-17 02:20:36 +04:00
u8 alarms ;
} ;
/* straight from the datasheet */
# define LM77_TEMP_MIN (-55000)
# define LM77_TEMP_MAX 125000
2012-01-15 01:42:20 +04:00
/*
* In the temperature registers , the low 3 bits are not part of the
* temperature values ; they are the status bits .
*/
2006-01-10 00:43:08 +03:00
static inline s16 LM77_TEMP_TO_REG ( int temp )
2005-04-17 02:20:36 +04:00
{
2014-07-31 18:58:04 +04:00
return ( temp / 500 ) * 8 ;
2005-04-17 02:20:36 +04:00
}
2006-01-10 00:43:08 +03:00
static inline int LM77_TEMP_FROM_REG ( s16 reg )
2005-04-17 02:20:36 +04:00
{
2006-01-10 00:43:08 +03:00
return ( reg / 8 ) * 500 ;
2005-04-17 02:20:36 +04:00
}
2014-04-12 19:45:20 +04:00
/*
* All registers are word - sized , except for the configuration register .
* The LM77 uses the high - byte first convention .
*/
static u16 lm77_read_value ( struct i2c_client * client , u8 reg )
{
if ( reg = = LM77_REG_CONF )
return i2c_smbus_read_byte_data ( client , reg ) ;
else
return i2c_smbus_read_word_swapped ( client , reg ) ;
}
static int lm77_write_value ( struct i2c_client * client , u8 reg , u16 value )
{
if ( reg = = LM77_REG_CONF )
return i2c_smbus_write_byte_data ( client , reg , value ) ;
else
return i2c_smbus_write_word_swapped ( client , reg , value ) ;
}
static struct lm77_data * lm77_update_device ( struct device * dev )
{
2014-04-12 20:25:25 +04:00
struct lm77_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-04-12 20:17:37 +04:00
int i ;
2014-04-12 19:45:20 +04:00
mutex_lock ( & data - > update_lock ) ;
if ( time_after ( jiffies , data - > last_updated + HZ + HZ / 2 )
| | ! data - > valid ) {
dev_dbg ( & client - > dev , " Starting lm77 update \n " ) ;
2014-04-12 20:17:37 +04:00
for ( i = 0 ; i < t_num_temp ; i + + ) {
data - > temp [ i ] =
LM77_TEMP_FROM_REG ( lm77_read_value ( client ,
temp_regs [ i ] ) ) ;
}
2014-04-12 19:45:20 +04:00
data - > alarms =
lm77_read_value ( client , LM77_REG_TEMP ) & 0x0007 ;
data - > last_updated = jiffies ;
data - > valid = 1 ;
}
mutex_unlock ( & data - > update_lock ) ;
return data ;
}
2005-04-17 02:20:36 +04:00
/* sysfs stuff */
2018-12-11 01:02:11 +03:00
static ssize_t temp_show ( struct device * dev , struct device_attribute * devattr ,
2014-04-12 20:17:37 +04:00
char * buf )
2005-04-17 02:20:36 +04:00
{
2014-04-12 20:17:37 +04:00
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
2005-04-17 02:20:36 +04:00
struct lm77_data * data = lm77_update_device ( dev ) ;
2014-04-12 20:17:37 +04:00
return sprintf ( buf , " %d \n " , data - > temp [ attr - > index ] ) ;
2005-04-17 02:20:36 +04:00
}
2014-04-12 20:17:37 +04:00
2018-12-11 01:02:11 +03:00
static ssize_t temp_hyst_show ( struct device * dev ,
2014-04-12 20:17:37 +04:00
struct device_attribute * devattr , char * buf )
2005-04-17 02:20:36 +04:00
{
2014-04-12 20:17:37 +04:00
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
2005-04-17 02:20:36 +04:00
struct lm77_data * data = lm77_update_device ( dev ) ;
2014-04-12 20:17:37 +04:00
int nr = attr - > index ;
int temp ;
2005-04-17 02:20:36 +04:00
2014-04-12 20:17:37 +04:00
temp = nr = = t_min ? data - > temp [ nr ] + data - > temp [ t_hyst ] :
data - > temp [ nr ] - data - > temp [ t_hyst ] ;
2005-04-17 02:20:36 +04:00
2014-04-12 20:17:37 +04:00
return sprintf ( buf , " %d \n " , temp ) ;
}
2005-04-17 02:20:36 +04:00
2018-12-11 01:02:11 +03:00
static ssize_t temp_store ( struct device * dev ,
struct device_attribute * devattr , const char * buf ,
size_t count )
2005-04-17 02:20:36 +04:00
{
2014-04-12 20:17:37 +04:00
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
2014-04-12 20:25:25 +04:00
struct lm77_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-04-12 20:17:37 +04:00
int nr = attr - > index ;
long val ;
2012-01-15 01:42:20 +04:00
int err ;
2014-04-12 20:17:37 +04:00
err = kstrtol ( buf , 10 , & val ) ;
2012-01-15 01:42:20 +04:00
if ( err )
return err ;
2005-04-17 02:20:36 +04:00
2014-07-31 18:58:04 +04:00
val = clamp_val ( val , LM77_TEMP_MIN , LM77_TEMP_MAX ) ;
2006-01-19 01:19:26 +03:00
mutex_lock ( & data - > update_lock ) ;
2014-04-12 20:17:37 +04:00
data - > temp [ nr ] = val ;
lm77_write_value ( client , temp_regs [ nr ] , LM77_TEMP_TO_REG ( val ) ) ;
2006-01-19 01:19:26 +03:00
mutex_unlock ( & data - > update_lock ) ;
2005-04-17 02:20:36 +04:00
return count ;
}
2014-04-12 20:17:37 +04:00
/*
* hysteresis is stored as a relative value on the chip , so it has to be
* converted first .
*/
2018-12-11 01:02:11 +03:00
static ssize_t temp_hyst_store ( struct device * dev ,
struct device_attribute * devattr ,
const char * buf , size_t count )
2005-04-17 02:20:36 +04:00
{
2014-04-12 20:25:25 +04:00
struct lm77_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-07-31 18:58:04 +04:00
long val ;
2012-01-15 01:42:20 +04:00
int err ;
2014-07-31 18:58:04 +04:00
err = kstrtol ( buf , 10 , & val ) ;
2012-01-15 01:42:20 +04:00
if ( err )
return err ;
2006-01-19 01:19:26 +03:00
mutex_lock ( & data - > update_lock ) ;
2014-07-31 18:58:04 +04:00
val = clamp_val ( data - > temp [ t_crit ] - val , LM77_TEMP_MIN , LM77_TEMP_MAX ) ;
data - > temp [ t_hyst ] = val ;
2014-04-12 20:17:37 +04:00
lm77_write_value ( client , LM77_REG_TEMP_HYST ,
LM77_TEMP_TO_REG ( data - > temp [ t_hyst ] ) ) ;
2006-01-19 01:19:26 +03:00
mutex_unlock ( & data - > update_lock ) ;
2005-04-17 02:20:36 +04:00
return count ;
}
2018-12-11 01:02:11 +03:00
static ssize_t alarm_show ( struct device * dev , struct device_attribute * attr ,
2008-01-04 01:35:33 +03:00
char * buf )
{
int bitnr = to_sensor_dev_attr ( attr ) - > index ;
struct lm77_data * data = lm77_update_device ( dev ) ;
return sprintf ( buf , " %u \n " , ( data - > alarms > > bitnr ) & 1 ) ;
}
2018-12-11 01:02:11 +03:00
static SENSOR_DEVICE_ATTR_RO ( temp1_input , temp , t_input ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_crit , temp , t_crit ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_min , temp , t_min ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_max , temp , t_max ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_crit_hyst , temp_hyst , t_crit ) ;
static SENSOR_DEVICE_ATTR_RO ( temp1_min_hyst , temp_hyst , t_min ) ;
static SENSOR_DEVICE_ATTR_RO ( temp1_max_hyst , temp_hyst , t_max ) ;
static SENSOR_DEVICE_ATTR_RO ( temp1_crit_alarm , alarm , 2 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp1_min_alarm , alarm , 0 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp1_max_alarm , alarm , 1 ) ;
2005-04-17 02:20:36 +04:00
2014-04-12 20:25:25 +04:00
static struct attribute * lm77_attrs [ ] = {
2014-04-12 20:17:37 +04:00
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp1_crit . dev_attr . attr ,
& sensor_dev_attr_temp1_min . dev_attr . attr ,
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp1_crit_hyst . dev_attr . attr ,
& sensor_dev_attr_temp1_min_hyst . dev_attr . attr ,
& sensor_dev_attr_temp1_max_hyst . dev_attr . attr ,
2008-01-04 01:35:33 +03:00
& sensor_dev_attr_temp1_crit_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_max_alarm . dev_attr . attr ,
2006-09-24 23:14:35 +04:00
NULL
} ;
2014-04-12 20:25:25 +04:00
ATTRIBUTE_GROUPS ( lm77 ) ;
2006-09-24 23:14:35 +04:00
2008-07-16 21:30:13 +04:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2012-06-02 20:58:09 +04:00
static int lm77_detect ( struct i2c_client * client , struct i2c_board_info * info )
2005-04-17 02:20:36 +04:00
{
2012-06-02 20:58:09 +04:00
struct i2c_adapter * adapter = client - > adapter ;
2009-12-09 22:35:52 +03:00
int i , cur , conf , hyst , crit , min , max ;
2005-04-17 02:20:36 +04:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA |
I2C_FUNC_SMBUS_WORD_DATA ) )
2008-07-16 21:30:13 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2012-01-15 01:42:20 +04:00
/*
* Here comes the remaining detection . Since the LM77 has no
* register dedicated to identification , we have to rely on the
* following tricks :
*
* 1. the high 4 bits represent the sign and thus they should
* always be the same
* 2. the high 3 bits are unused in the configuration register
* 3. addresses 0x06 and 0x07 return the last read value
* 4. registers cycling over 8 - address boundaries
*
* Word - sized registers are high - byte first .
*/
2005-04-17 02:20:36 +04:00
2009-12-09 22:35:52 +03:00
/* addresses cycling */
2012-06-02 20:58:09 +04:00
cur = i2c_smbus_read_word_data ( client , 0 ) ;
conf = i2c_smbus_read_byte_data ( client , 1 ) ;
hyst = i2c_smbus_read_word_data ( client , 2 ) ;
crit = i2c_smbus_read_word_data ( client , 3 ) ;
min = i2c_smbus_read_word_data ( client , 4 ) ;
max = i2c_smbus_read_word_data ( client , 5 ) ;
2009-12-09 22:35:52 +03:00
for ( i = 8 ; i < = 0xff ; i + = 8 ) {
2012-06-02 20:58:09 +04:00
if ( i2c_smbus_read_byte_data ( client , i + 1 ) ! = conf
| | i2c_smbus_read_word_data ( client , i + 2 ) ! = hyst
| | i2c_smbus_read_word_data ( client , i + 3 ) ! = crit
| | i2c_smbus_read_word_data ( client , i + 4 ) ! = min
| | i2c_smbus_read_word_data ( client , i + 5 ) ! = max )
2008-07-16 21:30:13 +04:00
return - ENODEV ;
2009-12-09 22:35:52 +03:00
}
2005-04-17 02:20:36 +04:00
2009-12-09 22:35:52 +03:00
/* sign bits */
if ( ( ( cur & 0x00f0 ) ! = 0xf0 & & ( cur & 0x00f0 ) ! = 0x0 )
| | ( ( hyst & 0x00f0 ) ! = 0xf0 & & ( hyst & 0x00f0 ) ! = 0x0 )
| | ( ( crit & 0x00f0 ) ! = 0xf0 & & ( crit & 0x00f0 ) ! = 0x0 )
| | ( ( min & 0x00f0 ) ! = 0xf0 & & ( min & 0x00f0 ) ! = 0x0 )
| | ( ( max & 0x00f0 ) ! = 0xf0 & & ( max & 0x00f0 ) ! = 0x0 ) )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2009-12-09 22:35:52 +03:00
/* unused bits */
if ( conf & 0xe0 )
return - ENODEV ;
/* 0x06 and 0x07 return the last read value */
2012-06-02 20:58:09 +04:00
cur = i2c_smbus_read_word_data ( client , 0 ) ;
if ( i2c_smbus_read_word_data ( client , 6 ) ! = cur
| | i2c_smbus_read_word_data ( client , 7 ) ! = cur )
2009-12-09 22:35:52 +03:00
return - ENODEV ;
2012-06-02 20:58:09 +04:00
hyst = i2c_smbus_read_word_data ( client , 2 ) ;
if ( i2c_smbus_read_word_data ( client , 6 ) ! = hyst
| | i2c_smbus_read_word_data ( client , 7 ) ! = hyst )
2009-12-09 22:35:52 +03:00
return - ENODEV ;
2012-06-02 20:58:09 +04:00
min = i2c_smbus_read_word_data ( client , 4 ) ;
if ( i2c_smbus_read_word_data ( client , 6 ) ! = min
| | i2c_smbus_read_word_data ( client , 7 ) ! = min )
2009-12-09 22:35:52 +03:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2008-07-16 21:30:13 +04:00
strlcpy ( info - > type , " lm77 " , I2C_NAME_SIZE ) ;
2005-04-17 02:20:36 +04:00
2008-07-16 21:30:13 +04:00
return 0 ;
}
2014-04-12 19:45:20 +04:00
static void lm77_init_client ( struct i2c_client * client )
{
/* Initialize the LM77 chip - turn off shutdown mode */
int conf = lm77_read_value ( client , LM77_REG_CONF ) ;
if ( conf & 1 )
lm77_write_value ( client , LM77_REG_CONF , conf & 0xfe ) ;
}
2020-08-13 19:02:22 +03:00
static int lm77_probe ( struct i2c_client * client )
2008-07-16 21:30:13 +04:00
{
2012-06-02 20:58:09 +04:00
struct device * dev = & client - > dev ;
2014-04-12 20:25:25 +04:00
struct device * hwmon_dev ;
2008-07-16 21:30:13 +04:00
struct lm77_data * data ;
2012-06-17 05:24:02 +04:00
data = devm_kzalloc ( dev , sizeof ( struct lm77_data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2005-04-17 02:20:36 +04:00
2014-04-12 20:25:25 +04:00
data - > client = client ;
2006-01-19 01:19:26 +03:00
mutex_init ( & data - > update_lock ) ;
2005-04-17 02:20:36 +04:00
/* Initialize the LM77 chip */
2012-06-02 20:58:09 +04:00
lm77_init_client ( client ) ;
2005-04-17 02:20:36 +04:00
2014-04-12 20:25:25 +04:00
hwmon_dev = devm_hwmon_device_register_with_groups ( dev , client - > name ,
data , lm77_groups ) ;
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2005-04-17 02:20:36 +04:00
}
2014-04-12 19:45:20 +04:00
static const struct i2c_device_id lm77_id [ ] = {
{ " lm77 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm77_id ) ;
2005-04-17 02:20:36 +04:00
2014-04-12 19:45:20 +04:00
/* This is the driver that will be inserted */
static struct i2c_driver lm77_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " lm77 " ,
} ,
2020-08-13 19:02:22 +03:00
. probe_new = lm77_probe ,
2014-04-12 19:45:20 +04:00
. id_table = lm77_id ,
. detect = lm77_detect ,
. address_list = normal_i2c ,
} ;
2005-04-17 02:20:36 +04:00
2012-01-20 11:38:18 +04:00
module_i2c_driver ( lm77_driver ) ;
2005-04-17 02:20:36 +04:00
MODULE_AUTHOR ( " Andras BALI <drewie@freemail.hu> " ) ;
MODULE_DESCRIPTION ( " LM77 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;