2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
* 60 xx Single Board Computer Watchdog Timer driver for Linux 2.2 . x
*
2009-03-18 11:35:09 +03:00
* Based on acquirewdt . c by Alan Cox .
2005-04-17 02:20:36 +04:00
*
* The author does NOT admit liability nor provide warranty for
* any of this software . This material is provided " AS-IS " in
* the hope that it may be useful for others .
*
* ( c ) Copyright 2000 Jakob Oestergaard < jakob @ unthought . net >
*
* 12 / 4 - 2000 [ Initial revision ]
* 25 / 4 - 2000 Added / dev / watchdog support
2008-05-19 17:08:11 +04:00
* 09 / 5 - 2001 [ smj @ oro . net ] fixed fop_write to " return 1 "
* on success
2005-04-17 02:20:36 +04:00
* 12 / 4 - 2002 [ rob @ osinvestor . com ] eliminate fop_read
* fix possible wdt_is_open race
* add CONFIG_WATCHDOG_NOWAYOUT support
* remove lock_kernel / unlock_kernel pairs
* added KERN_ * to printk ' s
* got rid of extraneous comments
2008-05-19 17:08:11 +04:00
* changed watchdog_info to correctly reflect what
* the driver offers
* added WDIOC_GETSTATUS , WDIOC_GETBOOTSTATUS ,
* WDIOC_SETTIMEOUT , WDIOC_GETTIMEOUT , and
* WDIOC_SETOPTIONS ioctls
2005-04-17 02:20:36 +04:00
* 09 / 8 - 2003 [ wim @ iguana . be ] cleanup of trailing spaces
* use module_param
2008-05-19 17:08:11 +04:00
* made timeout ( the emulated heartbeat ) a
* module_param
2005-04-17 02:20:36 +04:00
* made the keepalive ping an internal subroutine
* made wdt_stop and wdt_start module params
* added extra printk ' s for startup problems
* added MODULE_AUTHOR and MODULE_DESCRIPTION info
*
* This WDT driver is different from the other Linux WDT
* drivers in the following ways :
* * ) The driver will ping the watchdog by itself , because this
* particular WDT has a very short timeout ( one second ) and it
* would be insane to count on any userspace daemon always
* getting scheduled within that time frame .
*/
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/timer.h>
# include <linux/jiffies.h>
# include <linux/miscdevice.h>
# include <linux/watchdog.h>
# include <linux/fs.h>
# include <linux/ioport.h>
# include <linux/notifier.h>
# include <linux/reboot.h>
# include <linux/init.h>
2008-05-19 17:08:11 +04:00
# include <linux/io.h>
# include <linux/uaccess.h>
2005-04-17 02:20:36 +04:00
# define OUR_NAME "sbc60xxwdt"
# define PFX OUR_NAME ": "
/*
* You must set these - The driver cannot probe for the settings
*/
static int wdt_stop = 0x45 ;
module_param ( wdt_stop , int , 0 ) ;
MODULE_PARM_DESC ( wdt_stop , " SBC60xx WDT 'stop' io port (default 0x45) " ) ;
static int wdt_start = 0x443 ;
module_param ( wdt_start , int , 0 ) ;
MODULE_PARM_DESC ( wdt_start , " SBC60xx WDT 'start' io port (default 0x443) " ) ;
/*
* The 60 xx board can use watchdog timeout values from one second
* to several minutes . The default is one second , so if we reset
* the watchdog every ~ 250 ms we should be safe .
*/
# define WDT_INTERVAL (HZ / 4+1)
/*
* We must not require too good response from the userspace daemon .
* Here we require the userspace daemon to send us a heartbeat
* char to / dev / watchdog every 30 seconds .
* If the daemon pulses us every 25 seconds , we can still afford
* a 5 second scheduling delay on the ( high priority ) daemon . That
* should be sufficient for a box under any load .
*/
# define WATCHDOG_TIMEOUT 30 /* 30 sec default timeout */
2008-05-19 17:08:11 +04:00
static int timeout = WATCHDOG_TIMEOUT ; /* in seconds, multiplied by HZ to
get seconds to wait for a ping */
2005-04-17 02:20:36 +04:00
module_param ( timeout , int , 0 ) ;
2008-05-19 17:08:11 +04:00
MODULE_PARM_DESC ( timeout ,
" Watchdog timeout in seconds. (1<=timeout<=3600, default= "
__MODULE_STRING ( WATCHDOG_TIMEOUT ) " ) " ) ;
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:08:11 +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
2017-08-28 21:28:21 +03:00
static void wdt_timer_ping ( struct timer_list * ) ;
2017-10-05 02:27:04 +03:00
static DEFINE_TIMER ( timer , wdt_timer_ping ) ;
2005-04-17 02:20:36 +04:00
static unsigned long next_heartbeat ;
static unsigned long wdt_is_open ;
static char wdt_expect_close ;
/*
* Whack the dog
*/
2017-08-28 21:28:21 +03:00
static void wdt_timer_ping ( struct timer_list * unused )
2005-04-17 02:20:36 +04:00
{
/* If we got a heartbeat pulse within the WDT_US_INTERVAL
* we agree to ping the WDT
*/
2008-05-19 17:08:11 +04:00
if ( time_before ( jiffies , next_heartbeat ) ) {
2005-04-17 02:20:36 +04:00
/* Ping the WDT by reading from wdt_start */
inb_p ( wdt_start ) ;
/* Re-set the timer interval */
2007-02-08 20:39:36 +03:00
mod_timer ( & timer , jiffies + WDT_INTERVAL ) ;
2008-05-19 17:08:11 +04:00
} else
2012-02-16 03:06:19 +04:00
pr_warn ( " Heartbeat lost! Will not ping the watchdog \n " ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Utility routines
*/
static void wdt_startup ( void )
{
next_heartbeat = jiffies + ( timeout * HZ ) ;
/* Start the timer */
2007-02-08 20:39:36 +03:00
mod_timer ( & timer , jiffies + WDT_INTERVAL ) ;
2012-02-16 03:06:19 +04:00
pr_info ( " Watchdog timer is now enabled \n " ) ;
2005-04-17 02:20:36 +04:00
}
static void wdt_turnoff ( void )
{
/* Stop the timer */
del_timer ( & timer ) ;
inb_p ( wdt_stop ) ;
2012-02-16 03:06:19 +04:00
pr_info ( " Watchdog timer is now disabled... \n " ) ;
2005-04-17 02:20:36 +04:00
}
static void wdt_keepalive ( void )
{
/* user land ping */
next_heartbeat = jiffies + ( timeout * HZ ) ;
}
/*
* / dev / watchdog handling
*/
2008-05-19 17:08:11 +04:00
static ssize_t fop_write ( struct file * file , const char __user * buf ,
size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
/* See if we got the magic character 'V' and reload the timer */
2008-05-19 17:08:11 +04:00
if ( count ) {
if ( ! nowayout ) {
2005-04-17 02:20:36 +04:00
size_t ofs ;
2008-05-19 17:08:11 +04:00
/* note: just in case someone wrote the
magic character five months ago . . . */
2005-04-17 02:20:36 +04:00
wdt_expect_close = 0 ;
2008-05-19 17:08:11 +04:00
/* scan to see whether or not we got the
magic character */
for ( ofs = 0 ; ofs ! = count ; ofs + + ) {
2005-04-17 02:20:36 +04:00
char c ;
2008-08-07 00:19:41 +04:00
if ( get_user ( c , buf + ofs ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
2008-05-19 17:08:11 +04:00
if ( c = = ' V ' )
2005-04-17 02:20:36 +04:00
wdt_expect_close = 42 ;
}
}
2008-05-19 17:08:11 +04:00
/* Well, anyhow someone wrote to us, we should
return that favour */
2005-04-17 02:20:36 +04:00
wdt_keepalive ( ) ;
}
return count ;
}
2008-05-19 17:08:11 +04:00
static int fop_open ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
/* Just in case we're already talking to someone... */
2008-05-19 17:08:11 +04:00
if ( test_and_set_bit ( 0 , & wdt_is_open ) )
2005-04-17 02:20:36 +04:00
return - EBUSY ;
if ( nowayout )
__module_get ( THIS_MODULE ) ;
/* Good, fire up the show */
wdt_startup ( ) ;
2019-03-26 23:51:19 +03:00
return stream_open ( inode , file ) ;
2005-04-17 02:20:36 +04:00
}
2008-05-19 17:08:11 +04:00
static int fop_close ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
2008-05-19 17:08:11 +04:00
if ( wdt_expect_close = = 42 )
2005-04-17 02:20:36 +04:00
wdt_turnoff ( ) ;
else {
del_timer ( & timer ) ;
2012-02-16 03:06:19 +04:00
pr_crit ( " device file closed unexpectedly. Will not stop the WDT! \n " ) ;
2005-04-17 02:20:36 +04:00
}
clear_bit ( 0 , & wdt_is_open ) ;
wdt_expect_close = 0 ;
return 0 ;
}
2008-05-19 17:08:11 +04:00
static long fop_ioctl ( struct file * file , unsigned int cmd , unsigned long arg )
2005-04-17 02:20:36 +04:00
{
void __user * argp = ( void __user * ) arg ;
int __user * p = argp ;
2008-05-19 17:08:11 +04:00
static const struct watchdog_info ident = {
. options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT |
WDIOF_MAGICCLOSE ,
2005-04-17 02:20:36 +04:00
. firmware_version = 1 ,
. identity = " SBC60xx " ,
} ;
2008-05-19 17:08:11 +04:00
switch ( cmd ) {
case WDIOC_GETSUPPORT :
2008-08-07 00:19:41 +04:00
return copy_to_user ( argp , & ident , sizeof ( ident ) ) ? - EFAULT : 0 ;
2008-05-19 17:08:11 +04:00
case WDIOC_GETSTATUS :
case WDIOC_GETBOOTSTATUS :
return put_user ( 0 , p ) ;
case WDIOC_SETOPTIONS :
2005-04-17 02:20:36 +04:00
{
2008-05-19 17:08:11 +04:00
int new_options , retval = - EINVAL ;
if ( get_user ( new_options , p ) )
return - EFAULT ;
if ( new_options & WDIOS_DISABLECARD ) {
wdt_turnoff ( ) ;
retval = 0 ;
2005-04-17 02:20:36 +04:00
}
2008-05-19 17:08:11 +04:00
if ( new_options & WDIOS_ENABLECARD ) {
wdt_startup ( ) ;
retval = 0 ;
2005-04-17 02:20:36 +04:00
}
2008-05-19 17:08:11 +04:00
return retval ;
}
2008-07-18 15:41:17 +04:00
case WDIOC_KEEPALIVE :
wdt_keepalive ( ) ;
return 0 ;
2008-05-19 17:08:11 +04:00
case WDIOC_SETTIMEOUT :
{
int new_timeout ;
if ( get_user ( new_timeout , p ) )
return - EFAULT ;
/* arbitrary upper limit */
if ( new_timeout < 1 | | new_timeout > 3600 )
return - EINVAL ;
timeout = new_timeout ;
wdt_keepalive ( ) ;
}
2020-07-07 20:11:21 +03:00
fallthrough ;
2008-05-19 17:08:11 +04:00
case WDIOC_GETTIMEOUT :
return put_user ( timeout , p ) ;
2008-07-18 15:41:17 +04:00
default :
return - ENOTTY ;
2005-04-17 02:20:36 +04:00
}
}
2006-07-03 11:24:21 +04:00
static const struct file_operations wdt_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = fop_write ,
. open = fop_open ,
. release = fop_close ,
2008-05-19 17:08:11 +04:00
. unlocked_ioctl = fop_ioctl ,
2019-06-03 15:23:09 +03:00
. compat_ioctl = compat_ptr_ioctl ,
2005-04-17 02:20:36 +04:00
} ;
static struct miscdevice wdt_miscdev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & wdt_fops ,
} ;
/*
* Notifier for system down
*/
static int wdt_notify_sys ( struct notifier_block * this , unsigned long code ,
void * unused )
{
2008-05-19 17:08:11 +04:00
if ( code = = SYS_DOWN | | code = = SYS_HALT )
2005-04-17 02:20:36 +04:00
wdt_turnoff ( ) ;
return NOTIFY_DONE ;
}
/*
* The WDT needs to learn about soft shutdowns in order to
* turn the timebomb registers off .
*/
2008-05-19 17:08:11 +04:00
static struct notifier_block wdt_notifier = {
2005-04-17 02:20:36 +04:00
. notifier_call = wdt_notify_sys ,
} ;
static void __exit sbc60xxwdt_unload ( void )
{
wdt_turnoff ( ) ;
/* Deregister */
misc_deregister ( & wdt_miscdev ) ;
unregister_reboot_notifier ( & wdt_notifier ) ;
if ( ( wdt_stop ! = 0x45 ) & & ( wdt_stop ! = wdt_start ) )
2008-05-19 17:08:11 +04:00
release_region ( wdt_stop , 1 ) ;
release_region ( wdt_start , 1 ) ;
2005-04-17 02:20:36 +04:00
}
static int __init sbc60xxwdt_init ( void )
{
int rc = - EBUSY ;
2008-05-19 17:08:11 +04:00
if ( timeout < 1 | | timeout > 3600 ) { /* arbitrary upper limit */
2005-04-17 02:20:36 +04:00
timeout = WATCHDOG_TIMEOUT ;
2012-02-16 03:06:19 +04:00
pr_info ( " timeout value must be 1 <= x <= 3600, using %d \n " ,
timeout ) ;
2008-05-19 17:08:11 +04:00
}
2005-04-17 02:20:36 +04:00
2008-05-19 17:08:11 +04:00
if ( ! request_region ( wdt_start , 1 , " SBC 60XX WDT " ) ) {
2012-02-16 03:06:19 +04:00
pr_err ( " I/O address 0x%04x already in use \n " , wdt_start ) ;
2005-04-17 02:20:36 +04:00
rc = - EIO ;
goto err_out ;
}
/* We cannot reserve 0x45 - the kernel already has! */
2008-05-19 17:08:11 +04:00
if ( wdt_stop ! = 0x45 & & wdt_stop ! = wdt_start ) {
if ( ! request_region ( wdt_stop , 1 , " SBC 60XX WDT " ) ) {
2012-02-16 03:06:19 +04:00
pr_err ( " I/O address 0x%04x already in use \n " , wdt_stop ) ;
2005-04-17 02:20:36 +04:00
rc = - EIO ;
goto err_out_region1 ;
}
}
2007-12-26 23:32:51 +03:00
rc = register_reboot_notifier ( & wdt_notifier ) ;
2008-05-19 17:08:11 +04:00
if ( rc ) {
2012-02-16 03:06:19 +04:00
pr_err ( " cannot register reboot notifier (err=%d) \n " , rc ) ;
2005-04-17 02:20:36 +04:00
goto err_out_region2 ;
}
2007-12-26 23:32:51 +03:00
rc = misc_register ( & wdt_miscdev ) ;
2008-05-19 17:08:11 +04:00
if ( rc ) {
2012-02-16 03:06:19 +04:00
pr_err ( " cannot register miscdev on minor=%d (err=%d) \n " ,
wdt_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 ( " WDT driver for 60XX single board computer initialised. timeout=%d sec (nowayout=%d) \n " ,
timeout , nowayout ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2007-12-26 23:32:51 +03:00
err_out_reboot :
unregister_reboot_notifier ( & wdt_notifier ) ;
2005-04-17 02:20:36 +04:00
err_out_region2 :
2008-05-19 17:08:11 +04:00
if ( wdt_stop ! = 0x45 & & wdt_stop ! = wdt_start )
release_region ( wdt_stop , 1 ) ;
2005-04-17 02:20:36 +04:00
err_out_region1 :
2008-05-19 17:08:11 +04:00
release_region ( wdt_start , 1 ) ;
2005-04-17 02:20:36 +04:00
err_out :
return rc ;
}
module_init ( sbc60xxwdt_init ) ;
module_exit ( sbc60xxwdt_unload ) ;
MODULE_AUTHOR ( " Jakob Oestergaard <jakob@unthought.net> " ) ;
MODULE_DESCRIPTION ( " 60xx Single Board Computer Watchdog Timer driver " ) ;
MODULE_LICENSE ( " GPL " ) ;