2005-04-16 15:20:36 -07:00
/*
2010-10-28 20:31:44 +02:00
* lm75 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
* Copyright ( c ) 1998 , 1999 Frodo Looijaard < frodol @ dds . nl >
*
* 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 .
*/
2005-04-16 15:20:36 -07:00
# 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>
2007-05-08 17:22:01 +02: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>
2005-04-16 15:20:36 -07:00
# include "lm75.h"
2008-04-21 12:10:53 -07:00
/*
* This driver handles the LM75 and compatible digital temperature sensors .
*/
2008-05-03 19:33:15 -07:00
enum lm75_type { /* keep sorted in alphabetical order */
2011-10-13 04:43:31 -04:00
adt75 ,
2009-12-14 21:17:26 +01:00
ds1775 ,
2008-05-03 19:33:15 -07:00
ds75 ,
2009-12-14 21:17:26 +01:00
lm75 ,
2008-05-03 19:33:15 -07:00
lm75a ,
max6625 ,
max6626 ,
mcp980x ,
stds75 ,
tcn75 ,
tmp100 ,
tmp101 ,
2010-05-27 19:59:03 +02:00
tmp105 ,
2008-05-03 19:33:15 -07:00
tmp175 ,
tmp275 ,
tmp75 ,
} ;
2008-08-10 22:56:16 +02:00
/* Addresses scanned */
2008-02-17 22:28:03 -05:00
static const unsigned short normal_i2c [ ] = { 0x48 , 0x49 , 0x4a , 0x4b , 0x4c ,
2005-04-16 15:20:36 -07:00
0x4d , 0x4e , 0x4f , I2C_CLIENT_END } ;
/* The LM75 registers */
# define LM75_REG_CONF 0x01
2007-05-08 17:22:01 +02:00
static const u8 LM75_REG_TEMP [ 3 ] = {
0x00 , /* input */
0x03 , /* max */
0x02 , /* hyst */
} ;
2005-04-16 15:20:36 -07:00
/* Each client has this additional data */
struct lm75_data {
2008-04-21 12:10:53 -07:00
struct device * hwmon_dev ;
2006-01-18 23:19:26 +01:00
struct mutex update_lock ;
2008-05-03 19:33:15 -07:00
u8 orig_conf ;
2008-04-21 12:10:53 -07:00
char valid ; /* !=0 if registers are valid */
2005-04-16 15:20:36 -07:00
unsigned long last_updated ; /* In jiffies */
2007-05-08 17:22:01 +02:00
u16 temp [ 3 ] ; /* Register values,
0 = input
1 = max
2 = hyst */
2005-04-16 15:20:36 -07:00
} ;
static int lm75_read_value ( struct i2c_client * client , u8 reg ) ;
static int lm75_write_value ( struct i2c_client * client , u8 reg , u16 value ) ;
static struct lm75_data * lm75_update_device ( struct device * dev ) ;
2008-04-21 12:10:53 -07:00
/*-----------------------------------------------------------------------*/
/* sysfs attributes for hwmon */
2005-04-16 15:20:36 -07:00
2007-05-08 17:22:01 +02:00
static ssize_t show_temp ( struct device * dev , struct device_attribute * da ,
char * buf )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( da ) ;
struct lm75_data * data = lm75_update_device ( dev ) ;
return sprintf ( buf , " %d \n " ,
LM75_TEMP_FROM_REG ( data - > temp [ attr - > index ] ) ) ;
2005-04-16 15:20:36 -07:00
}
2007-05-08 17:22:01 +02:00
static ssize_t set_temp ( struct device * dev , struct device_attribute * da ,
const char * buf , size_t count )
{
struct sensor_device_attribute * attr = to_sensor_dev_attr ( da ) ;
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm75_data * data = i2c_get_clientdata ( client ) ;
int nr = attr - > index ;
2010-10-28 20:31:44 +02:00
long temp ;
int error ;
error = strict_strtol ( buf , 10 , & temp ) ;
if ( error )
return error ;
2007-05-08 17:22:01 +02:00
mutex_lock ( & data - > update_lock ) ;
data - > temp [ nr ] = LM75_TEMP_TO_REG ( temp ) ;
lm75_write_value ( client , LM75_REG_TEMP [ nr ] , data - > temp [ nr ] ) ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
2005-04-16 15:20:36 -07:00
}
2007-05-08 17:22:01 +02:00
static SENSOR_DEVICE_ATTR ( temp1_max , S_IWUSR | S_IRUGO ,
show_temp , set_temp , 1 ) ;
static SENSOR_DEVICE_ATTR ( temp1_max_hyst , S_IWUSR | S_IRUGO ,
show_temp , set_temp , 2 ) ;
static SENSOR_DEVICE_ATTR ( temp1_input , S_IRUGO , show_temp , NULL , 0 ) ;
2005-04-16 15:20:36 -07:00
2006-09-24 20:59:49 +02:00
static struct attribute * lm75_attributes [ ] = {
2007-05-08 17:22:01 +02:00
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp1_max_hyst . dev_attr . attr ,
2006-09-24 20:59:49 +02:00
NULL
} ;
static const struct attribute_group lm75_group = {
. attrs = lm75_attributes ,
} ;
2008-04-21 12:10:53 -07:00
/*-----------------------------------------------------------------------*/
2008-08-10 22:56:16 +02:00
/* device probe and removal */
2008-05-03 19:33:15 -07:00
static int
lm75_probe ( struct i2c_client * client , const struct i2c_device_id * id )
{
struct lm75_data * data ;
int status ;
u8 set_mask , clr_mask ;
int new ;
if ( ! i2c_check_functionality ( client - > adapter ,
I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA ) )
return - EIO ;
data = kzalloc ( sizeof ( struct lm75_data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
i2c_set_clientdata ( client , data ) ;
mutex_init ( & data - > update_lock ) ;
/* Set to LM75 resolution (9 bits, 1/2 degree C) and range.
* Then tweak to be more precise when appropriate .
*/
set_mask = 0 ;
clr_mask = ( 1 < < 0 ) /* continuous conversions */
| ( 1 < < 6 ) | ( 1 < < 5 ) ; /* 9-bit mode */
/* configure as specified */
status = lm75_read_value ( client , LM75_REG_CONF ) ;
if ( status < 0 ) {
dev_dbg ( & client - > dev , " Can't read config? %d \n " , status ) ;
goto exit_free ;
}
data - > orig_conf = status ;
new = status & ~ clr_mask ;
new | = set_mask ;
if ( status ! = new )
lm75_write_value ( client , LM75_REG_CONF , new ) ;
dev_dbg ( & client - > dev , " Config %02x \n " , new ) ;
/* Register sysfs hooks */
status = sysfs_create_group ( & client - > dev . kobj , & lm75_group ) ;
if ( status )
goto exit_free ;
data - > hwmon_dev = hwmon_device_register ( & client - > dev ) ;
if ( IS_ERR ( data - > hwmon_dev ) ) {
status = PTR_ERR ( data - > hwmon_dev ) ;
goto exit_remove ;
}
dev_info ( & client - > dev , " %s: sensor '%s' \n " ,
2009-01-06 10:44:41 -08:00
dev_name ( data - > hwmon_dev ) , client - > name ) ;
2008-05-03 19:33:15 -07:00
return 0 ;
exit_remove :
sysfs_remove_group ( & client - > dev . kobj , & lm75_group ) ;
exit_free :
kfree ( data ) ;
return status ;
}
static int lm75_remove ( struct i2c_client * client )
{
struct lm75_data * data = i2c_get_clientdata ( client ) ;
hwmon_device_unregister ( data - > hwmon_dev ) ;
sysfs_remove_group ( & client - > dev . kobj , & lm75_group ) ;
lm75_write_value ( client , LM75_REG_CONF , data - > orig_conf ) ;
kfree ( data ) ;
return 0 ;
}
static const struct i2c_device_id lm75_ids [ ] = {
2011-10-13 04:43:31 -04:00
{ " adt75 " , adt75 , } ,
2008-05-03 19:33:15 -07:00
{ " ds1775 " , ds1775 , } ,
{ " ds75 " , ds75 , } ,
{ " lm75 " , lm75 , } ,
{ " lm75a " , lm75a , } ,
{ " max6625 " , max6625 , } ,
{ " max6626 " , max6626 , } ,
{ " mcp980x " , mcp980x , } ,
{ " stds75 " , stds75 , } ,
{ " tcn75 " , tcn75 , } ,
{ " tmp100 " , tmp100 , } ,
{ " tmp101 " , tmp101 , } ,
2010-05-27 19:59:03 +02:00
{ " tmp105 " , tmp105 , } ,
2008-05-03 19:33:15 -07:00
{ " tmp175 " , tmp175 , } ,
{ " tmp275 " , tmp275 , } ,
{ " tmp75 " , tmp75 , } ,
{ /* LIST END */ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm75_ids ) ;
2011-03-21 17:59:36 +01:00
# define LM75A_ID 0xA1
2008-08-10 22:56:16 +02:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2009-12-14 21:17:23 +01:00
static int lm75_detect ( struct i2c_client * new_client ,
2008-08-10 22:56:16 +02:00
struct i2c_board_info * info )
2005-04-16 15:20:36 -07:00
{
2008-08-10 22:56:16 +02:00
struct i2c_adapter * adapter = new_client - > adapter ;
2005-04-16 15:20:36 -07:00
int i ;
2011-03-21 17:59:36 +01:00
int conf , hyst , os ;
2011-03-21 17:59:36 +01:00
bool is_lm75a = 0 ;
2005-04-16 15:20:36 -07:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA |
I2C_FUNC_SMBUS_WORD_DATA ) )
2008-08-10 22:56:16 +02:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
2011-10-13 17:15:11 -04:00
/*
* Now , we do the remaining detection . There is no identification -
* dedicated register so we have to rely on several tricks :
* unused bits , registers cycling over 8 - address boundaries ,
* addresses 0x04 - 0x07 returning the last read value .
* The cycling + unused addresses combination is not tested ,
* since it would significantly slow the detection down and would
* hardly add any value .
*
* The National Semiconductor LM75A is different than earlier
* LM75s . It has an ID byte of 0xa X ( where X is the chip
* revision , with 1 being the only revision in existence ) in
* register 7 , and unused registers return 0xff rather than the
* last read value .
*
* Note that this function only detects the original National
* Semiconductor LM75 and the LM75A . Clones from other vendors
* aren ' t detected , on purpose , because they are typically never
* found on PC hardware . They are found on embedded designs where
* they can be instantiated explicitly so detection is not needed .
* The absence of identification registers on all these clones
* would make their exhaustive detection very difficult and weak ,
* and odds are that the driver would bind to unsupported devices .
*/
2005-04-16 15:20:36 -07:00
2011-03-21 17:59:36 +01:00
/* Unused bits */
2009-12-09 20:35:57 +01:00
conf = i2c_smbus_read_byte_data ( new_client , 1 ) ;
2011-03-21 17:59:36 +01:00
if ( conf & 0xe0 )
return - ENODEV ;
2011-03-21 17:59:36 +01:00
/* First check for LM75A */
if ( i2c_smbus_read_byte_data ( new_client , 7 ) = = LM75A_ID ) {
/* LM75A returns 0xff on unused registers so
just to be sure we check for that too . */
if ( i2c_smbus_read_byte_data ( new_client , 4 ) ! = 0xff
| | i2c_smbus_read_byte_data ( new_client , 5 ) ! = 0xff
| | i2c_smbus_read_byte_data ( new_client , 6 ) ! = 0xff )
return - ENODEV ;
is_lm75a = 1 ;
2011-03-21 17:59:36 +01:00
hyst = i2c_smbus_read_byte_data ( new_client , 2 ) ;
os = i2c_smbus_read_byte_data ( new_client , 3 ) ;
2011-03-21 17:59:36 +01:00
} else { /* Traditional style LM75 detection */
/* Unused addresses */
2011-03-21 17:59:36 +01:00
hyst = i2c_smbus_read_byte_data ( new_client , 2 ) ;
if ( i2c_smbus_read_byte_data ( new_client , 4 ) ! = hyst
| | i2c_smbus_read_byte_data ( new_client , 5 ) ! = hyst
| | i2c_smbus_read_byte_data ( new_client , 6 ) ! = hyst
| | i2c_smbus_read_byte_data ( new_client , 7 ) ! = hyst )
2011-03-21 17:59:36 +01:00
return - ENODEV ;
2011-03-21 17:59:36 +01:00
os = i2c_smbus_read_byte_data ( new_client , 3 ) ;
if ( i2c_smbus_read_byte_data ( new_client , 4 ) ! = os
| | i2c_smbus_read_byte_data ( new_client , 5 ) ! = os
| | i2c_smbus_read_byte_data ( new_client , 6 ) ! = os
| | i2c_smbus_read_byte_data ( new_client , 7 ) ! = os )
2011-03-21 17:59:36 +01:00
return - ENODEV ;
}
2005-04-16 15:20:36 -07:00
2009-12-09 20:35:57 +01:00
/* Addresses cycling */
2011-03-21 17:59:36 +01:00
for ( i = 8 ; i < = 248 ; i + = 40 ) {
2009-12-09 20:35:57 +01:00
if ( i2c_smbus_read_byte_data ( new_client , i + 1 ) ! = conf
2011-03-21 17:59:36 +01:00
| | i2c_smbus_read_byte_data ( new_client , i + 2 ) ! = hyst
| | i2c_smbus_read_byte_data ( new_client , i + 3 ) ! = os )
2009-12-09 20:35:57 +01:00
return - ENODEV ;
2011-03-21 17:59:36 +01:00
if ( is_lm75a & & i2c_smbus_read_byte_data ( new_client , i + 7 )
! = LM75A_ID )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2011-03-21 17:59:36 +01:00
strlcpy ( info - > type , is_lm75a ? " lm75a " : " lm75 " , I2C_NAME_SIZE ) ;
2006-09-24 20:59:49 +02:00
2005-04-16 15:20:36 -07:00
return 0 ;
2008-04-21 12:10:53 -07:00
}
2010-08-14 21:08:50 +02:00
# ifdef CONFIG_PM
static int lm75_suspend ( struct device * dev )
{
int status ;
struct i2c_client * client = to_i2c_client ( dev ) ;
status = lm75_read_value ( client , LM75_REG_CONF ) ;
if ( status < 0 ) {
dev_dbg ( & client - > dev , " Can't read config? %d \n " , status ) ;
return status ;
}
status = status | LM75_SHUTDOWN ;
lm75_write_value ( client , LM75_REG_CONF , status ) ;
return 0 ;
}
static int lm75_resume ( struct device * dev )
{
int status ;
struct i2c_client * client = to_i2c_client ( dev ) ;
status = lm75_read_value ( client , LM75_REG_CONF ) ;
if ( status < 0 ) {
dev_dbg ( & client - > dev , " Can't read config? %d \n " , status ) ;
return status ;
}
status = status & ~ LM75_SHUTDOWN ;
lm75_write_value ( client , LM75_REG_CONF , status ) ;
return 0 ;
}
static const struct dev_pm_ops lm75_dev_pm_ops = {
. suspend = lm75_suspend ,
. resume = lm75_resume ,
} ;
# define LM75_DEV_PM_OPS (&lm75_dev_pm_ops)
# else
# define LM75_DEV_PM_OPS NULL
# endif /* CONFIG_PM */
2008-08-10 22:56:16 +02:00
static struct i2c_driver lm75_driver = {
. class = I2C_CLASS_HWMON ,
2008-04-21 12:10:53 -07:00
. driver = {
2008-08-10 22:56:16 +02:00
. name = " lm75 " ,
2010-08-14 21:08:50 +02:00
. pm = LM75_DEV_PM_OPS ,
2008-04-21 12:10:53 -07:00
} ,
2008-08-10 22:56:16 +02:00
. probe = lm75_probe ,
. remove = lm75_remove ,
. id_table = lm75_ids ,
. detect = lm75_detect ,
2009-12-14 21:17:25 +01:00
. address_list = normal_i2c ,
2008-04-21 12:10:53 -07:00
} ;
/*-----------------------------------------------------------------------*/
/* register access */
2010-10-28 20:31:44 +02:00
/*
* All registers are word - sized , except for the configuration register .
* LM75 uses a high - byte first convention , which is exactly opposite to
* the SMBus standard .
*/
2005-04-16 15:20:36 -07:00
static int lm75_read_value ( struct i2c_client * client , u8 reg )
{
if ( reg = = LM75_REG_CONF )
return i2c_smbus_read_byte_data ( client , reg ) ;
2011-11-04 12:00:47 +01:00
else
return i2c_smbus_read_word_swapped ( client , reg ) ;
2005-04-16 15:20:36 -07:00
}
static int lm75_write_value ( struct i2c_client * client , u8 reg , u16 value )
{
if ( reg = = LM75_REG_CONF )
return i2c_smbus_write_byte_data ( client , reg , value ) ;
else
2011-11-04 12:00:47 +01:00
return i2c_smbus_write_word_swapped ( client , reg , value ) ;
2005-04-16 15:20:36 -07:00
}
static struct lm75_data * lm75_update_device ( struct device * dev )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct lm75_data * data = i2c_get_clientdata ( client ) ;
2006-01-18 23:19:26 +01:00
mutex_lock ( & data - > update_lock ) ;
2005-04-16 15:20:36 -07:00
if ( time_after ( jiffies , data - > last_updated + HZ + HZ / 2 )
| | ! data - > valid ) {
2007-05-08 17:22:01 +02:00
int i ;
2005-04-16 15:20:36 -07:00
dev_dbg ( & client - > dev , " Starting lm75 update \n " ) ;
2008-05-03 19:19:16 -07:00
for ( i = 0 ; i < ARRAY_SIZE ( data - > temp ) ; i + + ) {
int status ;
status = lm75_read_value ( client , LM75_REG_TEMP [ i ] ) ;
if ( status < 0 )
dev_dbg ( & client - > dev , " reg %d, err %d \n " ,
LM75_REG_TEMP [ i ] , status ) ;
else
data - > temp [ i ] = status ;
}
2005-04-16 15:20:36 -07:00
data - > last_updated = jiffies ;
data - > valid = 1 ;
}
2006-01-18 23:19:26 +01:00
mutex_unlock ( & data - > update_lock ) ;
2005-04-16 15:20:36 -07:00
return data ;
}
2008-04-21 12:10:53 -07:00
/*-----------------------------------------------------------------------*/
/* module glue */
2005-04-16 15:20:36 -07:00
static int __init sensors_lm75_init ( void )
{
2008-08-10 22:56:16 +02:00
return i2c_add_driver ( & lm75_driver ) ;
2005-04-16 15:20:36 -07:00
}
static void __exit sensors_lm75_exit ( void )
{
i2c_del_driver ( & lm75_driver ) ;
}
MODULE_AUTHOR ( " Frodo Looijaard <frodol@dds.nl> " ) ;
MODULE_DESCRIPTION ( " LM75 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( sensors_lm75_init ) ;
module_exit ( sensors_lm75_exit ) ;