platform/x86: alienware-wmi-wmax: Add support for manual fan control

All models with the "AWCC" WMAX device support a way of manually
controlling fans.

The PWM duty cycle of a fan can't be controlled directly. Instead the
AWCC interface let's us tune a fan `boost` value, which has the
following empirically discovered, approximate behavior over the PWM
value:

	pwm = pwm_base + (fan_boost / 255) * (pwm_max - pwm_base)

Where the pwm_base is the locked PWM value controlled by the FW and
fan_boost is a value between 0 and 255.

Expose this fan_boost knob as a custom HWMON attribute.

Cc: Guenter Roeck <linux@roeck-us.net>
Cc: Jean Delvare <jdelvare@suse.com>
Cc: linux-hwmon@vger.kernel.org
Reviewed-by: Armin Wolf <W_Armin@gmx.de>
Signed-off-by: Kurt Borja <kuurtb@gmail.com>
Link: https://lore.kernel.org/r/20250329-hwm-v7-8-a14ea39d8a94@gmail.com
Signed-off-by: Ilpo Järvinen <ilpo.jarvinen@linux.intel.com>
This commit is contained in:
Kurt Borja 2025-03-29 04:32:25 -03:00 committed by Ilpo Järvinen
parent d699907834
commit 07ac275981
No known key found for this signature in database
GPG Key ID: 59AC4F6153E5CE31
1 changed files with 172 additions and 2 deletions

View File

