2005-04-17 02:20:36 +04:00
/*
* lm90 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
2010-03-06 00:17:14 +03:00
* Copyright ( C ) 2003 - 2010 Jean Delvare < khali @ linux - fr . org >
2005-04-17 02:20:36 +04:00
*
* Based on the lm83 driver . The LM90 is a sensor chip made by National
* Semiconductor . It reports up to two temperatures ( its own plus up to
* one external one ) with a 0.125 deg resolution ( 1 deg for local
2008-10-17 19:51:10 +04:00
* temperature ) and a 3 - 4 deg accuracy .
2005-04-17 02:20:36 +04:00
*
* This driver also supports the LM89 and LM99 , two other sensor chips
* made by National Semiconductor . Both have an increased remote
* temperature measurement accuracy ( 1 degree ) , and the LM99
* additionally shifts remote temperatures ( measured and limits ) by 16
2008-10-26 19:04:39 +03:00
* degrees , which allows for higher temperatures measurement .
2005-05-04 04:21:25 +04:00
* Note that there is no way to differentiate between both chips .
2008-10-26 19:04:39 +03:00
* When device is auto - detected , the driver will assume an LM99 .
2005-04-17 02:20:36 +04:00
*
* This driver also supports the LM86 , another sensor chip made by
* National Semiconductor . It is exactly similar to the LM90 except it
* has a higher accuracy .
*
* This driver also supports the ADM1032 , a sensor chip made by Analog
* Devices . That chip is similar to the LM90 , with a few differences
2008-10-17 19:51:10 +04:00
* that are not handled by this driver . Among others , it has a higher
* accuracy than the LM90 , much like the LM86 does .
2005-04-17 02:20:36 +04:00
*
* This driver also supports the MAX6657 , MAX6658 and MAX6659 sensor
2008-10-17 19:51:10 +04:00
* chips made by Maxim . These chips are similar to the LM86 .
2005-05-04 04:21:25 +04:00
* Note that there is no easy way to differentiate between the three
2010-10-28 22:31:43 +04:00
* variants . We use the device address to detect MAX6659 , which will result
* in a detection as max6657 if it is on address 0x4c . The extra address
* and features of the MAX6659 are only supported if the chip is configured
* explicitly as max6659 , or if its address is not 0x4c .
* These chips lack the remote temperature offset feature .
2005-04-17 02:20:36 +04:00
*
2009-03-12 15:36:38 +03:00
* This driver also supports the MAX6646 , MAX6647 , MAX6648 , MAX6649 and
* MAX6692 chips made by Maxim . These are again similar to the LM86 ,
* but they use unsigned temperature values and can report temperatures
* from 0 to 145 degrees .
2008-10-17 19:51:11 +04:00
*
2007-06-09 18:11:16 +04:00
* This driver also supports the MAX6680 and MAX6681 , two other sensor
* chips made by Maxim . These are quite similar to the other Maxim
2008-10-17 19:51:10 +04:00
* chips . The MAX6680 and MAX6681 only differ in the pinout so they can
* be treated identically .
2007-06-09 18:11:16 +04:00
*
2010-10-28 22:31:43 +04:00
* This driver also supports the MAX6695 and MAX6696 , two other sensor
* chips made by Maxim . These are also quite similar to other Maxim
* chips , but support three temperature sensors instead of two . MAX6695
* and MAX6696 only differ in the pinout so they can be treated identically .
*
2011-04-29 18:33:35 +04:00
* This driver also supports ADT7461 and ADT7461A from Analog Devices as well as
* NCT1008 from ON Semiconductor . The chips are supported in both compatibility
* and extended mode . They are mostly compatible with LM90 except for a data
* format difference for the temperature value registers .
2005-04-17 02:20:36 +04:00
*
2011-06-06 14:40:45 +04:00
* This driver also supports the SA56004 from Philips . This device is
* pin - compatible with the LM86 , the ED / EDP parts are also address - compatible .
*
2005-04-17 02:20:36 +04:00
* Since the LM90 was the first chipset supported by this driver , most
* comments will refer to this chipset , but are actually general and
* concern all supported chipsets , unless mentioned otherwise .
*
* 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 .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
2005-06-06 21:34:45 +04:00
# include <linux/hwmon-sysfs.h>
2005-07-16 05:39:18 +04:00
# include <linux/hwmon.h>
# include <linux/err.h>
2006-01-19 01:19:26 +03:00
# include <linux/mutex.h>
2006-09-24 23:16:40 +04:00
# include <linux/sysfs.h>
2005-04-17 02:20:36 +04:00
/*
* Addresses to scan
* Address is fully defined internally and cannot be changed except for
2007-06-09 18:11:16 +04:00
* MAX6659 , MAX6680 and MAX6681 .
2011-04-29 18:33:35 +04:00
* LM86 , LM89 , LM90 , LM99 , ADM1032 , ADM1032 - 1 , ADT7461 , ADT7461A , MAX6649 ,
* MAX6657 , MAX6658 , NCT1008 and W83L771 have address 0x4c .
* ADM1032 - 2 , ADT7461 - 2 , ADT7461A - 2 , LM89 - 1 , LM99 - 1 , MAX6646 , and NCT1008D
* have address 0x4d .
2008-10-17 19:51:11 +04:00
* MAX6647 has address 0x4e .
2010-10-28 22:31:43 +04:00
* MAX6659 can have address 0x4c , 0x4d or 0x4e .
2007-06-09 18:11:16 +04:00
* MAX6680 and MAX6681 can have address 0x18 , 0x19 , 0x1a , 0x29 , 0x2a , 0x2b ,
* 0x4c , 0x4d or 0x4e .
2011-06-06 14:40:45 +04:00
* SA56004 can have address 0x48 through 0x4F .
2005-04-17 02:20:36 +04:00
*/
2008-02-18 06:28:03 +03:00
static const unsigned short normal_i2c [ ] = {
2011-06-06 14:40:45 +04:00
0x18 , 0x19 , 0x1a , 0x29 , 0x2a , 0x2b , 0x48 , 0x49 , 0x4a , 0x4b , 0x4c ,
0x4d , 0x4e , 0x4f , I2C_CLIENT_END } ;
2005-04-17 02:20:36 +04:00
2010-10-28 22:31:43 +04:00
enum chips { lm90 , adm1032 , lm99 , lm86 , max6657 , max6659 , adt7461 , max6680 ,
2011-06-06 14:40:45 +04:00
max6646 , w83l771 , max6696 , sa56004 } ;
2005-04-17 02:20:36 +04:00
/*
* The LM90 registers
*/
# define LM90_REG_R_MAN_ID 0xFE
# define LM90_REG_R_CHIP_ID 0xFF
# define LM90_REG_R_CONFIG1 0x03
# define LM90_REG_W_CONFIG1 0x09
# define LM90_REG_R_CONFIG2 0xBF
# define LM90_REG_W_CONFIG2 0xBF
# define LM90_REG_R_CONVRATE 0x04
# define LM90_REG_W_CONVRATE 0x0A
# define LM90_REG_R_STATUS 0x02
# define LM90_REG_R_LOCAL_TEMP 0x00
# define LM90_REG_R_LOCAL_HIGH 0x05
# define LM90_REG_W_LOCAL_HIGH 0x0B
# define LM90_REG_R_LOCAL_LOW 0x06
# define LM90_REG_W_LOCAL_LOW 0x0C
# define LM90_REG_R_LOCAL_CRIT 0x20
# define LM90_REG_W_LOCAL_CRIT 0x20
# define LM90_REG_R_REMOTE_TEMPH 0x01
# define LM90_REG_R_REMOTE_TEMPL 0x10
# define LM90_REG_R_REMOTE_OFFSH 0x11
# define LM90_REG_W_REMOTE_OFFSH 0x11
# define LM90_REG_R_REMOTE_OFFSL 0x12
# define LM90_REG_W_REMOTE_OFFSL 0x12
# define LM90_REG_R_REMOTE_HIGHH 0x07
# define LM90_REG_W_REMOTE_HIGHH 0x0D
# define LM90_REG_R_REMOTE_HIGHL 0x13
# define LM90_REG_W_REMOTE_HIGHL 0x13
# define LM90_REG_R_REMOTE_LOWH 0x08
# define LM90_REG_W_REMOTE_LOWH 0x0E
# define LM90_REG_R_REMOTE_LOWL 0x14
# define LM90_REG_W_REMOTE_LOWL 0x14
# define LM90_REG_R_REMOTE_CRIT 0x19
# define LM90_REG_W_REMOTE_CRIT 0x19
# define LM90_REG_R_TCRIT_HYST 0x21
# define LM90_REG_W_TCRIT_HYST 0x21
2010-10-28 22:31:43 +04:00
/* MAX6646/6647/6649/6657/6658/6659/6695/6696 registers */
2008-10-17 19:51:09 +04:00
# define MAX6657_REG_R_LOCAL_TEMPL 0x11
2010-10-28 22:31:43 +04:00
# define MAX6696_REG_R_STATUS2 0x12
2010-10-28 22:31:43 +04:00
# define MAX6659_REG_R_REMOTE_EMERG 0x16
# define MAX6659_REG_W_REMOTE_EMERG 0x16
# define MAX6659_REG_R_LOCAL_EMERG 0x17
# define MAX6659_REG_W_LOCAL_EMERG 0x17
2008-10-17 19:51:09 +04:00
2011-06-06 14:40:45 +04:00
/* SA56004 registers */
# define SA56004_REG_R_LOCAL_TEMPL 0x22
2010-10-28 22:31:44 +04:00
# define LM90_DEF_CONVRATE_RVAL 6 /* Def conversion rate register value */
# define LM90_MAX_CONVRATE_MS 16000 /* Maximum conversion rate in ms */
2008-10-17 19:51:10 +04:00
/*
* Device flags
*/
2010-10-28 22:31:43 +04:00
# define LM90_FLAG_ADT7461_EXT (1 << 0) /* ADT7461 extended mode */
/* Device features */
# define LM90_HAVE_OFFSET (1 << 1) /* temperature offset register */
# define LM90_HAVE_REM_LIMIT_EXT (1 << 3) /* extended remote limit */
2010-10-28 22:31:43 +04:00
# define LM90_HAVE_EMERGENCY (1 << 4) /* 3rd upper (emergency) limit */
2010-10-28 22:31:43 +04:00
# define LM90_HAVE_EMERGENCY_ALARM (1 << 5) /* emergency alarm */
# define LM90_HAVE_TEMP3 (1 << 6) /* 3rd temperature sensor */
2010-10-28 22:31:44 +04:00
# define LM90_HAVE_BROKEN_ALERT (1 << 7) /* Broken alert */
2008-10-17 19:51:10 +04:00
2005-04-17 02:20:36 +04:00
/*
* Driver data ( common to all clients )
*/
2008-07-16 21:30:15 +04:00
static const struct i2c_device_id lm90_id [ ] = {
{ " adm1032 " , adm1032 } ,
{ " adt7461 " , adt7461 } ,
2011-04-29 18:33:35 +04:00
{ " adt7461a " , adt7461 } ,
2008-07-16 21:30:15 +04:00
{ " lm90 " , lm90 } ,
{ " lm86 " , lm86 } ,
2008-10-26 19:04:39 +03:00
{ " lm89 " , lm86 } ,
{ " lm99 " , lm99 } ,
2008-10-17 19:51:11 +04:00
{ " max6646 " , max6646 } ,
{ " max6647 " , max6646 } ,
{ " max6649 " , max6646 } ,
2008-07-16 21:30:15 +04:00
{ " max6657 " , max6657 } ,
{ " max6658 " , max6657 } ,
2010-10-28 22:31:43 +04:00
{ " max6659 " , max6659 } ,
2008-07-16 21:30:15 +04:00
{ " max6680 " , max6680 } ,
{ " max6681 " , max6680 } ,
2010-10-28 22:31:43 +04:00
{ " max6695 " , max6696 } ,
{ " max6696 " , max6696 } ,
2011-04-29 18:33:35 +04:00
{ " nct1008 " , adt7461 } ,
2010-03-06 00:17:13 +03:00
{ " w83l771 " , w83l771 } ,
2011-06-06 14:40:45 +04:00
{ " sa56004 " , sa56004 } ,
2008-07-16 21:30:15 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm90_id ) ;
2010-10-28 22:31:43 +04:00
/*
* chip type specific parameters
*/
struct lm90_params {
u32 flags ; /* Capabilities */
u16 alert_alarms ; /* Which alarm bits trigger ALERT# */
/* Upper 8 bits for max6695/96 */
2010-10-28 22:31:44 +04:00
u8 max_convrate ; /* Maximum conversion rate register value */
2011-07-28 10:22:25 +04:00
u8 reg_local_ext ; /* Extended local temp register (optional) */
2010-10-28 22:31:43 +04:00
} ;
static const struct lm90_params lm90_params [ ] = {
[ adm1032 ] = {
2010-10-28 22:31:44 +04:00
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT
| LM90_HAVE_BROKEN_ALERT ,
2010-10-28 22:31:43 +04:00
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 10 ,
2010-10-28 22:31:43 +04:00
} ,
[ adt7461 ] = {
2010-10-28 22:31:44 +04:00
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT
| LM90_HAVE_BROKEN_ALERT ,
2010-10-28 22:31:43 +04:00
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 10 ,
2010-10-28 22:31:43 +04:00
} ,
[ lm86 ] = {
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT ,
. alert_alarms = 0x7b ,
2010-10-28 22:31:44 +04:00
. max_convrate = 9 ,
2010-10-28 22:31:43 +04:00
} ,
[ lm90 ] = {
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT ,
. alert_alarms = 0x7b ,
2010-10-28 22:31:44 +04:00
. max_convrate = 9 ,
2010-10-28 22:31:43 +04:00
} ,
[ lm99 ] = {
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT ,
. alert_alarms = 0x7b ,
2010-10-28 22:31:44 +04:00
. max_convrate = 9 ,
2010-10-28 22:31:43 +04:00
} ,
[ max6646 ] = {
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 6 ,
2011-06-06 14:40:45 +04:00
. reg_local_ext = MAX6657_REG_R_LOCAL_TEMPL ,
2010-10-28 22:31:43 +04:00
} ,
[ max6657 ] = {
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 8 ,
2011-06-06 14:40:45 +04:00
. reg_local_ext = MAX6657_REG_R_LOCAL_TEMPL ,
2010-10-28 22:31:43 +04:00
} ,
[ max6659 ] = {
2011-07-28 10:22:25 +04:00
. flags = LM90_HAVE_EMERGENCY ,
2010-10-28 22:31:43 +04:00
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 8 ,
2011-06-06 14:40:45 +04:00
. reg_local_ext = MAX6657_REG_R_LOCAL_TEMPL ,
2010-10-28 22:31:43 +04:00
} ,
[ max6680 ] = {
. flags = LM90_HAVE_OFFSET ,
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 7 ,
2010-10-28 22:31:43 +04:00
} ,
[ max6696 ] = {
2011-07-28 10:22:25 +04:00
. flags = LM90_HAVE_EMERGENCY
2010-10-28 22:31:43 +04:00
| LM90_HAVE_EMERGENCY_ALARM | LM90_HAVE_TEMP3 ,
. alert_alarms = 0x187c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 6 ,
2011-06-06 14:40:45 +04:00
. reg_local_ext = MAX6657_REG_R_LOCAL_TEMPL ,
2010-10-28 22:31:43 +04:00
} ,
[ w83l771 ] = {
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT ,
. alert_alarms = 0x7c ,
2010-10-28 22:31:44 +04:00
. max_convrate = 8 ,
2010-10-28 22:31:43 +04:00
} ,
2011-06-06 14:40:45 +04:00
[ sa56004 ] = {
2011-07-28 10:22:25 +04:00
. flags = LM90_HAVE_OFFSET | LM90_HAVE_REM_LIMIT_EXT ,
2011-06-06 14:40:45 +04:00
. alert_alarms = 0x7b ,
. max_convrate = 9 ,
. reg_local_ext = SA56004_REG_R_LOCAL_TEMPL ,
} ,
2010-10-28 22:31:43 +04:00
} ;
2005-04-17 02:20:36 +04:00
/*
* Client data ( each client gets its own )
*/
struct lm90_data {
2007-08-21 00:46:20 +04:00
struct device * hwmon_dev ;
2006-01-19 01:19:26 +03:00
struct mutex update_lock ;
2005-04-17 02:20:36 +04:00
char valid ; /* zero until following fields are valid */
unsigned long last_updated ; /* in jiffies */
int kind ;
2010-10-28 22:31:43 +04:00
u32 flags ;
2005-04-17 02:20:36 +04:00
2010-10-28 22:31:44 +04:00
int update_interval ; /* in milliseconds */
2010-03-06 00:17:14 +03:00
u8 config_orig ; /* Original configuration register value */
2010-10-28 22:31:44 +04:00
u8 convrate_orig ; /* Original conversion rate register value */
2010-10-28 22:31:43 +04:00
u16 alert_alarms ; /* Which alarm bits trigger ALERT# */
/* Upper 8 bits for max6695/96 */
2010-10-28 22:31:44 +04:00
u8 max_convrate ; /* Maximum conversion rate */
2011-06-06 14:40:45 +04:00
u8 reg_local_ext ; /* local extension register offset */
2010-03-06 00:17:14 +03:00
2005-04-17 02:20:36 +04:00
/* registers values */
2010-10-28 22:31:43 +04:00
s8 temp8 [ 8 ] ; /* 0: local low limit
2008-10-17 19:51:09 +04:00
1 : local high limit
2 : local critical limit
2010-10-28 22:31:43 +04:00
3 : remote critical limit
2010-10-28 22:31:43 +04:00
4 : local emergency limit ( max6659 and max6695 / 96 )
5 : remote emergency limit ( max6659 and max6695 / 96 )
6 : remote 2 critical limit ( max6695 / 96 only )
7 : remote 2 emergency limit ( max6695 / 96 only ) */
s16 temp11 [ 8 ] ; /* 0: remote input
2005-06-05 23:27:28 +04:00
1 : remote low limit
2007-09-05 16:15:37 +04:00
2 : remote high limit
2010-10-28 22:31:43 +04:00
3 : remote offset ( except max6646 , max6657 / 58 / 59 ,
and max6695 / 96 )
4 : local input
5 : remote 2 input ( max6695 / 96 only )
6 : remote 2 low limit ( max6695 / 96 only )
7 : remote 2 high limit ( ma6695 / 96 only ) */
2005-04-17 02:20:36 +04:00
u8 temp_hyst ;
2010-10-28 22:31:43 +04:00
u16 alarms ; /* bitvector (upper 8 bits for max6695/96) */
2005-04-17 02:20:36 +04:00
} ;
2010-10-28 22:31:43 +04:00
/*
* Support functions
*/
/*
* The ADM1032 supports PEC but not on write byte transactions , so we need
* to explicitly ask for a transaction without PEC .
*/
static inline s32 adm1032_write_byte ( struct i2c_client * client , u8 value )
{
return i2c_smbus_xfer ( client - > adapter , client - > addr ,
client - > flags & ~ I2C_CLIENT_PEC ,
I2C_SMBUS_WRITE , value , I2C_SMBUS_BYTE , NULL ) ;
}
/*
* It is assumed that client - > update_lock is held ( unless we are in
* detection or initialization steps ) . This matters when PEC is enabled ,
* because we don ' t want the address pointer to change between the write
* byte and the read byte transactions .
*/
static int lm90_read_reg ( struct i2c_client * client , u8 reg , u8 * value )
{
int err ;
if ( client - > flags & I2C_CLIENT_PEC ) {
err = adm1032_write_byte ( client , reg ) ;
if ( err > = 0 )
err = i2c_smbus_read_byte ( client ) ;
} else
err = i2c_smbus_read_byte_data ( client , reg ) ;
if ( err < 0 ) {
dev_warn ( & client - > dev , " Register %#02x read failed (%d) \n " ,
reg , err ) ;
return err ;
}
* value = err ;
return 0 ;
}
static int lm90_read16 ( struct i2c_client * client , u8 regh , u8 regl , u16 * value )
{
int err ;
u8 oldh , newh , l ;
/*
* There is a trick here . We have to read two registers to have the
* sensor temperature , but we have to beware a conversion could occur
2011-03-31 05:57:33 +04:00
* between the readings . The datasheet says we should either use
2010-10-28 22:31:43 +04:00
* the one - shot conversion register , which we don ' t want to do
* ( disables hardware monitoring ) or monitor the busy bit , which is
* impossible ( we can ' t read the values and monitor that bit at the
* exact same time ) . So the solution used here is to read the high
* byte once , then the low byte , then the high byte again . If the new
* high byte matches the old one , then we have a valid reading . Else
* we have to read the low byte again , and now we believe we have a
* correct reading .
*/
if ( ( err = lm90_read_reg ( client , regh , & oldh ) )
| | ( err = lm90_read_reg ( client , regl , & l ) )
| | ( err = lm90_read_reg ( client , regh , & newh ) ) )
return err ;
if ( oldh ! = newh ) {
err = lm90_read_reg ( client , regl , & l ) ;
if ( err )
return err ;
}
* value = ( newh < < 8 ) | l ;
return 0 ;
}
/*
* client - > update_lock must be held when calling this function ( unless we are
* in detection or initialization steps ) , and while a remote channel other
* than channel 0 is selected . Also , calling code must make sure to re - select
* external channel 0 before releasing the lock . This is necessary because
* various registers have different meanings as a result of selecting a
* non - default remote channel .
*/
static inline void lm90_select_remote_channel ( struct i2c_client * client ,
struct lm90_data * data ,
int channel )
{
u8 config ;
if ( data - > kind = = max6696 ) {
lm90_read_reg ( client , LM90_REG_R_CONFIG1 , & config ) ;
config & = ~ 0x08 ;
if ( channel )
config | = 0x08 ;
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONFIG1 ,
config ) ;
}
}
2010-10-28 22:31:44 +04:00
/*
* Set conversion rate .
* client - > update_lock must be held when calling this function ( unless we are
* in detection or initialization steps ) .
*/
static void lm90_set_convrate ( struct i2c_client * client , struct lm90_data * data ,
unsigned int interval )
{
int i ;
unsigned int update_interval ;
/* Shift calculations to avoid rounding errors */
interval < < = 6 ;
/* find the nearest update rate */
for ( i = 0 , update_interval = LM90_MAX_CONVRATE_MS < < 6 ;
i < data - > max_convrate ; i + + , update_interval > > = 1 )
if ( interval > = update_interval * 3 / 4 )
break ;
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONVRATE , i ) ;
data - > update_interval = DIV_ROUND_CLOSEST ( update_interval , 64 ) ;
}
2010-10-28 22:31:43 +04:00
static struct lm90_data * lm90_update_device ( struct device * dev )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2010-10-28 22:31:44 +04:00
unsigned long next_update ;
2010-10-28 22:31:43 +04:00
mutex_lock ( & data - > update_lock ) ;
2010-10-28 22:31:44 +04:00
next_update = data - > last_updated
+ msecs_to_jiffies ( data - > update_interval ) + 1 ;
if ( time_after ( jiffies , next_update ) | | ! data - > valid ) {
2010-10-28 22:31:43 +04:00
u8 h , l ;
u8 alarms ;
dev_dbg ( & client - > dev , " Updating lm90 data. \n " ) ;
lm90_read_reg ( client , LM90_REG_R_LOCAL_LOW , & data - > temp8 [ 0 ] ) ;
lm90_read_reg ( client , LM90_REG_R_LOCAL_HIGH , & data - > temp8 [ 1 ] ) ;
lm90_read_reg ( client , LM90_REG_R_LOCAL_CRIT , & data - > temp8 [ 2 ] ) ;
lm90_read_reg ( client , LM90_REG_R_REMOTE_CRIT , & data - > temp8 [ 3 ] ) ;
lm90_read_reg ( client , LM90_REG_R_TCRIT_HYST , & data - > temp_hyst ) ;
2011-07-28 10:22:25 +04:00
if ( data - > reg_local_ext ) {
2010-10-28 22:31:43 +04:00
lm90_read16 ( client , LM90_REG_R_LOCAL_TEMP ,
2011-06-06 14:40:45 +04:00
data - > reg_local_ext ,
2010-10-28 22:31:43 +04:00
& data - > temp11 [ 4 ] ) ;
} else {
if ( lm90_read_reg ( client , LM90_REG_R_LOCAL_TEMP ,
& h ) = = 0 )
data - > temp11 [ 4 ] = h < < 8 ;
}
lm90_read16 ( client , LM90_REG_R_REMOTE_TEMPH ,
LM90_REG_R_REMOTE_TEMPL , & data - > temp11 [ 0 ] ) ;
if ( lm90_read_reg ( client , LM90_REG_R_REMOTE_LOWH , & h ) = = 0 ) {
data - > temp11 [ 1 ] = h < < 8 ;
if ( ( data - > flags & LM90_HAVE_REM_LIMIT_EXT )
& & lm90_read_reg ( client , LM90_REG_R_REMOTE_LOWL ,
& l ) = = 0 )
data - > temp11 [ 1 ] | = l ;
}
if ( lm90_read_reg ( client , LM90_REG_R_REMOTE_HIGHH , & h ) = = 0 ) {
data - > temp11 [ 2 ] = h < < 8 ;
if ( ( data - > flags & LM90_HAVE_REM_LIMIT_EXT )
& & lm90_read_reg ( client , LM90_REG_R_REMOTE_HIGHL ,
& l ) = = 0 )
data - > temp11 [ 2 ] | = l ;
}
if ( data - > flags & LM90_HAVE_OFFSET ) {
if ( lm90_read_reg ( client , LM90_REG_R_REMOTE_OFFSH ,
& h ) = = 0
& & lm90_read_reg ( client , LM90_REG_R_REMOTE_OFFSL ,
& l ) = = 0 )
data - > temp11 [ 3 ] = ( h < < 8 ) | l ;
}
if ( data - > flags & LM90_HAVE_EMERGENCY ) {
lm90_read_reg ( client , MAX6659_REG_R_LOCAL_EMERG ,
& data - > temp8 [ 4 ] ) ;
lm90_read_reg ( client , MAX6659_REG_R_REMOTE_EMERG ,
& data - > temp8 [ 5 ] ) ;
}
lm90_read_reg ( client , LM90_REG_R_STATUS , & alarms ) ;
data - > alarms = alarms ; /* save as 16 bit value */
if ( data - > kind = = max6696 ) {
lm90_select_remote_channel ( client , data , 1 ) ;
lm90_read_reg ( client , LM90_REG_R_REMOTE_CRIT ,
& data - > temp8 [ 6 ] ) ;
lm90_read_reg ( client , MAX6659_REG_R_REMOTE_EMERG ,
& data - > temp8 [ 7 ] ) ;
lm90_read16 ( client , LM90_REG_R_REMOTE_TEMPH ,
LM90_REG_R_REMOTE_TEMPL , & data - > temp11 [ 5 ] ) ;
if ( ! lm90_read_reg ( client , LM90_REG_R_REMOTE_LOWH , & h ) )
data - > temp11 [ 6 ] = h < < 8 ;
if ( ! lm90_read_reg ( client , LM90_REG_R_REMOTE_HIGHH , & h ) )
data - > temp11 [ 7 ] = h < < 8 ;
lm90_select_remote_channel ( client , data , 0 ) ;
if ( ! lm90_read_reg ( client , MAX6696_REG_R_STATUS2 ,
& alarms ) )
data - > alarms | = alarms < < 8 ;
}
/* Re-enable ALERT# output if it was originally enabled and
* relevant alarms are all clear */
if ( ( data - > config_orig & 0x80 ) = = 0
& & ( data - > alarms & data - > alert_alarms ) = = 0 ) {
u8 config ;
lm90_read_reg ( client , LM90_REG_R_CONFIG1 , & config ) ;
if ( config & 0x80 ) {
dev_dbg ( & client - > dev , " Re-enabling ALERT# \n " ) ;
i2c_smbus_write_byte_data ( client ,
LM90_REG_W_CONFIG1 ,
config & ~ 0x80 ) ;
}
}
data - > last_updated = jiffies ;
data - > valid = 1 ;
}
mutex_unlock ( & data - > update_lock ) ;
return data ;
}
2008-10-17 19:51:10 +04:00
/*
* Conversions
* For local temperatures and limits , critical limits and the hysteresis
* value , the LM90 uses signed 8 - bit values with LSB = 1 degree Celsius .
* For remote temperatures and limits , it uses signed 11 - bit values with
2008-10-17 19:51:11 +04:00
* LSB = 0.125 degree Celsius , left - justified in 16 - bit registers . Some
* Maxim chips use unsigned values .
2008-10-17 19:51:10 +04:00
*/
2008-10-17 19:51:10 +04:00
static inline int temp_from_s8 ( s8 val )
2008-10-17 19:51:10 +04:00
{
return val * 1000 ;
}
2008-10-17 19:51:11 +04:00
static inline int temp_from_u8 ( u8 val )
{
return val * 1000 ;
}
2008-10-17 19:51:10 +04:00
static inline int temp_from_s16 ( s16 val )
2008-10-17 19:51:10 +04:00
{
return val / 32 * 125 ;
}
2008-10-17 19:51:11 +04:00
static inline int temp_from_u16 ( u16 val )
{
return val / 32 * 125 ;
}
2008-10-17 19:51:10 +04:00
static s8 temp_to_s8 ( long val )
2008-10-17 19:51:10 +04:00
{
if ( val < = - 128000 )
return - 128 ;
if ( val > = 127000 )
return 127 ;
if ( val < 0 )
return ( val - 500 ) / 1000 ;
return ( val + 500 ) / 1000 ;
}
2008-10-17 19:51:11 +04:00
static u8 temp_to_u8 ( long val )
{
if ( val < = 0 )
return 0 ;
if ( val > = 255000 )
return 255 ;
return ( val + 500 ) / 1000 ;
}
2008-10-17 19:51:10 +04:00
static s16 temp_to_s16 ( long val )
2008-10-17 19:51:10 +04:00
{
if ( val < = - 128000 )
return 0x8000 ;
if ( val > = 127875 )
return 0x7FE0 ;
if ( val < 0 )
return ( val - 62 ) / 125 * 32 ;
return ( val + 62 ) / 125 * 32 ;
}
static u8 hyst_to_reg ( long val )
{
if ( val < = 0 )
return 0 ;
if ( val > = 30500 )
return 31 ;
return ( val + 500 ) / 1000 ;
}
/*
2008-10-17 19:51:10 +04:00
* ADT7461 in compatibility mode is almost identical to LM90 except that
* attempts to write values that are outside the range 0 < temp < 127 are
* treated as the boundary value .
*
* ADT7461 in " extended mode " operation uses unsigned integers offset by
* 64 ( e . g . , 0 - > - 64 degC ) . The range is restricted to - 64. .191 degC .
2008-10-17 19:51:10 +04:00
*/
2008-10-17 19:51:10 +04:00
static inline int temp_from_u8_adt7461 ( struct lm90_data * data , u8 val )
2008-10-17 19:51:10 +04:00
{
2008-10-17 19:51:10 +04:00
if ( data - > flags & LM90_FLAG_ADT7461_EXT )
return ( val - 64 ) * 1000 ;
else
2008-10-17 19:51:10 +04:00
return temp_from_s8 ( val ) ;
2008-10-17 19:51:10 +04:00
}
2008-10-17 19:51:10 +04:00
static inline int temp_from_u16_adt7461 ( struct lm90_data * data , u16 val )
2008-10-17 19:51:10 +04:00
{
2008-10-17 19:51:10 +04:00
if ( data - > flags & LM90_FLAG_ADT7461_EXT )
return ( val - 0x4000 ) / 64 * 250 ;
else
2008-10-17 19:51:10 +04:00
return temp_from_s16 ( val ) ;
2008-10-17 19:51:10 +04:00
}
2008-10-17 19:51:10 +04:00
static u8 temp_to_u8_adt7461 ( struct lm90_data * data , long val )
2008-10-17 19:51:10 +04:00
{
if ( data - > flags & LM90_FLAG_ADT7461_EXT ) {
if ( val < = - 64000 )
return 0 ;
if ( val > = 191000 )
return 0xFF ;
return ( val + 500 + 64000 ) / 1000 ;
} else {
if ( val < = 0 )
return 0 ;
if ( val > = 127000 )
return 127 ;
return ( val + 500 ) / 1000 ;
}
}
2008-10-17 19:51:10 +04:00
static u16 temp_to_u16_adt7461 ( struct lm90_data * data , long val )
2008-10-17 19:51:10 +04:00
{
if ( data - > flags & LM90_FLAG_ADT7461_EXT ) {
if ( val < = - 64000 )
return 0 ;
if ( val > = 191750 )
return 0xFFC0 ;
return ( val + 64000 + 125 ) / 250 * 64 ;
} else {
if ( val < = 0 )
return 0 ;
if ( val > = 127750 )
return 0x7FC0 ;
return ( val + 125 ) / 250 * 64 ;
}
2008-10-17 19:51:10 +04:00
}
2005-04-17 02:20:36 +04:00
/*
* Sysfs stuff
*/
2005-06-05 23:27:28 +04:00
static ssize_t show_temp8 ( struct device * dev , struct device_attribute * devattr ,
char * buf )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct lm90_data * data = lm90_update_device ( dev ) ;
2008-10-17 19:51:10 +04:00
int temp ;
if ( data - > kind = = adt7461 )
2008-10-17 19:51:10 +04:00
temp = temp_from_u8_adt7461 ( data , data - > temp8 [ attr - > index ] ) ;
2008-10-17 19:51:11 +04:00
else if ( data - > kind = = max6646 )
temp = temp_from_u8 ( data - > temp8 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
else
2008-10-17 19:51:10 +04:00
temp = temp_from_s8 ( data - > temp8 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
2008-10-26 19:04:39 +03:00
/* +16 degrees offset for temp2 for the LM99 */
if ( data - > kind = = lm99 & & attr - > index = = 3 )
temp + = 16000 ;
2008-10-17 19:51:10 +04:00
return sprintf ( buf , " %d \n " , temp ) ;
2005-06-05 23:27:28 +04:00
}
static ssize_t set_temp8 ( struct device * dev , struct device_attribute * devattr ,
const char * buf , size_t count )
{
2010-10-28 22:31:43 +04:00
static const u8 reg [ 8 ] = {
2005-06-05 23:27:28 +04:00
LM90_REG_W_LOCAL_LOW ,
LM90_REG_W_LOCAL_HIGH ,
LM90_REG_W_LOCAL_CRIT ,
LM90_REG_W_REMOTE_CRIT ,
2010-10-28 22:31:43 +04:00
MAX6659_REG_W_LOCAL_EMERG ,
MAX6659_REG_W_REMOTE_EMERG ,
2010-10-28 22:31:43 +04:00
LM90_REG_W_REMOTE_CRIT ,
MAX6659_REG_W_REMOTE_EMERG ,
2005-06-05 23:27:28 +04:00
} ;
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm90_data * data = i2c_get_clientdata ( client ) ;
int nr = attr - > index ;
2010-10-28 22:31:42 +04:00
long val ;
int err ;
err = strict_strtol ( buf , 10 , & val ) ;
if ( err < 0 )
return err ;
2005-06-05 23:27:28 +04:00
2008-10-26 19:04:39 +03:00
/* +16 degrees offset for temp2 for the LM99 */
if ( data - > kind = = lm99 & & attr - > index = = 3 )
val - = 16000 ;
2006-01-19 01:19:26 +03:00
mutex_lock ( & data - > update_lock ) ;
2005-06-05 23:27:28 +04:00
if ( data - > kind = = adt7461 )
2008-10-17 19:51:10 +04:00
data - > temp8 [ nr ] = temp_to_u8_adt7461 ( data , val ) ;
2008-10-17 19:51:11 +04:00
else if ( data - > kind = = max6646 )
data - > temp8 [ nr ] = temp_to_u8 ( val ) ;
2005-06-05 23:27:28 +04:00
else
2008-10-17 19:51:10 +04:00
data - > temp8 [ nr ] = temp_to_s8 ( val ) ;
2010-10-28 22:31:43 +04:00
lm90_select_remote_channel ( client , data , nr > = 6 ) ;
2008-10-17 19:51:09 +04:00
i2c_smbus_write_byte_data ( client , reg [ nr ] , data - > temp8 [ nr ] ) ;
2010-10-28 22:31:43 +04:00
lm90_select_remote_channel ( client , data , 0 ) ;
2006-01-19 01:19:26 +03:00
mutex_unlock ( & data - > update_lock ) ;
2005-06-05 23:27:28 +04:00
return count ;
2005-04-17 02:20:36 +04:00
}
2005-06-05 23:27:28 +04:00
static ssize_t show_temp11 ( struct device * dev , struct device_attribute * devattr ,
char * buf )
{
2010-10-28 22:31:43 +04:00
struct sensor_device_attribute_2 * attr = to_sensor_dev_attr_2 ( devattr ) ;
2005-06-05 23:27:28 +04:00
struct lm90_data * data = lm90_update_device ( dev ) ;
2008-10-17 19:51:10 +04:00
int temp ;
if ( data - > kind = = adt7461 )
2008-10-17 19:51:10 +04:00
temp = temp_from_u16_adt7461 ( data , data - > temp11 [ attr - > index ] ) ;
2008-10-17 19:51:11 +04:00
else if ( data - > kind = = max6646 )
temp = temp_from_u16 ( data - > temp11 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
else
2008-10-17 19:51:10 +04:00
temp = temp_from_s16 ( data - > temp11 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
2008-10-26 19:04:39 +03:00
/* +16 degrees offset for temp2 for the LM99 */
if ( data - > kind = = lm99 & & attr - > index < = 2 )
temp + = 16000 ;
2008-10-17 19:51:10 +04:00
return sprintf ( buf , " %d \n " , temp ) ;
2005-04-17 02:20:36 +04:00
}
2005-06-05 23:27:28 +04:00
static ssize_t set_temp11 ( struct device * dev , struct device_attribute * devattr ,
const char * buf , size_t count )
{
2010-10-28 22:31:43 +04:00
struct {
u8 high ;
u8 low ;
2010-10-28 22:31:43 +04:00
int channel ;
} reg [ 5 ] = {
{ LM90_REG_W_REMOTE_LOWH , LM90_REG_W_REMOTE_LOWL , 0 } ,
{ LM90_REG_W_REMOTE_HIGHH , LM90_REG_W_REMOTE_HIGHL , 0 } ,
{ LM90_REG_W_REMOTE_OFFSH , LM90_REG_W_REMOTE_OFFSL , 0 } ,
{ LM90_REG_W_REMOTE_LOWH , LM90_REG_W_REMOTE_LOWL , 1 } ,
{ LM90_REG_W_REMOTE_HIGHH , LM90_REG_W_REMOTE_HIGHL , 1 }
2005-06-05 23:27:28 +04:00
} ;
2010-10-28 22:31:43 +04:00
struct sensor_device_attribute_2 * attr = to_sensor_dev_attr_2 ( devattr ) ;
2005-06-05 23:27:28 +04:00
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2010-10-28 22:31:43 +04:00
int nr = attr - > nr ;
int index = attr - > index ;
2010-10-28 22:31:42 +04:00
long val ;
int err ;
err = strict_strtol ( buf , 10 , & val ) ;
if ( err < 0 )
return err ;
2005-06-05 23:27:28 +04:00
2008-10-26 19:04:39 +03:00
/* +16 degrees offset for temp2 for the LM99 */
2010-10-28 22:31:43 +04:00
if ( data - > kind = = lm99 & & index < = 2 )
2008-10-26 19:04:39 +03:00
val - = 16000 ;
2006-01-19 01:19:26 +03:00
mutex_lock ( & data - > update_lock ) ;
2005-06-05 23:27:28 +04:00
if ( data - > kind = = adt7461 )
2010-10-28 22:31:43 +04:00
data - > temp11 [ index ] = temp_to_u16_adt7461 ( data , val ) ;
2008-10-17 19:51:11 +04:00
else if ( data - > kind = = max6646 )
2010-10-28 22:31:43 +04:00
data - > temp11 [ index ] = temp_to_u8 ( val ) < < 8 ;
2010-10-28 22:31:43 +04:00
else if ( data - > flags & LM90_HAVE_REM_LIMIT_EXT )
2010-10-28 22:31:43 +04:00
data - > temp11 [ index ] = temp_to_s16 ( val ) ;
2010-10-28 22:31:43 +04:00
else
2010-10-28 22:31:43 +04:00
data - > temp11 [ index ] = temp_to_s8 ( val ) < < 8 ;
2008-10-17 19:51:09 +04:00
2010-10-28 22:31:43 +04:00
lm90_select_remote_channel ( client , data , reg [ nr ] . channel ) ;
2010-10-28 22:31:43 +04:00
i2c_smbus_write_byte_data ( client , reg [ nr ] . high ,
data - > temp11 [ index ] > > 8 ) ;
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_REM_LIMIT_EXT )
2010-10-28 22:31:43 +04:00
i2c_smbus_write_byte_data ( client , reg [ nr ] . low ,
data - > temp11 [ index ] & 0xff ) ;
2010-10-28 22:31:43 +04:00
lm90_select_remote_channel ( client , data , 0 ) ;
2006-01-19 01:19:26 +03:00
mutex_unlock ( & data - > update_lock ) ;
2005-06-05 23:27:28 +04:00
return count ;
2005-04-17 02:20:36 +04:00
}
2005-06-05 23:27:28 +04:00
2010-10-28 22:31:42 +04:00
static ssize_t show_temphyst ( struct device * dev ,
struct device_attribute * devattr ,
2005-06-05 23:27:28 +04:00
char * buf )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct lm90_data * data = lm90_update_device ( dev ) ;
2008-10-17 19:51:10 +04:00
int temp ;
if ( data - > kind = = adt7461 )
2008-10-17 19:51:10 +04:00
temp = temp_from_u8_adt7461 ( data , data - > temp8 [ attr - > index ] ) ;
2008-10-26 19:04:39 +03:00
else if ( data - > kind = = max6646 )
temp = temp_from_u8 ( data - > temp8 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
else
2008-10-17 19:51:10 +04:00
temp = temp_from_s8 ( data - > temp8 [ attr - > index ] ) ;
2008-10-17 19:51:10 +04:00
2008-10-26 19:04:39 +03:00
/* +16 degrees offset for temp2 for the LM99 */
if ( data - > kind = = lm99 & & attr - > index = = 3 )
temp + = 16000 ;
2008-10-17 19:51:10 +04:00
return sprintf ( buf , " %d \n " , temp - temp_from_s8 ( data - > temp_hyst ) ) ;
2005-04-17 02:20:36 +04:00
}
2005-06-05 23:27:28 +04:00
static ssize_t set_temphyst ( struct device * dev , struct device_attribute * dummy ,
const char * buf , size_t count )
2005-04-17 02:20:36 +04:00
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2010-10-28 22:31:42 +04:00
long val ;
int err ;
2008-10-26 19:04:39 +03:00
int temp ;
2005-04-17 02:20:36 +04:00
2010-10-28 22:31:42 +04:00
err = strict_strtol ( buf , 10 , & val ) ;
if ( err < 0 )
return err ;
2006-01-19 01:19:26 +03:00
mutex_lock ( & data - > update_lock ) ;
2008-10-26 19:04:39 +03:00
if ( data - > kind = = adt7461 )
temp = temp_from_u8_adt7461 ( data , data - > temp8 [ 2 ] ) ;
else if ( data - > kind = = max6646 )
temp = temp_from_u8 ( data - > temp8 [ 2 ] ) ;
else
temp = temp_from_s8 ( data - > temp8 [ 2 ] ) ;
data - > temp_hyst = hyst_to_reg ( temp - val ) ;
2005-04-17 02:20:36 +04:00
i2c_smbus_write_byte_data ( client , LM90_REG_W_TCRIT_HYST ,
2008-10-26 19:04:39 +03:00
data - > temp_hyst ) ;
2006-01-19 01:19:26 +03:00
mutex_unlock ( & data - > update_lock ) ;
2005-04-17 02:20:36 +04:00
return count ;
}
2005-06-05 23:27:28 +04:00
static ssize_t show_alarms ( struct device * dev , struct device_attribute * dummy ,
char * buf )
2005-04-17 02:20:36 +04:00
{
struct lm90_data * data = lm90_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , data - > alarms ) ;
}
2006-09-24 22:52:15 +04:00
static ssize_t show_alarm ( struct device * dev , struct device_attribute
* devattr , char * buf )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct lm90_data * data = lm90_update_device ( dev ) ;
int bitnr = attr - > index ;
return sprintf ( buf , " %d \n " , ( data - > alarms > > bitnr ) & 1 ) ;
}
2010-10-28 22:31:44 +04:00
static ssize_t show_update_interval ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct lm90_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %u \n " , data - > update_interval ) ;
}
static ssize_t set_update_interval ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm90_data * data = i2c_get_clientdata ( client ) ;
unsigned long val ;
int err ;
err = strict_strtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
mutex_lock ( & data - > update_lock ) ;
lm90_set_convrate ( client , data , val ) ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
2010-10-28 22:31:43 +04:00
static SENSOR_DEVICE_ATTR_2 ( temp1_input , S_IRUGO , show_temp11 , NULL , 0 , 4 ) ;
static SENSOR_DEVICE_ATTR_2 ( temp2_input , S_IRUGO , show_temp11 , NULL , 0 , 0 ) ;
2005-06-05 23:27:28 +04:00
static SENSOR_DEVICE_ATTR ( temp1_min , S_IWUSR | S_IRUGO , show_temp8 ,
2008-10-17 19:51:09 +04:00
set_temp8 , 0 ) ;
2010-10-28 22:31:43 +04:00
static SENSOR_DEVICE_ATTR_2 ( temp2_min , S_IWUSR | S_IRUGO , show_temp11 ,
set_temp11 , 0 , 1 ) ;
2005-06-05 23:27:28 +04:00
static SENSOR_DEVICE_ATTR ( temp1_max , S_IWUSR | S_IRUGO , show_temp8 ,
2008-10-17 19:51:09 +04:00
set_temp8 , 1 ) ;
2010-10-28 22:31:43 +04:00
static SENSOR_DEVICE_ATTR_2 ( temp2_max , S_IWUSR | S_IRUGO , show_temp11 ,
set_temp11 , 1 , 2 ) ;
2005-06-05 23:27:28 +04:00
static SENSOR_DEVICE_ATTR ( temp1_crit , S_IWUSR | S_IRUGO , show_temp8 ,
2008-10-17 19:51:09 +04:00
set_temp8 , 2 ) ;
2005-06-05 23:27:28 +04:00
static SENSOR_DEVICE_ATTR ( temp2_crit , S_IWUSR | S_IRUGO , show_temp8 ,
2008-10-17 19:51:09 +04:00
set_temp8 , 3 ) ;
2005-06-05 23:27:28 +04:00
static SENSOR_DEVICE_ATTR ( temp1_crit_hyst , S_IWUSR | S_IRUGO , show_temphyst ,
2008-10-17 19:51:09 +04:00
set_temphyst , 2 ) ;
static SENSOR_DEVICE_ATTR ( temp2_crit_hyst , S_IRUGO , show_temphyst , NULL , 3 ) ;
2010-10-28 22:31:43 +04:00
static SENSOR_DEVICE_ATTR_2 ( temp2_offset , S_IWUSR | S_IRUGO , show_temp11 ,
set_temp11 , 2 , 3 ) ;
2006-09-24 22:52:15 +04:00
/* Individual alarm files */
static SENSOR_DEVICE_ATTR ( temp1_crit_alarm , S_IRUGO , show_alarm , NULL , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp2_crit_alarm , S_IRUGO , show_alarm , NULL , 1 ) ;
2007-06-09 18:11:16 +04:00
static SENSOR_DEVICE_ATTR ( temp2_fault , S_IRUGO , show_alarm , NULL , 2 ) ;
2006-09-24 22:52:15 +04:00
static SENSOR_DEVICE_ATTR ( temp2_min_alarm , S_IRUGO , show_alarm , NULL , 3 ) ;
static SENSOR_DEVICE_ATTR ( temp2_max_alarm , S_IRUGO , show_alarm , NULL , 4 ) ;
static SENSOR_DEVICE_ATTR ( temp1_min_alarm , S_IRUGO , show_alarm , NULL , 5 ) ;
static SENSOR_DEVICE_ATTR ( temp1_max_alarm , S_IRUGO , show_alarm , NULL , 6 ) ;
/* Raw alarm file for compatibility */
2005-04-17 02:20:36 +04:00
static DEVICE_ATTR ( alarms , S_IRUGO , show_alarms , NULL ) ;
2010-10-28 22:31:44 +04:00
static DEVICE_ATTR ( update_interval , S_IRUGO | S_IWUSR , show_update_interval ,
set_update_interval ) ;
2006-09-24 23:16:40 +04:00
static struct attribute * lm90_attributes [ ] = {
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp2_input . dev_attr . attr ,
& sensor_dev_attr_temp1_min . dev_attr . attr ,
& sensor_dev_attr_temp2_min . dev_attr . attr ,
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp2_max . dev_attr . attr ,
& sensor_dev_attr_temp1_crit . dev_attr . attr ,
& sensor_dev_attr_temp2_crit . dev_attr . attr ,
& sensor_dev_attr_temp1_crit_hyst . dev_attr . attr ,
& sensor_dev_attr_temp2_crit_hyst . dev_attr . attr ,
& sensor_dev_attr_temp1_crit_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_crit_alarm . dev_attr . attr ,
2007-06-09 18:11:16 +04:00
& sensor_dev_attr_temp2_fault . dev_attr . attr ,
2006-09-24 23:16:40 +04:00
& sensor_dev_attr_temp2_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_max_alarm . dev_attr . attr ,
& dev_attr_alarms . attr ,
2010-10-28 22:31:44 +04:00
& dev_attr_update_interval . attr ,
2006-09-24 23:16:40 +04:00
NULL
} ;
static const struct attribute_group lm90_group = {
. attrs = lm90_attributes ,
} ;
2010-10-28 22:31:43 +04:00
/*
* Additional attributes for devices with emergency sensors
*/
static SENSOR_DEVICE_ATTR ( temp1_emergency , S_IWUSR | S_IRUGO , show_temp8 ,
set_temp8 , 4 ) ;
static SENSOR_DEVICE_ATTR ( temp2_emergency , S_IWUSR | S_IRUGO , show_temp8 ,
set_temp8 , 5 ) ;
static SENSOR_DEVICE_ATTR ( temp1_emergency_hyst , S_IRUGO , show_temphyst ,
NULL , 4 ) ;
static SENSOR_DEVICE_ATTR ( temp2_emergency_hyst , S_IRUGO , show_temphyst ,
NULL , 5 ) ;
static struct attribute * lm90_emergency_attributes [ ] = {
& sensor_dev_attr_temp1_emergency . dev_attr . attr ,
& sensor_dev_attr_temp2_emergency . dev_attr . attr ,
& sensor_dev_attr_temp1_emergency_hyst . dev_attr . attr ,
& sensor_dev_attr_temp2_emergency_hyst . dev_attr . attr ,
NULL
} ;
static const struct attribute_group lm90_emergency_group = {
. attrs = lm90_emergency_attributes ,
} ;
2010-10-28 22:31:43 +04:00
static SENSOR_DEVICE_ATTR ( temp1_emergency_alarm , S_IRUGO , show_alarm , NULL , 15 ) ;
static SENSOR_DEVICE_ATTR ( temp2_emergency_alarm , S_IRUGO , show_alarm , NULL , 13 ) ;
static struct attribute * lm90_emergency_alarm_attributes [ ] = {
& sensor_dev_attr_temp1_emergency_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_emergency_alarm . dev_attr . attr ,
NULL
} ;
static const struct attribute_group lm90_emergency_alarm_group = {
. attrs = lm90_emergency_alarm_attributes ,
} ;
/*
* Additional attributes for devices with 3 temperature sensors
*/
static SENSOR_DEVICE_ATTR_2 ( temp3_input , S_IRUGO , show_temp11 , NULL , 0 , 5 ) ;
static SENSOR_DEVICE_ATTR_2 ( temp3_min , S_IWUSR | S_IRUGO , show_temp11 ,
set_temp11 , 3 , 6 ) ;
static SENSOR_DEVICE_ATTR_2 ( temp3_max , S_IWUSR | S_IRUGO , show_temp11 ,
set_temp11 , 4 , 7 ) ;
static SENSOR_DEVICE_ATTR ( temp3_crit , S_IWUSR | S_IRUGO , show_temp8 ,
set_temp8 , 6 ) ;
static SENSOR_DEVICE_ATTR ( temp3_crit_hyst , S_IRUGO , show_temphyst , NULL , 6 ) ;
static SENSOR_DEVICE_ATTR ( temp3_emergency , S_IWUSR | S_IRUGO , show_temp8 ,
set_temp8 , 7 ) ;
static SENSOR_DEVICE_ATTR ( temp3_emergency_hyst , S_IRUGO , show_temphyst ,
NULL , 7 ) ;
static SENSOR_DEVICE_ATTR ( temp3_crit_alarm , S_IRUGO , show_alarm , NULL , 9 ) ;
static SENSOR_DEVICE_ATTR ( temp3_fault , S_IRUGO , show_alarm , NULL , 10 ) ;
static SENSOR_DEVICE_ATTR ( temp3_min_alarm , S_IRUGO , show_alarm , NULL , 11 ) ;
static SENSOR_DEVICE_ATTR ( temp3_max_alarm , S_IRUGO , show_alarm , NULL , 12 ) ;
static SENSOR_DEVICE_ATTR ( temp3_emergency_alarm , S_IRUGO , show_alarm , NULL , 14 ) ;
static struct attribute * lm90_temp3_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_emergency . dev_attr . attr ,
& sensor_dev_attr_temp3_emergency_hyst . dev_attr . attr ,
& sensor_dev_attr_temp3_fault . dev_attr . attr ,
& sensor_dev_attr_temp3_min_alarm . dev_attr . attr ,
& sensor_dev_attr_temp3_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp3_crit_alarm . dev_attr . attr ,
& sensor_dev_attr_temp3_emergency_alarm . dev_attr . attr ,
NULL
} ;
static const struct attribute_group lm90_temp3_group = {
. attrs = lm90_temp3_attributes ,
} ;
2005-10-26 23:39:40 +04:00
/* pec used for ADM1032 only */
static ssize_t show_pec ( struct device * dev , struct device_attribute * dummy ,
char * buf )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
return sprintf ( buf , " %d \n " , ! ! ( client - > flags & I2C_CLIENT_PEC ) ) ;
}
static ssize_t set_pec ( struct device * dev , struct device_attribute * dummy ,
const char * buf , size_t count )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
2010-10-28 22:31:42 +04:00
long val ;
int err ;
err = strict_strtol ( buf , 10 , & val ) ;
if ( err < 0 )
return err ;
2005-10-26 23:39:40 +04:00
switch ( val ) {
case 0 :
client - > flags & = ~ I2C_CLIENT_PEC ;
break ;
case 1 :
client - > flags | = I2C_CLIENT_PEC ;
break ;
default :
return - EINVAL ;
}
return count ;
}
static DEVICE_ATTR ( pec , S_IWUSR | S_IRUGO , show_pec , set_pec ) ;
2005-04-17 02:20:36 +04:00
/*
* Real code
*/
2010-10-28 22:31:43 +04:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2011-11-04 15:00:47 +04:00
static int lm90_detect ( struct i2c_client * client ,
2010-10-28 22:31:43 +04:00
struct i2c_board_info * info )
2005-10-26 23:37:52 +04:00
{
2011-11-04 15:00:47 +04:00
struct i2c_adapter * adapter = client - > adapter ;
int address = client - > addr ;
2010-10-28 22:31:43 +04:00
const char * name = NULL ;
2011-11-04 15:00:47 +04:00
int man_id , chip_id , config1 , config2 , convrate ;
2005-10-26 23:37:52 +04:00
2010-10-28 22:31:43 +04:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2009-12-09 22:35:53 +03:00
/* detection and identification */
2011-11-04 15:00:47 +04:00
man_id = i2c_smbus_read_byte_data ( client , LM90_REG_R_MAN_ID ) ;
chip_id = i2c_smbus_read_byte_data ( client , LM90_REG_R_CHIP_ID ) ;
config1 = i2c_smbus_read_byte_data ( client , LM90_REG_R_CONFIG1 ) ;
convrate = i2c_smbus_read_byte_data ( client , LM90_REG_R_CONVRATE ) ;
if ( man_id < 0 | | chip_id < 0 | | config1 < 0 | | convrate < 0 )
2009-12-09 22:35:53 +03:00
return - ENODEV ;
2011-07-24 22:37:05 +04:00
if ( man_id = = 0x01 | | man_id = = 0x5C | | man_id = = 0x41 ) {
2011-11-04 15:00:47 +04:00
config2 = i2c_smbus_read_byte_data ( client , LM90_REG_R_CONFIG2 ) ;
if ( config2 < 0 )
2008-07-16 21:30:15 +04:00
return - ENODEV ;
2011-07-24 22:37:05 +04:00
} else
2011-11-04 15:00:47 +04:00
config2 = 0 ; /* Make compiler happy */
2009-12-09 22:35:53 +03:00
2011-07-24 22:37:05 +04:00
if ( ( address = = 0x4C | | address = = 0x4D )
& & man_id = = 0x01 ) { /* National Semiconductor */
2011-11-04 15:00:47 +04:00
if ( ( config1 & 0x2A ) = = 0x00
& & ( config2 & 0xF8 ) = = 0x00
& & convrate < = 0x09 ) {
2009-12-09 22:35:53 +03:00
if ( address = = 0x4C
& & ( chip_id & 0xF0 ) = = 0x20 ) { /* LM90 */
name = " lm90 " ;
2007-06-09 18:11:16 +04:00
} else
2009-12-09 22:35:53 +03:00
if ( ( chip_id & 0xF0 ) = = 0x30 ) { /* LM89/LM99 */
name = " lm99 " ;
dev_info ( & adapter - > dev ,
" Assuming LM99 chip at 0x%02x \n " ,
address ) ;
dev_info ( & adapter - > dev ,
" If it is an LM89, instantiate it "
" with the new_device sysfs "
" interface \n " ) ;
2008-10-17 19:51:11 +04:00
} else
2009-12-09 22:35:53 +03:00
if ( address = = 0x4C
& & ( chip_id & 0xF0 ) = = 0x10 ) { /* LM86 */
name = " lm86 " ;
2005-04-17 02:20:36 +04:00
}
}
2009-12-09 22:35:53 +03:00
} else
if ( ( address = = 0x4C | | address = = 0x4D )
& & man_id = = 0x41 ) { /* Analog Devices */
if ( ( chip_id & 0xF0 ) = = 0x40 /* ADM1032 */
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x3F ) = = 0x00
& & convrate < = 0x0A ) {
2009-12-09 22:35:53 +03:00
name = " adm1032 " ;
/* The ADM1032 supports PEC, but only if combined
transactions are not used . */
if ( i2c_check_functionality ( adapter ,
I2C_FUNC_SMBUS_BYTE ) )
info - > flags | = I2C_CLIENT_PEC ;
} else
if ( chip_id = = 0x51 /* ADT7461 */
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x1B ) = = 0x00
& & convrate < = 0x0A ) {
2009-12-09 22:35:53 +03:00
name = " adt7461 " ;
2011-04-29 18:33:35 +04:00
} else
if ( chip_id = = 0x57 /* ADT7461A, NCT1008 */
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x1B ) = = 0x00
& & convrate < = 0x0A ) {
2011-04-29 18:33:35 +04:00
name = " adt7461a " ;
2009-12-09 22:35:53 +03:00
}
} else
if ( man_id = = 0x4D ) { /* Maxim */
2011-11-04 15:00:47 +04:00
int emerg , emerg2 , status2 ;
2010-10-28 22:31:43 +04:00
/*
* We read MAX6659_REG_R_REMOTE_EMERG twice , and re - read
* LM90_REG_R_MAN_ID in between . If MAX6659_REG_R_REMOTE_EMERG
* exists , both readings will reflect the same value . Otherwise ,
* the readings will be different .
*/
2011-11-04 15:00:47 +04:00
emerg = i2c_smbus_read_byte_data ( client ,
MAX6659_REG_R_REMOTE_EMERG ) ;
man_id = i2c_smbus_read_byte_data ( client ,
2011-11-04 15:00:46 +04:00
LM90_REG_R_MAN_ID ) ;
2011-11-04 15:00:47 +04:00
emerg2 = i2c_smbus_read_byte_data ( client ,
2011-11-04 15:00:46 +04:00
MAX6659_REG_R_REMOTE_EMERG ) ;
2011-11-04 15:00:47 +04:00
status2 = i2c_smbus_read_byte_data ( client ,
MAX6696_REG_R_STATUS2 ) ;
if ( emerg < 0 | | man_id < 0 | | emerg2 < 0 | | status2 < 0 )
2010-10-28 22:31:43 +04:00
return - ENODEV ;
2009-12-09 22:35:53 +03:00
/*
* The MAX6657 , MAX6658 and MAX6659 do NOT have a chip_id
* register . Reading from that address will return the last
* read value , which in our case is those of the man_id
* register . Likewise , the config1 register seems to lack a
* low nibble , so the value will be those of the previous
* read , so in our case those of the man_id register .
2010-10-28 22:31:43 +04:00
* MAX6659 has a third set of upper temperature limit registers .
* Those registers also return values on MAX6657 and MAX6658 ,
* thus the only way to detect MAX6659 is by its address .
* For this reason it will be mis - detected as MAX6657 if its
* address is 0x4C .
2009-12-09 22:35:53 +03:00
*/
if ( chip_id = = man_id
2010-10-28 22:31:43 +04:00
& & ( address = = 0x4C | | address = = 0x4D | | address = = 0x4E )
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x1F ) = = ( man_id & 0x0F )
& & convrate < = 0x09 ) {
2010-10-28 22:31:43 +04:00
if ( address = = 0x4C )
name = " max6657 " ;
else
name = " max6659 " ;
2009-12-09 22:35:53 +03:00
} else
2010-10-28 22:31:43 +04:00
/*
* Even though MAX6695 and MAX6696 do not have a chip ID
* register , reading it returns 0x01 . Bit 4 of the config1
* register is unused and should return zero when read . Bit 0 of
* the status2 register is unused and should return zero when
* read .
*
* MAX6695 and MAX6696 have an additional set of temperature
* limit registers . We can detect those chips by checking if
* one of those registers exists .
*/
if ( chip_id = = 0x01
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x10 ) = = 0x00
& & ( status2 & 0x01 ) = = 0x00
& & emerg = = emerg2
& & convrate < = 0x07 ) {
2010-10-28 22:31:43 +04:00
name = " max6696 " ;
} else
2009-12-09 22:35:53 +03:00
/*
* The chip_id register of the MAX6680 and MAX6681 holds the
* revision of the chip . The lowest bit of the config1 register
* is unused and should return zero when read , so should the
* second to last bit of config1 ( software reset ) .
*/
if ( chip_id = = 0x01
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x03 ) = = 0x00
& & convrate < = 0x07 ) {
2009-12-09 22:35:53 +03:00
name = " max6680 " ;
} else
/*
* The chip_id register of the MAX6646 / 6647 / 6649 holds the
* revision of the chip . The lowest 6 bits of the config1
* register are unused and should return zero when read .
*/
if ( chip_id = = 0x59
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x3f ) = = 0x00
& & convrate < = 0x07 ) {
2009-12-09 22:35:53 +03:00
name = " max6646 " ;
2005-04-17 02:20:36 +04:00
}
2010-03-06 00:17:13 +03:00
} else
if ( address = = 0x4C
& & man_id = = 0x5C ) { /* Winbond/Nuvoton */
2011-11-04 15:00:47 +04:00
if ( ( config1 & 0x2A ) = = 0x00
& & ( config2 & 0xF8 ) = = 0x00 ) {
2010-10-28 22:31:44 +04:00
if ( chip_id = = 0x01 /* W83L771W/G */
2011-11-04 15:00:47 +04:00
& & convrate < = 0x09 ) {
2010-10-28 22:31:44 +04:00
name = " w83l771 " ;
} else
if ( ( chip_id & 0xFE ) = = 0x10 /* W83L771AWG/ASG */
2011-11-04 15:00:47 +04:00
& & convrate < = 0x08 ) {
2010-10-28 22:31:44 +04:00
name = " w83l771 " ;
}
2010-03-06 00:17:13 +03:00
}
2011-06-06 14:40:45 +04:00
} else
2011-07-24 22:36:15 +04:00
if ( address > = 0x48 & & address < = 0x4F
& & man_id = = 0xA1 ) { /* NXP Semiconductor/Philips */
if ( chip_id = = 0x00
2011-11-04 15:00:47 +04:00
& & ( config1 & 0x2A ) = = 0x00
& & ( config2 & 0xFE ) = = 0x00
& & convrate < = 0x09 ) {
2011-06-06 14:40:45 +04:00
name = " sa56004 " ;
}
2005-04-17 02:20:36 +04:00
}
2009-12-09 22:35:53 +03:00
if ( ! name ) { /* identification failed */
dev_dbg ( & adapter - > dev ,
" Unsupported chip at 0x%02x (man_id=0x%02X, "
" chip_id=0x%02X) \n " , address , man_id , chip_id ) ;
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
2009-12-09 22:35:53 +03:00
2008-07-16 21:30:15 +04:00
strlcpy ( info - > type , name , I2C_NAME_SIZE ) ;
return 0 ;
}
2010-10-28 22:31:43 +04:00
static void lm90_remove_files ( struct i2c_client * client , struct lm90_data * data )
{
2011-11-04 15:00:47 +04:00
struct device * dev = & client - > dev ;
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_TEMP3 )
2011-11-04 15:00:47 +04:00
sysfs_remove_group ( & dev - > kobj , & lm90_temp3_group ) ;
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_EMERGENCY_ALARM )
2011-11-04 15:00:47 +04:00
sysfs_remove_group ( & dev - > kobj , & lm90_emergency_alarm_group ) ;
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_EMERGENCY )
2011-11-04 15:00:47 +04:00
sysfs_remove_group ( & dev - > kobj , & lm90_emergency_group ) ;
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_OFFSET )
2011-11-04 15:00:47 +04:00
device_remove_file ( dev , & sensor_dev_attr_temp2_offset . dev_attr ) ;
device_remove_file ( dev , & dev_attr_pec ) ;
sysfs_remove_group ( & dev - > kobj , & lm90_group ) ;
2010-10-28 22:31:43 +04:00
}
2010-10-28 22:31:43 +04:00
static void lm90_init_client ( struct i2c_client * client )
{
2010-10-28 22:31:44 +04:00
u8 config , convrate ;
2010-10-28 22:31:43 +04:00
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2010-10-28 22:31:44 +04:00
if ( lm90_read_reg ( client , LM90_REG_R_CONVRATE , & convrate ) < 0 ) {
dev_warn ( & client - > dev , " Failed to read convrate register! \n " ) ;
convrate = LM90_DEF_CONVRATE_RVAL ;
}
data - > convrate_orig = convrate ;
2010-10-28 22:31:43 +04:00
/*
* Start the conversions .
*/
2010-10-28 22:31:44 +04:00
lm90_set_convrate ( client , data , 500 ) ; /* 500ms; 2Hz conversion rate */
2010-10-28 22:31:43 +04:00
if ( lm90_read_reg ( client , LM90_REG_R_CONFIG1 , & config ) < 0 ) {
dev_warn ( & client - > dev , " Initialization failed! \n " ) ;
return ;
}
data - > config_orig = config ;
/* Check Temperature Range Select */
if ( data - > kind = = adt7461 ) {
if ( config & 0x04 )
data - > flags | = LM90_FLAG_ADT7461_EXT ;
}
/*
* Put MAX6680 / MAX8881 into extended resolution ( bit 0x10 ,
* 0.125 degree resolution ) and range ( 0x08 , extend range
* to - 64 degree ) mode for the remote temperature sensor .
*/
if ( data - > kind = = max6680 )
config | = 0x18 ;
/*
* Select external channel 0 for max6695 / 96
*/
if ( data - > kind = = max6696 )
config & = ~ 0x08 ;
config & = 0xBF ; /* run */
if ( config ! = data - > config_orig ) /* Only write if changed */
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONFIG1 , config ) ;
}
2011-11-04 15:00:47 +04:00
static int lm90_probe ( struct i2c_client * client ,
2008-07-16 21:30:15 +04:00
const struct i2c_device_id * id )
{
2011-11-04 15:00:47 +04:00
struct device * dev = & client - > dev ;
struct i2c_adapter * adapter = to_i2c_adapter ( dev - > parent ) ;
2008-07-16 21:30:15 +04:00
struct lm90_data * data ;
int err ;
2005-04-17 02:20:36 +04:00
2008-07-16 21:30:15 +04:00
data = kzalloc ( sizeof ( struct lm90_data ) , GFP_KERNEL ) ;
if ( ! data ) {
err = - ENOMEM ;
goto exit ;
}
2011-11-04 15:00:47 +04:00
i2c_set_clientdata ( client , data ) ;
2006-01-19 01:19:26 +03:00
mutex_init ( & data - > update_lock ) ;
2005-04-17 02:20:36 +04:00
2008-07-16 21:30:15 +04:00
/* Set the device type */
data - > kind = id - > driver_data ;
if ( data - > kind = = adm1032 ) {
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE ) )
2011-11-04 15:00:47 +04:00
client - > flags & = ~ I2C_CLIENT_PEC ;
2008-07-16 21:30:15 +04:00
}
2005-04-17 02:20:36 +04:00
2010-03-06 00:17:15 +03:00
/* Different devices have different alarm bits triggering the
* ALERT # output */
2010-10-28 22:31:43 +04:00
data - > alert_alarms = lm90_params [ data - > kind ] . alert_alarms ;
2010-03-06 00:17:15 +03:00
2010-10-28 22:31:43 +04:00
/* Set chip capabilities */
2010-10-28 22:31:43 +04:00
data - > flags = lm90_params [ data - > kind ] . flags ;
2011-07-28 10:22:25 +04:00
data - > reg_local_ext = lm90_params [ data - > kind ] . reg_local_ext ;
2010-10-28 22:31:43 +04:00
2010-10-28 22:31:44 +04:00
/* Set maximum conversion rate */
data - > max_convrate = lm90_params [ data - > kind ] . max_convrate ;
2005-04-17 02:20:36 +04:00
/* Initialize the LM90 chip */
2011-11-04 15:00:47 +04:00
lm90_init_client ( client ) ;
2005-04-17 02:20:36 +04:00
/* Register sysfs hooks */
2011-11-04 15:00:47 +04:00
err = sysfs_create_group ( & dev - > kobj , & lm90_group ) ;
2010-10-28 22:31:42 +04:00
if ( err )
2008-07-16 21:30:15 +04:00
goto exit_free ;
2011-11-04 15:00:47 +04:00
if ( client - > flags & I2C_CLIENT_PEC ) {
err = device_create_file ( dev , & dev_attr_pec ) ;
2010-10-28 22:31:42 +04:00
if ( err )
2006-09-24 23:16:40 +04:00
goto exit_remove_files ;
}
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_OFFSET ) {
2011-11-04 15:00:47 +04:00
err = device_create_file ( dev ,
2010-10-28 22:31:42 +04:00
& sensor_dev_attr_temp2_offset . dev_attr ) ;
if ( err )
2007-09-05 16:15:37 +04:00
goto exit_remove_files ;
}
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_EMERGENCY ) {
2011-11-04 15:00:47 +04:00
err = sysfs_create_group ( & dev - > kobj , & lm90_emergency_group ) ;
2010-10-28 22:31:43 +04:00
if ( err )
goto exit_remove_files ;
}
2010-10-28 22:31:43 +04:00
if ( data - > flags & LM90_HAVE_EMERGENCY_ALARM ) {
2011-11-04 15:00:47 +04:00
err = sysfs_create_group ( & dev - > kobj ,
2010-10-28 22:31:43 +04:00
& lm90_emergency_alarm_group ) ;
if ( err )
goto exit_remove_files ;
}
if ( data - > flags & LM90_HAVE_TEMP3 ) {
2011-11-04 15:00:47 +04:00
err = sysfs_create_group ( & dev - > kobj , & lm90_temp3_group ) ;
2010-10-28 22:31:43 +04:00
if ( err )
goto exit_remove_files ;
}
2006-09-24 23:16:40 +04:00
2011-11-04 15:00:47 +04:00
data - > hwmon_dev = hwmon_device_register ( dev ) ;
2007-08-21 00:46:20 +04:00
if ( IS_ERR ( data - > hwmon_dev ) ) {
err = PTR_ERR ( data - > hwmon_dev ) ;
2006-09-24 23:16:40 +04:00
goto exit_remove_files ;
2005-07-16 05:39:18 +04:00
}
2005-04-17 02:20:36 +04:00
return 0 ;
2006-09-24 23:16:40 +04:00
exit_remove_files :
2011-11-04 15:00:47 +04:00
lm90_remove_files ( client , data ) ;
2005-04-17 02:20:36 +04:00
exit_free :
kfree ( data ) ;
exit :
return err ;
}
2008-07-16 21:30:15 +04:00
static int lm90_remove ( struct i2c_client * client )
2005-04-17 02:20:36 +04:00
{
2005-07-16 05:39:18 +04:00
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2005-04-17 02:20:36 +04:00
2007-08-21 00:46:20 +04:00
hwmon_device_unregister ( data - > hwmon_dev ) ;
2010-10-28 22:31:43 +04:00
lm90_remove_files ( client , data ) ;
2005-07-16 05:39:18 +04:00
2010-03-06 00:17:14 +03:00
/* Restore initial configuration */
2010-10-28 22:31:44 +04:00
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONVRATE ,
data - > convrate_orig ) ;
2010-03-06 00:17:14 +03:00
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONFIG1 ,
data - > config_orig ) ;
2005-07-16 05:39:18 +04:00
kfree ( data ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2010-03-06 00:17:15 +03:00
static void lm90_alert ( struct i2c_client * client , unsigned int flag )
{
struct lm90_data * data = i2c_get_clientdata ( client ) ;
2010-10-28 22:31:43 +04:00
u8 config , alarms , alarms2 = 0 ;
2010-03-06 00:17:15 +03:00
lm90_read_reg ( client , LM90_REG_R_STATUS , & alarms ) ;
2010-10-28 22:31:43 +04:00
if ( data - > kind = = max6696 )
lm90_read_reg ( client , MAX6696_REG_R_STATUS2 , & alarms2 ) ;
if ( ( alarms & 0x7f ) = = 0 & & ( alarms2 & 0xfe ) = = 0 ) {
2010-03-06 00:17:15 +03:00
dev_info ( & client - > dev , " Everything OK \n " ) ;
} else {
if ( alarms & 0x61 )
dev_warn ( & client - > dev ,
" temp%d out of range, please check! \n " , 1 ) ;
if ( alarms & 0x1a )
dev_warn ( & client - > dev ,
" temp%d out of range, please check! \n " , 2 ) ;
if ( alarms & 0x04 )
dev_warn ( & client - > dev ,
" temp%d diode open, please check! \n " , 2 ) ;
2010-10-28 22:31:43 +04:00
if ( alarms2 & 0x18 )
dev_warn ( & client - > dev ,
" temp%d out of range, please check! \n " , 3 ) ;
2010-03-06 00:17:15 +03:00
/* Disable ALERT# output, because these chips don't implement
SMBus alert correctly ; they should only hold the alert line
low briefly . */
2010-10-28 22:31:44 +04:00
if ( ( data - > flags & LM90_HAVE_BROKEN_ALERT )
2010-03-06 00:17:15 +03:00
& & ( alarms & data - > alert_alarms ) ) {
dev_dbg ( & client - > dev , " Disabling ALERT# \n " ) ;
lm90_read_reg ( client , LM90_REG_R_CONFIG1 , & config ) ;
i2c_smbus_write_byte_data ( client , LM90_REG_W_CONFIG1 ,
config | 0x80 ) ;
}
}
}
2010-10-28 22:31:43 +04:00
static struct i2c_driver lm90_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " lm90 " ,
} ,
. probe = lm90_probe ,
. remove = lm90_remove ,
. alert = lm90_alert ,
. id_table = lm90_id ,
. detect = lm90_detect ,
. address_list = normal_i2c ,
} ;
2005-04-17 02:20:36 +04:00
static int __init sensors_lm90_init ( void )
{
return i2c_add_driver ( & lm90_driver ) ;
}
static void __exit sensors_lm90_exit ( void )
{
i2c_del_driver ( & lm90_driver ) ;
}
MODULE_AUTHOR ( " Jean Delvare <khali@linux-fr.org> " ) ;
MODULE_DESCRIPTION ( " LM90/ADM1032 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( sensors_lm90_init ) ;
module_exit ( sensors_lm90_exit ) ;