blob: 250f6b059a540641ed81101aec9e51286bd7199b [file] [log] [blame]
Jean Delvare08e7e272005-04-25 22:43:25 +02001/*
2 w83627ehf - Driver for the hardware monitoring functionality of
3 the Winbond W83627EHF Super-I/O chip
4 Copyright (C) 2005 Jean Delvare <khali@linux-fr.org>
5
6 Shamelessly ripped from the w83627hf driver
7 Copyright (C) 2003 Mark Studebaker
8
9 Thanks to Leon Moonen, Steve Cliffe and Grant Coady for their help
10 in testing and debugging this driver.
11
Jean Delvare8dd2d2c2005-07-27 21:33:15 +020012 This driver also supports the W83627EHG, which is the lead-free
13 version of the W83627EHF.
14
Jean Delvare08e7e272005-04-25 22:43:25 +020015 This program is free software; you can redistribute it and/or modify
16 it under the terms of the GNU General Public License as published by
17 the Free Software Foundation; either version 2 of the License, or
18 (at your option) any later version.
19
20 This program is distributed in the hope that it will be useful,
21 but WITHOUT ANY WARRANTY; without even the implied warranty of
22 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 GNU General Public License for more details.
24
25 You should have received a copy of the GNU General Public License
26 along with this program; if not, write to the Free Software
27 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
28
29
30 Supports the following chips:
31
32 Chip #vin #fan #pwm #temp chip_id man_id
33 w83627ehf - 5 - 3 0x88 0x5ca3
34
35 This is a preliminary version of the driver, only supporting the
36 fan and temperature inputs. The chip does much more than that.
37*/
38
39#include <linux/module.h>
40#include <linux/init.h>
41#include <linux/slab.h>
42#include <linux/i2c.h>
43#include <linux/i2c-sensor.h>
44#include <asm/io.h>
45#include "lm75.h"
46
47/* Addresses to scan
48 The actual ISA address is read from Super-I/O configuration space */
49static unsigned short normal_i2c[] = { I2C_CLIENT_END };
50static unsigned int normal_isa[] = { 0, I2C_CLIENT_ISA_END };
51
52/* Insmod parameters */
53SENSORS_INSMOD_1(w83627ehf);
54
55/*
56 * Super-I/O constants and functions
57 */
58
59static int REG; /* The register to read/write */
60static int VAL; /* The value to read/write */
61
62#define W83627EHF_LD_HWM 0x0b
63
64#define SIO_REG_LDSEL 0x07 /* Logical device select */
65#define SIO_REG_DEVID 0x20 /* Device ID (2 bytes) */
66#define SIO_REG_ENABLE 0x30 /* Logical device enable */
67#define SIO_REG_ADDR 0x60 /* Logical device address (2 bytes) */
68
69#define SIO_W83627EHF_ID 0x8840
70#define SIO_ID_MASK 0xFFC0
71
72static inline void
73superio_outb(int reg, int val)
74{
75 outb(reg, REG);
76 outb(val, VAL);
77}
78
79static inline int
80superio_inb(int reg)
81{
82 outb(reg, REG);
83 return inb(VAL);
84}
85
86static inline void
87superio_select(int ld)
88{
89 outb(SIO_REG_LDSEL, REG);
90 outb(ld, VAL);
91}
92
93static inline void
94superio_enter(void)
95{
96 outb(0x87, REG);
97 outb(0x87, REG);
98}
99
100static inline void
101superio_exit(void)
102{
103 outb(0x02, REG);
104 outb(0x02, VAL);
105}
106
107/*
108 * ISA constants
109 */
110
111#define REGION_LENGTH 8
112#define ADDR_REG_OFFSET 5
113#define DATA_REG_OFFSET 6
114
115#define W83627EHF_REG_BANK 0x4E
116#define W83627EHF_REG_CONFIG 0x40
117#define W83627EHF_REG_CHIP_ID 0x49
118#define W83627EHF_REG_MAN_ID 0x4F
119
120static const u16 W83627EHF_REG_FAN[] = { 0x28, 0x29, 0x2a, 0x3f, 0x553 };
121static const u16 W83627EHF_REG_FAN_MIN[] = { 0x3b, 0x3c, 0x3d, 0x3e, 0x55c };
122
123#define W83627EHF_REG_TEMP1 0x27
124#define W83627EHF_REG_TEMP1_HYST 0x3a
125#define W83627EHF_REG_TEMP1_OVER 0x39
126static const u16 W83627EHF_REG_TEMP[] = { 0x150, 0x250 };
127static const u16 W83627EHF_REG_TEMP_HYST[] = { 0x153, 0x253 };
128static const u16 W83627EHF_REG_TEMP_OVER[] = { 0x155, 0x255 };
129static const u16 W83627EHF_REG_TEMP_CONFIG[] = { 0x152, 0x252 };
130
131/* Fan clock dividers are spread over the following five registers */
132#define W83627EHF_REG_FANDIV1 0x47
133#define W83627EHF_REG_FANDIV2 0x4B
134#define W83627EHF_REG_VBAT 0x5D
135#define W83627EHF_REG_DIODE 0x59
136#define W83627EHF_REG_SMI_OVT 0x4C
137
138/*
139 * Conversions
140 */
141
142static inline unsigned int
143fan_from_reg(u8 reg, unsigned int div)
144{
145 if (reg == 0 || reg == 255)
146 return 0;
147 return 1350000U / (reg * div);
148}
149
150static inline unsigned int
151div_from_reg(u8 reg)
152{
153 return 1 << reg;
154}
155
156static inline int
157temp1_from_reg(s8 reg)
158{
159 return reg * 1000;
160}
161
162static inline s8
163temp1_to_reg(int temp)
164{
165 if (temp <= -128000)
166 return -128;
167 if (temp >= 127000)
168 return 127;
169 if (temp < 0)
170 return (temp - 500) / 1000;
171 return (temp + 500) / 1000;
172}
173
174/*
175 * Data structures and manipulation thereof
176 */
177
178struct w83627ehf_data {
179 struct i2c_client client;
180 struct semaphore lock;
181
182 struct semaphore update_lock;
183 char valid; /* !=0 if following fields are valid */
184 unsigned long last_updated; /* In jiffies */
185
186 /* Register values */
187 u8 fan[5];
188 u8 fan_min[5];
189 u8 fan_div[5];
190 u8 has_fan; /* some fan inputs can be disabled */
191 s8 temp1;
192 s8 temp1_max;
193 s8 temp1_max_hyst;
194 s16 temp[2];
195 s16 temp_max[2];
196 s16 temp_max_hyst[2];
197};
198
199static inline int is_word_sized(u16 reg)
200{
201 return (((reg & 0xff00) == 0x100
202 || (reg & 0xff00) == 0x200)
203 && ((reg & 0x00ff) == 0x50
204 || (reg & 0x00ff) == 0x53
205 || (reg & 0x00ff) == 0x55));
206}
207
208/* We assume that the default bank is 0, thus the following two functions do
209 nothing for registers which live in bank 0. For others, they respectively
210 set the bank register to the correct value (before the register is
211 accessed), and back to 0 (afterwards). */
212static inline void w83627ehf_set_bank(struct i2c_client *client, u16 reg)
213{
214 if (reg & 0xff00) {
215 outb_p(W83627EHF_REG_BANK, client->addr + ADDR_REG_OFFSET);
216 outb_p(reg >> 8, client->addr + DATA_REG_OFFSET);
217 }
218}
219
220static inline void w83627ehf_reset_bank(struct i2c_client *client, u16 reg)
221{
222 if (reg & 0xff00) {
223 outb_p(W83627EHF_REG_BANK, client->addr + ADDR_REG_OFFSET);
224 outb_p(0, client->addr + DATA_REG_OFFSET);
225 }
226}
227
228static u16 w83627ehf_read_value(struct i2c_client *client, u16 reg)
229{
230 struct w83627ehf_data *data = i2c_get_clientdata(client);
231 int res, word_sized = is_word_sized(reg);
232
233 down(&data->lock);
234
235 w83627ehf_set_bank(client, reg);
236 outb_p(reg & 0xff, client->addr + ADDR_REG_OFFSET);
237 res = inb_p(client->addr + DATA_REG_OFFSET);
238 if (word_sized) {
239 outb_p((reg & 0xff) + 1,
240 client->addr + ADDR_REG_OFFSET);
241 res = (res << 8) + inb_p(client->addr + DATA_REG_OFFSET);
242 }
243 w83627ehf_reset_bank(client, reg);
244
245 up(&data->lock);
246
247 return res;
248}
249
250static int w83627ehf_write_value(struct i2c_client *client, u16 reg, u16 value)
251{
252 struct w83627ehf_data *data = i2c_get_clientdata(client);
253 int word_sized = is_word_sized(reg);
254
255 down(&data->lock);
256
257 w83627ehf_set_bank(client, reg);
258 outb_p(reg & 0xff, client->addr + ADDR_REG_OFFSET);
259 if (word_sized) {
260 outb_p(value >> 8, client->addr + DATA_REG_OFFSET);
261 outb_p((reg & 0xff) + 1,
262 client->addr + ADDR_REG_OFFSET);
263 }
264 outb_p(value & 0xff, client->addr + DATA_REG_OFFSET);
265 w83627ehf_reset_bank(client, reg);
266
267 up(&data->lock);
268 return 0;
269}
270
271/* This function assumes that the caller holds data->update_lock */
272static void w83627ehf_write_fan_div(struct i2c_client *client, int nr)
273{
274 struct w83627ehf_data *data = i2c_get_clientdata(client);
275 u8 reg;
276
277 switch (nr) {
278 case 0:
279 reg = (w83627ehf_read_value(client, W83627EHF_REG_FANDIV1) & 0xcf)
280 | ((data->fan_div[0] & 0x03) << 4);
281 w83627ehf_write_value(client, W83627EHF_REG_FANDIV1, reg);
282 reg = (w83627ehf_read_value(client, W83627EHF_REG_VBAT) & 0xdf)
283 | ((data->fan_div[0] & 0x04) << 3);
284 w83627ehf_write_value(client, W83627EHF_REG_VBAT, reg);
285 break;
286 case 1:
287 reg = (w83627ehf_read_value(client, W83627EHF_REG_FANDIV1) & 0x3f)
288 | ((data->fan_div[1] & 0x03) << 6);
289 w83627ehf_write_value(client, W83627EHF_REG_FANDIV1, reg);
290 reg = (w83627ehf_read_value(client, W83627EHF_REG_VBAT) & 0xbf)
291 | ((data->fan_div[1] & 0x04) << 4);
292 w83627ehf_write_value(client, W83627EHF_REG_VBAT, reg);
293 break;
294 case 2:
295 reg = (w83627ehf_read_value(client, W83627EHF_REG_FANDIV2) & 0x3f)
296 | ((data->fan_div[2] & 0x03) << 6);
297 w83627ehf_write_value(client, W83627EHF_REG_FANDIV2, reg);
298 reg = (w83627ehf_read_value(client, W83627EHF_REG_VBAT) & 0x7f)
299 | ((data->fan_div[2] & 0x04) << 5);
300 w83627ehf_write_value(client, W83627EHF_REG_VBAT, reg);
301 break;
302 case 3:
303 reg = (w83627ehf_read_value(client, W83627EHF_REG_DIODE) & 0xfc)
304 | (data->fan_div[3] & 0x03);
305 w83627ehf_write_value(client, W83627EHF_REG_DIODE, reg);
306 reg = (w83627ehf_read_value(client, W83627EHF_REG_SMI_OVT) & 0x7f)
307 | ((data->fan_div[3] & 0x04) << 5);
308 w83627ehf_write_value(client, W83627EHF_REG_SMI_OVT, reg);
309 break;
310 case 4:
311 reg = (w83627ehf_read_value(client, W83627EHF_REG_DIODE) & 0x73)
312 | ((data->fan_div[4] & 0x03) << 3)
313 | ((data->fan_div[4] & 0x04) << 5);
314 w83627ehf_write_value(client, W83627EHF_REG_DIODE, reg);
315 break;
316 }
317}
318
319static struct w83627ehf_data *w83627ehf_update_device(struct device *dev)
320{
321 struct i2c_client *client = to_i2c_client(dev);
322 struct w83627ehf_data *data = i2c_get_clientdata(client);
323 int i;
324
325 down(&data->update_lock);
326
327 if (time_after(jiffies, data->last_updated + HZ)
328 || !data->valid) {
329 /* Fan clock dividers */
330 i = w83627ehf_read_value(client, W83627EHF_REG_FANDIV1);
331 data->fan_div[0] = (i >> 4) & 0x03;
332 data->fan_div[1] = (i >> 6) & 0x03;
333 i = w83627ehf_read_value(client, W83627EHF_REG_FANDIV2);
334 data->fan_div[2] = (i >> 6) & 0x03;
335 i = w83627ehf_read_value(client, W83627EHF_REG_VBAT);
336 data->fan_div[0] |= (i >> 3) & 0x04;
337 data->fan_div[1] |= (i >> 4) & 0x04;
338 data->fan_div[2] |= (i >> 5) & 0x04;
339 if (data->has_fan & ((1 << 3) | (1 << 4))) {
340 i = w83627ehf_read_value(client, W83627EHF_REG_DIODE);
341 data->fan_div[3] = i & 0x03;
342 data->fan_div[4] = ((i >> 2) & 0x03)
343 | ((i >> 5) & 0x04);
344 }
345 if (data->has_fan & (1 << 3)) {
346 i = w83627ehf_read_value(client, W83627EHF_REG_SMI_OVT);
347 data->fan_div[3] |= (i >> 5) & 0x04;
348 }
349
350 /* Measured fan speeds and limits */
351 for (i = 0; i < 5; i++) {
352 if (!(data->has_fan & (1 << i)))
353 continue;
354
355 data->fan[i] = w83627ehf_read_value(client,
356 W83627EHF_REG_FAN[i]);
357 data->fan_min[i] = w83627ehf_read_value(client,
358 W83627EHF_REG_FAN_MIN[i]);
359
360 /* If we failed to measure the fan speed and clock
361 divider can be increased, let's try that for next
362 time */
363 if (data->fan[i] == 0xff
364 && data->fan_div[i] < 0x07) {
365 dev_dbg(&client->dev, "Increasing fan %d "
366 "clock divider from %u to %u\n",
367 i, div_from_reg(data->fan_div[i]),
368 div_from_reg(data->fan_div[i] + 1));
369 data->fan_div[i]++;
370 w83627ehf_write_fan_div(client, i);
371 /* Preserve min limit if possible */
372 if (data->fan_min[i] >= 2
373 && data->fan_min[i] != 255)
374 w83627ehf_write_value(client,
375 W83627EHF_REG_FAN_MIN[i],
376 (data->fan_min[i] /= 2));
377 }
378 }
379
380 /* Measured temperatures and limits */
381 data->temp1 = w83627ehf_read_value(client,
382 W83627EHF_REG_TEMP1);
383 data->temp1_max = w83627ehf_read_value(client,
384 W83627EHF_REG_TEMP1_OVER);
385 data->temp1_max_hyst = w83627ehf_read_value(client,
386 W83627EHF_REG_TEMP1_HYST);
387 for (i = 0; i < 2; i++) {
388 data->temp[i] = w83627ehf_read_value(client,
389 W83627EHF_REG_TEMP[i]);
390 data->temp_max[i] = w83627ehf_read_value(client,
391 W83627EHF_REG_TEMP_OVER[i]);
392 data->temp_max_hyst[i] = w83627ehf_read_value(client,
393 W83627EHF_REG_TEMP_HYST[i]);
394 }
395
396 data->last_updated = jiffies;
397 data->valid = 1;
398 }
399
400 up(&data->update_lock);
401 return data;
402}
403
404/*
405 * Sysfs callback functions
406 */
407
408#define show_fan_reg(reg) \
409static ssize_t \
410show_##reg(struct device *dev, char *buf, int nr) \
411{ \
412 struct w83627ehf_data *data = w83627ehf_update_device(dev); \
413 return sprintf(buf, "%d\n", \
414 fan_from_reg(data->reg[nr], \
415 div_from_reg(data->fan_div[nr]))); \
416}
417show_fan_reg(fan);
418show_fan_reg(fan_min);
419
420static ssize_t
421show_fan_div(struct device *dev, char *buf, int nr)
422{
423 struct w83627ehf_data *data = w83627ehf_update_device(dev);
424 return sprintf(buf, "%u\n",
425 div_from_reg(data->fan_div[nr]));
426}
427
428static ssize_t
429store_fan_min(struct device *dev, const char *buf, size_t count, int nr)
430{
431 struct i2c_client *client = to_i2c_client(dev);
432 struct w83627ehf_data *data = i2c_get_clientdata(client);
433 unsigned int val = simple_strtoul(buf, NULL, 10);
434 unsigned int reg;
435 u8 new_div;
436
437 down(&data->update_lock);
438 if (!val) {
439 /* No min limit, alarm disabled */
440 data->fan_min[nr] = 255;
441 new_div = data->fan_div[nr]; /* No change */
442 dev_info(dev, "fan%u low limit and alarm disabled\n", nr + 1);
443 } else if ((reg = 1350000U / val) >= 128 * 255) {
444 /* Speed below this value cannot possibly be represented,
445 even with the highest divider (128) */
446 data->fan_min[nr] = 254;
447 new_div = 7; /* 128 == (1 << 7) */
448 dev_warn(dev, "fan%u low limit %u below minimum %u, set to "
449 "minimum\n", nr + 1, val, fan_from_reg(254, 128));
450 } else if (!reg) {
451 /* Speed above this value cannot possibly be represented,
452 even with the lowest divider (1) */
453 data->fan_min[nr] = 1;
454 new_div = 0; /* 1 == (1 << 0) */
455 dev_warn(dev, "fan%u low limit %u above maximum %u, set to "
Jean Delvareb9110b12005-05-02 23:08:22 +0200456 "maximum\n", nr + 1, val, fan_from_reg(1, 1));
Jean Delvare08e7e272005-04-25 22:43:25 +0200457 } else {
458 /* Automatically pick the best divider, i.e. the one such
459 that the min limit will correspond to a register value
460 in the 96..192 range */
461 new_div = 0;
462 while (reg > 192 && new_div < 7) {
463 reg >>= 1;
464 new_div++;
465 }
466 data->fan_min[nr] = reg;
467 }
468
469 /* Write both the fan clock divider (if it changed) and the new
470 fan min (unconditionally) */
471 if (new_div != data->fan_div[nr]) {
472 if (new_div > data->fan_div[nr])
473 data->fan[nr] >>= (data->fan_div[nr] - new_div);
474 else
475 data->fan[nr] <<= (new_div - data->fan_div[nr]);
476
477 dev_dbg(dev, "fan%u clock divider changed from %u to %u\n",
478 nr + 1, div_from_reg(data->fan_div[nr]),
479 div_from_reg(new_div));
480 data->fan_div[nr] = new_div;
481 w83627ehf_write_fan_div(client, nr);
482 }
483 w83627ehf_write_value(client, W83627EHF_REG_FAN_MIN[nr],
484 data->fan_min[nr]);
485 up(&data->update_lock);
486
487 return count;
488}
489
490#define sysfs_fan_offset(offset) \
491static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700492show_reg_fan_##offset(struct device *dev, struct device_attribute *attr, \
493 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200494{ \
495 return show_fan(dev, buf, offset-1); \
496} \
497static DEVICE_ATTR(fan##offset##_input, S_IRUGO, \
498 show_reg_fan_##offset, NULL);
499
500#define sysfs_fan_min_offset(offset) \
501static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700502show_reg_fan##offset##_min(struct device *dev, struct device_attribute *attr, \
503 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200504{ \
505 return show_fan_min(dev, buf, offset-1); \
506} \
507static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700508store_reg_fan##offset##_min(struct device *dev, struct device_attribute *attr, \
509 const char *buf, size_t count) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200510{ \
511 return store_fan_min(dev, buf, count, offset-1); \
512} \
513static DEVICE_ATTR(fan##offset##_min, S_IRUGO | S_IWUSR, \
514 show_reg_fan##offset##_min, \
515 store_reg_fan##offset##_min);
516
517#define sysfs_fan_div_offset(offset) \
518static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700519show_reg_fan##offset##_div(struct device *dev, struct device_attribute *attr, \
520 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200521{ \
522 return show_fan_div(dev, buf, offset - 1); \
523} \
524static DEVICE_ATTR(fan##offset##_div, S_IRUGO, \
525 show_reg_fan##offset##_div, NULL);
526
527sysfs_fan_offset(1);
528sysfs_fan_min_offset(1);
529sysfs_fan_div_offset(1);
530sysfs_fan_offset(2);
531sysfs_fan_min_offset(2);
532sysfs_fan_div_offset(2);
533sysfs_fan_offset(3);
534sysfs_fan_min_offset(3);
535sysfs_fan_div_offset(3);
536sysfs_fan_offset(4);
537sysfs_fan_min_offset(4);
538sysfs_fan_div_offset(4);
539sysfs_fan_offset(5);
540sysfs_fan_min_offset(5);
541sysfs_fan_div_offset(5);
542
543#define show_temp1_reg(reg) \
544static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700545show_##reg(struct device *dev, struct device_attribute *attr, \
546 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200547{ \
548 struct w83627ehf_data *data = w83627ehf_update_device(dev); \
549 return sprintf(buf, "%d\n", temp1_from_reg(data->reg)); \
550}
551show_temp1_reg(temp1);
552show_temp1_reg(temp1_max);
553show_temp1_reg(temp1_max_hyst);
554
555#define store_temp1_reg(REG, reg) \
556static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700557store_temp1_##reg(struct device *dev, struct device_attribute *attr, \
558 const char *buf, size_t count) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200559{ \
560 struct i2c_client *client = to_i2c_client(dev); \
561 struct w83627ehf_data *data = i2c_get_clientdata(client); \
562 u32 val = simple_strtoul(buf, NULL, 10); \
563 \
564 down(&data->update_lock); \
565 data->temp1_##reg = temp1_to_reg(val); \
566 w83627ehf_write_value(client, W83627EHF_REG_TEMP1_##REG, \
567 data->temp1_##reg); \
568 up(&data->update_lock); \
569 return count; \
570}
571store_temp1_reg(OVER, max);
572store_temp1_reg(HYST, max_hyst);
573
574static DEVICE_ATTR(temp1_input, S_IRUGO, show_temp1, NULL);
575static DEVICE_ATTR(temp1_max, S_IRUGO| S_IWUSR,
576 show_temp1_max, store_temp1_max);
577static DEVICE_ATTR(temp1_max_hyst, S_IRUGO| S_IWUSR,
578 show_temp1_max_hyst, store_temp1_max_hyst);
579
580#define show_temp_reg(reg) \
581static ssize_t \
582show_##reg (struct device *dev, char *buf, int nr) \
583{ \
584 struct w83627ehf_data *data = w83627ehf_update_device(dev); \
585 return sprintf(buf, "%d\n", \
586 LM75_TEMP_FROM_REG(data->reg[nr])); \
587}
588show_temp_reg(temp);
589show_temp_reg(temp_max);
590show_temp_reg(temp_max_hyst);
591
592#define store_temp_reg(REG, reg) \
593static ssize_t \
594store_##reg (struct device *dev, const char *buf, size_t count, int nr) \
595{ \
596 struct i2c_client *client = to_i2c_client(dev); \
597 struct w83627ehf_data *data = i2c_get_clientdata(client); \
598 u32 val = simple_strtoul(buf, NULL, 10); \
599 \
600 down(&data->update_lock); \
601 data->reg[nr] = LM75_TEMP_TO_REG(val); \
602 w83627ehf_write_value(client, W83627EHF_REG_TEMP_##REG[nr], \
603 data->reg[nr]); \
604 up(&data->update_lock); \
605 return count; \
606}
607store_temp_reg(OVER, temp_max);
608store_temp_reg(HYST, temp_max_hyst);
609
610#define sysfs_temp_offset(offset) \
611static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700612show_reg_temp##offset (struct device *dev, struct device_attribute *attr, \
613 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200614{ \
615 return show_temp(dev, buf, offset - 2); \
616} \
617static DEVICE_ATTR(temp##offset##_input, S_IRUGO, \
618 show_reg_temp##offset, NULL);
619
620#define sysfs_temp_reg_offset(reg, offset) \
621static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700622show_reg_temp##offset##_##reg(struct device *dev, struct device_attribute *attr, \
623 char *buf) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200624{ \
625 return show_temp_##reg(dev, buf, offset - 2); \
626} \
627static ssize_t \
Greg Kroah-Hartman6f637a62005-06-21 21:01:59 -0700628store_reg_temp##offset##_##reg(struct device *dev, struct device_attribute *attr, \
629 const char *buf, size_t count) \
Jean Delvare08e7e272005-04-25 22:43:25 +0200630{ \
631 return store_temp_##reg(dev, buf, count, offset - 2); \
632} \
633static DEVICE_ATTR(temp##offset##_##reg, S_IRUGO| S_IWUSR, \
634 show_reg_temp##offset##_##reg, \
635 store_reg_temp##offset##_##reg);
636
637sysfs_temp_offset(2);
638sysfs_temp_reg_offset(max, 2);
639sysfs_temp_reg_offset(max_hyst, 2);
640sysfs_temp_offset(3);
641sysfs_temp_reg_offset(max, 3);
642sysfs_temp_reg_offset(max_hyst, 3);
643
644/*
645 * Driver and client management
646 */
647
648static struct i2c_driver w83627ehf_driver;
649
650static void w83627ehf_init_client(struct i2c_client *client)
651{
652 int i;
653 u8 tmp;
654
655 /* Start monitoring is needed */
656 tmp = w83627ehf_read_value(client, W83627EHF_REG_CONFIG);
657 if (!(tmp & 0x01))
658 w83627ehf_write_value(client, W83627EHF_REG_CONFIG,
659 tmp | 0x01);
660
661 /* Enable temp2 and temp3 if needed */
662 for (i = 0; i < 2; i++) {
663 tmp = w83627ehf_read_value(client,
664 W83627EHF_REG_TEMP_CONFIG[i]);
665 if (tmp & 0x01)
666 w83627ehf_write_value(client,
667 W83627EHF_REG_TEMP_CONFIG[i],
668 tmp & 0xfe);
669 }
670}
671
672static int w83627ehf_detect(struct i2c_adapter *adapter, int address, int kind)
673{
674 struct i2c_client *client;
675 struct w83627ehf_data *data;
676 int i, err = 0;
677
678 if (!i2c_is_isa_adapter(adapter))
679 return 0;
680
681 if (!request_region(address, REGION_LENGTH, w83627ehf_driver.name)) {
682 err = -EBUSY;
683 goto exit;
684 }
685
686 if (!(data = kmalloc(sizeof(struct w83627ehf_data), GFP_KERNEL))) {
687 err = -ENOMEM;
688 goto exit_release;
689 }
690 memset(data, 0, sizeof(struct w83627ehf_data));
691
692 client = &data->client;
693 i2c_set_clientdata(client, data);
694 client->addr = address;
695 init_MUTEX(&data->lock);
696 client->adapter = adapter;
697 client->driver = &w83627ehf_driver;
698 client->flags = 0;
699
700 strlcpy(client->name, "w83627ehf", I2C_NAME_SIZE);
701 data->valid = 0;
702 init_MUTEX(&data->update_lock);
703
704 /* Tell the i2c layer a new client has arrived */
705 if ((err = i2c_attach_client(client)))
706 goto exit_free;
707
708 /* Initialize the chip */
709 w83627ehf_init_client(client);
710
711 /* A few vars need to be filled upon startup */
712 for (i = 0; i < 5; i++)
713 data->fan_min[i] = w83627ehf_read_value(client,
714 W83627EHF_REG_FAN_MIN[i]);
715
716 /* It looks like fan4 and fan5 pins can be alternatively used
717 as fan on/off switches */
718 data->has_fan = 0x07; /* fan1, fan2 and fan3 */
719 i = w83627ehf_read_value(client, W83627EHF_REG_FANDIV1);
720 if (i & (1 << 2))
721 data->has_fan |= (1 << 3);
722 if (i & (1 << 0))
723 data->has_fan |= (1 << 4);
724
725 /* Register sysfs hooks */
726 device_create_file(&client->dev, &dev_attr_fan1_input);
727 device_create_file(&client->dev, &dev_attr_fan1_min);
728 device_create_file(&client->dev, &dev_attr_fan1_div);
729 device_create_file(&client->dev, &dev_attr_fan2_input);
730 device_create_file(&client->dev, &dev_attr_fan2_min);
731 device_create_file(&client->dev, &dev_attr_fan2_div);
732 device_create_file(&client->dev, &dev_attr_fan3_input);
733 device_create_file(&client->dev, &dev_attr_fan3_min);
734 device_create_file(&client->dev, &dev_attr_fan3_div);
735
736 if (data->has_fan & (1 << 3)) {
737 device_create_file(&client->dev, &dev_attr_fan4_input);
738 device_create_file(&client->dev, &dev_attr_fan4_min);
739 device_create_file(&client->dev, &dev_attr_fan4_div);
740 }
741 if (data->has_fan & (1 << 4)) {
742 device_create_file(&client->dev, &dev_attr_fan5_input);
743 device_create_file(&client->dev, &dev_attr_fan5_min);
744 device_create_file(&client->dev, &dev_attr_fan5_div);
745 }
746
747 device_create_file(&client->dev, &dev_attr_temp1_input);
748 device_create_file(&client->dev, &dev_attr_temp1_max);
749 device_create_file(&client->dev, &dev_attr_temp1_max_hyst);
750 device_create_file(&client->dev, &dev_attr_temp2_input);
751 device_create_file(&client->dev, &dev_attr_temp2_max);
752 device_create_file(&client->dev, &dev_attr_temp2_max_hyst);
753 device_create_file(&client->dev, &dev_attr_temp3_input);
754 device_create_file(&client->dev, &dev_attr_temp3_max);
755 device_create_file(&client->dev, &dev_attr_temp3_max_hyst);
756
757 return 0;
758
759exit_free:
760 kfree(data);
761exit_release:
762 release_region(address, REGION_LENGTH);
763exit:
764 return err;
765}
766
767static int w83627ehf_attach_adapter(struct i2c_adapter *adapter)
768{
769 if (!(adapter->class & I2C_CLASS_HWMON))
770 return 0;
771 return i2c_detect(adapter, &addr_data, w83627ehf_detect);
772}
773
774static int w83627ehf_detach_client(struct i2c_client *client)
775{
776 int err;
777
778 if ((err = i2c_detach_client(client))) {
779 dev_err(&client->dev, "Client deregistration failed, "
780 "client not detached.\n");
781 return err;
782 }
783 release_region(client->addr, REGION_LENGTH);
784 kfree(i2c_get_clientdata(client));
785
786 return 0;
787}
788
789static struct i2c_driver w83627ehf_driver = {
790 .owner = THIS_MODULE,
791 .name = "w83627ehf",
792 .flags = I2C_DF_NOTIFY,
793 .attach_adapter = w83627ehf_attach_adapter,
794 .detach_client = w83627ehf_detach_client,
795};
796
797static int __init w83627ehf_find(int sioaddr, int *address)
798{
799 u16 val;
800
801 REG = sioaddr;
802 VAL = sioaddr + 1;
803 superio_enter();
804
805 val = (superio_inb(SIO_REG_DEVID) << 8)
806 | superio_inb(SIO_REG_DEVID + 1);
807 if ((val & SIO_ID_MASK) != SIO_W83627EHF_ID) {
808 superio_exit();
809 return -ENODEV;
810 }
811
812 superio_select(W83627EHF_LD_HWM);
813 val = (superio_inb(SIO_REG_ADDR) << 8)
814 | superio_inb(SIO_REG_ADDR + 1);
815 *address = val & ~(REGION_LENGTH - 1);
816 if (*address == 0) {
817 superio_exit();
818 return -ENODEV;
819 }
820
821 /* Activate logical device if needed */
822 val = superio_inb(SIO_REG_ENABLE);
823 if (!(val & 0x01))
824 superio_outb(SIO_REG_ENABLE, val | 0x01);
825
826 superio_exit();
827 return 0;
828}
829
830static int __init sensors_w83627ehf_init(void)
831{
832 if (w83627ehf_find(0x2e, &normal_isa[0])
833 && w83627ehf_find(0x4e, &normal_isa[0]))
834 return -ENODEV;
835
836 return i2c_add_driver(&w83627ehf_driver);
837}
838
839static void __exit sensors_w83627ehf_exit(void)
840{
841 i2c_del_driver(&w83627ehf_driver);
842}
843
844MODULE_AUTHOR("Jean Delvare <khali@linux-fr.org>");
845MODULE_DESCRIPTION("W83627EHF driver");
846MODULE_LICENSE("GPL");
847
848module_init(sensors_w83627ehf_init);
849module_exit(sensors_w83627ehf_exit);