sis5595.c 25.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5
/*
    sis5595.c - Part of lm_sensors, Linux kernel modules
		for hardware monitoring

    Copyright (C) 1998 - 2001 Frodo Looijaard <frodol@dds.nl>,
6
			Kyösti Mälkki <kmalkki@cc.hut.fi>, and
L
Linus Torvalds 已提交
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
			Mark D. Studebaker <mdsxyz123@yahoo.com>
    Ported to Linux 2.6 by Aurelien Jarno <aurelien@aurel32.net> with
    the help of Jean Delvare <khali@linux-fr.org>

    This program is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation; either version 2 of the License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program; if not, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/

/*
   SiS southbridge has a LM78-like chip integrated on the same IC.
   This driver is a customized copy of lm78.c
   
   Supports following revisions:
	Version		PCI ID		PCI Revision
	1		1039/0008	AF or less
	2		1039/0008	B0 or greater

   Note: these chips contain a 0008 device which is incompatible with the
	 5595. We recognize these by the presence of the listed
	 "blacklist" PCI ID and refuse to load.

   NOT SUPPORTED	PCI ID		BLACKLIST PCI ID	
	 540		0008		0540
	 550		0008		0550
	5513		0008		5511
	5581		0008		5597
	5582		0008		5597
	5597		0008		5597
	5598		0008		5597/5598
	 630		0008		0630
	 645		0008		0645
	 730		0008		0730
	 735		0008		0735
*/

53 54
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

L
Linus Torvalds 已提交
55 56 57 58
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/ioport.h>
#include <linux/pci.h>
59
#include <linux/platform_device.h>
60
#include <linux/hwmon.h>
61
#include <linux/hwmon-sysfs.h>
62
#include <linux/err.h>
L
Linus Torvalds 已提交
63
#include <linux/init.h>
64
#include <linux/jiffies.h>
65
#include <linux/mutex.h>
66
#include <linux/sysfs.h>
67
#include <linux/acpi.h>
68
#include <linux/io.h>
L
Linus Torvalds 已提交
69 70 71 72 73 74 75 76 77


/* If force_addr is set to anything different from 0, we forcibly enable
   the device at the given address. */
static u16 force_addr;
module_param(force_addr, ushort, 0);
MODULE_PARM_DESC(force_addr,
		 "Initialize the base address of the sensors");

78
static struct platform_device *pdev;
L
Linus Torvalds 已提交
79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163

/* Many SIS5595 constants specified below */

/* Length of ISA address segment */
#define SIS5595_EXTENT 8
/* PCI Config Registers */
#define SIS5595_BASE_REG 0x68
#define SIS5595_PIN_REG 0x7A
#define SIS5595_ENABLE_REG 0x7B

/* Where are the ISA address/data registers relative to the base address */
#define SIS5595_ADDR_REG_OFFSET 5
#define SIS5595_DATA_REG_OFFSET 6

/* The SIS5595 registers */
#define SIS5595_REG_IN_MAX(nr) (0x2b + (nr) * 2)
#define SIS5595_REG_IN_MIN(nr) (0x2c + (nr) * 2)
#define SIS5595_REG_IN(nr) (0x20 + (nr))

#define SIS5595_REG_FAN_MIN(nr) (0x3b + (nr))
#define SIS5595_REG_FAN(nr) (0x28 + (nr))

/* On the first version of the chip, the temp registers are separate.
   On the second version,
   TEMP pin is shared with IN4, configured in PCI register 0x7A.
   The registers are the same as well.
   OVER and HYST are really MAX and MIN. */

#define REV2MIN	0xb0
#define SIS5595_REG_TEMP 	(( data->revision) >= REV2MIN) ? \
					SIS5595_REG_IN(4) : 0x27
#define SIS5595_REG_TEMP_OVER	(( data->revision) >= REV2MIN) ? \
					SIS5595_REG_IN_MAX(4) : 0x39
#define SIS5595_REG_TEMP_HYST	(( data->revision) >= REV2MIN) ? \
					SIS5595_REG_IN_MIN(4) : 0x3a

#define SIS5595_REG_CONFIG 0x40
#define SIS5595_REG_ALARM1 0x41
#define SIS5595_REG_ALARM2 0x42
#define SIS5595_REG_FANDIV 0x47

/* Conversions. Limit checking is only done on the TO_REG
   variants. */

/* IN: mV, (0V to 4.08V)
   REG: 16mV/bit */
static inline u8 IN_TO_REG(unsigned long val)
{
	unsigned long nval = SENSORS_LIMIT(val, 0, 4080);
	return (nval + 8) / 16;
}
#define IN_FROM_REG(val) ((val) *  16)

