sis5595.c 24.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 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 53 54 55 56
/*
    sis5595.c - Part of lm_sensors, Linux kernel modules
		for hardware monitoring

    Copyright (C) 1998 - 2001 Frodo Looijaard <frodol@dds.nl>,
			Kyösti Mälkki <kmalkki@cc.hut.fi>, and
			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
*/

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


/* 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");

75
static struct platform_device *pdev;
L
Linus Torvalds 已提交
76 77 78 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

/* 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))

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

169
	struct mutex update_lock;
L
Linus Torvalds 已提交
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
	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 */

189
static int sis5595_probe(struct platform_device *pdev);
190
static int __devexit sis5595_remove(struct platform_device *pdev);
L
Linus Torvalds 已提交
191

192 193
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 已提交
194
static struct sis5595_data *sis5595_update_device(struct device *dev);
195
static void sis5595_init_device(struct sis5595_data *data);
L
Linus Torvalds 已提交
196

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

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

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

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

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

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

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

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

#define show_in_offset(offset)					\
265 266 267 268 269 270
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 已提交
271 272 273 274 275 276 277 278

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

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

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

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

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

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

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

314
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
315
	data->temp_hyst = TEMP_TO_REG(val);
316
	sis5595_write_value(data, SIS5595_REG_TEMP_HYST, data->temp_hyst);
317
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
318 319 320 321 322 323 324 325 326 327
	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 */
328 329
static ssize_t show_fan(struct device *dev, struct device_attribute *da,
			char *buf)
L
Linus Torvalds 已提交
330 331
{
	struct sis5595_data *data = sis5595_update_device(dev);
332 333
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
334 335 336 337
	return sprintf(buf, "%d\n", FAN_FROM_REG(data->fan[nr],
		DIV_FROM_REG(data->fan_div[nr])) );
}

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

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

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

363 364
static ssize_t show_fan_div(struct device *dev, struct device_attribute *da,
			    char *buf)
L
Linus Torvalds 已提交
365 366
{
	struct sis5595_data *data = sis5595_update_device(dev);
367 368
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
369 370 371 372 373
	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 已提交
374
   least surprise; the user doesn't expect the fan minimum to change just
L
Linus Torvalds 已提交
375
   because the divisor changed. */
376 377
static ssize_t set_fan_div(struct device *dev, struct device_attribute *da,
			   const char *buf, size_t count)
L
Linus Torvalds 已提交
378
{
379
	struct sis5595_data *data = dev_get_drvdata(dev);
380 381
	struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
	int nr = attr->index;
L
Linus Torvalds 已提交
382 383 384 385
	unsigned long min;
	unsigned long val = simple_strtoul(buf, NULL, 10);
	int reg;

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

	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:
397
		dev_err(dev, "fan_div value %ld not "
L
Linus Torvalds 已提交
398
			"supported. Choose one of 1, 2, 4 or 8!\n", val);
399
		mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
400 401 402 403 404 405 406 407 408 409 410
		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;
	}
411
	sis5595_write_value(data, SIS5595_REG_FANDIV, reg);
L
Linus Torvalds 已提交
412 413
	data->fan_min[nr] =
		FAN_TO_REG(min, DIV_FROM_REG(data->fan_div[nr]));
414
	sis5595_write_value(data, SIS5595_REG_FAN_MIN(nr), data->fan_min[nr]);
415
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
416 417 418 419
	return count;
}

#define show_fan_offset(offset)						\
420 421 422 423 424 425
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 已提交
426 427 428 429 430

show_fan_offset(1);
show_fan_offset(2);

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

438 439 440 441 442 443 444 445
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);

446
static struct attribute *sis5595_attributes[] = {
447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465
	&sensor_dev_attr_in0_input.dev_attr.attr,
	&sensor_dev_attr_in0_min.dev_attr.attr,
	&sensor_dev_attr_in0_max.dev_attr.attr,
	&sensor_dev_attr_in1_input.dev_attr.attr,
	&sensor_dev_attr_in1_min.dev_attr.attr,
	&sensor_dev_attr_in1_max.dev_attr.attr,
	&sensor_dev_attr_in2_input.dev_attr.attr,
	&sensor_dev_attr_in2_min.dev_attr.attr,
	&sensor_dev_attr_in2_max.dev_attr.attr,
	&sensor_dev_attr_in3_input.dev_attr.attr,
	&sensor_dev_attr_in3_min.dev_attr.attr,
	&sensor_dev_attr_in3_max.dev_attr.attr,

