Commit 3b43f416 authored by Guenter Roeck's avatar Guenter Roeck Committed by Greg Kroah-Hartman

hwmon: (it87) Do not overwrite bit 2..6 of pwm control registers

commit 4c7b8ca1 upstream.

In IT8620E, after setting pwm control to manual, it was observed that
pwm values for fan 4..6 have reversed results (writing 0 results in fans
running at full speed, writing 255 results in fans turned off).

With the new PWM control, pwm polarity for pwm control 4..6 is specified
in its pwm control registers. Those registers are overwritten when setting
the pwm mode or the temperature mapping. Do not touch bit 2..6 of pwm
control registers on register writes to fix the problem.
Signed-off-by: default avatarGuenter Roeck <linux@roeck-us.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 3d2c16ca
...@@ -1300,25 +1300,35 @@ static ssize_t set_pwm_enable(struct device *dev, struct device_attribute *attr, ...@@ -1300,25 +1300,35 @@ static ssize_t set_pwm_enable(struct device *dev, struct device_attribute *attr,
it87_write_value(data, IT87_REG_FAN_MAIN_CTRL, it87_write_value(data, IT87_REG_FAN_MAIN_CTRL,
data->fan_main_ctrl); data->fan_main_ctrl);
} else { } else {
u8 ctrl;
/* No on/off mode, set maximum pwm value */ /* No on/off mode, set maximum pwm value */
data->pwm_duty[nr] = pwm_to_reg(data, 0xff); data->pwm_duty[nr] = pwm_to_reg(data, 0xff);
it87_write_value(data, IT87_REG_PWM_DUTY[nr], it87_write_value(data, IT87_REG_PWM_DUTY[nr],
data->pwm_duty[nr]); data->pwm_duty[nr]);
/* and set manual mode */ /* and set manual mode */
data->pwm_ctrl[nr] = has_newer_autopwm(data) ? if (has_newer_autopwm(data)) {
data->pwm_temp_map[nr] : ctrl = (data->pwm_ctrl[nr] & 0x7c) |
data->pwm_duty[nr]; data->pwm_temp_map[nr];
it87_write_value(data, IT87_REG_PWM[nr], } else {
data->pwm_ctrl[nr]); ctrl = data->pwm_duty[nr];
}
data->pwm_ctrl[nr] = ctrl;
it87_write_value(data, IT87_REG_PWM[nr], ctrl);
} }
} else { } else {
if (val == 1) /* Manual mode */ u8 ctrl;
data->pwm_ctrl[nr] = has_newer_autopwm(data) ?
data->pwm_temp_map[nr] : if (has_newer_autopwm(data)) {
data->pwm_duty[nr]; ctrl = (data->pwm_ctrl[nr] & 0x7c) |
else /* Automatic mode */ data->pwm_temp_map[nr];
data->pwm_ctrl[nr] = 0x80 | data->pwm_temp_map[nr]; if (val != 1)
it87_write_value(data, IT87_REG_PWM[nr], data->pwm_ctrl[nr]); ctrl |= 0x80;
} else {
ctrl = (val == 1 ? data->pwm_duty[nr] : 0x80);
}
data->pwm_ctrl[nr] = ctrl;
it87_write_value(data, IT87_REG_PWM[nr], ctrl);
if (data->type != it8603 && nr < 3) { if (data->type != it8603 && nr < 3) {
/* set SmartGuardian mode */ /* set SmartGuardian mode */
...@@ -1462,7 +1472,8 @@ static ssize_t set_pwm_temp_map(struct device *dev, ...@@ -1462,7 +1472,8 @@ static ssize_t set_pwm_temp_map(struct device *dev,
* otherwise, just store it for later use. * otherwise, just store it for later use.
*/ */
if (data->pwm_ctrl[nr] & 0x80) { if (data->pwm_ctrl[nr] & 0x80) {
data->pwm_ctrl[nr] = 0x80 | data->pwm_temp_map[nr]; data->pwm_ctrl[nr] = (data->pwm_ctrl[nr] & 0xfc) |
data->pwm_temp_map[nr];
it87_write_value(data, IT87_REG_PWM[nr], data->pwm_ctrl[nr]); it87_write_value(data, IT87_REG_PWM[nr], data->pwm_ctrl[nr]);
} }
mutex_unlock(&data->update_lock); mutex_unlock(&data->update_lock);
......
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