pcwd.c 26.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4
/*
 * PC Watchdog Driver
 * by Ken Hollis (khollis@bitgate.com)
 *
5
 * Permission granted from Simon Machell (smachell@berkprod.com)
L
Linus Torvalds 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 * Written for the Linux Kernel, and GPLed by Ken Hollis
 *
 * 960107	Added request_region routines, modulized the whole thing.
 * 960108	Fixed end-of-file pointer (Thanks to Dan Hollis), added
 *		WD_TIMEOUT define.
 * 960216	Added eof marker on the file, and changed verbose messages.
 * 960716	Made functional and cosmetic changes to the source for
 *		inclusion in Linux 2.0.x kernels, thanks to Alan Cox.
 * 960717	Removed read/seek routines, replaced with ioctl.  Also, added
 *		check_region command due to Alan's suggestion.
 * 960821	Made changes to compile in newer 2.0.x kernels.  Added
 *		"cold reboot sense" entry.
 * 960825	Made a few changes to code, deleted some defines and made
 *		typedefs to replace them.  Made heartbeat reset only available
 *		via ioctl, and removed the write routine.
 * 960828	Added new items for PC Watchdog Rev.C card.
 * 960829	Changed around all of the IOCTLs, added new features,
 *		added watchdog disable/re-enable routines.  Added firmware
 *		version reporting.  Added read routine for temperature.
 *		Removed some extra defines, added an autodetect Revision
 *		routine.
27 28
 * 961006	Revised some documentation, fixed some cosmetic bugs.  Made
 *		drivers to panic the system if it's overheating at bootup.
L
Linus Torvalds 已提交
29 30
 * 961118	Changed some verbiage on some of the output, tidied up
 *		code bits, and added compatibility to 2.1.x.
31
 * 970912	Enabled board on open and disable on close.
L
Linus Torvalds 已提交
32
 * 971107	Took account of recent VFS changes (broke read).
33 34 35 36 37 38
 * 971210	Disable board on initialisation in case board already ticking.
 * 971222	Changed open/close for temperature handling
 *		Michael Meskes <meskes@debian.org>.
 * 980112	Used minor numbers from include/linux/miscdevice.h
 * 990403	Clear reset status after reading control status register in
 *		pcwd_showprevstate(). [Marc Boucher <marc@mbsi.ca>]
L
Linus Torvalds 已提交
39 40 41 42
 * 990605	Made changes to code to support Firmware 1.22a, added
 *		fairly useless proc entry.
 * 990610	removed said useless proc code for the merge <alan>
 * 000403	Removed last traces of proc code. <davej>
43 44
 * 011214	Added nowayout module option to override
 *		CONFIG_WATCHDOG_NOWAYOUT <Matt_Domsch@dell.com>
45
 *		Added timeout module option to override default
L
Linus Torvalds 已提交
46 47 48 49
 */

/*
 *	A bells and whistles driver is available from http://www.pcwd.de/
50 51
 *	More info available at http://www.berkprod.com/ or
 *	http://www.pcwatchdog.com/
L
Linus Torvalds 已提交
52 53
 */

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

56 57 58 59 60 61 62 63 64 65
#include <linux/module.h>	/* For module specific items */
#include <linux/moduleparam.h>	/* For new moduleparam's */
#include <linux/types.h>	/* For standard types (like size_t) */
#include <linux/errno.h>	/* For the -ENODEV/... values */
#include <linux/kernel.h>	/* For printk/panic/... */
#include <linux/delay.h>	/* For mdelay function */
#include <linux/timer.h>	/* For timer related operations */
#include <linux/jiffies.h>	/* For jiffies stuff */
#include <linux/miscdevice.h>	/* For MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR) */
#include <linux/watchdog.h>	/* For the watchdog specific items */
66
#include <linux/reboot.h>	/* For kernel_power_off() */
67 68
#include <linux/init.h>		/* For __init/__exit/... */
#include <linux/fs.h>		/* For file operations */
69
#include <linux/isa.h>		/* For isa devices */
70 71
#include <linux/ioport.h>	/* For io-port access */
#include <linux/spinlock.h>	/* For spin_lock/spin_unlock/... */
72 73
#include <linux/uaccess.h>	/* For copy_to_user/put_user/... */
#include <linux/io.h>		/* For inb/outb/... */
74 75

/* Module and version information */
76 77
#define WATCHDOG_VERSION "1.20"
#define WATCHDOG_DATE "18 Feb 2007"
78 79
#define WATCHDOG_DRIVER_NAME "ISA-PC Watchdog"
#define WATCHDOG_NAME "pcwd"
80
#define DRIVER_VERSION WATCHDOG_DRIVER_NAME " driver, v" WATCHDOG_VERSION "\n"
L
Linus Torvalds 已提交
81 82 83 84 85 86 87 88 89 90 91

/*
 * It should be noted that PCWD_REVISION_B was removed because A and B
 * are essentially the same types of card, with the exception that B
 * has temperature reporting.  Since I didn't receive a Rev.B card,
 * the Rev.B card is not supported.  (It's a good thing too, as they
 * are no longer in production.)
 */
#define	PCWD_REVISION_A		1
#define	PCWD_REVISION_C		2

92 93 94 95 96 97 98 99 100
/*
 * These are the auto-probe addresses available.
 *
 * Revision A only uses ports 0x270 and 0x370.  Revision C introduced 0x350.
 * Revision A has an address range of 2 addresses, while Revision C has 4.
 */
#define PCWD_ISA_NR_CARDS	3
static int pcwd_ioports[] = { 0x270, 0x350, 0x370, 0x000 };

L
Linus Torvalds 已提交
101
/*
102 103 104 105
 * These are the defines that describe the control status bits for the
 * PCI-PC Watchdog card.
*/
/* Port 1 : Control Status #1 for the PC Watchdog card, revision A. */
106 107 108 109 110
#define WD_WDRST		0x01	/* Previously reset state */
#define WD_T110			0x02	/* Temperature overheat sense */
#define WD_HRTBT		0x04	/* Heartbeat sense */
#define WD_RLY2			0x08	/* External relay triggered */
#define WD_SRLY2		0x80	/* Software external relay triggered */
111
/* Port 1 : Control Status #1 for the PC Watchdog card, revision C. */
112 113 114
#define WD_REVC_WTRP		0x01	/* Watchdog Trip status */
#define WD_REVC_HRBT		0x02	/* Watchdog Heartbeat */
#define WD_REVC_TTRP		0x04	/* Temperature Trip status */
115 116
#define WD_REVC_RL2A		0x08	/* Relay 2 activated by
							on-board processor */
117 118 119
#define WD_REVC_RL1A		0x10	/* Relay 1 active */
#define WD_REVC_R2DS		0x40	/* Relay 2 disable */
#define WD_REVC_RLY2		0x80	/* Relay 2 activated? */
120 121 122
/* Port 2 : Control Status #2 */
#define WD_WDIS			0x10	/* Watchdog Disabled */
#define WD_ENTP			0x20	/* Watchdog Enable Temperature Trip */
123 124
#define WD_SSEL			0x40	/* Watchdog Switch Select
							(1:SW1 <-> 0:SW2) */
125
#define WD_WCMD			0x80	/* Watchdog Command Mode */
L
Linus Torvalds 已提交
126 127 128 129 130 131

/* max. time we give an ISA watchdog card to process a command */
/* 500ms for each 4 bit response (according to spec.) */
#define ISA_COMMAND_TIMEOUT     1000

/* Watchdog's internal commands */
132 133 134 135 136 137
#define CMD_ISA_IDLE			0x00
#define CMD_ISA_VERSION_INTEGER		0x01
#define CMD_ISA_VERSION_TENTH		0x02
#define CMD_ISA_VERSION_HUNDRETH	0x03
#define CMD_ISA_VERSION_MINOR		0x04
#define CMD_ISA_SWITCH_SETTINGS		0x05
138 139 140 141
#define CMD_ISA_RESET_PC		0x06
#define CMD_ISA_ARM_0			0x07
#define CMD_ISA_ARM_30			0x08
#define CMD_ISA_ARM_60			0x09
142 143 144
#define CMD_ISA_DELAY_TIME_2SECS	0x0A
#define CMD_ISA_DELAY_TIME_4SECS	0x0B
#define CMD_ISA_DELAY_TIME_8SECS	0x0C
145
#define CMD_ISA_RESET_RELAYS		0x0D
L
Linus Torvalds 已提交
146

147
/* Watchdog's Dip Switch heartbeat values */
148
static const int heartbeat_tbl[] = {
149 150 151 152 153 154 155 156 157 158
	20,	/* OFF-OFF-OFF	= 20 Sec  */
	40,	/* OFF-OFF-ON	= 40 Sec  */
	60,	/* OFF-ON-OFF	=  1 Min  */
	300,	/* OFF-ON-ON	=  5 Min  */
	600,	/* ON-OFF-OFF	= 10 Min  */
	1800,	/* ON-OFF-ON	= 30 Min  */
	3600,	/* ON-ON-OFF	=  1 Hour */
	7200,	/* ON-ON-ON	=  2 hour */
};

L
Linus Torvalds 已提交
159 160 161 162 163 164 165 166 167 168 169 170
/*
 * We are using an kernel timer to do the pinging of the watchdog
 * every ~500ms. We try to set the internal heartbeat of the
 * watchdog to 2 ms.
 */

#define WDT_INTERVAL (HZ/2+1)

/* We can only use 1 card due to the /dev/watchdog restriction */
static int cards_found;

/* internal variables */
171
static unsigned long open_allowed;
L
Linus Torvalds 已提交
172 173
static char expect_close;
static int temp_panic;
174 175 176

/* this is private data for each ISA-PC watchdog card */
static struct {
177
	char fw_ver_str[6];		/* The cards firmware version */
178
	int revision;			/* The card's revision */
179 180 181 182
	int supports_temp;		/* Whether or not the card has
						a temperature device */
	int command_mode;		/* Whether or not the card is in
						command mode */
183 184 185 186 187 188
	int boot_status;		/* The card's boot status */
	int io_addr;			/* The cards I/O address */
	spinlock_t io_lock;		/* the lock for io operations */
	struct timer_list timer;	/* The timer that pings the watchdog */
	unsigned long next_heartbeat;	/* the next_heartbeat for the timer */
} pcwd_private;
L
Linus Torvalds 已提交
189 190

