mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-07 13:53:24 +00:00
15cb56bd52
If a cooling device is registered after a thermal zone it should be
bound to and the trip point it should be bound to has already been
crossed by the zone temperature on the way up, the cooling device's
state may need to be adjusted, but the Bang-bang governor will not
do that because its .manage() callback only looks at thermal instances
for trip points whose thresholds are below or at the zone temperature.
Address this by updating bang_bang_manage() to look at all of the
uninitialized thermal instances and setting their target states in
accordance with the position of the zone temperature with respect to
the threshold of the given trip point.
Fixes: 5f64b4a1ab
("thermal: gov_bang_bang: Add .manage() callback")
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Lukasz Luba <lukasz.luba@arm.com>
Link: https://patch.msgid.link/6103874.lOV4Wx5bFT@rjwysocki.net
134 lines
3.9 KiB
C
134 lines
3.9 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* gov_bang_bang.c - A simple thermal throttling governor using hysteresis
|
|
*
|
|
* Copyright (C) 2014 Peter Kaestle <peter@piie.net>
|
|
*
|
|
* Based on step_wise.c with following Copyrights:
|
|
* Copyright (C) 2012 Intel Corp
|
|
* Copyright (C) 2012 Durgadoss R <durgadoss.r@intel.com>
|
|
*/
|
|
|
|
#include <linux/thermal.h>
|
|
|
|
#include "thermal_core.h"
|
|
|
|
static void bang_bang_set_instance_target(struct thermal_instance *instance,
|
|
unsigned int target)
|
|
{
|
|
if (instance->target != 0 && instance->target != 1 &&
|
|
instance->target != THERMAL_NO_TARGET)
|
|
pr_debug("Unexpected state %ld of thermal instance %s in bang-bang\n",
|
|
instance->target, instance->name);
|
|
|
|
/*
|
|
* Enable the fan when the trip is crossed on the way up and disable it
|
|
* when the trip is crossed on the way down.
|
|
*/
|
|
instance->target = target;
|
|
instance->initialized = true;
|
|
|
|
dev_dbg(&instance->cdev->device, "target=%ld\n", instance->target);
|
|
|
|
mutex_lock(&instance->cdev->lock);
|
|
__thermal_cdev_update(instance->cdev);
|
|
mutex_unlock(&instance->cdev->lock);
|
|
}
|
|
|
|
/**
|
|
* bang_bang_control - controls devices associated with the given zone
|
|
* @tz: thermal_zone_device
|
|
* @trip: the trip point
|
|
* @crossed_up: whether or not the trip has been crossed on the way up
|
|
*
|
|
* Regulation Logic: a two point regulation, deliver cooling state depending
|
|
* on the previous state shown in this diagram:
|
|
*
|
|
* Fan: OFF ON
|
|
*
|
|
* |
|
|
* |
|
|
* trip_temp: +---->+
|
|
* | | ^
|
|
* | | |
|
|
* | | Temperature
|
|
* (trip_temp - hyst): +<----+
|
|
* |
|
|
* |
|
|
* |
|
|
*
|
|
* * If the fan is not running and temperature exceeds trip_temp, the fan
|
|
* gets turned on.
|
|
* * In case the fan is running, temperature must fall below
|
|
* (trip_temp - hyst) so that the fan gets turned off again.
|
|
*
|
|
*/
|
|
static void bang_bang_control(struct thermal_zone_device *tz,
|
|
const struct thermal_trip *trip,
|
|
bool crossed_up)
|
|
{
|
|
struct thermal_instance *instance;
|
|
|
|
lockdep_assert_held(&tz->lock);
|
|
|
|
dev_dbg(&tz->device, "Trip%d[temp=%d]:temp=%d:hyst=%d\n",
|
|
thermal_zone_trip_id(tz, trip), trip->temperature,
|
|
tz->temperature, trip->hysteresis);
|
|
|
|
list_for_each_entry(instance, &tz->thermal_instances, tz_node) {
|
|
if (instance->trip == trip)
|
|
bang_bang_set_instance_target(instance, crossed_up);
|
|
}
|
|
}
|
|
|
|
static void bang_bang_manage(struct thermal_zone_device *tz)
|
|
{
|
|
const struct thermal_trip_desc *td;
|
|
struct thermal_instance *instance;
|
|
|
|
/* If the code below has run already, nothing needs to be done. */
|
|
if (tz->governor_data)
|
|
return;
|
|
|
|
for_each_trip_desc(tz, td) {
|
|
const struct thermal_trip *trip = &td->trip;
|
|
bool turn_on;
|
|
|
|
if (trip->temperature == THERMAL_TEMP_INVALID ||
|
|
trip->type == THERMAL_TRIP_CRITICAL ||
|
|
trip->type == THERMAL_TRIP_HOT)
|
|
continue;
|
|
|
|
/*
|
|
* Adjust the target states for uninitialized thermal instances
|
|
* to the thermal zone temperature and the trip point threshold.
|
|
*/
|
|
turn_on = tz->temperature >= td->threshold;
|
|
list_for_each_entry(instance, &tz->thermal_instances, tz_node) {
|
|
if (!instance->initialized && instance->trip == trip)
|
|
bang_bang_set_instance_target(instance, turn_on);
|
|
}
|
|
}
|
|
|
|
tz->governor_data = (void *)true;
|
|
}
|
|
|
|
static void bang_bang_update_tz(struct thermal_zone_device *tz,
|
|
enum thermal_notify_event reason)
|
|
{
|
|
/*
|
|
* Let bang_bang_manage() know that it needs to walk trips after binding
|
|
* a new cdev and after system resume.
|
|
*/
|
|
if (reason == THERMAL_TZ_BIND_CDEV || reason == THERMAL_TZ_RESUME)
|
|
tz->governor_data = NULL;
|
|
}
|
|
|
|
static struct thermal_governor thermal_gov_bang_bang = {
|
|
.name = "bang_bang",
|
|
.trip_crossed = bang_bang_control,
|
|
.manage = bang_bang_manage,
|
|
.update_tz = bang_bang_update_tz,
|
|
};
|
|
THERMAL_GOVERNOR_DECLARE(thermal_gov_bang_bang);
|