@ -14,8 +14,12 @@
#include <linux/bits.h> #include <linux/bits.h>
#include <linux/dmi.h> #include <linux/dmi.h>
#include <linux/hwmon.h> #include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#include <linux/kstrtox.h>
#include <linux/minmax.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/platform_profile.h> #include <linux/platform_profile.h>
#include <linux/pm.h>
#include <linux/units.h> #include <linux/units.h>
#include <linux/wmi.h> #include <linux/wmi.h>
#include "alienware-wmi.h" #include "alienware-wmi.h"
@ -183,10 +187,12 @@ enum AWCC_THERMAL_INFORMATION_OPERATIONS {
AWCC_OP_GET_FAN_MIN_RPM = 0x08, AWCC_OP_GET_FAN_MIN_RPM = 0x08,
AWCC_OP_GET_FAN_MAX_RPM = 0x09, AWCC_OP_GET_FAN_MAX_RPM = 0x09,
AWCC_OP_GET_CURRENT_PROFILE = 0x0B, AWCC_OP_GET_CURRENT_PROFILE = 0x0B,
AWCC_OP_GET_FAN_BOOST = 0x0C,
}; };
enum AWCC_THERMAL_CONTROL_OPERATIONS { enum AWCC_THERMAL_CONTROL_OPERATIONS {
AWCC_OP_ACTIVATE_PROFILE = 0x01, AWCC_OP_ACTIVATE_PROFILE = 0x01,
AWCC_OP_SET_FAN_BOOST = 0x02,
}; };
enum AWCC_GAME_SHIFT_STATUS_OPERATIONS { enum AWCC_GAME_SHIFT_STATUS_OPERATIONS {
@ -250,6 +256,7 @@ struct awcc_fan_data {
const char *label; const char *label;
u32 min_rpm; u32 min_rpm;
u32 max_rpm; u32 max_rpm;
u8 suspend_cache;
u8 id; u8 id;
}; };
@ -638,6 +645,18 @@ static int awcc_op_get_temperature(struct wmi_device *wdev, u8 temp_id, u32 *out
return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_INFORMATION, &args, out); return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_INFORMATION, &args, out);
} }
static int awcc_op_get_fan_boost(struct wmi_device *wdev, u8 fan_id, u32 *out)
{
struct wmax_u32_args args = {
.operation = AWCC_OP_GET_FAN_BOOST,
.arg1 = fan_id,
.arg2 = 0,
.arg3 = 0,
};
return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_INFORMATION, &args, out);
}
static int awcc_op_get_current_profile(struct wmi_device *wdev, u32 *out) static int awcc_op_get_current_profile(struct wmi_device *wdev, u32 *out)
{ {
struct wmax_u32_args args = { struct wmax_u32_args args = {
@ -663,6 +682,19 @@ static int awcc_op_activate_profile(struct wmi_device *wdev, u8 profile)
return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_CONTROL, &args, &out); return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_CONTROL, &args, &out);
} }
static int awcc_op_set_fan_boost(struct wmi_device *wdev, u8 fan_id, u8 boost)
{
struct wmax_u32_args args = {
.operation = AWCC_OP_SET_FAN_BOOST,
.arg1 = fan_id,
.arg2 = boost,
.arg3 = 0,
};
u32 out;
return awcc_wmi_command(wdev, AWCC_METHOD_THERMAL_CONTROL, &args, &out);
}
/* /*
* HWMON * HWMON
* - Provides temperature and fan speed monitoring as well as manual fan * - Provides temperature and fan speed monitoring as well as manual fan
@ -827,6 +859,81 @@ static const struct hwmon_chip_info awcc_hwmon_chip_info = {
.info = awcc_hwmon_info, .info = awcc_hwmon_info,
}; };
static ssize_t fan_boost_show(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct awcc_priv *priv = dev_get_drvdata(dev);
int index = to_sensor_dev_attr(attr)->index;
struct awcc_fan_data *fan = priv->fan_data[index];
u32 boost;
int ret;
ret = awcc_op_get_fan_boost(priv->wdev, fan->id, &boost);
if (ret)
return ret;
return sysfs_emit(buf, "%u\n", boost);
}
static ssize_t fan_boost_store(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
struct awcc_priv *priv = dev_get_drvdata(dev);
int index = to_sensor_dev_attr(attr)->index;
struct awcc_fan_data *fan = priv->fan_data[index];
unsigned long val;
int ret;
ret = kstrtoul(buf, 0, &val);
if (ret)
return ret;
ret = awcc_op_set_fan_boost(priv->wdev, fan->id, clamp_val(val, 0, 255));
return ret ? ret : count;
}
static SENSOR_DEVICE_ATTR_RW(fan1_boost, fan_boost, 0);
static SENSOR_DEVICE_ATTR_RW(fan2_boost, fan_boost, 1);
static SENSOR_DEVICE_ATTR_RW(fan3_boost, fan_boost, 2);
static SENSOR_DEVICE_ATTR_RW(fan4_boost, fan_boost, 3);
static SENSOR_DEVICE_ATTR_RW(fan5_boost, fan_boost, 4);
static SENSOR_DEVICE_ATTR_RW(fan6_boost, fan_boost, 5);
static umode_t fan_boost_attr_visible(struct kobject *kobj, struct attribute *attr, int n)
{
struct awcc_priv *priv = dev_get_drvdata(kobj_to_dev(kobj));
return n < priv->fan_count ? attr->mode : 0;
}
static bool fan_boost_group_visible(struct kobject *kobj)
{
return true;
}
DEFINE_SYSFS_GROUP_VISIBLE(fan_boost);
static struct attribute *fan_boost_attrs[] = {
&sensor_dev_attr_fan1_boost.dev_attr.attr,
&sensor_dev_attr_fan2_boost.dev_attr.attr,
&sensor_dev_attr_fan3_boost.dev_attr.attr,
&sensor_dev_attr_fan4_boost.dev_attr.attr,
&sensor_dev_attr_fan5_boost.dev_attr.attr,
&sensor_dev_attr_fan6_boost.dev_attr.attr,
NULL
};
static const struct attribute_group fan_boost_group = {
.attrs = fan_boost_attrs,
.is_visible = SYSFS_GROUP_VISIBLE(fan_boost),
};
static const struct attribute_group *awcc_hwmon_groups[] = {
&fan_boost_group,
NULL
};
static int awcc_hwmon_temps_init(struct wmi_device *wdev) static int awcc_hwmon_temps_init(struct wmi_device *wdev)
{ {
struct awcc_priv *priv = dev_get_drvdata(&wdev->dev); struct awcc_priv *priv = dev_get_drvdata(&wdev->dev);
@ -964,12 +1071,56 @@ static int awcc_hwmon_init(struct wmi_device *wdev)
if (ret) if (ret)
return ret; return ret;
priv->hwdev = devm_hwmon_device_register_with_info(&wdev->dev, "alienware_wmi", priv, priv->hwdev = devm_hwmon_device_register_with_info(&wdev->dev, "alienware_wmi",
&awcc_hwmon_chip_info, NULL); priv, &awcc_hwmon_chip_info,
awcc_hwmon_groups);
return PTR_ERR_OR_ZERO(priv->hwdev); return PTR_ERR_OR_ZERO(priv->hwdev);
} }
static void awcc_hwmon_suspend(struct device *dev)
{
struct awcc_priv *priv = dev_get_drvdata(dev);
struct awcc_fan_data *fan;
unsigned int i;
u32 boost;
int ret;
for (i = 0; i < priv->fan_count; i++) {
fan = priv->fan_data[i];
ret = awcc_thermal_information(priv->wdev, AWCC_OP_GET_FAN_BOOST,
fan->id, &boost);
if (ret)
dev_err(dev, "Failed to store Fan %u boost while suspending\n", i);
fan->suspend_cache = ret ? 0 : clamp_val(boost, 0, 255);
awcc_op_set_fan_boost(priv->wdev, fan->id, 0);
if (ret)
dev_err(dev, "Failed to set Fan %u boost to 0 while suspending\n", i);
}
}
static void awcc_hwmon_resume(struct device *dev)
{
struct awcc_priv *priv = dev_get_drvdata(dev);
struct awcc_fan_data *fan;
unsigned int i;
int ret;
for (i = 0; i < priv->fan_count; i++) {
fan = priv->fan_data[i];
if (!fan->suspend_cache)
continue;
ret = awcc_op_set_fan_boost(priv->wdev, fan->id, fan->suspend_cache);
if (ret)
dev_err(dev, "Failed to restore Fan %u boost while resuming\n", i);
}
}
/* /*
* Thermal Profile control * Thermal Profile control
* - Provides thermal profile control through the Platform Profile API * - Provides thermal profile control through the Platform Profile API
@ -1196,6 +1347,24 @@ static int wmax_wmi_probe(struct wmi_device *wdev, const void *context)
return ret; return ret;
} }
static int wmax_wmi_suspend(struct device *dev)
{
if (awcc->hwmon)
awcc_hwmon_suspend(dev);
return 0;
}
static int wmax_wmi_resume(struct device *dev)
{
if (awcc->hwmon)
awcc_hwmon_resume(dev);
return 0;
}
static DEFINE_SIMPLE_DEV_PM_OPS(wmax_wmi_pm_ops, wmax_wmi_suspend, wmax_wmi_resume);
static const struct wmi_device_id alienware_wmax_device_id_table[] = { static const struct wmi_device_id alienware_wmax_device_id_table[] = {
{ WMAX_CONTROL_GUID, NULL }, { WMAX_CONTROL_GUID, NULL },
{ }, { },
@ -1206,6 +1375,7 @@ static struct wmi_driver alienware_wmax_wmi_driver = {
.driver = { .driver = {
.name = "alienware-wmi-wmax", .name = "alienware-wmi-wmax",
.probe_type = PROBE_PREFER_ASYNCHRONOUS, .probe_type = PROBE_PREFER_ASYNCHRONOUS,
.pm = pm_sleep_ptr(&wmax_wmi_pm_ops),
}, },
.id_table = alienware_wmax_device_id_table, .id_table = alienware_wmax_device_id_table,
.probe = wmax_wmi_probe, .probe = wmax_wmi_probe,