2018-05-23 22:24:40 +03:00
// SPDX-License-Identifier: GPL-2.0
//
// Copyright 2013 Freescale Semiconductor, Inc.
2013-06-24 10:30:44 +04:00
2013-12-24 00:49:22 +04:00
# include <linux/clk.h>
2017-04-25 13:27:14 +03:00
# include <linux/cpufreq.h>
2013-06-24 10:30:44 +04:00
# include <linux/cpu_cooling.h>
# include <linux/delay.h>
2013-08-01 20:33:12 +04:00
# include <linux/interrupt.h>
2013-06-24 10:30:44 +04:00
# include <linux/io.h>
# include <linux/mfd/syscon.h>
# include <linux/module.h>
# include <linux/of.h>
2014-08-06 11:12:09 +04:00
# include <linux/of_device.h>
2013-06-24 10:30:44 +04:00
# include <linux/regmap.h>
# include <linux/thermal.h>
2017-07-14 17:11:08 +03:00
# include <linux/nvmem-consumer.h>
2021-11-17 13:34:26 +03:00
# include <linux/pm_runtime.h>
2013-06-24 10:30:44 +04:00
# define REG_SET 0x4
# define REG_CLR 0x8
# define REG_TOG 0xc
2018-03-02 04:59:30 +03:00
/* i.MX6 specific */
# define IMX6_MISC0 0x0150
# define IMX6_MISC0_REFTOP_SELBIASOFF (1 << 3)
# define IMX6_MISC1 0x0160
# define IMX6_MISC1_IRQ_TEMPHIGH (1 << 29)
2014-08-06 11:12:09 +04:00
/* Below LOW and PANIC bits are only for TEMPMON_IMX6SX */
2018-03-02 04:59:30 +03:00
# define IMX6_MISC1_IRQ_TEMPLOW (1 << 28)
# define IMX6_MISC1_IRQ_TEMPPANIC (1 << 27)
# define IMX6_TEMPSENSE0 0x0180
# define IMX6_TEMPSENSE0_ALARM_VALUE_SHIFT 20
# define IMX6_TEMPSENSE0_ALARM_VALUE_MASK (0xfff << 20)
# define IMX6_TEMPSENSE0_TEMP_CNT_SHIFT 8
# define IMX6_TEMPSENSE0_TEMP_CNT_MASK (0xfff << 8)
# define IMX6_TEMPSENSE0_FINISHED (1 << 2)
# define IMX6_TEMPSENSE0_MEASURE_TEMP (1 << 1)
# define IMX6_TEMPSENSE0_POWER_DOWN (1 << 0)
# define IMX6_TEMPSENSE1 0x0190
# define IMX6_TEMPSENSE1_MEASURE_FREQ 0xffff
# define IMX6_TEMPSENSE1_MEASURE_FREQ_SHIFT 0
2013-06-24 10:30:44 +04:00
2015-11-19 17:49:40 +03:00
# define OCOTP_MEM0 0x0480
2013-06-24 10:30:44 +04:00
# define OCOTP_ANA1 0x04e0
2018-03-02 04:59:30 +03:00
/* Below TEMPSENSE2 is only for TEMPMON_IMX6SX */
# define IMX6_TEMPSENSE2 0x0290
# define IMX6_TEMPSENSE2_LOW_VALUE_SHIFT 0
# define IMX6_TEMPSENSE2_LOW_VALUE_MASK 0xfff
# define IMX6_TEMPSENSE2_PANIC_VALUE_SHIFT 16
# define IMX6_TEMPSENSE2_PANIC_VALUE_MASK 0xfff0000
/* i.MX7 specific */
# define IMX7_ANADIG_DIGPROG 0x800
# define IMX7_TEMPSENSE0 0x300
# define IMX7_TEMPSENSE0_PANIC_ALARM_SHIFT 18
# define IMX7_TEMPSENSE0_PANIC_ALARM_MASK (0x1ff << 18)
# define IMX7_TEMPSENSE0_HIGH_ALARM_SHIFT 9
# define IMX7_TEMPSENSE0_HIGH_ALARM_MASK (0x1ff << 9)
# define IMX7_TEMPSENSE0_LOW_ALARM_SHIFT 0
# define IMX7_TEMPSENSE0_LOW_ALARM_MASK 0x1ff
# define IMX7_TEMPSENSE1 0x310
# define IMX7_TEMPSENSE1_MEASURE_FREQ_SHIFT 16
# define IMX7_TEMPSENSE1_MEASURE_FREQ_MASK (0xffff << 16)
# define IMX7_TEMPSENSE1_FINISHED (1 << 11)
# define IMX7_TEMPSENSE1_MEASURE_TEMP (1 << 10)
# define IMX7_TEMPSENSE1_POWER_DOWN (1 << 9)
# define IMX7_TEMPSENSE1_TEMP_VALUE_SHIFT 0
# define IMX7_TEMPSENSE1_TEMP_VALUE_MASK 0x1ff
2013-06-24 10:30:44 +04:00
/* The driver supports 1 passive trip point and 1 critical trip point */
enum imx_thermal_trip {
IMX_TRIP_PASSIVE ,
IMX_TRIP_CRITICAL ,
} ;
# define IMX_POLLING_DELAY 2000 /* millisecond */
# define IMX_PASSIVE_DELAY 1000
2014-08-06 11:12:09 +04:00
# define TEMPMON_IMX6Q 1
# define TEMPMON_IMX6SX 2
2018-03-02 04:59:30 +03:00
# define TEMPMON_IMX7D 3
2014-08-06 11:12:09 +04:00
struct thermal_soc_data {
u32 version ;
2018-03-02 04:59:30 +03:00
u32 sensor_ctrl ;
u32 power_down_mask ;
u32 measure_temp_mask ;
u32 measure_freq_ctrl ;
u32 measure_freq_mask ;
u32 measure_freq_shift ;
u32 temp_data ;
u32 temp_value_mask ;
u32 temp_value_shift ;
u32 temp_valid_mask ;
u32 panic_alarm_ctrl ;
u32 panic_alarm_mask ;
u32 panic_alarm_shift ;
u32 high_alarm_ctrl ;
u32 high_alarm_mask ;
u32 high_alarm_shift ;
u32 low_alarm_ctrl ;
u32 low_alarm_mask ;
u32 low_alarm_shift ;
2014-08-06 11:12:09 +04:00
} ;
2022-10-03 12:25:54 +03:00
static struct thermal_trip trips [ ] = {
[ IMX_TRIP_PASSIVE ] = { . type = THERMAL_TRIP_PASSIVE } ,
[ IMX_TRIP_CRITICAL ] = { . type = THERMAL_TRIP_CRITICAL } ,
} ;
2014-08-06 11:12:09 +04:00
static struct thermal_soc_data thermal_imx6q_data = {
. version = TEMPMON_IMX6Q ,
2018-03-02 04:59:30 +03:00
. sensor_ctrl = IMX6_TEMPSENSE0 ,
. power_down_mask = IMX6_TEMPSENSE0_POWER_DOWN ,
. measure_temp_mask = IMX6_TEMPSENSE0_MEASURE_TEMP ,
. measure_freq_ctrl = IMX6_TEMPSENSE1 ,
. measure_freq_shift = IMX6_TEMPSENSE1_MEASURE_FREQ_SHIFT ,
. measure_freq_mask = IMX6_TEMPSENSE1_MEASURE_FREQ ,
. temp_data = IMX6_TEMPSENSE0 ,
. temp_value_mask = IMX6_TEMPSENSE0_TEMP_CNT_MASK ,
. temp_value_shift = IMX6_TEMPSENSE0_TEMP_CNT_SHIFT ,
. temp_valid_mask = IMX6_TEMPSENSE0_FINISHED ,
. high_alarm_ctrl = IMX6_TEMPSENSE0 ,
. high_alarm_mask = IMX6_TEMPSENSE0_ALARM_VALUE_MASK ,
. high_alarm_shift = IMX6_TEMPSENSE0_ALARM_VALUE_SHIFT ,
2014-08-06 11:12:09 +04:00
} ;
static struct thermal_soc_data thermal_imx6sx_data = {
. version = TEMPMON_IMX6SX ,
2018-03-02 04:59:30 +03:00
. sensor_ctrl = IMX6_TEMPSENSE0 ,
. power_down_mask = IMX6_TEMPSENSE0_POWER_DOWN ,
. measure_temp_mask = IMX6_TEMPSENSE0_MEASURE_TEMP ,
. measure_freq_ctrl = IMX6_TEMPSENSE1 ,
. measure_freq_shift = IMX6_TEMPSENSE1_MEASURE_FREQ_SHIFT ,
. measure_freq_mask = IMX6_TEMPSENSE1_MEASURE_FREQ ,
. temp_data = IMX6_TEMPSENSE0 ,
. temp_value_mask = IMX6_TEMPSENSE0_TEMP_CNT_MASK ,
. temp_value_shift = IMX6_TEMPSENSE0_TEMP_CNT_SHIFT ,
. temp_valid_mask = IMX6_TEMPSENSE0_FINISHED ,
. high_alarm_ctrl = IMX6_TEMPSENSE0 ,
. high_alarm_mask = IMX6_TEMPSENSE0_ALARM_VALUE_MASK ,
. high_alarm_shift = IMX6_TEMPSENSE0_ALARM_VALUE_SHIFT ,
. panic_alarm_ctrl = IMX6_TEMPSENSE2 ,
. panic_alarm_mask = IMX6_TEMPSENSE2_PANIC_VALUE_MASK ,
. panic_alarm_shift = IMX6_TEMPSENSE2_PANIC_VALUE_SHIFT ,
. low_alarm_ctrl = IMX6_TEMPSENSE2 ,
. low_alarm_mask = IMX6_TEMPSENSE2_LOW_VALUE_MASK ,
. low_alarm_shift = IMX6_TEMPSENSE2_LOW_VALUE_SHIFT ,
} ;
static struct thermal_soc_data thermal_imx7d_data = {
. version = TEMPMON_IMX7D ,
. sensor_ctrl = IMX7_TEMPSENSE1 ,
. power_down_mask = IMX7_TEMPSENSE1_POWER_DOWN ,
. measure_temp_mask = IMX7_TEMPSENSE1_MEASURE_TEMP ,
. measure_freq_ctrl = IMX7_TEMPSENSE1 ,
. measure_freq_shift = IMX7_TEMPSENSE1_MEASURE_FREQ_SHIFT ,
. measure_freq_mask = IMX7_TEMPSENSE1_MEASURE_FREQ_MASK ,
. temp_data = IMX7_TEMPSENSE1 ,
. temp_value_mask = IMX7_TEMPSENSE1_TEMP_VALUE_MASK ,
. temp_value_shift = IMX7_TEMPSENSE1_TEMP_VALUE_SHIFT ,
. temp_valid_mask = IMX7_TEMPSENSE1_FINISHED ,
. panic_alarm_ctrl = IMX7_TEMPSENSE1 ,
. panic_alarm_mask = IMX7_TEMPSENSE0_PANIC_ALARM_MASK ,
. panic_alarm_shift = IMX7_TEMPSENSE0_PANIC_ALARM_SHIFT ,
. high_alarm_ctrl = IMX7_TEMPSENSE0 ,
. high_alarm_mask = IMX7_TEMPSENSE0_HIGH_ALARM_MASK ,
. high_alarm_shift = IMX7_TEMPSENSE0_HIGH_ALARM_SHIFT ,
. low_alarm_ctrl = IMX7_TEMPSENSE0 ,
. low_alarm_mask = IMX7_TEMPSENSE0_LOW_ALARM_MASK ,
. low_alarm_shift = IMX7_TEMPSENSE0_LOW_ALARM_SHIFT ,
2014-08-06 11:12:09 +04:00
} ;
2013-06-24 10:30:44 +04:00
struct imx_thermal_data {
2021-11-17 13:34:26 +03:00
struct device * dev ;
2017-04-25 13:27:14 +03:00
struct cpufreq_policy * policy ;
2013-06-24 10:30:44 +04:00
struct thermal_zone_device * tz ;
struct thermal_cooling_device * cdev ;
struct regmap * tempmon ;
2017-07-14 17:11:08 +03:00
u32 c1 , c2 ; /* See formula in imx_init_calib() */
2015-11-19 17:49:40 +03:00
int temp_max ;
2015-07-24 09:12:54 +03:00
int alarm_temp ;
int last_temp ;
2013-08-01 20:33:12 +04:00
bool irq_enabled ;
int irq ;
2013-12-24 00:49:22 +04:00
struct clk * thermal_clk ;
2014-08-06 11:12:09 +04:00
const struct thermal_soc_data * socdata ;
2015-11-19 17:49:40 +03:00
const char * temp_grade ;
2013-06-24 10:30:44 +04:00
} ;
2014-08-06 11:12:09 +04:00
static void imx_set_panic_temp ( struct imx_thermal_data * data ,
2015-07-24 09:12:54 +03:00
int panic_temp )
2014-08-06 11:12:09 +04:00
{
2018-03-02 04:59:30 +03:00
const struct thermal_soc_data * soc_data = data - > socdata ;
2014-08-06 11:12:09 +04:00
struct regmap * map = data - > tempmon ;
int critical_value ;
critical_value = ( data - > c2 - panic_temp ) / data - > c1 ;
2018-03-02 04:59:30 +03:00
regmap_write ( map , soc_data - > panic_alarm_ctrl + REG_CLR ,
soc_data - > panic_alarm_mask ) ;
regmap_write ( map , soc_data - > panic_alarm_ctrl + REG_SET ,
critical_value < < soc_data - > panic_alarm_shift ) ;
2014-08-06 11:12:09 +04:00
}
2013-08-01 20:33:12 +04:00
static void imx_set_alarm_temp ( struct imx_thermal_data * data ,
2015-07-24 09:12:54 +03:00
int alarm_temp )
2013-08-01 20:33:12 +04:00
{
struct regmap * map = data - > tempmon ;
2018-03-02 04:59:30 +03:00
const struct thermal_soc_data * soc_data = data - > socdata ;
2013-08-01 20:33:12 +04:00
int alarm_value ;
data - > alarm_temp = alarm_temp ;
2018-03-02 04:59:30 +03:00
if ( data - > socdata - > version = = TEMPMON_IMX7D )
alarm_value = alarm_temp / 1000 + data - > c1 - 25 ;
else
alarm_value = ( data - > c2 - alarm_temp ) / data - > c1 ;
regmap_write ( map , soc_data - > high_alarm_ctrl + REG_CLR ,
soc_data - > high_alarm_mask ) ;
regmap_write ( map , soc_data - > high_alarm_ctrl + REG_SET ,
alarm_value < < soc_data - > high_alarm_shift ) ;
2013-08-01 20:33:12 +04:00
}
2015-07-24 09:12:54 +03:00
static int imx_get_temp ( struct thermal_zone_device * tz , int * temp )
2013-06-24 10:30:44 +04:00
{
2023-03-01 23:14:30 +03:00
struct imx_thermal_data * data = thermal_zone_device_priv ( tz ) ;
2018-03-02 04:59:30 +03:00
const struct thermal_soc_data * soc_data = data - > socdata ;
2013-06-24 10:30:44 +04:00
struct regmap * map = data - > tempmon ;
unsigned int n_meas ;
u32 val ;
2021-11-17 13:34:26 +03:00
int ret ;
2013-06-24 10:30:44 +04:00
2021-11-17 13:34:26 +03:00
ret = pm_runtime_resume_and_get ( data - > dev ) ;
if ( ret < 0 )
return ret ;
2013-06-24 10:30:44 +04:00
2018-03-02 04:59:30 +03:00
regmap_read ( map , soc_data - > temp_data , & val ) ;
2013-08-01 20:33:12 +04:00
2023-03-01 23:14:34 +03:00
if ( ( val & soc_data - > temp_valid_mask ) = = 0 )
2013-06-24 10:30:44 +04:00
return - EAGAIN ;
2018-03-02 04:59:30 +03:00
n_meas = ( val & soc_data - > temp_value_mask )
> > soc_data - > temp_value_shift ;
2013-06-24 10:30:44 +04:00
2017-07-14 17:11:08 +03:00
/* See imx_init_calib() for formula derivation */
2018-03-02 04:59:30 +03:00
if ( data - > socdata - > version = = TEMPMON_IMX7D )
* temp = ( n_meas - data - > c1 + 25 ) * 1000 ;
else
* temp = data - > c2 - n_meas * data - > c1 ;
2013-06-24 10:30:44 +04:00
2014-08-06 11:12:09 +04:00
/* Update alarm value to next higher trip point for TEMPMON_IMX6Q */
if ( data - > socdata - > version = = TEMPMON_IMX6Q ) {
2022-10-03 12:25:54 +03:00
if ( data - > alarm_temp = = trips [ IMX_TRIP_PASSIVE ] . temperature & &
* temp > = trips [ IMX_TRIP_PASSIVE ] . temperature )
imx_set_alarm_temp ( data , trips [ IMX_TRIP_CRITICAL ] . temperature ) ;
if ( data - > alarm_temp = = trips [ IMX_TRIP_CRITICAL ] . temperature & &
* temp < trips [ IMX_TRIP_PASSIVE ] . temperature ) {
imx_set_alarm_temp ( data , trips [ IMX_TRIP_PASSIVE ] . temperature ) ;
2023-03-01 23:14:37 +03:00
dev_dbg ( data - > dev , " thermal alarm off: T < %d \n " ,
2014-08-06 11:12:09 +04:00
data - > alarm_temp / 1000 ) ;
}
2013-08-01 20:33:12 +04:00
}
if ( * temp ! = data - > last_temp ) {
2023-03-01 23:14:37 +03:00
dev_dbg ( data - > dev , " millicelsius: %d \n " , * temp ) ;
2013-08-01 20:33:12 +04:00
data - > last_temp = * temp ;
}
/* Reenable alarm IRQ if temperature below alarm temperature */
if ( ! data - > irq_enabled & & * temp < data - > alarm_temp ) {
data - > irq_enabled = true ;
enable_irq ( data - > irq ) ;
2013-06-24 10:30:44 +04:00
}
2021-11-17 13:34:26 +03:00
pm_runtime_put ( data - > dev ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
}
2020-06-29 15:29:25 +03:00
static int imx_change_mode ( struct thermal_zone_device * tz ,
enum thermal_device_mode mode )
2013-06-24 10:30:44 +04:00
{
2023-03-01 23:14:30 +03:00
struct imx_thermal_data * data = thermal_zone_device_priv ( tz ) ;
2013-06-24 10:30:44 +04:00
if ( mode = = THERMAL_DEVICE_ENABLED ) {
2021-11-17 13:34:26 +03:00
pm_runtime_get ( data - > dev ) ;
2013-08-01 20:33:12 +04:00
if ( ! data - > irq_enabled ) {
data - > irq_enabled = true ;
enable_irq ( data - > irq ) ;
}
2013-06-24 10:30:44 +04:00
} else {
2021-11-17 13:34:26 +03:00
pm_runtime_put ( data - > dev ) ;
2013-08-01 20:33:12 +04:00
if ( data - > irq_enabled ) {
disable_irq ( data - > irq ) ;
data - > irq_enabled = false ;
}
2013-06-24 10:30:44 +04:00
}
return 0 ;
}
2023-03-09 12:28:21 +03:00
static int imx_set_trip_temp ( struct thermal_zone_device * tz , int trip_id ,
2015-07-24 09:12:54 +03:00
int temp )
2013-08-01 20:33:11 +04:00
{
2023-03-01 23:14:30 +03:00
struct imx_thermal_data * data = thermal_zone_device_priv ( tz ) ;
2023-03-09 12:28:21 +03:00
struct thermal_trip trip ;
2021-11-17 13:34:26 +03:00
int ret ;
ret = pm_runtime_resume_and_get ( data - > dev ) ;
if ( ret < 0 )
return ret ;
2013-08-01 20:33:11 +04:00
2023-03-09 12:28:21 +03:00
ret = __thermal_zone_get_trip ( tz , trip_id , & trip ) ;
if ( ret )
return ret ;
2015-11-19 17:49:40 +03:00
/* do not allow changing critical threshold */
2023-03-09 12:28:21 +03:00
if ( trip . type = = THERMAL_TRIP_CRITICAL )
2013-08-01 20:33:11 +04:00
return - EPERM ;
2015-11-19 17:49:40 +03:00
/* do not allow passive to be set higher than critical */
2022-10-03 12:25:54 +03:00
if ( temp < 0 | | temp > trips [ IMX_TRIP_CRITICAL ] . temperature )
2013-08-01 20:33:11 +04:00
return - EINVAL ;
2013-08-01 20:33:12 +04:00
imx_set_alarm_temp ( data , temp ) ;
2021-11-17 13:34:26 +03:00
pm_runtime_put ( data - > dev ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
}
static int imx_bind ( struct thermal_zone_device * tz ,
struct thermal_cooling_device * cdev )
{
2023-03-01 23:14:37 +03:00
return thermal_zone_bind_cooling_device ( tz , IMX_TRIP_PASSIVE , cdev ,
THERMAL_NO_LIMIT ,
THERMAL_NO_LIMIT ,
THERMAL_WEIGHT_DEFAULT ) ;
2013-06-24 10:30:44 +04:00
}
static int imx_unbind ( struct thermal_zone_device * tz ,
struct thermal_cooling_device * cdev )
{
2023-03-01 23:14:37 +03:00
return thermal_zone_unbind_cooling_device ( tz , IMX_TRIP_PASSIVE , cdev ) ;
2013-06-24 10:30:44 +04:00
}
2014-01-06 16:54:34 +04:00
static struct thermal_zone_device_ops imx_tz_ops = {
2013-06-24 10:30:44 +04:00
. bind = imx_bind ,
. unbind = imx_unbind ,
. get_temp = imx_get_temp ,
2020-06-29 15:29:25 +03:00
. change_mode = imx_change_mode ,
2013-08-01 20:33:11 +04:00
. set_trip_temp = imx_set_trip_temp ,
2013-06-24 10:30:44 +04:00
} ;
2017-11-30 12:17:35 +03:00
static int imx_init_calib ( struct platform_device * pdev , u32 ocotp_ana1 )
2013-06-24 10:30:44 +04:00
{
struct imx_thermal_data * data = platform_get_drvdata ( pdev ) ;
2017-11-30 12:17:38 +03:00
int n1 ;
2014-02-12 14:06:35 +04:00
u64 temp64 ;
2013-06-24 10:30:44 +04:00
2017-11-30 12:17:35 +03:00
if ( ocotp_ana1 = = 0 | | ocotp_ana1 = = ~ 0 ) {
2013-06-24 10:30:44 +04:00
dev_err ( & pdev - > dev , " invalid sensor calibration data \n " ) ;
return - EINVAL ;
}
2018-03-02 04:59:30 +03:00
/*
* On i . MX7D , we only use the calibration data at 25 C to get the temp ,
* Tmeas = ( Nmeas - n1 ) + 25 ; n1 is the fuse value for 25 C .
*/
if ( data - > socdata - > version = = TEMPMON_IMX7D ) {
data - > c1 = ( ocotp_ana1 > > 9 ) & 0x1ff ;
return 0 ;
}
2013-06-24 10:30:44 +04:00
/*
2017-11-30 12:17:36 +03:00
* The sensor is calibrated at 25 ° C ( aka T1 ) and the value measured
* ( aka N1 ) at this temperature is provided in bits [ 31 : 20 ] in the
* i . MX ' s OCOTP value ANA1 .
* To find the actual temperature T , the following formula has to be used
* when reading value n from the sensor :
*
2017-11-30 12:17:38 +03:00
* T = T1 + ( N - N1 ) / ( 0.4148468 - 0.0015423 * N1 ) ° C + 3.580661 ° C
* = [ T1 ' - N1 / ( 0.4148468 - 0.0015423 * N1 ) ° C ] + N / ( 0.4148468 - 0.0015423 * N1 ) ° C
* = [ T1 ' + N1 / ( 0.0015423 * N1 - 0.4148468 ) ° C ] - N / ( 0.0015423 * N1 - 0.4148468 ) ° C
2017-11-30 12:17:36 +03:00
* = c2 - c1 * N
*
* with
*
2017-11-30 12:17:38 +03:00
* T1 ' = 28.580661 ° C
* c1 = 1 / ( 0.0015423 * N1 - 0.4297157 ) ° C
* c2 = T1 ' + N1 / ( 0.0015423 * N1 - 0.4148468 ) ° C
* = T1 ' + N1 * c1
2013-06-24 10:30:44 +04:00
*/
2017-11-30 12:17:35 +03:00
n1 = ocotp_ana1 > > 20 ;
2013-06-24 10:30:44 +04:00
2017-11-30 12:17:38 +03:00
temp64 = 10000000 ; /* use 10^7 as fixed point constant for values in formula */
2017-11-30 12:17:36 +03:00
temp64 * = 1000 ; /* to get result in °mC */
2017-11-30 12:17:38 +03:00
do_div ( temp64 , 15423 * n1 - 4148468 ) ;
2014-02-12 14:06:35 +04:00
data - > c1 = temp64 ;
2017-11-30 12:17:38 +03:00
data - > c2 = n1 * data - > c1 + 28581 ;
2013-06-24 10:30:44 +04:00
2017-07-14 17:11:08 +03:00
return 0 ;
}
2017-11-30 12:17:35 +03:00
static void imx_init_temp_grade ( struct platform_device * pdev , u32 ocotp_mem0 )
2017-07-14 17:11:08 +03:00
{
struct imx_thermal_data * data = platform_get_drvdata ( pdev ) ;
2015-11-19 17:49:40 +03:00
/* The maximum die temp is specified by the Temperature Grade */
2017-11-30 12:17:35 +03:00
switch ( ( ocotp_mem0 > > 6 ) & 0x3 ) {
2017-11-30 12:17:37 +03:00
case 0 : /* Commercial (0 to 95 °C) */
2015-11-19 17:49:40 +03:00
data - > temp_grade = " Commercial " ;
data - > temp_max = 95000 ;
break ;
2017-11-30 12:17:37 +03:00
case 1 : /* Extended Commercial (-20 °C to 105 °C) */
2015-11-19 17:49:40 +03:00
data - > temp_grade = " Extended Commercial " ;
data - > temp_max = 105000 ;
break ;
2017-11-30 12:17:37 +03:00
case 2 : /* Industrial (-40 °C to 105 °C) */
2015-11-19 17:49:40 +03:00
data - > temp_grade = " Industrial " ;
data - > temp_max = 105000 ;
break ;
2017-11-30 12:17:37 +03:00
case 3 : /* Automotive (-40 °C to 125 °C) */
2015-11-19 17:49:40 +03:00
data - > temp_grade = " Automotive " ;
data - > temp_max = 125000 ;
break ;
}
2013-08-01 20:33:11 +04:00
/*
2017-11-30 12:17:37 +03:00
* Set the critical trip point at 5 ° C under max
* Set the passive trip point at 10 ° C under max ( changeable via sysfs )
2013-08-01 20:33:11 +04:00
*/
2022-10-03 12:25:54 +03:00
trips [ IMX_TRIP_PASSIVE ] . temperature = data - > temp_max - ( 1000 * 10 ) ;
trips [ IMX_TRIP_CRITICAL ] . temperature = data - > temp_max - ( 1000 * 5 ) ;
2017-07-14 17:11:08 +03:00
}
static int imx_init_from_tempmon_data ( struct platform_device * pdev )
{
struct regmap * map ;
int ret ;
u32 val ;
map = syscon_regmap_lookup_by_phandle ( pdev - > dev . of_node ,
" fsl,tempmon-data " ) ;
if ( IS_ERR ( map ) ) {
ret = PTR_ERR ( map ) ;
dev_err ( & pdev - > dev , " failed to get sensor regmap: %d \n " , ret ) ;
return ret ;
}
ret = regmap_read ( map , OCOTP_ANA1 , & val ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " failed to read sensor data: %d \n " , ret ) ;
return ret ;
}
ret = imx_init_calib ( pdev , val ) ;
if ( ret )
return ret ;
ret = regmap_read ( map , OCOTP_MEM0 , & val ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " failed to read sensor data: %d \n " , ret ) ;
return ret ;
}
imx_init_temp_grade ( pdev , val ) ;
return 0 ;
}
static int imx_init_from_nvmem_cells ( struct platform_device * pdev )
{
int ret ;
u32 val ;
ret = nvmem_cell_read_u32 ( & pdev - > dev , " calib " , & val ) ;
if ( ret )
return ret ;
2018-07-01 01:10:50 +03:00
ret = imx_init_calib ( pdev , val ) ;
if ( ret )
return ret ;
2017-07-14 17:11:08 +03:00
ret = nvmem_cell_read_u32 ( & pdev - > dev , " temp_grade " , & val ) ;
if ( ret )
return ret ;
imx_init_temp_grade ( pdev , val ) ;
2013-08-01 20:33:11 +04:00
2013-06-24 10:30:44 +04:00
return 0 ;
}
2013-08-01 20:33:12 +04:00
static irqreturn_t imx_thermal_alarm_irq ( int irq , void * dev )
{
struct imx_thermal_data * data = dev ;
disable_irq_nosync ( irq ) ;
data - > irq_enabled = false ;
return IRQ_WAKE_THREAD ;
}
static irqreturn_t imx_thermal_alarm_irq_thread ( int irq , void * dev )
{
struct imx_thermal_data * data = dev ;
2023-03-01 23:14:37 +03:00
dev_dbg ( data - > dev , " THERMAL ALARM: T > %d \n " , data - > alarm_temp / 1000 ) ;
2013-08-01 20:33:12 +04:00
2016-08-27 02:21:16 +03:00
thermal_zone_device_update ( data - > tz , THERMAL_EVENT_UNSPECIFIED ) ;
2013-08-01 20:33:12 +04:00
return IRQ_HANDLED ;
}
2014-08-06 11:12:09 +04:00
static const struct of_device_id of_imx_thermal_match [ ] = {
{ . compatible = " fsl,imx6q-tempmon " , . data = & thermal_imx6q_data , } ,
{ . compatible = " fsl,imx6sx-tempmon " , . data = & thermal_imx6sx_data , } ,
2018-03-02 04:59:30 +03:00
{ . compatible = " fsl,imx7d-tempmon " , . data = & thermal_imx7d_data , } ,
2014-08-06 11:12:09 +04:00
{ /* end */ }
} ;
MODULE_DEVICE_TABLE ( of , of_imx_thermal_match ) ;
2018-11-21 08:49:36 +03:00
# ifdef CONFIG_CPU_FREQ
2018-06-08 12:06:39 +03:00
/*
* Create cooling device in case no # cooling - cells property is available in
* CPU node
*/
static int imx_thermal_register_legacy_cooling ( struct imx_thermal_data * data )
{
2018-11-21 08:49:36 +03:00
struct device_node * np ;
2020-03-26 17:29:05 +03:00
int ret = 0 ;
2018-06-08 12:06:39 +03:00
2018-11-21 08:49:36 +03:00
data - > policy = cpufreq_cpu_get ( 0 ) ;
if ( ! data - > policy ) {
pr_debug ( " %s: CPUFreq policy not found \n " , __func__ ) ;
return - EPROBE_DEFER ;
}
np = of_get_cpu_node ( data - > policy - > cpu , NULL ) ;
2023-03-10 17:47:26 +03:00
if ( ! np | | ! of_property_present ( np , " #cooling-cells " ) ) {
2018-06-08 12:06:39 +03:00
data - > cdev = cpufreq_cooling_register ( data - > policy ) ;
if ( IS_ERR ( data - > cdev ) ) {
ret = PTR_ERR ( data - > cdev ) ;
cpufreq_cpu_put ( data - > policy ) ;
}
}
2020-03-26 17:29:05 +03:00
of_node_put ( np ) ;
return ret ;
2018-06-08 12:06:39 +03:00
}
2018-11-21 08:49:36 +03:00
static void imx_thermal_unregister_legacy_cooling ( struct imx_thermal_data * data )
{
cpufreq_cooling_unregister ( data - > cdev ) ;
cpufreq_cpu_put ( data - > policy ) ;
}
# else
static inline int imx_thermal_register_legacy_cooling ( struct imx_thermal_data * data )
{
return 0 ;
}
static inline void imx_thermal_unregister_legacy_cooling ( struct imx_thermal_data * data )
{
}
# endif
2013-06-24 10:30:44 +04:00
static int imx_thermal_probe ( struct platform_device * pdev )
{
struct imx_thermal_data * data ;
struct regmap * map ;
2013-08-01 20:33:12 +04:00
int measure_freq ;
2013-06-24 10:30:44 +04:00
int ret ;
data = devm_kzalloc ( & pdev - > dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2021-11-17 13:34:26 +03:00
data - > dev = & pdev - > dev ;
2013-06-24 10:30:44 +04:00
map = syscon_regmap_lookup_by_phandle ( pdev - > dev . of_node , " fsl,tempmon " ) ;
if ( IS_ERR ( map ) ) {
ret = PTR_ERR ( map ) ;
dev_err ( & pdev - > dev , " failed to get tempmon regmap: %d \n " , ret ) ;
return ret ;
}
data - > tempmon = map ;
2016-08-16 11:51:38 +03:00
data - > socdata = of_device_get_match_data ( & pdev - > dev ) ;
2017-01-30 08:04:58 +03:00
if ( ! data - > socdata ) {
dev_err ( & pdev - > dev , " no device match found \n " ) ;
return - ENODEV ;
}
2014-08-06 11:12:09 +04:00
/* make sure the IRQ flag is clear before enabling irq on i.MX6SX */
if ( data - > socdata - > version = = TEMPMON_IMX6SX ) {
2018-03-02 04:59:30 +03:00
regmap_write ( map , IMX6_MISC1 + REG_CLR ,
IMX6_MISC1_IRQ_TEMPHIGH | IMX6_MISC1_IRQ_TEMPLOW
| IMX6_MISC1_IRQ_TEMPPANIC ) ;
2014-08-06 11:12:09 +04:00
/*
* reset value of LOW ALARM is incorrect , set it to lowest
* value to avoid false trigger of low alarm .
*/
2018-03-02 04:59:30 +03:00
regmap_write ( map , data - > socdata - > low_alarm_ctrl + REG_SET ,
data - > socdata - > low_alarm_mask ) ;
2014-08-06 11:12:09 +04:00
}
2013-08-01 20:33:12 +04:00
data - > irq = platform_get_irq ( pdev , 0 ) ;
if ( data - > irq < 0 )
return data - > irq ;
2013-06-24 10:30:44 +04:00
platform_set_drvdata ( pdev , data ) ;
2023-03-10 17:47:26 +03:00
if ( of_property_present ( pdev - > dev . of_node , " nvmem-cells " ) ) {
2017-07-14 17:11:08 +03:00
ret = imx_init_from_nvmem_cells ( pdev ) ;
2020-08-11 09:59:44 +03:00
if ( ret )
return dev_err_probe ( & pdev - > dev , ret ,
" failed to init from nvmem \n " ) ;
2017-07-14 17:11:08 +03:00
} else {
ret = imx_init_from_tempmon_data ( pdev ) ;
if ( ret ) {
2018-09-13 12:13:05 +03:00
dev_err ( & pdev - > dev , " failed to init from fsl,tempmon-data \n " ) ;
2017-07-14 17:11:08 +03:00
return ret ;
}
2013-06-24 10:30:44 +04:00
}
/* Make sure sensor is in known good state for measurements */
2018-03-02 04:59:30 +03:00
regmap_write ( map , data - > socdata - > sensor_ctrl + REG_CLR ,
data - > socdata - > power_down_mask ) ;
regmap_write ( map , data - > socdata - > sensor_ctrl + REG_CLR ,
data - > socdata - > measure_temp_mask ) ;
regmap_write ( map , data - > socdata - > measure_freq_ctrl + REG_CLR ,
data - > socdata - > measure_freq_mask ) ;
if ( data - > socdata - > version ! = TEMPMON_IMX7D )
regmap_write ( map , IMX6_MISC0 + REG_SET ,
IMX6_MISC0_REFTOP_SELBIASOFF ) ;
regmap_write ( map , data - > socdata - > sensor_ctrl + REG_SET ,
data - > socdata - > power_down_mask ) ;
2013-06-24 10:30:44 +04:00
2018-06-08 12:06:39 +03:00
ret = imx_thermal_register_legacy_cooling ( data ) ;
2020-08-11 09:59:44 +03:00
if ( ret )
return dev_err_probe ( & pdev - > dev , ret ,
" failed to register cpufreq cooling device \n " ) ;
2013-06-24 10:30:44 +04:00
2014-11-08 22:35:54 +03:00
data - > thermal_clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( data - > thermal_clk ) ) {
ret = PTR_ERR ( data - > thermal_clk ) ;
if ( ret ! = - EPROBE_DEFER )
dev_err ( & pdev - > dev ,
" failed to get thermal clk: %d \n " , ret ) ;
2018-11-21 08:49:36 +03:00
goto legacy_cleanup ;
2014-11-08 22:35:54 +03:00
}
/*
* Thermal sensor needs clk on to get correct value , normally
* we should enable its clk before taking measurement and disable
* clk after measurement is done , but if alarm function is enabled ,
* hardware will auto measure the temperature periodically , so we
* need to keep the clk always on for alarm function .
*/
ret = clk_prepare_enable ( data - > thermal_clk ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " failed to enable thermal clk: %d \n " , ret ) ;
2018-11-21 08:49:36 +03:00
goto legacy_cleanup ;
2014-11-08 22:35:54 +03:00
}
2022-10-03 12:25:54 +03:00
data - > tz = thermal_zone_device_register_with_trips ( " imx_thermal_zone " ,
trips ,
ARRAY_SIZE ( trips ) ,
BIT ( IMX_TRIP_PASSIVE ) , data ,
& imx_tz_ops , NULL ,
IMX_PASSIVE_DELAY ,
IMX_POLLING_DELAY ) ;
2013-06-24 10:30:44 +04:00
if ( IS_ERR ( data - > tz ) ) {
ret = PTR_ERR ( data - > tz ) ;
dev_err ( & pdev - > dev ,
" failed to register thermal zone device %d \n " , ret ) ;
2018-09-13 12:13:06 +03:00
goto clk_disable ;
2013-06-24 10:30:44 +04:00
}
2015-11-19 17:49:40 +03:00
dev_info ( & pdev - > dev , " %s CPU temperature grade - max:%dC "
" critical:%dC passive:%dC \n " , data - > temp_grade ,
2022-10-03 12:25:54 +03:00
data - > temp_max / 1000 , trips [ IMX_TRIP_CRITICAL ] . temperature / 1000 ,
trips [ IMX_TRIP_PASSIVE ] . temperature / 1000 ) ;
2015-11-19 17:49:40 +03:00
2013-08-01 20:33:12 +04:00
/* Enable measurements at ~ 10 Hz */
2018-03-02 04:59:30 +03:00
regmap_write ( map , data - > socdata - > measure_freq_ctrl + REG_CLR ,
data - > socdata - > measure_freq_mask ) ;
2013-08-01 20:33:12 +04:00
measure_freq = DIV_ROUND_UP ( 32768 , 10 ) ; /* 10 Hz */
2018-03-02 04:59:30 +03:00
regmap_write ( map , data - > socdata - > measure_freq_ctrl + REG_SET ,
measure_freq < < data - > socdata - > measure_freq_shift ) ;
2022-10-03 12:25:54 +03:00
imx_set_alarm_temp ( data , trips [ IMX_TRIP_PASSIVE ] . temperature ) ;
2014-08-06 11:12:09 +04:00
if ( data - > socdata - > version = = TEMPMON_IMX6SX )
2022-10-03 12:25:54 +03:00
imx_set_panic_temp ( data , trips [ IMX_TRIP_CRITICAL ] . temperature ) ;
2014-08-06 11:12:09 +04:00
2018-03-02 04:59:30 +03:00
regmap_write ( map , data - > socdata - > sensor_ctrl + REG_CLR ,
data - > socdata - > power_down_mask ) ;
regmap_write ( map , data - > socdata - > sensor_ctrl + REG_SET ,
data - > socdata - > measure_temp_mask ) ;
2021-11-17 13:34:26 +03:00
/* After power up, we need a delay before first access can be done. */
usleep_range ( 20 , 50 ) ;
/* the core was configured and enabled just before */
pm_runtime_set_active ( & pdev - > dev ) ;
pm_runtime_enable ( data - > dev ) ;
ret = pm_runtime_resume_and_get ( data - > dev ) ;
if ( ret < 0 )
goto disable_runtime_pm ;
2013-08-01 20:33:12 +04:00
2018-02-02 21:17:46 +03:00
data - > irq_enabled = true ;
thermal: Use mode helpers in drivers
Use thermal_zone_device_{en|dis}able() and thermal_zone_device_is_enabled().
Consequently, all set_mode() implementations in drivers:
- can stop modifying tzd's "mode" member,
- shall stop taking tzd's lock, as it is taken in the helpers
- shall stop calling thermal_zone_device_update() as it is called in the
helpers
- can assume they are called when the mode truly changes, so checks to
verify that can be dropped
Not providing set_mode() by a driver no longer prevents the core from
being able to set tzd's mode, so the relevant check in mode_store() is
removed.
Other comments:
- acpi/thermal.c: tz->thermal_zone->mode will be updated only after we
return from set_mode(), so use function parameter in thermal_set_mode()
instead, no need to call acpi_thermal_check() in set_mode()
- thermal/imx_thermal.c: regmap writes and mode assignment are done in
thermal_zone_device_{en|dis}able() and set_mode() callback
- thermal/intel/intel_quark_dts_thermal.c: soc_dts_{en|dis}able() are a
part of set_mode() callback, so they don't need to modify tzd->mode, and
don't need to fall back to the opposite mode if unsuccessful, as the return
value will be propagated to thermal_zone_device_{en|dis}able() and
ultimately tzd's member will not be changed in thermal_zone_device_set_mode().
- thermal/of-thermal.c: no need to set zone->mode to DISABLED in
of_parse_thermal_zones() as a tzd is kzalloc'ed so mode is DISABLED anyway
Signed-off-by: Andrzej Pietrasiewicz <andrzej.p@collabora.com>
[for acerhdf]
Acked-by: Peter Kaestle <peter@piie.net>
Reviewed-by: Amit Kucheria <amit.kucheria@linaro.org>
Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Link: https://lore.kernel.org/r/20200629122925.21729-8-andrzej.p@collabora.com
2020-06-29 15:29:21 +03:00
ret = thermal_zone_device_enable ( data - > tz ) ;
if ( ret )
goto thermal_zone_unregister ;
2018-02-02 21:17:46 +03:00
2015-09-14 14:09:51 +03:00
ret = devm_request_threaded_irq ( & pdev - > dev , data - > irq ,
imx_thermal_alarm_irq , imx_thermal_alarm_irq_thread ,
0 , " imx_thermal " , data ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to request alarm irq: %d \n " , ret ) ;
2018-09-13 12:13:06 +03:00
goto thermal_zone_unregister ;
2015-09-14 14:09:51 +03:00
}
2021-11-17 13:34:26 +03:00
pm_runtime_put ( data - > dev ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
2018-09-13 12:13:06 +03:00
thermal_zone_unregister :
thermal_zone_device_unregister ( data - > tz ) ;
2021-11-17 13:34:26 +03:00
disable_runtime_pm :
pm_runtime_put_noidle ( data - > dev ) ;
pm_runtime_disable ( data - > dev ) ;
2018-09-13 12:13:06 +03:00
clk_disable :
clk_disable_unprepare ( data - > thermal_clk ) ;
2018-11-21 08:49:36 +03:00
legacy_cleanup :
imx_thermal_unregister_legacy_cooling ( data ) ;
2018-09-13 12:13:06 +03:00
return ret ;
2013-06-24 10:30:44 +04:00
}
static int imx_thermal_remove ( struct platform_device * pdev )
{
struct imx_thermal_data * data = platform_get_drvdata ( pdev ) ;
2013-08-01 20:33:12 +04:00
2021-11-17 13:34:26 +03:00
pm_runtime_put_noidle ( data - > dev ) ;
pm_runtime_disable ( data - > dev ) ;
2013-06-24 10:30:44 +04:00
thermal_zone_device_unregister ( data - > tz ) ;
2020-03-13 11:28:14 +03:00
imx_thermal_unregister_legacy_cooling ( data ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
}
2020-03-02 11:54:26 +03:00
static int __maybe_unused imx_thermal_suspend ( struct device * dev )
2013-06-24 10:30:44 +04:00
{
struct imx_thermal_data * data = dev_get_drvdata ( dev ) ;
thermal: Use mode helpers in drivers
Use thermal_zone_device_{en|dis}able() and thermal_zone_device_is_enabled().
Consequently, all set_mode() implementations in drivers:
- can stop modifying tzd's "mode" member,
- shall stop taking tzd's lock, as it is taken in the helpers
- shall stop calling thermal_zone_device_update() as it is called in the
helpers
- can assume they are called when the mode truly changes, so checks to
verify that can be dropped
Not providing set_mode() by a driver no longer prevents the core from
being able to set tzd's mode, so the relevant check in mode_store() is
removed.
Other comments:
- acpi/thermal.c: tz->thermal_zone->mode will be updated only after we
return from set_mode(), so use function parameter in thermal_set_mode()
instead, no need to call acpi_thermal_check() in set_mode()
- thermal/imx_thermal.c: regmap writes and mode assignment are done in
thermal_zone_device_{en|dis}able() and set_mode() callback
- thermal/intel/intel_quark_dts_thermal.c: soc_dts_{en|dis}able() are a
part of set_mode() callback, so they don't need to modify tzd->mode, and
don't need to fall back to the opposite mode if unsuccessful, as the return
value will be propagated to thermal_zone_device_{en|dis}able() and
ultimately tzd's member will not be changed in thermal_zone_device_set_mode().
- thermal/of-thermal.c: no need to set zone->mode to DISABLED in
of_parse_thermal_zones() as a tzd is kzalloc'ed so mode is DISABLED anyway
Signed-off-by: Andrzej Pietrasiewicz <andrzej.p@collabora.com>
[for acerhdf]
Acked-by: Peter Kaestle <peter@piie.net>
Reviewed-by: Amit Kucheria <amit.kucheria@linaro.org>
Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Link: https://lore.kernel.org/r/20200629122925.21729-8-andrzej.p@collabora.com
2020-06-29 15:29:21 +03:00
int ret ;
2013-06-24 10:30:44 +04:00
2013-12-24 18:43:24 +04:00
/*
* Need to disable thermal sensor , otherwise , when thermal core
* try to get temperature before thermal sensor resume , a wrong
* temperature will be read as the thermal sensor is powered
2020-06-29 15:29:25 +03:00
* down . This is done in change_mode ( ) operation called from
thermal: Use mode helpers in drivers
Use thermal_zone_device_{en|dis}able() and thermal_zone_device_is_enabled().
Consequently, all set_mode() implementations in drivers:
- can stop modifying tzd's "mode" member,
- shall stop taking tzd's lock, as it is taken in the helpers
- shall stop calling thermal_zone_device_update() as it is called in the
helpers
- can assume they are called when the mode truly changes, so checks to
verify that can be dropped
Not providing set_mode() by a driver no longer prevents the core from
being able to set tzd's mode, so the relevant check in mode_store() is
removed.
Other comments:
- acpi/thermal.c: tz->thermal_zone->mode will be updated only after we
return from set_mode(), so use function parameter in thermal_set_mode()
instead, no need to call acpi_thermal_check() in set_mode()
- thermal/imx_thermal.c: regmap writes and mode assignment are done in
thermal_zone_device_{en|dis}able() and set_mode() callback
- thermal/intel/intel_quark_dts_thermal.c: soc_dts_{en|dis}able() are a
part of set_mode() callback, so they don't need to modify tzd->mode, and
don't need to fall back to the opposite mode if unsuccessful, as the return
value will be propagated to thermal_zone_device_{en|dis}able() and
ultimately tzd's member will not be changed in thermal_zone_device_set_mode().
- thermal/of-thermal.c: no need to set zone->mode to DISABLED in
of_parse_thermal_zones() as a tzd is kzalloc'ed so mode is DISABLED anyway
Signed-off-by: Andrzej Pietrasiewicz <andrzej.p@collabora.com>
[for acerhdf]
Acked-by: Peter Kaestle <peter@piie.net>
Reviewed-by: Amit Kucheria <amit.kucheria@linaro.org>
Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Link: https://lore.kernel.org/r/20200629122925.21729-8-andrzej.p@collabora.com
2020-06-29 15:29:21 +03:00
* thermal_zone_device_disable ( )
2013-12-24 18:43:24 +04:00
*/
thermal: Use mode helpers in drivers
Use thermal_zone_device_{en|dis}able() and thermal_zone_device_is_enabled().
Consequently, all set_mode() implementations in drivers:
- can stop modifying tzd's "mode" member,
- shall stop taking tzd's lock, as it is taken in the helpers
- shall stop calling thermal_zone_device_update() as it is called in the
helpers
- can assume they are called when the mode truly changes, so checks to
verify that can be dropped
Not providing set_mode() by a driver no longer prevents the core from
being able to set tzd's mode, so the relevant check in mode_store() is
removed.
Other comments:
- acpi/thermal.c: tz->thermal_zone->mode will be updated only after we
return from set_mode(), so use function parameter in thermal_set_mode()
instead, no need to call acpi_thermal_check() in set_mode()
- thermal/imx_thermal.c: regmap writes and mode assignment are done in
thermal_zone_device_{en|dis}able() and set_mode() callback
- thermal/intel/intel_quark_dts_thermal.c: soc_dts_{en|dis}able() are a
part of set_mode() callback, so they don't need to modify tzd->mode, and
don't need to fall back to the opposite mode if unsuccessful, as the return
value will be propagated to thermal_zone_device_{en|dis}able() and
ultimately tzd's member will not be changed in thermal_zone_device_set_mode().
- thermal/of-thermal.c: no need to set zone->mode to DISABLED in
of_parse_thermal_zones() as a tzd is kzalloc'ed so mode is DISABLED anyway
Signed-off-by: Andrzej Pietrasiewicz <andrzej.p@collabora.com>
[for acerhdf]
Acked-by: Peter Kaestle <peter@piie.net>
Reviewed-by: Amit Kucheria <amit.kucheria@linaro.org>
Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Link: https://lore.kernel.org/r/20200629122925.21729-8-andrzej.p@collabora.com
2020-06-29 15:29:21 +03:00
ret = thermal_zone_device_disable ( data - > tz ) ;
if ( ret )
return ret ;
2021-11-17 13:34:26 +03:00
return pm_runtime_force_suspend ( data - > dev ) ;
}
static int __maybe_unused imx_thermal_resume ( struct device * dev )
{
struct imx_thermal_data * data = dev_get_drvdata ( dev ) ;
int ret ;
ret = pm_runtime_force_resume ( data - > dev ) ;
if ( ret )
return ret ;
/* Enabled thermal sensor after resume */
return thermal_zone_device_enable ( data - > tz ) ;
}
static int __maybe_unused imx_thermal_runtime_suspend ( struct device * dev )
{
struct imx_thermal_data * data = dev_get_drvdata ( dev ) ;
const struct thermal_soc_data * socdata = data - > socdata ;
struct regmap * map = data - > tempmon ;
int ret ;
ret = regmap_write ( map , socdata - > sensor_ctrl + REG_CLR ,
socdata - > measure_temp_mask ) ;
if ( ret )
return ret ;
ret = regmap_write ( map , socdata - > sensor_ctrl + REG_SET ,
socdata - > power_down_mask ) ;
if ( ret )
return ret ;
2015-01-06 13:50:22 +03:00
clk_disable_unprepare ( data - > thermal_clk ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
}
2021-11-17 13:34:26 +03:00
static int __maybe_unused imx_thermal_runtime_resume ( struct device * dev )
2013-06-24 10:30:44 +04:00
{
2013-12-24 18:43:24 +04:00
struct imx_thermal_data * data = dev_get_drvdata ( dev ) ;
2021-11-17 13:34:26 +03:00
const struct thermal_soc_data * socdata = data - > socdata ;
struct regmap * map = data - > tempmon ;
2017-06-06 12:42:37 +03:00
int ret ;
2013-12-24 18:43:24 +04:00
2017-06-06 12:42:37 +03:00
ret = clk_prepare_enable ( data - > thermal_clk ) ;
if ( ret )
return ret ;
2021-11-17 13:34:26 +03:00
ret = regmap_write ( map , socdata - > sensor_ctrl + REG_CLR ,
socdata - > power_down_mask ) ;
if ( ret )
return ret ;
ret = regmap_write ( map , socdata - > sensor_ctrl + REG_SET ,
socdata - > measure_temp_mask ) ;
thermal: Use mode helpers in drivers
Use thermal_zone_device_{en|dis}able() and thermal_zone_device_is_enabled().
Consequently, all set_mode() implementations in drivers:
- can stop modifying tzd's "mode" member,
- shall stop taking tzd's lock, as it is taken in the helpers
- shall stop calling thermal_zone_device_update() as it is called in the
helpers
- can assume they are called when the mode truly changes, so checks to
verify that can be dropped
Not providing set_mode() by a driver no longer prevents the core from
being able to set tzd's mode, so the relevant check in mode_store() is
removed.
Other comments:
- acpi/thermal.c: tz->thermal_zone->mode will be updated only after we
return from set_mode(), so use function parameter in thermal_set_mode()
instead, no need to call acpi_thermal_check() in set_mode()
- thermal/imx_thermal.c: regmap writes and mode assignment are done in
thermal_zone_device_{en|dis}able() and set_mode() callback
- thermal/intel/intel_quark_dts_thermal.c: soc_dts_{en|dis}able() are a
part of set_mode() callback, so they don't need to modify tzd->mode, and
don't need to fall back to the opposite mode if unsuccessful, as the return
value will be propagated to thermal_zone_device_{en|dis}able() and
ultimately tzd's member will not be changed in thermal_zone_device_set_mode().
- thermal/of-thermal.c: no need to set zone->mode to DISABLED in
of_parse_thermal_zones() as a tzd is kzalloc'ed so mode is DISABLED anyway
Signed-off-by: Andrzej Pietrasiewicz <andrzej.p@collabora.com>
[for acerhdf]
Acked-by: Peter Kaestle <peter@piie.net>
Reviewed-by: Amit Kucheria <amit.kucheria@linaro.org>
Reviewed-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org>
Link: https://lore.kernel.org/r/20200629122925.21729-8-andrzej.p@collabora.com
2020-06-29 15:29:21 +03:00
if ( ret )
return ret ;
2013-12-24 18:43:24 +04:00
2021-11-17 13:34:26 +03:00
/*
* According to the temp sensor designers , it may require up to ~ 17u s
* to complete a measurement .
*/
usleep_range ( 20 , 50 ) ;
2013-06-24 10:30:44 +04:00
return 0 ;
}
2021-11-17 13:34:26 +03:00
static const struct dev_pm_ops imx_thermal_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS ( imx_thermal_suspend , imx_thermal_resume )
SET_RUNTIME_PM_OPS ( imx_thermal_runtime_suspend ,
imx_thermal_runtime_resume , NULL )
} ;
2013-06-24 10:30:44 +04:00
static struct platform_driver imx_thermal = {
. driver = {
. name = " imx_thermal " ,
. pm = & imx_thermal_pm_ops ,
. of_match_table = of_imx_thermal_match ,
} ,
. probe = imx_thermal_probe ,
. remove = imx_thermal_remove ,
} ;
module_platform_driver ( imx_thermal ) ;
MODULE_AUTHOR ( " Freescale Semiconductor, Inc. " ) ;
MODULE_DESCRIPTION ( " Thermal driver for Freescale i.MX SoCs " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform:imx-thermal " ) ;