thermal/core: Protect sysfs accesses to thermal operations with thermal zone mutex
authorGuenter Roeck <linux@roeck-us.net>
Thu, 10 Nov 2022 15:24:58 +0000 (07:24 -0800)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 14 Nov 2022 18:04:37 +0000 (19:04 +0100)
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>
drivers/thermal/thermal_core.c
drivers/thermal/thermal_core.h
drivers/thermal/thermal_sysfs.c

index 0657e4bd47917f769280d37e969155b5e1c07cb3..173b049dca411caa3afe1eeba05a0cdb4ef0b3c4 100644 (file)
@@ -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;
 
index 1571917bd3c8917a4d1115bee42e7a3230babade..7b51b9a22e7eb764a1b5a5067d8ecd4088e14fb4 100644 (file)
@@ -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);
index febf9e76c440cc3ef4e3a7842632d54fbd17595b..d97f0bc0a26bc217cc30c263604356da6a424cef 100644 (file)
@@ -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;
 
@@ -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)
@@ -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;
 }
@@ -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;
@@ -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
@@ -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;
 }
@@ -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);
 }
@@ -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;
 }