2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
* AMD Elan SC520 processor Watchdog Timer driver
*
2009-03-18 08:35:09 +00:00
* Based on acquirewdt . c by Alan Cox ,
* and sbc60xxwdt . c by Jakob Oestergaard < jakob @ unthought . net >
2005-04-16 15:20:36 -07:00
*
* The authors do NOT admit liability nor provide warranty for
* any of this software . This material is provided " AS-IS " in
2009-03-18 08:35:09 +00:00
* the hope that it may be useful for others .
2005-04-16 15:20:36 -07:00
*
* ( c ) Copyright 2001 Scott Jennings < linuxdrivers @ oro . net >
* 9 / 27 - 2001 [ Initial release ]
*
* Additional fixes Alan Cox
* - Fixed formatting
* - Removed debug printks
* - Fixed SMP built kernel deadlock
* - Switched to private locks not lock_kernel
* - Used ioremap / writew / readw
* - Added NOWAYOUT support
* 4 / 12 - 2002 Changes by Rob Radez < rob @ osinvestor . com >
* - Change comments
* - Eliminate fop_llseek
* - Change CONFIG_WATCHDOG_NOWAYOUT semantics
* - Add KERN_ * tags to printks
* - fix possible wdt_is_open race
* - Report proper capabilities in watchdog_info
* - Add WDIOC_ { GETSTATUS , GETBOOTSTATUS , SETTIMEOUT ,
* GETTIMEOUT , SETOPTIONS } ioctls
* 09 / 8 - 2003 Changes by Wim Van Sebroeck < wim @ iguana . be >
* - cleanup of trailing spaces
* - added extra printk ' s for startup problems
* - use module_param
* - made timeout ( the emulated heartbeat ) a module_param
* - made the keepalive ping an internal subroutine
* 3 / 27 - 2004 Changes by Sean Young < sean @ mess . org >
* - set MMCR_BASE to 0xfffef000
* - CBAR does not need to be read
* - removed debugging printks
*
* This WDT driver is different from most other Linux WDT
* drivers in that the driver will ping the watchdog by itself ,
* because this particular WDT has a very short timeout ( 1.6
* seconds ) and it would be insane to count on any userspace
* daemon always getting scheduled within that time frame .
*
* This driver uses memory mapped IO , and spinlock .
*/
2012-02-15 15:06:19 -08:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/types.h>
# include <linux/timer.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>
2005-10-30 15:03:48 -08:00
# include <linux/jiffies.h>
2008-05-19 14:08:44 +01:00
# include <linux/io.h>
# include <linux/uaccess.h>
2005-04-16 15:20:36 -07:00
/*
* The AMD Elan SC520 timeout value is 492u s times a power of 2 ( 0 - 7 )
*
* 0 : 492u s 2 : 1.01 s 4 : 4.03 s 6 : 16.22 s
* 1 : 503 ms 3 : 2.01 s 5 : 8.05 s 7 : 32.21 s
*
* We will program the SC520 watchdog for a timeout of 2.01 s .
* 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 .
*/
# define WATCHDOG_TIMEOUT 30 /* 30 sec default timeout */
2008-05-19 14:08:44 +01:00
/* in seconds, will be multiplied by HZ to get seconds to wait for a ping */
static int timeout = WATCHDOG_TIMEOUT ;
2005-04-16 15:20:36 -07:00
module_param ( timeout , int , 0 ) ;
2008-05-19 14:08:44 +01:00
MODULE_PARM_DESC ( timeout ,
" Watchdog timeout in seconds. (1 <= timeout <= 3600, default= "
__MODULE_STRING ( WATCHDOG_TIMEOUT ) " ) " ) ;
2005-04-16 15:20:36 -07:00
2012-03-05 16:51:11 +01:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
module_param ( nowayout , bool , 0 ) ;
2008-05-19 14:08:44 +01:00
MODULE_PARM_DESC ( nowayout ,
" Watchdog cannot be stopped once started (default= "
__MODULE_STRING ( WATCHDOG_NOWAYOUT ) " ) " ) ;
2005-04-16 15:20:36 -07:00
/*
* AMD Elan SC520 - Watchdog Timer Registers
*/
# define MMCR_BASE 0xfffef000 /* The default base address */
# define OFFS_WDTMRCTL 0xCB0 /* Watchdog Timer Control Register */
/* WDT Control Register bit definitions */
# define WDT_EXP_SEL_01 0x0001 /* [01] Time-out = 496 us (with 33 Mhz clk). */
# define WDT_EXP_SEL_02 0x0002 /* [02] Time-out = 508 ms (with 33 Mhz clk). */
# define WDT_EXP_SEL_03 0x0004 /* [03] Time-out = 1.02 s (with 33 Mhz clk). */
# define WDT_EXP_SEL_04 0x0008 /* [04] Time-out = 2.03 s (with 33 Mhz clk). */
# define WDT_EXP_SEL_05 0x0010 /* [05] Time-out = 4.07 s (with 33 Mhz clk). */
# define WDT_EXP_SEL_06 0x0020 /* [06] Time-out = 8.13 s (with 33 Mhz clk). */
# define WDT_EXP_SEL_07 0x0040 /* [07] Time-out = 16.27s (with 33 Mhz clk). */
# define WDT_EXP_SEL_08 0x0080 /* [08] Time-out = 32.54s (with 33 Mhz clk). */
# define WDT_IRQ_FLG 0x1000 /* [12] Interrupt Request Flag */
# define WDT_WRST_ENB 0x4000 /* [14] Watchdog Timer Reset Enable */
# define WDT_ENB 0x8000 /* [15] Watchdog Timer Enable */
static __u16 __iomem * wdtmrctl ;
2017-08-28 11:28:21 -07:00
static void wdt_timer_ping ( struct timer_list * ) ;
2017-10-04 16:27:04 -07:00
static DEFINE_TIMER ( timer , wdt_timer_ping ) ;
2005-04-16 15:20:36 -07:00
static unsigned long next_heartbeat ;
static unsigned long wdt_is_open ;
static char wdt_expect_close ;
2007-11-01 16:27:08 -07:00
static DEFINE_SPINLOCK ( wdt_spinlock ) ;
2005-04-16 15:20:36 -07:00
/*
* Whack the dog
*/
2017-08-28 11:28:21 -07:00
static void wdt_timer_ping ( struct timer_list * unused )
2005-04-16 15:20:36 -07:00
{
/* If we got a heartbeat pulse within the WDT_US_INTERVAL
* we agree to ping the WDT
*/
2008-05-19 14:08:44 +01:00
if ( time_before ( jiffies , next_heartbeat ) ) {
2005-04-16 15:20:36 -07:00
/* Ping the WDT */
spin_lock ( & wdt_spinlock ) ;
writew ( 0xAAAA , wdtmrctl ) ;
writew ( 0x5555 , wdtmrctl ) ;
spin_unlock ( & wdt_spinlock ) ;
/* Re-set the timer interval */
2007-02-08 18:39:36 +01:00
mod_timer ( & timer , jiffies + WDT_INTERVAL ) ;
2008-05-19 14:08:44 +01:00
} else
2012-02-15 15:06:19 -08:00
pr_warn ( " Heartbeat lost! Will not ping the watchdog \n " ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Utility routines
*/
static void wdt_config ( int writeval )
{
unsigned long flags ;
/* buy some time (ping) */
spin_lock_irqsave ( & wdt_spinlock , flags ) ;
2014-02-15 13:22:49 +04:00
readw ( wdtmrctl ) ; /* ensure write synchronization */
2005-04-16 15:20:36 -07:00
writew ( 0xAAAA , wdtmrctl ) ;
writew ( 0x5555 , wdtmrctl ) ;
/* unlock WDT = make WDT configuration register writable one time */
writew ( 0x3333 , wdtmrctl ) ;
writew ( 0xCCCC , wdtmrctl ) ;
/* write WDT configuration register */
writew ( writeval , wdtmrctl ) ;
spin_unlock_irqrestore ( & wdt_spinlock , flags ) ;
}
static int wdt_startup ( void )
{
next_heartbeat = jiffies + ( timeout * HZ ) ;
/* Start the timer */
2007-02-08 18:39:36 +01:00
mod_timer ( & timer , jiffies + WDT_INTERVAL ) ;
2005-04-16 15:20:36 -07:00
/* Start the watchdog */
wdt_config ( WDT_ENB | WDT_WRST_ENB | WDT_EXP_SEL_04 ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " Watchdog timer is now enabled \n " ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int wdt_turnoff ( void )
{
/* Stop the timer */
2021-05-11 15:04:51 +08:00
del_timer_sync ( & timer ) ;
2005-04-16 15:20:36 -07:00
/* Stop the watchdog */
wdt_config ( 0 ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " Watchdog timer is now disabled... \n " ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int wdt_keepalive ( void )
{
/* user land ping */
next_heartbeat = jiffies + ( timeout * HZ ) ;
return 0 ;
}
static int wdt_set_heartbeat ( int t )
{
if ( ( t < 1 ) | | ( t > 3600 ) ) /* arbitrary upper limit */
return - EINVAL ;
timeout = t ;
return 0 ;
}
/*
* / dev / watchdog handling
*/
2008-05-19 14:08:44 +01:00
static ssize_t fop_write ( struct file * file , const char __user * buf ,
size_t count , loff_t * ppos )
2005-04-16 15:20:36 -07:00
{
/* See if we got the magic character 'V' and reload the timer */
2008-05-19 14:08:44 +01:00
if ( count ) {
2005-04-16 15:20:36 -07:00
if ( ! nowayout ) {
size_t ofs ;
/* note: just in case someone wrote the magic character
* five months ago . . . */
wdt_expect_close = 0 ;
/* now scan */
2008-05-19 14:08:44 +01:00
for ( ofs = 0 ; ofs ! = count ; ofs + + ) {
2005-04-16 15:20:36 -07:00
char c ;
if ( get_user ( c , buf + ofs ) )
return - EFAULT ;
2008-05-19 14:08:44 +01:00
if ( c = = ' V ' )
2005-04-16 15:20:36 -07:00
wdt_expect_close = 42 ;
}
}
2008-05-19 14:08:44 +01:00
/* Well, anyhow someone wrote to us, we should
return that favour */
2005-04-16 15:20:36 -07:00
wdt_keepalive ( ) ;
}
return count ;
}
2008-05-19 14:08:44 +01:00
static int fop_open ( struct inode * inode , struct file * file )
2005-04-16 15:20:36 -07:00
{
/* Just in case we're already talking to someone... */
2008-05-19 14:08:44 +01:00
if ( test_and_set_bit ( 0 , & wdt_is_open ) )
2005-04-16 15:20:36 -07: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-16 15:20:36 -07:00
}
2008-05-19 14:08:44 +01:00
static int fop_close ( struct inode * inode , struct file * file )
2005-04-16 15:20:36 -07:00
{
2008-05-19 14:08:44 +01:00
if ( wdt_expect_close = = 42 )
2005-04-16 15:20:36 -07:00
wdt_turnoff ( ) ;
2008-05-19 14:08:44 +01:00
else {
2012-02-15 15:06:19 -08:00
pr_crit ( " Unexpected close, not stopping watchdog! \n " ) ;
2005-04-16 15:20:36 -07:00
wdt_keepalive ( ) ;
}
clear_bit ( 0 , & wdt_is_open ) ;
wdt_expect_close = 0 ;
return 0 ;
}
2008-05-19 14:08:49 +01:00
static long fop_ioctl ( struct file * file , unsigned int cmd , unsigned long arg )
2005-04-16 15:20:36 -07:00
{
void __user * argp = ( void __user * ) arg ;
int __user * p = argp ;
2008-05-19 14:08:44 +01:00
static const struct watchdog_info ident = {
. options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT
| WDIOF_MAGICCLOSE ,
2005-04-16 15:20:36 -07:00
. firmware_version = 1 ,
. identity = " SC520 " ,
} ;
2008-07-17 18:08:47 +00:00
switch ( cmd ) {
2008-05-19 14:08:44 +01:00
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_SETOPTIONS :
{
int new_options , retval = - EINVAL ;
2005-04-16 15:20:36 -07:00
2008-05-19 14:08:44 +01:00
if ( get_user ( new_options , p ) )
return - EFAULT ;
if ( new_options & WDIOS_DISABLECARD ) {
wdt_turnoff ( ) ;
retval = 0 ;
}
2005-04-16 15:20:36 -07:00
2008-05-19 14:08:44 +01:00
if ( new_options & WDIOS_ENABLECARD ) {
wdt_startup ( ) ;
retval = 0 ;
2005-04-16 15:20:36 -07:00
}
2008-05-19 14:08:44 +01:00
return retval ;
}
2008-07-18 11:41:17 +00:00
case WDIOC_KEEPALIVE :
wdt_keepalive ( ) ;
return 0 ;
2008-05-19 14:08:44 +01:00
case WDIOC_SETTIMEOUT :
{
int new_timeout ;
2005-04-16 15:20:36 -07:00
2008-05-19 14:08:44 +01:00
if ( get_user ( new_timeout , p ) )
return - EFAULT ;
2005-04-16 15:20:36 -07:00
2008-05-19 14:08:44 +01:00
if ( wdt_set_heartbeat ( new_timeout ) )
return - EINVAL ;
wdt_keepalive ( ) ;
}
2020-07-07 12:11:21 -05:00
fallthrough ;
2008-05-19 14:08:44 +01:00
case WDIOC_GETTIMEOUT :
return put_user ( timeout , p ) ;
2008-07-18 11:41:17 +00:00
default :
return - ENOTTY ;
2005-04-16 15:20:36 -07:00
}
}
2006-07-03 00:24:21 -07:00
static const struct file_operations wdt_fops = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = fop_write ,
. open = fop_open ,
. release = fop_close ,
2008-05-19 14:08:49 +01:00
. unlocked_ioctl = fop_ioctl ,
2019-06-03 14:23:09 +02:00
. compat_ioctl = compat_ptr_ioctl ,
2005-04-16 15:20:36 -07: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 14:08:44 +01:00
if ( code = = SYS_DOWN | | code = = SYS_HALT )
2005-04-16 15:20:36 -07:00
wdt_turnoff ( ) ;
return NOTIFY_DONE ;
}
/*
* The WDT needs to learn about soft shutdowns in order to
* turn the timebomb registers off .
*/
static struct notifier_block wdt_notifier = {
. notifier_call = wdt_notify_sys ,
} ;
static void __exit sc520_wdt_unload ( void )
{
if ( ! nowayout )
wdt_turnoff ( ) ;
/* Deregister */
misc_deregister ( & wdt_miscdev ) ;
unregister_reboot_notifier ( & wdt_notifier ) ;
iounmap ( wdtmrctl ) ;
}
static int __init sc520_wdt_init ( void )
{
int rc = - EBUSY ;
2008-05-19 14:08:44 +01:00
/* Check that the timeout value is within it's range ;
if not reset to the default */
2005-04-16 15:20:36 -07:00
if ( wdt_set_heartbeat ( timeout ) ) {
wdt_set_heartbeat ( WATCHDOG_TIMEOUT ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " timeout value must be 1 <= timeout <= 3600, using %d \n " ,
WATCHDOG_TIMEOUT ) ;
2005-04-16 15:20:36 -07:00
}
2011-08-23 22:30:09 +01:00
wdtmrctl = ioremap ( MMCR_BASE + OFFS_WDTMRCTL , 2 ) ;
2005-04-16 15:20:36 -07:00
if ( ! wdtmrctl ) {
2012-02-15 15:06:19 -08:00
pr_err ( " Unable to remap memory \n " ) ;
2005-04-16 15:20:36 -07:00
rc = - ENOMEM ;
goto err_out_region2 ;
}
rc = register_reboot_notifier ( & wdt_notifier ) ;
if ( rc ) {
2012-02-15 15:06:19 -08:00
pr_err ( " cannot register reboot notifier (err=%d) \n " , rc ) ;
2005-04-16 15:20:36 -07:00
goto err_out_ioremap ;
}
rc = misc_register ( & wdt_miscdev ) ;
if ( rc ) {
2012-02-15 15:06:19 -08:00
pr_err ( " cannot register miscdev on minor=%d (err=%d) \n " ,
WATCHDOG_MINOR , rc ) ;
2005-04-16 15:20:36 -07:00
goto err_out_notifier ;
}
2012-02-15 15:06:19 -08:00
pr_info ( " WDT driver for SC520 initialised. timeout=%d sec (nowayout=%d) \n " ,
timeout , nowayout ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
err_out_notifier :
unregister_reboot_notifier ( & wdt_notifier ) ;
err_out_ioremap :
iounmap ( wdtmrctl ) ;
err_out_region2 :
return rc ;
}
module_init ( sc520_wdt_init ) ;
module_exit ( sc520_wdt_unload ) ;
MODULE_AUTHOR ( " Scott and Bill Jennings " ) ;
2009-03-18 08:35:09 +00:00
MODULE_DESCRIPTION (
" Driver for watchdog timer in AMD \" Elan \" SC520 uProcessor " ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;