static inline u8 FAN_TO_REG(long rpm, int div)
{
	if (rpm <= 0)
		return 255;
	return SENSORS_LIMIT((1350000 + rpm * div / 2) / (rpm * div), 1, 254);
}

static inline int FAN_FROM_REG(u8 val, int div)
{
	return val==0 ? -1 : val==255 ? 0 : 1350000/(val*div);
}

/* TEMP: mC (-54.12C to +157.53C)
   REG: 0.83C/bit + 52.12, two's complement  */
static inline int TEMP_FROM_REG(s8 val)
{
	return val * 830 + 52120;
}
static inline s8 TEMP_TO_REG(int val)
{
	int nval = SENSORS_LIMIT(val, -54120, 157530) ;
	return nval<0 ? (nval-5212-415)/830 : (nval-5212+415)/830;
}

/* FAN DIV: 1, 2, 4, or 8 (defaults to 2)
   REG: 0, 1, 2, or 3 (respectively) (defaults to 1) */
static inline u8 DIV_TO_REG(int val)
{
	return val==8 ? 3 : val==4 ? 2 : val==1 ? 0 : 1;
}
#define DIV_FROM_REG(val) (1 << (val))

164 165
/* For each registered chip, we need to keep some data in memory.
   The structure is dynamically allocated. */
L
Linus Torvalds 已提交
166
struct sis5595_data {
167 168
	unsigned short addr;
	const char *name;
169
	struct device *hwmon_dev;
170
	struct mutex lock;
L
Linus Torvalds 已提交
171

172
	struct mutex update_lock;
L
Linus Torvalds 已提交
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191
	char valid;		/* !=0 if following fields are valid */
	unsigned long last_updated;	/* In jiffies */
	char maxins;		/* == 3 if temp enabled, otherwise == 4 */
	u8 revision;		/* Reg. value */

	u8 in[5];		/* Register value */
	u8 in_max[5];		/* Register value */
	u8 in_min[5];		/* Register value */
	u8 fan[2];		/* Register value */
	u8 fan_min[2];		/* Register value */
	s8 temp;		/* Register value */
	s8 temp_over;		/* Register value */
	s8 temp_hyst;		/* Register value */
	u8 fan_div[2];		/* Register encoding, shifted right */
	u16 alarms;		/* Register encoding, combined */
};

static struct pci_dev *s_bridge;	/* pointer to the (only) sis5595 */

192
static int sis5595_probe(struct platform_device *pdev);
193
static int __devexit sis5595_remove(struct platform_device *pdev);
L
Linus Torvalds 已提交
194

195 196
static int sis5595_read_value(struct sis5595_data *data, u8 reg);
static void sis5595_write_value(struct sis5595_data *data, u8 reg, u8 value);
L
Linus Torvalds 已提交
197
static struct sis5595_data *sis5595_update_device(struct device *dev);
198
static void sis5595_init_device(struct sis5595_data *data);
L
Linus Torvalds 已提交
199

200
static struct platform_driver sis5595_driver = {
201
	.driver = {
J
Jean Delvare 已提交
202
		.owner	= THIS_MODULE,
203 204
		.name	= "sis5595",
	},
205 206
	.probe		= sis5595_probe,
	.remove		= __devexit_p(sis5595_remove),
L
Linus Torvalds 已提交
207 208 209
};

/* 4 Voltages */
210 211
static ssize_t show_in(struct device *dev, struct device_attribute *da,
		       char *buf)
L
Linus Torvalds 已提交
212 213
{
	struct sis5595_data *data = sis5595_update_device(dev);
214 215
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
216 217 218
	return sprintf(buf, "%d\n", IN_FROM_REG(data->in[nr]));
}

219 220
static ssize_t show_in_min(struct device *dev, struct device_attribute *da,
			   char *buf)
L
Linus Torvalds 已提交
221 222
{
	struct sis5595_data *data = sis5595_update_device(dev);
223 224
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
225 226 227
	return sprintf(buf, "%d\n", IN_FROM_REG(data->in_min[nr]));
}

228 229
static ssize_t show_in_max(struct device *dev, struct device_attribute *da,
			   char *buf)
L
Linus Torvalds 已提交
230 231
{
	struct sis5595_data *data = sis5595_update_device(dev);
232 233
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
234 235 236
	return sprintf(buf, "%d\n", IN_FROM_REG(data->in_max[nr]));
}

237 238
static ssize_t set_in_min(struct device *dev, struct device_attribute *da,
			  const char *buf, size_t count)
