diff options
| -rw-r--r-- | drivers/hwmon/adt7475.c | 114 |
1 files changed, 99 insertions, 15 deletions
diff --git a/drivers/hwmon/adt7475.c b/drivers/hwmon/adt7475.c index 716dae9ff0f7..d20130723d68 100644 --- a/drivers/hwmon/adt7475.c +++ b/drivers/hwmon/adt7475.c | |||
| @@ -80,6 +80,8 @@ | |||
| 80 | 80 | ||
| 81 | #define REG_EXTEND1 0x76 | 81 | #define REG_EXTEND1 0x76 |
| 82 | #define REG_EXTEND2 0x77 | 82 | #define REG_EXTEND2 0x77 |
| 83 | |||
| 84 | #define REG_CONFIG3 0x78 | ||
| 83 | #define REG_CONFIG5 0x7C | 85 | #define REG_CONFIG5 0x7C |
| 84 | #define REG_CONFIG4 0x7D | 86 | #define REG_CONFIG4 0x7D |
| 85 | 87 | ||
| @@ -88,6 +90,10 @@ | |||
| 88 | #define REG_VTT_MIN 0x84 /* ADT7490 only */ | 90 | #define REG_VTT_MIN 0x84 /* ADT7490 only */ |
| 89 | #define REG_VTT_MAX 0x86 /* ADT7490 only */ | 91 | #define REG_VTT_MAX 0x86 /* ADT7490 only */ |
| 90 | 92 | ||
| 93 | #define CONFIG3_SMBALERT 0x01 | ||
| 94 | #define CONFIG3_THERM 0x02 | ||
| 95 | |||
| 96 | #define CONFIG4_PINFUNC 0x03 | ||
| 91 | #define CONFIG4_MAXDUTY 0x08 | 97 | #define CONFIG4_MAXDUTY 0x08 |
| 92 | 98 | ||
| 93 | #define CONFIG5_TWOSCOMP 0x01 | 99 | #define CONFIG5_TWOSCOMP 0x01 |
| @@ -149,6 +155,8 @@ struct adt7475_data { | |||
| 149 | u8 config4; | 155 | u8 config4; |
| 150 | u8 config5; | 156 | u8 config5; |
| 151 | u8 has_voltage; | 157 | u8 has_voltage; |
| 158 | u8 has_pwm2:1; | ||
| 159 | u8 has_fan4:1; | ||
| 152 | u32 alarms; | 160 | u32 alarms; |
| 153 | u16 voltage[3][6]; | 161 | u16 voltage[3][6]; |
| 154 | u16 temp[7][3]; | 162 | u16 temp[7][3]; |
| @@ -1025,21 +1033,12 @@ static struct attribute *adt7475_attrs[] = { | |||
| 1025 | &sensor_dev_attr_fan3_input.dev_attr.attr, | 1033 | &sensor_dev_attr_fan3_input.dev_attr.attr, |
| 1026 | &sensor_dev_attr_fan3_min.dev_attr.attr, | 1034 | &sensor_dev_attr_fan3_min.dev_attr.attr, |
| 1027 | &sensor_dev_attr_fan3_alarm.dev_attr.attr, | 1035 | &sensor_dev_attr_fan3_alarm.dev_attr.attr, |
| 1028 | &sensor_dev_attr_fan4_input.dev_attr.attr, | ||
| 1029 | &sensor_dev_attr_fan4_min.dev_attr.attr, | ||
| 1030 | &sensor_dev_attr_fan4_alarm.dev_attr.attr, | ||
| 1031 | &sensor_dev_attr_pwm1.dev_attr.attr, | 1036 | &sensor_dev_attr_pwm1.dev_attr.attr, |
| 1032 | &sensor_dev_attr_pwm1_freq.dev_attr.attr, | 1037 | &sensor_dev_attr_pwm1_freq.dev_attr.attr, |
| 1033 | &sensor_dev_attr_pwm1_enable.dev_attr.attr, | 1038 | &sensor_dev_attr_pwm1_enable.dev_attr.attr, |
| 1034 | &sensor_dev_attr_pwm1_auto_channels_temp.dev_attr.attr, | 1039 | &sensor_dev_attr_pwm1_auto_channels_temp.dev_attr.attr, |
| 1035 | &sensor_dev_attr_pwm1_auto_point1_pwm.dev_attr.attr, | 1040 | &sensor_dev_attr_pwm1_auto_point1_pwm.dev_attr.attr, |
| 1036 | &sensor_dev_attr_pwm1_auto_point2_pwm.dev_attr.attr, | 1041 | &sensor_dev_attr_pwm1_auto_point2_pwm.dev_attr.attr, |
| 1037 | &sensor_dev_attr_pwm2.dev_attr.attr, | ||
| 1038 | &sensor_dev_attr_pwm2_freq.dev_attr.attr, | ||
| 1039 | &sensor_dev_attr_pwm2_enable.dev_attr.attr, | ||
| 1040 | &sensor_dev_attr_pwm2_auto_channels_temp.dev_attr.attr, | ||
| 1041 | &sensor_dev_attr_pwm2_auto_point1_pwm.dev_attr.attr, | ||
| 1042 | &sensor_dev_attr_pwm2_auto_point2_pwm.dev_attr.attr, | ||
| 1043 | &sensor_dev_attr_pwm3.dev_attr.attr, | 1042 | &sensor_dev_attr_pwm3.dev_attr.attr, |
| 1044 | &sensor_dev_attr_pwm3_freq.dev_attr.attr, | 1043 | &sensor_dev_attr_pwm3_freq.dev_attr.attr, |
| 1045 | &sensor_dev_attr_pwm3_enable.dev_attr.attr, | 1044 | &sensor_dev_attr_pwm3_enable.dev_attr.attr, |
| @@ -1050,12 +1049,33 @@ static struct attribute *adt7475_attrs[] = { | |||
| 1050 | NULL, | 1049 | NULL, |
| 1051 | }; | 1050 | }; |
| 1052 | 1051 | ||
| 1052 | static struct attribute *fan4_attrs[] = { | ||
| 1053 | &sensor_dev_attr_fan4_input.dev_attr.attr, | ||
| 1054 | &sensor_dev_attr_fan4_min.dev_attr.attr, | ||
| 1055 | &sensor_dev_attr_fan4_alarm.dev_attr.attr, | ||
| 1056 | NULL | ||
| 1057 | }; | ||
| 1058 | |||
| 1059 | static struct attribute *pwm2_attrs[] = { | ||
| 1060 | &sensor_dev_attr_pwm2.dev_attr.attr, | ||
| 1061 | &sensor_dev_attr_pwm2_freq.dev_attr.attr, | ||
| 1062 | &sensor_dev_attr_pwm2_enable.dev_attr.attr, | ||
| 1063 | &sensor_dev_attr_pwm2_auto_channels_temp.dev_attr.attr, | ||
| 1064 | &sensor_dev_attr_pwm2_auto_point1_pwm.dev_attr.attr, | ||
| 1065 | &sensor_dev_attr_pwm2_auto_point2_pwm.dev_attr.attr, | ||
| 1066 | NULL | ||
| 1067 | }; | ||
| 1068 | |||
| 1053 | /* Attributes specific to the ADT7490 */ | 1069 | /* Attributes specific to the ADT7490 */ |
| 1054 | static struct attribute *adt7490_attrs[] = { | 1070 | static struct attribute *in0_attrs[] = { |
| 1055 | &sensor_dev_attr_in0_input.dev_attr.attr, | 1071 | &sensor_dev_attr_in0_input.dev_attr.attr, |
| 1056 | &sensor_dev_attr_in0_max.dev_attr.attr, | 1072 | &sensor_dev_attr_in0_max.dev_attr.attr, |
| 1057 | &sensor_dev_attr_in0_min.dev_attr.attr, | 1073 | &sensor_dev_attr_in0_min.dev_attr.attr, |
| 1058 | &sensor_dev_attr_in0_alarm.dev_attr.attr, | 1074 | &sensor_dev_attr_in0_alarm.dev_attr.attr, |
| 1075 | NULL | ||
| 1076 | }; | ||
| 1077 | |||
| 1078 | static struct attribute *adt7490_attrs[] = { | ||
| 1059 | &sensor_dev_attr_in3_input.dev_attr.attr, | 1079 | &sensor_dev_attr_in3_input.dev_attr.attr, |
| 1060 | &sensor_dev_attr_in3_max.dev_attr.attr, | 1080 | &sensor_dev_attr_in3_max.dev_attr.attr, |
| 1061 | &sensor_dev_attr_in3_min.dev_attr.attr, | 1081 | &sensor_dev_attr_in3_min.dev_attr.attr, |
| @@ -1072,6 +1092,9 @@ static struct attribute *adt7490_attrs[] = { | |||
| 1072 | }; | 1092 | }; |
| 1073 | 1093 | ||
| 1074 | static struct attribute_group adt7475_attr_group = { .attrs = adt7475_attrs }; | 1094 | static struct attribute_group adt7475_attr_group = { .attrs = adt7475_attrs }; |
| 1095 | static struct attribute_group fan4_attr_group = { .attrs = fan4_attrs }; | ||
| 1096 | static struct attribute_group pwm2_attr_group = { .attrs = pwm2_attrs }; | ||
| 1097 | static struct attribute_group in0_attr_group = { .attrs = in0_attrs }; | ||
| 1075 | static struct attribute_group adt7490_attr_group = { .attrs = adt7490_attrs }; | 1098 | static struct attribute_group adt7490_attr_group = { .attrs = adt7490_attrs }; |
| 1076 | 1099 | ||
| 1077 | static int adt7475_detect(struct i2c_client *client, int kind, | 1100 | static int adt7475_detect(struct i2c_client *client, int kind, |
| @@ -1115,13 +1138,20 @@ static void adt7475_remove_files(struct i2c_client *client, | |||
| 1115 | sysfs_remove_group(&client->dev.kobj, &adt7475_attr_group); | 1138 | sysfs_remove_group(&client->dev.kobj, &adt7475_attr_group); |
| 1116 | if (data->has_voltage & 0x39) | 1139 | if (data->has_voltage & 0x39) |
| 1117 | sysfs_remove_group(&client->dev.kobj, &adt7490_attr_group); | 1140 | sysfs_remove_group(&client->dev.kobj, &adt7490_attr_group); |
| 1141 | if (data->has_fan4) | ||
| 1142 | sysfs_remove_group(&client->dev.kobj, &fan4_attr_group); | ||
| 1143 | if (data->has_pwm2) | ||
| 1144 | sysfs_remove_group(&client->dev.kobj, &pwm2_attr_group); | ||
| 1145 | if (data->has_voltage & (1 << 0)) | ||
| 1146 | sysfs_remove_group(&client->dev.kobj, &in0_attr_group); | ||
| 1118 | } | 1147 | } |
| 1119 | 1148 | ||
| 1120 | static int adt7475_probe(struct i2c_client *client, | 1149 | static int adt7475_probe(struct i2c_client *client, |
| 1121 | const struct i2c_device_id *id) | 1150 | const struct i2c_device_id *id) |
| 1122 | { | 1151 | { |
| 1123 | struct adt7475_data *data; | 1152 | struct adt7475_data *data; |
| 1124 | int i, ret = 0; | 1153 | int i, ret = 0, revision; |
| 1154 | u8 config3; | ||
| 1125 | 1155 | ||
| 1126 | data = kzalloc(sizeof(*data), GFP_KERNEL); | 1156 | data = kzalloc(sizeof(*data), GFP_KERNEL); |
| 1127 | if (data == NULL) | 1157 | if (data == NULL) |
| @@ -1133,10 +1163,36 @@ static int adt7475_probe(struct i2c_client *client, | |||
| 1133 | /* Initialize device-specific values */ | 1163 | /* Initialize device-specific values */ |
| 1134 | switch (id->driver_data) { | 1164 | switch (id->driver_data) { |
| 1135 | case adt7490: | 1165 | case adt7490: |
| 1136 | data->has_voltage = 0x3f; /* in0 to in5 */ | 1166 | data->has_voltage = 0x3e; /* in1 to in5 */ |
| 1167 | revision = adt7475_read(REG_DEVID2) & 0x03; | ||
| 1137 | break; | 1168 | break; |
| 1138 | default: | 1169 | default: |
| 1139 | data->has_voltage = 0x06; /* in1, in2 */ | 1170 | data->has_voltage = 0x06; /* in1, in2 */ |
| 1171 | revision = adt7475_read(REG_DEVID2) & 0x07; | ||
| 1172 | } | ||
| 1173 | |||
| 1174 | config3 = adt7475_read(REG_CONFIG3); | ||
| 1175 | /* Pin PWM2 may alternatively be used for ALERT output */ | ||
| 1176 | if (!(config3 & CONFIG3_SMBALERT)) | ||
| 1177 | data->has_pwm2 = 1; | ||
| 1178 | /* Meaning of this bit is inverted for the ADT7473-1 */ | ||
| 1179 | if (id->driver_data == adt7473 && revision >= 1) | ||
| 1180 | data->has_pwm2 = !data->has_pwm2; | ||
| 1181 | |||
| 1182 | data->config4 = adt7475_read(REG_CONFIG4); | ||
| 1183 | /* Pin TACH4 may alternatively be used for THERM */ | ||
| 1184 | if ((data->config4 & CONFIG4_PINFUNC) == 0x0) | ||
| 1185 | data->has_fan4 = 1; | ||
| 1186 | |||
| 1187 | /* THERM configuration is more complex on the ADT7490, because 2 | ||
| 1188 | different pins (TACH4 and +2.5 Vin) can be used for this function */ | ||
| 1189 | if (id->driver_data == adt7490) { | ||
| 1190 | if ((data->config4 & CONFIG4_PINFUNC) == 0x1 && | ||
| 1191 | !(config3 & CONFIG3_THERM)) | ||
| 1192 | data->has_fan4 = 1; | ||
| 1193 | if (!(config3 & CONFIG3_THERM) || | ||
| 1194 | (data->config4 & CONFIG4_PINFUNC) == 0x1) | ||
| 1195 | data->has_voltage |= (1 << 0); /* in0 */ | ||
| 1140 | } | 1196 | } |
| 1141 | 1197 | ||
| 1142 | /* Call adt7475_read_pwm for all pwm's as this will reprogram any | 1198 | /* Call adt7475_read_pwm for all pwm's as this will reprogram any |
| @@ -1155,6 +1211,23 @@ static int adt7475_probe(struct i2c_client *client, | |||
| 1155 | goto eremove; | 1211 | goto eremove; |
| 1156 | } | 1212 | } |
| 1157 | 1213 | ||
| 1214 | /* Features that can be disabled individually */ | ||
| 1215 | if (data->has_fan4) { | ||
| 1216 | ret = sysfs_create_group(&client->dev.kobj, &fan4_attr_group); | ||
| 1217 | if (ret) | ||
| 1218 | goto eremove; | ||
| 1219 | } | ||
| 1220 | if (data->has_pwm2) { | ||
| 1221 | ret = sysfs_create_group(&client->dev.kobj, &pwm2_attr_group); | ||
| 1222 | if (ret) | ||
| 1223 | goto eremove; | ||
| 1224 | } | ||
| 1225 | if (data->has_voltage & (1 << 0)) { | ||
| 1226 | ret = sysfs_create_group(&client->dev.kobj, &in0_attr_group); | ||
| 1227 | if (ret) | ||
| 1228 | goto eremove; | ||
| 1229 | } | ||
| 1230 | |||
| 1158 | data->hwmon_dev = hwmon_device_register(&client->dev); | 1231 | data->hwmon_dev = hwmon_device_register(&client->dev); |
| 1159 | if (IS_ERR(data->hwmon_dev)) { | 1232 | if (IS_ERR(data->hwmon_dev)) { |
| 1160 | ret = PTR_ERR(data->hwmon_dev); | 1233 | ret = PTR_ERR(data->hwmon_dev); |
| @@ -1293,13 +1366,19 @@ static struct adt7475_data *adt7475_update_device(struct device *dev) | |||
| 1293 | ((ext >> 4) & 3); | 1366 | ((ext >> 4) & 3); |
| 1294 | } | 1367 | } |
| 1295 | 1368 | ||
| 1296 | for (i = 0; i < ADT7475_TACH_COUNT; i++) | 1369 | for (i = 0; i < ADT7475_TACH_COUNT; i++) { |
| 1370 | if (i == 3 && !data->has_fan4) | ||
| 1371 | continue; | ||
| 1297 | data->tach[INPUT][i] = | 1372 | data->tach[INPUT][i] = |
| 1298 | adt7475_read_word(client, TACH_REG(i)); | 1373 | adt7475_read_word(client, TACH_REG(i)); |
| 1374 | } | ||
| 1299 | 1375 | ||
| 1300 | /* Updated by hw when in auto mode */ | 1376 | /* Updated by hw when in auto mode */ |
| 1301 | for (i = 0; i < ADT7475_PWM_COUNT; i++) | 1377 | for (i = 0; i < ADT7475_PWM_COUNT; i++) { |
| 1378 | if (i == 1 && !data->has_pwm2) | ||
| 1379 | continue; | ||
| 1302 | data->pwm[INPUT][i] = adt7475_read(PWM_REG(i)); | 1380 | data->pwm[INPUT][i] = adt7475_read(PWM_REG(i)); |
| 1381 | } | ||
| 1303 | 1382 | ||
| 1304 | data->measure_updated = jiffies; | 1383 | data->measure_updated = jiffies; |
| 1305 | } | 1384 | } |
| @@ -1340,11 +1419,16 @@ static struct adt7475_data *adt7475_update_device(struct device *dev) | |||
| 1340 | } | 1419 | } |
| 1341 | adt7475_read_hystersis(client); | 1420 | adt7475_read_hystersis(client); |
| 1342 | 1421 | ||
| 1343 | for (i = 0; i < ADT7475_TACH_COUNT; i++) | 1422 | for (i = 0; i < ADT7475_TACH_COUNT; i++) { |
| 1423 | if (i == 3 && !data->has_fan4) | ||
| 1424 | continue; | ||
| 1344 | data->tach[MIN][i] = | 1425 | data->tach[MIN][i] = |
| 1345 | adt7475_read_word(client, TACH_MIN_REG(i)); | 1426 | adt7475_read_word(client, TACH_MIN_REG(i)); |
| 1427 | } | ||
| 1346 | 1428 | ||
| 1347 | for (i = 0; i < ADT7475_PWM_COUNT; i++) { | 1429 | for (i = 0; i < ADT7475_PWM_COUNT; i++) { |
| 1430 | if (i == 1 && !data->has_pwm2) | ||
| 1431 | continue; | ||
| 1348 | data->pwm[MAX][i] = adt7475_read(PWM_MAX_REG(i)); | 1432 | data->pwm[MAX][i] = adt7475_read(PWM_MAX_REG(i)); |
| 1349 | data->pwm[MIN][i] = adt7475_read(PWM_MIN_REG(i)); | 1433 | data->pwm[MIN][i] = adt7475_read(PWM_MIN_REG(i)); |
| 1350 | /* Set the channel and control information */ | 1434 | /* Set the channel and control information */ |
