diff options
author | Anson Huang <b20788@freescale.com> | 2013-08-08 13:57:23 -0400 |
---|---|---|
committer | Nitin Garg <nitin.garg@freescale.com> | 2014-04-16 09:01:05 -0400 |
commit | ca7231fb849e1f40e9e4c0cac21b5f6c4e339c69 (patch) | |
tree | 632b407ae6ab12d3dc0adb976efcc148dfe8890c | |
parent | eab573f4f4986c3e2bb5ea1e7b2ae624dc5aafa6 (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.c | 43 |
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 | ||
71 | struct imx_thermal_data { | 73 | struct 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 | ||
227 | static 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 | |||
225 | static const struct thermal_zone_device_ops imx_tz_ops = { | 245 | static 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 | } |