2009-06-15 20:39:46 +04:00
/* tmp401.c
*
* Copyright ( C ) 2007 , 2008 Hans de Goede < hdegoede @ redhat . com >
2009-06-15 20:39:47 +04:00
* Preliminary tmp411 support by :
* Gabriel Konat , Sander Leget , Wouter Willems
* Copyright ( C ) 2009 Andre Prendel < andre . prendel @ gmx . de >
2009-06-15 20:39:46 +04:00
*
2013-03-28 12:36:44 +04:00
* Cleanup and support for TMP431 and TMP432 by Guenter Roeck
* Copyright ( c ) 2013 Guenter Roeck < linux @ roeck - us . net >
*
2009-06-15 20:39:46 +04:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
* Driver for the Texas Instruments TMP401 SMBUS temperature sensor IC .
*
* Note this IC is in some aspect similar to the LM90 , but it has quite a
* few differences too , for example the local temp has a higher resolution
* and thus has 16 bits registers for its value and limit instead of 8 bits .
*/
# include <linux/module.h>
# include <linux/init.h>
2013-03-27 19:48:03 +04:00
# include <linux/bitops.h>
2009-06-15 20:39:46 +04:00
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
# include <linux/hwmon.h>
# include <linux/hwmon-sysfs.h>
# include <linux/err.h>
# include <linux/mutex.h>
# include <linux/sysfs.h>
/* Addresses to scan */
2015-05-28 02:11:48 +03:00
static const unsigned short normal_i2c [ ] = { 0x48 , 0x49 , 0x4a , 0x4c , 0x4d ,
2014-12-05 21:15:03 +03:00
0x4e , 0x4f , I2C_CLIENT_END } ;
2009-06-15 20:39:46 +04:00
2016-06-08 20:00:54 +03:00
enum chips { tmp401 , tmp411 , tmp431 , tmp432 , tmp435 , tmp461 } ;
2009-06-15 20:39:46 +04:00
/*
* The TMP401 registers , note some registers have different addresses for
* reading and writing
*/
# define TMP401_STATUS 0x02
# define TMP401_CONFIG_READ 0x03
# define TMP401_CONFIG_WRITE 0x09
# define TMP401_CONVERSION_RATE_READ 0x04
# define TMP401_CONVERSION_RATE_WRITE 0x0A
# define TMP401_TEMP_CRIT_HYST 0x21
# define TMP401_MANUFACTURER_ID_REG 0xFE
# define TMP401_DEVICE_ID_REG 0xFF
2016-06-08 20:00:54 +03:00
static const u8 TMP401_TEMP_MSB_READ [ 7 ] [ 2 ] = {
2013-03-28 08:23:10 +04:00
{ 0x00 , 0x01 } , /* temp */
{ 0x06 , 0x08 } , /* low limit */
{ 0x05 , 0x07 } , /* high limit */
{ 0x20 , 0x19 } , /* therm (crit) limit */
{ 0x30 , 0x34 } , /* lowest */
{ 0x32 , 0x36 } , /* highest */
2016-06-08 20:00:54 +03:00
{ 0 , 0x11 } , /* offset */
2013-03-28 08:23:10 +04:00
} ;
2016-06-08 20:00:54 +03:00
static const u8 TMP401_TEMP_MSB_WRITE [ 7 ] [ 2 ] = {
2013-03-28 08:23:10 +04:00
{ 0 , 0 } , /* temp (unused) */
{ 0x0C , 0x0E } , /* low limit */
{ 0x0B , 0x0D } , /* high limit */
{ 0x20 , 0x19 } , /* therm (crit) limit */
{ 0x30 , 0x34 } , /* lowest */
{ 0x32 , 0x36 } , /* highest */
2016-06-08 20:00:54 +03:00
{ 0 , 0x11 } , /* offset */
2013-03-28 08:23:10 +04:00
} ;
2013-03-28 12:36:44 +04:00
static const u8 TMP432_TEMP_MSB_READ [ 4 ] [ 3 ] = {
{ 0x00 , 0x01 , 0x23 } , /* temp */
{ 0x06 , 0x08 , 0x16 } , /* low limit */
{ 0x05 , 0x07 , 0x15 } , /* high limit */
{ 0x20 , 0x19 , 0x1A } , /* therm (crit) limit */
} ;
static const u8 TMP432_TEMP_MSB_WRITE [ 4 ] [ 3 ] = {
{ 0 , 0 , 0 } , /* temp - unused */
{ 0x0C , 0x0E , 0x16 } , /* low limit */
{ 0x0B , 0x0D , 0x15 } , /* high limit */
{ 0x20 , 0x19 , 0x1A } , /* therm (crit) limit */
} ;
/* [0] = fault, [1] = low, [2] = high, [3] = therm/crit */
static const u8 TMP432_STATUS_REG [ ] = {
0x1b , 0x36 , 0x35 , 0x37 } ;
2009-06-15 20:39:46 +04:00
/* Flags */
2013-03-27 19:48:03 +04:00
# define TMP401_CONFIG_RANGE BIT(2)
# define TMP401_CONFIG_SHUTDOWN BIT(6)
# define TMP401_STATUS_LOCAL_CRIT BIT(0)
# define TMP401_STATUS_REMOTE_CRIT BIT(1)
# define TMP401_STATUS_REMOTE_OPEN BIT(2)
# define TMP401_STATUS_REMOTE_LOW BIT(3)
# define TMP401_STATUS_REMOTE_HIGH BIT(4)
# define TMP401_STATUS_LOCAL_LOW BIT(5)
# define TMP401_STATUS_LOCAL_HIGH BIT(6)
2009-06-15 20:39:46 +04:00
2013-03-28 12:36:44 +04:00
/* On TMP432, each status has its own register */
# define TMP432_STATUS_LOCAL BIT(0)
# define TMP432_STATUS_REMOTE1 BIT(1)
# define TMP432_STATUS_REMOTE2 BIT(2)
2009-06-15 20:39:46 +04:00
/* Manufacturer / Device ID's */
# define TMP401_MANUFACTURER_ID 0x55
# define TMP401_DEVICE_ID 0x11
2013-03-30 04:56:07 +04:00
# define TMP411A_DEVICE_ID 0x12
# define TMP411B_DEVICE_ID 0x13
# define TMP411C_DEVICE_ID 0x10
2013-03-15 23:55:08 +04:00
# define TMP431_DEVICE_ID 0x31
2013-03-28 12:36:44 +04:00
# define TMP432_DEVICE_ID 0x32
2014-12-04 19:45:51 +03:00
# define TMP435_DEVICE_ID 0x35
2009-06-15 20:39:46 +04:00
/*
* Driver data ( common to all clients )
*/
static const struct i2c_device_id tmp401_id [ ] = {
{ " tmp401 " , tmp401 } ,
2009-06-15 20:39:47 +04:00
{ " tmp411 " , tmp411 } ,
2013-03-15 23:55:08 +04:00
{ " tmp431 " , tmp431 } ,
2013-03-28 12:36:44 +04:00
{ " tmp432 " , tmp432 } ,
2014-12-04 19:45:51 +03:00
{ " tmp435 " , tmp435 } ,
2016-06-08 20:00:54 +03:00
{ " tmp461 " , tmp461 } ,
2009-06-15 20:39:46 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , tmp401_id ) ;
/*
* Client data ( each client gets its own )
*/
struct tmp401_data {
2013-09-05 03:24:09 +04:00
struct i2c_client * client ;
const struct attribute_group * groups [ 3 ] ;
2009-06-15 20:39:46 +04:00
struct mutex update_lock ;
char valid ; /* zero until following fields are valid */
unsigned long last_updated ; /* in jiffies */
2010-03-06 00:17:26 +03:00
enum chips kind ;
2009-06-15 20:39:46 +04:00
2013-03-28 13:03:10 +04:00
unsigned int update_interval ; /* in milliseconds */
2009-06-15 20:39:46 +04:00
/* register values */
2013-03-28 12:36:44 +04:00
u8 status [ 4 ] ;
2009-06-15 20:39:46 +04:00
u8 config ;
2016-06-08 20:00:54 +03:00
u16 temp [ 7 ] [ 3 ] ;
2009-06-15 20:39:46 +04:00
u8 temp_crit_hyst ;
} ;
/*
* Sysfs attr show / store functions
*/
static int tmp401_register_to_temp ( u16 reg , u8 config )
{
int temp = reg ;
if ( config & TMP401_CONFIG_RANGE )
temp - = 64 * 256 ;
2013-03-28 08:23:10 +04:00
return DIV_ROUND_CLOSEST ( temp * 125 , 32 ) ;
2009-06-15 20:39:46 +04:00
}
2013-03-28 08:23:10 +04:00
static u16 tmp401_temp_to_register ( long temp , u8 config , int zbits )
2009-06-15 20:39:46 +04:00
{
if ( config & TMP401_CONFIG_RANGE ) {
2013-01-09 20:09:34 +04:00
temp = clamp_val ( temp , - 64000 , 191000 ) ;
2009-06-15 20:39:46 +04:00
temp + = 64000 ;
} else
2013-01-09 20:09:34 +04:00
temp = clamp_val ( temp , 0 , 127000 ) ;
2009-06-15 20:39:46 +04:00
2013-03-28 08:23:10 +04:00
return DIV_ROUND_CLOSEST ( temp * ( 1 < < ( 8 - zbits ) ) , 1000 ) < < zbits ;
2009-06-15 20:39:46 +04:00
}
2013-03-28 08:23:10 +04:00
static int tmp401_update_device_reg16 ( struct i2c_client * client ,
struct tmp401_data * data )
2010-05-27 21:58:49 +04:00
{
2013-03-28 08:23:10 +04:00
int i , j , val ;
int num_regs = data - > kind = = tmp411 ? 6 : 4 ;
2013-03-28 12:36:44 +04:00
int num_sensors = data - > kind = = tmp432 ? 3 : 2 ;
2013-03-28 08:23:10 +04:00
2013-03-28 12:36:44 +04:00
for ( i = 0 ; i < num_sensors ; i + + ) { /* local / r1 / r2 */
2013-03-28 08:23:10 +04:00
for ( j = 0 ; j < num_regs ; j + + ) { /* temp / low / ... */
2013-03-28 12:36:44 +04:00
u8 regaddr ;
2017-01-09 20:55:00 +03:00
2013-03-28 12:36:44 +04:00
regaddr = data - > kind = = tmp432 ?
TMP432_TEMP_MSB_READ [ j ] [ i ] :
TMP401_TEMP_MSB_READ [ j ] [ i ] ;
2017-01-09 20:55:00 +03:00
if ( j = = 3 ) { /* crit is msb only */
val = i2c_smbus_read_byte_data ( client , regaddr ) ;
} else {
val = i2c_smbus_read_word_swapped ( client ,
regaddr ) ;
}
2013-03-28 08:23:10 +04:00
if ( val < 0 )
return val ;
2017-01-09 20:55:00 +03:00
data - > temp [ j ] [ i ] = j = = 3 ? val < < 8 : val ;
2010-05-27 21:58:49 +04:00
}
}
2013-03-28 08:23:10 +04:00
return 0 ;
2010-05-27 21:58:49 +04:00
}
static struct tmp401_data * tmp401_update_device ( struct device * dev )
{
2013-09-05 03:24:09 +04:00
struct tmp401_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2013-03-28 08:23:10 +04:00
struct tmp401_data * ret = data ;
2013-03-28 12:36:44 +04:00
int i , val ;
2013-03-28 13:03:10 +04:00
unsigned long next_update ;
2010-05-27 21:58:49 +04:00
mutex_lock ( & data - > update_lock ) ;
2013-03-28 13:03:10 +04:00
next_update = data - > last_updated +
2013-05-19 18:57:30 +04:00
msecs_to_jiffies ( data - > update_interval ) ;
2013-03-28 13:03:10 +04:00
if ( time_after ( jiffies , next_update ) | | ! data - > valid ) {
2013-03-28 12:36:44 +04:00
if ( data - > kind ! = tmp432 ) {
/*
* The driver uses the TMP432 status format internally .
* Convert status to TMP432 format for other chips .
*/
val = i2c_smbus_read_byte_data ( client , TMP401_STATUS ) ;
if ( val < 0 ) {
ret = ERR_PTR ( val ) ;
goto abort ;
}
data - > status [ 0 ] =
( val & TMP401_STATUS_REMOTE_OPEN ) > > 1 ;
data - > status [ 1 ] =
( ( val & TMP401_STATUS_REMOTE_LOW ) > > 2 ) |
( ( val & TMP401_STATUS_LOCAL_LOW ) > > 5 ) ;
data - > status [ 2 ] =
( ( val & TMP401_STATUS_REMOTE_HIGH ) > > 3 ) |
( ( val & TMP401_STATUS_LOCAL_HIGH ) > > 6 ) ;
data - > status [ 3 ] = val & ( TMP401_STATUS_LOCAL_CRIT
| TMP401_STATUS_REMOTE_CRIT ) ;
} else {
for ( i = 0 ; i < ARRAY_SIZE ( data - > status ) ; i + + ) {
val = i2c_smbus_read_byte_data ( client ,
TMP432_STATUS_REG [ i ] ) ;
if ( val < 0 ) {
ret = ERR_PTR ( val ) ;
goto abort ;
}
data - > status [ i ] = val ;
}
2013-03-28 08:23:10 +04:00
}
2013-03-28 12:36:44 +04:00
2013-03-28 08:23:10 +04:00
val = i2c_smbus_read_byte_data ( client , TMP401_CONFIG_READ ) ;
if ( val < 0 ) {
ret = ERR_PTR ( val ) ;
goto abort ;
}
data - > config = val ;
val = tmp401_update_device_reg16 ( client , data ) ;
if ( val < 0 ) {
ret = ERR_PTR ( val ) ;
goto abort ;
}
val = i2c_smbus_read_byte_data ( client , TMP401_TEMP_CRIT_HYST ) ;
if ( val < 0 ) {
ret = ERR_PTR ( val ) ;
goto abort ;
}
data - > temp_crit_hyst = val ;
2010-05-27 21:58:49 +04:00
data - > last_updated = jiffies ;
data - > valid = 1 ;
}
2013-03-28 08:23:10 +04:00
abort :
2010-05-27 21:58:49 +04:00
mutex_unlock ( & data - > update_lock ) ;
2013-03-28 08:23:10 +04:00
return ret ;
2009-06-15 20:39:46 +04:00
}
2018-12-06 21:59:49 +03:00
static ssize_t temp_show ( struct device * dev , struct device_attribute * devattr ,
char * buf )
2009-06-15 20:39:46 +04:00
{
2013-03-28 08:23:10 +04:00
int nr = to_sensor_dev_attr_2 ( devattr ) - > nr ;
int index = to_sensor_dev_attr_2 ( devattr ) - > index ;
2009-06-15 20:39:46 +04:00
struct tmp401_data * data = tmp401_update_device ( dev ) ;
2013-03-28 08:23:10 +04:00
if ( IS_ERR ( data ) )
return PTR_ERR ( data ) ;
2009-06-15 20:39:46 +04:00
return sprintf ( buf , " %d \n " ,
2013-03-28 08:23:10 +04:00
tmp401_register_to_temp ( data - > temp [ nr ] [ index ] , data - > config ) ) ;
2009-06-15 20:39:46 +04:00
}
2018-12-06 21:59:49 +03:00
static ssize_t temp_crit_hyst_show ( struct device * dev ,
struct device_attribute * devattr ,
char * buf )
2009-06-15 20:39:46 +04:00
{
int temp , index = to_sensor_dev_attr ( devattr ) - > index ;
struct tmp401_data * data = tmp401_update_device ( dev ) ;
2013-03-28 08:23:10 +04:00
if ( IS_ERR ( data ) )
return PTR_ERR ( data ) ;
2009-06-15 20:39:46 +04:00
mutex_lock ( & data - > update_lock ) ;
2013-03-28 08:23:10 +04:00
temp = tmp401_register_to_temp ( data - > temp [ 3 ] [ index ] , data - > config ) ;
2009-06-15 20:39:46 +04:00
temp - = data - > temp_crit_hyst * 1000 ;
mutex_unlock ( & data - > update_lock ) ;
return sprintf ( buf , " %d \n " , temp ) ;
}
2018-12-06 21:59:49 +03:00
static ssize_t status_show ( struct device * dev ,
struct device_attribute * devattr , char * buf )
2009-06-15 20:39:46 +04:00
{
2013-03-28 12:36:44 +04:00
int nr = to_sensor_dev_attr_2 ( devattr ) - > nr ;
int mask = to_sensor_dev_attr_2 ( devattr ) - > index ;
2009-06-15 20:39:46 +04:00
struct tmp401_data * data = tmp401_update_device ( dev ) ;
2013-03-28 08:23:10 +04:00
if ( IS_ERR ( data ) )
return PTR_ERR ( data ) ;
2009-06-15 20:39:46 +04:00
2013-03-28 12:36:44 +04:00
return sprintf ( buf , " %d \n " , ! ! ( data - > status [ nr ] & mask ) ) ;
2009-06-15 20:39:46 +04:00
}
2018-12-06 21:59:49 +03:00
static ssize_t temp_store ( struct device * dev ,
struct device_attribute * devattr , const char * buf ,
size_t count )
2009-06-15 20:39:46 +04:00
{
2013-03-28 08:23:10 +04:00
int nr = to_sensor_dev_attr_2 ( devattr ) - > nr ;
int index = to_sensor_dev_attr_2 ( devattr ) - > index ;
2013-09-05 03:24:09 +04:00
struct tmp401_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2009-06-15 20:39:46 +04:00
long val ;
u16 reg ;
2013-03-28 12:36:44 +04:00
u8 regaddr ;
2009-06-15 20:39:46 +04:00
2012-01-04 23:58:52 +04:00
if ( kstrtol ( buf , 10 , & val ) )
2009-06-15 20:39:46 +04:00
return - EINVAL ;
2013-03-28 08:23:10 +04:00
reg = tmp401_temp_to_register ( val , data - > config , nr = = 3 ? 8 : 4 ) ;
2009-06-15 20:39:46 +04:00
mutex_lock ( & data - > update_lock ) ;
2013-03-28 12:36:44 +04:00
regaddr = data - > kind = = tmp432 ? TMP432_TEMP_MSB_WRITE [ nr ] [ index ]
: TMP401_TEMP_MSB_WRITE [ nr ] [ index ] ;
2017-01-09 20:55:00 +03:00
if ( nr = = 3 ) { /* crit is msb only */
i2c_smbus_write_byte_data ( client , regaddr , reg > > 8 ) ;
} else {
/* Hardware expects big endian data --> use _swapped */
i2c_smbus_write_word_swapped ( client , regaddr , reg ) ;
2013-03-28 08:23:10 +04:00
}
data - > temp [ nr ] [ index ] = reg ;
2009-06-15 20:39:46 +04:00
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
2018-12-06 21:59:49 +03:00
static ssize_t temp_crit_hyst_store ( struct device * dev ,
struct device_attribute * devattr ,
const char * buf , size_t count )
2009-06-15 20:39:46 +04:00
{
int temp , index = to_sensor_dev_attr ( devattr ) - > index ;
struct tmp401_data * data = tmp401_update_device ( dev ) ;
long val ;
u8 reg ;
2013-03-28 08:23:10 +04:00
if ( IS_ERR ( data ) )
return PTR_ERR ( data ) ;
2012-01-04 23:58:52 +04:00
if ( kstrtol ( buf , 10 , & val ) )
2009-06-15 20:39:46 +04:00
return - EINVAL ;
if ( data - > config & TMP401_CONFIG_RANGE )
2013-01-09 20:09:34 +04:00
val = clamp_val ( val , - 64000 , 191000 ) ;
2009-06-15 20:39:46 +04:00
else
2013-01-09 20:09:34 +04:00
val = clamp_val ( val , 0 , 127000 ) ;
2009-06-15 20:39:46 +04:00
mutex_lock ( & data - > update_lock ) ;
2013-03-28 08:23:10 +04:00
temp = tmp401_register_to_temp ( data - > temp [ 3 ] [ index ] , data - > config ) ;
2013-01-09 20:09:34 +04:00
val = clamp_val ( val , temp - 255000 , temp ) ;
2009-06-15 20:39:46 +04:00
reg = ( ( temp - val ) + 500 ) / 1000 ;
2013-09-05 03:24:09 +04:00
i2c_smbus_write_byte_data ( data - > client , TMP401_TEMP_CRIT_HYST ,
2013-03-28 08:23:10 +04:00
reg ) ;
2009-06-15 20:39:46 +04:00
data - > temp_crit_hyst = reg ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
2009-06-15 20:39:47 +04:00
/*
* Resets the historical measurements of minimum and maximum temperatures .
* This is done by writing any value to any of the minimum / maximum registers
* ( 0x30 - 0x37 ) .
*/
2018-12-06 21:59:49 +03:00
static ssize_t reset_temp_history_store ( struct device * dev ,
struct device_attribute * devattr ,
const char * buf , size_t count )
2009-06-15 20:39:47 +04:00
{
2013-09-05 03:24:09 +04:00
struct tmp401_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2009-06-15 20:39:47 +04:00
long val ;
2012-01-04 23:58:52 +04:00
if ( kstrtol ( buf , 10 , & val ) )
2009-06-15 20:39:47 +04:00
return - EINVAL ;
if ( val ! = 1 ) {
2013-01-10 22:01:24 +04:00
dev_err ( dev ,
" temp_reset_history value %ld not supported. Use 1 to reset the history! \n " ,
val ) ;
2009-06-15 20:39:47 +04:00
return - EINVAL ;
}
2013-04-14 15:39:46 +04:00
mutex_lock ( & data - > update_lock ) ;
i2c_smbus_write_byte_data ( client , TMP401_TEMP_MSB_WRITE [ 5 ] [ 0 ] , val ) ;
data - > valid = 0 ;
mutex_unlock ( & data - > update_lock ) ;
2009-06-15 20:39:47 +04:00
return count ;
}
2016-12-22 15:05:35 +03:00
static ssize_t update_interval_show ( struct device * dev ,
2013-03-28 13:03:10 +04:00
struct device_attribute * attr , char * buf )
{
2013-09-05 03:24:09 +04:00
struct tmp401_data * data = dev_get_drvdata ( dev ) ;
2013-03-28 13:03:10 +04:00
return sprintf ( buf , " %u \n " , data - > update_interval ) ;
}
2016-12-22 15:05:35 +03:00
static ssize_t update_interval_store ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
2013-03-28 13:03:10 +04:00
{
2013-09-05 03:24:09 +04:00
struct tmp401_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2013-03-28 13:03:10 +04:00
unsigned long val ;
int err , rate ;
err = kstrtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
/*
* For valid rates , interval can be calculated as
* interval = ( 1 < < ( 7 - rate ) ) * 125 ;
* Rounded rate is therefore
* rate = 7 - __fls ( interval * 4 / ( 125 * 3 ) ) ;
* Use clamp_val ( ) to avoid overflows , and to ensure valid input
* for __fls .
*/
val = clamp_val ( val , 125 , 16000 ) ;
rate = 7 - __fls ( val * 4 / ( 125 * 3 ) ) ;
mutex_lock ( & data - > update_lock ) ;
i2c_smbus_write_byte_data ( client , TMP401_CONVERSION_RATE_WRITE , rate ) ;
data - > update_interval = ( 1 < < ( 7 - rate ) ) * 125 ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
2018-12-06 21:59:49 +03:00
static SENSOR_DEVICE_ATTR_2_RO ( temp1_input , temp , 0 , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp1_min , temp , 1 , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp1_max , temp , 2 , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp1_crit , temp , 3 , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_crit_hyst , temp_crit_hyst , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp1_min_alarm , status , 1 ,
TMP432_STATUS_LOCAL ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp1_max_alarm , status , 2 ,
TMP432_STATUS_LOCAL ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp1_crit_alarm , status , 3 ,
TMP432_STATUS_LOCAL ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_input , temp , 0 , 1 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp2_min , temp , 1 , 1 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp2_max , temp , 2 , 1 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp2_crit , temp , 3 , 1 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_crit_hyst , temp_crit_hyst , 1 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_fault , status , 0 , TMP432_STATUS_REMOTE1 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_min_alarm , status , 1 ,
TMP432_STATUS_REMOTE1 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_max_alarm , status , 2 ,
TMP432_STATUS_REMOTE1 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_crit_alarm , status , 3 ,
TMP432_STATUS_REMOTE1 ) ;
2013-03-27 19:58:46 +04:00
2016-12-22 15:05:35 +03:00
static DEVICE_ATTR_RW ( update_interval ) ;
2013-03-28 13:03:10 +04:00
2013-03-27 19:58:46 +04:00
static struct attribute * tmp401_attributes [ ] = {
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp1_min . dev_attr . attr ,
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp1_crit . dev_attr . attr ,
& sensor_dev_attr_temp1_crit_hyst . dev_attr . attr ,
& sensor_dev_attr_temp1_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_crit_alarm . 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 ,
& sensor_dev_attr_temp2_fault . dev_attr . attr ,
& sensor_dev_attr_temp2_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_crit_alarm . dev_attr . attr ,
2013-03-28 13:03:10 +04:00
& dev_attr_update_interval . attr ,
2013-03-27 19:58:46 +04:00
NULL
} ;
static const struct attribute_group tmp401_group = {
. attrs = tmp401_attributes ,
2009-06-15 20:39:46 +04:00
} ;
2009-06-15 20:39:47 +04:00
/*
* Additional features of the TMP411 chip .
* The TMP411 stores the minimum and maximum
* temperature measured since power - on , chip - reset , or
* minimum and maximum register reset for both the local
* and remote channels .
*/
2018-12-06 21:59:49 +03:00
static SENSOR_DEVICE_ATTR_2_RO ( temp1_lowest , temp , 4 , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp1_highest , temp , 5 , 0 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_lowest , temp , 4 , 1 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp2_highest , temp , 5 , 1 ) ;
static SENSOR_DEVICE_ATTR_WO ( temp_reset_history , reset_temp_history , 0 ) ;
2013-03-27 19:58:46 +04:00
static struct attribute * tmp411_attributes [ ] = {
& sensor_dev_attr_temp1_highest . dev_attr . attr ,
& sensor_dev_attr_temp1_lowest . dev_attr . attr ,
& sensor_dev_attr_temp2_highest . dev_attr . attr ,
& sensor_dev_attr_temp2_lowest . dev_attr . attr ,
& sensor_dev_attr_temp_reset_history . dev_attr . attr ,
NULL
} ;
static const struct attribute_group tmp411_group = {
. attrs = tmp411_attributes ,
2009-06-15 20:39:47 +04:00
} ;
2018-12-06 21:59:49 +03:00
static SENSOR_DEVICE_ATTR_2_RO ( temp3_input , temp , 0 , 2 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp3_min , temp , 1 , 2 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp3_max , temp , 2 , 2 ) ;
static SENSOR_DEVICE_ATTR_2_RW ( temp3_crit , temp , 3 , 2 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp3_crit_hyst , temp_crit_hyst , 2 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp3_fault , status , 0 , TMP432_STATUS_REMOTE2 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp3_min_alarm , status , 1 ,
TMP432_STATUS_REMOTE2 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp3_max_alarm , status , 2 ,
TMP432_STATUS_REMOTE2 ) ;
static SENSOR_DEVICE_ATTR_2_RO ( temp3_crit_alarm , status , 3 ,
TMP432_STATUS_REMOTE2 ) ;
2013-03-28 12:36:44 +04:00
static struct attribute * tmp432_attributes [ ] = {
& sensor_dev_attr_temp3_input . dev_attr . attr ,
& sensor_dev_attr_temp3_min . dev_attr . attr ,
& sensor_dev_attr_temp3_max . dev_attr . attr ,
& sensor_dev_attr_temp3_crit . dev_attr . attr ,
& sensor_dev_attr_temp3_crit_hyst . dev_attr . attr ,
& sensor_dev_attr_temp3_fault . dev_attr . attr ,
& sensor_dev_attr_temp3_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp3_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp3_crit_alarm . dev_attr . attr ,
NULL
} ;
static const struct attribute_group tmp432_group = {
. attrs = tmp432_attributes ,
} ;
2016-06-08 20:00:54 +03:00
/*
* Additional features of the TMP461 chip .
* The TMP461 temperature offset for the remote channel .
*/
2018-12-06 21:59:49 +03:00
static SENSOR_DEVICE_ATTR_2_RW ( temp2_offset , temp , 6 , 1 ) ;
2016-06-08 20:00:54 +03:00
static struct attribute * tmp461_attributes [ ] = {
& sensor_dev_attr_temp2_offset . dev_attr . attr ,
NULL
} ;
static const struct attribute_group tmp461_group = {
. attrs = tmp461_attributes ,
} ;
2009-06-15 20:39:46 +04:00
/*
* Begin non sysfs callback code ( aka Real code )
*/
2014-12-04 19:45:53 +03:00
static int tmp401_init_client ( struct tmp401_data * data ,
struct i2c_client * client )
2009-06-15 20:39:46 +04:00
{
2014-12-04 19:45:53 +03:00
int config , config_orig , status = 0 ;
2009-06-15 20:39:46 +04:00
/* Set the conversion rate to 2 Hz */
i2c_smbus_write_byte_data ( client , TMP401_CONVERSION_RATE_WRITE , 5 ) ;
2013-03-28 13:03:10 +04:00
data - > update_interval = 500 ;
2009-06-15 20:39:46 +04:00
/* Start conversions (disable shutdown if necessary) */
config = i2c_smbus_read_byte_data ( client , TMP401_CONFIG_READ ) ;
2014-12-04 19:45:53 +03:00
if ( config < 0 )
return config ;
2009-06-15 20:39:46 +04:00
config_orig = config ;
config & = ~ TMP401_CONFIG_SHUTDOWN ;
if ( config ! = config_orig )
2014-12-04 19:45:53 +03:00
status = i2c_smbus_write_byte_data ( client ,
TMP401_CONFIG_WRITE ,
config ) ;
return status ;
2009-06-15 20:39:46 +04:00
}
2009-12-14 23:17:23 +03:00
static int tmp401_detect ( struct i2c_client * client ,
2009-06-15 20:39:46 +04:00
struct i2c_board_info * info )
{
2009-12-09 22:35:54 +03:00
enum chips kind ;
2009-06-15 20:39:46 +04:00
struct i2c_adapter * adapter = client - > adapter ;
2009-12-09 22:35:54 +03:00
u8 reg ;
2009-06-15 20:39:46 +04:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
return - ENODEV ;
/* Detect and identify the chip */
2009-12-09 22:35:54 +03:00
reg = i2c_smbus_read_byte_data ( client , TMP401_MANUFACTURER_ID_REG ) ;
if ( reg ! = TMP401_MANUFACTURER_ID )
return - ENODEV ;
2009-06-15 20:39:46 +04:00
2009-12-09 22:35:54 +03:00
reg = i2c_smbus_read_byte_data ( client , TMP401_DEVICE_ID_REG ) ;
2009-06-15 20:39:46 +04:00
2009-12-09 22:35:54 +03:00
switch ( reg ) {
case TMP401_DEVICE_ID :
2013-03-15 23:55:08 +04:00
if ( client - > addr ! = 0x4c )
return - ENODEV ;
2009-12-09 22:35:54 +03:00
kind = tmp401 ;
break ;
2013-03-30 04:56:07 +04:00
case TMP411A_DEVICE_ID :
if ( client - > addr ! = 0x4c )
return - ENODEV ;
kind = tmp411 ;
break ;
case TMP411B_DEVICE_ID :
if ( client - > addr ! = 0x4d )
return - ENODEV ;
kind = tmp411 ;
break ;
case TMP411C_DEVICE_ID :
if ( client - > addr ! = 0x4e )
return - ENODEV ;
2009-12-09 22:35:54 +03:00
kind = tmp411 ;
break ;
2013-03-15 23:55:08 +04:00
case TMP431_DEVICE_ID :
2014-12-05 21:15:03 +03:00
if ( client - > addr ! = 0x4c & & client - > addr ! = 0x4d )
2013-03-15 23:55:08 +04:00
return - ENODEV ;
kind = tmp431 ;
break ;
2013-03-28 12:36:44 +04:00
case TMP432_DEVICE_ID :
2014-12-05 21:15:03 +03:00
if ( client - > addr ! = 0x4c & & client - > addr ! = 0x4d )
2013-03-28 12:36:44 +04:00
return - ENODEV ;
kind = tmp432 ;
break ;
2014-12-04 19:45:51 +03:00
case TMP435_DEVICE_ID :
kind = tmp435 ;
break ;
2009-12-09 22:35:54 +03:00
default :
return - ENODEV ;
2009-06-15 20:39:46 +04:00
}
2009-12-09 22:35:54 +03:00
reg = i2c_smbus_read_byte_data ( client , TMP401_CONFIG_READ ) ;
if ( reg & 0x1b )
return - ENODEV ;
reg = i2c_smbus_read_byte_data ( client , TMP401_CONVERSION_RATE_READ ) ;
/* Datasheet says: 0x1-0x6 */
if ( reg > 15 )
return - ENODEV ;
2010-03-06 00:17:26 +03:00
strlcpy ( info - > type , tmp401_id [ kind ] . name , I2C_NAME_SIZE ) ;
2009-06-15 20:39:46 +04:00
return 0 ;
}
static int tmp401_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
2014-12-04 19:45:51 +03:00
static const char * const names [ ] = {
2016-06-08 20:00:54 +03:00
" TMP401 " , " TMP411 " , " TMP431 " , " TMP432 " , " TMP435 " , " TMP461 "
2014-12-04 19:45:51 +03:00
} ;
2013-03-27 19:58:46 +04:00
struct device * dev = & client - > dev ;
2013-09-05 03:24:09 +04:00
struct device * hwmon_dev ;
2009-06-15 20:39:46 +04:00
struct tmp401_data * data ;
2014-12-04 19:45:53 +03:00
int groups = 0 , status ;
2009-06-15 20:39:46 +04:00
2013-03-27 19:58:46 +04:00
data = devm_kzalloc ( dev , sizeof ( struct tmp401_data ) , GFP_KERNEL ) ;
2009-06-15 20:39:46 +04:00
if ( ! data )
return - ENOMEM ;
2013-09-05 03:24:09 +04:00
data - > client = client ;
2009-06-15 20:39:46 +04:00
mutex_init ( & data - > update_lock ) ;
2009-06-15 20:39:47 +04:00
data - > kind = id - > driver_data ;
2009-06-15 20:39:46 +04:00
/* Initialize the TMP401 chip */
2014-12-04 19:45:53 +03:00
status = tmp401_init_client ( data , client ) ;
if ( status < 0 )
return status ;
2009-06-15 20:39:46 +04:00
/* Register sysfs hooks */
2013-09-05 03:24:09 +04:00
data - > groups [ groups + + ] = & tmp401_group ;
2009-06-15 20:39:46 +04:00
2012-02-11 17:55:58 +04:00
/* Register additional tmp411 sysfs hooks */
2013-09-05 03:24:09 +04:00
if ( data - > kind = = tmp411 )
data - > groups [ groups + + ] = & tmp411_group ;
2009-06-15 20:39:47 +04:00
2013-03-28 12:36:44 +04:00
/* Register additional tmp432 sysfs hooks */
2013-09-05 03:24:09 +04:00
if ( data - > kind = = tmp432 )
data - > groups [ groups + + ] = & tmp432_group ;
2013-03-28 12:36:44 +04:00
2016-06-08 20:00:54 +03:00
if ( data - > kind = = tmp461 )
data - > groups [ groups + + ] = & tmp461_group ;
2013-09-05 03:24:09 +04:00
hwmon_dev = devm_hwmon_device_register_with_groups ( dev , client - > name ,
data , data - > groups ) ;
if ( IS_ERR ( hwmon_dev ) )
return PTR_ERR ( hwmon_dev ) ;
2009-06-15 20:39:46 +04:00
2013-03-27 19:58:46 +04:00
dev_info ( dev , " Detected TI %s chip \n " , names [ data - > kind ] ) ;
2009-06-15 20:39:46 +04:00
return 0 ;
}
2010-05-27 21:58:49 +04:00
static struct i2c_driver tmp401_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " tmp401 " ,
} ,
. probe = tmp401_probe ,
. id_table = tmp401_id ,
. detect = tmp401_detect ,
. address_list = normal_i2c ,
} ;
2009-06-15 20:39:46 +04:00
2012-01-20 11:38:18 +04:00
module_i2c_driver ( tmp401_driver ) ;
2009-06-15 20:39:46 +04:00
MODULE_AUTHOR ( " Hans de Goede <hdegoede@redhat.com> " ) ;
MODULE_DESCRIPTION ( " Texas Instruments TMP401 temperature sensor driver " ) ;
MODULE_LICENSE ( " GPL " ) ;