L
Linus Torvalds 已提交
239
{
240
	struct sis5595_data *data = dev_get_drvdata(dev);
241 242
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
243 244
	unsigned long val = simple_strtoul(buf, NULL, 10);

245
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
246
	data->in_min[nr] = IN_TO_REG(val);
247
	sis5595_write_value(data, SIS5595_REG_IN_MIN(nr), data->in_min[nr]);
248
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
249 250 251
	return count;
}

252 253
static ssize_t set_in_max(struct device *dev, struct device_attribute *da,
			  const char *buf, size_t count)
L
Linus Torvalds 已提交
254
{
255
	struct sis5595_data *data = dev_get_drvdata(dev);
256 257
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
258 259
	unsigned long val = simple_strtoul(buf, NULL, 10);

260
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
261
	data->in_max[nr] = IN_TO_REG(val);
262
	sis5595_write_value(data, SIS5595_REG_IN_MAX(nr), data->in_max[nr]);
263
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
264 265 266 267
	return count;
}

#define show_in_offset(offset)					\
268 269 270 271 272 273
static SENSOR_DEVICE_ATTR(in##offset##_input, S_IRUGO,		\
		show_in, NULL, offset);				\
static SENSOR_DEVICE_ATTR(in##offset##_min, S_IRUGO | S_IWUSR,	\
		show_in_min, set_in_min, offset);		\
static SENSOR_DEVICE_ATTR(in##offset##_max, S_IRUGO | S_IWUSR,	\
		show_in_max, set_in_max, offset);
L
Linus Torvalds 已提交
274 275 276 277 278 279 280 281

show_in_offset(0);
show_in_offset(1);
show_in_offset(2);
show_in_offset(3);
show_in_offset(4);

/* Temperature */
282
static ssize_t show_temp(struct device *dev, struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
283 284 285 286 287
{
	struct sis5595_data *data = sis5595_update_device(dev);
	return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp));
}

288
static ssize_t show_temp_over(struct device *dev, struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
289 290 291 292 293
{
	struct sis5595_data *data = sis5595_update_device(dev);
	return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp_over));
}

294
static ssize_t set_temp_over(struct device *dev, struct device_attribute *attr, const char *buf, size_t count)
L
Linus Torvalds 已提交
295
{
296
	struct sis5595_data *data = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
297 298
	long val = simple_strtol(buf, NULL, 10);

299
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
300
	data->temp_over = TEMP_TO_REG(val);
301
	sis5595_write_value(data, SIS5595_REG_TEMP_OVER, data->temp_over);
302
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
303 304 305
	return count;
}

306
static ssize_t show_temp_hyst(struct device *dev, struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
307 308 309 310 311
{
	struct sis5595_data *data = sis5595_update_device(dev);
	return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp_hyst));
}

312
static ssize_t set_temp_hyst(struct device *dev, struct device_attribute *attr, const char *buf, size_t count)
L
Linus Torvalds 已提交
313
{
314
	struct sis5595_data *data = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
315 316
	long val = simple_strtol(buf, NULL, 10);

317
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
318
	data->temp_hyst = TEMP_TO_REG(val);
319
	sis5595_write_value(data, SIS5595_REG_TEMP_HYST, data->temp_hyst);
320
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
321 322 323 324 325 326 327 328 329 330
	return count;
}

static DEVICE_ATTR(temp1_input, S_IRUGO, show_temp, NULL);
static DEVICE_ATTR(temp1_max, S_IRUGO | S_IWUSR,
		show_temp_over, set_temp_over);
static DEVICE_ATTR(temp1_max_hyst, S_IRUGO | S_IWUSR,
		show_temp_hyst, set_temp_hyst);

/* 2 Fans */
331 332
static ssize_t show_fan(struct device *dev, struct device_attribute *da,
			char *buf)
L
Linus Torvalds 已提交
333 334
{
	struct sis5595_data *data = sis5595_update_device(dev);
335 336
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
337 338 339 340
	return sprintf(buf, "%d\n", FAN_FROM_REG(data->fan[nr],
		DIV_FROM_REG(data->fan_div[nr])) );
}

341 342
static ssize_t show_fan_min(struct device *dev, struct device_attribute *da,
			    char *buf)
L
Linus Torvalds 已提交
343 344
{
	struct sis5595_data *data = sis5595_update_device(dev);
345 346
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
347 348 349 350
	return sprintf(buf,"%d\n", FAN_FROM_REG(data->fan_min[nr],
		DIV_FROM_REG(data->fan_div[nr])) );
}

351 352
static ssize_t set_fan_min(struct device *dev, struct device_attribute *da,
			   const char *buf, size_t count)
