2019-05-20 10:19:02 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2006-03-23 18:37:23 +03:00
/*
2012-01-15 10:31:08 +04:00
* smsc47m192 . c - Support for hardware monitoring block of
* SMSC LPC47M192 and compatible Super I / O chips
*
* Copyright ( C ) 2006 Hartmut Rick < linux @ rick . claranet . de >
*
* Derived from lm78 . c and other chip drivers .
*/
2006-03-23 18:37:23 +03:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
# include <linux/hwmon.h>
# include <linux/hwmon-sysfs.h>
# include <linux/hwmon-vid.h>
# include <linux/err.h>
2006-09-24 23:25:12 +04:00
# include <linux/sysfs.h>
2007-06-09 18:11:15 +04:00
# include <linux/mutex.h>
2006-03-23 18:37:23 +03:00
/* Addresses to scan */
2008-02-18 06:28:03 +03:00
static const unsigned short normal_i2c [ ] = { 0x2c , 0x2d , I2C_CLIENT_END } ;
2006-03-23 18:37:23 +03:00
/* SMSC47M192 registers */
2012-01-15 10:31:08 +04:00
# define SMSC47M192_REG_IN(nr) ((nr) < 6 ? (0x20 + (nr)) : \
2006-03-23 18:37:23 +03:00
( 0x50 + ( nr ) - 6 ) )
2012-01-15 10:31:08 +04:00
# define SMSC47M192_REG_IN_MAX(nr) ((nr) < 6 ? (0x2b + (nr) * 2) : \
2006-03-23 18:37:23 +03:00
( 0x54 + ( ( ( nr ) - 6 ) * 2 ) ) )
2012-01-15 10:31:08 +04:00
# define SMSC47M192_REG_IN_MIN(nr) ((nr) < 6 ? (0x2c + (nr) * 2) : \
2006-03-23 18:37:23 +03:00
( 0x55 + ( ( ( nr ) - 6 ) * 2 ) ) )
static u8 SMSC47M192_REG_TEMP [ 3 ] = { 0x27 , 0x26 , 0x52 } ;
static u8 SMSC47M192_REG_TEMP_MAX [ 3 ] = { 0x39 , 0x37 , 0x58 } ;
static u8 SMSC47M192_REG_TEMP_MIN [ 3 ] = { 0x3A , 0x38 , 0x59 } ;
2012-01-15 10:31:08 +04:00
# define SMSC47M192_REG_TEMP_OFFSET(nr) ((nr) == 2 ? 0x1e : 0x1f)
2006-03-23 18:37:23 +03:00
# define SMSC47M192_REG_ALARM1 0x41
# define SMSC47M192_REG_ALARM2 0x42
# define SMSC47M192_REG_VID 0x47
# define SMSC47M192_REG_VID4 0x49
# define SMSC47M192_REG_CONFIG 0x40
# define SMSC47M192_REG_SFR 0x4f
# define SMSC47M192_REG_COMPANY_ID 0x3e
# define SMSC47M192_REG_VERSION 0x3f
/* generalised scaling with integer rounding */
static inline int SCALE ( long val , int mul , int div )
{
if ( val < 0 )
return ( val * mul - div / 2 ) / div ;
else
return ( val * mul + div / 2 ) / div ;
}
/* Conversions */
/* smsc47m192 internally scales voltage measurements */
static const u16 nom_mv [ ] = { 2500 , 2250 , 3300 , 5000 , 12000 , 3300 , 1500 , 1800 } ;
static inline unsigned int IN_FROM_REG ( u8 reg , int n )
{
return SCALE ( reg , nom_mv [ n ] , 192 ) ;
}
static inline u8 IN_TO_REG ( unsigned long val , int n )
{
2016-11-21 01:16:16 +03:00
val = clamp_val ( val , 0 , nom_mv [ n ] * 255 / 192 ) ;
return SCALE ( val , 192 , nom_mv [ n ] ) ;
2006-03-23 18:37:23 +03:00
}
2012-01-15 10:31:08 +04:00
/*
* TEMP : 0.001 degC units ( - 128 C to + 127 C )
* REG : 1 C / bit , two ' s complement
*/
2016-11-21 01:16:16 +03:00
static inline s8 TEMP_TO_REG ( long val )
2006-03-23 18:37:23 +03:00
{
2014-07-18 18:31:18 +04:00
return SCALE ( clamp_val ( val , - 128000 , 127000 ) , 1 , 1000 ) ;
2006-03-23 18:37:23 +03:00
}
static inline int TEMP_FROM_REG ( s8 val )
{
return val * 1000 ;
}
struct smsc47m192_data {
2014-07-17 15:21:44 +04:00
struct i2c_client * client ;
const struct attribute_group * groups [ 3 ] ;
2007-06-09 18:11:15 +04:00
struct mutex update_lock ;
2021-09-24 22:52:02 +03:00
bool valid ; /* true if following fields are valid */
2006-03-23 18:37:23 +03:00
unsigned long last_updated ; /* In jiffies */
u8 in [ 8 ] ; /* Register value */
u8 in_max [ 8 ] ; /* Register value */
u8 in_min [ 8 ] ; /* Register value */
s8 temp [ 3 ] ; /* Register value */
s8 temp_max [ 3 ] ; /* Register value */
s8 temp_min [ 3 ] ; /* Register value */
s8 temp_offset [ 3 ] ; /* Register value */
u16 alarms ; /* Register encoding, combined */
u8 vid ; /* Register encoding, combined */
u8 vrm ;
} ;
2014-07-17 15:20:53 +04:00
static struct smsc47m192_data * smsc47m192_update_device ( struct device * dev )
{
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2014-07-17 15:20:53 +04:00
int i , config ;
2006-03-23 18:37:23 +03:00
2014-07-17 15:20:53 +04:00
mutex_lock ( & data - > update_lock ) ;
2008-07-16 21:30:16 +04:00
2014-07-17 15:20:53 +04:00
if ( time_after ( jiffies , data - > last_updated + HZ + HZ / 2 )
| | ! data - > valid ) {
u8 sfr = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_SFR ) ;
dev_dbg ( & client - > dev , " Starting smsc47m192 update \n " ) ;
for ( i = 0 ; i < = 7 ; i + + ) {
data - > in [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_IN ( i ) ) ;
data - > in_min [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_IN_MIN ( i ) ) ;
data - > in_max [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_IN_MAX ( i ) ) ;
}
for ( i = 0 ; i < 3 ; i + + ) {
data - > temp [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_TEMP [ i ] ) ;
data - > temp_max [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_TEMP_MAX [ i ] ) ;
data - > temp_min [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_TEMP_MIN [ i ] ) ;
}
for ( i = 1 ; i < 3 ; i + + )
data - > temp_offset [ i ] = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_TEMP_OFFSET ( i ) ) ;
/*
* first offset is temp_offset [ 0 ] if SFR bit 4 is set ,
* temp_offset [ 1 ] otherwise
*/
if ( sfr & 0x10 ) {
data - > temp_offset [ 0 ] = data - > temp_offset [ 1 ] ;
data - > temp_offset [ 1 ] = 0 ;
} else
data - > temp_offset [ 0 ] = 0 ;
data - > vid = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_VID )
& 0x0f ;
config = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_CONFIG ) ;
if ( config & 0x20 )
data - > vid | = ( i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_VID4 ) & 0x01 ) < < 4 ;
data - > alarms = i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_ALARM1 ) |
( i2c_smbus_read_byte_data ( client ,
SMSC47M192_REG_ALARM2 ) < < 8 ) ;
data - > last_updated = jiffies ;
2021-09-24 22:52:02 +03:00
data - > valid = true ;
2014-07-17 15:20:53 +04:00
}
mutex_unlock ( & data - > update_lock ) ;
return data ;
}
2006-03-23 18:37:23 +03:00
/* Voltages */
2019-01-23 01:46:16 +03:00
static ssize_t in_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , IN_FROM_REG ( data - > in [ nr ] , nr ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t in_min_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , IN_FROM_REG ( data - > in_min [ nr ] , nr ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t in_max_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , IN_FROM_REG ( data - > in_max [ nr ] , nr ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t in_min_store ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2012-01-15 10:31:08 +04:00
unsigned long val ;
int err ;
err = kstrtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
2006-03-23 18:37:23 +03:00
2007-06-09 18:11:15 +04:00
mutex_lock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
data - > in_min [ nr ] = IN_TO_REG ( val , nr ) ;
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_IN_MIN ( nr ) ,
data - > in_min [ nr ] ) ;
2007-06-09 18:11:15 +04:00
mutex_unlock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
return count ;
}
2019-01-23 01:46:16 +03:00
static ssize_t in_max_store ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2012-01-15 10:31:08 +04:00
unsigned long val ;
int err ;
err = kstrtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
2006-03-23 18:37:23 +03:00
2007-06-09 18:11:15 +04:00
mutex_lock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
data - > in_max [ nr ] = IN_TO_REG ( val , nr ) ;
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_IN_MAX ( nr ) ,
data - > in_max [ nr ] ) ;
2007-06-09 18:11:15 +04:00
mutex_unlock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
return count ;
}
2019-01-23 01:46:16 +03:00
static SENSOR_DEVICE_ATTR_RO ( in0_input , in , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( in0_min , in_min , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( in0_max , in_max , 0 ) ;
static SENSOR_DEVICE_ATTR_RO ( in1_input , in , 1 ) ;
static SENSOR_DEVICE_ATTR_RW ( in1_min , in_min , 1 ) ;
static SENSOR_DEVICE_ATTR_RW ( in1_max , in_max , 1 ) ;
static SENSOR_DEVICE_ATTR_RO ( in2_input , in , 2 ) ;
static SENSOR_DEVICE_ATTR_RW ( in2_min , in_min , 2 ) ;
static SENSOR_DEVICE_ATTR_RW ( in2_max , in_max , 2 ) ;
static SENSOR_DEVICE_ATTR_RO ( in3_input , in , 3 ) ;
static SENSOR_DEVICE_ATTR_RW ( in3_min , in_min , 3 ) ;
static SENSOR_DEVICE_ATTR_RW ( in3_max , in_max , 3 ) ;
static SENSOR_DEVICE_ATTR_RO ( in4_input , in , 4 ) ;
static SENSOR_DEVICE_ATTR_RW ( in4_min , in_min , 4 ) ;
static SENSOR_DEVICE_ATTR_RW ( in4_max , in_max , 4 ) ;
static SENSOR_DEVICE_ATTR_RO ( in5_input , in , 5 ) ;
static SENSOR_DEVICE_ATTR_RW ( in5_min , in_min , 5 ) ;
static SENSOR_DEVICE_ATTR_RW ( in5_max , in_max , 5 ) ;
static SENSOR_DEVICE_ATTR_RO ( in6_input , in , 6 ) ;
static SENSOR_DEVICE_ATTR_RW ( in6_min , in_min , 6 ) ;
static SENSOR_DEVICE_ATTR_RW ( in6_max , in_max , 6 ) ;
static SENSOR_DEVICE_ATTR_RO ( in7_input , in , 7 ) ;
static SENSOR_DEVICE_ATTR_RW ( in7_min , in_min , 7 ) ;
static SENSOR_DEVICE_ATTR_RW ( in7_max , in_max , 7 ) ;
2006-03-23 18:37:23 +03:00
/* Temperatures */
2019-01-23 01:46:16 +03:00
static ssize_t temp_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , TEMP_FROM_REG ( data - > temp [ nr ] ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_min_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , TEMP_FROM_REG ( data - > temp_min [ nr ] ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_max_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , TEMP_FROM_REG ( data - > temp_max [ nr ] ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_min_store ( struct device * dev ,
struct device_attribute * attr , const char * buf ,
size_t count )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2012-01-15 10:31:08 +04:00
long val ;
int err ;
err = kstrtol ( buf , 10 , & val ) ;
if ( err )
return err ;
2006-03-23 18:37:23 +03:00
2007-06-09 18:11:15 +04:00
mutex_lock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
data - > temp_min [ nr ] = TEMP_TO_REG ( val ) ;
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_TEMP_MIN [ nr ] ,
data - > temp_min [ nr ] ) ;
2007-06-09 18:11:15 +04:00
mutex_unlock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
return count ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_max_store ( struct device * dev ,
struct device_attribute * attr , const char * buf ,
size_t count )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2012-01-15 10:31:08 +04:00
long val ;
int err ;
err = kstrtol ( buf , 10 , & val ) ;
if ( err )
return err ;
2006-03-23 18:37:23 +03:00
2007-06-09 18:11:15 +04:00
mutex_lock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
data - > temp_max [ nr ] = TEMP_TO_REG ( val ) ;
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_TEMP_MAX [ nr ] ,
data - > temp_max [ nr ] ) ;
2007-06-09 18:11:15 +04:00
mutex_unlock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
return count ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_offset_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , TEMP_FROM_REG ( data - > temp_offset [ nr ] ) ) ;
}
2019-01-23 01:46:16 +03:00
static ssize_t temp_offset_store ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
2014-07-17 15:21:44 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
struct i2c_client * client = data - > client ;
2006-03-23 18:37:23 +03:00
u8 sfr = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_SFR ) ;
2012-01-15 10:31:08 +04:00
long val ;
int err ;
err = kstrtol ( buf , 10 , & val ) ;
if ( err )
return err ;
2006-03-23 18:37:23 +03:00
2007-06-09 18:11:15 +04:00
mutex_lock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
data - > temp_offset [ nr ] = TEMP_TO_REG ( val ) ;
2012-01-15 10:31:08 +04:00
if ( nr > 1 )
2006-03-23 18:37:23 +03:00
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_TEMP_OFFSET ( nr ) , data - > temp_offset [ nr ] ) ;
else if ( data - > temp_offset [ nr ] ! = 0 ) {
2012-01-15 10:31:08 +04:00
/*
* offset [ 0 ] and offset [ 1 ] share the same register ,
* SFR bit 4 activates offset [ 0 ]
*/
2006-03-23 18:37:23 +03:00
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_SFR ,
2012-01-15 10:31:08 +04:00
( sfr & 0xef ) | ( nr = = 0 ? 0x10 : 0 ) ) ;
2006-03-23 18:37:23 +03:00
data - > temp_offset [ 1 - nr ] = 0 ;
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_TEMP_OFFSET ( nr ) , data - > temp_offset [ nr ] ) ;
2012-01-15 10:31:08 +04:00
} else if ( ( sfr & 0x10 ) = = ( nr = = 0 ? 0x10 : 0 ) )
2006-03-23 18:37:23 +03:00
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_TEMP_OFFSET ( nr ) , 0 ) ;
2007-06-09 18:11:15 +04:00
mutex_unlock ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
return count ;
}
2019-01-23 01:46:16 +03:00
static SENSOR_DEVICE_ATTR_RO ( temp1_input , temp , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_min , temp_min , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_max , temp_max , 0 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp1_offset , temp_offset , 0 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_input , temp , 1 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_min , temp_min , 1 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_max , temp_max , 1 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp2_offset , temp_offset , 1 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp3_input , temp , 2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp3_min , temp_min , 2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp3_max , temp_max , 2 ) ;
static SENSOR_DEVICE_ATTR_RW ( temp3_offset , temp_offset , 2 ) ;
2006-03-23 18:37:23 +03:00
/* VID */
2016-12-22 15:05:07 +03:00
static ssize_t cpu0_vid_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2006-03-23 18:37:23 +03:00
{
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %d \n " , vid_from_reg ( data - > vid , data - > vrm ) ) ;
}
2016-12-22 15:05:07 +03:00
static DEVICE_ATTR_RO ( cpu0_vid ) ;
2006-03-23 18:37:23 +03:00
2016-12-22 15:05:07 +03:00
static ssize_t vrm_show ( struct device * dev , struct device_attribute * attr ,
2006-03-23 18:37:23 +03:00
char * buf )
{
2007-10-08 20:24:35 +04:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
2006-03-23 18:37:23 +03:00
return sprintf ( buf , " %d \n " , data - > vrm ) ;
}
2016-12-22 15:05:07 +03:00
static ssize_t vrm_store ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
2006-03-23 18:37:23 +03:00
{
2007-12-01 13:25:33 +03:00
struct smsc47m192_data * data = dev_get_drvdata ( dev ) ;
2012-01-15 10:31:08 +04:00
unsigned long val ;
int err ;
err = kstrtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
2014-07-18 18:31:18 +04:00
if ( val > 255 )
return - EINVAL ;
2012-01-15 10:31:08 +04:00
data - > vrm = val ;
2006-03-23 18:37:23 +03:00
return count ;
}
2016-12-22 15:05:07 +03:00
static DEVICE_ATTR_RW ( vrm ) ;
2006-03-23 18:37:23 +03:00
/* Alarms */
2019-01-23 01:46:16 +03:00
static ssize_t alarm_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
2006-03-23 18:37:23 +03:00
{
struct sensor_device_attribute * sensor_attr = to_sensor_dev_attr ( attr ) ;
int nr = sensor_attr - > index ;
struct smsc47m192_data * data = smsc47m192_update_device ( dev ) ;
return sprintf ( buf , " %u \n " , ( data - > alarms & nr ) ? 1 : 0 ) ;
}
2019-01-23 01:46:16 +03:00
static SENSOR_DEVICE_ATTR_RO ( temp1_alarm , alarm , 0x0010 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_alarm , alarm , 0x0020 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp3_alarm , alarm , 0x0040 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp2_fault , alarm , 0x4000 ) ;
static SENSOR_DEVICE_ATTR_RO ( temp3_fault , alarm , 0x8000 ) ;
static SENSOR_DEVICE_ATTR_RO ( in0_alarm , alarm , 0x0001 ) ;
static SENSOR_DEVICE_ATTR_RO ( in1_alarm , alarm , 0x0002 ) ;
static SENSOR_DEVICE_ATTR_RO ( in2_alarm , alarm , 0x0004 ) ;
static SENSOR_DEVICE_ATTR_RO ( in3_alarm , alarm , 0x0008 ) ;
static SENSOR_DEVICE_ATTR_RO ( in4_alarm , alarm , 0x0100 ) ;
static SENSOR_DEVICE_ATTR_RO ( in5_alarm , alarm , 0x0200 ) ;
static SENSOR_DEVICE_ATTR_RO ( in6_alarm , alarm , 0x0400 ) ;
static SENSOR_DEVICE_ATTR_RO ( in7_alarm , alarm , 0x0800 ) ;
2006-03-23 18:37:23 +03:00
2006-09-24 23:25:12 +04:00
static struct attribute * smsc47m192_attributes [ ] = {
& sensor_dev_attr_in0_input . dev_attr . attr ,
& sensor_dev_attr_in0_min . dev_attr . attr ,
& sensor_dev_attr_in0_max . dev_attr . attr ,
& sensor_dev_attr_in0_alarm . dev_attr . attr ,
& sensor_dev_attr_in1_input . dev_attr . attr ,
& sensor_dev_attr_in1_min . dev_attr . attr ,
& sensor_dev_attr_in1_max . dev_attr . attr ,
& sensor_dev_attr_in1_alarm . dev_attr . attr ,
& sensor_dev_attr_in2_input . dev_attr . attr ,
& sensor_dev_attr_in2_min . dev_attr . attr ,
& sensor_dev_attr_in2_max . dev_attr . attr ,
& sensor_dev_attr_in2_alarm . dev_attr . attr ,
& sensor_dev_attr_in3_input . dev_attr . attr ,
& sensor_dev_attr_in3_min . dev_attr . attr ,
& sensor_dev_attr_in3_max . dev_attr . attr ,
& sensor_dev_attr_in3_alarm . dev_attr . attr ,
& sensor_dev_attr_in5_input . dev_attr . attr ,
& sensor_dev_attr_in5_min . dev_attr . attr ,
& sensor_dev_attr_in5_max . dev_attr . attr ,
& sensor_dev_attr_in5_alarm . dev_attr . attr ,
& sensor_dev_attr_in6_input . dev_attr . attr ,
& sensor_dev_attr_in6_min . dev_attr . attr ,
& sensor_dev_attr_in6_max . dev_attr . attr ,
& sensor_dev_attr_in6_alarm . dev_attr . attr ,
& sensor_dev_attr_in7_input . dev_attr . attr ,
& sensor_dev_attr_in7_min . dev_attr . attr ,
& sensor_dev_attr_in7_max . dev_attr . attr ,
& sensor_dev_attr_in7_alarm . dev_attr . attr ,
& sensor_dev_attr_temp1_input . dev_attr . attr ,
& sensor_dev_attr_temp1_max . dev_attr . attr ,
& sensor_dev_attr_temp1_min . dev_attr . attr ,
& sensor_dev_attr_temp1_offset . dev_attr . attr ,
& sensor_dev_attr_temp1_alarm . dev_attr . attr ,
& sensor_dev_attr_temp2_input . dev_attr . attr ,
& sensor_dev_attr_temp2_max . dev_attr . attr ,
& sensor_dev_attr_temp2_min . dev_attr . attr ,
& sensor_dev_attr_temp2_offset . dev_attr . attr ,
& sensor_dev_attr_temp2_alarm . dev_attr . attr ,
2007-06-09 18:11:16 +04:00
& sensor_dev_attr_temp2_fault . dev_attr . attr ,
2006-09-24 23:25:12 +04:00
& sensor_dev_attr_temp3_input . dev_attr . attr ,
& sensor_dev_attr_temp3_max . dev_attr . attr ,
& sensor_dev_attr_temp3_min . dev_attr . attr ,
& sensor_dev_attr_temp3_offset . dev_attr . attr ,
& sensor_dev_attr_temp3_alarm . dev_attr . attr ,
2007-06-09 18:11:16 +04:00
& sensor_dev_attr_temp3_fault . dev_attr . attr ,
2006-09-24 23:25:12 +04:00
& dev_attr_cpu0_vid . attr ,
& dev_attr_vrm . attr ,
NULL
} ;
static const struct attribute_group smsc47m192_group = {
. attrs = smsc47m192_attributes ,
} ;
static struct attribute * smsc47m192_attributes_in4 [ ] = {
& sensor_dev_attr_in4_input . dev_attr . attr ,
& sensor_dev_attr_in4_min . dev_attr . attr ,
& sensor_dev_attr_in4_max . dev_attr . attr ,
& sensor_dev_attr_in4_alarm . dev_attr . attr ,
NULL
} ;
static const struct attribute_group smsc47m192_group_in4 = {
. attrs = smsc47m192_attributes_in4 ,
} ;
2006-03-23 18:37:23 +03:00
static void smsc47m192_init_client ( struct i2c_client * client )
{
int i ;
u8 config = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_CONFIG ) ;
u8 sfr = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_SFR ) ;
/* select cycle mode (pause 1 sec between updates) */
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_SFR ,
( sfr & 0xfd ) | 0x02 ) ;
if ( ! ( config & 0x01 ) ) {
/* initialize alarm limits */
2012-01-15 10:31:08 +04:00
for ( i = 0 ; i < 8 ; i + + ) {
2006-03-23 18:37:23 +03:00
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_IN_MIN ( i ) , 0 ) ;
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_IN_MAX ( i ) , 0xff ) ;
}
2012-01-15 10:31:08 +04:00
for ( i = 0 ; i < 3 ; i + + ) {
2006-03-23 18:37:23 +03:00
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_TEMP_MIN [ i ] , 0x80 ) ;
i2c_smbus_write_byte_data ( client ,
SMSC47M192_REG_TEMP_MAX [ i ] , 0x7f ) ;
}
/* start monitoring */
i2c_smbus_write_byte_data ( client , SMSC47M192_REG_CONFIG ,
( config & 0xf7 ) | 0x01 ) ;
}
}
2008-07-16 21:30:16 +04:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2009-12-14 23:17:23 +03:00
static int smsc47m192_detect ( struct i2c_client * client ,
2008-07-16 21:30:16 +04:00
struct i2c_board_info * info )
2006-03-23 18:37:23 +03:00
{
2008-07-16 21:30:16 +04:00
struct i2c_adapter * adapter = client - > adapter ;
int version ;
2006-03-23 18:37:23 +03:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA ) )
2008-07-16 21:30:16 +04:00
return - ENODEV ;
2006-03-23 18:37:23 +03:00
/* Detection criteria from sensors_detect script */
2009-12-09 22:35:57 +03:00
version = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_VERSION ) ;
if ( i2c_smbus_read_byte_data ( client ,
2006-03-23 18:37:23 +03:00
SMSC47M192_REG_COMPANY_ID ) = = 0x55
2009-12-09 22:35:57 +03:00
& & ( version & 0xf0 ) = = 0x20
& & ( i2c_smbus_read_byte_data ( client ,
2006-03-23 18:37:23 +03:00
SMSC47M192_REG_VID ) & 0x70 ) = = 0x00
2009-12-09 22:35:57 +03:00
& & ( i2c_smbus_read_byte_data ( client ,
2006-03-23 18:37:23 +03:00
SMSC47M192_REG_VID4 ) & 0xfe ) = = 0x80 ) {
2009-12-09 22:35:57 +03:00
dev_info ( & adapter - > dev ,
" found SMSC47M192 or compatible, "
" version 2, stepping A%d \n " , version & 0x0f ) ;
} else {
dev_dbg ( & adapter - > dev ,
" SMSC47M192 detection failed at 0x%02x \n " ,
client - > addr ) ;
return - ENODEV ;
2006-03-23 18:37:23 +03:00
}
2008-07-16 21:30:16 +04:00
strlcpy ( info - > type , " smsc47m192 " , I2C_NAME_SIZE ) ;
return 0 ;
}
2020-08-13 19:02:22 +03:00
static int smsc47m192_probe ( struct i2c_client * client )
2008-07-16 21:30:16 +04:00
{
2014-07-17 15:21:44 +04:00
struct device * dev = & client - > dev ;
struct device * hwmon_dev ;
2008-07-16 21:30:16 +04:00
struct smsc47m192_data * data ;
int config ;
2014-07-17 15:21:44 +04:00
data = devm_kzalloc ( dev , sizeof ( struct smsc47m192_data ) , GFP_KERNEL ) ;
2012-06-02 23:04:07 +04:00
if ( ! data )
return - ENOMEM ;
2008-07-16 21:30:16 +04:00
2014-07-17 15:21:44 +04:00
data - > client = client ;
2006-03-23 18:37:23 +03:00
data - > vrm = vid_which_vrm ( ) ;
2007-06-09 18:11:15 +04:00
mutex_init ( & data - > update_lock ) ;
2006-03-23 18:37:23 +03:00
/* Initialize the SMSC47M192 chip */
smsc47m192_init_client ( client ) ;
2014-07-17 15:21:44 +04:00
/* sysfs hooks */
data - > groups [ 0 ] = & smsc47m192_group ;
2006-03-23 18:37:23 +03:00
/* Pin 110 is either in4 (+12V) or VID4 */
config = i2c_smbus_read_byte_data ( client , SMSC47M192_REG_CONFIG ) ;
2014-07-17 15:21:44 +04:00
if ( ! ( config & 0x20 ) )
data - > groups [ 1 ] = & smsc47m192_group_in4 ;
2006-03-23 18:37:23 +03:00
2014-07-17 15:21:44 +04:00
hwmon_dev = devm_hwmon_device_register_with_groups ( dev , client - > name ,
data , data - > groups ) ;
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2006-03-23 18:37:23 +03:00
}
2014-07-17 15:20:53 +04:00
static const struct i2c_device_id smsc47m192_id [ ] = {
{ " smsc47m192 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , smsc47m192_id ) ;
2006-03-23 18:37:23 +03:00
2014-07-17 15:20:53 +04:00
static struct i2c_driver smsc47m192_driver = {
. class = I2C_CLASS_HWMON ,
. driver = {
. name = " smsc47m192 " ,
} ,
2020-08-13 19:02:22 +03:00
. probe_new = smsc47m192_probe ,
2014-07-17 15:20:53 +04:00
. id_table = smsc47m192_id ,
. detect = smsc47m192_detect ,
. address_list = normal_i2c ,
} ;
2006-03-23 18:37:23 +03:00
2012-01-20 11:38:18 +04:00
module_i2c_driver ( smsc47m192_driver ) ;
2006-03-23 18:37:23 +03:00
MODULE_AUTHOR ( " Hartmut Rick <linux@rick.claranet.de> " ) ;
MODULE_DESCRIPTION ( " SMSC47M192 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;