2019-05-20 09:19:02 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2013-03-12 11:38:46 +01:00
/*
* adt7x10 . c - Part of lm_sensors , Linux kernel modules for hardware
* monitoring
* This driver handles the ADT7410 and compatible digital temperature sensors .
* Hartmut Knaack < knaack . h @ gmx . de > 2012 - 07 - 22
* based on lm75 . c by Frodo Looijaard < frodol @ dds . nl >
* and adt7410 . c from iio - staging by Sonic Zhang < sonic . zhang @ analog . com >
*/
2021-12-21 23:58:37 +02:00
# include <linux/device.h>
2013-03-12 11:38:46 +01:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/jiffies.h>
# include <linux/hwmon.h>
# include <linux/err.h>
# include <linux/mutex.h>
# include <linux/delay.h>
2013-03-12 11:38:47 +01:00
# include <linux/interrupt.h>
2021-12-23 10:32:05 -08:00
# include <linux/regmap.h>
2013-03-12 11:38:46 +01:00
# include "adt7x10.h"
/*
* ADT7X10 status
*/
# define ADT7X10_STAT_T_LOW (1 << 4)
# define ADT7X10_STAT_T_HIGH (1 << 5)
# define ADT7X10_STAT_T_CRIT (1 << 6)
# define ADT7X10_STAT_NOT_RDY (1 << 7)
/*
* ADT7X10 config
*/
# define ADT7X10_FAULT_QUEUE_MASK (1 << 0 | 1 << 1)
# define ADT7X10_CT_POLARITY (1 << 2)
# define ADT7X10_INT_POLARITY (1 << 3)
# define ADT7X10_EVENT_MODE (1 << 4)
# define ADT7X10_MODE_MASK (1 << 5 | 1 << 6)
# define ADT7X10_FULL (0 << 5 | 0 << 6)
# define ADT7X10_PD (1 << 5 | 1 << 6)
# define ADT7X10_RESOLUTION (1 << 7)
/*
* ADT7X10 masks
*/
# define ADT7X10_T13_VALUE_MASK 0xFFF8
# define ADT7X10_T_HYST_MASK 0xF
/* straight from the datasheet */
# define ADT7X10_TEMP_MIN (-55000)
# define ADT7X10_TEMP_MAX 150000
/* Each client has this additional data */
struct adt7x10_data {
2021-12-23 10:32:05 -08:00
struct regmap * regmap ;
2013-03-12 11:38:46 +01:00
struct mutex update_lock ;
u8 config ;
u8 oldconfig ;
2021-12-23 10:32:05 -08:00
bool valid ; /* true if temperature valid */
2013-03-12 11:38:46 +01:00
} ;
2021-12-21 23:58:37 +02:00
enum {
adt7x10_temperature = 0 ,
adt7x10_t_alarm_high ,
adt7x10_t_alarm_low ,
adt7x10_t_crit ,
} ;
static const u8 ADT7X10_REG_TEMP [ ] = {
[ adt7x10_temperature ] = ADT7X10_TEMPERATURE , /* input */
[ adt7x10_t_alarm_high ] = ADT7X10_T_ALARM_HIGH , /* high */
[ adt7x10_t_alarm_low ] = ADT7X10_T_ALARM_LOW , /* low */
[ adt7x10_t_crit ] = ADT7X10_T_CRIT , /* critical */
2013-03-12 11:38:46 +01:00
} ;
2013-03-12 11:38:47 +01:00
static irqreturn_t adt7x10_irq_handler ( int irq , void * private )
{
struct device * dev = private ;
2021-12-23 10:32:05 -08:00
struct adt7x10_data * d = dev_get_drvdata ( dev ) ;
unsigned int status ;
int ret ;
2013-03-12 11:38:47 +01:00
2021-12-23 10:32:05 -08:00
ret = regmap_read ( d - > regmap , ADT7X10_STATUS , & status ) ;
if ( ret < 0 )
2013-03-12 11:38:47 +01:00
return IRQ_HANDLED ;
if ( status & ADT7X10_STAT_T_HIGH )
2021-12-21 23:58:41 +02:00
hwmon_notify_event ( dev , hwmon_temp , hwmon_temp_max_alarm , 0 ) ;
2013-03-12 11:38:47 +01:00
if ( status & ADT7X10_STAT_T_LOW )
2021-12-21 23:58:41 +02:00
hwmon_notify_event ( dev , hwmon_temp , hwmon_temp_min_alarm , 0 ) ;
2013-03-12 11:38:47 +01:00
if ( status & ADT7X10_STAT_T_CRIT )
2021-12-21 23:58:41 +02:00
hwmon_notify_event ( dev , hwmon_temp , hwmon_temp_crit_alarm , 0 ) ;
2013-03-12 11:38:47 +01:00
return IRQ_HANDLED ;
}
2021-12-23 10:32:05 -08:00
static int adt7x10_temp_ready ( struct regmap * regmap )
2013-03-12 11:38:46 +01:00
{
2021-12-23 10:32:05 -08:00
unsigned int status ;
int i , ret ;
2013-03-12 11:38:46 +01:00
for ( i = 0 ; i < 6 ; i + + ) {
2021-12-23 10:32:05 -08:00
ret = regmap_read ( regmap , ADT7X10_STATUS , & status ) ;
if ( ret < 0 )
return ret ;
2013-03-12 11:38:46 +01:00
if ( ! ( status & ADT7X10_STAT_NOT_RDY ) )
return 0 ;
msleep ( 60 ) ;
}
return - ETIMEDOUT ;
}
static s16 ADT7X10_TEMP_TO_REG ( long temp )
{
return DIV_ROUND_CLOSEST ( clamp_val ( temp , ADT7X10_TEMP_MIN ,
2021-12-23 10:32:05 -08:00
ADT7X10_TEMP_MAX ) * 128 , 1000 ) ;
2013-03-12 11:38:46 +01:00
}
static int ADT7X10_REG_TO_TEMP ( struct adt7x10_data * data , s16 reg )
{
/* in 13 bit mode, bits 0-2 are status flags - mask them out */
if ( ! ( data - > config & ADT7X10_RESOLUTION ) )
reg & = ADT7X10_T13_VALUE_MASK ;
/*
* temperature is stored in twos complement format , in steps of
* 1 / 128 ° C
*/
return DIV_ROUND_CLOSEST ( reg * 1000 , 128 ) ;
}
/*-----------------------------------------------------------------------*/
2021-12-21 23:58:37 +02:00
static int adt7x10_temp_read ( struct adt7x10_data * data , int index , long * val )
2013-03-12 11:38:46 +01:00
{
2021-12-21 23:58:37 +02:00
unsigned int regval ;
2021-12-23 10:32:05 -08:00
int ret ;
2013-03-12 11:38:46 +01:00
2021-12-23 10:32:05 -08:00
mutex_lock ( & data - > update_lock ) ;
2021-12-21 23:58:37 +02:00
if ( index = = adt7x10_temperature & & ! data - > valid ) {
2021-12-23 10:32:05 -08:00
/* wait for valid temperature */
ret = adt7x10_temp_ready ( data - > regmap ) ;
if ( ret ) {
mutex_unlock ( & data - > update_lock ) ;
2013-03-12 11:38:46 +01:00
return ret ;
2021-12-23 10:32:05 -08:00
}
data - > valid = true ;
2013-03-12 11:38:46 +01:00
}
2021-12-23 10:32:05 -08:00
mutex_unlock ( & data - > update_lock ) ;
2013-03-12 11:38:46 +01:00
2021-12-21 23:58:37 +02:00
ret = regmap_read ( data - > regmap , ADT7X10_REG_TEMP [ index ] , & regval ) ;
2021-12-23 10:32:05 -08:00
if ( ret )
return ret ;
2021-12-21 23:58:37 +02:00
* val = ADT7X10_REG_TO_TEMP ( data , regval ) ;
return 0 ;
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static int adt7x10_temp_write ( struct adt7x10_data * data , int index , long temp )
2013-03-12 11:38:46 +01:00
{
int ret ;
mutex_lock ( & data - > update_lock ) ;
2021-12-21 23:58:37 +02:00
ret = regmap_write ( data - > regmap , ADT7X10_REG_TEMP [ index ] ,
2021-12-23 10:32:05 -08:00
ADT7X10_TEMP_TO_REG ( temp ) ) ;
2013-03-12 11:38:46 +01:00
mutex_unlock ( & data - > update_lock ) ;
2021-12-21 23:58:37 +02:00
return ret ;
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static int adt7x10_hyst_read ( struct adt7x10_data * data , int index , long * val )
2013-03-12 11:38:46 +01:00
{
2021-12-23 10:32:05 -08:00
int hyst , temp , ret ;
mutex_lock ( & data - > update_lock ) ;
ret = regmap_read ( data - > regmap , ADT7X10_T_HYST , & hyst ) ;
if ( ret ) {
mutex_unlock ( & data - > update_lock ) ;
return ret ;
}
2021-12-21 23:58:37 +02:00
ret = regmap_read ( data - > regmap , ADT7X10_REG_TEMP [ index ] , & temp ) ;
2021-12-23 10:32:05 -08:00
mutex_unlock ( & data - > update_lock ) ;
if ( ret )
return ret ;
2013-03-12 11:38:46 +01:00
2021-12-23 10:32:05 -08:00
hyst = ( hyst & ADT7X10_T_HYST_MASK ) * 1000 ;
2013-03-12 11:38:46 +01:00
/*
* hysteresis is stored as a 4 bit offset in the device , convert it
* to an absolute value
*/
2021-12-21 23:58:37 +02:00
/* min has positive offset, others have negative */
if ( index = = adt7x10_t_alarm_low )
2013-03-12 11:38:46 +01:00
hyst = - hyst ;
2021-12-23 10:32:05 -08:00
2021-12-21 23:58:37 +02:00
* val = ADT7X10_REG_TO_TEMP ( data , temp ) - hyst ;
return 0 ;
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static int adt7x10_hyst_write ( struct adt7x10_data * data , long hyst )
2013-03-12 11:38:46 +01:00
{
2021-12-23 10:32:05 -08:00
unsigned int regval ;
2013-03-12 11:38:46 +01:00
int limit , ret ;
2021-12-23 10:32:05 -08:00
mutex_lock ( & data - > update_lock ) ;
2013-03-12 11:38:46 +01:00
/* convert absolute hysteresis value to a 4 bit delta value */
2021-12-23 10:32:05 -08:00
ret = regmap_read ( data - > regmap , ADT7X10_T_ALARM_HIGH , & regval ) ;
if ( ret < 0 )
goto abort ;
2013-03-12 11:38:46 +01:00
2021-12-23 10:32:05 -08:00
limit = ADT7X10_REG_TO_TEMP ( data , regval ) ;
hyst = clamp_val ( hyst , ADT7X10_TEMP_MIN , ADT7X10_TEMP_MAX ) ;
regval = clamp_val ( DIV_ROUND_CLOSEST ( limit - hyst , 1000 ) , 0 ,
ADT7X10_T_HYST_MASK ) ;
ret = regmap_write ( data - > regmap , ADT7X10_T_HYST , regval ) ;
abort :
mutex_unlock ( & data - > update_lock ) ;
2021-12-21 23:58:37 +02:00
return ret ;
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static int adt7x10_alarm_read ( struct adt7x10_data * data , int index , long * val )
2013-03-12 11:38:46 +01:00
{
2021-12-23 10:32:05 -08:00
unsigned int status ;
2013-03-12 11:38:46 +01:00
int ret ;
2021-12-23 10:32:05 -08:00
ret = regmap_read ( data - > regmap , ADT7X10_STATUS , & status ) ;
2013-03-12 11:38:46 +01:00
if ( ret < 0 )
return ret ;
2021-12-21 23:58:37 +02:00
* val = ! ! ( status & index ) ;
return 0 ;
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static umode_t adt7x10_is_visible ( const void * data ,
enum hwmon_sensor_types type ,
u32 attr , int channel )
{
switch ( attr ) {
case hwmon_temp_max :
case hwmon_temp_min :
case hwmon_temp_crit :
case hwmon_temp_max_hyst :
return 0644 ;
case hwmon_temp_input :
case hwmon_temp_min_alarm :
case hwmon_temp_max_alarm :
case hwmon_temp_crit_alarm :
case hwmon_temp_min_hyst :
case hwmon_temp_crit_hyst :
return 0444 ;
default :
break ;
}
return 0 ;
}
static int adt7x10_read ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
2013-03-12 11:38:46 +01:00
{
struct adt7x10_data * data = dev_get_drvdata ( dev ) ;
2021-12-21 23:58:37 +02:00
switch ( attr ) {
case hwmon_temp_input :
return adt7x10_temp_read ( data , adt7x10_temperature , val ) ;
case hwmon_temp_max :
return adt7x10_temp_read ( data , adt7x10_t_alarm_high , val ) ;
case hwmon_temp_min :
return adt7x10_temp_read ( data , adt7x10_t_alarm_low , val ) ;
case hwmon_temp_crit :
return adt7x10_temp_read ( data , adt7x10_t_crit , val ) ;
case hwmon_temp_max_hyst :
return adt7x10_hyst_read ( data , adt7x10_t_alarm_high , val ) ;
case hwmon_temp_min_hyst :
return adt7x10_hyst_read ( data , adt7x10_t_alarm_low , val ) ;
case hwmon_temp_crit_hyst :
return adt7x10_hyst_read ( data , adt7x10_t_crit , val ) ;
case hwmon_temp_min_alarm :
return adt7x10_alarm_read ( data , ADT7X10_STAT_T_LOW , val ) ;
case hwmon_temp_max_alarm :
return adt7x10_alarm_read ( data , ADT7X10_STAT_T_HIGH , val ) ;
case hwmon_temp_crit_alarm :
return adt7x10_alarm_read ( data , ADT7X10_STAT_T_CRIT , val ) ;
default :
return - EOPNOTSUPP ;
}
2013-03-12 11:38:46 +01:00
}
2021-12-21 23:58:37 +02:00
static int adt7x10_write ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long val )
{
struct adt7x10_data * data = dev_get_drvdata ( dev ) ;
switch ( attr ) {
case hwmon_temp_max :
return adt7x10_temp_write ( data , adt7x10_t_alarm_high , val ) ;
case hwmon_temp_min :
return adt7x10_temp_write ( data , adt7x10_t_alarm_low , val ) ;
case hwmon_temp_crit :
return adt7x10_temp_write ( data , adt7x10_t_crit , val ) ;
case hwmon_temp_max_hyst :
return adt7x10_hyst_write ( data , val ) ;
default :
return - EOPNOTSUPP ;
}
}
static const struct hwmon_channel_info * adt7x10_info [ ] = {
HWMON_CHANNEL_INFO ( temp , HWMON_T_INPUT | HWMON_T_MAX | HWMON_T_MIN |
HWMON_T_CRIT | HWMON_T_MAX_HYST | HWMON_T_MIN_HYST |
HWMON_T_CRIT_HYST | HWMON_T_MIN_ALARM |
HWMON_T_MAX_ALARM | HWMON_T_CRIT_ALARM ) ,
NULL ,
} ;
static const struct hwmon_ops adt7x10_hwmon_ops = {
. is_visible = adt7x10_is_visible ,
. read = adt7x10_read ,
. write = adt7x10_write ,
2013-03-12 11:38:46 +01:00
} ;
2021-12-21 23:58:37 +02:00
static const struct hwmon_chip_info adt7x10_chip_info = {
. ops = & adt7x10_hwmon_ops ,
. info = adt7x10_info ,
2013-03-12 11:38:46 +01:00
} ;
2021-12-21 23:58:36 +02:00
static void adt7x10_restore_config ( void * private )
{
struct adt7x10_data * data = private ;
regmap_write ( data - > regmap , ADT7X10_CONFIG , data - > oldconfig ) ;
}
2013-03-12 11:38:47 +01:00
int adt7x10_probe ( struct device * dev , const char * name , int irq ,
2021-12-23 10:32:05 -08:00
struct regmap * regmap )
2013-03-12 11:38:46 +01:00
{
struct adt7x10_data * data ;
2021-12-23 10:32:05 -08:00
unsigned int config ;
2021-12-21 23:58:37 +02:00
struct device * hdev ;
2013-03-12 11:38:46 +01:00
int ret ;
data = devm_kzalloc ( dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2021-12-23 10:32:05 -08:00
data - > regmap = regmap ;
2013-03-12 11:38:46 +01:00
dev_set_drvdata ( dev , data ) ;
mutex_init ( & data - > update_lock ) ;
/* configure as specified */
2021-12-23 10:32:05 -08:00
ret = regmap_read ( regmap , ADT7X10_CONFIG , & config ) ;
2013-03-12 11:38:46 +01:00
if ( ret < 0 ) {
dev_dbg ( dev , " Can't read config? %d \n " , ret ) ;
return ret ;
}
2021-12-23 10:32:05 -08:00
data - > oldconfig = config ;
2013-03-12 11:38:46 +01:00
/*
* Set to 16 bit resolution , continous conversion and comparator mode .
*/
data - > config = data - > oldconfig ;
2013-03-12 11:38:47 +01:00
data - > config & = ~ ( ADT7X10_MODE_MASK | ADT7X10_CT_POLARITY |
ADT7X10_INT_POLARITY ) ;
2013-03-12 11:38:46 +01:00
data - > config | = ADT7X10_FULL | ADT7X10_RESOLUTION | ADT7X10_EVENT_MODE ;
2013-03-12 11:38:47 +01:00
2013-03-12 11:38:46 +01:00
if ( data - > config ! = data - > oldconfig ) {
2021-12-23 10:32:05 -08:00
ret = regmap_write ( regmap , ADT7X10_CONFIG , data - > config ) ;
2013-03-12 11:38:46 +01:00
if ( ret )
return ret ;
2021-12-21 23:58:36 +02:00
ret = devm_add_action_or_reset ( dev , adt7x10_restore_config , data ) ;
if ( ret )
return ret ;
2013-03-12 11:38:46 +01:00
}
dev_dbg ( dev , " Config %02x \n " , data - > config ) ;
2021-12-21 23:58:37 +02:00
hdev = devm_hwmon_device_register_with_info ( dev , name , data ,
& adt7x10_chip_info , NULL ) ;
if ( IS_ERR ( hdev ) )
return PTR_ERR ( hdev ) ;
2013-03-12 11:38:46 +01:00
2013-03-12 11:38:47 +01:00
if ( irq > 0 ) {
2021-12-21 23:58:38 +02:00
ret = devm_request_threaded_irq ( dev , irq , NULL ,
adt7x10_irq_handler ,
IRQF_TRIGGER_FALLING |
IRQF_ONESHOT ,
2021-12-21 23:58:41 +02:00
dev_name ( dev ) , hdev ) ;
2013-03-12 11:38:47 +01:00
if ( ret )
2021-12-21 23:58:37 +02:00
return ret ;
2013-03-12 11:38:47 +01:00
}
2013-03-12 11:38:46 +01:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( adt7x10_probe ) ;
static int adt7x10_suspend ( struct device * dev )
{
struct adt7x10_data * data = dev_get_drvdata ( dev ) ;
2021-12-23 10:32:05 -08:00
return regmap_write ( data - > regmap , ADT7X10_CONFIG ,
data - > config | ADT7X10_PD ) ;
2013-03-12 11:38:46 +01:00
}
static int adt7x10_resume ( struct device * dev )
{
struct adt7x10_data * data = dev_get_drvdata ( dev ) ;
2021-12-23 10:32:05 -08:00
return regmap_write ( data - > regmap , ADT7X10_CONFIG , data - > config ) ;
2013-03-12 11:38:46 +01:00
}
2022-09-25 18:27:45 +01:00
EXPORT_SIMPLE_DEV_PM_OPS ( adt7x10_dev_pm_ops , adt7x10_suspend , adt7x10_resume ) ;
2013-03-12 11:38:46 +01:00
MODULE_AUTHOR ( " Hartmut Knaack " ) ;
MODULE_DESCRIPTION ( " ADT7410/ADT7420, ADT7310/ADT7320 common code " ) ;
MODULE_LICENSE ( " GPL " ) ;