Commit ecf2a80a authored by Henrique de Moraes Holschuh's avatar Henrique de Moraes Holschuh Committed by Len Brown

ACPI: thinkpad-acpi: add a fan-control feature master toggle

Len Brown considers that an active by default fan control interface in
laptops may be too close to giving users enough rope.  There is a good
chance he is quite correct on this, especially if someone decides to use
that interface in applets and users are not aware of its risks.

This patch adds a master switch to thinkpad-acpi that enables or disables
the entire fan-control feature as a module parameter: "fan_control".  It
defaults to disabled.  Set it to non-zero to enable fan control.

Also, the patch removes the expermiental status from fan control, since it
is stable enough to not be called experimental, and the master switch makes
it safe enough to do so.
Signed-off-by: default avatarHenrique de Moraes Holschuh <hmh@hmh.eng.br>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent 7d5a015e
...@@ -38,7 +38,7 @@ detailed description): ...@@ -38,7 +38,7 @@ detailed description):
- Experimental: embedded controller register dump - Experimental: embedded controller register dump
- LCD brightness control - LCD brightness control
- Volume control - Volume control
- Experimental: fan speed, fan enable/disable - Fan control and monitoring: fan speed, fan enable/disable
- Experimental: WAN enable and disable - Experimental: WAN enable and disable
A compatibility table by model and feature is maintained on the web A compatibility table by model and feature is maintained on the web
...@@ -681,21 +681,20 @@ distinct. The unmute the volume after the mute command, use either the ...@@ -681,21 +681,20 @@ distinct. The unmute the volume after the mute command, use either the
up or down command (the level command will not unmute the volume). up or down command (the level command will not unmute the volume).
The current volume level and mute state is shown in the file. The current volume level and mute state is shown in the file.
EXPERIMENTAL: fan speed, fan enable/disable Fan control and monitoring: fan speed, fan enable/disable
------------------------------------------- ---------------------------------------------------------
procfs: /proc/acpi/ibm/fan procfs: /proc/acpi/ibm/fan
sysfs device attributes: (hwmon) fan_input, pwm1, pwm1_enable sysfs device attributes: (hwmon) fan_input, pwm1, pwm1_enable
This feature is marked EXPERIMENTAL because the implementation NOTE NOTE NOTE: fan control operations are disabled by default for
directly accesses hardware registers and may not work as expected. USE safety reasons. To enable them, the module parameter "fan_control=1"
WITH CAUTION! To use this feature, you need to supply the must be given to thinkpad-acpi.
experimental=1 parameter when loading the module.
This feature attempts to show the current fan speed, control mode and This feature attempts to show the current fan speed, control mode and
other fan data that might be available. The speed is read directly other fan data that might be available. The speed is read directly
from the hardware registers of the embedded controller. This is known from the hardware registers of the embedded controller. This is known
to work on later R, T and X series ThinkPads but may show a bogus to work on later R, T, X and Z series ThinkPads but may show a bogus
value on other models. value on other models.
Fan levels: Fan levels:
......
...@@ -2935,6 +2935,9 @@ static ssize_t fan_fan_watchdog_store(struct device_driver *drv, ...@@ -2935,6 +2935,9 @@ static ssize_t fan_fan_watchdog_store(struct device_driver *drv,
if (parse_strtoul(buf, 120, &t)) if (parse_strtoul(buf, 120, &t))
return -EINVAL; return -EINVAL;
if (!fan_control_allowed)
return -EPERM;
fan_watchdog_maxinterval = t; fan_watchdog_maxinterval = t;
fan_watchdog_reset(); fan_watchdog_reset();
...@@ -3046,6 +3049,14 @@ static int __init fan_init(struct ibm_init_struct *iibm) ...@@ -3046,6 +3049,14 @@ static int __init fan_init(struct ibm_init_struct *iibm)
fan_control_access_mode != TPACPI_FAN_WR_NONE), fan_control_access_mode != TPACPI_FAN_WR_NONE),
fan_status_access_mode, fan_control_access_mode); fan_status_access_mode, fan_control_access_mode);
/* fan control master switch */
if (!fan_control_allowed) {
fan_control_access_mode = TPACPI_FAN_WR_NONE;
fan_control_commands = 0;
dbg_printk(TPACPI_DBG_INIT,
"fan control features disabled by parameter\n");
}
/* update fan_control_desired_level */ /* update fan_control_desired_level */
if (fan_status_access_mode != TPACPI_FAN_NONE) if (fan_status_access_mode != TPACPI_FAN_NONE)
fan_get_status_safe(NULL); fan_get_status_safe(NULL);
...@@ -3203,6 +3214,9 @@ static void fan_watchdog_reset(void) ...@@ -3203,6 +3214,9 @@ static void fan_watchdog_reset(void)
static int fan_set_level(int level) static int fan_set_level(int level)
{ {
if (!fan_control_allowed)
return -EPERM;
switch (fan_control_access_mode) { switch (fan_control_access_mode) {
case TPACPI_FAN_WR_ACPI_SFAN: case TPACPI_FAN_WR_ACPI_SFAN:
if (level >= 0 && level <= 7) { if (level >= 0 && level <= 7) {
...@@ -3242,6 +3256,9 @@ static int fan_set_level_safe(int level) ...@@ -3242,6 +3256,9 @@ static int fan_set_level_safe(int level)
{ {
int rc; int rc;
if (!fan_control_allowed)
return -EPERM;
rc = mutex_lock_interruptible(&fan_mutex); rc = mutex_lock_interruptible(&fan_mutex);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -3262,6 +3279,9 @@ static int fan_set_enable(void) ...@@ -3262,6 +3279,9 @@ static int fan_set_enable(void)
u8 s; u8 s;
int rc; int rc;
if (!fan_control_allowed)
return -EPERM;
rc = mutex_lock_interruptible(&fan_mutex); rc = mutex_lock_interruptible(&fan_mutex);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -3315,6 +3335,9 @@ static int fan_set_disable(void) ...@@ -3315,6 +3335,9 @@ static int fan_set_disable(void)
{ {
int rc; int rc;
if (!fan_control_allowed)
return -EPERM;
rc = mutex_lock_interruptible(&fan_mutex); rc = mutex_lock_interruptible(&fan_mutex);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -3351,6 +3374,9 @@ static int fan_set_speed(int speed) ...@@ -3351,6 +3374,9 @@ static int fan_set_speed(int speed)
{ {
int rc; int rc;
if (!fan_control_allowed)
return -EPERM;
rc = mutex_lock_interruptible(&fan_mutex); rc = mutex_lock_interruptible(&fan_mutex);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -3558,7 +3584,6 @@ static struct ibm_struct fan_driver_data = { ...@@ -3558,7 +3584,6 @@ static struct ibm_struct fan_driver_data = {
.read = fan_read, .read = fan_read,
.write = fan_write, .write = fan_write,
.exit = fan_exit, .exit = fan_exit,
.flags.experimental = 1,
}; };
/**************************************************************************** /****************************************************************************
...@@ -3879,6 +3904,9 @@ module_param_named(debug, dbg_level, uint, 0); ...@@ -3879,6 +3904,9 @@ module_param_named(debug, dbg_level, uint, 0);
static int force_load; static int force_load;
module_param(force_load, int, 0); module_param(force_load, int, 0);
static int fan_control_allowed;
module_param_named(fan_control, fan_control_allowed, int, 0);
#define IBM_PARAM(feature) \ #define IBM_PARAM(feature) \
module_param_call(feature, set_ibm_param, NULL, NULL, 0) module_param_call(feature, set_ibm_param, NULL, NULL, 0)
......
...@@ -375,6 +375,8 @@ enum fan_control_commands { ...@@ -375,6 +375,8 @@ enum fan_control_commands {
* and also watchdog cmd */ * and also watchdog cmd */
}; };
static int fan_control_allowed;
static enum fan_status_access_mode fan_status_access_mode; static enum fan_status_access_mode fan_status_access_mode;
static enum fan_control_access_mode fan_control_access_mode; static enum fan_control_access_mode fan_control_access_mode;
static enum fan_control_commands fan_control_commands; static enum fan_control_commands fan_control_commands;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment