2006-05-21 16:37:44 +04:00
/*
* intel TCO Watchdog Driver ( Used in i82801 and i6300ESB chipsets )
*
2008-11-19 22:39:58 +03:00
* ( c ) Copyright 2006 - 2008 Wim Van Sebroeck < wim @ iguana . be > .
2006-05-21 16:37:44 +04:00
*
* 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 .
*
* Neither Wim Van Sebroeck nor Iguana vzw . admit liability nor
* provide warranty for any of this software . This material is
* provided " AS-IS " and at no charge .
*
* The TCO watchdog is implemented in the following I / O controller hubs :
* ( See the intel documentation on http : //developer.intel.com.)
* 82801 AA ( ICH ) : document number 290655 - 003 , 290677 - 014 ,
* 82801 AB ( ICHO ) : document number 290655 - 003 , 290677 - 014 ,
* 82801 BA ( ICH2 ) : document number 290687 - 002 , 298242 - 027 ,
* 82801 BAM ( ICH2 - M ) : document number 290687 - 002 , 298242 - 027 ,
* 82801 CA ( ICH3 - S ) : document number 290733 - 003 , 290739 - 013 ,
* 82801 CAM ( ICH3 - M ) : document number 290716 - 001 , 290718 - 007 ,
* 82801 DB ( ICH4 ) : document number 290744 - 001 , 290745 - 020 ,
* 82801 DBM ( ICH4 - M ) : document number 252337 - 001 , 252663 - 005 ,
* 82801 E ( C - ICH ) : document number 273599 - 001 , 273645 - 002 ,
* 82801 EB ( ICH5 ) : document number 252516 - 001 , 252517 - 003 ,
* 82801 ER ( ICH5R ) : document number 252516 - 001 , 252517 - 003 ,
* 82801F B ( ICH6 ) : document number 301473 - 002 , 301474 - 007 ,
* 82801F R ( ICH6R ) : document number 301473 - 002 , 301474 - 007 ,
* 82801F BM ( ICH6 - M ) : document number 301473 - 002 , 301474 - 007 ,
* 82801F W ( ICH6W ) : document number 301473 - 001 , 301474 - 007 ,
* 82801F RW ( ICH6RW ) : document number 301473 - 001 , 301474 - 007 ,
* 82801 GB ( ICH7 ) : document number 307013 - 002 , 307014 - 00 9 ,
* 82801 GR ( ICH7R ) : document number 307013 - 002 , 307014 - 00 9 ,
* 82801 GDH ( ICH7DH ) : document number 307013 - 002 , 307014 - 00 9 ,
* 82801 GBM ( ICH7 - M ) : document number 307013 - 002 , 307014 - 00 9 ,
* 82801 GHM ( ICH7 - M DH ) : document number 307013 - 002 , 307014 - 00 9 ,
2007-08-31 12:23:10 +04:00
* 82801 HB ( ICH8 ) : document number 313056 - 003 , 313057 - 00 9 ,
* 82801 HR ( ICH8R ) : document number 313056 - 003 , 313057 - 00 9 ,
* 82801 HBM ( ICH8M ) : document number 313056 - 003 , 313057 - 00 9 ,
* 82801 HH ( ICH8DH ) : document number 313056 - 003 , 313057 - 00 9 ,
* 82801 HO ( ICH8DO ) : document number 313056 - 003 , 313057 - 00 9 ,
* 82801 HEM ( ICH8M - E ) : document number 313056 - 003 , 313057 - 00 9 ,
2008-04-30 18:51:10 +04:00
* 82801 IB ( ICH9 ) : document number 316972 - 001 , 316973 - 006 ,
* 82801 IR ( ICH9R ) : document number 316972 - 001 , 316973 - 006 ,
* 82801 IH ( ICH9DH ) : document number 316972 - 001 , 316973 - 006 ,
* 82801 IO ( ICH9DO ) : document number 316972 - 001 , 316973 - 006 ,
2007-07-27 01:11:28 +04:00
* 6300 ESB ( 6300 ESB ) : document number 300641 - 003 , 300884 - 010 ,
2007-07-26 18:28:35 +04:00
* 631 xESB ( 631 xESB ) : document number 313082 - 001 , 313075 - 005 ,
* 632 xESB ( 632 xESB ) : document number 313082 - 001 , 313075 - 005
2006-05-21 16:37:44 +04:00
*/
/*
* Includes , defines , variables , module parameters , . . .
*/
/* Module and version information */
2008-08-07 00:19:41 +04:00
# define DRV_NAME "iTCO_wdt"
2008-11-19 22:39:58 +03:00
# define DRV_VERSION "1.04"
2006-05-21 16:37:44 +04:00
# define PFX DRV_NAME ": "
/* Includes */
2006-06-30 10:44:53 +04:00
# include <linux/module.h> /* For module specific items */
# include <linux/moduleparam.h> /* For new moduleparam's */
# include <linux/types.h> /* For standard types (like size_t) */
# include <linux/errno.h> /* For the -ENODEV/... values */
# include <linux/kernel.h> /* For printk/panic/... */
2008-05-19 17:06:25 +04:00
# include <linux/miscdevice.h> /* For MODULE_ALIAS_MISCDEV
( WATCHDOG_MINOR ) */
2006-06-30 10:44:53 +04:00
# include <linux/watchdog.h> /* For the watchdog specific items */
# include <linux/init.h> /* For __init/__exit/... */
# include <linux/fs.h> /* For file operations */
# include <linux/platform_device.h> /* For platform_driver framework */
# include <linux/pci.h> /* For pci functions */
# include <linux/ioport.h> /* For io-port access */
# include <linux/spinlock.h> /* For spin_lock/spin_unlock/... */
2008-05-19 17:06:25 +04:00
# include <linux/uaccess.h> /* For copy_to_user/put_user/... */
# include <linux/io.h> /* For inb/outb/... */
2006-06-30 10:44:53 +04:00
2008-05-19 17:06:25 +04:00
# include "iTCO_vendor.h"
2006-05-21 16:37:44 +04:00
/* TCO related info */
enum iTCO_chipsets {
TCO_ICH = 0 , /* ICH */
TCO_ICH0 , /* ICH0 */
TCO_ICH2 , /* ICH2 */
TCO_ICH2M , /* ICH2-M */
TCO_ICH3 , /* ICH3-S */
TCO_ICH3M , /* ICH3-M */
TCO_ICH4 , /* ICH4 */
TCO_ICH4M , /* ICH4-M */
TCO_CICH , /* C-ICH */
TCO_ICH5 , /* ICH5 & ICH5R */
TCO_6300ESB , /* 6300ESB */
TCO_ICH6 , /* ICH6 & ICH6R */
TCO_ICH6M , /* ICH6-M */
TCO_ICH6W , /* ICH6W & ICH6RW */
TCO_ICH7 , /* ICH7 & ICH7R */
TCO_ICH7M , /* ICH7-M */
TCO_ICH7MDH , /* ICH7-M DH */
2006-10-08 23:05:21 +04:00
TCO_ICH8 , /* ICH8 & ICH8R */
2007-08-31 12:23:10 +04:00
TCO_ICH8ME , /* ICH8M-E */
2006-10-08 23:05:21 +04:00
TCO_ICH8DH , /* ICH8DH */
TCO_ICH8DO , /* ICH8DO */
2007-08-31 12:23:10 +04:00
TCO_ICH8M , /* ICH8M */
2007-07-27 01:11:28 +04:00
TCO_ICH9 , /* ICH9 */
TCO_ICH9R , /* ICH9R */
TCO_ICH9DH , /* ICH9DH */
2008-08-07 00:19:41 +04:00
TCO_ICH9DO , /* ICH9DO */
2007-07-26 18:28:35 +04:00
TCO_631XESB , /* 631xESB/632xESB */
2006-05-21 16:37:44 +04:00
} ;
static struct {
char * name ;
unsigned int iTCO_version ;
} iTCO_chipset_info [ ] __devinitdata = {
{ " ICH " , 1 } ,
{ " ICH0 " , 1 } ,
{ " ICH2 " , 1 } ,
{ " ICH2-M " , 1 } ,
{ " ICH3-S " , 1 } ,
{ " ICH3-M " , 1 } ,
{ " ICH4 " , 1 } ,
{ " ICH4-M " , 1 } ,
{ " C-ICH " , 1 } ,
{ " ICH5 or ICH5R " , 1 } ,
{ " 6300ESB " , 1 } ,
{ " ICH6 or ICH6R " , 2 } ,
{ " ICH6-M " , 2 } ,
{ " ICH6W or ICH6RW " , 2 } ,
{ " ICH7 or ICH7R " , 2 } ,
{ " ICH7-M " , 2 } ,
{ " ICH7-M DH " , 2 } ,
2006-10-04 16:18:29 +04:00
{ " ICH8 or ICH8R " , 2 } ,
2007-08-31 12:23:10 +04:00
{ " ICH8M-E " , 2 } ,
2006-10-08 23:05:21 +04:00
{ " ICH8DH " , 2 } ,
{ " ICH8DO " , 2 } ,
2007-08-31 12:23:10 +04:00
{ " ICH8M " , 2 } ,
2007-07-27 01:11:28 +04:00
{ " ICH9 " , 2 } ,
{ " ICH9R " , 2 } ,
{ " ICH9DH " , 2 } ,
2008-04-30 18:51:10 +04:00
{ " ICH9DO " , 2 } ,
2007-07-26 18:28:35 +04:00
{ " 631xESB/632xESB " , 2 } ,
2008-05-19 17:06:25 +04:00
{ NULL , 0 }
2006-05-21 16:37:44 +04:00
} ;
2007-08-20 00:17:58 +04:00
# define ITCO_PCI_DEVICE(dev, data) \
. vendor = PCI_VENDOR_ID_INTEL , \
. device = dev , \
. subvendor = PCI_ANY_ID , \
. subdevice = PCI_ANY_ID , \
. class = 0 , \
. class_mask = 0 , \
. driver_data = data
2006-05-21 16:37:44 +04:00
/*
* This data only exists for exporting the supported PCI ids
* via MODULE_DEVICE_TABLE . We do not actually register a
* pci_driver , because the I / O Controller Hub has also other
* functions that probably will be registered by other drivers .
*/
static struct pci_device_id iTCO_wdt_pci_tbl [ ] = {
2008-05-19 17:06:25 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801AA_0 , TCO_ICH ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801AB_0 , TCO_ICH0 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801BA_0 , TCO_ICH2 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801BA_10 , TCO_ICH2M ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801CA_0 , TCO_ICH3 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801CA_12 , TCO_ICH3M ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801DB_0 , TCO_ICH4 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801DB_12 , TCO_ICH4M ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801E_0 , TCO_CICH ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_82801EB_0 , TCO_ICH5 ) } ,
2007-08-20 00:17:58 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ESB_1 , TCO_6300ESB ) } ,
2008-05-19 17:06:25 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH6_0 , TCO_ICH6 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH6_1 , TCO_ICH6M ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH6_2 , TCO_ICH6W ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH7_0 , TCO_ICH7 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH7_1 , TCO_ICH7M ) } ,
2007-08-20 00:17:58 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH7_31 , TCO_ICH7MDH ) } ,
2008-05-19 17:06:25 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH8_0 , TCO_ICH8 ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH8_1 , TCO_ICH8ME ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH8_2 , TCO_ICH8DH ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH8_3 , TCO_ICH8DO ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH8_4 , TCO_ICH8M ) } ,
{ ITCO_PCI_DEVICE ( 0x2918 , TCO_ICH9 ) } ,
{ ITCO_PCI_DEVICE ( 0x2916 , TCO_ICH9R ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH9_2 , TCO_ICH9DH ) } ,
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ICH9_4 , TCO_ICH9DO ) } ,
2007-08-20 00:17:58 +04:00
{ ITCO_PCI_DEVICE ( PCI_DEVICE_ID_INTEL_ESB2_0 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2671 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2672 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2673 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2674 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2675 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2676 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2677 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2678 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x2679 , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267a , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267b , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267c , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267d , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267e , TCO_631XESB ) } ,
{ ITCO_PCI_DEVICE ( 0x267f , TCO_631XESB ) } ,
2006-05-21 16:37:44 +04:00
{ 0 , } , /* End of list */
} ;
2008-05-19 17:06:25 +04:00
MODULE_DEVICE_TABLE ( pci , iTCO_wdt_pci_tbl ) ;
2006-05-21 16:37:44 +04:00
/* Address definitions for the TCO */
2008-05-19 17:06:25 +04:00
/* TCO base address */
# define TCOBASE iTCO_wdt_private.ACPIBASE + 0x60
/* SMI Control and Enable Register */
# define SMI_EN iTCO_wdt_private.ACPIBASE + 0x30
2006-05-21 16:37:44 +04:00
2008-05-19 17:06:25 +04:00
# define TCO_RLD TCOBASE + 0x00 /* TCO Timer Reload and Curr. Value */
2006-05-21 16:37:44 +04:00
# define TCOv1_TMR TCOBASE + 0x01 /* TCOv1 Timer Initial Value */
# define TCO_DAT_IN TCOBASE + 0x02 /* TCO Data In Register */
# define TCO_DAT_OUT TCOBASE + 0x03 /* TCO Data Out Register */
# define TCO1_STS TCOBASE + 0x04 /* TCO1 Status Register */
# define TCO2_STS TCOBASE + 0x06 /* TCO2 Status Register */
# define TCO1_CNT TCOBASE + 0x08 /* TCO1 Control Register */
# define TCO2_CNT TCOBASE + 0x0a /* TCO2 Control Register */
# define TCOv2_TMR TCOBASE + 0x12 /* TCOv2 Timer Initial Value */
/* internal variables */
static unsigned long is_active ;
static char expect_release ;
2008-05-19 17:06:25 +04:00
static struct { /* this is private data for the iTCO_wdt device */
/* TCO version/generation */
unsigned int iTCO_version ;
/* The cards ACPIBASE address (TCOBASE = ACPIBASE+0x60) */
unsigned long ACPIBASE ;
/* NO_REBOOT flag is Memory-Mapped GCS register bit 5 (TCO version 2)*/
unsigned long __iomem * gcs ;
/* the lock for io operations */
spinlock_t io_lock ;
/* the PCI-device */
struct pci_dev * pdev ;
2006-05-21 16:37:44 +04:00
} iTCO_wdt_private ;
2008-05-19 17:06:25 +04:00
/* the watchdog platform device */
static struct platform_device * iTCO_wdt_platform_device ;
2006-06-30 10:44:53 +04:00
2006-05-21 16:37:44 +04:00
/* module parameters */
# define WATCHDOG_HEARTBEAT 30 /* 30 sec default heartbeat */
static int heartbeat = WATCHDOG_HEARTBEAT ; /* in seconds */
module_param ( heartbeat , int , 0 ) ;
MODULE_PARM_DESC ( heartbeat , " Watchdog heartbeat in seconds. (2<heartbeat<39 (TCO v1) or 613 (TCO v2), default= " __MODULE_STRING ( WATCHDOG_HEARTBEAT ) " ) " ) ;
static int nowayout = WATCHDOG_NOWAYOUT ;
module_param ( nowayout , int , 0 ) ;
2008-05-19 17:06:25 +04:00
MODULE_PARM_DESC ( nowayout ,
" Watchdog cannot be stopped once started (default= "
__MODULE_STRING ( WATCHDOG_NOWAYOUT ) " ) " ) ;
2006-11-12 20:05:09 +03:00
2006-05-21 16:37:44 +04:00
/*
* Some TCO specific functions
*/
static inline unsigned int seconds_to_ticks ( int seconds )
{
/* the internal timer is stored as ticks which decrement
* every 0.6 seconds */
return ( seconds * 10 ) / 6 ;
}
static void iTCO_wdt_set_NO_REBOOT_bit ( void )
{
u32 val32 ;
/* Set the NO_REBOOT bit: this disables reboots */
if ( iTCO_wdt_private . iTCO_version = = 2 ) {
val32 = readl ( iTCO_wdt_private . gcs ) ;
val32 | = 0x00000020 ;
writel ( val32 , iTCO_wdt_private . gcs ) ;
} else if ( iTCO_wdt_private . iTCO_version = = 1 ) {
pci_read_config_dword ( iTCO_wdt_private . pdev , 0xd4 , & val32 ) ;
val32 | = 0x00000002 ;
pci_write_config_dword ( iTCO_wdt_private . pdev , 0xd4 , val32 ) ;
}
}
static int iTCO_wdt_unset_NO_REBOOT_bit ( void )
{
int ret = 0 ;
u32 val32 ;
/* Unset the NO_REBOOT bit: this enables reboots */
if ( iTCO_wdt_private . iTCO_version = = 2 ) {
val32 = readl ( iTCO_wdt_private . gcs ) ;
val32 & = 0xffffffdf ;
writel ( val32 , iTCO_wdt_private . gcs ) ;
val32 = readl ( iTCO_wdt_private . gcs ) ;
if ( val32 & 0x00000020 )
ret = - EIO ;
} else if ( iTCO_wdt_private . iTCO_version = = 1 ) {
pci_read_config_dword ( iTCO_wdt_private . pdev , 0xd4 , & val32 ) ;
val32 & = 0xfffffffd ;
pci_write_config_dword ( iTCO_wdt_private . pdev , 0xd4 , val32 ) ;
pci_read_config_dword ( iTCO_wdt_private . pdev , 0xd4 , & val32 ) ;
if ( val32 & 0x00000002 )
ret = - EIO ;
}
return ret ; /* returns: 0 = OK, -EIO = Error */
}
static int iTCO_wdt_start ( void )
{
unsigned int val ;
2008-11-19 22:39:58 +03:00
unsigned long val32 ;
2006-05-21 16:37:44 +04:00
spin_lock ( & iTCO_wdt_private . io_lock ) ;
2006-11-12 20:05:09 +03:00
iTCO_vendor_pre_start ( iTCO_wdt_private . ACPIBASE , heartbeat ) ;
2006-05-21 16:37:44 +04:00
/* disable chipset's NO_REBOOT bit */
if ( iTCO_wdt_unset_NO_REBOOT_bit ( ) ) {
2007-10-23 05:08:27 +04:00
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
2006-05-21 16:37:44 +04:00
printk ( KERN_ERR PFX " failed to reset NO_REBOOT flag, reboot disabled by hardware \n " ) ;
return - EIO ;
}
2008-11-19 22:39:58 +03:00
/* Bit 13: TCO_EN -> 0 = Disables TCO logic generating an SMI# */
val32 = inl ( SMI_EN ) ;
val32 & = 0xffffdfff ; /* Turn off SMI clearing watchdog */
outl ( val32 , SMI_EN ) ;
/* Force the timer to its reload value by writing to the TCO_RLD
register */
if ( iTCO_wdt_private . iTCO_version = = 2 )
outw ( 0x01 , TCO_RLD ) ;
else if ( iTCO_wdt_private . iTCO_version = = 1 )
outb ( 0x01 , TCO_RLD ) ;
2006-05-21 16:37:44 +04:00
/* Bit 11: TCO Timer Halt -> 0 = The TCO timer is enabled to count */
val = inw ( TCO1_CNT ) ;
val & = 0xf7ff ;
outw ( val , TCO1_CNT ) ;
val = inw ( TCO1_CNT ) ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
if ( val & 0x0800 )
return - 1 ;
return 0 ;
}
static int iTCO_wdt_stop ( void )
{
unsigned int val ;
2008-11-19 22:39:58 +03:00
unsigned long val32 ;
2006-05-21 16:37:44 +04:00
spin_lock ( & iTCO_wdt_private . io_lock ) ;
2006-11-12 20:05:09 +03:00
iTCO_vendor_pre_stop ( iTCO_wdt_private . ACPIBASE ) ;
2006-05-21 16:37:44 +04:00
/* Bit 11: TCO Timer Halt -> 1 = The TCO timer is disabled */
val = inw ( TCO1_CNT ) ;
val | = 0x0800 ;
outw ( val , TCO1_CNT ) ;
val = inw ( TCO1_CNT ) ;
2008-11-19 22:39:58 +03:00
/* Bit 13: TCO_EN -> 1 = Enables the TCO logic to generate SMI# */
val32 = inl ( SMI_EN ) ;
val32 & = 0x00002000 ;
outl ( val32 , SMI_EN ) ;
2006-05-21 16:37:44 +04:00
/* Set the NO_REBOOT bit to prevent later reboots, just for sure */
iTCO_wdt_set_NO_REBOOT_bit ( ) ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
if ( ( val & 0x0800 ) = = 0 )
return - 1 ;
return 0 ;
}
static int iTCO_wdt_keepalive ( void )
{
spin_lock ( & iTCO_wdt_private . io_lock ) ;
2006-11-12 20:05:09 +03:00
iTCO_vendor_pre_keepalive ( iTCO_wdt_private . ACPIBASE , heartbeat ) ;
2006-05-21 16:37:44 +04:00
/* Reload the timer by writing to the TCO Timer Counter register */
2008-05-19 17:06:25 +04:00
if ( iTCO_wdt_private . iTCO_version = = 2 )
2006-05-21 16:37:44 +04:00
outw ( 0x01 , TCO_RLD ) ;
2008-05-19 17:06:25 +04:00
else if ( iTCO_wdt_private . iTCO_version = = 1 )
2006-05-21 16:37:44 +04:00
outb ( 0x01 , TCO_RLD ) ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
return 0 ;
}
static int iTCO_wdt_set_heartbeat ( int t )
{
unsigned int val16 ;
unsigned char val8 ;
unsigned int tmrval ;
tmrval = seconds_to_ticks ( t ) ;
/* from the specs: */
/* "Values of 0h-3h are ignored and should not be attempted" */
if ( tmrval < 0x04 )
return - EINVAL ;
if ( ( ( iTCO_wdt_private . iTCO_version = = 2 ) & & ( tmrval > 0x3ff ) ) | |
( ( iTCO_wdt_private . iTCO_version = = 1 ) & & ( tmrval > 0x03f ) ) )
return - EINVAL ;
2006-11-12 20:05:09 +03:00
iTCO_vendor_pre_set_heartbeat ( tmrval ) ;
2006-05-21 16:37:44 +04:00
/* Write new heartbeat to watchdog */
if ( iTCO_wdt_private . iTCO_version = = 2 ) {
spin_lock ( & iTCO_wdt_private . io_lock ) ;
val16 = inw ( TCOv2_TMR ) ;
val16 & = 0xfc00 ;
val16 | = tmrval ;
outw ( val16 , TCOv2_TMR ) ;
val16 = inw ( TCOv2_TMR ) ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
if ( ( val16 & 0x3ff ) ! = tmrval )
return - EINVAL ;
} else if ( iTCO_wdt_private . iTCO_version = = 1 ) {
spin_lock ( & iTCO_wdt_private . io_lock ) ;
val8 = inb ( TCOv1_TMR ) ;
val8 & = 0xc0 ;
val8 | = ( tmrval & 0xff ) ;
outb ( val8 , TCOv1_TMR ) ;
val8 = inb ( TCOv1_TMR ) ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
if ( ( val8 & 0x3f ) ! = tmrval )
return - EINVAL ;
}
heartbeat = t ;
return 0 ;
}
2008-05-19 17:06:25 +04:00
static int iTCO_wdt_get_timeleft ( int * time_left )
2006-05-21 16:37:44 +04:00
{
unsigned int val16 ;
unsigned char val8 ;
/* read the TCO Timer */
if ( iTCO_wdt_private . iTCO_version = = 2 ) {
spin_lock ( & iTCO_wdt_private . io_lock ) ;
val16 = inw ( TCO_RLD ) ;
val16 & = 0x3ff ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
* time_left = ( val16 * 6 ) / 10 ;
} else if ( iTCO_wdt_private . iTCO_version = = 1 ) {
spin_lock ( & iTCO_wdt_private . io_lock ) ;
val8 = inb ( TCO_RLD ) ;
val8 & = 0x3f ;
spin_unlock ( & iTCO_wdt_private . io_lock ) ;
* time_left = ( val8 * 6 ) / 10 ;
2006-10-10 11:40:44 +04:00
} else
return - EINVAL ;
2006-05-21 16:37:44 +04:00
return 0 ;
}
/*
* / dev / watchdog handling
*/
2008-05-19 17:06:25 +04:00
static int iTCO_wdt_open ( struct inode * inode , struct file * file )
2006-05-21 16:37:44 +04:00
{
/* /dev/watchdog can only be opened once */
if ( test_and_set_bit ( 0 , & is_active ) )
return - EBUSY ;
/*
* Reload and activate timer
*/
iTCO_wdt_start ( ) ;
return nonseekable_open ( inode , file ) ;
}
2008-05-19 17:06:25 +04:00
static int iTCO_wdt_release ( struct inode * inode , struct file * file )
2006-05-21 16:37:44 +04:00
{
/*
* Shut off the timer .
*/
if ( expect_release = = 42 ) {
iTCO_wdt_stop ( ) ;
} else {
2008-05-19 17:06:25 +04:00
printk ( KERN_CRIT PFX
" Unexpected close, not stopping watchdog! \n " ) ;
2006-05-21 16:37:44 +04:00
iTCO_wdt_keepalive ( ) ;
}
clear_bit ( 0 , & is_active ) ;
expect_release = 0 ;
return 0 ;
}
2008-05-19 17:06:25 +04:00
static ssize_t iTCO_wdt_write ( struct file * file , const char __user * data ,
size_t len , loff_t * ppos )
2006-05-21 16:37:44 +04:00
{
/* See if we got the magic character 'V' and reload the timer */
if ( len ) {
if ( ! nowayout ) {
size_t i ;
2008-05-19 17:06:25 +04:00
/* note: just in case someone wrote the magic
character five months ago . . . */
2006-05-21 16:37:44 +04:00
expect_release = 0 ;
2008-05-19 17:06:25 +04:00
/* scan to see whether or not we got the
magic character */
2006-05-21 16:37:44 +04:00
for ( i = 0 ; i ! = len ; i + + ) {
char c ;
2008-08-07 00:19:41 +04:00
if ( get_user ( c , data + i ) )
2006-05-21 16:37:44 +04:00
return - EFAULT ;
if ( c = = ' V ' )
expect_release = 42 ;
}
}
/* someone wrote to us, we should reload the timer */
iTCO_wdt_keepalive ( ) ;
}
return len ;
}
2008-05-19 17:06:25 +04:00
static long iTCO_wdt_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
2006-05-21 16:37:44 +04:00
{
int new_options , retval = - EINVAL ;
int new_heartbeat ;
void __user * argp = ( void __user * ) arg ;
int __user * p = argp ;
static struct watchdog_info ident = {
. options = WDIOF_SETTIMEOUT |
WDIOF_KEEPALIVEPING |
WDIOF_MAGICCLOSE ,
. firmware_version = 0 ,
. identity = DRV_NAME ,
} ;
switch ( cmd ) {
2008-05-19 17:06:25 +04:00
case WDIOC_GETSUPPORT :
return copy_to_user ( argp , & ident , sizeof ( ident ) ) ? - EFAULT : 0 ;
case WDIOC_GETSTATUS :
case WDIOC_GETBOOTSTATUS :
return put_user ( 0 , p ) ;
2006-05-21 16:37:44 +04:00
2008-05-19 17:06:25 +04:00
case WDIOC_SETOPTIONS :
{
if ( get_user ( new_options , p ) )
return - EFAULT ;
2006-05-21 16:37:44 +04:00
2008-05-19 17:06:25 +04:00
if ( new_options & WDIOS_DISABLECARD ) {
iTCO_wdt_stop ( ) ;
retval = 0 ;
2006-05-21 16:37:44 +04:00
}
2008-05-19 17:06:25 +04:00
if ( new_options & WDIOS_ENABLECARD ) {
2006-05-21 16:37:44 +04:00
iTCO_wdt_keepalive ( ) ;
2008-05-19 17:06:25 +04:00
iTCO_wdt_start ( ) ;
retval = 0 ;
2006-05-21 16:37:44 +04:00
}
2008-05-19 17:06:25 +04:00
return retval ;
}
2008-07-18 15:41:17 +04:00
case WDIOC_KEEPALIVE :
iTCO_wdt_keepalive ( ) ;
return 0 ;
2008-05-19 17:06:25 +04:00
case WDIOC_SETTIMEOUT :
{
if ( get_user ( new_heartbeat , p ) )
return - EFAULT ;
if ( iTCO_wdt_set_heartbeat ( new_heartbeat ) )
return - EINVAL ;
iTCO_wdt_keepalive ( ) ;
/* Fall */
}
case WDIOC_GETTIMEOUT :
return put_user ( heartbeat , p ) ;
case WDIOC_GETTIMELEFT :
{
int time_left ;
if ( iTCO_wdt_get_timeleft ( & time_left ) )
return - EINVAL ;
return put_user ( time_left , p ) ;
}
default :
return - ENOTTY ;
2006-05-21 16:37:44 +04:00
}
}
/*
* Kernel Interfaces
*/
2007-02-12 11:55:32 +03:00
static const struct file_operations iTCO_wdt_fops = {
2008-05-19 17:06:25 +04:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. write = iTCO_wdt_write ,
. unlocked_ioctl = iTCO_wdt_ioctl ,
. open = iTCO_wdt_open ,
. release = iTCO_wdt_release ,
2006-05-21 16:37:44 +04:00
} ;
static struct miscdevice iTCO_wdt_miscdev = {
. minor = WATCHDOG_MINOR ,
. name = " watchdog " ,
. fops = & iTCO_wdt_fops ,
} ;
/*
* Init & exit routines
*/
2008-05-19 17:06:25 +04:00
static int __devinit iTCO_wdt_init ( struct pci_dev * pdev ,
const struct pci_device_id * ent , struct platform_device * dev )
2006-05-21 16:37:44 +04:00
{
int ret ;
u32 base_address ;
unsigned long RCBA ;
/*
* Find the ACPI / PM base I / O address which is the base
* for the TCO registers ( TCOBASE = ACPIBASE + 0x60 )
* ACPIBASE is bits [ 15 : 7 ] from 0x40 - 0x43
*/
pci_read_config_dword ( pdev , 0x40 , & base_address ) ;
2007-05-11 22:59:24 +04:00
base_address & = 0x0000ff80 ;
2006-05-21 16:37:44 +04:00
if ( base_address = = 0x00000000 ) {
/* Something's wrong here, ACPIBASE has to be set */
printk ( KERN_ERR PFX " failed to get TCOBASE address \n " ) ;
2006-07-20 00:39:13 +04:00
pci_dev_put ( pdev ) ;
2006-05-21 16:37:44 +04:00
return - ENODEV ;
}
2008-05-19 17:06:25 +04:00
iTCO_wdt_private . iTCO_version =
iTCO_chipset_info [ ent - > driver_data ] . iTCO_version ;
2006-05-21 16:37:44 +04:00
iTCO_wdt_private . ACPIBASE = base_address ;
iTCO_wdt_private . pdev = pdev ;
2008-05-19 17:06:25 +04:00
/* Get the Memory-Mapped GCS register, we need it for the
NO_REBOOT flag ( TCO v2 ) . To get access to it you have to
read RCBA from PCI Config space 0xf0 and use it as base .
GCS = RCBA + ICH6_GCS ( 0x3410 ) . */
2006-05-21 16:37:44 +04:00
if ( iTCO_wdt_private . iTCO_version = = 2 ) {
pci_read_config_dword ( pdev , 0xf0 , & base_address ) ;
RCBA = base_address & 0xffffc000 ;
2008-05-19 17:06:25 +04:00
iTCO_wdt_private . gcs = ioremap ( ( RCBA + 0x3410 ) , 4 ) ;
2006-05-21 16:37:44 +04:00
}
/* Check chipset's NO_REBOOT bit */
2006-11-12 20:05:09 +03:00
if ( iTCO_wdt_unset_NO_REBOOT_bit ( ) & & iTCO_vendor_check_noreboot_on ( ) ) {
2006-05-21 16:37:44 +04:00
printk ( KERN_ERR PFX " failed to reset NO_REBOOT flag, reboot disabled by hardware \n " ) ;
ret = - ENODEV ; /* Cannot reset NO_REBOOT bit */
goto out ;
}
/* Set the NO_REBOOT bit to prevent later reboots, just for sure */
iTCO_wdt_set_NO_REBOOT_bit ( ) ;
2008-11-19 22:39:58 +03:00
/* The TCO logic uses the TCO_EN bit in the SMI_EN register */
2006-05-21 16:37:44 +04:00
if ( ! request_region ( SMI_EN , 4 , " iTCO_wdt " ) ) {
2008-05-19 17:06:25 +04:00
printk ( KERN_ERR PFX
" I/O address 0x%04lx already in use \n " , SMI_EN ) ;
2006-05-21 16:37:44 +04:00
ret = - EIO ;
goto out ;
}
2008-05-19 17:06:25 +04:00
/* The TCO I/O registers reside in a 32-byte range pointed to
by the TCOBASE value */
if ( ! request_region ( TCOBASE , 0x20 , " iTCO_wdt " ) ) {
printk ( KERN_ERR PFX " I/O address 0x%04lx already in use \n " ,
2006-05-21 16:37:44 +04:00
TCOBASE ) ;
ret = - EIO ;
2008-11-19 22:39:58 +03:00
goto unreg_smi_en ;
2006-05-21 16:37:44 +04:00
}
2008-05-19 17:06:25 +04:00
printk ( KERN_INFO PFX
" Found a %s TCO device (Version=%d, TCOBASE=0x%04lx) \n " ,
iTCO_chipset_info [ ent - > driver_data ] . name ,
iTCO_chipset_info [ ent - > driver_data ] . iTCO_version ,
TCOBASE ) ;
2006-05-21 16:37:44 +04:00
/* Clear out the (probably old) status */
2008-11-19 23:02:02 +03:00
outb ( 8 , TCO1_STS ) ; /* Clear the Time Out Status bit */
outb ( 2 , TCO2_STS ) ; /* Clear SECOND_TO_STS bit */
outb ( 4 , TCO2_STS ) ; /* Clear BOOT_STS bit */
2006-05-21 16:37:44 +04:00
/* Make sure the watchdog is not running */
iTCO_wdt_stop ( ) ;
2008-05-19 17:06:25 +04:00
/* Check that the heartbeat value is within it's range;
if not reset to the default */
2006-05-21 16:37:44 +04:00
if ( iTCO_wdt_set_heartbeat ( heartbeat ) ) {
iTCO_wdt_set_heartbeat ( WATCHDOG_HEARTBEAT ) ;
2008-05-19 17:06:25 +04:00
printk ( KERN_INFO PFX " heartbeat value must be 2 < heartbeat < 39 (TCO v1) or 613 (TCO v2), using %d \n " ,
heartbeat ) ;
2006-05-21 16:37:44 +04:00
}
ret = misc_register ( & iTCO_wdt_miscdev ) ;
if ( ret ! = 0 ) {
2008-05-19 17:06:25 +04:00
printk ( KERN_ERR PFX
" cannot register miscdev on minor=%d (err=%d) \n " ,
WATCHDOG_MINOR , ret ) ;
2006-08-05 22:59:01 +04:00
goto unreg_region ;
2006-05-21 16:37:44 +04:00
}
2008-05-19 17:06:25 +04:00
printk ( KERN_INFO PFX " initialized. heartbeat=%d sec (nowayout=%d) \n " ,
heartbeat , nowayout ) ;
2006-05-21 16:37:44 +04:00
return 0 ;
unreg_region :
2008-05-19 17:06:25 +04:00
release_region ( TCOBASE , 0x20 ) ;
2008-11-19 22:39:58 +03:00
unreg_smi_en :
release_region ( SMI_EN , 4 ) ;
2006-05-21 16:37:44 +04:00
out :
if ( iTCO_wdt_private . iTCO_version = = 2 )
iounmap ( iTCO_wdt_private . gcs ) ;
2006-07-20 00:39:13 +04:00
pci_dev_put ( iTCO_wdt_private . pdev ) ;
2006-08-05 22:59:01 +04:00
iTCO_wdt_private . ACPIBASE = 0 ;
2006-05-21 16:37:44 +04:00
return ret ;
}
2007-08-31 12:15:34 +04:00
static void __devexit iTCO_wdt_cleanup ( void )
2006-05-21 16:37:44 +04:00
{
/* Stop the timer before we leave */
if ( ! nowayout )
iTCO_wdt_stop ( ) ;
/* Deregister */
misc_deregister ( & iTCO_wdt_miscdev ) ;
release_region ( TCOBASE , 0x20 ) ;
2008-11-19 22:39:58 +03:00
release_region ( SMI_EN , 4 ) ;
2006-05-21 16:37:44 +04:00
if ( iTCO_wdt_private . iTCO_version = = 2 )
iounmap ( iTCO_wdt_private . gcs ) ;
2006-07-20 00:39:13 +04:00
pci_dev_put ( iTCO_wdt_private . pdev ) ;
2006-08-05 22:59:01 +04:00
iTCO_wdt_private . ACPIBASE = 0 ;
2006-05-21 16:37:44 +04:00
}
2007-08-31 12:15:34 +04:00
static int __devinit iTCO_wdt_probe ( struct platform_device * dev )
2006-05-21 16:37:44 +04:00
{
int found = 0 ;
struct pci_dev * pdev = NULL ;
const struct pci_device_id * ent ;
spin_lock_init ( & iTCO_wdt_private . io_lock ) ;
for_each_pci_dev ( pdev ) {
ent = pci_match_id ( iTCO_wdt_pci_tbl , pdev ) ;
if ( ent ) {
2006-06-30 10:44:53 +04:00
if ( ! ( iTCO_wdt_init ( pdev , ent , dev ) ) ) {
2006-05-21 16:37:44 +04:00
found + + ;
break ;
}
}
}
if ( ! found ) {
printk ( KERN_INFO PFX " No card detected \n " ) ;
return - ENODEV ;
}
return 0 ;
}
2007-08-31 12:15:34 +04:00
static int __devexit iTCO_wdt_remove ( struct platform_device * dev )
2006-05-21 16:37:44 +04:00
{
if ( iTCO_wdt_private . ACPIBASE )
iTCO_wdt_cleanup ( ) ;
2006-06-30 10:44:53 +04:00
return 0 ;
}
static void iTCO_wdt_shutdown ( struct platform_device * dev )
{
iTCO_wdt_stop ( ) ;
}
# define iTCO_wdt_suspend NULL
# define iTCO_wdt_resume NULL
static struct platform_driver iTCO_wdt_driver = {
. probe = iTCO_wdt_probe ,
2007-08-31 12:15:34 +04:00
. remove = __devexit_p ( iTCO_wdt_remove ) ,
2006-06-30 10:44:53 +04:00
. shutdown = iTCO_wdt_shutdown ,
. suspend = iTCO_wdt_suspend ,
. resume = iTCO_wdt_resume ,
. driver = {
. owner = THIS_MODULE ,
. name = DRV_NAME ,
} ,
} ;
static int __init iTCO_wdt_init_module ( void )
{
int err ;
2008-11-19 22:39:58 +03:00
printk ( KERN_INFO PFX " Intel TCO WatchDog Timer Driver v%s \n " ,
DRV_VERSION ) ;
2006-06-30 10:44:53 +04:00
err = platform_driver_register ( & iTCO_wdt_driver ) ;
if ( err )
return err ;
2008-05-19 17:06:25 +04:00
iTCO_wdt_platform_device = platform_device_register_simple ( DRV_NAME ,
- 1 , NULL , 0 ) ;
2006-06-30 10:44:53 +04:00
if ( IS_ERR ( iTCO_wdt_platform_device ) ) {
err = PTR_ERR ( iTCO_wdt_platform_device ) ;
goto unreg_platform_driver ;
}
return 0 ;
unreg_platform_driver :
platform_driver_unregister ( & iTCO_wdt_driver ) ;
return err ;
}
static void __exit iTCO_wdt_cleanup_module ( void )
{
platform_device_unregister ( iTCO_wdt_platform_device ) ;
platform_driver_unregister ( & iTCO_wdt_driver ) ;
2006-05-21 16:37:44 +04:00
printk ( KERN_INFO PFX " Watchdog Module Unloaded. \n " ) ;
}
module_init ( iTCO_wdt_init_module ) ;
module_exit ( iTCO_wdt_cleanup_module ) ;
MODULE_AUTHOR ( " Wim Van Sebroeck <wim@iguana.be> " ) ;
MODULE_DESCRIPTION ( " Intel TCO WatchDog Timer Driver " ) ;
2006-06-30 10:44:53 +04:00
MODULE_VERSION ( DRV_VERSION ) ;
2006-05-21 16:37:44 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_MISCDEV ( WATCHDOG_MINOR ) ;