L
Linus Torvalds 已提交
353
{
354
	struct sis5595_data *data = dev_get_drvdata(dev);
355 356
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
357 358
	unsigned long val = simple_strtoul(buf, NULL, 10);

359
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
360
	data->fan_min[nr] = FAN_TO_REG(val, DIV_FROM_REG(data->fan_div[nr]));
361
	sis5595_write_value(data, SIS5595_REG_FAN_MIN(nr), data->fan_min[nr]);
362
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
363 364 365
	return count;
}

366 367
static ssize_t show_fan_div(struct device *dev, struct device_attribute *da,
			    char *buf)
L
Linus Torvalds 已提交
368 369
{
	struct sis5595_data *data = sis5595_update_device(dev);
370 371
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
372 373 374 375 376
	return sprintf(buf, "%d\n", DIV_FROM_REG(data->fan_div[nr]) );
}

/* Note: we save and restore the fan minimum here, because its value is
   determined in part by the fan divisor.  This follows the principle of
A
Andreas Mohr 已提交
377
   least surprise; the user doesn't expect the fan minimum to change just
L
Linus Torvalds 已提交
378
   because the divisor changed. */
379 380
static ssize_t set_fan_div(struct device *dev, struct device_attribute *da,
			   const char *buf, size_t count)
L
Linus Torvalds 已提交
381
{
382
	struct sis5595_data *data = dev_get_drvdata(dev);
383 384
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
385 386 387 388
	unsigned long min;
	unsigned long val = simple_strtoul(buf, NULL, 10);
	int reg;

389
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
390 391
	min = FAN_FROM_REG(data->fan_min[nr],
			DIV_FROM_REG(data->fan_div[nr]));
392
	reg = sis5595_read_value(data, SIS5595_REG_FANDIV);
L
Linus Torvalds 已提交
393 394 395 396 397 398 399

	switch (val) {
	case 1: data->fan_div[nr] = 0; break;
	case 2: data->fan_div[nr] = 1; break;
	case 4: data->fan_div[nr] = 2; break;
	case 8: data->fan_div[nr] = 3; break;
	default:
400
		dev_err(dev, "fan_div value %ld not "
L
Linus Torvalds 已提交
401
			"supported. Choose one of 1, 2, 4 or 8!\n", val);
402
		mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
403 404 405 406 407 408 409 410 411 412 413
		return -EINVAL;
	}
	
	switch (nr) {
	case 0:
		reg = (reg & 0xcf) | (data->fan_div[nr] << 4);
		break;
	case 1:
		reg = (reg & 0x3f) | (data->fan_div[nr] << 6);
		break;
	}
414
	sis5595_write_value(data, SIS5595_REG_FANDIV, reg);
L
Linus Torvalds 已提交
415 416
	data->fan_min[nr] =
		FAN_TO_REG(min, DIV_FROM_REG(data->fan_div[nr]));
417
	sis5595_write_value(data, SIS5595_REG_FAN_MIN(nr), data->fan_min[nr]);
418
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
419 420 421 422
	return count;
}

#define show_fan_offset(offset)						\
423 424 425 426 427 428
static SENSOR_DEVICE_ATTR(fan##offset##_input, S_IRUGO,			\
		show_fan, NULL, offset - 1);				\
static SENSOR_DEVICE_ATTR(fan##offset##_min, S_IRUGO | S_IWUSR,		\
		show_fan_min, set_fan_min, offset - 1);			\
static SENSOR_DEVICE_ATTR(fan##offset##_div, S_IRUGO | S_IWUSR,		\
		show_fan_div, set_fan_div, offset - 1);
L
Linus Torvalds 已提交
429 430 431 432 433

show_fan_offset(1);
show_fan_offset(2);

