Skip to content

Commit

Permalink
thermal/core: Protect sysfs accesses to thermal operations with therm…
Browse files Browse the repository at this point in the history
…al zone mutex

Protect access to thermal operations against thermal zone removal by
acquiring the thermal zone device mutex. After acquiring the mutex, check
if the thermal zone device is registered and abort the operation if not.

With this change, we can call __thermal_zone_device_update() instead of
thermal_zone_device_update() from trip_point_temp_store() and from
emul_temp_store(). Similar, we can call __thermal_zone_set_trips() instead
of thermal_zone_set_trips() from trip_point_hyst_store().

Signed-off-by: Guenter Roeck <linux@roeck-us.net>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
  • Loading branch information
Guenter Roeck authored and Rafael J. Wysocki committed Nov 14, 2022
1 parent ea37bec commit 05eeee2
Show file tree
Hide file tree
Showing 3 changed files with 69 additions and 16 deletions.
4 changes: 2 additions & 2 deletions drivers/thermal/thermal_core.c
Original file line number Diff line number Diff line change
Expand Up @@ -403,8 +403,8 @@ static void thermal_zone_device_init(struct thermal_zone_device *tz)
pos->initialized = false;
}

static void __thermal_zone_device_update(struct thermal_zone_device *tz,
enum thermal_notify_event event)
void __thermal_zone_device_update(struct thermal_zone_device *tz,
enum thermal_notify_event event)
{
int count;

Expand Down
2 changes: 2 additions & 0 deletions drivers/thermal/thermal_core.h
Original file line number Diff line number Diff line change
Expand Up @@ -109,6 +109,8 @@ int thermal_register_governor(struct thermal_governor *);
void thermal_unregister_governor(struct thermal_governor *);
int thermal_zone_device_set_policy(struct thermal_zone_device *, char *);
int thermal_build_list_of_policies(char *buf);
void __thermal_zone_device_update(struct thermal_zone_device *tz,
enum thermal_notify_event event);

/* Helpers */
void thermal_zone_set_trips(struct thermal_zone_device *tz);
Expand Down
79 changes: 65 additions & 14 deletions drivers/thermal/thermal_sysfs.c
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,14 @@ trip_point_type_show(struct device *dev, struct device_attribute *attr,
if (sscanf(attr->attr.name, "trip_point_%d_type", &trip) != 1)
return -EINVAL;

result = tz->ops->get_trip_type(tz, trip, &type);
mutex_lock(&tz->lock);

if (device_is_registered(dev))
result = tz->ops->get_trip_type(tz, trip, &type);
else
result = -ENODEV;

mutex_unlock(&tz->lock);
if (result)
return result;

Expand Down Expand Up @@ -128,10 +135,17 @@ trip_point_temp_store(struct device *dev, struct device_attribute *attr,
if (kstrtoint(buf, 10, &temperature))
return -EINVAL;

mutex_lock(&tz->lock);

if (!device_is_registered(dev)) {
ret = -ENODEV;
goto unlock;
}

if (tz->ops->set_trip_temp) {
ret = tz->ops->set_trip_temp(tz, trip, temperature);
if (ret)
return ret;
goto unlock;
}

if (tz->trips)
Expand All @@ -140,16 +154,22 @@ trip_point_temp_store(struct device *dev, struct device_attribute *attr,
if (tz->ops->get_trip_hyst) {
ret = tz->ops->get_trip_hyst(tz, trip, &hyst);
if (ret)
return ret;
goto unlock;
}

ret = tz->ops->get_trip_type(tz, trip, &type);
if (ret)
return ret;
goto unlock;

thermal_notify_tz_trip_change(tz->id, trip, type, temperature, hyst);

thermal_zone_device_update(tz, THERMAL_EVENT_UNSPECIFIED);
__thermal_zone_device_update(tz, THERMAL_EVENT_UNSPECIFIED);

unlock:
mutex_unlock(&tz->lock);

if (ret)
return ret;

return count;
}
Expand All @@ -168,7 +188,14 @@ trip_point_temp_show(struct device *dev, struct device_attribute *attr,
if (sscanf(attr->attr.name, "trip_point_%d_temp", &trip) != 1)
return -EINVAL;

ret = tz->ops->get_trip_temp(tz, trip, &temperature);
mutex_lock(&tz->lock);

if (device_is_registered(dev))
ret = tz->ops->get_trip_temp(tz, trip, &temperature);
else
ret = -ENODEV;

mutex_unlock(&tz->lock);

if (ret)
return ret;
Expand All @@ -193,6 +220,13 @@ trip_point_hyst_store(struct device *dev, struct device_attribute *attr,
if (kstrtoint(buf, 10, &temperature))
return -EINVAL;

mutex_lock(&tz->lock);

if (!device_is_registered(dev)) {
ret = -ENODEV;
goto unlock;
}

/*
* We are not doing any check on the 'temperature' value
* here. The driver implementing 'set_trip_hyst' has to
Expand All @@ -201,7 +235,10 @@ trip_point_hyst_store(struct device *dev, struct device_attribute *attr,
ret = tz->ops->set_trip_hyst(tz, trip, temperature);

if (!ret)
thermal_zone_set_trips(tz);
__thermal_zone_set_trips(tz);

unlock:
mutex_unlock(&tz->lock);

return ret ? ret : count;
}
Expand All @@ -220,7 +257,14 @@ trip_point_hyst_show(struct device *dev, struct device_attribute *attr,
if (sscanf(attr->attr.name, "trip_point_%d_hyst", &trip) != 1)
return -EINVAL;

ret = tz->ops->get_trip_hyst(tz, trip, &temperature);
mutex_lock(&tz->lock);

if (device_is_registered(dev))
ret = tz->ops->get_trip_hyst(tz, trip, &temperature);
else
ret = -ENODEV;

mutex_unlock(&tz->lock);

return ret ? ret : sprintf(buf, "%d\n", temperature);
}
Expand Down Expand Up @@ -269,16 +313,23 @@ emul_temp_store(struct device *dev, struct device_attribute *attr,
if (kstrtoint(buf, 10, &temperature))
return -EINVAL;

if (!tz->ops->set_emul_temp) {
mutex_lock(&tz->lock);
mutex_lock(&tz->lock);

if (!device_is_registered(dev)) {
ret = -ENODEV;
goto unlock;
}

if (!tz->ops->set_emul_temp)
tz->emul_temperature = temperature;
mutex_unlock(&tz->lock);
} else {
else
ret = tz->ops->set_emul_temp(tz, temperature);
}

if (!ret)
thermal_zone_device_update(tz, THERMAL_EVENT_UNSPECIFIED);
__thermal_zone_device_update(tz, THERMAL_EVENT_UNSPECIFIED);

unlock:
mutex_unlock(&tz->lock);

return ret ? ret : count;
}
Expand Down

0 comments on commit 05eeee2

Please sign in to comment.