2005-04-16 15:20:36 -07:00
/*
adm1021 . c - Part of lm_sensors , Linux kernel modules for hardware
2007-07-19 08:00:17 +02:00
monitoring
2005-04-16 15:20:36 -07:00
Copyright ( c ) 1998 , 1999 Frodo Looijaard < frodol @ dds . nl > and
Philip Edelbrock < phil @ netroedge . com >
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-07-15 21:39:18 -04:00
# include <linux/hwmon.h>
2007-09-09 19:51:14 +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
/* Addresses to scan */
static unsigned short normal_i2c [ ] = { 0x18 , 0x19 , 0x1a ,
0x29 , 0x2a , 0x2b ,
2007-07-19 08:00:17 +02:00
0x4c , 0x4d , 0x4e ,
2005-04-16 15:20:36 -07:00
I2C_CLIENT_END } ;
/* Insmod parameters */
2007-07-19 08:00:17 +02:00
I2C_CLIENT_INSMOD_8 ( adm1021 , adm1023 , max1617 , max1617a , thmc10 , lm84 , gl523sm ,
mc1066 ) ;
2005-04-16 15:20:36 -07:00
/* adm1021 constants specified below */
/* The adm1021 registers */
/* Read-only */
2007-09-09 19:51:14 +02:00
/* For nr in 0-1 */
# define ADM1021_REG_TEMP(nr) (nr)
2005-04-16 15:20:36 -07:00
# define ADM1021_REG_STATUS 0x02
2007-07-19 08:00:17 +02:00
/* 0x41 = AD, 0x49 = TI, 0x4D = Maxim, 0x23 = Genesys , 0x54 = Onsemi */
# define ADM1021_REG_MAN_ID 0xFE
/* ADM1021 = 0x0X, ADM1023 = 0x3X */
# define ADM1021_REG_DEV_ID 0xFF
2005-04-16 15:20:36 -07:00
/* These use different addresses for reading/writing */
# define ADM1021_REG_CONFIG_R 0x03
# define ADM1021_REG_CONFIG_W 0x09
# define ADM1021_REG_CONV_RATE_R 0x04
# define ADM1021_REG_CONV_RATE_W 0x0A
/* These are for the ADM1023's additional precision on the remote temp sensor */
2007-07-19 08:00:17 +02:00
# define ADM1023_REG_REM_TEMP_PREC 0x10
# define ADM1023_REG_REM_OFFSET 0x11
# define ADM1023_REG_REM_OFFSET_PREC 0x12
# define ADM1023_REG_REM_TOS_PREC 0x13
# define ADM1023_REG_REM_THYST_PREC 0x14
2005-04-16 15:20:36 -07:00
/* limits */
2007-09-09 19:51:14 +02:00
/* For nr in 0-1 */
# define ADM1021_REG_TOS_R(nr) (0x05 + 2 * (nr))
# define ADM1021_REG_TOS_W(nr) (0x0B + 2 * (nr))
# define ADM1021_REG_THYST_R(nr) (0x06 + 2 * (nr))
# define ADM1021_REG_THYST_W(nr) (0x0C + 2 * (nr))
2005-04-16 15:20:36 -07:00
/* write-only */
# define ADM1021_REG_ONESHOT 0x0F
/* Initial values */
2007-07-19 08:00:17 +02:00
/* Note: Even though I left the low and high limits named os and hyst,
they don ' t quite work like a thermostat the way the LM75 does . I . e . ,
a lower temp than THYST actually triggers an alarm instead of
2005-04-16 15:20:36 -07:00
clearing it . Weird , ey ? - - Phil */
/* Each client has this additional data */
struct adm1021_data {
struct i2c_client client ;
2007-08-20 13:46:20 -07:00
struct device * hwmon_dev ;
2005-04-16 15:20:36 -07:00
enum chips type ;
2006-01-18 23:19:26 +01:00
struct mutex update_lock ;
2005-04-16 15:20:36 -07:00
char valid ; /* !=0 if following fields are valid */
unsigned long last_updated ; /* In jiffies */
2007-09-09 19:51:14 +02:00
s8 temp_max [ 2 ] ; /* Register values */
s8 temp_min [ 2 ] ;
s8 temp [ 2 ] ;
u8 alarms ;
/* Special values for ADM1023 only */
u8 remote_temp_prec ;
u8 remote_temp_os_prec ;
u8 remote_temp_hyst_prec ;
u8 remote_temp_offset ;
u8 remote_temp_offset_prec ;
2005-04-16 15:20:36 -07:00
} ;
static int adm1021_attach_adapter ( struct i2c_adapter * adapter ) ;
static int adm1021_detect ( struct i2c_adapter * adapter , int address , int kind ) ;
static void adm1021_init_client ( struct i2c_client * client ) ;
static int adm1021_detach_client ( struct i2c_client * client ) ;
static struct adm1021_data * adm1021_update_device ( struct device * dev ) ;
/* (amalysh) read only mode, otherwise any limit's writing confuse BIOS */
2005-09-25 16:26:44 +02:00
static int read_only ;
2005-04-16 15:20:36 -07:00
/* This is the driver that will be inserted */
static struct i2c_driver adm1021_driver = {
2005-11-26 20:37:41 +01:00
. driver = {
. name = " adm1021 " ,
} ,
2005-04-16 15:20:36 -07:00
. id = I2C_DRIVERID_ADM1021 ,
. attach_adapter = adm1021_attach_adapter ,
. detach_client = adm1021_detach_client ,
} ;
2007-09-09 19:51:14 +02:00
static ssize_t show_temp ( struct device * dev ,
struct device_attribute * devattr , char * buf )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
struct adm1021_data * data = adm1021_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , 1000 * data - > temp [ index ] ) ;
}
static ssize_t show_temp_max ( struct device * dev ,
struct device_attribute * devattr , char * buf )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
struct adm1021_data * data = adm1021_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , 1000 * data - > temp_max [ index ] ) ;
}
static ssize_t show_temp_min ( struct device * dev ,
struct device_attribute * devattr , char * buf )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
struct adm1021_data * data = adm1021_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , 1000 * data - > temp_min [ index ] ) ;
2005-04-16 15:20:36 -07:00
}
2007-09-09 19:16:03 +02:00
static ssize_t show_alarm ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
int index = to_sensor_dev_attr ( attr ) - > index ;
struct adm1021_data * data = adm1021_update_device ( dev ) ;
return sprintf ( buf , " %u \n " , ( data - > alarms > > index ) & 1 ) ;
}
2007-07-19 08:00:17 +02:00
static ssize_t show_alarms ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
{
struct adm1021_data * data = adm1021_update_device ( dev ) ;
return sprintf ( buf , " %u \n " , data - > alarms ) ;
2005-04-16 15:20:36 -07:00
}
2007-07-19 08:00:17 +02:00
2007-09-09 19:51:14 +02:00
static ssize_t set_temp_max ( struct device * dev ,
struct device_attribute * devattr ,
const char * buf , size_t count )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
struct i2c_client * client = to_i2c_client ( dev ) ;
struct adm1021_data * data = i2c_get_clientdata ( client ) ;
long temp = simple_strtol ( buf , NULL , 10 ) / 1000 ;
mutex_lock ( & data - > update_lock ) ;
data - > temp_max [ index ] = SENSORS_LIMIT ( temp , - 128 , 127 ) ;
if ( ! read_only )
i2c_smbus_write_byte_data ( client , ADM1021_REG_TOS_W ( index ) ,
data - > temp_max [ index ] ) ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
}
static ssize_t set_temp_min ( struct device * dev ,
struct device_attribute * devattr ,
const char * buf , size_t count )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
struct i2c_client * client = to_i2c_client ( dev ) ;
struct adm1021_data * data = i2c_get_clientdata ( client ) ;
long temp = simple_strtol ( buf , NULL , 10 ) / 1000 ;
mutex_lock ( & data - > update_lock ) ;
data - > temp_min [ index ] = SENSORS_LIMIT ( temp , - 128 , 127 ) ;
if ( ! read_only )
i2c_smbus_write_byte_data ( client , ADM1021_REG_THYST_W ( index ) ,
data - > temp_min [ index ] ) ;
mutex_unlock ( & data - > update_lock ) ;
return count ;
2005-04-16 15:20:36 -07:00
}
2007-09-09 19:51:14 +02:00
static SENSOR_DEVICE_ATTR ( temp1_input , S_IRUGO , show_temp , NULL , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp1_max , S_IWUSR | S_IRUGO , show_temp_max ,
set_temp_max , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp1_min , S_IWUSR | S_IRUGO , show_temp_min ,
set_temp_min , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp2_input , S_IRUGO , show_temp , NULL , 1 ) ;
static SENSOR_DEVICE_ATTR ( temp2_max , S_IWUSR | S_IRUGO , show_temp_max ,
set_temp_max , 1 ) ;
static SENSOR_DEVICE_ATTR ( temp2_min , S_IWUSR | S_IRUGO , show_temp_min ,
set_temp_min , 1 ) ;
2007-09-09 19:16:03 +02:00
static SENSOR_DEVICE_ATTR ( temp1_max_alarm , S_IRUGO , show_alarm , NULL , 6 ) ;
static SENSOR_DEVICE_ATTR ( temp1_min_alarm , S_IRUGO , show_alarm , NULL , 5 ) ;
static SENSOR_DEVICE_ATTR ( temp2_max_alarm , S_IRUGO , show_alarm , NULL , 4 ) ;
static SENSOR_DEVICE_ATTR ( temp2_min_alarm , S_IRUGO , show_alarm , NULL , 3 ) ;
static SENSOR_DEVICE_ATTR ( temp2_fault , S_IRUGO , show_alarm , NULL , 2 ) ;
2007-09-09 19:51:14 +02:00
static DEVICE_ATTR ( alarms , S_IRUGO , show_alarms , NULL ) ;
2005-04-16 15:20:36 -07:00
static int adm1021_attach_adapter ( struct i2c_adapter * adapter )
{
if ( ! ( adapter - > class & I2C_CLASS_HWMON ) )
return 0 ;
2005-07-31 21:42:02 +02:00
return i2c_probe ( adapter , & addr_data , adm1021_detect ) ;
2005-04-16 15:20:36 -07:00
}
2006-09-24 21:15:35 +02:00
static struct attribute * adm1021_attributes [ ] = {
2007-09-09 19:51:14 +02:00
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp1_min . dev_attr . attr ,
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp2_max . dev_attr . attr ,
& sensor_dev_attr_temp2_min . dev_attr . attr ,
& sensor_dev_attr_temp2_input . dev_attr . attr ,
2007-09-09 19:16:03 +02:00
& sensor_dev_attr_temp1_max_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_min_alarm . 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_fault . dev_attr . attr ,
2006-09-24 21:15:35 +02:00
& dev_attr_alarms . attr ,
NULL
} ;
static const struct attribute_group adm1021_group = {
. attrs = adm1021_attributes ,
} ;
2005-04-16 15:20:36 -07:00
static int adm1021_detect ( struct i2c_adapter * adapter , int address , int kind )
{
int i ;
2007-07-19 08:00:17 +02:00
struct i2c_client * client ;
2005-04-16 15:20:36 -07:00
struct adm1021_data * data ;
int err = 0 ;
const char * type_name = " " ;
2007-07-19 08:00:17 +02:00
int conv_rate , status , config ;
2005-04-16 15:20:36 -07:00
2007-07-19 08:00:17 +02:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) ) {
pr_debug ( " adm1021: detect failed, "
" smbus byte data not supported! \n " ) ;
2005-04-16 15:20:36 -07:00
goto error0 ;
2007-07-19 08:00:17 +02:00
}
2005-04-16 15:20:36 -07:00
/* OK. For now, we presume we have a valid client. We now create the
client structure , even though we cannot fill it completely yet .
2007-07-19 08:00:17 +02:00
But it allows us to access adm1021 register values . */
2005-04-16 15:20:36 -07:00
2005-10-17 23:08:32 +02:00
if ( ! ( data = kzalloc ( sizeof ( struct adm1021_data ) , GFP_KERNEL ) ) ) {
2007-07-19 08:00:17 +02:00
pr_debug ( " adm1021: detect failed, kzalloc failed! \n " ) ;
2005-04-16 15:20:36 -07:00
err = - ENOMEM ;
goto error0 ;
}
2007-07-19 08:00:17 +02:00
client = & data - > client ;
i2c_set_clientdata ( client , data ) ;
client - > addr = address ;
client - > adapter = adapter ;
client - > driver = & adm1021_driver ;
status = i2c_smbus_read_byte_data ( client , ADM1021_REG_STATUS ) ;
conv_rate = i2c_smbus_read_byte_data ( client ,
ADM1021_REG_CONV_RATE_R ) ;
config = i2c_smbus_read_byte_data ( client , ADM1021_REG_CONFIG_R ) ;
2005-04-16 15:20:36 -07:00
/* Now, we do the remaining detection. */
if ( kind < 0 ) {
2007-07-19 08:00:17 +02:00
if ( ( status & 0x03 ) ! = 0x00 | | ( config & 0x3F ) ! = 0x00
| | ( conv_rate & 0xF8 ) ! = 0x00 ) {
pr_debug ( " adm1021: detect failed, "
" chip not detected! \n " ) ;
2005-04-16 15:20:36 -07:00
err = - ENODEV ;
goto error1 ;
}
}
/* Determine the chip type. */
if ( kind < = 0 ) {
2007-07-19 08:00:17 +02:00
i = i2c_smbus_read_byte_data ( client , ADM1021_REG_MAN_ID ) ;
2005-04-16 15:20:36 -07:00
if ( i = = 0x41 )
2007-07-19 08:00:17 +02:00
if ( ( i2c_smbus_read_byte_data ( client ,
ADM1021_REG_DEV_ID ) & 0xF0 ) = = 0x30 )
2005-04-16 15:20:36 -07:00
kind = adm1023 ;
else
kind = adm1021 ;
else if ( i = = 0x49 )
kind = thmc10 ;
else if ( i = = 0x23 )
kind = gl523sm ;
else if ( ( i = = 0x4d ) & &
2007-07-19 08:00:17 +02:00
( i2c_smbus_read_byte_data ( client ,
ADM1021_REG_DEV_ID ) = = 0x01 ) )
2005-04-16 15:20:36 -07:00
kind = max1617a ;
else if ( i = = 0x54 )
kind = mc1066 ;
/* LM84 Mfr ID in a different place, and it has more unused bits */
2007-07-19 08:00:17 +02:00
else if ( conv_rate = = 0x00
& & ( kind = = 0 /* skip extra detection */
| | ( ( config & 0x7F ) = = 0x00
& & ( status & 0xAB ) = = 0x00 ) ) )
2005-04-16 15:20:36 -07:00
kind = lm84 ;
else
kind = max1617 ;
}
if ( kind = = max1617 ) {
type_name = " max1617 " ;
} else if ( kind = = max1617a ) {
type_name = " max1617a " ;
} else if ( kind = = adm1021 ) {
type_name = " adm1021 " ;
} else if ( kind = = adm1023 ) {
type_name = " adm1023 " ;
} else if ( kind = = thmc10 ) {
type_name = " thmc10 " ;
} else if ( kind = = lm84 ) {
type_name = " lm84 " ;
} else if ( kind = = gl523sm ) {
type_name = " gl523sm " ;
} else if ( kind = = mc1066 ) {
type_name = " mc1066 " ;
}
2007-07-19 08:00:17 +02:00
pr_debug ( " adm1021: Detected chip %s at adapter %d, address 0x%02x. \n " ,
type_name , i2c_adapter_id ( adapter ) , address ) ;
2005-04-16 15:20:36 -07:00
2007-07-19 08:00:17 +02:00
/* Fill in the remaining client fields */
strlcpy ( client - > name , type_name , I2C_NAME_SIZE ) ;
2005-04-16 15:20:36 -07:00
data - > type = kind ;
2006-01-18 23:19:26 +01:00
mutex_init ( & data - > update_lock ) ;
2005-04-16 15:20:36 -07:00
/* Tell the I2C layer a new client has arrived */
2007-07-19 08:00:17 +02:00
if ( ( err = i2c_attach_client ( client ) ) )
2005-04-16 15:20:36 -07:00
goto error1 ;
/* Initialize the ADM1021 chip */
2007-07-19 08:00:17 +02:00
if ( kind ! = lm84 & & ! read_only )
adm1021_init_client ( client ) ;
2005-04-16 15:20:36 -07:00
/* Register sysfs hooks */
2007-07-19 08:00:17 +02:00
if ( ( err = sysfs_create_group ( & client - > dev . kobj , & adm1021_group ) ) )
2006-09-24 21:15:35 +02:00
goto error2 ;
2007-08-20 13:46:20 -07:00
data - > hwmon_dev = hwmon_device_register ( & client - > dev ) ;
if ( IS_ERR ( data - > hwmon_dev ) ) {
err = PTR_ERR ( data - > hwmon_dev ) ;
2006-09-24 21:15:35 +02:00
goto error3 ;
2005-07-15 21:39:18 -04:00
}
2005-04-16 15:20:36 -07:00
return 0 ;
2006-09-24 21:15:35 +02:00
error3 :
2007-07-19 08:00:17 +02:00
sysfs_remove_group ( & client - > dev . kobj , & adm1021_group ) ;
2005-07-15 21:39:18 -04:00
error2 :
2007-07-19 08:00:17 +02:00
i2c_detach_client ( client ) ;
2005-04-16 15:20:36 -07:00
error1 :
kfree ( data ) ;
error0 :
return err ;
}
static void adm1021_init_client ( struct i2c_client * client )
{
/* Enable ADC and disable suspend mode */
2007-07-19 08:00:17 +02:00
i2c_smbus_write_byte_data ( client , ADM1021_REG_CONFIG_W ,
i2c_smbus_read_byte_data ( client , ADM1021_REG_CONFIG_R ) & 0xBF ) ;
2005-04-16 15:20:36 -07:00
/* Set Conversion rate to 1/sec (this can be tinkered with) */
2007-07-19 08:00:17 +02:00
i2c_smbus_write_byte_data ( client , ADM1021_REG_CONV_RATE_W , 0x04 ) ;
2005-04-16 15:20:36 -07:00
}
static int adm1021_detach_client ( struct i2c_client * client )
{
2005-07-15 21:39:18 -04:00
struct adm1021_data * data = i2c_get_clientdata ( client ) ;
2005-04-16 15:20:36 -07:00
int err ;
2007-08-20 13:46:20 -07:00
hwmon_device_unregister ( data - > hwmon_dev ) ;
2006-09-24 21:15:35 +02:00
sysfs_remove_group ( & client - > dev . kobj , & adm1021_group ) ;
2005-07-15 21:39:18 -04:00
2005-07-27 22:14:49 +02:00
if ( ( err = i2c_detach_client ( client ) ) )
2005-04-16 15:20:36 -07:00
return err ;
2005-07-15 21:39:18 -04:00
kfree ( data ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static struct adm1021_data * adm1021_update_device ( struct device * dev )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct adm1021_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-09-09 19:51:14 +02:00
int i ;
2005-04-16 15:20:36 -07:00
dev_dbg ( & client - > dev , " Starting adm1021 update \n " ) ;
2007-09-09 19:51:14 +02:00
for ( i = 0 ; i < 2 ; i + + ) {
data - > temp [ i ] = i2c_smbus_read_byte_data ( client ,
ADM1021_REG_TEMP ( i ) ) ;
data - > temp_max [ i ] = i2c_smbus_read_byte_data ( client ,
ADM1021_REG_TOS_R ( i ) ) ;
data - > temp_min [ i ] = i2c_smbus_read_byte_data ( client ,
ADM1021_REG_THYST_R ( i ) ) ;
}
2007-07-19 08:00:17 +02:00
data - > alarms = i2c_smbus_read_byte_data ( client ,
ADM1021_REG_STATUS ) & 0x7c ;
2005-04-16 15:20:36 -07:00
if ( data - > type = = adm1023 ) {
2007-07-19 08:00:17 +02:00
data - > remote_temp_prec =
i2c_smbus_read_byte_data ( client ,
ADM1023_REG_REM_TEMP_PREC ) ;
data - > remote_temp_os_prec =
i2c_smbus_read_byte_data ( client ,
ADM1023_REG_REM_TOS_PREC ) ;
data - > remote_temp_hyst_prec =
i2c_smbus_read_byte_data ( client ,
ADM1023_REG_REM_THYST_PREC ) ;
data - > remote_temp_offset =
i2c_smbus_read_byte_data ( client ,
ADM1023_REG_REM_OFFSET ) ;
data - > remote_temp_offset_prec =
i2c_smbus_read_byte_data ( client ,
ADM1023_REG_REM_OFFSET_PREC ) ;
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 ;
}
static int __init sensors_adm1021_init ( void )
{
return i2c_add_driver ( & adm1021_driver ) ;
}
static void __exit sensors_adm1021_exit ( void )
{
i2c_del_driver ( & adm1021_driver ) ;
}
MODULE_AUTHOR ( " Frodo Looijaard <frodol@dds.nl> and "
" Philip Edelbrock <phil@netroedge.com> " ) ;
MODULE_DESCRIPTION ( " adm1021 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_param ( read_only , bool , 0 ) ;
MODULE_PARM_DESC ( read_only , " Don't set any values, read only mode " ) ;
module_init ( sensors_adm1021_init )
module_exit ( sensors_adm1021_exit )