2005-04-17 02:20:36 +04:00
/*
2010-10-28 22:31:44 +04: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-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/i2c.h>
2005-07-16 05:39:18 +04:00
# include <linux/hwmon.h>
2007-05-08 19:22:01 +04:00
# include <linux/hwmon-sysfs.h>
2005-07-16 05:39:18 +04:00
# include <linux/err.h>
2017-02-24 16:13:02 +03:00
# include <linux/of_device.h>
2013-07-16 22:54:55 +04:00
# include <linux/of.h>
2016-06-20 03:49:19 +03:00
# include <linux/regmap.h>
2005-04-17 02:20:36 +04:00
# include "lm75.h"
2008-04-21 23:10:53 +04:00
/*
* This driver handles the LM75 and compatible digital temperature sensors .
*/
2008-05-04 06:33:15 +04:00
enum lm75_type { /* keep sorted in alphabetical order */
2011-10-13 12:43:31 +04:00
adt75 ,
2009-12-14 23:17:26 +03:00
ds1775 ,
2008-05-04 06:33:15 +04:00
ds75 ,
2013-05-04 16:49:36 +04:00
ds7505 ,
2013-11-09 21:39:14 +04:00
g751 ,
2009-12-14 23:17:26 +03:00
lm75 ,
2008-05-04 06:33:15 +04:00
lm75a ,
2014-11-18 19:08:04 +03:00
lm75b ,
2008-05-04 06:33:15 +04:00
max6625 ,
max6626 ,
2018-09-11 23:25:58 +03:00
max31725 ,
2008-05-04 06:33:15 +04:00
mcp980x ,
stds75 ,
2018-12-06 00:14:22 +03:00
stlm75 ,
2008-05-04 06:33:15 +04:00
tcn75 ,
tmp100 ,
tmp101 ,
2010-05-27 21:59:03 +04:00
tmp105 ,
2014-06-26 13:21:11 +04:00
tmp112 ,
2008-05-04 06:33:15 +04:00
tmp175 ,
tmp275 ,
tmp75 ,
2015-10-08 05:55:20 +03:00
tmp75c ,
2008-05-04 06:33:15 +04:00
} ;
2008-08-11 00:56:16 +04:00
/* Addresses scanned */
2008-02-18 06:28:03 +03:00
static const unsigned short normal_i2c [ ] = { 0x48 , 0x49 , 0x4a , 0x4b , 0x4c ,
2005-04-17 02:20:36 +04:00
0x4d , 0x4e , 0x4f , I2C_CLIENT_END } ;
/* The LM75 registers */
2016-06-20 03:49:19 +03:00
# define LM75_REG_TEMP 0x00
2005-04-17 02:20:36 +04:00
# define LM75_REG_CONF 0x01
2016-06-20 03:49:19 +03:00
# define LM75_REG_HYST 0x02
# define LM75_REG_MAX 0x03
2005-04-17 02:20:36 +04:00
/* Each client has this additional data */
struct lm75_data {
2014-01-14 04:00:37 +04:00
struct i2c_client * client ;
2016-06-20 03:49:19 +03:00
struct regmap * regmap ;
2008-05-04 06:33:15 +04:00
u8 orig_conf ;
2018-09-11 23:25:58 +03:00
u8 resolution ; /* In bits, between 9 and 16 */
2013-05-04 16:49:36 +04:00
u8 resolution_limits ;
2016-06-20 03:49:19 +03:00
unsigned int sample_time ; /* In ms */
2005-04-17 02:20:36 +04:00
} ;
2008-04-21 23:10:53 +04:00
/*-----------------------------------------------------------------------*/
2013-07-16 22:54:55 +04:00
static inline long lm75_reg_to_mc ( s16 temp , u8 resolution )
{
return ( ( temp > > ( 16 - resolution ) ) * 1000 ) > > ( resolution - 8 ) ;
}
2016-06-20 05:15:05 +03:00
static int lm75_read ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
2013-07-16 22:54:55 +04:00
{
2016-06-20 03:49:19 +03:00
struct lm75_data * data = dev_get_drvdata ( dev ) ;
2016-06-20 05:15:05 +03:00
unsigned int regval ;
int err , reg ;
switch ( type ) {
case hwmon_chip :
switch ( attr ) {
case hwmon_chip_update_interval :
* val = data - > sample_time ;
2018-01-17 21:24:48 +03:00
break ;
2016-06-20 05:15:05 +03:00
default :
return - EINVAL ;
}
break ;
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_input :
reg = LM75_REG_TEMP ;
break ;
case hwmon_temp_max :
reg = LM75_REG_MAX ;
break ;
case hwmon_temp_max_hyst :
reg = LM75_REG_HYST ;
break ;
default :
return - EINVAL ;
}
err = regmap_read ( data - > regmap , reg , & regval ) ;
if ( err < 0 )
return err ;
* val = lm75_reg_to_mc ( regval , data - > resolution ) ;
break ;
default :
return - EINVAL ;
}
2013-07-16 22:54:55 +04:00
return 0 ;
}
2016-06-20 05:15:05 +03:00
static int lm75_write ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long temp )
2007-05-08 19:22:01 +04:00
{
2016-06-20 03:49:19 +03:00
struct lm75_data * data = dev_get_drvdata ( dev ) ;
2013-05-04 16:49:36 +04:00
u8 resolution ;
2016-06-20 05:15:05 +03:00
int reg ;
if ( type ! = hwmon_temp )
return - EINVAL ;
2010-10-28 22:31:44 +04:00
2016-06-20 05:15:05 +03:00
switch ( attr ) {
case hwmon_temp_max :
reg = LM75_REG_MAX ;
break ;
case hwmon_temp_max_hyst :
reg = LM75_REG_HYST ;
break ;
default :
return - EINVAL ;
}
2007-05-08 19:22:01 +04:00
2013-05-04 16:49:36 +04:00
/*
* Resolution of limit registers is assumed to be the same as the
* temperature input register resolution unless given explicitly .
*/
2016-06-20 05:15:05 +03:00
if ( data - > resolution_limits )
2013-05-04 16:49:36 +04:00
resolution = data - > resolution_limits ;
else
resolution = data - > resolution ;
temp = clamp_val ( temp , LM75_TEMP_MIN , LM75_TEMP_MAX ) ;
2016-06-20 03:49:19 +03:00
temp = DIV_ROUND_CLOSEST ( temp < < ( resolution - 8 ) ,
1000 ) < < ( 16 - resolution ) ;
2016-06-20 05:15:05 +03:00
return regmap_write ( data - > regmap , reg , temp ) ;
2005-04-17 02:20:36 +04:00
}
2016-06-20 05:15:05 +03:00
static umode_t lm75_is_visible ( const void * data , enum hwmon_sensor_types type ,
u32 attr , int channel )
2016-06-20 03:56:22 +03:00
{
2016-06-20 05:15:05 +03:00
switch ( type ) {
case hwmon_chip :
switch ( attr ) {
case hwmon_chip_update_interval :
2018-12-11 01:02:11 +03:00
return 0444 ;
2016-06-20 05:15:05 +03:00
}
break ;
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_input :
2018-12-11 01:02:11 +03:00
return 0444 ;
2016-06-20 05:15:05 +03:00
case hwmon_temp_max :
case hwmon_temp_max_hyst :
2018-12-11 01:02:11 +03:00
return 0644 ;
2016-06-20 05:15:05 +03:00
}
break ;
default :
break ;
}
return 0 ;
2016-06-20 03:56:22 +03:00
}
2016-06-20 05:15:05 +03:00
/*-----------------------------------------------------------------------*/
2005-04-17 02:20:36 +04:00
2016-06-20 05:15:05 +03:00
/* device probe and removal */
2006-09-24 22:59:49 +04:00
2016-06-20 05:15:05 +03:00
/* chip configuration */
static const u32 lm75_chip_config [ ] = {
HWMON_C_REGISTER_TZ | HWMON_C_UPDATE_INTERVAL ,
0
2006-09-24 22:59:49 +04:00
} ;
2016-06-20 05:15:05 +03:00
static const struct hwmon_channel_info lm75_chip = {
. type = hwmon_chip ,
. config = lm75_chip_config ,
2014-11-08 04:24:39 +03:00
} ;
2016-06-20 05:15:05 +03:00
static const u32 lm75_temp_config [ ] = {
HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MAX_HYST ,
0
} ;
2008-04-21 23:10:53 +04:00
2016-06-20 05:15:05 +03:00
static const struct hwmon_channel_info lm75_temp = {
. type = hwmon_temp ,
. config = lm75_temp_config ,
} ;
static const struct hwmon_channel_info * lm75_info [ ] = {
& lm75_chip ,
& lm75_temp ,
NULL
} ;
static const struct hwmon_ops lm75_hwmon_ops = {
. is_visible = lm75_is_visible ,
. read = lm75_read ,
. write = lm75_write ,
} ;
static const struct hwmon_chip_info lm75_chip_info = {
. ops = & lm75_hwmon_ops ,
. info = lm75_info ,
} ;
2008-05-04 06:33:15 +04:00
2016-06-20 03:49:19 +03:00
static bool lm75_is_writeable_reg ( struct device * dev , unsigned int reg )
{
return reg ! = LM75_REG_TEMP ;
}
static bool lm75_is_volatile_reg ( struct device * dev , unsigned int reg )
{
return reg = = LM75_REG_TEMP ;
}
static const struct regmap_config lm75_regmap_config = {
. reg_bits = 8 ,
. val_bits = 16 ,
. max_register = LM75_REG_MAX ,
. writeable_reg = lm75_is_writeable_reg ,
. volatile_reg = lm75_is_volatile_reg ,
. val_format_endian = REGMAP_ENDIAN_BIG ,
. cache_type = REGCACHE_RBTREE ,
2018-09-01 19:50:41 +03:00
. use_single_read = true ,
. use_single_write = true ,
2016-06-20 03:49:19 +03:00
} ;
2016-06-20 03:06:48 +03:00
static void lm75_remove ( void * data )
{
struct lm75_data * lm75 = data ;
struct i2c_client * client = lm75 - > client ;
i2c_smbus_write_byte_data ( client , LM75_REG_CONF , lm75 - > orig_conf ) ;
}
2008-05-04 06:33:15 +04:00
static int
lm75_probe ( struct i2c_client * client , const struct i2c_device_id * id )
{
2014-01-14 04:00:37 +04:00
struct device * dev = & client - > dev ;
2016-06-20 03:06:48 +03:00
struct device * hwmon_dev ;
2008-05-04 06:33:15 +04:00
struct lm75_data * data ;
2016-07-26 00:56:00 +03:00
int status , err ;
2008-05-04 06:33:15 +04:00
u8 set_mask , clr_mask ;
int new ;
2017-02-24 16:13:02 +03:00
enum lm75_type kind ;
if ( client - > dev . of_node )
kind = ( enum lm75_type ) of_device_get_match_data ( & client - > dev ) ;
else
kind = id - > driver_data ;
2008-05-04 06:33:15 +04:00
if ( ! i2c_check_functionality ( client - > adapter ,
I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA ) )
return - EIO ;
2014-01-14 04:00:37 +04:00
data = devm_kzalloc ( dev , sizeof ( struct lm75_data ) , GFP_KERNEL ) ;
2008-05-04 06:33:15 +04:00
if ( ! data )
return - ENOMEM ;
2014-01-14 04:00:37 +04:00
data - > client = client ;
2016-06-20 03:49:19 +03:00
data - > regmap = devm_regmap_init_i2c ( client , & lm75_regmap_config ) ;
if ( IS_ERR ( data - > regmap ) )
return PTR_ERR ( data - > regmap ) ;
2008-05-04 06:33:15 +04:00
/* Set to LM75 resolution (9 bits, 1/2 degree C) and range.
* Then tweak to be more precise when appropriate .
*/
set_mask = 0 ;
2013-05-04 16:49:36 +04:00
clr_mask = LM75_SHUTDOWN ; /* continuous conversions */
2013-05-04 16:49:36 +04:00
switch ( kind ) {
2013-05-04 16:49:36 +04:00
case adt75 :
clr_mask | = 1 < < 5 ; /* not one-shot mode */
2013-05-04 16:49:36 +04:00
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 8 ;
2013-05-04 16:49:36 +04:00
break ;
case ds1775 :
case ds75 :
case stds75 :
2013-05-04 16:49:36 +04:00
clr_mask | = 3 < < 5 ;
set_mask | = 2 < < 5 ; /* 11-bit mode */
data - > resolution = 11 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC ;
2013-05-04 16:49:36 +04:00
break ;
2018-12-06 00:14:22 +03:00
case stlm75 :
data - > resolution = 9 ;
data - > sample_time = MSEC_PER_SEC / 5 ;
break ;
2013-05-04 16:49:36 +04:00
case ds7505 :
set_mask | = 3 < < 5 ; /* 12-bit mode */
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2013-05-04 16:49:36 +04:00
break ;
2013-11-09 21:39:14 +04:00
case g751 :
2013-05-04 16:49:36 +04:00
case lm75 :
case lm75a :
data - > resolution = 9 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 2 ;
2013-05-04 16:49:36 +04:00
break ;
2014-11-18 19:08:04 +03:00
case lm75b :
data - > resolution = 11 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2014-11-18 19:08:04 +03:00
break ;
2013-05-04 16:49:36 +04:00
case max6625 :
data - > resolution = 9 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2013-05-04 16:49:36 +04:00
break ;
case max6626 :
data - > resolution = 12 ;
data - > resolution_limits = 9 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2013-05-04 16:49:36 +04:00
break ;
2018-09-11 23:25:58 +03:00
case max31725 :
data - > resolution = 16 ;
data - > sample_time = MSEC_PER_SEC / 8 ;
break ;
2013-05-04 16:49:36 +04:00
case tcn75 :
data - > resolution = 9 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 8 ;
2013-05-04 16:49:36 +04:00
break ;
case mcp980x :
2013-05-04 16:49:36 +04:00
data - > resolution_limits = 9 ;
/* fall through */
2013-05-04 16:49:36 +04:00
case tmp100 :
case tmp101 :
2013-05-04 16:49:36 +04:00
set_mask | = 3 < < 5 ; /* 12-bit mode */
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC ;
2013-05-04 16:49:36 +04:00
clr_mask | = 1 < < 7 ; /* not one-shot mode */
break ;
2014-06-26 13:21:11 +04:00
case tmp112 :
set_mask | = 3 < < 5 ; /* 12-bit mode */
clr_mask | = 1 < < 7 ; /* not one-shot mode */
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2014-06-26 13:21:11 +04:00
break ;
2013-05-04 16:49:36 +04:00
case tmp105 :
case tmp175 :
case tmp275 :
case tmp75 :
2013-05-04 16:49:36 +04:00
set_mask | = 3 < < 5 ; /* 12-bit mode */
2013-05-04 16:49:36 +04:00
clr_mask | = 1 < < 7 ; /* not one-shot mode */
2013-05-04 16:49:36 +04:00
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 2 ;
2013-05-04 16:49:36 +04:00
break ;
2015-10-08 05:55:20 +03:00
case tmp75c :
clr_mask | = 1 < < 5 ; /* not one-shot mode */
data - > resolution = 12 ;
2016-06-20 03:49:19 +03:00
data - > sample_time = MSEC_PER_SEC / 4 ;
2015-10-08 05:55:20 +03:00
break ;
2013-05-04 16:49:36 +04:00
}
2008-05-04 06:33:15 +04:00
/* configure as specified */
2016-06-20 03:11:13 +03:00
status = i2c_smbus_read_byte_data ( client , LM75_REG_CONF ) ;
2008-05-04 06:33:15 +04:00
if ( status < 0 ) {
2014-01-14 04:00:37 +04:00
dev_dbg ( dev , " Can't read config? %d \n " , status ) ;
2012-06-02 20:58:08 +04:00
return status ;
2008-05-04 06:33:15 +04:00
}
data - > orig_conf = status ;
new = status & ~ clr_mask ;
new | = set_mask ;
if ( status ! = new )
2016-06-20 03:11:13 +03:00
i2c_smbus_write_byte_data ( client , LM75_REG_CONF , new ) ;
2008-05-04 06:33:15 +04:00
2016-07-26 00:56:00 +03:00
err = devm_add_action_or_reset ( dev , lm75_remove , data ) ;
if ( err )
return err ;
2008-05-04 06:33:15 +04:00
2016-06-20 03:06:48 +03:00
dev_dbg ( dev , " Config %02x \n " , new ) ;
2013-07-16 22:54:55 +04:00
2016-06-20 05:15:05 +03:00
hwmon_dev = devm_hwmon_device_register_with_info ( dev , client - > name ,
data , & lm75_chip_info ,
NULL ) ;
2016-06-20 03:06:48 +03:00
if ( IS_ERR ( hwmon_dev ) )
return PTR_ERR ( hwmon_dev ) ;
2008-05-04 06:33:15 +04:00
2016-06-20 03:06:48 +03:00
dev_info ( dev , " %s: sensor '%s' \n " , dev_name ( hwmon_dev ) , client - > name ) ;
2008-05-04 06:33:15 +04:00
return 0 ;
}
static const struct i2c_device_id lm75_ids [ ] = {
2011-10-13 12:43:31 +04:00
{ " adt75 " , adt75 , } ,
2008-05-04 06:33:15 +04:00
{ " ds1775 " , ds1775 , } ,
{ " ds75 " , ds75 , } ,
2013-05-04 16:49:36 +04:00
{ " ds7505 " , ds7505 , } ,
2013-11-09 21:39:14 +04:00
{ " g751 " , g751 , } ,
2008-05-04 06:33:15 +04:00
{ " lm75 " , lm75 , } ,
{ " lm75a " , lm75a , } ,
2014-11-18 19:08:04 +03:00
{ " lm75b " , lm75b , } ,
2008-05-04 06:33:15 +04:00
{ " max6625 " , max6625 , } ,
{ " max6626 " , max6626 , } ,
2018-09-11 23:25:58 +03:00
{ " max31725 " , max31725 , } ,
{ " max31726 " , max31725 , } ,
2008-05-04 06:33:15 +04:00
{ " mcp980x " , mcp980x , } ,
{ " stds75 " , stds75 , } ,
2018-12-06 00:14:22 +03:00
{ " stlm75 " , stlm75 , } ,
2008-05-04 06:33:15 +04:00
{ " tcn75 " , tcn75 , } ,
{ " tmp100 " , tmp100 , } ,
{ " tmp101 " , tmp101 , } ,
2010-05-27 21:59:03 +04:00
{ " tmp105 " , tmp105 , } ,
2014-06-26 13:21:11 +04:00
{ " tmp112 " , tmp112 , } ,
2008-05-04 06:33:15 +04:00
{ " tmp175 " , tmp175 , } ,
{ " tmp275 " , tmp275 , } ,
{ " tmp75 " , tmp75 , } ,
2015-10-08 05:55:20 +03:00
{ " tmp75c " , tmp75c , } ,
2008-05-04 06:33:15 +04:00
{ /* LIST END */ }
} ;
MODULE_DEVICE_TABLE ( i2c , lm75_ids ) ;
2017-02-24 16:13:02 +03:00
static const struct of_device_id lm75_of_match [ ] = {
{
. compatible = " adi,adt75 " ,
. data = ( void * ) adt75
} ,
{
. compatible = " dallas,ds1775 " ,
. data = ( void * ) ds1775
} ,
{
. compatible = " dallas,ds75 " ,
. data = ( void * ) ds75
} ,
{
. compatible = " dallas,ds7505 " ,
. data = ( void * ) ds7505
} ,
{
. compatible = " gmt,g751 " ,
. data = ( void * ) g751
} ,
{
. compatible = " national,lm75 " ,
. data = ( void * ) lm75
} ,
{
. compatible = " national,lm75a " ,
. data = ( void * ) lm75a
} ,
{
. compatible = " national,lm75b " ,
. data = ( void * ) lm75b
} ,
{
. compatible = " maxim,max6625 " ,
. data = ( void * ) max6625
} ,
{
. compatible = " maxim,max6626 " ,
. data = ( void * ) max6626
} ,
2018-09-11 23:25:58 +03:00
{
. compatible = " maxim,max31725 " ,
. data = ( void * ) max31725
} ,
{
. compatible = " maxim,max31726 " ,
. data = ( void * ) max31725
} ,
2017-02-24 16:13:02 +03:00
{
. compatible = " maxim,mcp980x " ,
. data = ( void * ) mcp980x
} ,
{
. compatible = " st,stds75 " ,
. data = ( void * ) stds75
} ,
2018-12-06 00:14:22 +03:00
{
. compatible = " st,stlm75 " ,
. data = ( void * ) stlm75
} ,
2017-02-24 16:13:02 +03:00
{
. compatible = " microchip,tcn75 " ,
. data = ( void * ) tcn75
} ,
{
. compatible = " ti,tmp100 " ,
. data = ( void * ) tmp100
} ,
{
. compatible = " ti,tmp101 " ,
. data = ( void * ) tmp101
} ,
{
. compatible = " ti,tmp105 " ,
. data = ( void * ) tmp105
} ,
{
. compatible = " ti,tmp112 " ,
. data = ( void * ) tmp112
} ,
{
. compatible = " ti,tmp175 " ,
. data = ( void * ) tmp175
} ,
{
. compatible = " ti,tmp275 " ,
. data = ( void * ) tmp275
} ,
{
. compatible = " ti,tmp75 " ,
. data = ( void * ) tmp75
} ,
{
. compatible = " ti,tmp75c " ,
. data = ( void * ) tmp75c
} ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , lm75_of_match ) ;
2011-03-21 19:59:36 +03:00
# define LM75A_ID 0xA1
2008-08-11 00:56:16 +04:00
/* Return 0 if detection is successful, -ENODEV otherwise */
2009-12-14 23:17:23 +03:00
static int lm75_detect ( struct i2c_client * new_client ,
2008-08-11 00:56:16 +04:00
struct i2c_board_info * info )
2005-04-17 02:20:36 +04:00
{
2008-08-11 00:56:16 +04:00
struct i2c_adapter * adapter = new_client - > adapter ;
2005-04-17 02:20:36 +04:00
int i ;
2011-03-21 19:59:36 +03:00
int conf , hyst , os ;
2011-03-21 19:59:36 +03:00
bool is_lm75a = 0 ;
2005-04-17 02:20:36 +04:00
if ( ! i2c_check_functionality ( adapter , I2C_FUNC_SMBUS_BYTE_DATA |
I2C_FUNC_SMBUS_WORD_DATA ) )
2008-08-11 00:56:16 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2011-10-14 01: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-17 02:20:36 +04:00
2011-03-21 19:59:36 +03:00
/* Unused bits */
2009-12-09 22:35:57 +03:00
conf = i2c_smbus_read_byte_data ( new_client , 1 ) ;
2011-03-21 19:59:36 +03:00
if ( conf & 0xe0 )
return - ENODEV ;
2011-03-21 19:59:36 +03: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 19:59:36 +03:00
hyst = i2c_smbus_read_byte_data ( new_client , 2 ) ;
os = i2c_smbus_read_byte_data ( new_client , 3 ) ;
2011-03-21 19:59:36 +03:00
} else { /* Traditional style LM75 detection */
/* Unused addresses */
2011-03-21 19:59:36 +03: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 19:59:36 +03:00
return - ENODEV ;
2011-03-21 19:59:36 +03: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 19:59:36 +03:00
return - ENODEV ;
}
2014-12-04 20:58:15 +03:00
/*
* It is very unlikely that this is a LM75 if both
* hysteresis and temperature limit registers are 0.
*/
if ( hyst = = 0 & & os = = 0 )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2009-12-09 22:35:57 +03:00
/* Addresses cycling */
2011-03-21 19:59:36 +03:00
for ( i = 8 ; i < = 248 ; i + = 40 ) {
2009-12-09 22:35:57 +03:00
if ( i2c_smbus_read_byte_data ( new_client , i + 1 ) ! = conf
2011-03-21 19:59:36 +03:00
| | i2c_smbus_read_byte_data ( new_client , i + 2 ) ! = hyst
| | i2c_smbus_read_byte_data ( new_client , i + 3 ) ! = os )
2009-12-09 22:35:57 +03:00
return - ENODEV ;
2011-03-21 19:59:36 +03:00
if ( is_lm75a & & i2c_smbus_read_byte_data ( new_client , i + 7 )
! = LM75A_ID )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
2011-03-21 19:59:36 +03:00
strlcpy ( info - > type , is_lm75a ? " lm75a " : " lm75 " , I2C_NAME_SIZE ) ;
2006-09-24 22:59:49 +04:00
2005-04-17 02:20:36 +04:00
return 0 ;
2008-04-21 23:10:53 +04:00
}
2010-08-14 23:08:50 +04:00
# ifdef CONFIG_PM
static int lm75_suspend ( struct device * dev )
{
int status ;
struct i2c_client * client = to_i2c_client ( dev ) ;
2016-06-20 03:11:13 +03:00
status = i2c_smbus_read_byte_data ( client , LM75_REG_CONF ) ;
2010-08-14 23:08:50 +04:00
if ( status < 0 ) {
dev_dbg ( & client - > dev , " Can't read config? %d \n " , status ) ;
return status ;
}
status = status | LM75_SHUTDOWN ;
2016-06-20 03:11:13 +03:00
i2c_smbus_write_byte_data ( client , LM75_REG_CONF , status ) ;
2010-08-14 23:08:50 +04:00
return 0 ;
}
static int lm75_resume ( struct device * dev )
{
int status ;
struct i2c_client * client = to_i2c_client ( dev ) ;
2016-06-20 03:11:13 +03:00
status = i2c_smbus_read_byte_data ( client , LM75_REG_CONF ) ;
2010-08-14 23:08:50 +04:00
if ( status < 0 ) {
dev_dbg ( & client - > dev , " Can't read config? %d \n " , status ) ;
return status ;
}
status = status & ~ LM75_SHUTDOWN ;
2016-06-20 03:11:13 +03:00
i2c_smbus_write_byte_data ( client , LM75_REG_CONF , status ) ;
2010-08-14 23:08:50 +04:00
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-11 00:56:16 +04:00
static struct i2c_driver lm75_driver = {
. class = I2C_CLASS_HWMON ,
2008-04-21 23:10:53 +04:00
. driver = {
2008-08-11 00:56:16 +04:00
. name = " lm75 " ,
2017-02-24 16:13:02 +03:00
. of_match_table = of_match_ptr ( lm75_of_match ) ,
2010-08-14 23:08:50 +04:00
. pm = LM75_DEV_PM_OPS ,
2008-04-21 23:10:53 +04:00
} ,
2008-08-11 00:56:16 +04:00
. probe = lm75_probe ,
. id_table = lm75_ids ,
. detect = lm75_detect ,
2009-12-14 23:17:25 +03:00
. address_list = normal_i2c ,
2008-04-21 23:10:53 +04:00
} ;
2012-01-20 11:38:18 +04:00
module_i2c_driver ( lm75_driver ) ;
2005-04-17 02:20:36 +04:00
MODULE_AUTHOR ( " Frodo Looijaard <frodol@dds.nl> " ) ;
MODULE_DESCRIPTION ( " LM75 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;