aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/thermal
diff options
context:
space:
mode:
authorSascha Hauer <s.hauer@pengutronix.de>2016-06-22 04:42:01 -0400
committerZhang Rui <rui.zhang@intel.com>2016-09-27 02:02:16 -0400
commit060c034a974187e930b790957cafc5047cc30a40 (patch)
tree0afdf73e38e94d0bd704ba40ae283ee088df6179 /drivers/thermal
parent032d4057291e9d017cf50be017cf09b169ff438f (diff)
thermal: Add support for hardware-tracked trip points
This adds support for hardware-tracked trip points to the device tree thermal sensor framework. The framework supports an arbitrary number of trip points. Whenever the current temperature is updated, the trip points immediately below and above the current temperature are found. A .set_trips callback is then called with the temperatures. If there is no trip point above or below the current temperature, the passed trip temperature will be -INT_MAX or INT_MAX respectively. In this callback, the driver should program the hardware such that it is notified when either of these trip points are triggered. When a trip point is triggered, the driver should call `thermal_zone_device_update' for the respective thermal zone. This will cause the trip points to be updated again. If .set_trips is not implemented, the framework behaves as before. This patch is based on an earlier version from Mikko Perttunen <mikko.perttunen@kapsi.fi> Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de> Signed-off-by: Caesar Wang <wxt@rock-chips.com> Cc: Zhang Rui <rui.zhang@intel.com> Cc: Eduardo Valentin <edubezval@gmail.com> Cc: linux-pm@vger.kernel.org Reviewed-by: Javi Merino <javi.merino@arm.com> Signed-off-by: Eduardo Valentin <edubezval@gmail.com> Signed-off-by: Zhang Rui <rui.zhang@intel.com>
Diffstat (limited to 'drivers/thermal')
-rw-r--r--drivers/thermal/thermal_core.c55
1 files changed, 55 insertions, 0 deletions
diff --git a/drivers/thermal/thermal_core.c b/drivers/thermal/thermal_core.c
index 8728cc615452..f2d55e478b2a 100644
--- a/drivers/thermal/thermal_core.c
+++ b/drivers/thermal/thermal_core.c
@@ -520,6 +520,56 @@ exit:
520} 520}
521EXPORT_SYMBOL_GPL(thermal_zone_get_temp); 521EXPORT_SYMBOL_GPL(thermal_zone_get_temp);
522 522
523void thermal_zone_set_trips(struct thermal_zone_device *tz)
524{
525 int low = -INT_MAX;
526 int high = INT_MAX;
527 int trip_temp, hysteresis;
528 int i, ret;
529
530 mutex_lock(&tz->lock);
531
532 if (!tz->ops->set_trips || !tz->ops->get_trip_hyst)
533 goto exit;
534
535 for (i = 0; i < tz->trips; i++) {
536 int trip_low;
537
538 tz->ops->get_trip_temp(tz, i, &trip_temp);
539 tz->ops->get_trip_hyst(tz, i, &hysteresis);
540
541 trip_low = trip_temp - hysteresis;
542
543 if (trip_low < tz->temperature && trip_low > low)
544 low = trip_low;
545
546 if (trip_temp > tz->temperature && trip_temp < high)
547 high = trip_temp;
548 }
549
550 /* No need to change trip points */
551 if (tz->prev_low_trip == low && tz->prev_high_trip == high)
552 goto exit;
553
554 tz->prev_low_trip = low;
555 tz->prev_high_trip = high;
556
557 dev_dbg(&tz->device,
558 "new temperature boundaries: %d < x < %d\n", low, high);
559
560 /*
561 * Set a temperature window. When this window is left the driver
562 * must inform the thermal core via thermal_zone_device_update.
563 */
564 ret = tz->ops->set_trips(tz, low, high);
565 if (ret)
566 dev_err(&tz->device, "Failed to set trips: %d\n", ret);
567
568exit:
569 mutex_unlock(&tz->lock);
570}
571EXPORT_SYMBOL_GPL(thermal_zone_set_trips);
572
523static void update_temperature(struct thermal_zone_device *tz) 573static void update_temperature(struct thermal_zone_device *tz)
524{ 574{
525 int temp, ret; 575 int temp, ret;
@@ -569,6 +619,8 @@ void thermal_zone_device_update(struct thermal_zone_device *tz)
569 619
570 update_temperature(tz); 620 update_temperature(tz);
571 621
622 thermal_zone_set_trips(tz);
623
572 for (count = 0; count < tz->trips; count++) 624 for (count = 0; count < tz->trips; count++)
573 handle_thermal_trip(tz, count); 625 handle_thermal_trip(tz, count);
574} 626}
@@ -754,6 +806,9 @@ trip_point_hyst_store(struct device *dev, struct device_attribute *attr,
754 */ 806 */
755 ret = tz->ops->set_trip_hyst(tz, trip, temperature); 807 ret = tz->ops->set_trip_hyst(tz, trip, temperature);
756 808
809 if (!ret)
810 thermal_zone_set_trips(tz);
811
757 return ret ? ret : count; 812 return ret ? ret : count;
758} 813}
759 814