2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-16 15:20:36 -07:00
/*
* ALi M7101 PMU Computer Watchdog Timer driver
*
* Based on w83877f_wdt . c by Scott Jennings < linuxdrivers @ oro . net >
* and the Cobalt kernel WDT timer driver by Tim Hockin
* < thockin @ cobaltnet . com >
*
* ( c ) 2002 Steve Hill < steve @ navaho . co . uk >
*
* 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 .
*
* Additions :
* Aug 23 , 2004 - Added use_gpio module parameter for use on revision a1d PMUs
* found on very old cobalt hardware .
* - - Mike Waychison < michael . waychison @ sun . com >
*/
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/ioport.h>
# include <linux/notifier.h>
# include <linux/reboot.h>
# include <linux/init.h>
# include <linux/fs.h>
# include <linux/pci.h>
2008-05-19 14:04:57 +01:00
# include <linux/io.h>
# include <linux/uaccess.h>
2008-07-15 11:46:11 +00:00
2005-04-16 15:20:36 -07:00
# define WDT_ENABLE 0x9C
# define WDT_DISABLE 0x8C
# define ALI_7101_WDT 0x92
# define ALI_7101_GPIO 0x7D
# define ALI_7101_GPIO_O 0x7E
# define ALI_WDT_ARM 0x01
/*
* We ' re going to use a 1 second timeout .
* 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:04:57 +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:04:57 +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
2008-05-19 14:04:57 +01:00
static int use_gpio ; /* Use the pic (for a1d revision alim7101) */
2005-04-16 15:20:36 -07:00
module_param ( use_gpio , int , 0 ) ;
2008-05-19 14:04:57 +01:00
MODULE_PARM_DESC ( use_gpio ,
" Use the gpio watchdog (required by old cobalt boards). " ) ;
2005-04-16 15:20:36 -07:00
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 ;
static struct pci_dev * alim7101_pmu ;
2012-03-05 16:51:11 +01:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
module_param ( nowayout , bool , 0 ) ;
2008-05-19 14:04:57 +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
/*
* 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:04:57 +01:00
char tmp ;
2005-04-16 15:20:36 -07:00
2008-05-19 14:04:57 +01:00
if ( time_before ( jiffies , next_heartbeat ) ) {
2005-04-16 15:20:36 -07:00
/* Ping the WDT (this is actually a disarm/arm sequence) */
pci_read_config_byte ( alim7101_pmu , 0x92 , & tmp ) ;
2008-05-19 14:04:57 +01:00
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_WDT , ( tmp & ~ ALI_WDT_ARM ) ) ;
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_WDT , ( tmp | ALI_WDT_ARM ) ) ;
2005-04-16 15:20:36 -07:00
if ( use_gpio ) {
2008-05-19 14:04:57 +01:00
pci_read_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , & tmp ) ;
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , tmp | 0x20 ) ;
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , tmp & ~ 0x20 ) ;
2005-04-16 15:20:36 -07: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
}
/* Re-set the timer interval */
2007-02-08 18:39:36 +01:00
mod_timer ( & timer , jiffies + WDT_INTERVAL ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Utility routines
*/
static void wdt_change ( int writeval )
{
2008-08-06 20:19:41 +00:00
char tmp ;
2005-04-16 15:20:36 -07:00
pci_read_config_byte ( alim7101_pmu , ALI_7101_WDT , & tmp ) ;
if ( writeval = = WDT_ENABLE ) {
2008-05-19 14:04:57 +01:00
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_WDT , ( tmp | ALI_WDT_ARM ) ) ;
2005-04-16 15:20:36 -07:00
if ( use_gpio ) {
2008-05-19 14:04:57 +01:00
pci_read_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , & tmp ) ;
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , tmp & ~ 0x20 ) ;
2005-04-16 15:20:36 -07:00
}
} else {
2008-05-19 14:04:57 +01:00
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_WDT , ( tmp & ~ ALI_WDT_ARM ) ) ;
2005-04-16 15:20:36 -07:00
if ( use_gpio ) {
2008-05-19 14:04:57 +01:00
pci_read_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , & tmp ) ;
pci_write_config_byte ( alim7101_pmu ,
ALI_7101_GPIO_O , tmp | 0x20 ) ;
2005-04-16 15:20:36 -07:00
}
}
}
static void wdt_startup ( void )
{
next_heartbeat = jiffies + ( timeout * HZ ) ;
/* We must enable before we kick off the timer in case the timer
occurs as we ping it */
wdt_change ( WDT_ENABLE ) ;
/* 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
2012-02-15 15:06:19 -08:00
pr_info ( " Watchdog timer is now enabled \n " ) ;
2005-04-16 15:20:36 -07:00
}
static void wdt_turnoff ( void )
{
/* Stop the timer */
del_timer_sync ( & timer ) ;
wdt_change ( WDT_DISABLE ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " Watchdog timer is now disabled... \n " ) ;
2005-04-16 15:20:36 -07:00
}
static void wdt_keepalive ( void )
{
/* user land ping */
next_heartbeat = jiffies + ( timeout * HZ ) ;
}
/*
* / dev / watchdog handling
*/
2008-05-19 14:04:57 +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:04:57 +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 */
for ( ofs = 0 ; ofs ! = count ; ofs + + ) {
char c ;
2008-08-06 20:19:41 +00:00
if ( get_user ( c , buf + ofs ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
if ( c = = ' V ' )
wdt_expect_close = 42 ;
}
}
/* someone wrote to us, we should restart timer */
wdt_keepalive ( ) ;
}
return count ;
}
2008-05-19 14:04:57 +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:04:57 +01:00
if ( test_and_set_bit ( 0 , & wdt_is_open ) )
2005-04-16 15:20:36 -07:00
return - EBUSY ;
/* 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:04:57 +01:00
static int fop_close ( struct inode * inode , struct file * file )
2005-04-16 15:20:36 -07:00
{
2008-05-19 14:04:57 +01:00
if ( wdt_expect_close = = 42 )
2005-04-16 15:20:36 -07:00
wdt_turnoff ( ) ;
else {
/* wim: shouldn't there be a: del_timer(&timer); */
2012-02-15 15:06:19 -08:00
pr_crit ( " device file closed unexpectedly. Will not stop the WDT! \n " ) ;
2005-04-16 15:20:36 -07:00
}
clear_bit ( 0 , & wdt_is_open ) ;
wdt_expect_close = 0 ;
return 0 ;
}
2008-05-19 14:04:57 +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 ;
2009-12-26 18:55:22 +00:00
static const struct watchdog_info ident = {
2008-05-19 14:04:57 +01:00
. options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT
| WDIOF_MAGICCLOSE ,
2005-04-16 15:20:36 -07:00
. firmware_version = 1 ,
. identity = " ALiM7101 " ,
} ;
2008-05-19 14:04:57 +01:00
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_SETOPTIONS :
2005-04-16 15:20:36 -07:00
{
2008-05-19 14:04:57 +01:00
int new_options , retval = - EINVAL ;
2005-04-16 15:20:36 -07:00
2008-05-19 14:04:57 +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:04:57 +01:00
if ( new_options & WDIOS_ENABLECARD ) {
wdt_startup ( ) ;
retval = 0 ;
2005-04-16 15:20:36 -07:00
}
2008-05-19 14:04:57 +01:00
return retval ;
}
2008-07-18 11:41:17 +00:00
case WDIOC_KEEPALIVE :
wdt_keepalive ( ) ;
return 0 ;
2008-05-19 14:04:57 +01: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 ( ) ;
}
2019-03-20 13:16:58 -05:00
/* Fall through */
2008-05-19 14:04:57 +01:00
case WDIOC_GETTIMEOUT :
return put_user ( timeout , p ) ;
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 = {
2008-05-19 14:04:57 +01:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = fop_write ,
. open = fop_open ,
. release = fop_close ,
. unlocked_ioctl = fop_ioctl ,
2005-04-16 15:20:36 -07:00
} ;
static struct miscdevice wdt_miscdev = {
2008-05-19 14:04:57 +01:00
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & wdt_fops ,
2005-04-16 15:20:36 -07:00
} ;
2014-09-26 00:03:17 +00:00
static int wdt_restart_handle ( struct notifier_block * this , unsigned long mode ,
void * cmd )
{
/*
* Cobalt devices have no way of rebooting themselves other
* than getting the watchdog to pull reset , so we restart the
* watchdog on reboot with no heartbeat .
*/
wdt_change ( WDT_ENABLE ) ;
/* loop until the watchdog fires */
while ( true )
;
return NOTIFY_DONE ;
}
static struct notifier_block wdt_restart_handler = {
. notifier_call = wdt_restart_handle ,
. priority = 128 ,
} ;
2005-04-16 15:20:36 -07:00
/*
* Notifier for system down
*/
2008-05-19 14:04:57 +01:00
static int wdt_notify_sys ( struct notifier_block * this ,
unsigned long code , void * unused )
2005-04-16 15:20:36 -07:00
{
2008-05-19 14:04:57 +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 .
*/
2008-05-19 14:04:57 +01:00
static struct notifier_block wdt_notifier = {
2005-04-16 15:20:36 -07:00
. notifier_call = wdt_notify_sys ,
} ;
static void __exit alim7101_wdt_unload ( void )
{
wdt_turnoff ( ) ;
/* Deregister */
misc_deregister ( & wdt_miscdev ) ;
unregister_reboot_notifier ( & wdt_notifier ) ;
2014-09-26 00:03:17 +00:00
unregister_restart_handler ( & wdt_restart_handler ) ;
2006-07-18 18:29:00 +01:59
pci_dev_put ( alim7101_pmu ) ;
2005-04-16 15:20:36 -07:00
}
static int __init alim7101_wdt_init ( void )
{
int rc = - EBUSY ;
struct pci_dev * ali1543_south ;
char tmp ;
2012-02-15 15:06:19 -08:00
pr_info ( " Steve Hill <steve@navaho.co.uk> \n " ) ;
2006-07-18 18:29:00 +01:59
alim7101_pmu = pci_get_device ( PCI_VENDOR_ID_AL , PCI_DEVICE_ID_AL_M7101 ,
NULL ) ;
2005-04-16 15:20:36 -07:00
if ( ! alim7101_pmu ) {
2012-02-15 15:06:19 -08:00
pr_info ( " ALi M7101 PMU not present - WDT not set \n " ) ;
2005-04-16 15:20:36 -07:00
return - EBUSY ;
}
/* Set the WDT in the PMU to 1 second */
pci_write_config_byte ( alim7101_pmu , ALI_7101_WDT , 0x02 ) ;
2006-07-18 18:29:00 +01:59
ali1543_south = pci_get_device ( PCI_VENDOR_ID_AL , PCI_DEVICE_ID_AL_M1533 ,
NULL ) ;
2005-04-16 15:20:36 -07:00
if ( ! ali1543_south ) {
2012-02-15 15:06:19 -08:00
pr_info ( " ALi 1543 South-Bridge not present - WDT not set \n " ) ;
2006-07-18 18:29:00 +01:59
goto err_out ;
2005-04-16 15:20:36 -07:00
}
pci_read_config_byte ( ali1543_south , 0x5e , & tmp ) ;
2006-07-18 18:29:00 +01:59
pci_dev_put ( ali1543_south ) ;
2005-04-16 15:20:36 -07:00
if ( ( tmp & 0x1e ) = = 0x00 ) {
if ( ! use_gpio ) {
2012-02-15 15:06:19 -08:00
pr_info ( " Detected old alim7101 revision 'a1d'. If this is a cobalt board, set the 'use_gpio' module parameter. \n " ) ;
2006-07-18 18:29:00 +01:59
goto err_out ;
2008-05-19 14:04:57 +01:00
}
2005-04-16 15:20:36 -07:00
nowayout = 1 ;
} else if ( ( tmp & 0x1e ) ! = 0x12 & & ( tmp & 0x1e ) ! = 0x00 ) {
2012-02-15 15:06:19 -08:00
pr_info ( " ALi 1543 South-Bridge does not have the correct revision number (???1001?) - WDT not set \n " ) ;
2006-07-18 18:29:00 +01:59
goto err_out ;
2005-04-16 15:20:36 -07:00
}
2008-05-19 14:04:57 +01:00
if ( timeout < 1 | | timeout > 3600 ) {
/* arbitrary upper limit */
2005-04-16 15:20:36 -07:00
timeout = WATCHDOG_TIMEOUT ;
2012-02-15 15:06:19 -08:00
pr_info ( " timeout value must be 1 <= x <= 3600, using %d \n " ,
timeout ) ;
2005-04-16 15:20:36 -07:00
}
2007-12-26 20:32:51 +00:00
rc = register_reboot_notifier ( & wdt_notifier ) ;
2005-04-16 15:20:36 -07:00
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 ;
}
2014-09-26 00:03:17 +00:00
rc = register_restart_handler ( & wdt_restart_handler ) ;
if ( rc ) {
pr_err ( " cannot register restart handler (err=%d) \n " , rc ) ;
goto err_out_reboot ;
}
2007-12-26 20:32:51 +00:00
rc = misc_register ( & wdt_miscdev ) ;
2005-04-16 15:20:36 -07:00
if ( rc ) {
2012-02-15 15:06:19 -08:00
pr_err ( " cannot register miscdev on minor=%d (err=%d) \n " ,
wdt_miscdev . minor , rc ) ;
2014-09-26 00:03:17 +00:00
goto err_out_restart ;
2005-04-16 15:20:36 -07:00
}
2008-05-19 14:04:57 +01:00
if ( nowayout )
2005-04-16 15:20:36 -07:00
__module_get ( THIS_MODULE ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " WDT driver for ALi M7101 initialised. timeout=%d sec (nowayout=%d) \n " ,
2005-04-16 15:20:36 -07:00
timeout , nowayout ) ;
return 0 ;
2014-09-26 00:03:17 +00:00
err_out_restart :
unregister_restart_handler ( & wdt_restart_handler ) ;
2007-12-26 20:32:51 +00:00
err_out_reboot :
unregister_reboot_notifier ( & wdt_notifier ) ;
2005-04-16 15:20:36 -07:00
err_out :
2006-07-18 18:29:00 +01:59
pci_dev_put ( alim7101_pmu ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
module_init ( alim7101_wdt_init ) ;
module_exit ( alim7101_wdt_unload ) ;
2013-12-03 08:30:22 +09:00
static const struct pci_device_id alim7101_pci_tbl [ ] __used = {
2007-02-12 00:51:48 -08:00
{ PCI_DEVICE ( PCI_VENDOR_ID_AL , PCI_DEVICE_ID_AL_M1533 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_AL , PCI_DEVICE_ID_AL_M7101 ) } ,
2006-10-18 08:24:30 -04:00
{ }
} ;
MODULE_DEVICE_TABLE ( pci , alim7101_pci_tbl ) ;
2005-04-16 15:20:36 -07:00
MODULE_AUTHOR ( " Steve Hill " ) ;
MODULE_DESCRIPTION ( " ALi M7101 PMU Computer Watchdog Timer driver " ) ;
MODULE_LICENSE ( " GPL " ) ;