	&sensor_dev_attr_fan1_input.dev_attr.attr,
	&sensor_dev_attr_fan1_min.dev_attr.attr,
	&sensor_dev_attr_fan1_div.dev_attr.attr,
	&sensor_dev_attr_fan2_input.dev_attr.attr,
	&sensor_dev_attr_fan2_min.dev_attr.attr,
	&sensor_dev_attr_fan2_div.dev_attr.attr,
466 467

	&dev_attr_alarms.attr,
468
	&dev_attr_name.attr,
469 470 471 472 473 474 475 476
	NULL
};

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

static struct attribute *sis5595_attributes_opt[] = {
477 478 479
	&sensor_dev_attr_in4_input.dev_attr.attr,
	&sensor_dev_attr_in4_min.dev_attr.attr,
	&sensor_dev_attr_in4_max.dev_attr.attr,
480 481 482 483 484 485 486 487 488 489

	&dev_attr_temp1_input.attr,
	&dev_attr_temp1_max.attr,
	&dev_attr_temp1_max_hyst.attr,
	NULL
};

static const struct attribute_group sis5595_group_opt = {
	.attrs = sis5595_attributes_opt,
};
L
Linus Torvalds 已提交
490 491
 
/* This is called when the module is loaded */
492
static int __devinit sis5595_probe(struct platform_device *pdev)
L
Linus Torvalds 已提交
493 494 495 496
{
	int err = 0;
	int i;
	struct sis5595_data *data;
497
	struct resource *res;
L
Linus Torvalds 已提交
498 499 500
	char val;

	/* Reserve the ISA region */
501 502
	res = platform_get_resource(pdev, IORESOURCE_IO, 0);
	if (!request_region(res->start, SIS5595_EXTENT,
503
			    sis5595_driver.driver.name)) {
L
Linus Torvalds 已提交
504 505 506 507
		err = -EBUSY;
		goto exit;
	}

D
Deepak Saxena 已提交
508
	if (!(data = kzalloc(sizeof(struct sis5595_data), GFP_KERNEL))) {
L
Linus Torvalds 已提交
509 510 511 512
		err = -ENOMEM;
		goto exit_release;
	}

513
	mutex_init(&data->lock);
514 515 516 517
	mutex_init(&data->update_lock);
	data->addr = res->start;
	data->name = "sis5595";
	platform_set_drvdata(pdev, data);
L
Linus Torvalds 已提交
518 519

	/* Check revision and pin registers to determine whether 4 or 5 voltages */
520
	pci_read_config_byte(s_bridge, PCI_REVISION_ID, &data->revision);
L
Linus Torvalds 已提交
521 522 523 524 525 526 527 528 529 530
	/* 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 */
531
	sis5595_init_device(data);
L
Linus Torvalds 已提交
532 533 534

	/* A few vars need to be filled upon startup */
	for (i = 0; i < 2; i++) {
535
		data->fan_min[i] = sis5595_read_value(data,
L
Linus Torvalds 已提交
536 537 538 539
					SIS5595_REG_FAN_MIN(i));
	}

	/* Register sysfs hooks */
540 541
	if ((err = sysfs_create_group(&pdev->dev.kobj, &sis5595_group)))
		goto exit_free;
542
	if (data->maxins == 4) {
543
		if ((err = device_create_file(&pdev->dev,
544
					&sensor_dev_attr_in4_input.dev_attr))
545
		 || (err = device_create_file(&pdev->dev,
546
					&sensor_dev_attr_in4_min.dev_attr))
547
		 || (err = device_create_file(&pdev->dev,
548
					&sensor_dev_attr_in4_max.dev_attr)))
549 550
			goto exit_remove_files;
	} else {
551
		if ((err = device_create_file(&pdev->dev,
552
					      &dev_attr_temp1_input))
553
		 || (err = device_create_file(&pdev->dev,
554
					      &dev_attr_temp1_max))
555
		 || (err = device_create_file(&pdev->dev,
556 557 558 559
					      &dev_attr_temp1_max_hyst)))
			goto exit_remove_files;
	}

560 561 562
	data->hwmon_dev = hwmon_device_register(&pdev->dev);
	if (IS_ERR(data->hwmon_dev)) {
		err = PTR_ERR(data->hwmon_dev);
563
		goto exit_remove_files;
564 565
	}

L
Linus Torvalds 已提交
566
	return 0;
567

568
exit_remove_files:
569 570
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group);
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_opt);
L
Linus Torvalds 已提交
571 572 573
exit_free:
	kfree(data);