/* Alarms */
434
static ssize_t show_alarms(struct device *dev, struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
435 436 437 438 439
{
	struct sis5595_data *data = sis5595_update_device(dev);
	return sprintf(buf, "%d\n", data->alarms);
}
static DEVICE_ATTR(alarms, S_IRUGO, show_alarms, NULL);
440

441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456
static ssize_t show_alarm(struct device *dev, struct device_attribute *da,
			  char *buf)
{
	struct sis5595_data *data = sis5595_update_device(dev);
	int nr = to_sensor_dev_attr(da)->index;
	return sprintf(buf, "%u\n", (data->alarms >> nr) & 1);
}
static SENSOR_DEVICE_ATTR(in0_alarm, S_IRUGO, show_alarm, NULL, 0);
static SENSOR_DEVICE_ATTR(in1_alarm, S_IRUGO, show_alarm, NULL, 1);
static SENSOR_DEVICE_ATTR(in2_alarm, S_IRUGO, show_alarm, NULL, 2);
static SENSOR_DEVICE_ATTR(in3_alarm, S_IRUGO, show_alarm, NULL, 3);
static SENSOR_DEVICE_ATTR(in4_alarm, S_IRUGO, show_alarm, NULL, 15);
static SENSOR_DEVICE_ATTR(fan1_alarm, S_IRUGO, show_alarm, NULL, 6);
static SENSOR_DEVICE_ATTR(fan2_alarm, S_IRUGO, show_alarm, NULL, 7);
static SENSOR_DEVICE_ATTR(temp1_alarm, S_IRUGO, show_alarm, NULL, 15);

457 458 459 460 461 462 463 464
static ssize_t show_name(struct device *dev, struct device_attribute *attr,
			 char *buf)
{
	struct sis5595_data *data = dev_get_drvdata(dev);
	return sprintf(buf, "%s\n", data->name);
}
static DEVICE_ATTR(name, S_IRUGO, show_name, NULL);

465
static struct attribute *sis5595_attributes[] = {
466 467 468
	&sensor_dev_attr_in0_input.dev_attr.attr,
	&sensor_dev_attr_in0_min.dev_attr.attr,
	&sensor_dev_attr_in0_max.dev_attr.attr,
469
	&sensor_dev_attr_in0_alarm.dev_attr.attr,
470 471 472
	&sensor_dev_attr_in1_input.dev_attr.attr,
	&sensor_dev_attr_in1_min.dev_attr.attr,
	&sensor_dev_attr_in1_max.dev_attr.attr,
473
	&sensor_dev_attr_in1_alarm.dev_attr.attr,
474 475 476
	&sensor_dev_attr_in2_input.dev_attr.attr,
	&sensor_dev_attr_in2_min.dev_attr.attr,
	&sensor_dev_attr_in2_max.dev_attr.attr,
477
	&sensor_dev_attr_in2_alarm.dev_attr.attr,
478 479 480
	&sensor_dev_attr_in3_input.dev_attr.attr,
	&sensor_dev_attr_in3_min.dev_attr.attr,
	&sensor_dev_attr_in3_max.dev_attr.attr,
481
	&sensor_dev_attr_in3_alarm.dev_attr.attr,
482 483 484 485

	&sensor_dev_attr_fan1_input.dev_attr.attr,
	&sensor_dev_attr_fan1_min.dev_attr.attr,
	&sensor_dev_attr_fan1_div.dev_attr.attr,
486
	&sensor_dev_attr_fan1_alarm.dev_attr.attr,
487 488 489
	&sensor_dev_attr_fan2_input.dev_attr.attr,
	&sensor_dev_attr_fan2_min.dev_attr.attr,
	&sensor_dev_attr_fan2_div.dev_attr.attr,
490
	&sensor_dev_attr_fan2_alarm.dev_attr.attr,
491 492

	&dev_attr_alarms.attr,
493
	&dev_attr_name.attr,
494 495 496 497 498 499 500
	NULL
};

static const struct attribute_group sis5595_group = {
	.attrs = sis5595_attributes,
};

501
static struct attribute *sis5595_attributes_in4[] = {
502 503 504
	&sensor_dev_attr_in4_input.dev_attr.attr,
	&sensor_dev_attr_in4_min.dev_attr.attr,
	&sensor_dev_attr_in4_max.dev_attr.attr,
505
	&sensor_dev_attr_in4_alarm.dev_attr.attr,
506 507 508 509 510 511
	NULL
};

static const struct attribute_group sis5595_group_in4 = {
	.attrs = sis5595_attributes_in4,
};
512

513
static struct attribute *sis5595_attributes_temp1[] = {
514 515 516
	&dev_attr_temp1_input.attr,
	&dev_attr_temp1_max.attr,
	&dev_attr_temp1_max_hyst.attr,
517
	&sensor_dev_attr_temp1_alarm.dev_attr.attr,
518 519 520
	NULL
};

521 522
static const struct attribute_group sis5595_group_temp1 = {
	.attrs = sis5595_attributes_temp1,
523
};
L
Linus Torvalds 已提交
524 525
 
/* This is called when the module is loaded */
526
static int __devinit sis5595_probe(struct platform_device *pdev)
L
Linus Torvalds 已提交
527 528 529 530
{
	int err = 0;
	int i;
	struct sis5595_data *data;
531
	struct resource *res;
L
Linus Torvalds 已提交
532 533 534
	char val;

	/* Reserve the ISA region */
535 536
	res = platform_get_resource(pdev, IORESOURCE_IO, 0);
	if (!request_region(res->start, SIS5595_EXTENT,
537
			    sis5595_driver.driver.name)) {
L
Linus Torvalds 已提交
538 539 540 541
		err = -EBUSY;
		goto exit;
	}

D
Deepak Saxena 已提交
542
	if (!(data = kzalloc(sizeof(struct sis5595_data), GFP_KERNEL))) {
L
Linus Torvalds 已提交
543 544 545 546
		err = -ENOMEM;
		goto exit_release;
	}

547
	mutex_init(&data->lock);
548 549 550 551
	mutex_init(&data->update_lock);
	data->addr = res->start;
	data->name = "sis5595";
	platform_set_drvdata(pdev, data);
L
Linus Torvalds 已提交
552 553

	/* Check revision and pin registers to determine whether 4 or 5 voltages */
554
	data->revision = s_bridge->revision;
L
Linus Torvalds 已提交
555 556 557 558 559 560 561 562 563 564
	/* 4 voltages, 1 temp */
	data->maxins = 3;
	if (data->revision >= REV2MIN) {
		pci_read_config_byte(s_bridge, SIS5595_PIN_REG, &val);
		if (!(val & 0x80))
			/* 5 voltages, no temps */
			data->maxins = 4;
	}
	
	/* Initialize the SIS5595 chip */
565
	sis5595_init_device(data);
L
Linus Torvalds 已提交
566 567 568

	/* A few vars need to be filled upon startup */
	for (i = 0; i < 2; i++) {
569
		data->fan_min[i] = sis5595_read_value(data,
L
Linus Torvalds 已提交
570 571 572 573
					SIS5595_REG_FAN_MIN(i));
	}

	/* Register sysfs hooks */
574 575
	if ((err = sysfs_create_group(&pdev->dev.kobj, &sis5595_group)))
		goto exit_free;
576
	if (data->maxins == 4) {
577 578
		if ((err = sysfs_create_group(&pdev->dev.kobj,
					      &sis5595_group_in4)))
579 580
			goto exit_remove_files;
	} else {
581 582
		if ((err = sysfs_create_group(&pdev->dev.kobj,
					      &sis5595_group_temp1)))
583 584 585
			goto exit_remove_files;
	}

586 587 588
	data->hwmon_dev = hwmon_device_register(&pdev->dev);
	if (IS_ERR(data->hwmon_dev)) {
		err = PTR_ERR(data->hwmon_dev);
589
		goto exit_remove_files;
590 591
	}

L
Linus Torvalds 已提交
592
	return 0;
593

594
exit_remove_files:
595
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group);
596 597
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_in4);
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_temp1);
L
Linus Torvalds 已提交
598 599 600
exit_free:
	kfree(data);
