2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
2006-01-08 12:00:31 +03:00
* Wdt977 0.04 : A Watchdog Device for Netwinder W83977AF chip
2005-04-17 02:20:36 +04:00
*
* ( c ) Copyright 1998 Rebel . com ( Woody Suwalski < woody @ netwinder . org > )
*
* - - - - - - - - - - - - - - - - - - - - - - -
*
* - - - - - - - - - - - - - - - - - - - - - - -
* 14 - Dec - 2001 Matt Domsch < Matt_Domsch @ dell . com >
* Added nowayout module option to override CONFIG_WATCHDOG_NOWAYOUT
* 19 - Dec - 2001 Woody Suwalski : Netwinder fixes , ioctl interface
* 06 - Jan - 2002 Woody Suwalski : For compatibility , convert all timeouts
* from minutes to seconds .
* 07 - Jul - 2003 Daniele Bellucci : Audit return code of misc_register in
* nwwatchdog_init .
2006-01-08 12:00:31 +03:00
* 25 - Oct - 2005 Woody Suwalski : Convert addresses to # defs , add spinlocks
2008-05-19 17:09:57 +04:00
* remove limitiation to be used on
* Netwinders only
2005-04-17 02:20:36 +04:00
*/
2012-02-16 03:06:19 +04:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/fs.h>
# include <linux/miscdevice.h>
# include <linux/init.h>
2006-01-08 12:00:31 +03:00
# include <linux/ioport.h>
2005-04-17 02:20:36 +04:00
# include <linux/watchdog.h>
# include <linux/notifier.h>
# include <linux/reboot.h>
2008-05-19 17:09:57 +04:00
# include <linux/io.h>
# include <linux/uaccess.h>
2005-04-17 02:20:36 +04:00
# include <asm/mach-types.h>
2006-01-08 12:00:31 +03:00
# define WATCHDOG_VERSION "0.04"
# define WATCHDOG_NAME "Wdt977"
# define IO_INDEX_PORT 0x370 /* on some systems it can be 0x3F0 */
2008-05-19 17:09:57 +04:00
# define IO_DATA_PORT (IO_INDEX_PORT + 1)
2006-01-08 12:00:31 +03:00
# define UNLOCK_DATA 0x87
# define LOCK_DATA 0xAA
# define DEVICE_REGISTER 0x07
2005-04-17 02:20:36 +04:00
# define DEFAULT_TIMEOUT 60 /* default timeout in seconds */
static int timeout = DEFAULT_TIMEOUT ;
static int timeoutM ; /* timeout in minutes */
static unsigned long timer_alive ;
static int testmode ;
static char expect_close ;
2007-11-02 02:27:08 +03:00
static DEFINE_SPINLOCK ( spinlock ) ;
2005-04-17 02:20:36 +04:00
module_param ( timeout , int , 0 ) ;
2010-05-01 20:46:15 +04:00
MODULE_PARM_DESC ( timeout , " Watchdog timeout in seconds (60..15300, default= "
2008-05-19 17:09:57 +04:00
__MODULE_STRING ( DEFAULT_TIMEOUT ) " ) " ) ;
2005-04-17 02:20:36 +04:00
module_param ( testmode , int , 0 ) ;
2008-05-19 17:09:57 +04:00
MODULE_PARM_DESC ( testmode , " Watchdog testmode (1 = no reboot), default=0 " ) ;
2005-04-17 02:20:36 +04:00
2012-03-05 19:51:11 +04:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
module_param ( nowayout , bool , 0 ) ;
2008-05-19 17:09:57 +04:00
MODULE_PARM_DESC ( nowayout ,
" Watchdog cannot be stopped once started (default= "
__MODULE_STRING ( WATCHDOG_NOWAYOUT ) " ) " ) ;
2005-04-17 02:20:36 +04:00
/*
* Start the watchdog
*/
static int wdt977_start ( void )
{
2006-01-08 12:00:31 +03:00
unsigned long flags ;
spin_lock_irqsave ( & spinlock , flags ) ;
2005-04-17 02:20:36 +04:00
/* unlock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
/* select device Aux2 (device=8) and set watchdog regs F2, F3 and F4
* F2 has the timeout in minutes
* F3 could be set to the POWER LED blink ( with GP17 set to PowerLed )
* at timeout , and to reset timer on kbd / mouse activity ( not impl . )
* F4 is used to just clear the TIMEOUT ' ed state ( bit 0 )
*/
2006-01-08 12:00:31 +03:00
outb_p ( DEVICE_REGISTER , IO_INDEX_PORT ) ;
outb_p ( 0x08 , IO_DATA_PORT ) ;
outb_p ( 0xF2 , IO_INDEX_PORT ) ;
outb_p ( timeoutM , IO_DATA_PORT ) ;
outb_p ( 0xF3 , IO_INDEX_PORT ) ;
2008-05-19 17:09:57 +04:00
outb_p ( 0x00 , IO_DATA_PORT ) ; /* another setting is 0E for
kbd / mouse / LED */
2006-01-08 12:00:31 +03:00
outb_p ( 0xF4 , IO_INDEX_PORT ) ;
outb_p ( 0x00 , IO_DATA_PORT ) ;
2005-04-17 02:20:36 +04:00
2008-05-19 17:09:57 +04:00
/* At last select device Aux1 (dev=7) and set GP16 as a
* watchdog output . In test mode watch the bit 1 on F4 to
* indicate " triggered "
*/
if ( ! testmode ) {
2006-01-08 12:00:31 +03:00
outb_p ( DEVICE_REGISTER , IO_INDEX_PORT ) ;
outb_p ( 0x07 , IO_DATA_PORT ) ;
outb_p ( 0xE6 , IO_INDEX_PORT ) ;
outb_p ( 0x08 , IO_DATA_PORT ) ;
2005-04-17 02:20:36 +04:00
}
/* lock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( LOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:00:31 +03:00
spin_unlock_irqrestore ( & spinlock , flags ) ;
2012-02-16 03:06:19 +04:00
pr_info ( " activated \n " ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/*
* Stop the watchdog
*/
static int wdt977_stop ( void )
{
2006-01-08 12:00:31 +03:00
unsigned long flags ;
spin_lock_irqsave ( & spinlock , flags ) ;
2005-04-17 02:20:36 +04:00
/* unlock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
/* select device Aux2 (device=8) and set watchdog regs F2,F3 and F4
* F3 is reset to its default state
* F4 can clear the TIMEOUT ' ed state ( bit 0 ) - back to default
* We can not use GP17 as a PowerLed , as we use its usage as a RedLed
*/
2006-01-08 12:00:31 +03:00
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 ) ;
2005-04-17 02:20:36 +04:00
2008-05-19 17:09:57 +04:00
/* at last select device Aux1 (dev=7) and set
GP16 as a watchdog output */
2006-01-08 12:00:31 +03:00
outb_p ( DEVICE_REGISTER , IO_INDEX_PORT ) ;
outb_p ( 0x07 , IO_DATA_PORT ) ;
outb_p ( 0xE6 , IO_INDEX_PORT ) ;
outb_p ( 0x08 , IO_DATA_PORT ) ;
2005-04-17 02:20:36 +04:00
/* lock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( LOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:00:31 +03:00
spin_unlock_irqrestore ( & spinlock , flags ) ;
2012-02-16 03:06:19 +04:00
pr_info ( " shutdown \n " ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/*
* Send a keepalive ping to the watchdog
* This is done by simply re - writing the timeout to reg . 0xF2
*/
static int wdt977_keepalive ( void )
{
2006-01-08 12:00:31 +03:00
unsigned long flags ;
spin_lock_irqsave ( & spinlock , flags ) ;
2005-04-17 02:20:36 +04:00
/* unlock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
/* select device Aux2 (device=8) and kicks watchdog reg F2 */
/* F2 has the timeout in minutes */
2006-01-08 12:00:31 +03:00
outb_p ( DEVICE_REGISTER , IO_INDEX_PORT ) ;
outb_p ( 0x08 , IO_DATA_PORT ) ;
outb_p ( 0xF2 , IO_INDEX_PORT ) ;
outb_p ( timeoutM , IO_DATA_PORT ) ;
2005-04-17 02:20:36 +04:00
/* lock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( LOCK_DATA , IO_INDEX_PORT ) ;
spin_unlock_irqrestore ( & spinlock , flags ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/*
* Set the watchdog timeout value
*/
static int wdt977_set_timeout ( int t )
{
int tmrval ;
/* convert seconds to minutes, rounding up */
tmrval = ( t + 59 ) / 60 ;
if ( machine_is_netwinder ( ) ) {
2008-05-19 17:09:57 +04:00
/* we have a hw bug somewhere, so each 977 minute is actually
* only 30 sec . This limits the max timeout to half of device
* max of 255 minutes . . .
2005-04-17 02:20:36 +04:00
*/
tmrval + = tmrval ;
}
2008-05-19 17:09:57 +04:00
if ( tmrval < 1 | | tmrval > 255 )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
2008-05-19 17:09:57 +04:00
/* timeout is the timeout in seconds, timeoutM is
the timeout in minutes ) */
2005-04-17 02:20:36 +04:00
timeout = t ;
timeoutM = tmrval ;
return 0 ;
}
/*
* Get the watchdog status
*/
static int wdt977_get_status ( int * status )
{
int new_status ;
2006-01-08 12:00:31 +03:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:00:31 +03:00
spin_lock_irqsave ( & spinlock , flags ) ;
2005-04-17 02:20:36 +04:00
/* unlock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
outb_p ( UNLOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
/* select device Aux2 (device=8) and read watchdog reg F4 */
2006-01-08 12:00:31 +03:00
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 ) ;
2005-04-17 02:20:36 +04:00
/* lock the SuperIO chip */
2006-01-08 12:00:31 +03:00
outb_p ( LOCK_DATA , IO_INDEX_PORT ) ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:00:31 +03:00
spin_unlock_irqrestore ( & spinlock , flags ) ;
2008-05-19 17:09:57 +04:00
* status = 0 ;
2005-04-17 02:20:36 +04:00
if ( new_status & 1 )
* status | = WDIOF_CARDRESET ;
return 0 ;
}
/*
* / dev / watchdog handling
*/
static int wdt977_open ( struct inode * inode , struct file * file )
{
/* If the watchdog is alive we don't need to start it again */
2008-05-19 17:09:57 +04:00
if ( test_and_set_bit ( 0 , & timer_alive ) )
2005-04-17 02:20:36 +04:00
return - EBUSY ;
if ( nowayout )
__module_get ( THIS_MODULE ) ;
wdt977_start ( ) ;
2019-03-26 23:51:19 +03:00
return stream_open ( inode , file ) ;
2005-04-17 02:20:36 +04:00
}
static int wdt977_release ( struct inode * inode , struct file * file )
{
/*
* Shut off the timer .
2011-02-23 23:04:38 +03:00
* Lock it in if it ' s a module and we set nowayout
2005-04-17 02:20:36 +04:00
*/
2008-05-19 17:09:57 +04:00
if ( expect_close = = 42 ) {
2005-04-17 02:20:36 +04:00
wdt977_stop ( ) ;
2008-05-19 17:09:57 +04:00
clear_bit ( 0 , & timer_alive ) ;
2005-04-17 02:20:36 +04:00
} else {
wdt977_keepalive ( ) ;
2012-02-16 03:06:19 +04:00
pr_crit ( " Unexpected close, not stopping watchdog! \n " ) ;
2005-04-17 02:20:36 +04:00
}
expect_close = 0 ;
return 0 ;
}
/*
* wdt977_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 wdt977_write ( struct file * file , const char __user * buf ,
size_t count , loff_t * ppos )
{
2008-05-19 17:09:57 +04:00
if ( count ) {
if ( ! nowayout ) {
2005-04-17 02:20:36 +04:00
size_t i ;
/* In case it was set long ago */
expect_close = 0 ;
2008-05-19 17:09:57 +04:00
for ( i = 0 ; i ! = count ; i + + ) {
2005-04-17 02:20:36 +04:00
char c ;
if ( get_user ( c , buf + i ) )
return - EFAULT ;
if ( c = = ' V ' )
expect_close = 42 ;
}
}
2006-01-08 12:00:31 +03:00
/* someone wrote to us, we should restart timer */
2005-04-17 02:20:36 +04:00
wdt977_keepalive ( ) ;
}
return count ;
}
2008-05-19 17:09:57 +04:00
static const struct watchdog_info ident = {
. options = WDIOF_SETTIMEOUT |
WDIOF_MAGICCLOSE |
WDIOF_KEEPALIVEPING ,
. firmware_version = 1 ,
. identity = WATCHDOG_NAME ,
} ;
2005-04-17 02:20:36 +04:00
/*
* wdt977_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 .
*/
2008-05-19 17:09:57 +04:00
static long wdt977_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
2005-04-17 02:20:36 +04:00
{
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 ;
2008-05-19 17:09:57 +04:00
switch ( cmd ) {
2005-04-17 02:20:36 +04:00
case WDIOC_GETSUPPORT :
return copy_to_user ( uarg . ident , & ident ,
sizeof ( ident ) ) ? - EFAULT : 0 ;
case WDIOC_GETSTATUS :
wdt977_get_status ( & status ) ;
return put_user ( status , uarg . i ) ;
case WDIOC_GETBOOTSTATUS :
return put_user ( 0 , uarg . i ) ;
case WDIOC_SETOPTIONS :
2008-05-19 17:09:57 +04:00
if ( get_user ( new_options , uarg . i ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
if ( new_options & WDIOS_DISABLECARD ) {
wdt977_stop ( ) ;
retval = 0 ;
}
if ( new_options & WDIOS_ENABLECARD ) {
wdt977_start ( ) ;
retval = 0 ;
}
return retval ;
2008-07-18 15:41:17 +04:00
case WDIOC_KEEPALIVE :
wdt977_keepalive ( ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
case WDIOC_SETTIMEOUT :
if ( get_user ( new_timeout , uarg . i ) )
return - EFAULT ;
if ( wdt977_set_timeout ( new_timeout ) )
2009-03-18 11:35:09 +03:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
wdt977_keepalive ( ) ;
2020-07-07 20:11:21 +03:00
fallthrough ;
2005-04-17 02:20:36 +04:00
case WDIOC_GETTIMEOUT :
return put_user ( timeout , uarg . i ) ;
2008-07-18 15:41:17 +04:00
default :
return - ENOTTY ;
2005-04-17 02:20:36 +04:00
}
}
static int wdt977_notify_sys ( struct notifier_block * this , unsigned long code ,
void * unused )
{
2008-05-19 17:09:57 +04:00
if ( code = = SYS_DOWN | | code = = SYS_HALT )
2005-04-17 02:20:36 +04:00
wdt977_stop ( ) ;
return NOTIFY_DONE ;
}
2008-05-19 17:09:57 +04:00
static const struct file_operations wdt977_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = wdt977_write ,
2008-05-19 17:09:57 +04:00
. unlocked_ioctl = wdt977_ioctl ,
2019-06-03 15:23:09 +03:00
. compat_ioctl = compat_ptr_ioctl ,
2005-04-17 02:20:36 +04:00
. open = wdt977_open ,
. release = wdt977_release ,
} ;
2008-05-19 17:09:57 +04:00
static struct miscdevice wdt977_miscdev = {
2005-04-17 02:20:36 +04:00
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & wdt977_fops ,
} ;
static struct notifier_block wdt977_notifier = {
. notifier_call = wdt977_notify_sys ,
} ;
2006-01-08 12:00:31 +03:00
static int __init wd977_init ( void )
2005-04-17 02:20:36 +04:00
{
2006-01-08 12:00:31 +03:00
int rc ;
2012-02-16 03:06:19 +04:00
pr_info ( " driver v%s \n " , WATCHDOG_VERSION ) ;
2006-01-08 12:00:31 +03:00
2008-05-19 17:09:57 +04:00
/* Check that the timeout value is within its range;
if not reset to the default */
if ( wdt977_set_timeout ( timeout ) ) {
2005-04-17 02:20:36 +04:00
wdt977_set_timeout ( DEFAULT_TIMEOUT ) ;
2012-02-16 03:06:19 +04:00
pr_info ( " timeout value must be 60 < timeout < 15300, using %d \n " ,
DEFAULT_TIMEOUT ) ;
2005-04-17 02:20:36 +04:00
}
2006-01-08 12:00:31 +03:00
/* on Netwinder the IOports are already reserved by
* arch / arm / mach - footbridge / netwinder - hw . c
*/
2008-05-19 17:09:57 +04:00
if ( ! machine_is_netwinder ( ) ) {
if ( ! request_region ( IO_INDEX_PORT , 2 , WATCHDOG_NAME ) ) {
2012-02-16 03:06:19 +04:00
pr_err ( " I/O address 0x%04x already in use \n " ,
IO_INDEX_PORT ) ;
2006-01-08 12:00:31 +03:00
rc = - EIO ;
goto err_out ;
}
2005-04-17 02:20:36 +04:00
}
2007-12-26 23:32:51 +03:00
rc = register_reboot_notifier ( & wdt977_notifier ) ;
2008-05-19 17:09:57 +04:00
if ( rc ) {
2012-02-16 03:06:19 +04:00
pr_err ( " cannot register reboot notifier (err=%d) \n " , rc ) ;
2006-01-08 12:00:31 +03:00
goto err_out_region ;
}
2007-12-26 23:32:51 +03:00
rc = misc_register ( & wdt977_miscdev ) ;
2008-05-19 17:09:57 +04:00
if ( rc ) {
2012-02-16 03:06:19 +04:00
pr_err ( " cannot register miscdev on minor=%d (err=%d) \n " ,
wdt977_miscdev . minor , rc ) ;
2007-12-26 23:32:51 +03:00
goto err_out_reboot ;
2005-04-17 02:20:36 +04:00
}
2012-02-16 03:06:19 +04:00
pr_info ( " initialized. timeout=%d sec (nowayout=%d, testmode=%i) \n " ,
timeout , nowayout , testmode ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2006-01-08 12:00:31 +03:00
2007-12-26 23:32:51 +03:00
err_out_reboot :
unregister_reboot_notifier ( & wdt977_notifier ) ;
2006-01-08 12:00:31 +03:00
err_out_region :
if ( ! machine_is_netwinder ( ) )
2008-05-19 17:09:57 +04:00
release_region ( IO_INDEX_PORT , 2 ) ;
2006-01-08 12:00:31 +03:00
err_out :
return rc ;
2005-04-17 02:20:36 +04:00
}
2006-01-08 12:00:31 +03:00
static void __exit wd977_exit ( void )
2005-04-17 02:20:36 +04:00
{
2006-01-08 12:00:31 +03:00
wdt977_stop ( ) ;
2005-04-17 02:20:36 +04:00
misc_deregister ( & wdt977_miscdev ) ;
unregister_reboot_notifier ( & wdt977_notifier ) ;
2008-05-19 17:09:57 +04:00
release_region ( IO_INDEX_PORT , 2 ) ;
2005-04-17 02:20:36 +04:00
}
2006-01-08 12:00:31 +03:00
module_init ( wd977_init ) ;
module_exit ( wd977_exit ) ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:00:31 +03:00
MODULE_AUTHOR ( " Woody Suwalski <woodys@xandros.com> " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " W83977AF Watchdog driver " ) ;
MODULE_LICENSE ( " GPL " ) ;