2005-04-16 15:20:36 -07:00
/*
* MixCom Watchdog : A Simple Hardware Watchdog Device
* Based on Softdog driver by Alan Cox and PC Watchdog driver by Ken Hollis
*
* Author : Gergely Madarasz < gorgo @ itc . hu >
*
* Copyright ( c ) 1999 ITConsult - Pro Co . < info @ itc . hu >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
* Version 0.1 ( 99 / 04 / 15 ) :
* - first version
*
* Version 0.2 ( 99 / 06 / 16 ) :
* - added kernel timer watchdog ping after close
* since the hardware does not support watchdog shutdown
*
* Version 0.3 ( 99 / 06 / 21 ) :
* - added WDIOC_GETSTATUS and WDIOC_GETSUPPORT ioctl calls
*
* Version 0.3 .1 ( 99 / 06 / 22 ) :
* - allow module removal while internal timer is active ,
* print warning about probable reset
*
* Version 0.4 ( 99 / 11 / 15 ) :
* - support for one more type board
*
* Version 0.5 ( 2001 / 12 / 14 ) Matt Domsch < Matt_Domsch @ dell . com >
2007-06-03 15:22:32 +00:00
* - added nowayout module option to override CONFIG_WATCHDOG_NOWAYOUT
*
* Version 0.6 ( 2002 / 04 / 12 ) : Rob Radez < rob @ osinvestor . com >
* - make mixcomwd_opened unsigned ,
* removed lock_kernel / unlock_kernel from mixcomwd_release ,
* modified ioctl a bit to conform to API
2005-04-16 15:20:36 -07:00
*
*/
2007-06-03 15:22:32 +00:00
# define VERSION "0.6"
2007-06-04 18:35:41 +00:00
# define WATCHDOG_NAME "mixcomwd"
# define PFX WATCHDOG_NAME ": "
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/types.h>
# include <linux/miscdevice.h>
# include <linux/ioport.h>
# include <linux/watchdog.h>
# include <linux/fs.h>
# include <linux/reboot.h>
# include <linux/init.h>
2005-10-30 15:03:48 -08:00
# include <linux/jiffies.h>
# include <linux/timer.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
# include <asm/io.h>
2007-06-03 15:39:25 +00:00
/*
* We have two types of cards that can be probed :
* 1 ) The Mixcom cards : these cards can be found at addresses
* 0x180 , 0x280 , 0x380 with an additional offset of 0xc10 .
* ( Or 0xd90 , 0xe90 , 0xf90 ) .
* 2 ) The FlashCOM cards : these cards can be set up at
* 0x300 - > 0x378 , in 0x8 jumps with an offset of 0x04 .
* ( Or 0x304 - > 0x37c in 0x8 jumps ) .
* Each card has it ' s own ID .
*/
2005-04-16 15:20:36 -07:00
# define MIXCOM_ID 0x11
# define FLASHCOM_ID 0x18
2007-06-03 20:46:05 +00:00
static struct {
int ioport ;
int id ;
} mixcomwd_io_info [ ] __devinitdata = {
/* The Mixcom cards */
{ 0x0d90 , MIXCOM_ID } ,
{ 0x0e90 , MIXCOM_ID } ,
{ 0x0f90 , MIXCOM_ID } ,
/* The FlashCOM cards */
{ 0x0304 , FLASHCOM_ID } ,
{ 0x030c , FLASHCOM_ID } ,
{ 0x0314 , FLASHCOM_ID } ,
{ 0x031c , FLASHCOM_ID } ,
{ 0x0324 , FLASHCOM_ID } ,
{ 0x032c , FLASHCOM_ID } ,
{ 0x0334 , FLASHCOM_ID } ,
{ 0x033c , FLASHCOM_ID } ,
{ 0x0344 , FLASHCOM_ID } ,
{ 0x034c , FLASHCOM_ID } ,
{ 0x0354 , FLASHCOM_ID } ,
{ 0x035c , FLASHCOM_ID } ,
{ 0x0364 , FLASHCOM_ID } ,
{ 0x036c , FLASHCOM_ID } ,
{ 0x0374 , FLASHCOM_ID } ,
{ 0x037c , FLASHCOM_ID } ,
/* The end of the list */
{ 0x0000 , 0 } ,
} ;
2005-04-16 15:20:36 -07:00
2007-02-08 18:39:36 +01:00
static void mixcomwd_timerfun ( unsigned long d ) ;
2005-04-16 15:20:36 -07:00
static unsigned long mixcomwd_opened ; /* long req'd for setbit --RR */
static int watchdog_port ;
static int mixcomwd_timer_alive ;
2007-02-08 18:39:36 +01:00
static DEFINE_TIMER ( mixcomwd_timer , mixcomwd_timerfun , 0 , 0 ) ;
2005-04-16 15:20:36 -07:00
static char expect_close ;
2005-07-27 11:43:58 -07:00
static int nowayout = WATCHDOG_NOWAYOUT ;
2005-04-16 15:20:36 -07:00
module_param ( nowayout , int , 0 ) ;
2007-01-27 20:54:24 +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
static void mixcomwd_ping ( void )
{
outb_p ( 55 , watchdog_port ) ;
return ;
}
static void mixcomwd_timerfun ( unsigned long d )
{
mixcomwd_ping ( ) ;
2007-02-08 18:39:36 +01:00
mod_timer ( & mixcomwd_timer , jiffies + 5 * HZ ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Allow only one person to hold it open
*/
static int mixcomwd_open ( struct inode * inode , struct file * file )
{
if ( test_and_set_bit ( 0 , & mixcomwd_opened ) ) {
return - EBUSY ;
}
mixcomwd_ping ( ) ;
if ( nowayout ) {
/*
* fops_get ( ) code via open ( ) has already done
* a try_module_get ( ) so it is safe to do the
* __module_get ( ) .
*/
__module_get ( THIS_MODULE ) ;
} else {
if ( mixcomwd_timer_alive ) {
del_timer ( & mixcomwd_timer ) ;
mixcomwd_timer_alive = 0 ;
}
}
return nonseekable_open ( inode , file ) ;
}
static int mixcomwd_release ( struct inode * inode , struct file * file )
{
if ( expect_close = = 42 ) {
if ( mixcomwd_timer_alive ) {
2007-06-04 18:35:41 +00:00
printk ( KERN_ERR PFX " release called while internal timer alive " ) ;
2005-04-16 15:20:36 -07:00
return - EBUSY ;
}
mixcomwd_timer_alive = 1 ;
2007-02-08 18:39:36 +01:00
mod_timer ( & mixcomwd_timer , jiffies + 5 * HZ ) ;
2005-04-16 15:20:36 -07:00
} else {
2007-06-04 18:35:41 +00:00
printk ( KERN_CRIT PFX " WDT device closed unexpectedly. WDT will not stop! \n " ) ;
2005-04-16 15:20:36 -07:00
}
clear_bit ( 0 , & mixcomwd_opened ) ;
expect_close = 0 ;
return 0 ;
}
static ssize_t mixcomwd_write ( struct file * file , const char __user * data , size_t len , loff_t * ppos )
{
if ( len )
{
if ( ! nowayout ) {
size_t i ;
/* In case it was set long ago */
expect_close = 0 ;
for ( i = 0 ; i ! = len ; i + + ) {
char c ;
if ( get_user ( c , data + i ) )
return - EFAULT ;
if ( c = = ' V ' )
expect_close = 42 ;
}
}
mixcomwd_ping ( ) ;
}
return len ;
}
static int mixcomwd_ioctl ( struct inode * inode , struct file * file ,
unsigned int cmd , unsigned long arg )
{
void __user * argp = ( void __user * ) arg ;
int __user * p = argp ;
int status ;
static struct watchdog_info ident = {
. options = WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE ,
. firmware_version = 1 ,
. identity = " MixCOM watchdog " ,
} ;
switch ( cmd )
{
case WDIOC_GETSTATUS :
status = mixcomwd_opened ;
if ( ! nowayout ) {
status | = mixcomwd_timer_alive ;
}
if ( copy_to_user ( p , & status , sizeof ( int ) ) ) {
return - EFAULT ;
}
break ;
2007-07-21 13:42:18 +00:00
case WDIOC_GETBOOTSTATUS :
if ( copy_to_user ( p , & status , sizeof ( int ) ) ) {
return - EFAULT ;
}
break ;
2005-04-16 15:20:36 -07:00
case WDIOC_GETSUPPORT :
if ( copy_to_user ( argp , & ident , sizeof ( ident ) ) ) {
return - EFAULT ;
}
break ;
case WDIOC_KEEPALIVE :
mixcomwd_ping ( ) ;
break ;
default :
2006-09-09 17:34:31 +02:00
return - ENOTTY ;
2005-04-16 15:20:36 -07:00
}
return 0 ;
}
2007-06-04 19:13:28 +00:00
static const struct file_operations mixcomwd_fops = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = mixcomwd_write ,
. ioctl = mixcomwd_ioctl ,
. open = mixcomwd_open ,
. release = mixcomwd_release ,
} ;
2007-06-04 19:13:28 +00:00
static struct miscdevice mixcomwd_miscdev = {
2005-04-16 15:20:36 -07:00
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & mixcomwd_fops ,
} ;
2007-06-03 19:01:38 +00:00
static int __init checkcard ( int port , int card_id )
2005-04-16 15:20:36 -07:00
{
int id ;
if ( ! request_region ( port , 1 , " MixCOM watchdog " ) ) {
return 0 ;
}
id = inb_p ( port ) ;
2007-06-03 19:01:38 +00:00
if ( card_id = = MIXCOM_ID )
id & = 0x3f ;
if ( id ! = card_id ) {
2005-04-16 15:20:36 -07:00
release_region ( port , 1 ) ;
return 0 ;
}
2007-06-03 19:01:38 +00:00
return 1 ;
}
2005-04-16 15:20:36 -07:00
static int __init mixcomwd_init ( void )
{
int i ;
int ret ;
int found = 0 ;
2007-06-03 20:46:05 +00:00
for ( i = 0 ; ! found & & mixcomwd_io_info [ i ] . ioport ! = 0 ; i + + ) {
if ( checkcard ( mixcomwd_io_info [ i ] . ioport ,
mixcomwd_io_info [ i ] . id ) ) {
2005-04-16 15:20:36 -07:00
found = 1 ;
2007-06-03 20:46:05 +00:00
watchdog_port = mixcomwd_io_info [ i ] . ioport ;
2005-04-16 15:20:36 -07:00
}
}
if ( ! found ) {
2007-06-04 18:35:41 +00:00
printk ( KERN_ERR PFX " No card detected, or port not available. \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
ret = misc_register ( & mixcomwd_miscdev ) ;
if ( ret )
{
2007-06-04 18:35:41 +00:00
printk ( KERN_ERR PFX " cannot register miscdev on minor=%d (err=%d) \n " ,
WATCHDOG_MINOR , ret ) ;
goto error_misc_register_watchdog ;
2005-04-16 15:20:36 -07:00
}
2007-06-04 19:13:28 +00:00
printk ( KERN_INFO " MixCOM watchdog driver v%s, watchdog port at 0x%3x \n " ,
VERSION , watchdog_port ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
2007-06-04 18:35:41 +00:00
error_misc_register_watchdog :
release_region ( watchdog_port , 1 ) ;
watchdog_port = 0x0000 ;
return ret ;
2005-04-16 15:20:36 -07:00
}
static void __exit mixcomwd_exit ( void )
{
if ( ! nowayout ) {
if ( mixcomwd_timer_alive ) {
2007-06-04 18:35:41 +00:00
printk ( KERN_WARNING PFX " I quit now, hardware will "
2005-04-16 15:20:36 -07:00
" probably reboot! \n " ) ;
2007-02-08 18:39:36 +01:00
del_timer_sync ( & mixcomwd_timer ) ;
2005-04-16 15:20:36 -07:00
mixcomwd_timer_alive = 0 ;
}
}
misc_deregister ( & mixcomwd_miscdev ) ;
2007-06-04 18:35:41 +00:00
release_region ( watchdog_port , 1 ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( mixcomwd_init ) ;
module_exit ( mixcomwd_exit ) ;
MODULE_AUTHOR ( " Gergely Madarasz <gorgo@itc.hu> " ) ;
MODULE_DESCRIPTION ( " MixCom Watchdog driver " ) ;
2007-06-04 18:35:41 +00:00
MODULE_VERSION ( VERSION ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_MISCDEV ( WATCHDOG_MINOR ) ;