exit_release:
601
	release_region(res->start, SIS5595_EXTENT);
L
Linus Torvalds 已提交
602 603 604 605
exit:
	return err;
}

606
static int __devexit sis5595_remove(struct platform_device *pdev)
L
Linus Torvalds 已提交
607
{
608
	struct sis5595_data *data = platform_get_drvdata(pdev);
L
Linus Torvalds 已提交
609

610
	hwmon_device_unregister(data->hwmon_dev);
611
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group);
612 613
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_in4);
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_temp1);
L
Linus Torvalds 已提交
614

615 616
	release_region(data->addr, SIS5595_EXTENT);
	platform_set_drvdata(pdev, NULL);
617
	kfree(data);
L
Linus Torvalds 已提交
618 619 620 621 622 623

	return 0;
}


/* ISA access must be locked explicitly. */
624
static int sis5595_read_value(struct sis5595_data *data, u8 reg)
L
Linus Torvalds 已提交
625 626 627
{
	int res;

628
	mutex_lock(&data->lock);
629 630
	outb_p(reg, data->addr + SIS5595_ADDR_REG_OFFSET);
	res = inb_p(data->addr + SIS5595_DATA_REG_OFFSET);
631
	mutex_unlock(&data->lock);
L
Linus Torvalds 已提交
632 633 634
	return res;
}

635
static void sis5595_write_value(struct sis5595_data *data, u8 reg, u8 value)
L
Linus Torvalds 已提交
636
{
637
	mutex_lock(&data->lock);
638 639
	outb_p(reg, data->addr + SIS5595_ADDR_REG_OFFSET);
	outb_p(value, data->addr + SIS5595_DATA_REG_OFFSET);
640
	mutex_unlock(&data->lock);
L
Linus Torvalds 已提交
641 642 643
}

/* Called when we have found a new SIS5595. */
644
static void __devinit sis5595_init_device(struct sis5595_data *data)
L
Linus Torvalds 已提交
645
{
646
	u8 config = sis5595_read_value(data, SIS5595_REG_CONFIG);
L
Linus Torvalds 已提交
647
	if (!(config & 0x01))
648
		sis5595_write_value(data, SIS5595_REG_CONFIG,
L
Linus Torvalds 已提交
649 650 651 652 653
				(config & 0xf7) | 0x01);
}

static struct sis5595_data *sis5595_update_device(struct device *dev)
{
654
	struct sis5595_data *data = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
655 656
	int i;

657
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
658 659 660 661 662 663

	if (time_after(jiffies, data->last_updated + HZ + HZ / 2)
	    || !data->valid) {

		for (i = 0; i <= data->maxins; i++) {
			data->in[i] =
664
			    sis5595_read_value(data, SIS5595_REG_IN(i));
L
Linus Torvalds 已提交
665
			data->in_min[i] =
666
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
667 668
					       SIS5595_REG_IN_MIN(i));
			data->in_max[i] =
669
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
670 671 672 673
					       SIS5595_REG_IN_MAX(i));
		}
		for (i = 0; i < 2; i++) {
			data->fan[i] =
674
			    sis5595_read_value(data, SIS5595_REG_FAN(i));
L
Linus Torvalds 已提交
675
			data->fan_min[i] =
676
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
677 678 679 680
					       SIS5595_REG_FAN_MIN(i));
		}
		if (data->maxins == 3) {
			data->temp =
681
			    sis5595_read_value(data, SIS5595_REG_TEMP);
L
Linus Torvalds 已提交
682
			data->temp_over =
683
			    sis5595_read_value(data, SIS5595_REG_TEMP_OVER);
L
Linus Torvalds 已提交
684
			data->temp_hyst =
685
			    sis5595_read_value(data, SIS5595_REG_TEMP_HYST);
L
Linus Torvalds 已提交
686
		}
687
		i = sis5595_read_value(data, SIS5595_REG_FANDIV);
L
Linus Torvalds 已提交
688 689 690
		data->fan_div[0] = (i >> 4) & 0x03;
		data->fan_div[1] = i >> 6;
		data->alarms =
691 692
		    sis5595_read_value(data, SIS5595_REG_ALARM1) |
		    (sis5595_read_value(data, SIS5595_REG_ALARM2) << 8);
L
Linus Torvalds 已提交
693 694 695 696
		data->last_updated = jiffies;
		data->valid = 1;
	}

697
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
698 699 700 701

	return data;
}

702
static DEFINE_PCI_DEVICE_TABLE(sis5595_pci_ids) = {
L
Linus Torvalds 已提交
703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722
	{ PCI_DEVICE(PCI_VENDOR_ID_SI, PCI_DEVICE_ID_SI_503) },
	{ 0, }
};

MODULE_DEVICE_TABLE(pci, sis5595_pci_ids);

static int blacklist[] __devinitdata = {
	PCI_DEVICE_ID_SI_540,
	PCI_DEVICE_ID_SI_550,
	PCI_DEVICE_ID_SI_630,
	PCI_DEVICE_ID_SI_645,
	PCI_DEVICE_ID_SI_730,
	PCI_DEVICE_ID_SI_735,
	PCI_DEVICE_ID_SI_5511, /* 5513 chip has the 0008 device but
				  that ID shows up in other chips so we
				  use the 5511 ID for recognition */
	PCI_DEVICE_ID_SI_5597,
	PCI_DEVICE_ID_SI_5598,
	0 };

723 724 725 726 727 728 729 730 731 732
static int __devinit sis5595_device_add(unsigned short address)
{
	struct resource res = {
		.start	= address,
		.end	= address + SIS5595_EXTENT - 1,
		.name	= "sis5595",
		.flags	= IORESOURCE_IO,
	};
	int err;

733 734 735 736
	err = acpi_check_resource_conflict(&res);
	if (err)
		goto exit;

737 738 739
	pdev = platform_device_alloc("sis5595", address);
	if (!pdev) {
		err = -ENOMEM;
740
		pr_err("Device allocation failed\n");
741 742 743 744 745
		goto exit;
	}

	err = platform_device_add_resources(pdev, &res, 1);
	if (err) {
746
		pr_err("Device resource addition failed (%d)\n", err);
747 748 749 750 751
		goto exit_device_put;
	}

	err = platform_device_add(pdev);
	if (err) {
752
		pr_err("Device addition failed (%d)\n", err);
753 754 755 756 757 758 759 760 761 762 763
		goto exit_device_put;
	}

	return 0;

exit_device_put:
	platform_device_put(pdev);
exit:
	return err;
}

