Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
a8cd2e50
Commit
a8cd2e50
authored
Sep 13, 2005
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge master.kernel.org:/pub/scm/linux/kernel/git/wim/linux-2.6-watchdog
parents
0160f53e
c315b7e8
Changes
10
Hide whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
2252 additions
and
43 deletions
+2252
-43
drivers/char/watchdog/Kconfig
drivers/char/watchdog/Kconfig
+72
-21
drivers/char/watchdog/Makefile
drivers/char/watchdog/Makefile
+6
-1
drivers/char/watchdog/i6300esb.c
drivers/char/watchdog/i6300esb.c
+527
-0
drivers/char/watchdog/ibmasr.c
drivers/char/watchdog/ibmasr.c
+405
-0
drivers/char/watchdog/mv64x60_wdt.c
drivers/char/watchdog/mv64x60_wdt.c
+252
-0
drivers/char/watchdog/pcwd_pci.c
drivers/char/watchdog/pcwd_pci.c
+24
-20
drivers/char/watchdog/s3c2410_wdt.c
drivers/char/watchdog/s3c2410_wdt.c
+1
-1
drivers/char/watchdog/sbc8360.c
drivers/char/watchdog/sbc8360.c
+414
-0
drivers/char/watchdog/w83977f_wdt.c
drivers/char/watchdog/w83977f_wdt.c
+543
-0
include/asm-ppc/mv64x60.h
include/asm-ppc/mv64x60.h
+8
-0
No files found.
drivers/char/watchdog/Kconfig
View file @
a8cd2e50
...
...
@@ -84,6 +84,17 @@ config 977_WATCHDOG
Not sure? It's safe to say N.
config IXP2000_WATCHDOG
tristate "IXP2000 Watchdog"
depends on WATCHDOG && ARCH_IXP2000
help
Say Y here if to include support for the watchdog timer
in the Intel IXP2000(2400, 2800, 2850) network processors.
This driver can be built as a module by choosing M. The module
will be called ixp2000_wdt.
Say N if you are unsure.
config IXP4XX_WATCHDOG
tristate "IXP4xx Watchdog"
depends on WATCHDOG && ARCH_IXP4XX
...
...
@@ -100,17 +111,6 @@ config IXP4XX_WATCHDOG
Say N if you are unsure.
config IXP2000_WATCHDOG
tristate "IXP2000 Watchdog"
depends on WATCHDOG && ARCH_IXP2000
help
Say Y here if to include support for the watchdog timer
in the Intel IXP2000(2400, 2800, 2850) network processors.
This driver can be built as a module by choosing M. The module
will be called ixp2000_wdt.
Say N if you are unsure.
config S3C2410_WATCHDOG
tristate "S3C2410 Watchdog"
depends on WATCHDOG && ARCH_S3C2410
...
...
@@ -233,6 +233,16 @@ config IB700_WDT
Most people will say N.
config IBMASR
tristate "IBM Automatic Server Restart"
depends on WATCHDOG && X86
help
This is the driver for the IBM Automatic Server Restart watchdog
timer builtin into some eServer xSeries machines.
To compile this driver as a module, choose M here: the
module will be called ibmasr.
config WAFER_WDT
tristate "ICP Wafer 5823 Single Board Computer Watchdog"
depends on WATCHDOG && X86
...
...
@@ -243,6 +253,16 @@ config WAFER_WDT
To compile this driver as a module, choose M here: the
module will be called wafer5823wdt.
config I6300ESB_WDT
tristate "Intel 6300ESB Timer/Watchdog"
depends on WATCHDOG && X86 && PCI
---help---
Hardware driver for the watchdog timer built into the Intel
6300ESB controller hub.
To compile this driver as a module, choose M here: the
module will be called i6300esb.
config I8XX_TCO
tristate "Intel i8xx TCO Timer/Watchdog"
depends on WATCHDOG && (X86 || IA64) && PCI
...
...
@@ -298,6 +318,19 @@ config 60XX_WDT
You can compile this driver directly into the kernel, or use
it as a module. The module will be called sbc60xxwdt.
config SBC8360_WDT
tristate "SBC8360 Watchdog Timer"
depends on WATCHDOG && X86
---help---
This is the driver for the hardware watchdog on the SBC8360 Single
Board Computer produced by Axiomtek Co., Ltd. (www.axiomtek.com).
To compile this driver as a module, choose M here: the
module will be called sbc8360.ko.
Most people will say N.
config CPU5_WDT
tristate "SMA CPU5 Watchdog"
depends on WATCHDOG && X86
...
...
@@ -336,6 +369,19 @@ config W83877F_WDT
Most people will say N.
config W83977F_WDT
tristate "W83977F (PCM-5335) Watchdog Timer"
depends on WATCHDOG && X86
---help---
This is the driver for the hardware watchdog on the W83977F I/O chip
as used in AAEON's PCM-5335 SBC (and likely others). This
watchdog simply watches your kernel to make sure it doesn't freeze,
and if it does, it reboots your computer after a certain amount of
time.
To compile this driver as a module, choose M here: the
module will be called w83977f_wdt.
config MACHZ_WDT
tristate "ZF MachZ Watchdog"
depends on WATCHDOG && X86
...
...
@@ -355,6 +401,10 @@ config 8xx_WDT
tristate "MPC8xx Watchdog Timer"
depends on WATCHDOG && 8xx
config MV64X60_WDT
tristate "MV64X60 (Marvell Discovery) Watchdog Timer"
depends on WATCHDOG && MV64X60
config BOOKE_WDT
tristate "PowerPC Book-E Watchdog Timer"
depends on WATCHDOG && (BOOKE || 4xx)
...
...
@@ -362,6 +412,17 @@ config BOOKE_WDT
Please see Documentation/watchdog/watchdog-api.txt for
more information.
# PPC64 Architecture
config WATCHDOG_RTAS
tristate "RTAS watchdog"
depends on WATCHDOG && PPC_RTAS
help
This driver adds watchdog support for the RTAS watchdog.
To compile this driver as a module, choose M here. The module
will be called wdrtas.
# MIPS Architecture
config INDYDOG
...
...
@@ -430,16 +491,6 @@ config WATCHDOG_RIO
machines. The watchdog timeout period is normally one minute but
can be changed with a boot-time parameter.
# ppc64 RTAS watchdog
config WATCHDOG_RTAS
tristate "RTAS watchdog"
depends on WATCHDOG && PPC_RTAS
help
This driver adds watchdog support for the RTAS watchdog.
To compile this driver as a module, choose M here. The module
will be called wdrtas.
#
# ISA-based Watchdog Cards
#
...
...
drivers/char/watchdog/Makefile
View file @
a8cd2e50
...
...
@@ -39,22 +39,27 @@ obj-$(CONFIG_ALIM7101_WDT) += alim7101_wdt.o
obj-$(CONFIG_SC520_WDT)
+=
sc520_wdt.o
obj-$(CONFIG_EUROTECH_WDT)
+=
eurotechwdt.o
obj-$(CONFIG_IB700_WDT)
+=
ib700wdt.o
obj-$(CONFIG_IBMASR)
+=
ibmasr.o
obj-$(CONFIG_WAFER_WDT)
+=
wafer5823wdt.o
obj-$(CONFIG_I6300ESB_WDT)
+=
i6300esb.o
obj-$(CONFIG_I8XX_TCO)
+=
i8xx_tco.o
obj-$(CONFIG_SC1200_WDT)
+=
sc1200wdt.o
obj-$(CONFIG_SCx200_WDT)
+=
scx200_wdt.o
obj-$(CONFIG_60XX_WDT)
+=
sbc60xxwdt.o
obj-$(CONFIG_SBC8360_WDT)
+=
sbc8360.o
obj-$(CONFIG_CPU5_WDT)
+=
cpu5wdt.o
obj-$(CONFIG_W83627HF_WDT)
+=
w83627hf_wdt.o
obj-$(CONFIG_W83877F_WDT)
+=
w83877f_wdt.o
obj-$(CONFIG_W83977F_WDT)
+=
w83977f_wdt.o
obj-$(CONFIG_MACHZ_WDT)
+=
machzwd.o
# PowerPC Architecture
obj-$(CONFIG_8xx_WDT)
+=
mpc8xx_wdt.o
obj-$(CONFIG_MV64X60_WDT)
+=
mv64x60_wdt.o
obj-$(CONFIG_BOOKE_WDT)
+=
booke_wdt.o
# PPC64 Architecture
obj-$(CONFIG_WATCHDOG_RTAS)
+=
wdrtas.o
obj-$(CONFIG_BOOKE_WDT)
+=
booke_wdt.o
# MIPS Architecture
obj-$(CONFIG_INDYDOG)
+=
indydog.o
...
...
drivers/char/watchdog/i6300esb.c
0 → 100644
View file @
a8cd2e50
/*
* i6300esb: Watchdog timer driver for Intel 6300ESB chipset
*
* (c) Copyright 2004 Google Inc.
* (c) Copyright 2005 David Hrdeman <david@2gen.com>
*
* 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.
*
* based on i810-tco.c which is in turn based on softdog.c
*
* The timer is implemented in the following I/O controller hubs:
* (See the intel documentation on http://developer.intel.com.)
* 6300ESB chip : document number 300641-003
*
* 2004YYZZ Ross Biro
* Initial version 0.01
* 2004YYZZ Ross Biro
* Version 0.02
* 20050210 David Hrdeman <david@2gen.com>
* Ported driver to kernel 2.6
*/
/*
* Includes, defines, variables, module parameters, ...
*/
#include <linux/module.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/mm.h>
#include <linux/miscdevice.h>
#include <linux/watchdog.h>
#include <linux/reboot.h>
#include <linux/init.h>
#include <linux/pci.h>
#include <linux/ioport.h>
#include <asm/uaccess.h>
#include <asm/io.h>
/* Module and version information */
#define ESB_VERSION "0.03"
#define ESB_MODULE_NAME "i6300ESB timer"
#define ESB_DRIVER_NAME ESB_MODULE_NAME ", v" ESB_VERSION
#define PFX ESB_MODULE_NAME ": "
/* PCI configuration registers */
#define ESB_CONFIG_REG 0x60
/* Config register */
#define ESB_LOCK_REG 0x68
/* WDT lock register */
/* Memory mapped registers */
#define ESB_TIMER1_REG BASEADDR + 0x00
/* Timer1 value after each reset */
#define ESB_TIMER2_REG BASEADDR + 0x04
/* Timer2 value after each reset */
#define ESB_GINTSR_REG BASEADDR + 0x08
/* General Interrupt Status Register */
#define ESB_RELOAD_REG BASEADDR + 0x0c
/* Reload register */
/* Lock register bits */
#define ESB_WDT_FUNC ( 0x01 << 2 )
/* Watchdog functionality */
#define ESB_WDT_ENABLE ( 0x01 << 1 )
/* Enable WDT */
#define ESB_WDT_LOCK ( 0x01 << 0 )
/* Lock (nowayout) */
/* Config register bits */
#define ESB_WDT_REBOOT ( 0x01 << 5 )
/* Enable reboot on timeout */
#define ESB_WDT_FREQ ( 0x01 << 2 )
/* Decrement frequency */
#define ESB_WDT_INTTYPE ( 0x11 << 0 )
/* Interrupt type on timer1 timeout */
/* Reload register bits */
#define ESB_WDT_RELOAD ( 0x01 << 8 )
/* prevent timeout */
/* Magic constants */
#define ESB_UNLOCK1 0x80
/* Step 1 to unlock reset registers */
#define ESB_UNLOCK2 0x86
/* Step 2 to unlock reset registers */
/* internal variables */
static
void
__iomem
*
BASEADDR
;
static
spinlock_t
esb_lock
;
/* Guards the hardware */
static
unsigned
long
timer_alive
;
static
struct
pci_dev
*
esb_pci
;
static
unsigned
short
triggered
;
/* The status of the watchdog upon boot */
static
char
esb_expect_close
;
/* module parameters */
#define WATCHDOG_HEARTBEAT 30
/* 30 sec default heartbeat (1<heartbeat<2*1023) */
static
int
heartbeat
=
WATCHDOG_HEARTBEAT
;
/* in seconds */
module_param
(
heartbeat
,
int
,
0
);
MODULE_PARM_DESC
(
heartbeat
,
"Watchdog heartbeat in seconds. (1<heartbeat<2046, default="
__MODULE_STRING
(
WATCHDOG_HEARTBEAT
)
")"
);
static
int
nowayout
=
WATCHDOG_NOWAYOUT
;
module_param
(
nowayout
,
int
,
0
);
MODULE_PARM_DESC
(
nowayout
,
"Watchdog cannot be stopped once started (default=CONFIG_WATCHDOG_NOWAYOUT)"
);
/*
* Some i6300ESB specific functions
*/
/*
* Prepare for reloading the timer by unlocking the proper registers.
* This is performed by first writing 0x80 followed by 0x86 to the
* reload register. After this the appropriate registers can be written
* to once before they need to be unlocked again.
*/
static
inline
void
esb_unlock_registers
(
void
)
{
writeb
(
ESB_UNLOCK1
,
ESB_RELOAD_REG
);
writeb
(
ESB_UNLOCK2
,
ESB_RELOAD_REG
);
}
static
void
esb_timer_start
(
void
)
{
u8
val
;
/* Enable or Enable + Lock? */
val
=
0x02
|
(
nowayout
?
0x01
:
0x00
);
pci_write_config_byte
(
esb_pci
,
ESB_LOCK_REG
,
val
);
}
static
int
esb_timer_stop
(
void
)
{
u8
val
;
spin_lock
(
&
esb_lock
);
/* First, reset timers as suggested by the docs */
esb_unlock_registers
();
writew
(
ESB_WDT_RELOAD
,
ESB_RELOAD_REG
);
/* Then disable the WDT */
pci_write_config_byte
(
esb_pci
,
ESB_LOCK_REG
,
0x0
);
pci_read_config_byte
(
esb_pci
,
ESB_LOCK_REG
,
&
val
);
spin_unlock
(
&
esb_lock
);
/* Returns 0 if the timer was disabled, non-zero otherwise */
return
(
val
&
0x01
);
}
static
void
esb_timer_keepalive
(
void
)
{
spin_lock
(
&
esb_lock
);
esb_unlock_registers
();
writew
(
ESB_WDT_RELOAD
,
ESB_RELOAD_REG
);
/* FIXME: Do we need to flush anything here? */
spin_unlock
(
&
esb_lock
);
}
static
int
esb_timer_set_heartbeat
(
int
time
)
{
u32
val
;
if
(
time
<
0x1
||
time
>
(
2
*
0x03ff
))
return
-
EINVAL
;
spin_lock
(
&
esb_lock
);
/* We shift by 9, so if we are passed a value of 1 sec,
* val will be 1 << 9 = 512, then write that to two
* timers => 2 * 512 = 1024 (which is decremented at 1KHz)
*/
val
=
time
<<
9
;
/* Write timer 1 */
esb_unlock_registers
();
writel
(
val
,
ESB_TIMER1_REG
);
/* Write timer 2 */
esb_unlock_registers
();
writel
(
val
,
ESB_TIMER2_REG
);
/* Reload */
esb_unlock_registers
();
writew
(
ESB_WDT_RELOAD
,
ESB_RELOAD_REG
);
/* FIXME: Do we need to flush everything out? */
/* Done */
heartbeat
=
time
;
spin_unlock
(
&
esb_lock
);
return
0
;
}
static
int
esb_timer_read
(
void
)
{
u32
count
;
/* This isn't documented, and doesn't take into
* acount which stage is running, but it looks
* like a 20 bit count down, so we might as well report it.
*/
pci_read_config_dword
(
esb_pci
,
0x64
,
&
count
);
return
(
int
)
count
;
}
/*
* /dev/watchdog handling
*/
static
int
esb_open
(
struct
inode
*
inode
,
struct
file
*
file
)
{
/* /dev/watchdog can only be opened once */
if
(
test_and_set_bit
(
0
,
&
timer_alive
))
return
-
EBUSY
;
/* Reload and activate timer */
esb_timer_keepalive
();
esb_timer_start
();
return
nonseekable_open
(
inode
,
file
);
}
static
int
esb_release
(
struct
inode
*
inode
,
struct
file
*
file
)
{
/* Shut off the timer. */
if
(
esb_expect_close
==
42
)
{
esb_timer_stop
();
}
else
{
printk
(
KERN_CRIT
PFX
"Unexpected close, not stopping watchdog!
\n
"
);
esb_timer_keepalive
();
}
clear_bit
(
0
,
&
timer_alive
);
esb_expect_close
=
0
;
return
0
;
}
static
ssize_t
esb_write
(
struct
file
*
file
,
const
char
__user
*
data
,
size_t
len
,
loff_t
*
ppos
)
{
/* See if we got the magic character 'V' and reload the timer */
if
(
len
)
{
if
(
!
nowayout
)
{
size_t
i
;
/* note: just in case someone wrote the magic character
* five months ago... */
esb_expect_close
=
0
;
/* scan to see whether or not we got the magic character */
for
(
i
=
0
;
i
!=
len
;
i
++
)
{
char
c
;
if
(
get_user
(
c
,
data
+
i
))
return
-
EFAULT
;
if
(
c
==
'V'
)
esb_expect_close
=
42
;
}
}
/* someone wrote to us, we should reload the timer */
esb_timer_keepalive
();
}
return
len
;
}
static
int
esb_ioctl
(
struct
inode
*
inode
,
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
int
new_options
,
retval
=
-
EINVAL
;
int
new_heartbeat
;
void
__user
*
argp
=
(
void
__user
*
)
arg
;
int
__user
*
p
=
argp
;
static
struct
watchdog_info
ident
=
{
.
options
=
WDIOF_SETTIMEOUT
|
WDIOF_KEEPALIVEPING
|
WDIOF_MAGICCLOSE
,
.
firmware_version
=
0
,
.
identity
=
ESB_MODULE_NAME
,
};
switch
(
cmd
)
{
case
WDIOC_GETSUPPORT
:
return
copy_to_user
(
argp
,
&
ident
,
sizeof
(
ident
))
?
-
EFAULT
:
0
;
case
WDIOC_GETSTATUS
:
return
put_user
(
esb_timer_read
(),
p
);
case
WDIOC_GETBOOTSTATUS
:
return
put_user
(
triggered
,
p
);
case
WDIOC_KEEPALIVE
:
esb_timer_keepalive
();
return
0
;
case
WDIOC_SETOPTIONS
:
{
if
(
get_user
(
new_options
,
p
))
return
-
EFAULT
;
if
(
new_options
&
WDIOS_DISABLECARD
)
{
esb_timer_stop
();
retval
=
0
;
}
if
(
new_options
&
WDIOS_ENABLECARD
)
{
esb_timer_keepalive
();
esb_timer_start
();
retval
=
0
;
}
return
retval
;
}
case
WDIOC_SETTIMEOUT
:
{
if
(
get_user
(
new_heartbeat
,
p
))
return
-
EFAULT
;
if
(
esb_timer_set_heartbeat
(
new_heartbeat
))
return
-
EINVAL
;
esb_timer_keepalive
();
/* Fall */
}
case
WDIOC_GETTIMEOUT
:
return
put_user
(
heartbeat
,
p
);
default:
return
-
ENOIOCTLCMD
;
}
}
/*
* Notify system
*/
static
int
esb_notify_sys
(
struct
notifier_block
*
this
,
unsigned
long
code
,
void
*
unused
)
{
if
(
code
==
SYS_DOWN
||
code
==
SYS_HALT
)
{
/* Turn the WDT off */
esb_timer_stop
();
}
return
NOTIFY_DONE
;
}
/*
* Kernel Interfaces
*/
static
struct
file_operations
esb_fops
=
{
.
owner
=
THIS_MODULE
,
.
llseek
=
no_llseek
,
.
write
=
esb_write
,
.
ioctl
=
esb_ioctl
,
.
open
=
esb_open
,
.
release
=
esb_release
,
};
static
struct
miscdevice
esb_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"watchdog"
,
.
fops
=
&
esb_fops
,
};
static
struct
notifier_block
esb_notifier
=
{
.
notifier_call
=
esb_notify_sys
,
};
/*
* Data for PCI driver interface
*
* This data only exists for exporting the supported
* PCI ids via MODULE_DEVICE_TABLE. We do not actually
* register a pci_driver, because someone else might one day
* want to register another driver on the same PCI id.
*/
static
struct
pci_device_id
esb_pci_tbl
[]
=
{
{
PCI_DEVICE
(
PCI_VENDOR_ID_INTEL
,
PCI_DEVICE_ID_INTEL_ESB_9
),
},
{
0
,
},
/* End of list */
};
MODULE_DEVICE_TABLE
(
pci
,
esb_pci_tbl
);
/*
* Init & exit routines
*/
static
unsigned
char
__init
esb_getdevice
(
void
)
{
u8
val1
;
unsigned
short
val2
;
struct
pci_dev
*
dev
=
NULL
;
/*
* Find the PCI device
*/
for_each_pci_dev
(
dev
)
{
if
(
pci_match_id
(
esb_pci_tbl
,
dev
))
{
esb_pci
=
dev
;
break
;
}
}
if
(
esb_pci
)
{
if
(
pci_enable_device
(
esb_pci
))
{
printk
(
KERN_ERR
PFX
"failed to enable device
\n
"
);
goto
err_devput
;
}
if
(
pci_request_region
(
esb_pci
,
0
,
ESB_MODULE_NAME
))
{
printk
(
KERN_ERR
PFX
"failed to request region
\n
"
);
goto
err_disable
;
}
BASEADDR
=
ioremap
(
pci_resource_start
(
esb_pci
,
0
),
pci_resource_len
(
esb_pci
,
0
));
if
(
BASEADDR
==
NULL
)
{
/* Something's wrong here, BASEADDR has to be set */
printk
(
KERN_ERR
PFX
"failed to get BASEADDR
\n
"
);
goto
err_release
;
}
/*
* The watchdog has two timers, it can be setup so that the
* expiry of timer1 results in an interrupt and the expiry of
* timer2 results in a reboot. We set it to not generate
* any interrupts as there is not much we can do with it
* right now.
*
* We also enable reboots and set the timer frequency to
* the PCI clock divided by 2^15 (approx 1KHz).
*/
pci_write_config_word
(
esb_pci
,
ESB_CONFIG_REG
,
0x0003
);
/* Check that the WDT isn't already locked */
pci_read_config_byte
(
esb_pci
,
ESB_LOCK_REG
,
&
val1
);
if
(
val1
&
ESB_WDT_LOCK
)
printk
(
KERN_WARNING
PFX
"nowayout already set
\n
"
);
/* Set the timer to watchdog mode and disable it for now */
pci_write_config_byte
(
esb_pci
,
ESB_LOCK_REG
,
0x00
);
/* Check if the watchdog was previously triggered */
esb_unlock_registers
();
val2
=
readw
(
ESB_RELOAD_REG
);
triggered
=
(
val2
&
(
0x01
<<
9
)
>>
9
);
/* Reset trigger flag and timers */
esb_unlock_registers
();
writew
((
0x11
<<
8
),
ESB_RELOAD_REG
);
/* Done */
return
1
;
err_release:
pci_release_region
(
esb_pci
,
0
);
err_disable:
pci_disable_device
(
esb_pci
);
err_devput:
pci_dev_put
(
esb_pci
);
}
return
0
;
}
static
int
__init
watchdog_init
(
void
)
{
int
ret
;
spin_lock_init
(
&
esb_lock
);
/* Check whether or not the hardware watchdog is there */
if
(
!
esb_getdevice
()
||
esb_pci
==
NULL
)
return
-
ENODEV
;
/* Check that the heartbeat value is within it's range ; if not reset to the default */
if
(
esb_timer_set_heartbeat
(
heartbeat
))
{
esb_timer_set_heartbeat
(
WATCHDOG_HEARTBEAT
);
printk
(
KERN_INFO
PFX
"heartbeat value must be 1<heartbeat<2046, using %d
\n
"
,
heartbeat
);
}
ret
=
register_reboot_notifier
(
&
esb_notifier
);
if
(
ret
!=
0
)
{
printk
(
KERN_ERR
PFX
"cannot register reboot notifier (err=%d)
\n
"
,
ret
);
goto
err_unmap
;
}
ret
=
misc_register
(
&
esb_miscdev
);
if
(
ret
!=
0
)
{
printk
(
KERN_ERR
PFX
"cannot register miscdev on minor=%d (err=%d)
\n
"
,
WATCHDOG_MINOR
,
ret
);
goto
err_notifier
;
}
esb_timer_stop
();
printk
(
KERN_INFO
PFX
"initialized (0x%p). heartbeat=%d sec (nowayout=%d)
\n
"
,
BASEADDR
,
heartbeat
,
nowayout
);
return
0
;
err_notifier:
unregister_reboot_notifier
(
&
esb_notifier
);
err_unmap:
iounmap
(
BASEADDR
);
/* err_release: */
pci_release_region
(
esb_pci
,
0
);
/* err_disable: */
pci_disable_device
(
esb_pci
);
/* err_devput: */
pci_dev_put
(
esb_pci
);
return
ret
;
}
static
void
__exit
watchdog_cleanup
(
void
)
{
/* Stop the timer before we leave */
if
(
!
nowayout
)
esb_timer_stop
();
/* Deregister */
misc_deregister
(
&
esb_miscdev
);
unregister_reboot_notifier
(
&
esb_notifier
);
iounmap
(
BASEADDR
);
pci_release_region
(
esb_pci
,
0
);
pci_disable_device
(
esb_pci
);
pci_dev_put
(
esb_pci
);
}
module_init
(
watchdog_init
);
module_exit
(
watchdog_cleanup
);
MODULE_AUTHOR
(
"Ross Biro and David Hrdeman"
);
MODULE_DESCRIPTION
(
"Watchdog driver for Intel 6300ESB chipsets"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
drivers/char/watchdog/ibmasr.c
0 → 100644
View file @
a8cd2e50
/*
* IBM Automatic Server Restart driver.
*
* Copyright (c) 2005 Andrey Panin <pazke@donpac.ru>
*
* Based on driver written by Pete Reynolds.
* Copyright (c) IBM Corporation, 1998-2004.
*
* This software may be used and distributed according to the terms
* of the GNU Public License, incorporated herein by reference.
*/
#include <linux/config.h>
#include <linux/fs.h>
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/module.h>
#include <linux/pci.h>
#include <linux/timer.h>
#include <linux/miscdevice.h>
#include <linux/watchdog.h>
#include <linux/dmi.h>
#include <asm/io.h>
#include <asm/uaccess.h>
enum
{
ASMTYPE_UNKNOWN
,
ASMTYPE_TOPAZ
,
ASMTYPE_JASPER
,
ASMTYPE_PEARL
,
ASMTYPE_JUNIPER
,
ASMTYPE_SPRUCE
,
};
#define PFX "ibmasr: "
#define TOPAZ_ASR_REG_OFFSET 4
#define TOPAZ_ASR_TOGGLE 0x40
#define TOPAZ_ASR_DISABLE 0x80
/* PEARL ASR S/W REGISTER SUPERIO PORT ADDRESSES */
#define PEARL_BASE 0xe04
#define PEARL_WRITE 0xe06
#define PEARL_READ 0xe07
#define PEARL_ASR_DISABLE_MASK 0x80
/* bit 7: disable = 1, enable = 0 */
#define PEARL_ASR_TOGGLE_MASK 0x40
/* bit 6: 0, then 1, then 0 */
/* JASPER OFFSET FROM SIO BASE ADDR TO ASR S/W REGISTERS. */
#define JASPER_ASR_REG_OFFSET 0x38
#define JASPER_ASR_DISABLE_MASK 0x01
/* bit 0: disable = 1, enable = 0 */
#define JASPER_ASR_TOGGLE_MASK 0x02
/* bit 1: 0, then 1, then 0 */
#define JUNIPER_BASE_ADDRESS 0x54b
/* Base address of Juniper ASR */
#define JUNIPER_ASR_DISABLE_MASK 0x01
/* bit 0: disable = 1 enable = 0 */
#define JUNIPER_ASR_TOGGLE_MASK 0x02
/* bit 1: 0, then 1, then 0 */
#define SPRUCE_BASE_ADDRESS 0x118e
/* Base address of Spruce ASR */
#define SPRUCE_ASR_DISABLE_MASK 0x01
/* bit 1: disable = 1 enable = 0 */
#define SPRUCE_ASR_TOGGLE_MASK 0x02
/* bit 0: 0, then 1, then 0 */
static
int
nowayout
=
WATCHDOG_NOWAYOUT
;
static
unsigned
long
asr_is_open
;
static
char
asr_expect_close
;
static
unsigned
int
asr_type
,
asr_base
,
asr_length
;
static
unsigned
int
asr_read_addr
,
asr_write_addr
;
static
unsigned
char
asr_toggle_mask
,
asr_disable_mask
;
static
void
asr_toggle
(
void
)
{
unsigned
char
reg
=
inb
(
asr_read_addr
);
outb
(
reg
&
~
asr_toggle_mask
,
asr_write_addr
);
reg
=
inb
(
asr_read_addr
);
outb
(
reg
|
asr_toggle_mask
,
asr_write_addr
);
reg
=
inb
(
asr_read_addr
);
outb
(
reg
&
~
asr_toggle_mask
,
asr_write_addr
);
reg
=
inb
(
asr_read_addr
);
}
static
void
asr_enable
(
void
)
{
unsigned
char
reg
;
if
(
asr_type
==
ASMTYPE_TOPAZ
)
{
/* asr_write_addr == asr_read_addr */
reg
=
inb
(
asr_read_addr
);
outb
(
reg
&
~
(
TOPAZ_ASR_TOGGLE
|
TOPAZ_ASR_DISABLE
),
asr_read_addr
);
}
else
{
/*
* First make sure the hardware timer is reset by toggling
* ASR hardware timer line.
*/
asr_toggle
();
reg
=
inb
(
asr_read_addr
);
outb
(
reg
&
~
asr_disable_mask
,
asr_write_addr
);
}
reg
=
inb
(
asr_read_addr
);
}
static
void
asr_disable
(
void
)
{
unsigned
char
reg
=
inb
(
asr_read_addr
);
if
(
asr_type
==
ASMTYPE_TOPAZ
)
/* asr_write_addr == asr_read_addr */
outb
(
reg
|
TOPAZ_ASR_TOGGLE
|
TOPAZ_ASR_DISABLE
,
asr_read_addr
);
else
{
outb
(
reg
|
asr_toggle_mask
,
asr_write_addr
);
reg
=
inb
(
asr_read_addr
);
outb
(
reg
|
asr_disable_mask
,
asr_write_addr
);
}
reg
=
inb
(
asr_read_addr
);
}
static
int
__init
asr_get_base_address
(
void
)
{
unsigned
char
low
,
high
;
const
char
*
type
=
""
;
asr_length
=
1
;
switch
(
asr_type
)
{
case
ASMTYPE_TOPAZ
:
/* SELECT SuperIO CHIP FOR QUERYING (WRITE 0x07 TO BOTH 0x2E and 0x2F) */
outb
(
0x07
,
0x2e
);
outb
(
0x07
,
0x2f
);
/* SELECT AND READ THE HIGH-NIBBLE OF THE GPIO BASE ADDRESS */
outb
(
0x60
,
0x2e
);
high
=
inb
(
0x2f
);
/* SELECT AND READ THE LOW-NIBBLE OF THE GPIO BASE ADDRESS */
outb
(
0x61
,
0x2e
);
low
=
inb
(
0x2f
);
asr_base
=
(
high
<<
16
)
|
low
;
asr_read_addr
=
asr_write_addr
=
asr_base
+
TOPAZ_ASR_REG_OFFSET
;
asr_length
=
5
;
break
;
case
ASMTYPE_JASPER
:
type
=
"Jaspers "
;
/* FIXME: need to use pci_config_lock here, but it's not exported */
/* spin_lock_irqsave(&pci_config_lock, flags);*/
/* Select the SuperIO chip in the PCI I/O port register */
outl
(
0x8000f858
,
0xcf8
);
/*
* Read the base address for the SuperIO chip.
* Only the lower 16 bits are valid, but the address is word
* aligned so the last bit must be masked off.
*/
asr_base
=
inl
(
0xcfc
)
&
0xfffe
;
/* spin_unlock_irqrestore(&pci_config_lock, flags);*/
asr_read_addr
=
asr_write_addr
=
asr_base
+
JASPER_ASR_REG_OFFSET
;
asr_toggle_mask
=
JASPER_ASR_TOGGLE_MASK
;
asr_disable_mask
=
JASPER_ASR_DISABLE_MASK
;
asr_length
=
JASPER_ASR_REG_OFFSET
+
1
;
break
;
case
ASMTYPE_PEARL
:
type
=
"Pearls "
;
asr_base
=
PEARL_BASE
;
asr_read_addr
=
PEARL_READ
;
asr_write_addr
=
PEARL_WRITE
;
asr_toggle_mask
=
PEARL_ASR_TOGGLE_MASK
;
asr_disable_mask
=
PEARL_ASR_DISABLE_MASK
;
asr_length
=
4
;
break
;
case
ASMTYPE_JUNIPER
:
type
=
"Junipers "
;
asr_base
=
JUNIPER_BASE_ADDRESS
;
asr_read_addr
=
asr_write_addr
=
asr_base
;
asr_toggle_mask
=
JUNIPER_ASR_TOGGLE_MASK
;
asr_disable_mask
=
JUNIPER_ASR_DISABLE_MASK
;
break
;
case
ASMTYPE_SPRUCE
:
type
=
"Spruce's "
;
asr_base
=
SPRUCE_BASE_ADDRESS
;
asr_read_addr
=
asr_write_addr
=
asr_base
;
asr_toggle_mask
=
SPRUCE_ASR_TOGGLE_MASK
;
asr_disable_mask
=
SPRUCE_ASR_DISABLE_MASK
;
break
;
}
if
(
!
request_region
(
asr_base
,
asr_length
,
"ibmasr"
))
{
printk
(
KERN_ERR
PFX
"address %#x already in use
\n
"
,
asr_base
);
return
-
EBUSY
;
}
printk
(
KERN_INFO
PFX
"found %sASR @ addr %#x
\n
"
,
type
,
asr_base
);
return
0
;
}
static
ssize_t
asr_write
(
struct
file
*
file
,
const
char
__user
*
buf
,
size_t
count
,
loff_t
*
ppos
)
{
if
(
count
)
{
if
(
!
nowayout
)
{
size_t
i
;
/* In case it was set long ago */
asr_expect_close
=
0
;
for
(
i
=
0
;
i
!=
count
;
i
++
)
{
char
c
;
if
(
get_user
(
c
,
buf
+
i
))
return
-
EFAULT
;
if
(
c
==
'V'
)
asr_expect_close
=
42
;
}
}
asr_toggle
();
}
return
count
;
}
static
int
asr_ioctl
(
struct
inode
*
inode
,
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
static
const
struct
watchdog_info
ident
=
{
.
options
=
WDIOF_KEEPALIVEPING
|
WDIOF_MAGICCLOSE
,
.
identity
=
"IBM ASR"
};
void
__user
*
argp
=
(
void
__user
*
)
arg
;
int
__user
*
p
=
argp
;
int
heartbeat
;
switch
(
cmd
)
{
case
WDIOC_GETSUPPORT
:
return
copy_to_user
(
argp
,
&
ident
,
sizeof
(
ident
))
?
-
EFAULT
:
0
;
case
WDIOC_GETSTATUS
:
case
WDIOC_GETBOOTSTATUS
:
return
put_user
(
0
,
p
);
case
WDIOC_KEEPALIVE
:
asr_toggle
();
return
0
;
/*
* The hardware has a fixed timeout value, so no WDIOC_SETTIMEOUT
* and WDIOC_GETTIMEOUT always returns 256.
*/
case
WDIOC_GETTIMEOUT
:
heartbeat
=
256
;
return
put_user
(
heartbeat
,
p
);
case
WDIOC_SETOPTIONS
:
{
int
new_options
,
retval
=
-
EINVAL
;
if
(
get_user
(
new_options
,
p
))
return
-
EFAULT
;
if
(
new_options
&
WDIOS_DISABLECARD
)
{
asr_disable
();
retval
=
0
;
}
if
(
new_options
&
WDIOS_ENABLECARD
)
{
asr_enable
();
asr_toggle
();
retval
=
0
;
}
return
retval
;
}
}
return
-
ENOIOCTLCMD
;
}
static
int
asr_open
(
struct
inode
*
inode
,
struct
file
*
file
)
{
if
(
test_and_set_bit
(
0
,
&
asr_is_open
))
return
-
EBUSY
;
asr_toggle
();
asr_enable
();
return
nonseekable_open
(
inode
,
file
);
}
static
int
asr_release
(
struct
inode
*
inode
,
struct
file
*
file
)
{
if
(
asr_expect_close
==
42
)
asr_disable
();
else
{
printk
(
KERN_CRIT
PFX
"unexpected close, not stopping watchdog!
\n
"
);
asr_toggle
();
}
clear_bit
(
0
,
&
asr_is_open
);
asr_expect_close
=
0
;
return
0
;
}
static
struct
file_operations
asr_fops
=
{
.
owner
=
THIS_MODULE
,
.
llseek
=
no_llseek
,
.
write
=
asr_write
,
.
ioctl
=
asr_ioctl
,
.
open
=
asr_open
,
.
release
=
asr_release
,
};
static
struct
miscdevice
asr_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"watchdog"
,
.
fops
=
&
asr_fops
,
};
struct
ibmasr_id
{
const
char
*
desc
;
int
type
;
};
static
struct
ibmasr_id
__initdata
ibmasr_id_table
[]
=
{
{
"IBM Automatic Server Restart - eserver xSeries 220"
,
ASMTYPE_TOPAZ
},
{
"IBM Automatic Server Restart - Machine Type 8673"
,
ASMTYPE_PEARL
},
{
"IBM Automatic Server Restart - Machine Type 8480"
,
ASMTYPE_JASPER
},
{
"IBM Automatic Server Restart - Machine Type 8482"
,
ASMTYPE_JUNIPER
},
{
"IBM Automatic Server Restart - Machine Type 8648"
,
ASMTYPE_SPRUCE
},
{
NULL
}
};
static
int
__init
ibmasr_init
(
void
)
{
struct
ibmasr_id
*
id
;
int
rc
;
for
(
id
=
ibmasr_id_table
;
id
->
desc
;
id
++
)
{
if
(
dmi_find_device
(
DMI_DEV_TYPE_OTHER
,
id
->
desc
,
NULL
))
{
asr_type
=
id
->
type
;
break
;
}
}
if
(
!
asr_type
)
return
-
ENODEV
;
rc
=
misc_register
(
&
asr_miscdev
);
if
(
rc
<
0
)
{
printk
(
KERN_ERR
PFX
"failed to register misc device
\n
"
);
return
rc
;
}
rc
=
asr_get_base_address
();
if
(
rc
)
{
misc_deregister
(
&
asr_miscdev
);
return
rc
;
}
return
0
;
}
static
void
__exit
ibmasr_exit
(
void
)
{
if
(
!
nowayout
)
asr_disable
();
misc_deregister
(
&
asr_miscdev
);
release_region
(
asr_base
,
asr_length
);
}
module_init
(
ibmasr_init
);
module_exit
(
ibmasr_exit
);
module_param
(
nowayout
,
int
,
0
);
MODULE_PARM_DESC
(
nowayout
,
"Watchdog cannot be stopped once started (default=CONFIG_WATCHDOG_NOWAYOUT)"
);
MODULE_DESCRIPTION
(
"IBM Automatic Server Restart driver"
);
MODULE_AUTHOR
(
"Andrey Panin"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
drivers/char/watchdog/mv64x60_wdt.c
0 → 100644
View file @
a8cd2e50
/*
* mv64x60_wdt.c - MV64X60 (Marvell Discovery) watchdog userspace interface
*
* Author: James Chapman <jchapman@katalix.com>
*
* Platform-specific setup code should configure the dog to generate
* interrupt or reset as required. This code only enables/disables
* and services the watchdog.
*
* Derived from mpc8xx_wdt.c, with the following copyright.
*
* 2002 (c) Florian Schirmer <jolt@tuxbox.org> This file is licensed under
* the terms of the GNU General Public License version 2. This program
* is licensed "as is" without any warranty of any kind, whether express
* or implied.
*/
#include <linux/config.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/miscdevice.h>
#include <linux/module.h>
#include <linux/watchdog.h>
#include <asm/mv64x60.h>
#include <asm/uaccess.h>
#include <asm/io.h>
/* MV64x60 WDC (config) register access definitions */
#define MV64x60_WDC_CTL1_MASK (3 << 24)
#define MV64x60_WDC_CTL1(val) ((val & 3) << 24)
#define MV64x60_WDC_CTL2_MASK (3 << 26)
#define MV64x60_WDC_CTL2(val) ((val & 3) << 26)
/* Flags bits */
#define MV64x60_WDOG_FLAG_OPENED 0
#define MV64x60_WDOG_FLAG_ENABLED 1
static
unsigned
long
wdt_flags
;
static
int
wdt_status
;
static
void
__iomem
*
mv64x60_regs
;
static
int
mv64x60_wdt_timeout
;
static
void
mv64x60_wdt_reg_write
(
u32
val
)
{
/* Allow write only to CTL1 / CTL2 fields, retaining values in
* other fields.
*/
u32
data
=
readl
(
mv64x60_regs
+
MV64x60_WDT_WDC
);
data
&=
~
(
MV64x60_WDC_CTL1_MASK
|
MV64x60_WDC_CTL2_MASK
);
data
|=
val
;
writel
(
data
,
mv64x60_regs
+
MV64x60_WDT_WDC
);
}
static
void
mv64x60_wdt_service
(
void
)
{
/* Write 01 followed by 10 to CTL2 */
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL2
(
0x01
));
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL2
(
0x02
));
}
static
void
mv64x60_wdt_handler_disable
(
void
)
{
if
(
test_and_clear_bit
(
MV64x60_WDOG_FLAG_ENABLED
,
&
wdt_flags
))
{
/* Write 01 followed by 10 to CTL1 */
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL1
(
0x01
));
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL1
(
0x02
));
printk
(
KERN_NOTICE
"mv64x60_wdt: watchdog deactivated
\n
"
);
}
}
static
void
mv64x60_wdt_handler_enable
(
void
)
{
if
(
!
test_and_set_bit
(
MV64x60_WDOG_FLAG_ENABLED
,
&
wdt_flags
))
{
/* Write 01 followed by 10 to CTL1 */
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL1
(
0x01
));
mv64x60_wdt_reg_write
(
MV64x60_WDC_CTL1
(
0x02
));
printk
(
KERN_NOTICE
"mv64x60_wdt: watchdog activated
\n
"
);
}
}
static
int
mv64x60_wdt_open
(
struct
inode
*
inode
,
struct
file
*
file
)
{
if
(
test_and_set_bit
(
MV64x60_WDOG_FLAG_OPENED
,
&
wdt_flags
))
return
-
EBUSY
;
mv64x60_wdt_service
();
mv64x60_wdt_handler_enable
();
return
0
;
}
static
int
mv64x60_wdt_release
(
struct
inode
*
inode
,
struct
file
*
file
)
{
mv64x60_wdt_service
();
#if !defined(CONFIG_WATCHDOG_NOWAYOUT)
mv64x60_wdt_handler_disable
();
#endif
clear_bit
(
MV64x60_WDOG_FLAG_OPENED
,
&
wdt_flags
);
return
0
;
}
static
ssize_t
mv64x60_wdt_write
(
struct
file
*
file
,
const
char
*
data
,
size_t
len
,
loff_t
*
ppos
)
{
if
(
*
ppos
!=
file
->
f_pos
)
return
-
ESPIPE
;
if
(
len
)
mv64x60_wdt_service
();
return
len
;
}
static
int
mv64x60_wdt_ioctl
(
struct
inode
*
inode
,
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
int
timeout
;
static
struct
watchdog_info
info
=
{
.
options
=
WDIOF_KEEPALIVEPING
,
.
firmware_version
=
0
,
.
identity
=
"MV64x60 watchdog"
,
};
switch
(
cmd
)
{
case
WDIOC_GETSUPPORT
:
if
(
copy_to_user
((
void
*
)
arg
,
&
info
,
sizeof
(
info
)))
return
-
EFAULT
;
break
;
case
WDIOC_GETSTATUS
:
case
WDIOC_GETBOOTSTATUS
:
if
(
put_user
(
wdt_status
,
(
int
*
)
arg
))
return
-
EFAULT
;
wdt_status
&=
~
WDIOF_KEEPALIVEPING
;
break
;
case
WDIOC_GETTEMP
:
return
-
EOPNOTSUPP
;
case
WDIOC_SETOPTIONS
:
return
-
EOPNOTSUPP
;
case
WDIOC_KEEPALIVE
:
mv64x60_wdt_service
();
wdt_status
|=
WDIOF_KEEPALIVEPING
;
break
;
case
WDIOC_SETTIMEOUT
:
return
-
EOPNOTSUPP
;
case
WDIOC_GETTIMEOUT
:
timeout
=
mv64x60_wdt_timeout
*
HZ
;
if
(
put_user
(
timeout
,
(
int
*
)
arg
))
return
-
EFAULT
;
break
;
default:
return
-
ENOIOCTLCMD
;
}
return
0
;
}
static
struct
file_operations
mv64x60_wdt_fops
=
{
.
owner
=
THIS_MODULE
,
.
llseek
=
no_llseek
,
.
write
=
mv64x60_wdt_write
,
.
ioctl
=
mv64x60_wdt_ioctl
,
.
open
=
mv64x60_wdt_open
,
.
release
=
mv64x60_wdt_release
,
};
static
struct
miscdevice
mv64x60_wdt_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"watchdog"
,
.
fops
=
&
mv64x60_wdt_fops
,
};
static
int
__devinit
mv64x60_wdt_probe
(
struct
device
*
dev
)
{
struct
platform_device
*
pd
=
to_platform_device
(
dev
);
struct
mv64x60_wdt_pdata
*
pdata
=
pd
->
dev
.
platform_data
;
int
bus_clk
=
133
;
mv64x60_wdt_timeout
=
10
;
if
(
pdata
)
{
mv64x60_wdt_timeout
=
pdata
->
timeout
;
bus_clk
=
pdata
->
bus_clk
;
}
mv64x60_regs
=
mv64x60_get_bridge_vbase
();
writel
((
mv64x60_wdt_timeout
*
(
bus_clk
*
1000000
))
>>
8
,
mv64x60_regs
+
MV64x60_WDT_WDC
);
return
misc_register
(
&
mv64x60_wdt_miscdev
);
}
static
int
__devexit
mv64x60_wdt_remove
(
struct
device
*
dev
)
{
misc_deregister
(
&
mv64x60_wdt_miscdev
);
mv64x60_wdt_service
();
mv64x60_wdt_handler_disable
();
return
0
;
}
static
struct
device_driver
mv64x60_wdt_driver
=
{
.
name
=
MV64x60_WDT_NAME
,
.
bus
=
&
platform_bus_type
,
.
probe
=
mv64x60_wdt_probe
,
.
remove
=
__devexit_p
(
mv64x60_wdt_remove
),
};
static
struct
platform_device
*
mv64x60_wdt_dev
;
static
int
__init
mv64x60_wdt_init
(
void
)
{
int
ret
;
printk
(
KERN_INFO
"MV64x60 watchdog driver
\n
"
);
mv64x60_wdt_dev
=
platform_device_register_simple
(
MV64x60_WDT_NAME
,
-
1
,
NULL
,
0
);
if
(
IS_ERR
(
mv64x60_wdt_dev
))
{
ret
=
PTR_ERR
(
mv64x60_wdt_dev
);
goto
out
;
}
ret
=
driver_register
(
&
mv64x60_wdt_driver
);
out:
return
ret
;
}
static
void
__exit
mv64x60_wdt_exit
(
void
)
{
driver_unregister
(
&
mv64x60_wdt_driver
);
platform_device_unregister
(
mv64x60_wdt_dev
);
}
module_init
(
mv64x60_wdt_init
);
module_exit
(
mv64x60_wdt_exit
);
MODULE_AUTHOR
(
"James Chapman <jchapman@katalix.com>"
);
MODULE_DESCRIPTION
(
"MV64x60 watchdog driver"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
drivers/char/watchdog/pcwd_pci.c
View file @
a8cd2e50
...
...
@@ -29,27 +29,29 @@
* Includes, defines, variables, module parameters, ...
*/
#include <linux/config.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/types.h>
#include <linux/delay.h>
#include <linux/miscdevice.h>
#include <linux/watchdog.h>
#include <linux/notifier.h>
#include <linux/reboot.h>
#include <linux/init.h>
#include <linux/fs.h>
#include <linux/pci.h>
#include <linux/ioport.h>
#include <linux/spinlock.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/config.h>
/* For CONFIG_WATCHDOG_NOWAYOUT/... */
#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/miscdevice.h>
/* For MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR) */
#include <linux/watchdog.h>
/* For the watchdog specific items */
#include <linux/notifier.h>
/* For notifier support */
#include <linux/reboot.h>
/* For reboot_notifier stuff */
#include <linux/init.h>
/* For __init/__exit/... */
#include <linux/fs.h>
/* For file operations */
#include <linux/pci.h>
/* For pci functions */
#include <linux/ioport.h>
/* For io-port access */
#include <linux/spinlock.h>
/* For spin_lock/spin_unlock/... */
#include <asm/uaccess.h>
/* For copy_to_user/put_user/... */
#include <asm/io.h>
/* For inb/outb/... */
/* Module and version information */
#define WATCHDOG_VERSION "1.01"
#define WATCHDOG_DATE "
15 Mar
2005"
#define WATCHDOG_DATE "
02 Sep
2005"
#define WATCHDOG_DRIVER_NAME "PCI-PC Watchdog"
#define WATCHDOG_NAME "pcwd_pci"
#define PFX WATCHDOG_NAME ": "
...
...
@@ -335,12 +337,14 @@ static int pcipcwd_ioctl(struct inode *inode, struct file *file,
return
-
EFAULT
;
if
(
new_options
&
WDIOS_DISABLECARD
)
{
pcipcwd_stop
();
if
(
pcipcwd_stop
())
return
-
EIO
;
retval
=
0
;
}
if
(
new_options
&
WDIOS_ENABLECARD
)
{
pcipcwd_start
();
if
(
pcipcwd_start
())
return
-
EIO
;
retval
=
0
;
}
...
...
drivers/char/watchdog/s3c2410_wdt.c
View file @
a8cd2e50
...
...
@@ -464,7 +464,7 @@ static void s3c2410wdt_shutdown(struct device *dev)
static
unsigned
long
wtcon_save
;
static
unsigned
long
wtdat_save
;
static
int
s3c2410wdt_suspend
(
struct
device
*
dev
,
u32
state
,
u32
level
)
static
int
s3c2410wdt_suspend
(
struct
device
*
dev
,
pm_message_t
state
,
u32
level
)
{
if
(
level
==
SUSPEND_POWER_DOWN
)
{
/* Save watchdog state, and turn it off. */
...
...
drivers/char/watchdog/sbc8360.c
0 → 100644
View file @
a8cd2e50
/*
* SBC8360 Watchdog driver
*
* (c) Copyright 2005 Webcon, Inc.
*
* Based on ib700wdt.c, which is based on advantechwdt.c which is based
* on acquirewdt.c which is based on wdt.c.
*
* (c) Copyright 2001 Charles Howes <chowes@vsol.net>
*
* Based on advantechwdt.c which is based on acquirewdt.c which
* is based on wdt.c.
*
* (c) Copyright 2000-2001 Marek Michalkiewicz <marekm@linux.org.pl>
*
* Based on acquirewdt.c which is based on wdt.c.
* Original copyright messages:
*
* (c) Copyright 1996 Alan Cox <alan@redhat.com>, All Rights Reserved.
* http://www.redhat.com
*
* 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.
*
* Neither Alan Cox nor CymruNet Ltd. admit liability nor provide
* warranty for any of this software. This material is provided
* "AS-IS" and at no charge.
*
* (c) Copyright 1995 Alan Cox <alan@redhat.com>
*
* 14-Dec-2001 Matt Domsch <Matt_Domsch@dell.com>
* Added nowayout module option to override CONFIG_WATCHDOG_NOWAYOUT
* Added timeout module option to override default
*
*/
#include <linux/config.h>
#include <linux/module.h>
#include <linux/types.h>
#include <linux/miscdevice.h>
#include <linux/watchdog.h>
#include <linux/ioport.h>
#include <linux/delay.h>
#include <linux/notifier.h>
#include <linux/fs.h>
#include <linux/reboot.h>
#include <linux/init.h>
#include <linux/spinlock.h>
#include <linux/moduleparam.h>
#include <asm/io.h>
#include <asm/uaccess.h>
#include <asm/system.h>
static
unsigned
long
sbc8360_is_open
;
static
spinlock_t
sbc8360_lock
;
static
char
expect_close
;
#define PFX "sbc8360: "
/*
*
* Watchdog Timer Configuration
*
* The function of the watchdog timer is to reset the system automatically
* and is defined at I/O port 0120H and 0121H. To enable the watchdog timer
* and allow the system to reset, write appropriate values from the table
* below to I/O port 0120H and 0121H. To disable the timer, write a zero
* value to I/O port 0121H for the system to stop the watchdog function.
*
* The following describes how the timer should be programmed (according to
* the vendor documentation)
*
* Enabling Watchdog:
* MOV AX,000AH (enable, phase I)
* MOV DX,0120H
* OUT DX,AX
* MOV AX,000BH (enable, phase II)
* MOV DX,0120H
* OUT DX,AX
* MOV AX,000nH (set multiplier n, from 1-4)
* MOV DX,0120H
* OUT DX,AX
* MOV AX,000mH (set base timer m, from 0-F)
* MOV DX,0121H
* OUT DX,AX
*
* Reset timer:
* MOV AX,000mH (same as set base timer, above)
* MOV DX,0121H
* OUT DX,AX
*
* Disabling Watchdog:
* MOV AX,0000H (a zero value)
* MOV DX,0120H
* OUT DX,AX
*
* Watchdog timeout configuration values:
* N
* M | 1 2 3 4
* --|----------------------------------
* 0 | 0.5s 5s 50s 100s
* 1 | 1s 10s 100s 200s
* 2 | 1.5s 15s 150s 300s
* 3 | 2s 20s 200s 400s
* 4 | 2.5s 25s 250s 500s
* 5 | 3s 30s 300s 600s
* 6 | 3.5s 35s 350s 700s
* 7 | 4s 40s 400s 800s
* 8 | 4.5s 45s 450s 900s
* 9 | 5s 50s 500s 1000s
* A | 5.5s 55s 550s 1100s
* B | 6s 60s 600s 1200s
* C | 6.5s 65s 650s 1300s
* D | 7s 70s 700s 1400s
* E | 7.5s 75s 750s 1500s
* F | 8s 80s 800s 1600s
*
* Another way to say the same things is:
* For N=1, Timeout = (M+1) * 0.5s
* For N=2, Timeout = (M+1) * 5s
* For N=3, Timeout = (M+1) * 50s
* For N=4, Timeout = (M+1) * 100s
*
*/
static
int
wd_times
[
64
][
2
]
=
{
{
0
,
1
},
/* 0 = 0.5s */
{
1
,
1
},
/* 1 = 1s */
{
2
,
1
},
/* 2 = 1.5s */
{
3
,
1
},
/* 3 = 2s */
{
4
,
1
},
/* 4 = 2.5s */
{
5
,
1
},
/* 5 = 3s */
{
6
,
1
},
/* 6 = 3.5s */
{
7
,
1
},
/* 7 = 4s */
{
8
,
1
},
/* 8 = 4.5s */
{
9
,
1
},
/* 9 = 5s */
{
0xA
,
1
},
/* 10 = 5.5s */
{
0xB
,
1
},
/* 11 = 6s */
{
0xC
,
1
},
/* 12 = 6.5s */
{
0xD
,
1
},
/* 13 = 7s */
{
0xE
,
1
},
/* 14 = 7.5s */
{
0xF
,
1
},
/* 15 = 8s */
{
0
,
2
},
/* 16 = 5s */
{
1
,
2
},
/* 17 = 10s */
{
2
,
2
},
/* 18 = 15s */
{
3
,
2
},
/* 19 = 20s */
{
4
,
2
},
/* 20 = 25s */
{
5
,
2
},
/* 21 = 30s */
{
6
,
2
},
/* 22 = 35s */
{
7
,
2
},
/* 23 = 40s */
{
8
,
2
},
/* 24 = 45s */
{
9
,
2
},
/* 25 = 50s */
{
0xA
,
2
},
/* 26 = 55s */
{
0xB
,
2
},
/* 27 = 60s */
{
0xC
,
2
},
/* 28 = 65s */
{
0xD
,
2
},
/* 29 = 70s */
{
0xE
,
2
},
/* 30 = 75s */
{
0xF
,
2
},
/* 31 = 80s */
{
0
,
3
},
/* 32 = 50s */
{
1
,
3
},
/* 33 = 100s */
{
2
,
3
},
/* 34 = 150s */
{
3
,
3
},
/* 35 = 200s */
{
4
,
3
},
/* 36 = 250s */
{
5
,
3
},
/* 37 = 300s */
{
6
,
3
},
/* 38 = 350s */
{
7
,
3
},
/* 39 = 400s */
{
8
,
3
},
/* 40 = 450s */
{
9
,
3
},
/* 41 = 500s */
{
0xA
,
3
},
/* 42 = 550s */
{
0xB
,
3
},
/* 43 = 600s */
{
0xC
,
3
},
/* 44 = 650s */
{
0xD
,
3
},
/* 45 = 700s */
{
0xE
,
3
},
/* 46 = 750s */
{
0xF
,
3
},
/* 47 = 800s */
{
0
,
4
},
/* 48 = 100s */
{
1
,
4
},
/* 49 = 200s */
{
2
,
4
},
/* 50 = 300s */
{
3
,
4
},
/* 51 = 400s */
{
4
,
4
},
/* 52 = 500s */
{
5
,
4
},
/* 53 = 600s */
{
6
,
4
},
/* 54 = 700s */
{
7
,
4
},
/* 55 = 800s */
{
8
,
4
},
/* 56 = 900s */
{
9
,
4
},
/* 57 = 1000s */
{
0xA
,
4
},
/* 58 = 1100s */
{
0xB
,
4
},
/* 59 = 1200s */
{
0xC
,
4
},
/* 60 = 1300s */
{
0xD
,
4
},
/* 61 = 1400s */
{
0xE
,
4
},
/* 62 = 1500s */
{
0xF
,
4
}
/* 63 = 1600s */
};
#define SBC8360_ENABLE 0x120
#define SBC8360_BASETIME 0x121
static
int
timeout
=
27
;
static
int
wd_margin
=
0xB
;
static
int
wd_multiplier
=
2
;
static
int
nowayout
=
WATCHDOG_NOWAYOUT
;
module_param
(
timeout
,
int
,
27
);
MODULE_PARM_DESC
(
timeout
,
"Index into timeout table (0-63) (default=27 (60s))"
);
module_param
(
nowayout
,
int
,
0
);
MODULE_PARM_DESC
(
nowayout
,
"Watchdog cannot be stopped once started (default=CONFIG_WATCHDOG_NOWAYOUT)"
);
/*
* Kernel methods.
*/
/* Activate and pre-configure watchdog */
static
void
sbc8360_activate
(
void
)
{
/* Enable the watchdog */
outb
(
0x0A
,
SBC8360_ENABLE
);
msleep_interruptible
(
100
);
outb
(
0x0B
,
SBC8360_ENABLE
);
msleep_interruptible
(
100
);
/* Set timeout multiplier */
outb
(
wd_multiplier
,
SBC8360_ENABLE
);
msleep_interruptible
(
100
);
/* Nothing happens until first sbc8360_ping() */
}
/* Kernel pings watchdog */
static
void
sbc8360_ping
(
void
)
{
/* Write the base timer register */
outb
(
wd_margin
,
SBC8360_BASETIME
);
}
/* Userspace pings kernel driver, or requests clean close */
static
ssize_t
sbc8360_write
(
struct
file
*
file
,
const
char
__user
*
buf
,
size_t
count
,
loff_t
*
ppos
)
{
if
(
count
)
{
if
(
!
nowayout
)
{
size_t
i
;
/* In case it was set long ago */
expect_close
=
0
;
for
(
i
=
0
;
i
!=
count
;
i
++
)
{
char
c
;
if
(
get_user
(
c
,
buf
+
i
))
return
-
EFAULT
;
if
(
c
==
'V'
)
expect_close
=
42
;
}
}
sbc8360_ping
();
}
return
count
;
}
static
int
sbc8360_open
(
struct
inode
*
inode
,
struct
file
*
file
)
{
spin_lock
(
&
sbc8360_lock
);
if
(
test_and_set_bit
(
0
,
&
sbc8360_is_open
))
{
spin_unlock
(
&
sbc8360_lock
);
return
-
EBUSY
;
}
if
(
nowayout
)
__module_get
(
THIS_MODULE
);
/* Activate and ping once to start the countdown */
spin_unlock
(
&
sbc8360_lock
);
sbc8360_activate
();
sbc8360_ping
();
return
nonseekable_open
(
inode
,
file
);
}
static
int
sbc8360_close
(
struct
inode
*
inode
,
struct
file
*
file
)
{
spin_lock
(
&
sbc8360_lock
);
if
(
expect_close
==
42
)
outb
(
0
,
SBC8360_ENABLE
);
else
printk
(
KERN_CRIT
PFX
"SBC8360 device closed unexpectedly. SBC8360 will not stop!
\n
"
);
clear_bit
(
0
,
&
sbc8360_is_open
);
expect_close
=
0
;
spin_unlock
(
&
sbc8360_lock
);
return
0
;
}
/*
* Notifier for system down
*/
static
int
sbc8360_notify_sys
(
struct
notifier_block
*
this
,
unsigned
long
code
,
void
*
unused
)
{
if
(
code
==
SYS_DOWN
||
code
==
SYS_HALT
)
{
/* Disable the SBC8360 Watchdog */
outb
(
0
,
SBC8360_ENABLE
);
}
return
NOTIFY_DONE
;
}
/*
* Kernel Interfaces
*/
static
struct
file_operations
sbc8360_fops
=
{
.
owner
=
THIS_MODULE
,
.
llseek
=
no_llseek
,
.
write
=
sbc8360_write
,
.
open
=
sbc8360_open
,
.
release
=
sbc8360_close
,
};
static
struct
miscdevice
sbc8360_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"watchdog"
,
.
fops
=
&
sbc8360_fops
,
};
/*
* The SBC8360 needs to learn about soft shutdowns in order to
* turn the timebomb registers off.
*/
static
struct
notifier_block
sbc8360_notifier
=
{
.
notifier_call
=
sbc8360_notify_sys
,
};
static
int
__init
sbc8360_init
(
void
)
{
int
res
;
unsigned
long
int
mseconds
=
60000
;
spin_lock_init
(
&
sbc8360_lock
);
res
=
misc_register
(
&
sbc8360_miscdev
);
if
(
res
)
{
printk
(
KERN_ERR
PFX
"failed to register misc device
\n
"
);
goto
out_nomisc
;
}
if
(
!
request_region
(
SBC8360_ENABLE
,
1
,
"SBC8360"
))
{
printk
(
KERN_ERR
PFX
"ENABLE method I/O %X is not available.
\n
"
,
SBC8360_ENABLE
);
res
=
-
EIO
;
goto
out_noenablereg
;
}
if
(
!
request_region
(
SBC8360_BASETIME
,
1
,
"SBC8360"
))
{
printk
(
KERN_ERR
PFX
"BASETIME method I/O %X is not available.
\n
"
,
SBC8360_BASETIME
);
res
=
-
EIO
;
goto
out_nobasetimereg
;
}
res
=
register_reboot_notifier
(
&
sbc8360_notifier
);
if
(
res
)
{
printk
(
KERN_ERR
PFX
"Failed to register reboot notifier.
\n
"
);
goto
out_noreboot
;
}
if
(
timeout
<
0
||
timeout
>
63
)
{
printk
(
KERN_ERR
PFX
"Invalid timeout index (must be 0-63).
\n
"
);
res
=
-
EINVAL
;
goto
out_noreboot
;
}
wd_margin
=
wd_times
[
timeout
][
0
];
wd_multiplier
=
wd_times
[
timeout
][
1
];
if
(
wd_multiplier
==
1
)
mseconds
=
(
wd_margin
+
1
)
*
500
;
else
if
(
wd_multiplier
==
2
)
mseconds
=
(
wd_margin
+
1
)
*
5000
;
else
if
(
wd_multiplier
==
3
)
mseconds
=
(
wd_margin
+
1
)
*
50000
;
else
if
(
wd_multiplier
==
4
)
mseconds
=
(
wd_margin
+
1
)
*
100000
;
/* My kingdom for the ability to print "0.5 seconds" in the kernel! */
printk
(
KERN_INFO
PFX
"Timeout set at %ld ms.
\n
"
,
mseconds
);
return
0
;
out_noreboot:
release_region
(
SBC8360_ENABLE
,
1
);
release_region
(
SBC8360_BASETIME
,
1
);
out_noenablereg:
out_nobasetimereg:
misc_deregister
(
&
sbc8360_miscdev
);
out_nomisc:
return
res
;
}
static
void
__exit
sbc8360_exit
(
void
)
{
misc_deregister
(
&
sbc8360_miscdev
);
unregister_reboot_notifier
(
&
sbc8360_notifier
);
release_region
(
SBC8360_ENABLE
,
1
);
release_region
(
SBC8360_BASETIME
,
1
);
}
module_init
(
sbc8360_init
);
module_exit
(
sbc8360_exit
);
MODULE_AUTHOR
(
"Ian E. Morgan <imorgan@webcon.ca>"
);
MODULE_DESCRIPTION
(
"SBC8360 watchdog driver"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_VERSION
(
"1.0"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
/* end of sbc8360.c */
drivers/char/watchdog/w83977f_wdt.c
0 → 100644
View file @
a8cd2e50
/*
* W83977F Watchdog Timer Driver for Winbond W83977F I/O Chip
*
* (c) Copyright 2005 Jose Goncalves <jose.goncalves@inov.pt>
*
* Based on w83877f_wdt.c by Scott Jennings,
* and wdt977.c by Woody Suwalski
*
* -----------------------
*
* 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.
*
*/
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/miscdevice.h>
#include <linux/init.h>
#include <linux/ioport.h>
#include <linux/watchdog.h>
#include <linux/notifier.h>
#include <linux/reboot.h>
#include <asm/io.h>
#include <asm/system.h>
#include <asm/uaccess.h>
#define WATCHDOG_VERSION "1.00"
#define WATCHDOG_NAME "W83977F WDT"
#define PFX WATCHDOG_NAME ": "
#define DRIVER_VERSION WATCHDOG_NAME " driver, v" WATCHDOG_VERSION "\n"
#define IO_INDEX_PORT 0x3F0
#define IO_DATA_PORT (IO_INDEX_PORT+1)
#define UNLOCK_DATA 0x87
#define LOCK_DATA 0xAA
#define DEVICE_REGISTER 0x07
#define DEFAULT_TIMEOUT 45
/* default timeout in seconds */
static
int
timeout
=
DEFAULT_TIMEOUT
;
static
int
timeoutW
;
/* timeout in watchdog counter units */
static
unsigned
long
timer_alive
;
static
int
testmode
;
static
char
expect_close
;
static
spinlock_t
spinlock
;
module_param
(
timeout
,
int
,
0
);
MODULE_PARM_DESC
(
timeout
,
"Watchdog timeout in seconds (15..7635), default="
__MODULE_STRING
(
DEFAULT_TIMEOUT
)
")"
);
module_param
(
testmode
,
int
,
0
);
MODULE_PARM_DESC
(
testmode
,
"Watchdog testmode (1 = no reboot), default=0"
);
static
int
nowayout
=
WATCHDOG_NOWAYOUT
;
module_param
(
nowayout
,
int
,
0
);
MODULE_PARM_DESC
(
nowayout
,
"Watchdog cannot be stopped once started (default=CONFIG_WATCHDOG_NOWAYOUT)"
);
/*
* Start the watchdog
*/
static
int
wdt_start
(
void
)
{
unsigned
long
flags
;
spin_lock_irqsave
(
&
spinlock
,
flags
);
/* Unlock the SuperIO chip */
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
/*
* Select device Aux2 (device=8) to set watchdog regs F2, F3 and F4.
* F2 has the timeout in watchdog counter units.
* F3 is set to enable watchdog LED blink at timeout.
* F4 is used to just clear the TIMEOUT'ed state (bit 0).
*/
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
outb_p
(
0xF2
,
IO_INDEX_PORT
);
outb_p
(
timeoutW
,
IO_DATA_PORT
);
outb_p
(
0xF3
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
outb_p
(
0xF4
,
IO_INDEX_PORT
);
outb_p
(
0x00
,
IO_DATA_PORT
);
/* Set device Aux2 active */
outb_p
(
0x30
,
IO_INDEX_PORT
);
outb_p
(
0x01
,
IO_DATA_PORT
);
/*
* Select device Aux1 (dev=7) to set GP16 as the watchdog output
* (in reg E6) and GP13 as the watchdog LED output (in reg E3).
* Map GP16 at pin 119.
* In test mode watch the bit 0 on F4 to indicate "triggered" or
* check watchdog LED on SBC.
*/
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x07
,
IO_DATA_PORT
);
if
(
!
testmode
)
{
unsigned
pin_map
;
outb_p
(
0xE6
,
IO_INDEX_PORT
);
outb_p
(
0x0A
,
IO_DATA_PORT
);
outb_p
(
0x2C
,
IO_INDEX_PORT
);
pin_map
=
inb_p
(
IO_DATA_PORT
);
pin_map
|=
0x10
;
pin_map
&=
~
(
0x20
);
outb_p
(
0x2C
,
IO_INDEX_PORT
);
outb_p
(
pin_map
,
IO_DATA_PORT
);
}
outb_p
(
0xE3
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
/* Set device Aux1 active */
outb_p
(
0x30
,
IO_INDEX_PORT
);
outb_p
(
0x01
,
IO_DATA_PORT
);
/* Lock the SuperIO chip */
outb_p
(
LOCK_DATA
,
IO_INDEX_PORT
);
spin_unlock_irqrestore
(
&
spinlock
,
flags
);
printk
(
KERN_INFO
PFX
"activated.
\n
"
);
return
0
;
}
/*
* Stop the watchdog
*/
static
int
wdt_stop
(
void
)
{
unsigned
long
flags
;
spin_lock_irqsave
(
&
spinlock
,
flags
);
/* Unlock the SuperIO chip */
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
/*
* Select device Aux2 (device=8) to set watchdog regs F2, F3 and F4.
* F2 is reset to its default value (watchdog timer disabled).
* F3 is reset to its default state.
* F4 clears the TIMEOUT'ed state (bit 0) - back to default.
*/
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
outb_p
(
0xF2
,
IO_INDEX_PORT
);
outb_p
(
0xFF
,
IO_DATA_PORT
);
outb_p
(
0xF3
,
IO_INDEX_PORT
);
outb_p
(
0x00
,
IO_DATA_PORT
);
outb_p
(
0xF4
,
IO_INDEX_PORT
);
outb_p
(
0x00
,
IO_DATA_PORT
);
outb_p
(
0xF2
,
IO_INDEX_PORT
);
outb_p
(
0x00
,
IO_DATA_PORT
);
/*
* Select device Aux1 (dev=7) to set GP16 (in reg E6) and
* Gp13 (in reg E3) as inputs.
*/
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x07
,
IO_DATA_PORT
);
if
(
!
testmode
)
{
outb_p
(
0xE6
,
IO_INDEX_PORT
);
outb_p
(
0x01
,
IO_DATA_PORT
);
}
outb_p
(
0xE3
,
IO_INDEX_PORT
);
outb_p
(
0x01
,
IO_DATA_PORT
);
/* Lock the SuperIO chip */
outb_p
(
LOCK_DATA
,
IO_INDEX_PORT
);
spin_unlock_irqrestore
(
&
spinlock
,
flags
);
printk
(
KERN_INFO
PFX
"shutdown.
\n
"
);
return
0
;
}
/*
* Send a keepalive ping to the watchdog
* This is done by simply re-writing the timeout to reg. 0xF2
*/
static
int
wdt_keepalive
(
void
)
{
unsigned
long
flags
;
spin_lock_irqsave
(
&
spinlock
,
flags
);
/* Unlock the SuperIO chip */
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
/* Select device Aux2 (device=8) to kick watchdog reg F2 */
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
outb_p
(
0xF2
,
IO_INDEX_PORT
);
outb_p
(
timeoutW
,
IO_DATA_PORT
);
/* Lock the SuperIO chip */
outb_p
(
LOCK_DATA
,
IO_INDEX_PORT
);
spin_unlock_irqrestore
(
&
spinlock
,
flags
);
return
0
;
}
/*
* Set the watchdog timeout value
*/
static
int
wdt_set_timeout
(
int
t
)
{
int
tmrval
;
/*
* Convert seconds to watchdog counter time units, rounding up.
* On PCM-5335 watchdog units are 30 seconds/step with 15 sec startup
* value. This information is supplied in the PCM-5335 manual and was
* checked by me on a real board. This is a bit strange because W83977f
* datasheet says counter unit is in minutes!
*/
if
(
t
<
15
)
return
-
EINVAL
;
tmrval
=
((
t
+
15
)
+
29
)
/
30
;
if
(
tmrval
>
255
)
return
-
EINVAL
;
/*
* timeout is the timeout in seconds,
* timeoutW is the timeout in watchdog counter units.
*/
timeoutW
=
tmrval
;
timeout
=
(
timeoutW
*
30
)
-
15
;
return
0
;
}
/*
* Get the watchdog status
*/
static
int
wdt_get_status
(
int
*
status
)
{
int
new_status
;
unsigned
long
flags
;
spin_lock_irqsave
(
&
spinlock
,
flags
);
/* Unlock the SuperIO chip */
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
outb_p
(
UNLOCK_DATA
,
IO_INDEX_PORT
);
/* Select device Aux2 (device=8) to read watchdog reg F4 */
outb_p
(
DEVICE_REGISTER
,
IO_INDEX_PORT
);
outb_p
(
0x08
,
IO_DATA_PORT
);
outb_p
(
0xF4
,
IO_INDEX_PORT
);
new_status
=
inb_p
(
IO_DATA_PORT
);
/* Lock the SuperIO chip */
outb_p
(
LOCK_DATA
,
IO_INDEX_PORT
);
spin_unlock_irqrestore
(
&
spinlock
,
flags
);
*
status
=
0
;
if
(
new_status
&
1
)
*
status
|=
WDIOF_CARDRESET
;
return
0
;
}
/*
* /dev/watchdog handling
*/
static
int
wdt_open
(
struct
inode
*
inode
,
struct
file
*
file
)
{
/* If the watchdog is alive we don't need to start it again */
if
(
test_and_set_bit
(
0
,
&
timer_alive
)
)
return
-
EBUSY
;
if
(
nowayout
)
__module_get
(
THIS_MODULE
);
wdt_start
();
return
nonseekable_open
(
inode
,
file
);
}
static
int
wdt_release
(
struct
inode
*
inode
,
struct
file
*
file
)
{
/*
* Shut off the timer.
* Lock it in if it's a module and we set nowayout
*/
if
(
expect_close
==
42
)
{
wdt_stop
();
clear_bit
(
0
,
&
timer_alive
);
}
else
{
wdt_keepalive
();
printk
(
KERN_CRIT
PFX
"unexpected close, not stopping watchdog!
\n
"
);
}
expect_close
=
0
;
return
0
;
}
/*
* wdt_write:
* @file: file handle to the watchdog
* @buf: buffer to write (unused as data does not matter here
* @count: count of bytes
* @ppos: pointer to the position to write. No seeks allowed
*
* A write to a watchdog device is defined as a keepalive signal. Any
* write of data will do, as we we don't define content meaning.
*/
static
ssize_t
wdt_write
(
struct
file
*
file
,
const
char
__user
*
buf
,
size_t
count
,
loff_t
*
ppos
)
{
/* See if we got the magic character 'V' and reload the timer */
if
(
count
)
{
if
(
!
nowayout
)
{
size_t
ofs
;
/* note: just in case someone wrote the magic character long ago */
expect_close
=
0
;
/* scan to see whether or not we got the magic character */
for
(
ofs
=
0
;
ofs
!=
count
;
ofs
++
)
{
char
c
;
if
(
get_user
(
c
,
buf
+
ofs
))
return
-
EFAULT
;
if
(
c
==
'V'
)
{
expect_close
=
42
;
}
}
}
/* someone wrote to us, we should restart timer */
wdt_keepalive
();
}
return
count
;
}
/*
* wdt_ioctl:
* @inode: inode of the device
* @file: file handle to the device
* @cmd: watchdog command
* @arg: argument pointer
*
* The watchdog API defines a common set of functions for all watchdogs
* according to their available features.
*/
static
struct
watchdog_info
ident
=
{
.
options
=
WDIOF_SETTIMEOUT
|
WDIOF_MAGICCLOSE
|
WDIOF_KEEPALIVEPING
,
.
firmware_version
=
1
,
.
identity
=
WATCHDOG_NAME
,
};
static
int
wdt_ioctl
(
struct
inode
*
inode
,
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
int
status
;
int
new_options
,
retval
=
-
EINVAL
;
int
new_timeout
;
union
{
struct
watchdog_info
__user
*
ident
;
int
__user
*
i
;
}
uarg
;
uarg
.
i
=
(
int
__user
*
)
arg
;
switch
(
cmd
)
{
default:
return
-
ENOIOCTLCMD
;
case
WDIOC_GETSUPPORT
:
return
copy_to_user
(
uarg
.
ident
,
&
ident
,
sizeof
(
ident
))
?
-
EFAULT
:
0
;
case
WDIOC_GETSTATUS
:
wdt_get_status
(
&
status
);
return
put_user
(
status
,
uarg
.
i
);
case
WDIOC_GETBOOTSTATUS
:
return
put_user
(
0
,
uarg
.
i
);
case
WDIOC_KEEPALIVE
:
wdt_keepalive
();
return
0
;
case
WDIOC_SETOPTIONS
:
if
(
get_user
(
new_options
,
uarg
.
i
))
return
-
EFAULT
;
if
(
new_options
&
WDIOS_DISABLECARD
)
{
wdt_stop
();
retval
=
0
;
}
if
(
new_options
&
WDIOS_ENABLECARD
)
{
wdt_start
();
retval
=
0
;
}
return
retval
;
case
WDIOC_SETTIMEOUT
:
if
(
get_user
(
new_timeout
,
uarg
.
i
))
return
-
EFAULT
;
if
(
wdt_set_timeout
(
new_timeout
))
return
-
EINVAL
;
wdt_keepalive
();
/* Fall */
case
WDIOC_GETTIMEOUT
:
return
put_user
(
timeout
,
uarg
.
i
);
}
}
static
int
wdt_notify_sys
(
struct
notifier_block
*
this
,
unsigned
long
code
,
void
*
unused
)
{
if
(
code
==
SYS_DOWN
||
code
==
SYS_HALT
)
wdt_stop
();
return
NOTIFY_DONE
;
}
static
struct
file_operations
wdt_fops
=
{
.
owner
=
THIS_MODULE
,
.
llseek
=
no_llseek
,
.
write
=
wdt_write
,
.
ioctl
=
wdt_ioctl
,
.
open
=
wdt_open
,
.
release
=
wdt_release
,
};
static
struct
miscdevice
wdt_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"watchdog"
,
.
fops
=
&
wdt_fops
,
};
static
struct
notifier_block
wdt_notifier
=
{
.
notifier_call
=
wdt_notify_sys
,
};
static
int
__init
w83977f_wdt_init
(
void
)
{
int
rc
;
printk
(
KERN_INFO
PFX
DRIVER_VERSION
);
spin_lock_init
(
&
spinlock
);
/*
* Check that the timeout value is within it's range ;
* if not reset to the default
*/
if
(
wdt_set_timeout
(
timeout
))
{
wdt_set_timeout
(
DEFAULT_TIMEOUT
);
printk
(
KERN_INFO
PFX
"timeout value must be 15<=timeout<=7635, using %d
\n
"
,
DEFAULT_TIMEOUT
);
}
if
(
!
request_region
(
IO_INDEX_PORT
,
2
,
WATCHDOG_NAME
))
{
printk
(
KERN_ERR
PFX
"I/O address 0x%04x already in use
\n
"
,
IO_INDEX_PORT
);
rc
=
-
EIO
;
goto
err_out
;
}
rc
=
misc_register
(
&
wdt_miscdev
);
if
(
rc
)
{
printk
(
KERN_ERR
PFX
"cannot register miscdev on minor=%d (err=%d)
\n
"
,
wdt_miscdev
.
minor
,
rc
);
goto
err_out_region
;
}
rc
=
register_reboot_notifier
(
&
wdt_notifier
);
if
(
rc
)
{
printk
(
KERN_ERR
PFX
"cannot register reboot notifier (err=%d)
\n
"
,
rc
);
goto
err_out_miscdev
;
}
printk
(
KERN_INFO
PFX
"initialized. timeout=%d sec (nowayout=%d testmode=%d)
\n
"
,
timeout
,
nowayout
,
testmode
);
return
0
;
err_out_miscdev:
misc_deregister
(
&
wdt_miscdev
);
err_out_region:
release_region
(
IO_INDEX_PORT
,
2
);
err_out:
return
rc
;
}
static
void
__exit
w83977f_wdt_exit
(
void
)
{
wdt_stop
();
misc_deregister
(
&
wdt_miscdev
);
unregister_reboot_notifier
(
&
wdt_notifier
);
release_region
(
IO_INDEX_PORT
,
2
);
}
module_init
(
w83977f_wdt_init
);
module_exit
(
w83977f_wdt_exit
);
MODULE_AUTHOR
(
"Jose Goncalves <jose.goncalves@inov.pt>"
);
MODULE_DESCRIPTION
(
"Driver for watchdog timer in W83977F I/O chip"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
include/asm-ppc/mv64x60.h
View file @
a8cd2e50
...
...
@@ -119,6 +119,14 @@ extern spinlock_t mv64x60_lock;
#define MV64x60_64BIT_WIN_COUNT 24
/* Watchdog Platform Device, Driver Data */
#define MV64x60_WDT_NAME "wdt"
struct
mv64x60_wdt_pdata
{
int
timeout
;
/* watchdog expiry in seconds, default 10 */
int
bus_clk
;
/* bus clock in MHz, default 133 */
};
/*
* Define a structure that's used to pass in config information to the
* core routines.
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment