diff options
author | Guenter Roeck <linux@roeck-us.net> | 2013-03-28 04:36:44 -0400 |
---|---|---|
committer | Guenter Roeck <linux@roeck-us.net> | 2013-04-21 11:56:48 -0400 |
commit | 29dd3b64b9edba3dd3dc8bb4d589869a4597a710 (patch) | |
tree | fdc8d81542108ea8ee93e9c4ae722232f5eedac8 | |
parent | 0846e30dd403ca613c203ccb6e4a89c09235480c (diff) |
hwmon: (tmp401) Add support for TMP432
TMP432 is similar to TMP431 with a second external temperature sensor.
Signed-off-by: Guenter Roeck <linux@roeck-us.net>
Acked-by: Jean Delvare <khali@linux-fr.org>
-rw-r--r-- | Documentation/hwmon/tmp401 | 11 | ||||
-rw-r--r-- | drivers/hwmon/Kconfig | 2 | ||||
-rw-r--r-- | drivers/hwmon/tmp401.c | 192 |
3 files changed, 165 insertions, 40 deletions
diff --git a/Documentation/hwmon/tmp401 b/Documentation/hwmon/tmp401 index 12e4781c0bc1..f91e3fa7e5ec 100644 --- a/Documentation/hwmon/tmp401 +++ b/Documentation/hwmon/tmp401 | |||
@@ -14,6 +14,10 @@ Supported chips: | |||
14 | Prefix: 'tmp431' | 14 | Prefix: 'tmp431' |
15 | Addresses scanned: I2C 0x4c, 0x4d | 15 | Addresses scanned: I2C 0x4c, 0x4d |
16 | Datasheet: http://focus.ti.com/docs/prod/folders/print/tmp431.html | 16 | Datasheet: http://focus.ti.com/docs/prod/folders/print/tmp431.html |
17 | * Texas Instruments TMP432 | ||
18 | Prefix: 'tmp432' | ||
19 | Addresses scanned: I2C 0x4c, 0x4d | ||
20 | Datasheet: http://focus.ti.com/docs/prod/folders/print/tmp432.html | ||
17 | 21 | ||
18 | Authors: | 22 | Authors: |
19 | Hans de Goede <hdegoede@redhat.com> | 23 | Hans de Goede <hdegoede@redhat.com> |
@@ -23,8 +27,8 @@ Description | |||
23 | ----------- | 27 | ----------- |
24 | 28 | ||
25 | This driver implements support for Texas Instruments TMP401, TMP411, | 29 | This driver implements support for Texas Instruments TMP401, TMP411, |
26 | and TMP431 chips. These chips implement one remote and one local | 30 | TMP431, and TMP432 chips. These chips implement one or two remote and |
27 | temperature sensor. Temperature is measured in degrees | 31 | one local temperature sensors. Temperature is measured in degrees |
28 | Celsius. Resolution of the remote sensor is 0.0625 degree. Local | 32 | Celsius. Resolution of the remote sensor is 0.0625 degree. Local |
29 | sensor resolution can be set to 0.5, 0.25, 0.125 or 0.0625 degree (not | 33 | sensor resolution can be set to 0.5, 0.25, 0.125 or 0.0625 degree (not |
30 | supported by the driver so far, so using the default resolution of 0.5 | 34 | supported by the driver so far, so using the default resolution of 0.5 |
@@ -44,3 +48,6 @@ some additional features. | |||
44 | 48 | ||
45 | Exported via sysfs attribute temp_reset_history. Writing 1 to this | 49 | Exported via sysfs attribute temp_reset_history. Writing 1 to this |
46 | file triggers a reset. | 50 | file triggers a reset. |
51 | |||
52 | TMP432 is compatible with TMP401 and TMP431. It supports two external | ||
53 | temperature sensors. | ||
diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig index 26ebff0b45ca..4f29117651b6 100644 --- a/drivers/hwmon/Kconfig +++ b/drivers/hwmon/Kconfig | |||
@@ -1249,7 +1249,7 @@ config SENSORS_TMP401 | |||
1249 | depends on I2C | 1249 | depends on I2C |
1250 | help | 1250 | help |
1251 | If you say yes here you get support for Texas Instruments TMP401, | 1251 | If you say yes here you get support for Texas Instruments TMP401, |
1252 | TMP411, and TMP431 temperature sensor chips. | 1252 | TMP411, TMP431, and TMP432 temperature sensor chips. |
1253 | 1253 | ||
1254 | This driver can also be built as a module. If so, the module | 1254 | This driver can also be built as a module. If so, the module |
1255 | will be called tmp401. | 1255 | will be called tmp401. |
diff --git a/drivers/hwmon/tmp401.c b/drivers/hwmon/tmp401.c index fa6af51b300d..a478454f690f 100644 --- a/drivers/hwmon/tmp401.c +++ b/drivers/hwmon/tmp401.c | |||
@@ -5,6 +5,9 @@ | |||
5 | * Gabriel Konat, Sander Leget, Wouter Willems | 5 | * Gabriel Konat, Sander Leget, Wouter Willems |
6 | * Copyright (C) 2009 Andre Prendel <andre.prendel@gmx.de> | 6 | * Copyright (C) 2009 Andre Prendel <andre.prendel@gmx.de> |
7 | * | 7 | * |
8 | * Cleanup and support for TMP431 and TMP432 by Guenter Roeck | ||
9 | * Copyright (c) 2013 Guenter Roeck <linux@roeck-us.net> | ||
10 | * | ||
8 | * This program is free software; you can redistribute it and/or modify | 11 | * This program is free software; you can redistribute it and/or modify |
9 | * it under the terms of the GNU General Public License as published by | 12 | * it under the terms of the GNU General Public License as published by |
10 | * the Free Software Foundation; either version 2 of the License, or | 13 | * the Free Software Foundation; either version 2 of the License, or |
@@ -43,7 +46,7 @@ | |||
43 | /* Addresses to scan */ | 46 | /* Addresses to scan */ |
44 | static const unsigned short normal_i2c[] = { 0x4c, 0x4d, 0x4e, I2C_CLIENT_END }; | 47 | static const unsigned short normal_i2c[] = { 0x4c, 0x4d, 0x4e, I2C_CLIENT_END }; |
45 | 48 | ||
46 | enum chips { tmp401, tmp411, tmp431 }; | 49 | enum chips { tmp401, tmp411, tmp431, tmp432 }; |
47 | 50 | ||
48 | /* | 51 | /* |
49 | * The TMP401 registers, note some registers have different addresses for | 52 | * The TMP401 registers, note some registers have different addresses for |
@@ -85,6 +88,30 @@ static const u8 TMP401_TEMP_LSB[6][2] = { | |||
85 | { 0x33, 0x37 }, /* highest */ | 88 | { 0x33, 0x37 }, /* highest */ |
86 | }; | 89 | }; |
87 | 90 | ||
91 | static const u8 TMP432_TEMP_MSB_READ[4][3] = { | ||
92 | { 0x00, 0x01, 0x23 }, /* temp */ | ||
93 | { 0x06, 0x08, 0x16 }, /* low limit */ | ||
94 | { 0x05, 0x07, 0x15 }, /* high limit */ | ||
95 | { 0x20, 0x19, 0x1A }, /* therm (crit) limit */ | ||
96 | }; | ||
97 | |||
98 | static const u8 TMP432_TEMP_MSB_WRITE[4][3] = { | ||
99 | { 0, 0, 0 }, /* temp - unused */ | ||
100 | { 0x0C, 0x0E, 0x16 }, /* low limit */ | ||
101 | { 0x0B, 0x0D, 0x15 }, /* high limit */ | ||
102 | { 0x20, 0x19, 0x1A }, /* therm (crit) limit */ | ||
103 | }; | ||
104 | |||
105 | static const u8 TMP432_TEMP_LSB[3][3] = { | ||
106 | { 0x29, 0x10, 0x24 }, /* temp */ | ||
107 | { 0x3E, 0x14, 0x18 }, /* low limit */ | ||
108 | { 0x3D, 0x13, 0x17 }, /* high limit */ | ||
109 | }; | ||
110 | |||
111 | /* [0] = fault, [1] = low, [2] = high, [3] = therm/crit */ | ||
112 | static const u8 TMP432_STATUS_REG[] = { | ||
113 | 0x1b, 0x36, 0x35, 0x37 }; | ||
114 | |||
88 | /* Flags */ | 115 | /* Flags */ |
89 | #define TMP401_CONFIG_RANGE BIT(2) | 116 | #define TMP401_CONFIG_RANGE BIT(2) |
90 | #define TMP401_CONFIG_SHUTDOWN BIT(6) | 117 | #define TMP401_CONFIG_SHUTDOWN BIT(6) |
@@ -96,6 +123,11 @@ static const u8 TMP401_TEMP_LSB[6][2] = { | |||
96 | #define TMP401_STATUS_LOCAL_LOW BIT(5) | 123 | #define TMP401_STATUS_LOCAL_LOW BIT(5) |
97 | #define TMP401_STATUS_LOCAL_HIGH BIT(6) | 124 | #define TMP401_STATUS_LOCAL_HIGH BIT(6) |
98 | 125 | ||
126 | /* On TMP432, each status has its own register */ | ||
127 | #define TMP432_STATUS_LOCAL BIT(0) | ||
128 | #define TMP432_STATUS_REMOTE1 BIT(1) | ||
129 | #define TMP432_STATUS_REMOTE2 BIT(2) | ||
130 | |||
99 | /* Manufacturer / Device ID's */ | 131 | /* Manufacturer / Device ID's */ |
100 | #define TMP401_MANUFACTURER_ID 0x55 | 132 | #define TMP401_MANUFACTURER_ID 0x55 |
101 | #define TMP401_DEVICE_ID 0x11 | 133 | #define TMP401_DEVICE_ID 0x11 |
@@ -103,6 +135,7 @@ static const u8 TMP401_TEMP_LSB[6][2] = { | |||
103 | #define TMP411B_DEVICE_ID 0x13 | 135 | #define TMP411B_DEVICE_ID 0x13 |
104 | #define TMP411C_DEVICE_ID 0x10 | 136 | #define TMP411C_DEVICE_ID 0x10 |
105 | #define TMP431_DEVICE_ID 0x31 | 137 | #define TMP431_DEVICE_ID 0x31 |
138 | #define TMP432_DEVICE_ID 0x32 | ||
106 | 139 | ||
107 | /* | 140 | /* |
108 | * Driver data (common to all clients) | 141 | * Driver data (common to all clients) |
@@ -112,6 +145,7 @@ static const struct i2c_device_id tmp401_id[] = { | |||
112 | { "tmp401", tmp401 }, | 145 | { "tmp401", tmp401 }, |
113 | { "tmp411", tmp411 }, | 146 | { "tmp411", tmp411 }, |
114 | { "tmp431", tmp431 }, | 147 | { "tmp431", tmp431 }, |
148 | { "tmp432", tmp432 }, | ||
115 | { } | 149 | { } |
116 | }; | 150 | }; |
117 | MODULE_DEVICE_TABLE(i2c, tmp401_id); | 151 | MODULE_DEVICE_TABLE(i2c, tmp401_id); |
@@ -130,9 +164,9 @@ struct tmp401_data { | |||
130 | unsigned int update_interval; /* in milliseconds */ | 164 | unsigned int update_interval; /* in milliseconds */ |
131 | 165 | ||
132 | /* register values */ | 166 | /* register values */ |
133 | u8 status; | 167 | u8 status[4]; |
134 | u8 config; | 168 | u8 config; |
135 | u16 temp[6][2]; | 169 | u16 temp[6][3]; |
136 | u8 temp_crit_hyst; | 170 | u8 temp_crit_hyst; |
137 | }; | 171 | }; |
138 | 172 | ||
@@ -166,22 +200,27 @@ static int tmp401_update_device_reg16(struct i2c_client *client, | |||
166 | { | 200 | { |
167 | int i, j, val; | 201 | int i, j, val; |
168 | int num_regs = data->kind == tmp411 ? 6 : 4; | 202 | int num_regs = data->kind == tmp411 ? 6 : 4; |
203 | int num_sensors = data->kind == tmp432 ? 3 : 2; | ||
169 | 204 | ||
170 | for (i = 0; i < 2; i++) { /* local / rem1 */ | 205 | for (i = 0; i < num_sensors; i++) { /* local / r1 / r2 */ |
171 | for (j = 0; j < num_regs; j++) { /* temp / low / ... */ | 206 | for (j = 0; j < num_regs; j++) { /* temp / low / ... */ |
207 | u8 regaddr; | ||
172 | /* | 208 | /* |
173 | * High byte must be read first immediately followed | 209 | * High byte must be read first immediately followed |
174 | * by the low byte | 210 | * by the low byte |
175 | */ | 211 | */ |
176 | val = i2c_smbus_read_byte_data(client, | 212 | regaddr = data->kind == tmp432 ? |
177 | TMP401_TEMP_MSB_READ[j][i]); | 213 | TMP432_TEMP_MSB_READ[j][i] : |
214 | TMP401_TEMP_MSB_READ[j][i]; | ||
215 | val = i2c_smbus_read_byte_data(client, regaddr); | ||
178 | if (val < 0) | 216 | if (val < 0) |
179 | return val; | 217 | return val; |
180 | data->temp[j][i] = val << 8; | 218 | data->temp[j][i] = val << 8; |
181 | if (j == 3) /* crit is msb only */ | 219 | if (j == 3) /* crit is msb only */ |
182 | continue; | 220 | continue; |
183 | val = i2c_smbus_read_byte_data(client, | 221 | regaddr = data->kind == tmp432 ? TMP432_TEMP_LSB[j][i] |
184 | TMP401_TEMP_LSB[j][i]); | 222 | : TMP401_TEMP_LSB[j][i]; |
223 | val = i2c_smbus_read_byte_data(client, regaddr); | ||
185 | if (val < 0) | 224 | if (val < 0) |
186 | return val; | 225 | return val; |
187 | data->temp[j][i] |= val; | 226 | data->temp[j][i] |= val; |
@@ -195,7 +234,7 @@ static struct tmp401_data *tmp401_update_device(struct device *dev) | |||
195 | struct i2c_client *client = to_i2c_client(dev); | 234 | struct i2c_client *client = to_i2c_client(dev); |
196 | struct tmp401_data *data = i2c_get_clientdata(client); | 235 | struct tmp401_data *data = i2c_get_clientdata(client); |
197 | struct tmp401_data *ret = data; | 236 | struct tmp401_data *ret = data; |
198 | int val; | 237 | int i, val; |
199 | unsigned long next_update; | 238 | unsigned long next_update; |
200 | 239 | ||
201 | mutex_lock(&data->update_lock); | 240 | mutex_lock(&data->update_lock); |
@@ -203,12 +242,38 @@ static struct tmp401_data *tmp401_update_device(struct device *dev) | |||
203 | next_update = data->last_updated + | 242 | next_update = data->last_updated + |
204 | msecs_to_jiffies(data->update_interval) + 1; | 243 | msecs_to_jiffies(data->update_interval) + 1; |
205 | if (time_after(jiffies, next_update) || !data->valid) { | 244 | if (time_after(jiffies, next_update) || !data->valid) { |
206 | val = i2c_smbus_read_byte_data(client, TMP401_STATUS); | 245 | if (data->kind != tmp432) { |
207 | if (val < 0) { | 246 | /* |
208 | ret = ERR_PTR(val); | 247 | * The driver uses the TMP432 status format internally. |
209 | goto abort; | 248 | * Convert status to TMP432 format for other chips. |
249 | */ | ||
250 | val = i2c_smbus_read_byte_data(client, TMP401_STATUS); | ||
251 | if (val < 0) { | ||
252 | ret = ERR_PTR(val); | ||
253 | goto abort; | ||
254 | } | ||
255 | data->status[0] = | ||
256 | (val & TMP401_STATUS_REMOTE_OPEN) >> 1; | ||
257 | data->status[1] = | ||
258 | ((val & TMP401_STATUS_REMOTE_LOW) >> 2) | | ||
259 | ((val & TMP401_STATUS_LOCAL_LOW) >> 5); | ||
260 | data->status[2] = | ||
261 | ((val & TMP401_STATUS_REMOTE_HIGH) >> 3) | | ||
262 | ((val & TMP401_STATUS_LOCAL_HIGH) >> 6); | ||
263 | data->status[3] = val & (TMP401_STATUS_LOCAL_CRIT | ||
264 | | TMP401_STATUS_REMOTE_CRIT); | ||
265 | } else { | ||
266 | for (i = 0; i < ARRAY_SIZE(data->status); i++) { | ||
267 | val = i2c_smbus_read_byte_data(client, | ||
268 | TMP432_STATUS_REG[i]); | ||
269 | if (val < 0) { | ||
270 | ret = ERR_PTR(val); | ||
271 | goto abort; | ||
272 | } | ||
273 | data->status[i] = val; | ||
274 | } | ||
210 | } | 275 | } |
211 | data->status = val; | 276 | |
212 | val = i2c_smbus_read_byte_data(client, TMP401_CONFIG_READ); | 277 | val = i2c_smbus_read_byte_data(client, TMP401_CONFIG_READ); |
213 | if (val < 0) { | 278 | if (val < 0) { |
214 | ret = ERR_PTR(val); | 279 | ret = ERR_PTR(val); |
@@ -270,13 +335,14 @@ static ssize_t show_temp_crit_hyst(struct device *dev, | |||
270 | static ssize_t show_status(struct device *dev, | 335 | static ssize_t show_status(struct device *dev, |
271 | struct device_attribute *devattr, char *buf) | 336 | struct device_attribute *devattr, char *buf) |
272 | { | 337 | { |
273 | int mask = to_sensor_dev_attr(devattr)->index; | 338 | int nr = to_sensor_dev_attr_2(devattr)->nr; |
339 | int mask = to_sensor_dev_attr_2(devattr)->index; | ||
274 | struct tmp401_data *data = tmp401_update_device(dev); | 340 | struct tmp401_data *data = tmp401_update_device(dev); |
275 | 341 | ||
276 | if (IS_ERR(data)) | 342 | if (IS_ERR(data)) |
277 | return PTR_ERR(data); | 343 | return PTR_ERR(data); |
278 | 344 | ||
279 | return sprintf(buf, "%d\n", !!(data->status & mask)); | 345 | return sprintf(buf, "%d\n", !!(data->status[nr] & mask)); |
280 | } | 346 | } |
281 | 347 | ||
282 | static ssize_t store_temp(struct device *dev, struct device_attribute *devattr, | 348 | static ssize_t store_temp(struct device *dev, struct device_attribute *devattr, |
@@ -288,6 +354,7 @@ static ssize_t store_temp(struct device *dev, struct device_attribute *devattr, | |||
288 | struct tmp401_data *data = tmp401_update_device(dev); | 354 | struct tmp401_data *data = tmp401_update_device(dev); |
289 | long val; | 355 | long val; |
290 | u16 reg; | 356 | u16 reg; |
357 | u8 regaddr; | ||
291 | 358 | ||
292 | if (IS_ERR(data)) | 359 | if (IS_ERR(data)) |
293 | return PTR_ERR(data); | 360 | return PTR_ERR(data); |
@@ -299,13 +366,13 @@ static ssize_t store_temp(struct device *dev, struct device_attribute *devattr, | |||
299 | 366 | ||
300 | mutex_lock(&data->update_lock); | 367 | mutex_lock(&data->update_lock); |
301 | 368 | ||
302 | i2c_smbus_write_byte_data(client, | 369 | regaddr = data->kind == tmp432 ? TMP432_TEMP_MSB_WRITE[nr][index] |
303 | TMP401_TEMP_MSB_WRITE[nr][index], | 370 | : TMP401_TEMP_MSB_WRITE[nr][index]; |
304 | reg >> 8); | 371 | i2c_smbus_write_byte_data(client, regaddr, reg >> 8); |
305 | if (nr != 3) { | 372 | if (nr != 3) { |
306 | i2c_smbus_write_byte_data(client, | 373 | regaddr = data->kind == tmp432 ? TMP432_TEMP_LSB[nr][index] |
307 | TMP401_TEMP_LSB[nr][index], | 374 | : TMP401_TEMP_LSB[nr][index]; |
308 | reg & 0xFF); | 375 | i2c_smbus_write_byte_data(client, regaddr, reg & 0xFF); |
309 | } | 376 | } |
310 | data->temp[nr][index] = reg; | 377 | data->temp[nr][index] = reg; |
311 | 378 | ||
@@ -426,12 +493,12 @@ static SENSOR_DEVICE_ATTR_2(temp1_crit, S_IWUSR | S_IRUGO, show_temp, | |||
426 | store_temp, 3, 0); | 493 | store_temp, 3, 0); |
427 | static SENSOR_DEVICE_ATTR(temp1_crit_hyst, S_IWUSR | S_IRUGO, | 494 | static SENSOR_DEVICE_ATTR(temp1_crit_hyst, S_IWUSR | S_IRUGO, |
428 | show_temp_crit_hyst, store_temp_crit_hyst, 0); | 495 | show_temp_crit_hyst, store_temp_crit_hyst, 0); |
429 | static SENSOR_DEVICE_ATTR(temp1_min_alarm, S_IRUGO, show_status, NULL, | 496 | static SENSOR_DEVICE_ATTR_2(temp1_min_alarm, S_IRUGO, show_status, NULL, |
430 | TMP401_STATUS_LOCAL_LOW); | 497 | 1, TMP432_STATUS_LOCAL); |
431 | static SENSOR_DEVICE_ATTR(temp1_max_alarm, S_IRUGO, show_status, NULL, | 498 | static SENSOR_DEVICE_ATTR_2(temp1_max_alarm, S_IRUGO, show_status, NULL, |
432 | TMP401_STATUS_LOCAL_HIGH); | 499 | 2, TMP432_STATUS_LOCAL); |
433 | static SENSOR_DEVICE_ATTR(temp1_crit_alarm, S_IRUGO, show_status, NULL, | 500 | static SENSOR_DEVICE_ATTR_2(temp1_crit_alarm, S_IRUGO, show_status, NULL, |
434 | TMP401_STATUS_LOCAL_CRIT); | 501 | 3, TMP432_STATUS_LOCAL); |
435 | static SENSOR_DEVICE_ATTR_2(temp2_input, S_IRUGO, show_temp, NULL, 0, 1); | 502 | static SENSOR_DEVICE_ATTR_2(temp2_input, S_IRUGO, show_temp, NULL, 0, 1); |
436 | static SENSOR_DEVICE_ATTR_2(temp2_min, S_IWUSR | S_IRUGO, show_temp, | 503 | static SENSOR_DEVICE_ATTR_2(temp2_min, S_IWUSR | S_IRUGO, show_temp, |
437 | store_temp, 1, 1); | 504 | store_temp, 1, 1); |
@@ -441,14 +508,14 @@ static SENSOR_DEVICE_ATTR_2(temp2_crit, S_IWUSR | S_IRUGO, show_temp, | |||
441 | store_temp, 3, 1); | 508 | store_temp, 3, 1); |
442 | static SENSOR_DEVICE_ATTR(temp2_crit_hyst, S_IRUGO, show_temp_crit_hyst, | 509 | static SENSOR_DEVICE_ATTR(temp2_crit_hyst, S_IRUGO, show_temp_crit_hyst, |
443 | NULL, 1); | 510 | NULL, 1); |
444 | static SENSOR_DEVICE_ATTR(temp2_fault, S_IRUGO, show_status, NULL, | 511 | static SENSOR_DEVICE_ATTR_2(temp2_fault, S_IRUGO, show_status, NULL, |
445 | TMP401_STATUS_REMOTE_OPEN); | 512 | 0, TMP432_STATUS_REMOTE1); |
446 | static SENSOR_DEVICE_ATTR(temp2_min_alarm, S_IRUGO, show_status, NULL, | 513 | static SENSOR_DEVICE_ATTR_2(temp2_min_alarm, S_IRUGO, show_status, NULL, |
447 | TMP401_STATUS_REMOTE_LOW); | 514 | 1, TMP432_STATUS_REMOTE1); |
448 | static SENSOR_DEVICE_ATTR(temp2_max_alarm, S_IRUGO, show_status, NULL, | 515 | static SENSOR_DEVICE_ATTR_2(temp2_max_alarm, S_IRUGO, show_status, NULL, |
449 | TMP401_STATUS_REMOTE_HIGH); | 516 | 2, TMP432_STATUS_REMOTE1); |
450 | static SENSOR_DEVICE_ATTR(temp2_crit_alarm, S_IRUGO, show_status, NULL, | 517 | static SENSOR_DEVICE_ATTR_2(temp2_crit_alarm, S_IRUGO, show_status, NULL, |
451 | TMP401_STATUS_REMOTE_CRIT); | 518 | 3, TMP432_STATUS_REMOTE1); |
452 | 519 | ||
453 | static DEVICE_ATTR(update_interval, S_IRUGO | S_IWUSR, show_update_interval, | 520 | static DEVICE_ATTR(update_interval, S_IRUGO | S_IWUSR, show_update_interval, |
454 | set_update_interval); | 521 | set_update_interval); |
@@ -509,6 +576,42 @@ static const struct attribute_group tmp411_group = { | |||
509 | .attrs = tmp411_attributes, | 576 | .attrs = tmp411_attributes, |
510 | }; | 577 | }; |
511 | 578 | ||
579 | static SENSOR_DEVICE_ATTR_2(temp3_input, S_IRUGO, show_temp, NULL, 0, 2); | ||
580 | static SENSOR_DEVICE_ATTR_2(temp3_min, S_IWUSR | S_IRUGO, show_temp, | ||
581 | store_temp, 1, 2); | ||
582 | static SENSOR_DEVICE_ATTR_2(temp3_max, S_IWUSR | S_IRUGO, show_temp, | ||
583 | store_temp, 2, 2); | ||
584 | static SENSOR_DEVICE_ATTR_2(temp3_crit, S_IWUSR | S_IRUGO, show_temp, | ||
585 | store_temp, 3, 2); | ||
586 | static SENSOR_DEVICE_ATTR(temp3_crit_hyst, S_IRUGO, show_temp_crit_hyst, | ||
587 | NULL, 2); | ||
588 | static SENSOR_DEVICE_ATTR_2(temp3_fault, S_IRUGO, show_status, NULL, | ||
589 | 0, TMP432_STATUS_REMOTE2); | ||
590 | static SENSOR_DEVICE_ATTR_2(temp3_min_alarm, S_IRUGO, show_status, NULL, | ||
591 | 1, TMP432_STATUS_REMOTE2); | ||
592 | static SENSOR_DEVICE_ATTR_2(temp3_max_alarm, S_IRUGO, show_status, NULL, | ||
593 | 2, TMP432_STATUS_REMOTE2); | ||
594 | static SENSOR_DEVICE_ATTR_2(temp3_crit_alarm, S_IRUGO, show_status, NULL, | ||
595 | 3, TMP432_STATUS_REMOTE2); | ||
596 | |||
597 | static struct attribute *tmp432_attributes[] = { | ||
598 | &sensor_dev_attr_temp3_input.dev_attr.attr, | ||
599 | &sensor_dev_attr_temp3_min.dev_attr.attr, | ||
600 | &sensor_dev_attr_temp3_max.dev_attr.attr, | ||
601 | &sensor_dev_attr_temp3_crit.dev_attr.attr, | ||
602 | &sensor_dev_attr_temp3_crit_hyst.dev_attr.attr, | ||
603 | &sensor_dev_attr_temp3_fault.dev_attr.attr, | ||
604 | &sensor_dev_attr_temp3_max_alarm.dev_attr.attr, | ||
605 | &sensor_dev_attr_temp3_min_alarm.dev_attr.attr, | ||
606 | &sensor_dev_attr_temp3_crit_alarm.dev_attr.attr, | ||
607 | |||
608 | NULL | ||
609 | }; | ||
610 | |||
611 | static const struct attribute_group tmp432_group = { | ||
612 | .attrs = tmp432_attributes, | ||
613 | }; | ||
614 | |||
512 | /* | 615 | /* |
513 | * Begin non sysfs callback code (aka Real code) | 616 | * Begin non sysfs callback code (aka Real code) |
514 | */ | 617 | */ |
@@ -579,6 +682,11 @@ static int tmp401_detect(struct i2c_client *client, | |||
579 | return -ENODEV; | 682 | return -ENODEV; |
580 | kind = tmp431; | 683 | kind = tmp431; |
581 | break; | 684 | break; |
685 | case TMP432_DEVICE_ID: | ||
686 | if (client->addr == 0x4e) | ||
687 | return -ENODEV; | ||
688 | kind = tmp432; | ||
689 | break; | ||
582 | default: | 690 | default: |
583 | return -ENODEV; | 691 | return -ENODEV; |
584 | } | 692 | } |
@@ -610,6 +718,9 @@ static int tmp401_remove(struct i2c_client *client) | |||
610 | if (data->kind == tmp411) | 718 | if (data->kind == tmp411) |
611 | sysfs_remove_group(&dev->kobj, &tmp411_group); | 719 | sysfs_remove_group(&dev->kobj, &tmp411_group); |
612 | 720 | ||
721 | if (data->kind == tmp432) | ||
722 | sysfs_remove_group(&dev->kobj, &tmp432_group); | ||
723 | |||
613 | return 0; | 724 | return 0; |
614 | } | 725 | } |
615 | 726 | ||
@@ -619,7 +730,7 @@ static int tmp401_probe(struct i2c_client *client, | |||
619 | struct device *dev = &client->dev; | 730 | struct device *dev = &client->dev; |
620 | int err; | 731 | int err; |
621 | struct tmp401_data *data; | 732 | struct tmp401_data *data; |
622 | const char *names[] = { "TMP401", "TMP411", "TMP431" }; | 733 | const char *names[] = { "TMP401", "TMP411", "TMP431", "TMP432" }; |
623 | 734 | ||
624 | data = devm_kzalloc(dev, sizeof(struct tmp401_data), GFP_KERNEL); | 735 | data = devm_kzalloc(dev, sizeof(struct tmp401_data), GFP_KERNEL); |
625 | if (!data) | 736 | if (!data) |
@@ -644,6 +755,13 @@ static int tmp401_probe(struct i2c_client *client, | |||
644 | goto exit_remove; | 755 | goto exit_remove; |
645 | } | 756 | } |
646 | 757 | ||
758 | /* Register additional tmp432 sysfs hooks */ | ||
759 | if (data->kind == tmp432) { | ||
760 | err = sysfs_create_group(&dev->kobj, &tmp432_group); | ||
761 | if (err) | ||
762 | goto exit_remove; | ||
763 | } | ||
764 | |||
647 | data->hwmon_dev = hwmon_device_register(dev); | 765 | data->hwmon_dev = hwmon_device_register(dev); |
648 | if (IS_ERR(data->hwmon_dev)) { | 766 | if (IS_ERR(data->hwmon_dev)) { |
649 | err = PTR_ERR(data->hwmon_dev); | 767 | err = PTR_ERR(data->hwmon_dev); |