L
Linus Torvalds 已提交
764 765 766
static int __devinit sis5595_pci_probe(struct pci_dev *dev,
				       const struct pci_device_id *id)
{
767 768
	u16 address;
	u8 enable;
L
Linus Torvalds 已提交
769 770 771
	int *i;

	for (i = blacklist; *i != 0; i++) {
772 773 774 775
		struct pci_dev *d;
		if ((d = pci_get_device(PCI_VENDOR_ID_SI, *i, NULL))) {
			dev_err(&d->dev, "Looked for SIS5595 but found unsupported device %.4x\n", *i);
			pci_dev_put(d);
L
Linus Torvalds 已提交
776 777 778 779
			return -ENODEV;
		}
	}
	
780 781 782 783 784 785
	force_addr &= ~(SIS5595_EXTENT - 1);
	if (force_addr) {
		dev_warn(&dev->dev, "Forcing ISA address 0x%x\n", force_addr);
		pci_write_config_word(dev, SIS5595_BASE_REG, force_addr);
	}

L
Linus Torvalds 已提交
786
	if (PCIBIOS_SUCCESSFUL !=
787 788
	    pci_read_config_word(dev, SIS5595_BASE_REG, &address)) {
		dev_err(&dev->dev, "Failed to read ISA address\n");
L
Linus Torvalds 已提交
789
		return -ENODEV;
790
	}
L
Linus Torvalds 已提交
791
	
792 793
	address &= ~(SIS5595_EXTENT - 1);
	if (!address) {
L
Linus Torvalds 已提交
794 795 796
		dev_err(&dev->dev, "Base address not set - upgrade BIOS or use force_addr=0xaddr\n");
		return -ENODEV;
	}
797 798 799 800 801
	if (force_addr && address != force_addr) {
		/* doesn't work for some chips? */
		dev_err(&dev->dev, "Failed to force ISA address\n");
		return -ENODEV;
	}
L
Linus Torvalds 已提交
802

803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818
	if (PCIBIOS_SUCCESSFUL !=
	    pci_read_config_byte(dev, SIS5595_ENABLE_REG, &enable)) {
		dev_err(&dev->dev, "Failed to read enable register\n");
		return -ENODEV;
	}
	if (!(enable & 0x80)) {
		if ((PCIBIOS_SUCCESSFUL !=
		     pci_write_config_byte(dev, SIS5595_ENABLE_REG,
					   enable | 0x80))
		 || (PCIBIOS_SUCCESSFUL !=
		     pci_read_config_byte(dev, SIS5595_ENABLE_REG, &enable))
		 || (!(enable & 0x80))) {
			/* doesn't work for some chips! */
			dev_err(&dev->dev, "Failed to enable HWM device\n");
			return -ENODEV;
		}
L
Linus Torvalds 已提交
819 820
	}

821 822 823 824 825 826 827 828 829 830
	if (platform_driver_register(&sis5595_driver)) {
		dev_dbg(&dev->dev, "Failed to register sis5595 driver\n");
		goto exit;
	}

	s_bridge = pci_dev_get(dev);
	/* Sets global pdev as a side effect */
	if (sis5595_device_add(address))
		goto exit_unregister;

L
Linus Torvalds 已提交
831 832 833 834 835
	/* Always return failure here.  This is to allow other drivers to bind
	 * to this pci device.  We don't really want to have control over the
	 * pci device, we only wanted to read as few register values from it.
	 */
	return -ENODEV;
836 837 838 839 840 841

exit_unregister:
	pci_dev_put(dev);
	platform_driver_unregister(&sis5595_driver);
exit:
	return -ENODEV;
L
Linus Torvalds 已提交
842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858
}

static struct pci_driver sis5595_pci_driver = {
	.name            = "sis5595",
	.id_table        = sis5595_pci_ids,
	.probe           = sis5595_pci_probe,
};

static int __init sm_sis5595_init(void)
{
	return pci_register_driver(&sis5595_pci_driver);
}

static void __exit sm_sis5595_exit(void)
{
	pci_unregister_driver(&sis5595_pci_driver);
	if (s_bridge != NULL) {
859 860
		platform_device_unregister(pdev);
		platform_driver_unregister(&sis5595_driver);
L
Linus Torvalds 已提交
861 862 863 864 865 866 867 868 869 870 871
		pci_dev_put(s_bridge);
		s_bridge = NULL;
	}
}

MODULE_AUTHOR("Aurelien Jarno <aurelien@aurel32.net>");
MODULE_DESCRIPTION("SiS 5595 Sensor device");
MODULE_LICENSE("GPL");

module_init(sm_sis5595_init);
module_exit(sm_sis5595_exit);