aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAnson Huang <b20788@freescale.com>2013-08-08 13:57:23 -0400
committerNitin Garg <nitin.garg@freescale.com>2014-04-16 09:01:05 -0400
commitca7231fb849e1f40e9e4c0cac21b5f6c4e339c69 (patch)
tree632b407ae6ab12d3dc0adb976efcc148dfe8890c
parenteab573f4f4986c3e2bb5ea1e7b2ae624dc5aafa6 (diff)
ENGR00274056-3 thermal: imx: binding device cooling to thermal
1. imx thermal depends on device cooling config, as only cpu cooling is not enough for cooling down SOC; 2. binding device cooling to imx thermal driver. 3. add temperature buffer for passive trip, which means when temperature cross passive trip, cooling devices will be triggered, but only when temperature drop to more than the number we defined(10 C) lower than passive trip, cooling devices will be canceled. this is to avoid triggering/canceling cooling device back and forth when temperature is around passive trip. Signed-off-by: Anson Huang <b20788@freescale.com>
-rw-r--r--drivers/thermal/imx_thermal.c43
1 files changed, 37 insertions, 6 deletions
diff --git a/drivers/thermal/imx_thermal.c b/drivers/thermal/imx_thermal.c
index 0f78121c0892..7343476c5964 100644
--- a/drivers/thermal/imx_thermal.c
+++ b/drivers/thermal/imx_thermal.c
@@ -11,6 +11,7 @@
11#include <linux/cpufreq.h> 11#include <linux/cpufreq.h>
12#include <linux/delay.h> 12#include <linux/delay.h>
13#include <linux/device.h> 13#include <linux/device.h>
14#include <linux/device_cooling.h>
14#include <linux/init.h> 15#include <linux/init.h>
15#include <linux/io.h> 16#include <linux/io.h>
16#include <linux/kernel.h> 17#include <linux/kernel.h>
@@ -54,6 +55,7 @@ enum imx_thermal_trip {
54 * that will trigger cooling action when crossed. 55 * that will trigger cooling action when crossed.
55 */ 56 */
56#define IMX_TEMP_PASSIVE 85000 57#define IMX_TEMP_PASSIVE 85000
58#define IMX_TEMP_PASSIVE_COOL_DELTA 10000
57 59
58/* 60/*
59 * The maximum die temperature on imx parts is 105C, let's give some cushion 61 * The maximum die temperature on imx parts is 105C, let's give some cushion
@@ -70,7 +72,7 @@ enum imx_thermal_trip {
70 72
71struct imx_thermal_data { 73struct imx_thermal_data {
72 struct thermal_zone_device *tz; 74 struct thermal_zone_device *tz;
73 struct thermal_cooling_device *cdev; 75 struct thermal_cooling_device *cdev[2];
74 enum thermal_device_mode mode; 76 enum thermal_device_mode mode;
75 struct regmap *tempmon; 77 struct regmap *tempmon;
76 unsigned long trip_temp[IMX_TRIP_NUM]; 78 unsigned long trip_temp[IMX_TRIP_NUM];
@@ -222,6 +224,24 @@ static int imx_unbind(struct thermal_zone_device *tz,
222 return 0; 224 return 0;
223} 225}
224 226
227static int imx_get_trend(struct thermal_zone_device *tz,
228 int trip, enum thermal_trend *trend)
229{
230 int ret;
231 unsigned long trip_temp;
232
233 ret = imx_get_trip_temp(tz, trip, &trip_temp);
234 if (ret < 0)
235 return ret;
236
237 if (tz->temperature >= (trip_temp - IMX_TEMP_PASSIVE_COOL_DELTA))
238 *trend = THERMAL_TREND_RAISE_FULL;
239 else
240 *trend = THERMAL_TREND_DROP_FULL;
241
242 return 0;
243}
244
225static const struct thermal_zone_device_ops imx_tz_ops = { 245static const struct thermal_zone_device_ops imx_tz_ops = {
226 .bind = imx_bind, 246 .bind = imx_bind,
227 .unbind = imx_unbind, 247 .unbind = imx_unbind,
@@ -231,6 +251,7 @@ static const struct thermal_zone_device_ops imx_tz_ops = {
231 .get_trip_type = imx_get_trip_type, 251 .get_trip_type = imx_get_trip_type,
232 .get_trip_temp = imx_get_trip_temp, 252 .get_trip_temp = imx_get_trip_temp,
233 .get_crit_temp = imx_get_crit_temp, 253 .get_crit_temp = imx_get_crit_temp,
254 .get_trend = imx_get_trend,
234 .set_trip_temp = imx_set_trip_temp, 255 .set_trip_temp = imx_set_trip_temp,
235}; 256};
236 257
@@ -331,14 +352,22 @@ static int imx_thermal_probe(struct platform_device *pdev)
331 regmap_write(map, TEMPSENSE0 + REG_SET, TEMPSENSE0_POWER_DOWN); 352 regmap_write(map, TEMPSENSE0 + REG_SET, TEMPSENSE0_POWER_DOWN);
332 353
333 cpumask_set_cpu(0, &clip_cpus); 354 cpumask_set_cpu(0, &clip_cpus);
334 data->cdev = cpufreq_cooling_register(&clip_cpus); 355 data->cdev[0] = cpufreq_cooling_register(&clip_cpus);
335 if (IS_ERR(data->cdev)) { 356 if (IS_ERR(data->cdev[0])) {
336 ret = PTR_ERR(data->cdev); 357 ret = PTR_ERR(data->cdev[0]);
337 dev_err(&pdev->dev, 358 dev_err(&pdev->dev,
338 "failed to register cpufreq cooling device: %d\n", ret); 359 "failed to register cpufreq cooling device: %d\n", ret);
339 return ret; 360 return ret;
340 } 361 }
341 362
363 data->cdev[1] = devfreq_cooling_register();
364 if (IS_ERR(data->cdev[1])) {
365 ret = PTR_ERR(data->cdev[1]);
366 dev_err(&pdev->dev,
367 "failed to register devfreq cooling device: %d\n", ret);
368 return ret;
369 }
370
342 data->trip_temp[IMX_TRIP_PASSIVE] = IMX_TEMP_PASSIVE; 371 data->trip_temp[IMX_TRIP_PASSIVE] = IMX_TEMP_PASSIVE;
343 data->trip_temp[IMX_TRIP_CRITICAL] = IMX_TEMP_CRITICAL; 372 data->trip_temp[IMX_TRIP_CRITICAL] = IMX_TEMP_CRITICAL;
344 data->tz = thermal_zone_device_register("imx_thermal_zone", 373 data->tz = thermal_zone_device_register("imx_thermal_zone",
@@ -352,7 +381,8 @@ static int imx_thermal_probe(struct platform_device *pdev)
352 ret = PTR_ERR(data->tz); 381 ret = PTR_ERR(data->tz);
353 dev_err(&pdev->dev, 382 dev_err(&pdev->dev,
354 "failed to register thermal zone device %d\n", ret); 383 "failed to register thermal zone device %d\n", ret);
355 cpufreq_cooling_unregister(data->cdev); 384 cpufreq_cooling_unregister(data->cdev[0]);
385 devfreq_cooling_unregister(data->cdev[1]);
356 return ret; 386 return ret;
357 } 387 }
358 388
@@ -366,7 +396,8 @@ static int imx_thermal_remove(struct platform_device *pdev)
366 struct imx_thermal_data *data = platform_get_drvdata(pdev); 396 struct imx_thermal_data *data = platform_get_drvdata(pdev);
367 397
368 thermal_zone_device_unregister(data->tz); 398 thermal_zone_device_unregister(data->tz);
369 cpufreq_cooling_unregister(data->cdev); 399 cpufreq_cooling_unregister(data->cdev[0]);
400 devfreq_cooling_unregister(data->cdev[1]);
370 401
371 return 0; 402 return 0;
372} 403}