2005-04-17 02:20:36 +04:00
/*
* Watchdog implementation based on z / VM Watchdog Timer API
*
2009-06-16 12:30:35 +04:00
* Copyright IBM Corp . 2004 , 2009
*
2005-04-17 02:20:36 +04:00
* The user space watchdog daemon can use this driver as
* / dev / vmwatchdog to have z / VM execute the specified CP
* command when the timeout expires . The default command is
* " IPL " , which which cause an immediate reboot .
*/
2009-06-16 12:30:35 +04:00
# define KMSG_COMPONENT "vmwatchdog"
# define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
2005-04-17 02:20:36 +04:00
# include <linux/init.h>
# include <linux/fs.h>
# include <linux/kernel.h>
# include <linux/miscdevice.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
2009-06-16 12:30:35 +04:00
# include <linux/suspend.h>
2005-04-17 02:20:36 +04:00
# include <linux/watchdog.h>
2008-05-20 21:17:01 +04:00
# include <linux/smp_lock.h>
2005-04-17 02:20:36 +04:00
# include <asm/ebcdic.h>
# include <asm/io.h>
# include <asm/uaccess.h>
# define MAX_CMDLEN 240
# define MIN_INTERVAL 15
static char vmwdt_cmd [ MAX_CMDLEN ] = " IPL " ;
static int vmwdt_conceal ;
2005-07-27 22:43:58 +04:00
static int vmwdt_nowayout = WATCHDOG_NOWAYOUT ;
2005-04-17 02:20:36 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Arnd Bergmann <arndb@de.ibm.com> " ) ;
MODULE_DESCRIPTION ( " z/VM Watchdog Timer " ) ;
module_param_string ( cmd , vmwdt_cmd , MAX_CMDLEN , 0644 ) ;
MODULE_PARM_DESC ( cmd , " CP command that is run when the watchdog triggers " ) ;
module_param_named ( conceal , vmwdt_conceal , bool , 0644 ) ;
MODULE_PARM_DESC ( conceal , " Enable the CONCEAL CP option while the watchdog "
" is active " ) ;
module_param_named ( nowayout , vmwdt_nowayout , bool , 0 ) ;
MODULE_PARM_DESC ( nowayout , " Watchdog cannot be stopped once started "
" (default=CONFIG_WATCHDOG_NOWAYOUT) " ) ;
MODULE_ALIAS_MISCDEV ( WATCHDOG_MINOR ) ;
static unsigned int vmwdt_interval = 60 ;
static unsigned long vmwdt_is_open ;
static int vmwdt_expect_close ;
2009-06-16 12:30:35 +04:00
# define VMWDT_OPEN 0 /* devnode is open or suspend in progress */
# define VMWDT_RUNNING 1 /* The watchdog is armed */
2005-04-17 02:20:36 +04:00
enum vmwdt_func {
/* function codes */
wdt_init = 0 ,
wdt_change = 1 ,
wdt_cancel = 2 ,
/* flags */
wdt_conceal = 0x80000000 ,
} ;
static int __diag288 ( enum vmwdt_func func , unsigned int timeout ,
char * cmd , size_t len )
{
2006-09-28 18:56:43 +04:00
register unsigned long __func asm ( " 2 " ) = func ;
register unsigned long __timeout asm ( " 3 " ) = timeout ;
register unsigned long __cmdp asm ( " 4 " ) = virt_to_phys ( cmd ) ;
register unsigned long __cmdl asm ( " 5 " ) = len ;
2005-04-17 02:20:36 +04:00
int err ;
2006-09-28 18:56:43 +04:00
err = - EINVAL ;
asm volatile (
" diag %1,%3,0x288 \n "
" 0: la %0,0 \n "
" 1: \n "
EX_TABLE ( 0 b , 1 b )
2007-10-12 18:11:48 +04:00
: " +d " ( err ) : " d " ( __func ) , " d " ( __timeout ) ,
" d " ( __cmdp ) , " d " ( __cmdl ) : " 1 " , " cc " ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
static int vmwdt_keepalive ( void )
{
/* we allocate new memory every time to avoid having
* to track the state . static allocation is not an
* option since that might not be contiguous in real
* storage in case of a modular build */
static char * ebc_cmd ;
size_t len ;
int ret ;
unsigned int func ;
ebc_cmd = kmalloc ( MAX_CMDLEN , GFP_KERNEL ) ;
if ( ! ebc_cmd )
return - ENOMEM ;
len = strlcpy ( ebc_cmd , vmwdt_cmd , MAX_CMDLEN ) ;
ASCEBC ( ebc_cmd , MAX_CMDLEN ) ;
EBC_TOUPPER ( ebc_cmd , MAX_CMDLEN ) ;
func = vmwdt_conceal ? ( wdt_init | wdt_conceal ) : wdt_init ;
2009-06-16 12:30:35 +04:00
set_bit ( VMWDT_RUNNING , & vmwdt_is_open ) ;
2005-04-17 02:20:36 +04:00
ret = __diag288 ( func , vmwdt_interval , ebc_cmd , len ) ;
2008-07-14 11:59:40 +04:00
WARN_ON ( ret ! = 0 ) ;
2005-04-17 02:20:36 +04:00
kfree ( ebc_cmd ) ;
return ret ;
}
static int vmwdt_disable ( void )
{
int ret = __diag288 ( wdt_cancel , 0 , " " , 0 ) ;
2008-07-14 11:59:40 +04:00
WARN_ON ( ret ! = 0 ) ;
2009-06-16 12:30:35 +04:00
clear_bit ( VMWDT_RUNNING , & vmwdt_is_open ) ;
2005-04-17 02:20:36 +04:00
return ret ;
}
static int __init vmwdt_probe ( void )
{
/* there is no real way to see if the watchdog is supported,
* so we try initializing it with a NOP command ( " BEGIN " )
* that won ' t cause any harm even if the following disable
* fails for some reason */
static char __initdata ebc_begin [ ] = {
194 , 197 , 199 , 201 , 213
} ;
2008-07-14 11:59:40 +04:00
if ( __diag288 ( wdt_init , 15 , ebc_begin , sizeof ( ebc_begin ) ) ! = 0 )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
return vmwdt_disable ( ) ;
}
static int vmwdt_open ( struct inode * i , struct file * f )
{
int ret ;
2008-05-20 21:17:01 +04:00
lock_kernel ( ) ;
2009-06-16 12:30:35 +04:00
if ( test_and_set_bit ( VMWDT_OPEN , & vmwdt_is_open ) ) {
2008-05-20 21:17:01 +04:00
unlock_kernel ( ) ;
2005-04-17 02:20:36 +04:00
return - EBUSY ;
2008-05-20 21:17:01 +04:00
}
2005-04-17 02:20:36 +04:00
ret = vmwdt_keepalive ( ) ;
if ( ret )
2009-06-16 12:30:35 +04:00
clear_bit ( VMWDT_OPEN , & vmwdt_is_open ) ;
2008-05-20 21:17:01 +04:00
unlock_kernel ( ) ;
2005-04-17 02:20:36 +04:00
return ret ? ret : nonseekable_open ( i , f ) ;
}
static int vmwdt_close ( struct inode * i , struct file * f )
{
if ( vmwdt_expect_close = = 42 )
vmwdt_disable ( ) ;
vmwdt_expect_close = 0 ;
2009-06-16 12:30:35 +04:00
clear_bit ( VMWDT_OPEN , & vmwdt_is_open ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static struct watchdog_info vmwdt_info = {
. options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE ,
. firmware_version = 0 ,
. identity = " z/VM Watchdog Timer " ,
} ;
static int vmwdt_ioctl ( struct inode * i , struct file * f ,
unsigned int cmd , unsigned long arg )
{
switch ( cmd ) {
case WDIOC_GETSUPPORT :
if ( copy_to_user ( ( void __user * ) arg , & vmwdt_info ,
sizeof ( vmwdt_info ) ) )
return - EFAULT ;
return 0 ;
case WDIOC_GETSTATUS :
case WDIOC_GETBOOTSTATUS :
2006-07-12 18:41:55 +04:00
return put_user ( 0 , ( int __user * ) arg ) ;
2005-04-17 02:20:36 +04:00
case WDIOC_GETTEMP :
return - EINVAL ;
case WDIOC_SETOPTIONS :
{
int options , ret ;
if ( get_user ( options , ( int __user * ) arg ) )
return - EFAULT ;
ret = - EINVAL ;
if ( options & WDIOS_DISABLECARD ) {
ret = vmwdt_disable ( ) ;
if ( ret )
return ret ;
}
if ( options & WDIOS_ENABLECARD ) {
ret = vmwdt_keepalive ( ) ;
}
return ret ;
}
case WDIOC_GETTIMEOUT :
return put_user ( vmwdt_interval , ( int __user * ) arg ) ;
case WDIOC_SETTIMEOUT :
{
int interval ;
if ( get_user ( interval , ( int __user * ) arg ) )
return - EFAULT ;
if ( interval < MIN_INTERVAL )
return - EINVAL ;
vmwdt_interval = interval ;
}
return vmwdt_keepalive ( ) ;
case WDIOC_KEEPALIVE :
return vmwdt_keepalive ( ) ;
}
return - EINVAL ;
}
static ssize_t vmwdt_write ( struct file * f , const char __user * buf ,
size_t count , loff_t * ppos )
{
if ( count ) {
if ( ! vmwdt_nowayout ) {
size_t i ;
/* note: just in case someone wrote the magic character
* five months ago . . . */
vmwdt_expect_close = 0 ;
for ( i = 0 ; i ! = count ; i + + ) {
char c ;
if ( get_user ( c , buf + i ) )
return - EFAULT ;
if ( c = = ' V ' )
vmwdt_expect_close = 42 ;
}
}
/* someone wrote to us, we should restart timer */
vmwdt_keepalive ( ) ;
}
return count ;
}
2009-06-16 12:30:35 +04:00
static int vmwdt_resume ( void )
{
clear_bit ( VMWDT_OPEN , & vmwdt_is_open ) ;
return NOTIFY_DONE ;
}
/*
* It makes no sense to go into suspend while the watchdog is running .
* Depending on the memory size , the watchdog might trigger , while we
* are still saving the memory .
* We reuse the open flag to ensure that suspend and watchdog open are
* exclusive operations
*/
static int vmwdt_suspend ( void )
{
if ( test_and_set_bit ( VMWDT_OPEN , & vmwdt_is_open ) ) {
2009-07-07 18:37:11 +04:00
pr_err ( " The system cannot be suspended while the watchdog "
" is in use \n " ) ;
2009-06-16 12:30:35 +04:00
return NOTIFY_BAD ;
}
if ( test_bit ( VMWDT_RUNNING , & vmwdt_is_open ) ) {
clear_bit ( VMWDT_OPEN , & vmwdt_is_open ) ;
2009-07-07 18:37:11 +04:00
pr_err ( " The system cannot be suspended while the watchdog "
" is running \n " ) ;
2009-06-16 12:30:35 +04:00
return NOTIFY_BAD ;
}
return NOTIFY_DONE ;
}
/*
* This function is called for suspend and resume .
*/
static int vmwdt_power_event ( struct notifier_block * this , unsigned long event ,
void * ptr )
{
switch ( event ) {
case PM_POST_HIBERNATION :
case PM_POST_SUSPEND :
return vmwdt_resume ( ) ;
case PM_HIBERNATION_PREPARE :
case PM_SUSPEND_PREPARE :
return vmwdt_suspend ( ) ;
default :
return NOTIFY_DONE ;
}
}
static struct notifier_block vmwdt_power_notifier = {
. notifier_call = vmwdt_power_event ,
} ;
2007-02-12 11:55:34 +03:00
static const struct file_operations vmwdt_fops = {
2005-04-17 02:20:36 +04:00
. open = & vmwdt_open ,
. release = & vmwdt_close ,
. ioctl = & vmwdt_ioctl ,
. write = & vmwdt_write ,
. owner = THIS_MODULE ,
} ;
static struct miscdevice vmwdt_dev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & vmwdt_fops ,
} ;
static int __init vmwdt_init ( void )
{
int ret ;
ret = vmwdt_probe ( ) ;
if ( ret )
return ret ;
2009-06-16 12:30:35 +04:00
ret = register_pm_notifier ( & vmwdt_power_notifier ) ;
if ( ret )
return ret ;
ret = misc_register ( & vmwdt_dev ) ;
if ( ret ) {
unregister_pm_notifier ( & vmwdt_power_notifier ) ;
return ret ;
}
return 0 ;
2005-04-17 02:20:36 +04:00
}
module_init ( vmwdt_init ) ;
static void __exit vmwdt_exit ( void )
{
2009-06-16 12:30:35 +04:00
unregister_pm_notifier ( & vmwdt_power_notifier ) ;
misc_deregister ( & vmwdt_dev ) ;
2005-04-17 02:20:36 +04:00
}
module_exit ( vmwdt_exit ) ;