exit_release:
574
	release_region(res->start, SIS5595_EXTENT);
L
Linus Torvalds 已提交
575 576 577 578
exit:
	return err;
}

579
static int __devexit sis5595_remove(struct platform_device *pdev)
L
Linus Torvalds 已提交
580
{
581
	struct sis5595_data *data = platform_get_drvdata(pdev);
L
Linus Torvalds 已提交
582

583
	hwmon_device_unregister(data->hwmon_dev);
584 585
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group);
	sysfs_remove_group(&pdev->dev.kobj, &sis5595_group_opt);
L
Linus Torvalds 已提交
586

587 588
	release_region(data->addr, SIS5595_EXTENT);
	platform_set_drvdata(pdev, NULL);
589
	kfree(data);
L
Linus Torvalds 已提交
590 591 592 593 594 595

	return 0;
}


/* ISA access must be locked explicitly. */
596
static int sis5595_read_value(struct sis5595_data *data, u8 reg)
L
Linus Torvalds 已提交
597 598 599
{
	int res;

600
	mutex_lock(&data->lock);
601 602
	outb_p(reg, data->addr + SIS5595_ADDR_REG_OFFSET);
	res = inb_p(data->addr + SIS5595_DATA_REG_OFFSET);
603
	mutex_unlock(&data->lock);
L
Linus Torvalds 已提交
604 605 606
	return res;
}

607
static void sis5595_write_value(struct sis5595_data *data, u8 reg, u8 value)
L
Linus Torvalds 已提交
608
{
609
	mutex_lock(&data->lock);
610 611
	outb_p(reg, data->addr + SIS5595_ADDR_REG_OFFSET);
	outb_p(value, data->addr + SIS5595_DATA_REG_OFFSET);
612
	mutex_unlock(&data->lock);
L
Linus Torvalds 已提交
613 614 615
}

/* Called when we have found a new SIS5595. */
616
static void __devinit sis5595_init_device(struct sis5595_data *data)
L
Linus Torvalds 已提交
617
{
618
	u8 config = sis5595_read_value(data, SIS5595_REG_CONFIG);
L
Linus Torvalds 已提交
619
	if (!(config & 0x01))
620
		sis5595_write_value(data, SIS5595_REG_CONFIG,
L
Linus Torvalds 已提交
621 622 623 624 625
				(config & 0xf7) | 0x01);
}

static struct sis5595_data *sis5595_update_device(struct device *dev)
{
626
	struct sis5595_data *data = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
627 628
	int i;

629
	mutex_lock(&data->update_lock);
L
Linus Torvalds 已提交
630 631 632 633 634 635

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

