2008-08-30 02:40:24 +04:00
/* riowd.c - driver for hw watchdog inside Super I/O of RIO
2005-04-17 02:20:36 +04:00
*
2008-08-30 02:35:59 +04:00
* Copyright ( C ) 2001 , 2008 David S . Miller ( davem @ davemloft . net )
2005-04-17 02:20:36 +04:00
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/types.h>
# include <linux/fs.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/miscdevice.h>
2008-05-20 21:16:35 +04:00
# include <linux/smp_lock.h>
2008-08-30 02:35:59 +04:00
# include <linux/watchdog.h>
# include <linux/of.h>
# include <linux/of_device.h>
2005-04-17 02:20:36 +04:00
# include <asm/io.h>
# include <asm/uaccess.h>
/* RIO uses the NatSemi Super I/O power management logical device
* as its ' watchdog .
*
* When the watchdog triggers , it asserts a line to the BBC ( Boot Bus
* Controller ) of the machine . The BBC can only be configured to
* trigger a power - on reset when the signal is asserted . The BBC
* can be configured to ignore the signal entirely as well .
*
* The only Super I / O device register we care about is at index
* 0x05 ( WDTO_INDEX ) which is the watchdog time - out in minutes ( 1 - 255 ) .
* If set to zero , this disables the watchdog . When set , the system
* must periodically ( before watchdog expires ) clear ( set to zero ) and
* re - set the watchdog else it will trigger .
*
* There are two other indexed watchdog registers inside this Super I / O
* logical device , but they are unused . The first , at index 0x06 is
* the watchdog control and can be used to make the watchdog timer re - set
* when the PS / 2 mouse or serial lines show activity . The second , at
* index 0x07 is merely a sampling of the line from the watchdog to the
* BBC .
*
* The watchdog device generates no interrupts .
*/
2008-08-30 02:35:59 +04:00
MODULE_AUTHOR ( " David S. Miller <davem@davemloft.net> " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " Hardware watchdog driver for Sun RIO " ) ;
MODULE_SUPPORTED_DEVICE ( " watchdog " ) ;
MODULE_LICENSE ( " GPL " ) ;
2008-08-30 02:42:31 +04:00
# define DRIVER_NAME "riowd"
# define PFX DRIVER_NAME ": "
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
struct riowd {
void __iomem * regs ;
spinlock_t lock ;
} ;
static struct riowd * riowd_device ;
2005-04-17 02:20:36 +04:00
# define WDTO_INDEX 0x05
static int riowd_timeout = 1 ; /* in minutes */
module_param ( riowd_timeout , int , 0 ) ;
MODULE_PARM_DESC ( riowd_timeout , " Watchdog timeout in minutes " ) ;
2008-08-30 02:35:59 +04:00
static void riowd_writereg ( struct riowd * p , u8 val , int index )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
2008-08-30 02:35:59 +04:00
spin_lock_irqsave ( & p - > lock , flags ) ;
writeb ( index , p - > regs + 0 ) ;
writeb ( val , p - > regs + 1 ) ;
spin_unlock_irqrestore ( & p - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
}
static int riowd_open ( struct inode * inode , struct file * filp )
{
2008-05-20 21:16:35 +04:00
cycle_kernel_lock ( ) ;
2005-04-17 02:20:36 +04:00
nonseekable_open ( inode , filp ) ;
return 0 ;
}
static int riowd_release ( struct inode * inode , struct file * filp )
{
return 0 ;
}
static int riowd_ioctl ( struct inode * inode , struct file * filp ,
unsigned int cmd , unsigned long arg )
{
static struct watchdog_info info = {
2008-08-30 02:35:59 +04:00
. options = WDIOF_SETTIMEOUT ,
. firmware_version = 1 ,
2008-08-30 02:42:31 +04:00
. identity = DRIVER_NAME ,
2005-04-17 02:20:36 +04:00
} ;
void __user * argp = ( void __user * ) arg ;
2008-08-30 02:35:59 +04:00
struct riowd * p = riowd_device ;
2005-04-17 02:20:36 +04:00
unsigned int options ;
int new_margin ;
switch ( cmd ) {
case WDIOC_GETSUPPORT :
if ( copy_to_user ( argp , & info , sizeof ( info ) ) )
return - EFAULT ;
break ;
case WDIOC_GETSTATUS :
case WDIOC_GETBOOTSTATUS :
if ( put_user ( 0 , ( int __user * ) argp ) )
return - EFAULT ;
break ;
case WDIOC_KEEPALIVE :
2008-08-30 02:35:59 +04:00
riowd_writereg ( p , riowd_timeout , WDTO_INDEX ) ;
2005-04-17 02:20:36 +04:00
break ;
case WDIOC_SETOPTIONS :
if ( copy_from_user ( & options , argp , sizeof ( options ) ) )
return - EFAULT ;
if ( options & WDIOS_DISABLECARD )
2008-08-30 02:35:59 +04:00
riowd_writereg ( p , 0 , WDTO_INDEX ) ;
2005-04-17 02:20:36 +04:00
else if ( options & WDIOS_ENABLECARD )
2008-08-30 02:35:59 +04:00
riowd_writereg ( p , riowd_timeout , WDTO_INDEX ) ;
2005-04-17 02:20:36 +04:00
else
return - EINVAL ;
break ;
case WDIOC_SETTIMEOUT :
if ( get_user ( new_margin , ( int __user * ) argp ) )
return - EFAULT ;
if ( ( new_margin < 60 ) | | ( new_margin > ( 255 * 60 ) ) )
2008-08-30 02:35:59 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
riowd_timeout = ( new_margin + 59 ) / 60 ;
2008-08-30 02:35:59 +04:00
riowd_writereg ( p , riowd_timeout , WDTO_INDEX ) ;
2005-04-17 02:20:36 +04:00
/* Fall */
case WDIOC_GETTIMEOUT :
return put_user ( riowd_timeout * 60 , ( int __user * ) argp ) ;
default :
return - EINVAL ;
} ;
return 0 ;
}
static ssize_t riowd_write ( struct file * file , const char __user * buf , size_t count , loff_t * ppos )
{
2008-08-30 02:35:59 +04:00
struct riowd * p = riowd_device ;
2005-04-17 02:20:36 +04:00
if ( count ) {
2008-08-30 02:35:59 +04:00
riowd_writereg ( p , riowd_timeout , WDTO_INDEX ) ;
2005-04-17 02:20:36 +04:00
return 1 ;
}
return 0 ;
}
2007-02-12 11:55:34 +03:00
static const struct file_operations riowd_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
2008-08-30 02:35:59 +04:00
. llseek = no_llseek ,
2005-04-17 02:20:36 +04:00
. ioctl = riowd_ioctl ,
. open = riowd_open ,
. write = riowd_write ,
. release = riowd_release ,
} ;
2008-08-30 02:35:59 +04:00
static struct miscdevice riowd_miscdev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & riowd_fops
} ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
static int __devinit riowd_probe ( struct of_device * op ,
const struct of_device_id * match )
2005-04-17 02:20:36 +04:00
{
2008-08-30 02:35:59 +04:00
struct riowd * p ;
int err = - EINVAL ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
if ( riowd_device )
goto out ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
err = - ENOMEM ;
p = kzalloc ( sizeof ( * p ) , GFP_KERNEL ) ;
if ( ! p )
goto out ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
spin_lock_init ( & p - > lock ) ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:42:31 +04:00
p - > regs = of_ioremap ( & op - > resource [ 0 ] , 0 , 2 , DRIVER_NAME ) ;
2008-08-30 02:35:59 +04:00
if ( ! p - > regs ) {
printk ( KERN_ERR PFX " Cannot map registers. \n " ) ;
goto out_free ;
2005-04-17 02:20:36 +04:00
}
2008-08-30 02:35:59 +04:00
err = misc_register ( & riowd_miscdev ) ;
if ( err ) {
printk ( KERN_ERR PFX " Cannot register watchdog misc device. \n " ) ;
goto out_iounmap ;
2005-04-17 02:20:36 +04:00
}
2008-08-30 02:35:59 +04:00
printk ( KERN_INFO PFX " Hardware watchdog [%i minutes], "
" regs at %p \n " , riowd_timeout , p - > regs ) ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
dev_set_drvdata ( & op - > dev , p ) ;
riowd_device = p ;
err = 0 ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
out_iounmap :
of_iounmap ( & op - > resource [ 0 ] , p - > regs , 2 ) ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
out_free :
kfree ( p ) ;
2005-04-17 02:20:36 +04:00
2008-08-30 02:35:59 +04:00
out :
return err ;
2005-04-17 02:20:36 +04:00
}
2008-08-30 02:35:59 +04:00
static int __devexit riowd_remove ( struct of_device * op )
2005-04-17 02:20:36 +04:00
{
2008-08-30 02:35:59 +04:00
struct riowd * p = dev_get_drvdata ( & op - > dev ) ;
2005-04-17 02:20:36 +04:00
misc_deregister ( & riowd_miscdev ) ;
2008-08-30 02:35:59 +04:00
of_iounmap ( & op - > resource [ 0 ] , p - > regs , 2 ) ;
kfree ( p ) ;
return 0 ;
}
2008-08-31 12:23:17 +04:00
static const struct of_device_id riowd_match [ ] = {
2008-08-30 02:35:59 +04:00
{
. name = " pmc " ,
} ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , riowd_match ) ;
static struct of_platform_driver riowd_driver = {
2008-08-30 02:42:31 +04:00
. name = DRIVER_NAME ,
2008-08-30 02:35:59 +04:00
. match_table = riowd_match ,
. probe = riowd_probe ,
. remove = __devexit_p ( riowd_remove ) ,
} ;
static int __init riowd_init ( void )
{
return of_register_driver ( & riowd_driver , & of_bus_type ) ;
}
static void __exit riowd_exit ( void )
{
of_unregister_driver ( & riowd_driver ) ;
2005-04-17 02:20:36 +04:00
}
module_init ( riowd_init ) ;
2008-08-30 02:35:59 +04:00
module_exit ( riowd_exit ) ;