2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2007-11-19 15:09:21 +01:00
/*
* IT8712F " Smart Guardian " Watchdog support
*
* Copyright ( c ) 2006 - 2007 Jorge Boncompte - DTI2 < jorge @ dti2 . net >
*
* Based on info and code taken from :
*
* drivers / char / watchdog / scx200_wdt . c
* drivers / hwmon / it87 . c
2008-02-10 22:11:15 -05:00
* IT8712F EC - LPC I / O Preliminary Specification 0.8 .2
* IT8712F EC - LPC I / O Preliminary Specification 0.9 .3
2007-11-19 15:09:21 +01:00
*
* The author ( s ) of this software shall not be held liable for damages
* of any nature resulting due to the use of this software . This
* software is provided AS - IS with no warranties .
*/
2012-02-15 15:06:19 -08:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2007-11-19 15:09:21 +01:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/miscdevice.h>
# include <linux/watchdog.h>
# include <linux/notifier.h>
# include <linux/reboot.h>
# include <linux/fs.h>
# include <linux/spinlock.h>
2008-08-04 17:54:01 +01:00
# include <linux/uaccess.h>
# include <linux/io.h>
2011-07-25 18:53:00 +00:00
# include <linux/ioport.h>
2007-11-19 15:09:21 +01:00
# define NAME "it8712f_wdt"
MODULE_AUTHOR ( " Jorge Boncompte - DTI2 <jorge@dti2.net> " ) ;
MODULE_DESCRIPTION ( " IT8712F Watchdog Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
2008-02-10 22:11:15 -05:00
static int max_units = 255 ;
2007-11-19 15:09:21 +01:00
static int margin = 60 ; /* in seconds */
module_param ( margin , int , 0 ) ;
MODULE_PARM_DESC ( margin , " Watchdog margin in seconds " ) ;
2012-03-05 16:51:11 +01:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
module_param ( nowayout , bool , 0 ) ;
2007-11-19 15:09:21 +01:00
MODULE_PARM_DESC ( nowayout , " Disable watchdog shutdown on close " ) ;
2008-08-04 17:54:01 +01:00
static unsigned long wdt_open ;
2007-11-19 15:09:21 +01:00
static unsigned expect_close ;
2008-02-10 22:11:15 -05:00
static unsigned char revision ;
2007-11-19 15:09:21 +01:00
/* Dog Food address - We use the game port address */
static unsigned short address ;
# define REG 0x2e /* The register to read/write */
# define VAL 0x2f /* The value to read/write */
# define LDN 0x07 /* Register: Logical device select */
# define DEVID 0x20 /* Register: Device ID */
# define DEVREV 0x22 /* Register: Device Revision */
# define ACT_REG 0x30 /* LDN Register: Activation */
# define BASE_REG 0x60 /* LDN Register: Base address */
# define IT8712F_DEVID 0x8712
# define LDN_GPIO 0x07 /* GPIO and Watch Dog Timer */
2011-02-23 20:04:38 +00:00
# define LDN_GAME 0x09 /* Game Port */
2007-11-19 15:09:21 +01:00
# define WDT_CONTROL 0x71 /* WDT Register: Control */
# define WDT_CONFIG 0x72 /* WDT Register: Configuration */
# define WDT_TIMEOUT 0x73 /* WDT Register: Timeout Value */
2010-09-30 17:08:03 +03:00
# define WDT_RESET_GAME 0x10 /* Reset timer on read or write to game port */
# define WDT_RESET_KBD 0x20 /* Reset timer on keyboard interrupt */
# define WDT_RESET_MOUSE 0x40 /* Reset timer on mouse interrupt */
# define WDT_RESET_CIR 0x80 /* Reset timer on consumer IR interrupt */
2007-11-19 15:09:21 +01:00
# define WDT_UNIT_SEC 0x80 /* If 0 in MINUTES */
2010-09-30 17:08:03 +03:00
# define WDT_OUT_PWROK 0x10 /* Pulse PWROK on timeout */
# define WDT_OUT_KRST 0x40 /* Pulse reset on timeout */
2007-11-19 15:09:21 +01:00
2010-09-30 17:08:04 +03:00
static int wdt_control_reg = WDT_RESET_GAME ;
module_param ( wdt_control_reg , int , 0 ) ;
MODULE_PARM_DESC ( wdt_control_reg , " Value to write to watchdog control "
" register. The default WDT_RESET_GAME resets the timer on "
" game port reads that this driver generates. You can also "
" use KBD, MOUSE or CIR if you have some external way to "
" generate those interrupts. " ) ;
2008-08-04 17:54:01 +01:00
static int superio_inb ( int reg )
2007-11-19 15:09:21 +01:00
{
outb ( reg , REG ) ;
return inb ( VAL ) ;
}
2008-08-04 17:54:01 +01:00
static void superio_outb ( int val , int reg )
2007-11-19 15:09:21 +01:00
{
outb ( reg , REG ) ;
outb ( val , VAL ) ;
}
2008-08-04 17:54:01 +01:00
static int superio_inw ( int reg )
2007-11-19 15:09:21 +01:00
{
int val ;
outb ( reg + + , REG ) ;
val = inb ( VAL ) < < 8 ;
outb ( reg , REG ) ;
val | = inb ( VAL ) ;
return val ;
}
2008-08-04 17:54:01 +01:00
static inline void superio_select ( int ldn )
2007-11-19 15:09:21 +01:00
{
outb ( LDN , REG ) ;
outb ( ldn , VAL ) ;
}
2011-05-09 11:45:07 -07:00
static inline int superio_enter ( void )
2007-11-19 15:09:21 +01:00
{
2011-05-09 11:45:07 -07:00
/*
* Try to reserve REG and REG + 1 for exclusive access .
*/
if ( ! request_muxed_region ( REG , 2 , NAME ) )
return - EBUSY ;
2007-11-19 15:09:21 +01:00
outb ( 0x87 , REG ) ;
outb ( 0x01 , REG ) ;
outb ( 0x55 , REG ) ;
outb ( 0x55 , REG ) ;
2011-05-09 11:45:07 -07:00
return 0 ;
2007-11-19 15:09:21 +01:00
}
2008-08-04 17:54:01 +01:00
static inline void superio_exit ( void )
2007-11-19 15:09:21 +01:00
{
outb ( 0x02 , REG ) ;
outb ( 0x02 , VAL ) ;
2011-05-09 11:45:07 -07:00
release_region ( REG , 2 ) ;
2007-11-19 15:09:21 +01:00
}
2008-08-04 17:54:01 +01:00
static inline void it8712f_wdt_ping ( void )
2007-11-19 15:09:21 +01:00
{
2010-09-30 17:08:04 +03:00
if ( wdt_control_reg & WDT_RESET_GAME )
inb ( address ) ;
2007-11-19 15:09:21 +01:00
}
2008-08-04 17:54:01 +01:00
static void it8712f_wdt_update_margin ( void )
2007-11-19 15:09:21 +01:00
{
int config = WDT_OUT_KRST | WDT_OUT_PWROK ;
2008-02-10 22:11:15 -05:00
int units = margin ;
/* Switch to minutes precision if the configured margin
* value does not fit within the register width .
*/
if ( units < = max_units ) {
config | = WDT_UNIT_SEC ; /* else UNIT is MINUTES */
2012-02-15 15:06:19 -08:00
pr_info ( " timer margin %d seconds \n " , units ) ;
2008-02-10 22:11:15 -05:00
} else {
units / = 60 ;
2012-02-15 15:06:19 -08:00
pr_info ( " timer margin %d minutes \n " , units ) ;
2008-02-10 22:11:15 -05:00
}
2007-11-19 15:09:21 +01:00
superio_outb ( config , WDT_CONFIG ) ;
2008-02-10 22:11:15 -05:00
if ( revision > = 0x08 )
2008-04-01 17:06:21 +02:00
superio_outb ( units > > 8 , WDT_TIMEOUT + 1 ) ;
superio_outb ( units , WDT_TIMEOUT ) ;
2008-02-10 22:11:15 -05:00
}
2008-08-04 17:54:01 +01:00
static int it8712f_wdt_get_status ( void )
2008-02-10 22:11:15 -05:00
{
if ( superio_inb ( WDT_CONTROL ) & 0x01 )
return WDIOF_CARDRESET ;
else
return 0 ;
2007-11-19 15:09:21 +01:00
}
2011-05-09 11:45:07 -07:00
static int it8712f_wdt_enable ( void )
2007-11-19 15:09:21 +01:00
{
2011-05-09 11:45:07 -07:00
int ret = superio_enter ( ) ;
if ( ret )
return ret ;
2012-02-15 15:06:19 -08:00
pr_debug ( " enabling watchdog timer \n " ) ;
2007-11-19 15:09:21 +01:00
superio_select ( LDN_GPIO ) ;
2010-09-30 17:08:04 +03:00
superio_outb ( wdt_control_reg , WDT_CONTROL ) ;
2007-11-19 15:09:21 +01:00
it8712f_wdt_update_margin ( ) ;
superio_exit ( ) ;
it8712f_wdt_ping ( ) ;
2011-05-09 11:45:07 -07:00
return 0 ;
2007-11-19 15:09:21 +01:00
}
2011-05-09 11:45:07 -07:00
static int it8712f_wdt_disable ( void )
2007-11-19 15:09:21 +01:00
{
2011-05-09 11:45:07 -07:00
int ret = superio_enter ( ) ;
if ( ret )
return ret ;
2007-11-19 15:09:21 +01:00
2012-02-15 15:06:19 -08:00
pr_debug ( " disabling watchdog timer \n " ) ;
2007-11-19 15:09:21 +01:00
superio_select ( LDN_GPIO ) ;
superio_outb ( 0 , WDT_CONFIG ) ;
superio_outb ( 0 , WDT_CONTROL ) ;
2008-04-02 02:43:19 -04:00
if ( revision > = 0x08 )
superio_outb ( 0 , WDT_TIMEOUT + 1 ) ;
2007-11-19 15:09:21 +01:00
superio_outb ( 0 , WDT_TIMEOUT ) ;
superio_exit ( ) ;
2011-05-09 11:45:07 -07:00
return 0 ;
2007-11-19 15:09:21 +01:00
}
2008-08-04 17:54:01 +01:00
static int it8712f_wdt_notify ( struct notifier_block * this ,
2007-11-19 15:09:21 +01:00
unsigned long code , void * unused )
{
if ( code = = SYS_HALT | | code = = SYS_POWER_OFF )
if ( ! nowayout )
it8712f_wdt_disable ( ) ;
return NOTIFY_DONE ;
}
static struct notifier_block it8712f_wdt_notifier = {
. notifier_call = it8712f_wdt_notify ,
} ;
2008-08-04 17:54:01 +01:00
static ssize_t it8712f_wdt_write ( struct file * file , const char __user * data ,
size_t len , loff_t * ppos )
2007-11-19 15:09:21 +01:00
{
/* check for a magic close character */
if ( len ) {
size_t i ;
it8712f_wdt_ping ( ) ;
expect_close = 0 ;
for ( i = 0 ; i < len ; + + i ) {
char c ;
2008-08-06 20:19:41 +00:00
if ( get_user ( c , data + i ) )
2007-11-19 15:09:21 +01:00
return - EFAULT ;
if ( c = = ' V ' )
expect_close = 42 ;
}
}
return len ;
}
2008-08-04 17:54:01 +01:00
static long it8712f_wdt_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
2007-11-19 15:09:21 +01: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 = {
2007-11-19 15:09:21 +01:00
. identity = " IT8712F Watchdog " ,
. firmware_version = 1 ,
2009-05-11 18:33:00 +00:00
. options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING |
WDIOF_MAGICCLOSE ,
2007-11-19 15:09:21 +01:00
} ;
2008-02-10 22:11:15 -05:00
int value ;
2011-05-09 11:45:07 -07:00
int ret ;
2007-11-19 15:09:21 +01:00
switch ( cmd ) {
case WDIOC_GETSUPPORT :
if ( copy_to_user ( argp , & ident , sizeof ( ident ) ) )
return - EFAULT ;
return 0 ;
case WDIOC_GETSTATUS :
2011-05-09 11:45:07 -07:00
ret = superio_enter ( ) ;
if ( ret )
return ret ;
2008-02-10 22:11:15 -05:00
superio_select ( LDN_GPIO ) ;
value = it8712f_wdt_get_status ( ) ;
superio_exit ( ) ;
return put_user ( value , p ) ;
2007-11-19 15:09:21 +01:00
case WDIOC_GETBOOTSTATUS :
return put_user ( 0 , p ) ;
case WDIOC_KEEPALIVE :
it8712f_wdt_ping ( ) ;
return 0 ;
case WDIOC_SETTIMEOUT :
2008-02-10 22:11:15 -05:00
if ( get_user ( value , p ) )
2007-11-19 15:09:21 +01:00
return - EFAULT ;
2008-02-10 22:11:15 -05:00
if ( value < 1 )
return - EINVAL ;
if ( value > ( max_units * 60 ) )
2007-11-19 15:09:21 +01:00
return - EINVAL ;
2008-02-10 22:11:15 -05:00
margin = value ;
2011-05-09 11:45:07 -07:00
ret = superio_enter ( ) ;
if ( ret )
return ret ;
2007-11-19 15:09:21 +01:00
superio_select ( LDN_GPIO ) ;
it8712f_wdt_update_margin ( ) ;
superio_exit ( ) ;
it8712f_wdt_ping ( ) ;
2020-07-07 12:11:21 -05:00
fallthrough ;
2007-11-19 15:09:21 +01:00
case WDIOC_GETTIMEOUT :
if ( put_user ( margin , p ) )
return - EFAULT ;
return 0 ;
2008-07-18 11:41:17 +00:00
default :
return - ENOTTY ;
2007-11-19 15:09:21 +01:00
}
}
2008-08-04 17:54:01 +01:00
static int it8712f_wdt_open ( struct inode * inode , struct file * file )
2007-11-19 15:09:21 +01:00
{
2011-05-09 11:45:07 -07:00
int ret ;
2007-11-19 15:09:21 +01:00
/* only allow one at a time */
2008-08-04 17:54:01 +01:00
if ( test_and_set_bit ( 0 , & wdt_open ) )
2007-11-19 15:09:21 +01:00
return - EBUSY ;
2011-05-09 11:45:07 -07:00
ret = it8712f_wdt_enable ( ) ;
if ( ret )
return ret ;
2019-03-26 23:51:19 +03:00
return stream_open ( inode , file ) ;
2007-11-19 15:09:21 +01:00
}
2008-08-04 17:54:01 +01:00
static int it8712f_wdt_release ( struct inode * inode , struct file * file )
2007-11-19 15:09:21 +01:00
{
if ( expect_close ! = 42 ) {
2012-02-15 15:06:19 -08:00
pr_warn ( " watchdog device closed unexpectedly, will not disable the watchdog timer \n " ) ;
2007-11-19 15:09:21 +01:00
} else if ( ! nowayout ) {
2011-05-09 11:45:07 -07:00
if ( it8712f_wdt_disable ( ) )
2012-02-15 15:06:19 -08:00
pr_warn ( " Watchdog disable failed \n " ) ;
2007-11-19 15:09:21 +01:00
}
expect_close = 0 ;
2008-08-04 17:54:01 +01:00
clear_bit ( 0 , & wdt_open ) ;
2007-11-19 15:09:21 +01:00
return 0 ;
}
2008-01-22 20:48:10 +01:00
static const struct file_operations it8712f_wdt_fops = {
2007-11-19 15:09:21 +01:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = it8712f_wdt_write ,
2008-08-04 17:54:01 +01:00
. unlocked_ioctl = it8712f_wdt_ioctl ,
2019-06-03 14:23:09 +02:00
. compat_ioctl = compat_ptr_ioctl ,
2007-11-19 15:09:21 +01:00
. open = it8712f_wdt_open ,
. release = it8712f_wdt_release ,
} ;
static struct miscdevice it8712f_wdt_miscdev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & it8712f_wdt_fops ,
} ;
2008-08-04 17:54:01 +01:00
static int __init it8712f_wdt_find ( unsigned short * address )
2007-11-19 15:09:21 +01:00
{
int err = - ENODEV ;
int chip_type ;
2011-05-09 11:45:07 -07:00
int ret = superio_enter ( ) ;
if ( ret )
return ret ;
2007-11-19 15:09:21 +01:00
chip_type = superio_inw ( DEVID ) ;
if ( chip_type ! = IT8712F_DEVID )
goto exit ;
superio_select ( LDN_GAME ) ;
superio_outb ( 1 , ACT_REG ) ;
if ( ! ( superio_inb ( ACT_REG ) & 0x01 ) ) {
2012-02-15 15:06:19 -08:00
pr_err ( " Device not activated, skipping \n " ) ;
2007-11-19 15:09:21 +01:00
goto exit ;
}
* address = superio_inw ( BASE_REG ) ;
if ( * address = = 0 ) {
2012-02-15 15:06:19 -08:00
pr_err ( " Base address not set, skipping \n " ) ;
2007-11-19 15:09:21 +01:00
goto exit ;
}
err = 0 ;
2008-02-10 22:11:15 -05:00
revision = superio_inb ( DEVREV ) & 0x0f ;
/* Later revisions have 16-bit values per datasheet 0.9.1 */
if ( revision > = 0x08 )
max_units = 65535 ;
if ( margin > ( max_units * 60 ) )
margin = ( max_units * 60 ) ;
2012-02-15 15:06:19 -08:00
pr_info ( " Found IT%04xF chip revision %d - using DogFood address 0x%x \n " ,
2008-02-10 22:11:15 -05:00
chip_type , revision , * address ) ;
2007-11-19 15:09:21 +01:00
exit :
superio_exit ( ) ;
return err ;
}
2008-08-04 17:54:01 +01:00
static int __init it8712f_wdt_init ( void )
2007-11-19 15:09:21 +01:00
{
int err = 0 ;
if ( it8712f_wdt_find ( & address ) )
return - ENODEV ;
if ( ! request_region ( address , 1 , " IT8712F Watchdog " ) ) {
2012-02-15 15:06:19 -08:00
pr_warn ( " watchdog I/O region busy \n " ) ;
2007-11-19 15:09:21 +01:00
return - EBUSY ;
}
2011-05-09 11:45:07 -07:00
err = it8712f_wdt_disable ( ) ;
if ( err ) {
2012-02-15 15:06:19 -08:00
pr_err ( " unable to disable watchdog timer \n " ) ;
2011-05-09 11:45:07 -07:00
goto out ;
}
2007-11-19 15:09:21 +01:00
err = register_reboot_notifier ( & it8712f_wdt_notifier ) ;
if ( err ) {
2012-02-15 15:06:19 -08:00
pr_err ( " unable to register reboot notifier \n " ) ;
2007-11-19 15:09:21 +01:00
goto out ;
}
err = misc_register ( & it8712f_wdt_miscdev ) ;
if ( err ) {
2012-02-15 15:06:19 -08:00
pr_err ( " cannot register miscdev on minor=%d (err=%d) \n " ,
WATCHDOG_MINOR , err ) ;
2007-11-19 15:09:21 +01:00
goto reboot_out ;
}
return 0 ;
reboot_out :
unregister_reboot_notifier ( & it8712f_wdt_notifier ) ;
out :
release_region ( address , 1 ) ;
return err ;
}
2008-08-04 17:54:01 +01:00
static void __exit it8712f_wdt_exit ( void )
2007-11-19 15:09:21 +01:00
{
misc_deregister ( & it8712f_wdt_miscdev ) ;
unregister_reboot_notifier ( & it8712f_wdt_notifier ) ;
release_region ( address , 1 ) ;
}
module_init ( it8712f_wdt_init ) ;
module_exit ( it8712f_wdt_exit ) ;