2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
* max1619 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
2012-10-10 15:25:56 +02:00
* Copyright ( C ) 2003 - 2004 Oleksij Rempel < bug - track @ fisher - privat . net >
2014-01-29 20:40:08 +01:00
* Jean Delvare < jdelvare @ suse . de >
2005-04-16 15:20:36 -07:00
*
* Based on the lm90 driver . The MAX1619 is a sensor chip made by Maxim .
* It reports up to two temperatures ( its own plus up to
* one external one ) . Complete datasheet can be
* obtained from Maxim ' s website at :
* http : //pdfserv.maxim-ic.com/en/ds/MAX1619.pdf
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
2005-07-15 21:39:18 -04:00
# include <linux/hwmon.h>
2008-01-06 15:27:59 +01:00
# include <linux/hwmon-sysfs.h>
2005-07-15 21:39:18 -04:00
# include <linux/err.h>
2006-01-18 23:19:26 +01:00
# include <linux/mutex.h>
2006-09-24 21:24:46 +02:00
# include <linux/sysfs.h>
2005-04-16 15:20:36 -07:00
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
/*
* The MAX1619 registers
*/
# define MAX1619_REG_R_MAN_ID 0xFE
# define MAX1619_REG_R_CHIP_ID 0xFF
# define MAX1619_REG_R_CONFIG 0x03
# define MAX1619_REG_W_CONFIG 0x09
# define MAX1619_REG_R_CONVRATE 0x04
# define MAX1619_REG_W_CONVRATE 0x0A
# define MAX1619_REG_R_STATUS 0x02
# define MAX1619_REG_R_LOCAL_TEMP 0x00
# define MAX1619_REG_R_REMOTE_TEMP 0x01
# define MAX1619_REG_R_REMOTE_HIGH 0x07
# define MAX1619_REG_W_REMOTE_HIGH 0x0D
# define MAX1619_REG_R_REMOTE_LOW 0x08
# define MAX1619_REG_W_REMOTE_LOW 0x0E
# define MAX1619_REG_R_REMOTE_CRIT 0x10
# define MAX1619_REG_W_REMOTE_CRIT 0x12
# define MAX1619_REG_R_TCRIT_HYST 0x11
# define MAX1619_REG_W_TCRIT_HYST 0x13
/*
2008-10-17 17:51:16 +02:00
* Conversions
2005-04-16 15:20:36 -07:00
*/
2008-10-17 17:51:16 +02:00
static int temp_from_reg ( int val )
{
return ( val & 0x80 ? val - 0x100 : val ) * 1000 ;
}
static int temp_to_reg ( int val )
{
return ( val < 0 ? val + 0x100 * 1000 : val ) / 1000 ;
}
2005-04-16 15:20:36 -07:00
2014-04-12 10:28:50 -07:00
enum temp_index {
t_input1 = 0 ,
t_input2 ,
t_low2 ,
t_high2 ,
t_crit2 ,
t_hyst2 ,
t_num_regs
} ;
2005-04-16 15:20:36 -07:00
/*
* Client data ( each client gets its own )
*/
struct max1619_data {
2014-04-12 10:35:29 -07:00
struct i2c_client * client ;
2006-01-18 23:19:26 +01:00
struct mutex update_lock ;
2005-04-16 15:20:36 -07:00
char valid ; /* zero until following fields are valid */
unsigned long last_updated ; /* in jiffies */
/* registers values */
2014-04-12 10:28:50 -07:00
u8 temp [ t_num_regs ] ; /* index with enum temp_index */
2012-01-14 21:29:27 -08:00
u8 alarms ;
2005-04-16 15:20:36 -07:00
} ;
2014-04-12 10:28:50 -07:00
static const u8 regs_read [ t_num_regs ] = {
[ t_input1 ] = MAX1619_REG_R_LOCAL_TEMP ,
[ t_input2 ] = MAX1619_REG_R_REMOTE_TEMP ,
[ t_low2 ] = MAX1619_REG_R_REMOTE_LOW ,
[ t_high2 ] = MAX1619_REG_R_REMOTE_HIGH ,
[ t_crit2 ] = MAX1619_REG_R_REMOTE_CRIT ,
[ t_hyst2 ] = MAX1619_REG_R_TCRIT_HYST ,
} ;
static const u8 regs_write [ t_num_regs ] = {
[ t_low2 ] = MAX1619_REG_W_REMOTE_LOW ,
[ t_high2 ] = MAX1619_REG_W_REMOTE_HIGH ,
[ t_crit2 ] = MAX1619_REG_W_REMOTE_CRIT ,
[ t_hyst2 ] = MAX1619_REG_W_TCRIT_HYST ,
} ;
2014-04-12 10:14:33 -07:00
static struct max1619_data * max1619_update_device ( struct device * dev )
{
2014-04-12 10:35:29 -07:00
struct max1619_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-04-12 10:28:50 -07:00
int config , i ;
2014-04-12 10:14:33 -07:00
mutex_lock ( & data - > update_lock ) ;
if ( time_after ( jiffies , data - > last_updated + HZ * 2 ) | | ! data - > valid ) {
dev_dbg ( & client - > dev , " Updating max1619 data. \n " ) ;
2014-04-12 10:28:50 -07:00
for ( i = 0 ; i < t_num_regs ; i + + )
data - > temp [ i ] = i2c_smbus_read_byte_data ( client ,
regs_read [ i ] ) ;
2014-04-12 10:14:33 -07:00
data - > alarms = i2c_smbus_read_byte_data ( client ,
MAX1619_REG_R_STATUS ) ;
/* If OVERT polarity is low, reverse alarm bit */
config = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_CONFIG ) ;
if ( ! ( config & 0x20 ) )
data - > alarms ^ = 0x02 ;
data - > last_updated = jiffies ;
data - > valid = 1 ;
}
mutex_unlock ( & data - > update_lock ) ;
return data ;
}
2005-04-16 15:20:36 -07:00
/*
* Sysfs stuff
*/
2018-12-10 14:02:15 -08:00
static ssize_t temp_show ( struct device * dev , struct device_attribute * devattr ,
2014-04-12 10:28:50 -07:00
char * buf )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct max1619_data * data = max1619_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , temp_from_reg ( data - > temp [ attr - > index ] ) ) ;
2005-04-16 15:20:36 -07:00
}
2018-12-10 14:02:15 -08:00
static ssize_t temp_store ( struct device * dev ,
struct device_attribute * devattr , const char * buf ,
size_t count )
2014-04-12 10:28:50 -07:00
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
2014-04-12 10:35:29 -07:00
struct max1619_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-04-12 10:28:50 -07:00
long val ;
int err = kstrtol ( buf , 10 , & val ) ;
if ( err )
return err ;
mutex_lock ( & data - > update_lock ) ;
data - > temp [ attr - > index ] = temp_to_reg ( val ) ;
i2c_smbus_write_byte_data ( client , regs_write [ attr - > index ] ,
data - > temp [ attr - > index ] ) ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
2005-04-16 15:20:36 -07:00
2016-12-22 13:04:57 +01:00
static ssize_t alarms_show ( struct device * dev , struct device_attribute * attr ,
2012-01-14 21:29:27 -08:00
char * buf )
2005-04-16 15:20:36 -07:00
{
struct max1619_data * data = max1619_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , data - > alarms ) ;
}
2018-12-10 14:02:15 -08:00
static ssize_t alarm_show ( struct device * dev , struct device_attribute * attr ,
2008-01-06 15:27:59 +01:00
char * buf )
{
int bitnr = to_sensor_dev_attr ( attr ) - > index ;
struct max1619_data * data = max1619_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , ( data - > alarms > > bitnr ) & 1 ) ;
}
2018-12-10 14:02:15 -08:00
static SENSOR_DEVICE_ATTR_RO ( temp1_input , temp , t_input1 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_input , temp , t_input2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_min , temp , t_low2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_max , temp , t_high2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_crit , temp , t_crit2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_crit_hyst , temp , t_hyst2 ) ;
2014-04-12 10:28:50 -07:00
2016-12-22 13:04:57 +01:00
static DEVICE_ATTR_RO ( alarms ) ;
2018-12-10 14:02:15 -08:00
static SENSOR_DEVICE_ATTR_RO ( temp2_crit_alarm , alarm , 1 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_fault , alarm , 2 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_min_alarm , alarm , 3 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_max_alarm , alarm , 4 ) ;
2005-04-16 15:20:36 -07:00
2014-04-12 10:35:29 -07:00
static struct attribute * max1619_attrs [ ] = {
2014-04-12 10:28:50 -07:00
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp2_input . dev_attr . attr ,
& sensor_dev_attr_temp2_min . dev_attr . attr ,
& sensor_dev_attr_temp2_max . dev_attr . attr ,
& sensor_dev_attr_temp2_crit . dev_attr . attr ,
& sensor_dev_attr_temp2_crit_hyst . dev_attr . attr ,
2006-09-24 21:24:46 +02:00
& dev_attr_alarms . attr ,
2008-01-06 15:27:59 +01:00
& sensor_dev_attr_temp2_crit_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_fault . dev_attr . attr ,
& sensor_dev_attr_temp2_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_max_alarm . dev_attr . attr ,
2006-09-24 21:24:46 +02:00
NULL
} ;
2014-04-12 10:35:29 -07:00
ATTRIBUTE_GROUPS ( max1619 ) ;
2006-09-24 21:24:46 +02:00
2008-07-16 19:30:15 +02:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2009-12-14 21:17:23 +01:00
static int max1619_detect ( struct i2c_client * client ,
2008-07-16 19:30:15 +02:00
struct i2c_board_info * info )
2005-04-16 15:20:36 -07:00
{
2009-12-09 20:35:57 +01:00
struct i2c_adapter * adapter = client - > adapter ;
u8 reg_config , reg_convrate , reg_status , man_id , chip_id ;
2005-11-07 01:01:49 -08:00
2005-04-16 15:20:36 -07:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
2008-07-16 19:30:15 +02:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
2009-12-09 20:35:57 +01:00
/* detection */
reg_config = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_CONFIG ) ;
reg_convrate = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_CONVRATE ) ;
reg_status = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_STATUS ) ;
if ( ( reg_config & 0x03 ) ! = 0x00
| | reg_convrate > 0x07 | | ( reg_status & 0x61 ) ! = 0x00 ) {
dev_dbg ( & adapter - > dev , " MAX1619 detection failed at 0x%02x \n " ,
client - > addr ) ;
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2009-12-09 20:35:57 +01:00
/* identification */
man_id = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_MAN_ID ) ;
chip_id = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_CHIP_ID ) ;
if ( man_id ! = 0x4D | | chip_id ! = 0x04 ) {
dev_info ( & adapter - > dev ,
" Unsupported chip (man_id=0x%02X, chip_id=0x%02X). \n " ,
man_id , chip_id ) ;
return - ENODEV ;
2005-11-07 01:01:49 -08:00
}
2005-04-16 15:20:36 -07:00
2008-07-16 19:30:15 +02:00
strlcpy ( info - > type , " max1619 " , I2C_NAME_SIZE ) ;
return 0 ;
}
2014-04-12 10:14:33 -07:00
static void max1619_init_client ( struct i2c_client * client )
{
u8 config ;
/*
* Start the conversions .
*/
i2c_smbus_write_byte_data ( client , MAX1619_REG_W_CONVRATE ,
5 ) ; /* 2 Hz */
config = i2c_smbus_read_byte_data ( client , MAX1619_REG_R_CONFIG ) ;
if ( config & 0x40 )
i2c_smbus_write_byte_data ( client , MAX1619_REG_W_CONFIG ,
config & 0xBF ) ; /* run */
}
2020-08-13 18:02:22 +02:00
static int max1619_probe ( struct i2c_client * new_client )
2008-07-16 19:30:15 +02:00
{
struct max1619_data * data ;
2014-04-12 10:35:29 -07:00
struct device * hwmon_dev ;
2008-07-16 19:30:15 +02:00
2012-06-02 09:58:12 -07:00
data = devm_kzalloc ( & new_client - > dev , sizeof ( struct max1619_data ) ,
GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2005-04-16 15:20:36 -07:00
2014-04-12 10:35:29 -07:00
data - > client = new_client ;
2006-01-18 23:19:26 +01:00
mutex_init ( & data - > update_lock ) ;
2005-04-16 15:20:36 -07:00
/* Initialize the MAX1619 chip */
max1619_init_client ( new_client ) ;
2014-04-12 10:35:29 -07:00
hwmon_dev = devm_hwmon_device_register_with_groups ( & new_client - > dev ,
new_client - > name ,
data ,
max1619_groups ) ;
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2005-04-16 15:20:36 -07:00
}
2014-04-12 10:14:33 -07:00
static const struct i2c_device_id max1619_id [ ] = {
{ " max1619 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , max1619_id ) ;
2005-04-16 15:20:36 -07:00
2017-09-11 14:16:49 -05:00
# ifdef CONFIG_OF
static const struct of_device_id max1619_of_match [ ] = {
{ . compatible = " maxim,max1619 " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , max1619_of_match ) ;
# endif
2014-04-12 10:14:33 -07:00
static struct i2c_driver max1619_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " max1619 " ,
2017-09-11 14:16:49 -05:00
. of_match_table = of_match_ptr ( max1619_of_match ) ,
2014-04-12 10:14:33 -07:00
} ,
2020-08-13 18:02:22 +02:00
. probe_new = max1619_probe ,
2014-04-12 10:14:33 -07:00
. id_table = max1619_id ,
. detect = max1619_detect ,
. address_list = normal_i2c ,
} ;
2005-04-16 15:20:36 -07:00
2012-01-20 15:38:18 +08:00
module_i2c_driver ( max1619_driver ) ;
2005-04-16 15:20:36 -07:00
2014-01-29 20:40:08 +01:00
MODULE_AUTHOR ( " Oleksij Rempel <bug-track@fisher-privat.net>, Jean Delvare <jdelvare@suse.de> " ) ;
2005-04-16 15:20:36 -07:00
MODULE_DESCRIPTION ( " MAX1619 sensor driver " ) ;
MODULE_LICENSE ( " GPL " ) ;