diff options
| -rw-r--r-- | Documentation/hwmon/w83791d | 19 | ||||
| -rw-r--r-- | drivers/hwmon/w83791d.c | 148 |
2 files changed, 162 insertions, 5 deletions
diff --git a/Documentation/hwmon/w83791d b/Documentation/hwmon/w83791d index b1e4798764e8..5663e491655c 100644 --- a/Documentation/hwmon/w83791d +++ b/Documentation/hwmon/w83791d | |||
| @@ -77,6 +77,9 @@ readings can be divided by a programmable divider (1, 2, 4, 8, 16, | |||
| 77 | 77 | ||
| 78 | Each fan controlled is controlled by PWM. The PWM duty cycle can be read and | 78 | Each fan controlled is controlled by PWM. The PWM duty cycle can be read and |
| 79 | set for each fan separately. Valid values range from 0 (stop) to 255 (full). | 79 | set for each fan separately. Valid values range from 0 (stop) to 255 (full). |
| 80 | PWM 1-3 support Thermal Cruise mode, in which the PWMs are automatically | ||
| 81 | regulated to keep respectively temp 1-3 at a certain target temperature. | ||
| 82 | See below for the description of the sysfs-interface. | ||
| 80 | 83 | ||
| 81 | The w83791d has a global bit used to enable beeping from the speaker when an | 84 | The w83791d has a global bit used to enable beeping from the speaker when an |
| 82 | alarm is triggered as well as a bitmask to enable or disable the beep for | 85 | alarm is triggered as well as a bitmask to enable or disable the beep for |
| @@ -116,9 +119,19 @@ chip-specific options are documented here. | |||
| 116 | pwm[1-3]_enable - this file controls mode of fan/temperature control for | 119 | pwm[1-3]_enable - this file controls mode of fan/temperature control for |
| 117 | fan 1-3. Fan/PWM 4-5 only support manual mode. | 120 | fan 1-3. Fan/PWM 4-5 only support manual mode. |
| 118 | * 1 Manual mode | 121 | * 1 Manual mode |
| 119 | * 2 Thermal Cruise mode (no further support) | 122 | * 2 Thermal Cruise mode |
| 120 | * 3 Fan Speed Cruise mode (no further support) | 123 | * 3 Fan Speed Cruise mode (no further support) |
| 121 | 124 | ||
| 125 | temp[1-3]_target - defines the target temperature for Thermal Cruise mode. | ||
| 126 | Unit: millidegree Celsius | ||
| 127 | RW | ||
| 128 | |||
| 129 | temp[1-3]_tolerance - temperature tolerance for Thermal Cruise mode. | ||
| 130 | Specifies an interval around the target temperature | ||
| 131 | in which the fan speed is not changed. | ||
| 132 | Unit: millidegree Celsius | ||
| 133 | RW | ||
| 134 | |||
| 122 | Alarms bitmap vs. beep_mask bitmask | 135 | Alarms bitmap vs. beep_mask bitmask |
| 123 | ------------------------------------ | 136 | ------------------------------------ |
| 124 | For legacy code using the alarms and beep_mask files: | 137 | For legacy code using the alarms and beep_mask files: |
| @@ -146,7 +159,3 @@ tart2 : alarms: 0x020000 beep_mask: 0x080000 <== mismatch | |||
| 146 | tart3 : alarms: 0x040000 beep_mask: 0x100000 <== mismatch | 159 | tart3 : alarms: 0x040000 beep_mask: 0x100000 <== mismatch |
| 147 | case_open : alarms: 0x001000 beep_mask: 0x001000 | 160 | case_open : alarms: 0x001000 beep_mask: 0x001000 |
| 148 | global_enable: alarms: -------- beep_mask: 0x800000 (modified via beep_enable) | 161 | global_enable: alarms: -------- beep_mask: 0x800000 (modified via beep_enable) |
| 149 | |||
| 150 | W83791D TODO: | ||
| 151 | --------------- | ||
| 152 | Provide a patch for Thermal Cruise registers. | ||
diff --git a/drivers/hwmon/w83791d.c b/drivers/hwmon/w83791d.c index a4d2b02d9e04..5768def8a4f2 100644 --- a/drivers/hwmon/w83791d.c +++ b/drivers/hwmon/w83791d.c | |||
| @@ -125,6 +125,17 @@ static const u8 W83791D_REG_PWM[NUMBER_OF_PWM] = { | |||
| 125 | 0xA1, /* PWM 5 duty cycle register in DataSheet */ | 125 | 0xA1, /* PWM 5 duty cycle register in DataSheet */ |
| 126 | }; | 126 | }; |
| 127 | 127 | ||
| 128 | static const u8 W83791D_REG_TEMP_TARGET[3] = { | ||
| 129 | 0x85, /* PWM 1 target temperature for temp 1 */ | ||
| 130 | 0x86, /* PWM 2 target temperature for temp 2 */ | ||
| 131 | 0x96, /* PWM 3 target temperature for temp 3 */ | ||
| 132 | }; | ||
| 133 | |||
| 134 | static const u8 W83791D_REG_TEMP_TOL[2] = { | ||
| 135 | 0x87, /* PWM 1/2 temperature tolerance */ | ||
| 136 | 0x97, /* PWM 3 temperature tolerance */ | ||
| 137 | }; | ||
| 138 | |||
| 128 | static const u8 W83791D_REG_FAN_CFG[2] = { | 139 | static const u8 W83791D_REG_FAN_CFG[2] = { |
| 129 | 0x84, /* FAN 1/2 configuration */ | 140 | 0x84, /* FAN 1/2 configuration */ |
| 130 | 0x95, /* FAN 3 configuration */ | 141 | 0x95, /* FAN 3 configuration */ |
| @@ -234,6 +245,15 @@ static u8 fan_to_reg(long rpm, int div) | |||
| 234 | (val) < 0 ? ((val) - 250) / 500 * 128 : \ | 245 | (val) < 0 ? ((val) - 250) / 500 * 128 : \ |
| 235 | ((val) + 250) / 500 * 128) | 246 | ((val) + 250) / 500 * 128) |
| 236 | 247 | ||
| 248 | /* for thermal cruise target temp, 7-bits, LSB = 1 degree Celsius */ | ||
| 249 | #define TARGET_TEMP_TO_REG(val) ((val) < 0 ? 0 : \ | ||
| 250 | (val) >= 127000 ? 127 : \ | ||
| 251 | ((val) + 500) / 1000) | ||
| 252 | |||
| 253 | /* for thermal cruise temp tolerance, 4-bits, LSB = 1 degree Celsius */ | ||
| 254 | #define TOL_TEMP_TO_REG(val) ((val) < 0 ? 0 : \ | ||
| 255 | (val) >= 15000 ? 15 : \ | ||
| 256 | ((val) + 500) / 1000) | ||
| 237 | 257 | ||
| 238 | #define BEEP_MASK_TO_REG(val) ((val) & 0xffffff) | 258 | #define BEEP_MASK_TO_REG(val) ((val) & 0xffffff) |
| 239 | #define BEEP_MASK_FROM_REG(val) ((val) & 0xffffff) | 259 | #define BEEP_MASK_FROM_REG(val) ((val) & 0xffffff) |
| @@ -290,6 +310,9 @@ struct w83791d_data { | |||
| 290 | u8 pwm_enable[3]; /* pwm enable status for fan 1-3 | 310 | u8 pwm_enable[3]; /* pwm enable status for fan 1-3 |
| 291 | (fan 4-5 only support manual mode) */ | 311 | (fan 4-5 only support manual mode) */ |
| 292 | 312 | ||
| 313 | u8 temp_target[3]; /* pwm 1-3 target temperature */ | ||
| 314 | u8 temp_tolerance[3]; /* pwm 1-3 temperature tolerance */ | ||
| 315 | |||
| 293 | /* Misc */ | 316 | /* Misc */ |
| 294 | u32 alarms; /* realtime status register encoding,combined */ | 317 | u32 alarms; /* realtime status register encoding,combined */ |
| 295 | u8 beep_enable; /* Global beep enable */ | 318 | u8 beep_enable; /* Global beep enable */ |
| @@ -774,6 +797,110 @@ static struct sensor_device_attribute sda_pwmenable[] = { | |||
| 774 | show_pwmenable, store_pwmenable, 2), | 797 | show_pwmenable, store_pwmenable, 2), |
| 775 | }; | 798 | }; |
| 776 | 799 | ||
| 800 | /* For Smart Fan I / Thermal Cruise */ | ||
| 801 | static ssize_t show_temp_target(struct device *dev, | ||
| 802 | struct device_attribute *attr, char *buf) | ||
| 803 | { | ||
| 804 | struct sensor_device_attribute *sensor_attr = to_sensor_dev_attr(attr); | ||
| 805 | struct w83791d_data *data = w83791d_update_device(dev); | ||
| 806 | int nr = sensor_attr->index; | ||
| 807 | return sprintf(buf, "%d\n", TEMP1_FROM_REG(data->temp_target[nr])); | ||
| 808 | } | ||
| 809 | |||
| 810 | static ssize_t store_temp_target(struct device *dev, | ||
| 811 | struct device_attribute *attr, const char *buf, size_t count) | ||
| 812 | { | ||
| 813 | struct sensor_device_attribute *sensor_attr = to_sensor_dev_attr(attr); | ||
| 814 | struct i2c_client *client = to_i2c_client(dev); | ||
| 815 | struct w83791d_data *data = i2c_get_clientdata(client); | ||
| 816 | int nr = sensor_attr->index; | ||
| 817 | unsigned long val; | ||
| 818 | u8 target_mask; | ||
| 819 | |||
| 820 | if (strict_strtoul(buf, 10, &val)) | ||
| 821 | return -EINVAL; | ||
| 822 | |||
| 823 | mutex_lock(&data->update_lock); | ||
| 824 | data->temp_target[nr] = TARGET_TEMP_TO_REG(val); | ||
| 825 | target_mask = w83791d_read(client, | ||
| 826 | W83791D_REG_TEMP_TARGET[nr]) & 0x80; | ||
| 827 | w83791d_write(client, W83791D_REG_TEMP_TARGET[nr], | ||
| 828 | data->temp_target[nr] | target_mask); | ||
| 829 | mutex_unlock(&data->update_lock); | ||
| 830 | return count; | ||
| 831 | } | ||
| 832 | |||
| 833 | static struct sensor_device_attribute sda_temp_target[] = { | ||
| 834 | SENSOR_ATTR(temp1_target, S_IWUSR | S_IRUGO, | ||
| 835 | show_temp_target, store_temp_target, 0), | ||
| 836 | SENSOR_ATTR(temp2_target, S_IWUSR | S_IRUGO, | ||
| 837 | show_temp_target, store_temp_target, 1), | ||
| 838 | SENSOR_ATTR(temp3_target, S_IWUSR | S_IRUGO, | ||
| 839 | show_temp_target, store_temp_target, 2), | ||
| 840 | }; | ||
| 841 | |||
| 842 | static ssize_t show_temp_tolerance(struct device *dev, | ||
| 843 | struct device_attribute *attr, char *buf) | ||
| 844 | { | ||
| 845 | struct sensor_device_attribute *sensor_attr = to_sensor_dev_attr(attr); | ||
| 846 | struct w83791d_data *data = w83791d_update_device(dev); | ||
| 847 | int nr = sensor_attr->index; | ||
| 848 | return sprintf(buf, "%d\n", TEMP1_FROM_REG(data->temp_tolerance[nr])); | ||
| 849 | } | ||
| 850 | |||
| 851 | static ssize_t store_temp_tolerance(struct device *dev, | ||
| 852 | struct device_attribute *attr, const char *buf, size_t count) | ||
| 853 | { | ||
| 854 | struct sensor_device_attribute *sensor_attr = to_sensor_dev_attr(attr); | ||
| 855 | struct i2c_client *client = to_i2c_client(dev); | ||
| 856 | struct w83791d_data *data = i2c_get_clientdata(client); | ||
| 857 | int nr = sensor_attr->index; | ||
| 858 | unsigned long val; | ||
| 859 | u8 target_mask; | ||
| 860 | u8 reg_idx = 0; | ||
| 861 | u8 val_shift = 0; | ||
| 862 | u8 keep_mask = 0; | ||
| 863 | |||
| 864 | if (strict_strtoul(buf, 10, &val)) | ||
| 865 | return -EINVAL; | ||
| 866 | |||
| 867 | switch (nr) { | ||
| 868 | case 0: | ||
| 869 | reg_idx = 0; | ||
| 870 | val_shift = 0; | ||
| 871 | keep_mask = 0xf0; | ||
| 872 | break; | ||
| 873 | case 1: | ||
| 874 | reg_idx = 0; | ||
| 875 | val_shift = 4; | ||
| 876 | keep_mask = 0x0f; | ||
| 877 | break; | ||
| 878 | case 2: | ||
| 879 | reg_idx = 1; | ||
| 880 | val_shift = 0; | ||
| 881 | keep_mask = 0xf0; | ||
| 882 | break; | ||
| 883 | } | ||
| 884 | |||
| 885 | mutex_lock(&data->update_lock); | ||
| 886 | data->temp_tolerance[nr] = TOL_TEMP_TO_REG(val); | ||
| 887 | target_mask = w83791d_read(client, | ||
| 888 | W83791D_REG_TEMP_TOL[reg_idx]) & keep_mask; | ||
| 889 | w83791d_write(client, W83791D_REG_TEMP_TOL[reg_idx], | ||
| 890 | (data->temp_tolerance[nr] << val_shift) | target_mask); | ||
| 891 | mutex_unlock(&data->update_lock); | ||
| 892 | return count; | ||
| 893 | } | ||
| 894 | |||
| 895 | static struct sensor_device_attribute sda_temp_tolerance[] = { | ||
| 896 | SENSOR_ATTR(temp1_tolerance, S_IWUSR | S_IRUGO, | ||
| 897 | show_temp_tolerance, store_temp_tolerance, 0), | ||
| 898 | SENSOR_ATTR(temp2_tolerance, S_IWUSR | S_IRUGO, | ||
| 899 | show_temp_tolerance, store_temp_tolerance, 1), | ||
| 900 | SENSOR_ATTR(temp3_tolerance, S_IWUSR | S_IRUGO, | ||
| 901 | show_temp_tolerance, store_temp_tolerance, 2), | ||
| 902 | }; | ||
| 903 | |||
| 777 | /* read/write the temperature1, includes measured value and limits */ | 904 | /* read/write the temperature1, includes measured value and limits */ |
| 778 | static ssize_t show_temp1(struct device *dev, struct device_attribute *devattr, | 905 | static ssize_t show_temp1(struct device *dev, struct device_attribute *devattr, |
| 779 | char *buf) | 906 | char *buf) |
| @@ -1044,6 +1171,12 @@ static struct attribute *w83791d_attributes[] = { | |||
| 1044 | &sda_pwmenable[0].dev_attr.attr, | 1171 | &sda_pwmenable[0].dev_attr.attr, |
| 1045 | &sda_pwmenable[1].dev_attr.attr, | 1172 | &sda_pwmenable[1].dev_attr.attr, |
| 1046 | &sda_pwmenable[2].dev_attr.attr, | 1173 | &sda_pwmenable[2].dev_attr.attr, |
| 1174 | &sda_temp_target[0].dev_attr.attr, | ||
| 1175 | &sda_temp_target[1].dev_attr.attr, | ||
| 1176 | &sda_temp_target[2].dev_attr.attr, | ||
| 1177 | &sda_temp_tolerance[0].dev_attr.attr, | ||
| 1178 | &sda_temp_tolerance[1].dev_attr.attr, | ||
| 1179 | &sda_temp_tolerance[2].dev_attr.attr, | ||
| 1047 | NULL | 1180 | NULL |
| 1048 | }; | 1181 | }; |
| 1049 | 1182 | ||
| @@ -1404,6 +1537,21 @@ static struct w83791d_data *w83791d_update_device(struct device *dev) | |||
| 1404 | data->pwm_enable[1] = (reg_array_tmp[0] >> 4) & 0x03; | 1537 | data->pwm_enable[1] = (reg_array_tmp[0] >> 4) & 0x03; |
| 1405 | data->pwm_enable[2] = (reg_array_tmp[1] >> 2) & 0x03; | 1538 | data->pwm_enable[2] = (reg_array_tmp[1] >> 2) & 0x03; |
| 1406 | 1539 | ||
| 1540 | /* Update PWM target temperature */ | ||
| 1541 | for (i = 0; i < 3; i++) { | ||
| 1542 | data->temp_target[i] = w83791d_read(client, | ||
| 1543 | W83791D_REG_TEMP_TARGET[i]) & 0x7f; | ||
| 1544 | } | ||
| 1545 | |||
| 1546 | /* Update PWM temperature tolerance */ | ||
| 1547 | for (i = 0; i < 2; i++) { | ||
| 1548 | reg_array_tmp[i] = w83791d_read(client, | ||
| 1549 | W83791D_REG_TEMP_TOL[i]); | ||
| 1550 | } | ||
| 1551 | data->temp_tolerance[0] = reg_array_tmp[0] & 0x0f; | ||
| 1552 | data->temp_tolerance[1] = (reg_array_tmp[0] >> 4) & 0x0f; | ||
| 1553 | data->temp_tolerance[2] = reg_array_tmp[1] & 0x0f; | ||
| 1554 | |||
| 1407 | /* Update the first temperature sensor */ | 1555 | /* Update the first temperature sensor */ |
| 1408 | for (i = 0; i < 3; i++) { | 1556 | for (i = 0; i < 3; i++) { |
| 1409 | data->temp1[i] = w83791d_read(client, | 1557 | data->temp1[i] = w83791d_read(client, |