		for (i = 0; i <= data->maxins; i++) {
			data->in[i] =
636
			    sis5595_read_value(data, SIS5595_REG_IN(i));
L
Linus Torvalds 已提交
637
			data->in_min[i] =
638
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
639 640
					       SIS5595_REG_IN_MIN(i));
			data->in_max[i] =
641
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
642 643 644 645
					       SIS5595_REG_IN_MAX(i));
		}
		for (i = 0; i < 2; i++) {
			data->fan[i] =
646
			    sis5595_read_value(data, SIS5595_REG_FAN(i));
L
Linus Torvalds 已提交
647
			data->fan_min[i] =
648
			    sis5595_read_value(data,
L
Linus Torvalds 已提交
649 650 651 652
					       SIS5595_REG_FAN_MIN(i));
		}
		if (data->maxins == 3) {
			data->temp =
653
			    sis5595_read_value(data, SIS5595_REG_TEMP);
L
Linus Torvalds 已提交
654
			data->temp_over =
655
			    sis5595_read_value(data, SIS5595_REG_TEMP_OVER);
L
Linus Torvalds 已提交
656
			data->temp_hyst =
657
			    sis5595_read_value(data, SIS5595_REG_TEMP_HYST);
L
Linus Torvalds 已提交
658
		}
659
		i = sis5595_read_value(data, SIS5595_REG_FANDIV);
L
Linus Torvalds 已提交
660 661 662
		data->fan_div[0] = (i >> 4) & 0x03;
		data->fan_div[1] = i >> 6;
		data->alarms =
663 664
		    sis5595_read_value(data, SIS5595_REG_ALARM1) |
		    (sis5595_read_value(data, SIS5595_REG_ALARM2) << 8);
L
Linus Torvalds 已提交
665 666 667 668
		data->last_updated = jiffies;
		data->valid = 1;
	}

669
	mutex_unlock(&data->update_lock);
L
Linus Torvalds 已提交
670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694

	return data;
}

static struct pci_device_id sis5595_pci_ids[] = {
	{ 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 };

695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733
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;

	pdev = platform_device_alloc("sis5595", address);
	if (!pdev) {
		err = -ENOMEM;
		printk(KERN_ERR "sis5595: Device allocation failed\n");
		goto exit;
	}

	err = platform_device_add_resources(pdev, &res, 1);
	if (err) {
		printk(KERN_ERR "sis5595: Device resource addition failed "
		       "(%d)\n", err);
		goto exit_device_put;
	}

	err = platform_device_add(pdev);
	if (err) {
		printk(KERN_ERR "sis5595: Device addition failed (%d)\n",
		       err);
		goto exit_device_put;
	}

	return 0;

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

L
Linus Torvalds 已提交
734 735 736
static int __devinit sis5595_pci_probe(struct pci_dev *dev,
				       const struct pci_device_id *id)
{
737 738
	u16 address;
	u8 enable;
L
Linus Torvalds 已提交
739 740 741 742 743 744 745 746 747 748 749 750
	int *i;

	for (i = blacklist; *i != 0; i++) {
		struct pci_dev *dev;
		dev = pci_get_device(PCI_VENDOR_ID_SI, *i, NULL);
		if (dev) {
			dev_err(&dev->dev, "Looked for SIS5595 but found unsupported device %.4x\n", *i);
			pci_dev_put(dev);
			return -ENODEV;
		}
	}
	
751 752 753 754 755 756
	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 已提交
757
	if (PCIBIOS_SUCCESSFUL !=
758 759
	    pci_read_config_word(dev, SIS5595_BASE_REG, &address)) {
		dev_err(&dev->dev, "Failed to read ISA address\n");
L
Linus Torvalds 已提交
760
		return -ENODEV;
761
	}
L
Linus Torvalds 已提交
762
	
763 764
	address &= ~(SIS5595_EXTENT - 1);
	if (!address) {
L
Linus Torvalds 已提交
765 766 767
		dev_err(&dev->dev, "Base address not set - upgrade BIOS or use force_addr=0xaddr\n");
		return -ENODEV;
	}
768 769 770 771 772
	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 已提交
773

774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789
	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 已提交
790 791
	}

792 793 794 795 796 797 798 799 800 801
	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 已提交
802 803 804 805 806
	/* 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;
807 808 809 810 811 812

exit_unregister:
	pci_dev_put(dev);
	platform_driver_unregister(&sis5595_driver);
exit:
	return -ENODEV;
L
Linus Torvalds 已提交
813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829
}

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) {
830 831
		platform_device_unregister(pdev);
		platform_driver_unregister(&sis5595_driver);
L
Linus Torvalds 已提交
832 833 834 835 836 837 838 839 840 841 842
		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);