/* module parameters */
191 192 193 194 195
#define QUIET	0	/* Default */
#define VERBOSE	1	/* Verbose */
#define DEBUG	2	/* print fancy stuff too */
static int debug = QUIET;
module_param(debug, int, 0);
196 197
MODULE_PARM_DESC(debug,
		"Debug level: 0=Quiet, 1=Verbose, 2=Debug (default=0)");
198

199 200
/* default heartbeat = delay-time from dip-switches */
#define WATCHDOG_HEARTBEAT 0
L
Linus Torvalds 已提交
201 202
static int heartbeat = WATCHDOG_HEARTBEAT;
module_param(heartbeat, int, 0);
203 204 205
MODULE_PARM_DESC(heartbeat, "Watchdog heartbeat in seconds. "
	"(2 <= heartbeat <= 7200 or 0=delay-time from dip-switches, default="
				__MODULE_STRING(WATCHDOG_HEARTBEAT) ")");
L
Linus Torvalds 已提交
206

W
Wim Van Sebroeck 已提交
207 208
static bool nowayout = WATCHDOG_NOWAYOUT;
module_param(nowayout, bool, 0);
209 210 211
MODULE_PARM_DESC(nowayout,
		"Watchdog cannot be stopped once started (default="
				__MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
L
Linus Torvalds 已提交
212 213 214 215 216 217 218 219 220 221 222

/*
 *	Internal functions
 */

static int send_isa_command(int cmd)
{
	int i;
	int control_status;
	int port0, last_port0;	/* Double read for stabilising */

223
	if (debug >= DEBUG)
224
		pr_debug("sending following data cmd=0x%02x\n", cmd);
225

L
Linus Torvalds 已提交
226
	/* The WCMD bit must be 1 and the command is only 4 bits in size */
227
	control_status = (cmd & 0x0F) | WD_WCMD;
228
	outb_p(control_status, pcwd_private.io_addr + 2);
L
Linus Torvalds 已提交
229 230
	udelay(ISA_COMMAND_TIMEOUT);

231
	port0 = inb_p(pcwd_private.io_addr);
L
Linus Torvalds 已提交
232 233
	for (i = 0; i < 25; ++i) {
		last_port0 = port0;
234
		port0 = inb_p(pcwd_private.io_addr);
L
Linus Torvalds 已提交
235 236 237 238

		if (port0 == last_port0)
			break;	/* Data is stable */

239
		udelay(250);
L
Linus Torvalds 已提交
240 241
	}

242
	if (debug >= DEBUG)
243 244
		pr_debug("received following data for cmd=0x%02x: port0=0x%02x last_port0=0x%02x\n",
			 cmd, port0, last_port0);
245

L
Linus Torvalds 已提交
246 247 248 249 250
	return port0;
}

static int set_command_mode(void)
{
251
	int i, found = 0, count = 0;
L
Linus Torvalds 已提交
252 253

	/* Set the card into command mode */
254
	spin_lock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
255 256 257 258 259 260 261
	while ((!found) && (count < 3)) {
		i = send_isa_command(CMD_ISA_IDLE);

		if (i == 0x00)
			found = 1;
		else if (i == 0xF3) {
			/* Card does not like what we've done to it */
262
			outb_p(0x00, pcwd_private.io_addr + 2);
L
Linus Torvalds 已提交
263
			udelay(1200);	/* Spec says wait 1ms */
264
			outb_p(0x00, pcwd_private.io_addr + 2);
L
Linus Torvalds 已提交
265 266 267 268
			udelay(ISA_COMMAND_TIMEOUT);
		}
		count++;
	}
269 270
	spin_unlock(&pcwd_private.io_lock);
	pcwd_private.command_mode = found;
L
Linus Torvalds 已提交
271

272
	if (debug >= DEBUG)
273
		pr_debug("command_mode=%d\n", pcwd_private.command_mode);
274

275
	return found;
L
Linus Torvalds 已提交
276 277 278 279 280
}

static void unset_command_mode(void)
{
	/* Set the card into normal mode */
281 282
	spin_lock(&pcwd_private.io_lock);
	outb_p(0x00, pcwd_private.io_addr + 2);
L
Linus Torvalds 已提交
283
	udelay(ISA_COMMAND_TIMEOUT);
284
	spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
285

286
	pcwd_private.command_mode = 0;
287 288

	if (debug >= DEBUG)
289
		pr_debug("command_mode=%d\n", pcwd_private.command_mode);
L
Linus Torvalds 已提交
290 291
}

292 293 294 295 296 297
static inline void pcwd_check_temperature_support(void)
{
	if (inb(pcwd_private.io_addr) != 0xF0)
		pcwd_private.supports_temp = 1;
}

298
static inline void pcwd_get_firmware(void)
299 300 301
{
	int one, ten, hund, minor;

302
	strcpy(pcwd_private.fw_ver_str, "ERROR");
303 304 305 306 307 308

	if (set_command_mode()) {
		one = send_isa_command(CMD_ISA_VERSION_INTEGER);
		ten = send_isa_command(CMD_ISA_VERSION_TENTH);
		hund = send_isa_command(CMD_ISA_VERSION_HUNDRETH);
		minor = send_isa_command(CMD_ISA_VERSION_MINOR);
309 310
		sprintf(pcwd_private.fw_ver_str, "%c.%c%c%c",
					one, ten, hund, minor);
311 312
	}
	unset_command_mode();
313 314

	return;
315 316 317 318
}

static inline int pcwd_get_option_switches(void)
{
319
	int option_switches = 0;
320 321 322 323 324 325 326

	if (set_command_mode()) {
		/* Get switch settings */
		option_switches = send_isa_command(CMD_ISA_SWITCH_SETTINGS);
	}

	unset_command_mode();
327
	return option_switches;
328 329 330 331 332 333 334 335
}

static void pcwd_show_card_info(void)
{
	int option_switches;

	/* Get some extra info from the hardware (in command/debug/diag mode) */
	if (pcwd_private.revision == PCWD_REVISION_A)
336 337
		pr_info("ISA-PC Watchdog (REV.A) detected at port 0x%04x\n",
			pcwd_private.io_addr);
338
	else if (pcwd_private.revision == PCWD_REVISION_C) {
339
		pcwd_get_firmware();
340
		pr_info("ISA-PC Watchdog (REV.C) detected at port 0x%04x (Firmware version: %s)\n",
341
			pcwd_private.io_addr, pcwd_private.fw_ver_str);
342
		option_switches = pcwd_get_option_switches();
343
		pr_info("Option switches (0x%02x): Temperature Reset Enable=%s, Power On Delay=%s\n",
344 345 346 347 348 349 350 351 352 353 354 355
			option_switches,
			((option_switches & 0x10) ? "ON" : "OFF"),
			((option_switches & 0x08) ? "ON" : "OFF"));

		/* Reprogram internal heartbeat to 2 seconds */
		if (set_command_mode()) {
			send_isa_command(CMD_ISA_DELAY_TIME_2SECS);
			unset_command_mode();
		}
	}

	if (pcwd_private.supports_temp)
356
		pr_info("Temperature Option Detected\n");
357 358

	if (pcwd_private.boot_status & WDIOF_CARDRESET)
359
		pr_info("Previous reboot was caused by the card\n");
360 361

	if (pcwd_private.boot_status & WDIOF_OVERHEAT) {
362 363
		pr_emerg("Card senses a CPU Overheat. Panicking!\n");
		pr_emerg("CPU Overheat\n");
364 365 366
	}

	if (pcwd_private.boot_status == 0)
367
		pr_info("No previous trip detected - Cold boot or reset\n");
368 369
}

L
Linus Torvalds 已提交
370 371 372 373 374 375
static void pcwd_timer_ping(unsigned long data)
{
	int wdrst_stat;

	/* If we got a heartbeat pulse within the WDT_INTERVAL
	 * we agree to ping the WDT */
376
	if (time_before(jiffies, pcwd_private.next_heartbeat)) {
L
Linus Torvalds 已提交
377
		/* Ping the watchdog */
378 379
		spin_lock(&pcwd_private.io_lock);
		if (pcwd_private.revision == PCWD_REVISION_A) {
380 381
			/*  Rev A cards are reset by setting the
			    WD_WDRST bit in register 1 */
382
			wdrst_stat = inb_p(pcwd_private.io_addr);
L
Linus Torvalds 已提交
383 384 385
			wdrst_stat &= 0x0F;
			wdrst_stat |= WD_WDRST;

386
			outb_p(wdrst_stat, pcwd_private.io_addr + 1);
L
Linus Torvalds 已提交
387 388
		} else {
			/* Re-trigger watchdog by writing to port 0 */
389
			outb_p(0x00, pcwd_private.io_addr);
L
Linus Torvalds 已提交
390 391 392
		}

		/* Re-set the timer interval */
393
		mod_timer(&pcwd_private.timer, jiffies + WDT_INTERVAL);
L
Linus Torvalds 已提交
394

395
		spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
396
	} else {
397
		pr_warn("Heartbeat lost! Will not ping the watchdog\n");
L
Linus Torvalds 已提交
398 399 400 401 402 403 404
	}
}

static int pcwd_start(void)
{
	int stat_reg;

405
	pcwd_private.next_heartbeat = jiffies + (heartbeat * HZ);
L
Linus Torvalds 已提交
406 407

	/* Start the timer */
408
	mod_timer(&pcwd_private.timer, jiffies + WDT_INTERVAL);
L
Linus Torvalds 已提交
409 410

	/* Enable the port */
411 412 413
	if (pcwd_private.revision == PCWD_REVISION_C) {
		spin_lock(&pcwd_private.io_lock);
		outb_p(0x00, pcwd_private.io_addr + 3);
L
Linus Torvalds 已提交
414
		udelay(ISA_COMMAND_TIMEOUT);
415 416
		stat_reg = inb_p(pcwd_private.io_addr + 2);
		spin_unlock(&pcwd_private.io_lock);
417
		if (stat_reg & WD_WDIS) {
418
			pr_info("Could not start watchdog\n");
L
Linus Torvalds 已提交
419 420 421
			return -EIO;
		}
	}
422 423

	if (debug >= VERBOSE)
424
		pr_debug("Watchdog started\n");
425

L
Linus Torvalds 已提交
426 427 428 429 430 431 432 433
	return 0;
}

static int pcwd_stop(void)
{
	int stat_reg;

	/* Stop the timer */
434
	del_timer(&pcwd_private.timer);
L
Linus Torvalds 已提交
435 436

	/*  Disable the board  */
437 438 439
	if (pcwd_private.revision == PCWD_REVISION_C) {
		spin_lock(&pcwd_private.io_lock);
		outb_p(0xA5, pcwd_private.io_addr + 3);
L
Linus Torvalds 已提交
440
		udelay(ISA_COMMAND_TIMEOUT);
441
		outb_p(0xA5, pcwd_private.io_addr + 3);
L
Linus Torvalds 已提交
442
		udelay(ISA_COMMAND_TIMEOUT);
443 444
		stat_reg = inb_p(pcwd_private.io_addr + 2);
		spin_unlock(&pcwd_private.io_lock);
445
		if ((stat_reg & WD_WDIS) == 0) {
446
			pr_info("Could not stop watchdog\n");
L
Linus Torvalds 已提交
447 448 449
			return -EIO;
		}
	}
450 451

	if (debug >= VERBOSE)
452
		pr_debug("Watchdog stopped\n");
453

L
Linus Torvalds 已提交
454 455 456 457 458 459
	return 0;
}

static int pcwd_keepalive(void)
{
	/* user land ping */
460
	pcwd_private.next_heartbeat = jiffies + (heartbeat * HZ);
461 462

	if (debug >= DEBUG)
463
		pr_debug("Watchdog keepalive signal send\n");
464

L
Linus Torvalds 已提交
465 466 467 468 469
	return 0;
}

static int pcwd_set_heartbeat(int t)
{
470
	if (t < 2 || t > 7200) /* arbitrary upper limit */
L
Linus Torvalds 已提交
471 472 473
		return -EINVAL;

	heartbeat = t;
474 475

	if (debug >= VERBOSE)
476
		pr_debug("New heartbeat: %d\n", heartbeat);
477

L
Linus Torvalds 已提交
478 479 480 481 482
	return 0;
}

static int pcwd_get_status(int *status)
{
483
	int control_status;
L
Linus Torvalds 已提交
484

485
	*status = 0;
486 487
	spin_lock(&pcwd_private.io_lock);
	if (pcwd_private.revision == PCWD_REVISION_A)
L
Linus Torvalds 已提交
488 489 490
		/* Rev A cards return status information from
		 * the base register, which is used for the
		 * temperature in other cards. */
491
		control_status = inb(pcwd_private.io_addr);
L
Linus Torvalds 已提交
492 493 494 495 496 497
	else {
		/* Rev C cards return card status in the base
		 * address + 1 register. And use different bits
		 * to indicate a card initiated reset, and an
		 * over-temperature condition. And the reboot
		 * status can be reset. */
498
		control_status = inb(pcwd_private.io_addr + 1);
L
Linus Torvalds 已提交
499
	}
500
	spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
501

502
	if (pcwd_private.revision == PCWD_REVISION_A) {
503
		if (control_status & WD_WDRST)
L
Linus Torvalds 已提交
504 505
			*status |= WDIOF_CARDRESET;

506
		if (control_status & WD_T110) {
L
Linus Torvalds 已提交
507 508
			*status |= WDIOF_OVERHEAT;
			if (temp_panic) {
509
				pr_info("Temperature overheat trip!\n");
510
				kernel_power_off();
L
Linus Torvalds 已提交
511 512 513
			}
		}
	} else {
514
		if (control_status & WD_REVC_WTRP)
L
Linus Torvalds 已提交
515 516
			*status |= WDIOF_CARDRESET;

517
		if (control_status & WD_REVC_TTRP) {
L
Linus Torvalds 已提交
518 519
			*status |= WDIOF_OVERHEAT;
			if (temp_panic) {
520
				pr_info("Temperature overheat trip!\n");
521
				kernel_power_off();
L
Linus Torvalds 已提交
522 523 524 525 526 527 528 529 530
			}
		}
	}

	return 0;
}

static int pcwd_clear_status(void)
{
531 532
	int control_status;

533 534
	if (pcwd_private.revision == PCWD_REVISION_C) {
		spin_lock(&pcwd_private.io_lock);
535

536
		if (debug >= VERBOSE)
537
			pr_info("clearing watchdog trip status\n");
538

539 540
		control_status = inb_p(pcwd_private.io_addr + 1);

541
		if (debug >= DEBUG) {
542 543 544
			pr_debug("status was: 0x%02x\n", control_status);
			pr_debug("sending: 0x%02x\n",
				 (control_status & WD_REVC_R2DS));
545 546
		}

547
		/* clear reset status & Keep Relay 2 disable state as it is */
548 549
		outb_p((control_status & WD_REVC_R2DS),
						pcwd_private.io_addr + 1);
550

551
		spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
552 553 554 555 556 557 558
	}
	return 0;
}

static int pcwd_get_temperature(int *temperature)
{
	/* check that port 0 gives temperature info and no command results */
559
	if (pcwd_private.command_mode)
L
Linus Torvalds 已提交
560 561 562
		return -1;

	*temperature = 0;
563
	if (!pcwd_private.supports_temp)
L
Linus Torvalds 已提交
564 565 566 567 568 569
		return -ENODEV;

	/*
	 * Convert celsius to fahrenheit, since this was
	 * the decided 'standard' for this return value.
	 */
570 571 572
	spin_lock(&pcwd_private.io_lock);
	*temperature = ((inb(pcwd_private.io_addr)) * 9 / 5) + 32;
	spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
573

574
	if (debug >= DEBUG) {
575
		pr_debug("temperature is: %d F\n", *temperature);
576 577
	}

L
Linus Torvalds 已提交
578 579 580 581 582 583 584
	return 0;
}

/*
 *	/dev/watchdog handling
 */

585
static long pcwd_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
586 587 588 589 590 591
{
	int rv;
	int status;
	int temperature;
	int new_heartbeat;
	int __user *argp = (int __user *)arg;
592
	static const struct watchdog_info ident = {
L
Linus Torvalds 已提交
593 594 595 596 597 598 599 600 601
		.options =		WDIOF_OVERHEAT |
					WDIOF_CARDRESET |
					WDIOF_KEEPALIVEPING |
					WDIOF_SETTIMEOUT |
					WDIOF_MAGICCLOSE,
		.firmware_version =	1,
		.identity =		"PCWD",
	};

602
	switch (cmd) {
L
Linus Torvalds 已提交
603
	case WDIOC_GETSUPPORT:
604
		if (copy_to_user(argp, &ident, sizeof(ident)))
L
Linus Torvalds 已提交
605 606 607 608 609 610 611 612
			return -EFAULT;
		return 0;

	case WDIOC_GETSTATUS:
		pcwd_get_status(&status);
		return put_user(status, argp);

	case WDIOC_GETBOOTSTATUS:
613
		return put_user(pcwd_private.boot_status, argp);
L
Linus Torvalds 已提交
614 615 616 617 618 619 620 621

	case WDIOC_GETTEMP:
		if (pcwd_get_temperature(&temperature))
			return -EFAULT;

		return put_user(temperature, argp);

	case WDIOC_SETOPTIONS:
622 623
		if (pcwd_private.revision == PCWD_REVISION_C) {
			if (get_user(rv, argp))
L
Linus Torvalds 已提交
624 625
				return -EFAULT;

626 627 628 629
			if (rv & WDIOS_DISABLECARD) {
				status = pcwd_stop();
				if (status < 0)
					return status;
L
Linus Torvalds 已提交
630
			}
631 632 633 634
			if (rv & WDIOS_ENABLECARD) {
				status = pcwd_start();
				if (status < 0)
					return status;
L
Linus Torvalds 已提交
635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656
			}
			if (rv & WDIOS_TEMPPANIC)
				temp_panic = 1;
		}
		return -EINVAL;

	case WDIOC_KEEPALIVE:
		pcwd_keepalive();
		return 0;

	case WDIOC_SETTIMEOUT:
		if (get_user(new_heartbeat, argp))
			return -EFAULT;

		if (pcwd_set_heartbeat(new_heartbeat))
			return -EINVAL;

		pcwd_keepalive();
		/* Fall */

	case WDIOC_GETTIMEOUT:
		return put_user(heartbeat, argp);
657 658 659

	default:
		return -ENOTTY;
L
Linus Torvalds 已提交
660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690
	}

	return 0;
}

static ssize_t pcwd_write(struct file *file, const char __user *buf, size_t len,
			  loff_t *ppos)
{
	if (len) {
		if (!nowayout) {
			size_t i;

			/* In case it was set long ago */
			expect_close = 0;

			for (i = 0; i != len; i++) {
				char c;

				if (get_user(c, buf + i))
					return -EFAULT;
				if (c == 'V')
					expect_close = 42;
			}
		}
		pcwd_keepalive();
	}
	return len;
}

static int pcwd_open(struct inode *inode, struct file *file)
{
691
	if (test_and_set_bit(0, &open_allowed))
L
Linus Torvalds 已提交
692 693 694 695 696 697 698 699 700 701 702
		return -EBUSY;
	if (nowayout)
		__module_get(THIS_MODULE);
	/* Activate */
	pcwd_start();
	pcwd_keepalive();
	return nonseekable_open(inode, file);
}

static int pcwd_close(struct inode *inode, struct file *file)
{
703
	if (expect_close == 42)
L
Linus Torvalds 已提交
704
		pcwd_stop();
705
	else {
706
		pr_crit("Unexpected close, not stopping watchdog!\n");
L
Linus Torvalds 已提交
707 708 709
		pcwd_keepalive();
	}
	expect_close = 0;
710
	clear_bit(0, &open_allowed);
L
Linus Torvalds 已提交
711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733
	return 0;
}

/*
 *	/dev/temperature handling
 */

static ssize_t pcwd_temp_read(struct file *file, char __user *buf, size_t count,
			 loff_t *ppos)
{
	int temperature;

	if (pcwd_get_temperature(&temperature))
		return -EFAULT;

	if (copy_to_user(buf, &temperature, 1))
		return -EFAULT;

	return 1;
}

static int pcwd_temp_open(struct inode *inode, struct file *file)
{
734
	if (!pcwd_private.supports_temp)
L
Linus Torvalds 已提交
735 736 737 738 739 740 741 742 743 744 745 746 747 748
		return -ENODEV;

	return nonseekable_open(inode, file);
}

static int pcwd_temp_close(struct inode *inode, struct file *file)
{
	return 0;
}

/*
 *	Kernel Interfaces
 */

749
static const struct file_operations pcwd_fops = {
L
Linus Torvalds 已提交
750 751 752
	.owner		= THIS_MODULE,
	.llseek		= no_llseek,
	.write		= pcwd_write,
753
	.unlocked_ioctl	= pcwd_ioctl,
L
Linus Torvalds 已提交
754 755 756 757 758 759 760 761 762 763
	.open		= pcwd_open,
	.release	= pcwd_close,
};

static struct miscdevice pcwd_miscdev = {
	.minor =	WATCHDOG_MINOR,
	.name =		"watchdog",
	.fops =		&pcwd_fops,
};

764
static const struct file_operations pcwd_temp_fops = {
L
Linus Torvalds 已提交
765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785
	.owner		= THIS_MODULE,
	.llseek		= no_llseek,
	.read		= pcwd_temp_read,
	.open		= pcwd_temp_open,
	.release	= pcwd_temp_close,
};

static struct miscdevice temp_miscdev = {
	.minor =	TEMP_MINOR,
	.name =		"temperature",
	.fops =		&pcwd_temp_fops,
};

/*
 *	Init & exit routines
 */

static inline int get_revision(void)
{
	int r = PCWD_REVISION_C;

786
	spin_lock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
787 788
	/* REV A cards use only 2 io ports; test
	 * presumes a floating bus reads as 0xff. */
789 790
	if ((inb(pcwd_private.io_addr + 2) == 0xFF) ||
	    (inb(pcwd_private.io_addr + 3) == 0xFF))
791
		r = PCWD_REVISION_A;
792
	spin_unlock(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
793 794 795 796

	return r;
}

797 798 799 800 801 802 803 804 805
/*
 *  The ISA cards have a heartbeat bit in one of the registers, which
 *  register is card dependent.  The heartbeat bit is monitored, and if
 *  found, is considered proof that a Berkshire card has been found.
 *  The initial rate is once per second at board start up, then twice
 *  per second for normal operation.
 */
static int __devinit pcwd_isa_match(struct device *dev, unsigned int id)
{
806
	int base_addr = pcwd_ioports[id];
807 808 809 810 811 812
	int port0, last_port0;	/* Reg 0, in case it's REV A */
	int port1, last_port1;	/* Register 1 for REV C cards */
	int i;
	int retval;

	if (debug >= DEBUG)
813
		pr_debug("pcwd_isa_match id=%d\n", id);
814

815
	if (!request_region(base_addr, 4, "PCWD")) {
816
		pr_info("Port 0x%04x unavailable\n", base_addr);
817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843
		return 0;
	}

	retval = 0;

	port0 = inb_p(base_addr);	/* For REV A boards */
	port1 = inb_p(base_addr + 1);	/* For REV C boards */
	if (port0 != 0xff || port1 != 0xff) {
		/* Not an 'ff' from a floating bus, so must be a card! */
		for (i = 0; i < 4; ++i) {

			msleep(500);

			last_port0 = port0;
			last_port1 = port1;

			port0 = inb_p(base_addr);
			port1 = inb_p(base_addr + 1);

			/* Has either hearbeat bit changed?  */
			if ((port0 ^ last_port0) & WD_HRTBT ||
			    (port1 ^ last_port1) & WD_REVC_HRBT) {
				retval = 1;
				break;
			}
		}
	}
844
	release_region(base_addr, 4);
845 846 847 848 849

	return retval;
}

static int __devinit pcwd_isa_probe(struct device *dev, unsigned int id)
L
Linus Torvalds 已提交
850 851 852
{
	int ret;

853
	if (debug >= DEBUG)
854
		pr_debug("pcwd_isa_probe id=%d\n", id);
855

L
Linus Torvalds 已提交
856 857
	cards_found++;
	if (cards_found == 1)
858
		pr_info("v%s Ken Hollis (kenji@bitgate.com)\n",
859
							WATCHDOG_VERSION);
L
Linus Torvalds 已提交
860 861

	if (cards_found > 1) {
862
		pr_err("This driver only supports 1 device\n");
L
Linus Torvalds 已提交
863 864 865
		return -ENODEV;
	}

866
	if (pcwd_ioports[id] == 0x0000) {
867
		pr_err("No I/O-Address for card detected\n");
L
Linus Torvalds 已提交
868 869
		return -ENODEV;
	}
870 871 872
	pcwd_private.io_addr = pcwd_ioports[id];

	spin_lock_init(&pcwd_private.io_lock);
L
Linus Torvalds 已提交
873 874

	/* Check card's revision */
875
	pcwd_private.revision = get_revision();
L
Linus Torvalds 已提交
876

877 878
	if (!request_region(pcwd_private.io_addr,
		(pcwd_private.revision == PCWD_REVISION_A) ? 2 : 4, "PCWD")) {
879 880
		pr_err("I/O address 0x%04x already in use\n",
		       pcwd_private.io_addr);
881
		ret = -EIO;
882
		goto error_request_region;
L
Linus Torvalds 已提交
883 884 885
	}

	/* Initial variables */
886
	pcwd_private.supports_temp = 0;
L
Linus Torvalds 已提交
887
	temp_panic = 0;
888
	pcwd_private.boot_status = 0x0000;
L
Linus Torvalds 已提交
889 890

	/* get the boot_status */
891
	pcwd_get_status(&pcwd_private.boot_status);
L
Linus Torvalds 已提交
892 893 894 895

	/* clear the "card caused reboot" flag */
	pcwd_clear_status();

J
Jiri Slaby 已提交
896
	setup_timer(&pcwd_private.timer, pcwd_timer_ping, 0);
L
Linus Torvalds 已提交
897 898 899 900 901

	/*  Disable the board  */
	pcwd_stop();

	/*  Check whether or not the card supports the temperature device */
902
	pcwd_check_temperature_support();
L
Linus Torvalds 已提交
903

904 905
	/* Show info about the card itself */
	pcwd_show_card_info();
L
Linus Torvalds 已提交
906

907 908 909 910
	/* If heartbeat = 0 then we use the heartbeat from the dip-switches */
	if (heartbeat == 0)
		heartbeat = heartbeat_tbl[(pcwd_get_option_switches() & 0x07)];

911 912
	/* Check that the heartbeat value is within it's range;
	   if not reset to the default */
913 914
	if (pcwd_set_heartbeat(heartbeat)) {
		pcwd_set_heartbeat(WATCHDOG_HEARTBEAT);
915 916
		pr_info("heartbeat value must be 2 <= heartbeat <= 7200, using %d\n",
			WATCHDOG_HEARTBEAT);
L
Linus Torvalds 已提交
917 918
	}

919
	if (pcwd_private.supports_temp) {
L
Linus Torvalds 已提交
920 921
		ret = misc_register(&temp_miscdev);
		if (ret) {
922 923
			pr_err("cannot register miscdev on minor=%d (err=%d)\n",
			       TEMP_MINOR, ret);
924
			goto error_misc_register_temp;
L
Linus Torvalds 已提交
925 926 927 928 929
		}
	}

	ret = misc_register(&pcwd_miscdev);
	if (ret) {
930 931
		pr_err("cannot register miscdev on minor=%d (err=%d)\n",
		       WATCHDOG_MINOR, ret);
932
		goto error_misc_register_watchdog;
L
Linus Torvalds 已提交
933 934
	}

935
	pr_info("initialized. heartbeat=%d sec (nowayout=%d)\n",
L
Linus Torvalds 已提交
936 937 938
		heartbeat, nowayout);

	return 0;
939 940 941 942 943

error_misc_register_watchdog:
	if (pcwd_private.supports_temp)
		misc_deregister(&temp_miscdev);
error_misc_register_temp:
944 945
	release_region(pcwd_private.io_addr,
			(pcwd_private.revision == PCWD_REVISION_A) ? 2 : 4);
946 947 948 949
error_request_region:
	pcwd_private.io_addr = 0x0000;
	cards_found--;
	return ret;
L
Linus Torvalds 已提交
950 951
}

952
static int __devexit pcwd_isa_remove(struct device *dev, unsigned int id)
L
Linus Torvalds 已提交
953
{
954
	if (debug >= DEBUG)
955
		pr_debug("pcwd_isa_remove id=%d\n", id);
956 957 958 959

	if (!pcwd_private.io_addr)
		return 1;

L
Linus Torvalds 已提交
960 961 962 963 964 965
	/*  Disable the board  */
	if (!nowayout)
		pcwd_stop();

	/* Deregister */
	misc_deregister(&pcwd_miscdev);
966
	if (pcwd_private.supports_temp)
L
Linus Torvalds 已提交
967
		misc_deregister(&temp_miscdev);
968 969
	release_region(pcwd_private.io_addr,
			(pcwd_private.revision == PCWD_REVISION_A) ? 2 : 4);
970
	pcwd_private.io_addr = 0x0000;
971
	cards_found--;
972 973

	return 0;
L
Linus Torvalds 已提交
974 975
}

976
static void pcwd_isa_shutdown(struct device *dev, unsigned int id)
L
Linus Torvalds 已提交
977
{
978
	if (debug >= DEBUG)
979
		pr_debug("pcwd_isa_shutdown id=%d\n", id);
L
Linus Torvalds 已提交
980

981
	pcwd_stop();
L
Linus Torvalds 已提交
982 983
}

984 985 986
static struct isa_driver pcwd_isa_driver = {
	.match		= pcwd_isa_match,
	.probe		= pcwd_isa_probe,
987
	.remove		= pcwd_isa_remove,
988 989 990 991 992 993
	.shutdown	= pcwd_isa_shutdown,
	.driver		= {
		.owner	= THIS_MODULE,
		.name	= WATCHDOG_NAME,
	},
};
L
Linus Torvalds 已提交
994 995 996

static int __init pcwd_init_module(void)
{
997
	return isa_register_driver(&pcwd_isa_driver, PCWD_ISA_NR_CARDS);
L
Linus Torvalds 已提交
998 999 1000 1001
}

static void __exit pcwd_cleanup_module(void)
{
1002
	isa_unregister_driver(&pcwd_isa_driver);
1003
	pr_info("Watchdog Module Unloaded\n");
L
Linus Torvalds 已提交
1004 1005 1006 1007 1008
}

module_init(pcwd_init_module);
module_exit(pcwd_cleanup_module);

1009 1010
MODULE_AUTHOR("Ken Hollis <kenji@bitgate.com>, "
		"Wim Van Sebroeck <wim@iguana.be>");
L
Linus Torvalds 已提交
1011
MODULE_DESCRIPTION("Berkshire ISA-PC Watchdog driver");
1012
MODULE_VERSION(WATCHDOG_VERSION);
L
Linus Torvalds 已提交
1013 1014 1015
MODULE_LICENSE("GPL");
MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR);
MODULE_ALIAS_MISCDEV(TEMP_MINOR);