2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2007-12-04 20:41:54 +03:00
/*
2015-12-14 23:22:09 +03:00
* HPE WatchDog Driver
2007-12-04 20:41:54 +03:00
* based on
*
* SoftDog 0.05 : A Software Watchdog Device
*
2018-02-26 06:22:19 +03:00
* ( c ) Copyright 2018 Hewlett Packard Enterprise Development LP
2015-12-14 23:22:09 +03:00
* Thomas Mingarelli < thomas . mingarelli @ hpe . com >
2007-12-04 20:41:54 +03:00
*/
2012-02-16 03:06:19 +04:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2007-12-04 20:41:54 +03:00
# include <linux/device.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/pci.h>
# include <linux/pci_ids.h>
# include <linux/types.h>
# include <linux/watchdog.h>
2011-10-06 16:20:27 +04:00
# include <asm/nmi.h>
2020-11-23 05:08:39 +03:00
# include <linux/crash_dump.h>
2007-12-04 20:41:54 +03:00
2020-11-23 05:08:40 +03:00
# define HPWDT_VERSION "2.0.4"
2010-06-03 02:23:39 +04:00
# define SECS_TO_TICKS(secs) ((secs) * 1000 / 128)
2010-06-03 02:23:40 +04:00
# define TICKS_TO_SECS(ticks) ((ticks) * 128 / 1000)
2019-05-17 23:59:41 +03:00
# define HPWDT_MAX_TICKS 65535
# define HPWDT_MAX_TIMER TICKS_TO_SECS(HPWDT_MAX_TICKS)
2010-07-28 03:50:54 +04:00
# define DEFAULT_MARGIN 30
2018-02-26 06:22:25 +03:00
# define PRETIMEOUT_SEC 9
2010-07-28 03:50:54 +04:00
2018-02-26 06:22:23 +03:00
static bool ilo5 ;
2010-07-28 03:50:54 +04:00
static unsigned int soft_margin = DEFAULT_MARGIN ; /* in seconds */
2012-03-05 19:51:11 +04:00
static bool nowayout = WATCHDOG_NOWAYOUT ;
2018-02-26 06:22:25 +03:00
static bool pretimeout = IS_ENABLED ( CONFIG_HPWDT_NMI_DECODING ) ;
2019-05-17 23:59:41 +03:00
static int kdumptimeout = - 1 ;
2010-07-28 03:50:54 +04:00
static void __iomem * pci_mem_addr ; /* the PCI-memory address */
2017-10-24 01:46:17 +03:00
static unsigned long __iomem * hpwdt_nmistat ;
2010-07-28 03:50:54 +04:00
static unsigned long __iomem * hpwdt_timer_reg ;
static unsigned long __iomem * hpwdt_timer_con ;
2013-12-03 03:30:22 +04:00
static const struct pci_device_id hpwdt_devices [ ] = {
2010-07-28 03:50:57 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_COMPAQ , 0xB203 ) } , /* iLO2 */
{ PCI_DEVICE ( PCI_VENDOR_ID_HP , 0x3306 ) } , /* iLO3 */
2021-05-13 07:03:32 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_HP_3PAR , 0x0389 ) } , /* PCtrl */
2010-07-28 03:50:54 +04:00
{ 0 } , /* terminate list */
} ;
MODULE_DEVICE_TABLE ( pci , hpwdt_devices ) ;
2018-12-06 03:42:21 +03:00
static const struct pci_device_id hpwdt_blacklist [ ] = {
{ PCI_DEVICE_SUB ( PCI_VENDOR_ID_HP , 0x3306 , PCI_VENDOR_ID_HP , 0x1979 ) } , /* auxilary iLO */
2018-12-06 03:42:22 +03:00
{ PCI_DEVICE_SUB ( PCI_VENDOR_ID_HP , 0x3306 , PCI_VENDOR_ID_HP_3PAR , 0x0289 ) } , /* CL */
2018-12-06 03:42:21 +03:00
{ 0 } , /* terminate list */
} ;
2007-12-04 20:41:54 +03:00
2019-05-17 23:59:41 +03:00
static struct watchdog_device hpwdt_dev ;
2007-12-04 20:41:54 +03:00
/*
* Watchdog operations
*/
2019-05-17 23:59:40 +03:00
static int hpwdt_hw_is_running ( void )
{
return ioread8 ( hpwdt_timer_con ) & 0x01 ;
}
2018-02-26 06:22:22 +03:00
static int hpwdt_start ( struct watchdog_device * wdd )
2007-12-04 20:41:54 +03:00
{
2018-02-26 06:22:25 +03:00
int control = 0x81 | ( pretimeout ? 0x4 : 0 ) ;
2019-05-17 23:59:39 +03:00
int reload = SECS_TO_TICKS ( min ( wdd - > timeout , wdd - > max_hw_heartbeat_ms / 1000 ) ) ;
2018-02-26 06:22:22 +03:00
2019-05-17 23:59:39 +03:00
dev_dbg ( wdd - > parent , " start watchdog 0x%08x:0x%08x:0x%02x \n " , wdd - > timeout , reload , control ) ;
2007-12-04 20:41:54 +03:00
iowrite16 ( reload , hpwdt_timer_reg ) ;
2018-02-26 06:22:25 +03:00
iowrite8 ( control , hpwdt_timer_con ) ;
2018-02-26 06:22:22 +03:00
return 0 ;
2007-12-04 20:41:54 +03:00
}
static void hpwdt_stop ( void )
{
unsigned long data ;
2018-02-26 06:22:26 +03:00
pr_debug ( " stop watchdog \n " ) ;
2012-04-03 09:37:01 +04:00
data = ioread8 ( hpwdt_timer_con ) ;
2007-12-04 20:41:54 +03:00
data & = 0xFE ;
2012-04-03 09:37:01 +04:00
iowrite8 ( data , hpwdt_timer_con ) ;
2007-12-04 20:41:54 +03:00
}
2018-02-26 06:22:22 +03:00
static int hpwdt_stop_core ( struct watchdog_device * wdd )
2007-12-04 20:41:54 +03:00
{
2018-02-26 06:22:22 +03:00
hpwdt_stop ( ) ;
return 0 ;
2007-12-04 20:41:54 +03:00
}
2019-05-17 23:59:41 +03:00
static void hpwdt_ping_ticks ( int val )
{
val = min ( val , HPWDT_MAX_TICKS ) ;
iowrite16 ( val , hpwdt_timer_reg ) ;
}
2018-02-26 06:22:22 +03:00
static int hpwdt_ping ( struct watchdog_device * wdd )
2007-12-04 20:41:54 +03:00
{
2019-05-17 23:59:39 +03:00
int reload = SECS_TO_TICKS ( min ( wdd - > timeout , wdd - > max_hw_heartbeat_ms / 1000 ) ) ;
2018-02-26 06:22:25 +03:00
2019-05-17 23:59:39 +03:00
dev_dbg ( wdd - > parent , " ping watchdog 0x%08x:0x%08x \n " , wdd - > timeout , reload ) ;
2019-05-17 23:59:41 +03:00
hpwdt_ping_ticks ( reload ) ;
2018-02-26 06:22:25 +03:00
2007-12-04 20:41:54 +03:00
return 0 ;
}
2018-02-26 06:22:22 +03:00
static unsigned int hpwdt_gettimeleft ( struct watchdog_device * wdd )
2010-06-03 02:23:41 +04:00
{
return TICKS_TO_SECS ( ioread16 ( hpwdt_timer_reg ) ) ;
}
2018-02-26 06:22:22 +03:00
static int hpwdt_settimeout ( struct watchdog_device * wdd , unsigned int val )
{
2018-02-26 06:22:26 +03:00
dev_dbg ( wdd - > parent , " set_timeout = %d \n " , val ) ;
2018-02-26 06:22:22 +03:00
wdd - > timeout = val ;
2018-02-26 06:22:25 +03:00
if ( val < = wdd - > pretimeout ) {
2018-02-26 06:22:26 +03:00
dev_dbg ( wdd - > parent , " pretimeout < timeout. Setting to zero \n " ) ;
2018-02-26 06:22:25 +03:00
wdd - > pretimeout = 0 ;
2021-01-20 10:48:10 +03:00
pretimeout = false ;
2018-02-26 06:22:25 +03:00
if ( watchdog_active ( wdd ) )
hpwdt_start ( wdd ) ;
}
2018-02-26 06:22:22 +03:00
hpwdt_ping ( wdd ) ;
return 0 ;
}
2017-12-07 00:02:37 +03:00
# ifdef CONFIG_HPWDT_NMI_DECODING
2018-02-26 06:22:25 +03:00
static int hpwdt_set_pretimeout ( struct watchdog_device * wdd , unsigned int req )
{
unsigned int val = 0 ;
2018-02-26 06:22:26 +03:00
dev_dbg ( wdd - > parent , " set_pretimeout = %d \n " , req ) ;
2018-02-26 06:22:25 +03:00
if ( req ) {
val = PRETIMEOUT_SEC ;
if ( val > = wdd - > timeout )
return - EINVAL ;
}
2018-02-26 06:22:26 +03:00
if ( val ! = req )
dev_dbg ( wdd - > parent , " Rounding pretimeout to: %d \n " , val ) ;
2018-02-26 06:22:25 +03:00
wdd - > pretimeout = val ;
pretimeout = ! ! val ;
if ( watchdog_active ( wdd ) )
hpwdt_start ( wdd ) ;
return 0 ;
}
2017-10-24 01:46:17 +03:00
static int hpwdt_my_nmi ( void )
{
return ioread8 ( hpwdt_nmistat ) & 0x6 ;
}
2008-07-15 23:40:41 +04:00
/*
* NMI Handler
*/
2011-09-30 23:06:21 +04:00
static int hpwdt_pretimeout ( unsigned int ulReason , struct pt_regs * regs )
2008-07-15 23:40:41 +04:00
{
2018-02-26 06:22:21 +03:00
unsigned int mynmi = hpwdt_my_nmi ( ) ;
static char panic_msg [ ] =
" 00: An NMI occurred. Depending on your system the reason "
" for the NMI is logged in any one of the following resources: \n "
" 1. Integrated Management Log (IML) \n "
" 2. OA Syslog \n "
" 3. OA Forward Progress Log \n "
" 4. iLO Event Log " ;
2018-05-04 00:00:55 +03:00
if ( ilo5 & & ulReason = = NMI_UNKNOWN & & ! mynmi )
2017-10-24 01:46:17 +03:00
return NMI_DONE ;
2018-08-08 22:13:24 +03:00
if ( ilo5 & & ! pretimeout & & ! mynmi )
2018-02-26 06:22:25 +03:00
return NMI_DONE ;
2019-05-17 23:59:41 +03:00
if ( kdumptimeout < 0 )
hpwdt_stop ( ) ;
else if ( kdumptimeout = = 0 )
;
else {
unsigned int val = max ( ( unsigned int ) kdumptimeout , hpwdt_dev . timeout ) ;
hpwdt_ping_ticks ( SECS_TO_TICKS ( val ) ) ;
}
2011-08-10 02:27:26 +04:00
2018-02-26 06:22:21 +03:00
hex_byte_pack ( panic_msg , mynmi ) ;
nmi_panic ( regs , panic_msg ) ;
2011-07-26 17:05:53 +04:00
2016-03-23 00:27:24 +03:00
return NMI_HANDLED ;
2008-07-15 23:40:41 +04:00
}
2010-07-28 03:51:02 +04:00
# endif /* CONFIG_HPWDT_NMI_DECODING */
2008-07-15 23:40:41 +04:00
2007-12-04 20:41:54 +03:00
2009-12-26 21:55:22 +03:00
static const struct watchdog_info ident = {
2018-02-26 06:22:25 +03:00
. options = WDIOF_PRETIMEOUT |
WDIOF_SETTIMEOUT |
2007-12-04 20:41:54 +03:00
WDIOF_KEEPALIVEPING |
WDIOF_MAGICCLOSE ,
2015-12-14 23:22:09 +03:00
. identity = " HPE iLO2+ HW Watchdog Timer " ,
2007-12-04 20:41:54 +03:00
} ;
/*
* Kernel interfaces
*/
2018-02-26 06:22:22 +03:00
static const struct watchdog_ops hpwdt_ops = {
. owner = THIS_MODULE ,
. start = hpwdt_start ,
. stop = hpwdt_stop_core ,
. ping = hpwdt_ping ,
. set_timeout = hpwdt_settimeout ,
. get_timeleft = hpwdt_gettimeleft ,
2018-02-26 06:22:25 +03:00
# ifdef CONFIG_HPWDT_NMI_DECODING
. set_pretimeout = hpwdt_set_pretimeout ,
# endif
2007-12-04 20:41:54 +03:00
} ;
2018-02-26 06:22:22 +03:00
static struct watchdog_device hpwdt_dev = {
. info = & ident ,
. ops = & hpwdt_ops ,
. min_timeout = 1 ,
. timeout = DEFAULT_MARGIN ,
2018-02-26 06:22:25 +03:00
. pretimeout = PRETIMEOUT_SEC ,
2019-05-17 23:59:39 +03:00
. max_hw_heartbeat_ms = HPWDT_MAX_TIMER * 1000 ,
2007-12-04 20:41:54 +03:00
} ;
2018-02-26 06:22:22 +03:00
2007-12-04 20:41:54 +03:00
/*
* Init & Exit
*/
2012-11-19 22:21:41 +04:00
static int hpwdt_init_nmi_decoding ( struct pci_dev * dev )
2010-07-28 22:38:43 +04:00
{
2018-02-26 06:22:20 +03:00
# ifdef CONFIG_HPWDT_NMI_DECODING
2010-07-28 22:38:43 +04:00
int retval ;
/*
2012-03-30 00:11:15 +04:00
* Only one function can register for NMI_UNKNOWN
2010-07-28 22:38:43 +04:00
*/
2012-03-30 00:11:15 +04:00
retval = register_nmi_handler ( NMI_UNKNOWN , hpwdt_pretimeout , 0 , " hpwdt " ) ;
2012-03-30 00:11:16 +04:00
if ( retval )
goto error ;
retval = register_nmi_handler ( NMI_SERR , hpwdt_pretimeout , 0 , " hpwdt " ) ;
if ( retval )
goto error1 ;
retval = register_nmi_handler ( NMI_IO_CHECK , hpwdt_pretimeout , 0 , " hpwdt " ) ;
if ( retval )
goto error2 ;
2010-07-28 22:38:43 +04:00
dev_info ( & dev - > dev ,
2018-02-26 06:22:24 +03:00
" HPE Watchdog Timer Driver: NMI decoding initialized \n " ) ;
2010-07-28 22:38:43 +04:00
return 0 ;
2012-03-30 00:11:16 +04:00
error2 :
unregister_nmi_handler ( NMI_SERR , " hpwdt " ) ;
error1 :
unregister_nmi_handler ( NMI_UNKNOWN , " hpwdt " ) ;
error :
dev_warn ( & dev - > dev ,
" Unable to register a die notifier (err=%d). \n " ,
retval ) ;
return retval ;
2018-02-26 06:22:20 +03:00
# endif /* CONFIG_HPWDT_NMI_DECODING */
return 0 ;
2010-07-28 22:38:43 +04:00
}
2011-03-02 06:49:44 +03:00
static void hpwdt_exit_nmi_decoding ( void )
2010-07-28 22:38:43 +04:00
{
2018-02-26 06:22:20 +03:00
# ifdef CONFIG_HPWDT_NMI_DECODING
2011-09-30 23:06:21 +04:00
unregister_nmi_handler ( NMI_UNKNOWN , " hpwdt " ) ;
2012-06-26 12:27:00 +04:00
unregister_nmi_handler ( NMI_SERR , " hpwdt " ) ;
unregister_nmi_handler ( NMI_IO_CHECK , " hpwdt " ) ;
2018-02-26 06:22:20 +03:00
# endif
2010-07-28 03:51:02 +04:00
}
2012-11-19 22:21:41 +04:00
static int hpwdt_init_one ( struct pci_dev * dev ,
2008-07-15 23:40:41 +04:00
const struct pci_device_id * ent )
2007-12-04 20:41:54 +03:00
{
int retval ;
/*
2010-07-28 03:50:57 +04:00
* First let ' s find out if we are on an iLO2 + server . We will
2007-12-04 20:41:54 +03:00
* not run on a legacy ASM box .
2008-07-15 23:40:41 +04:00
* So we only support the G5 ProLiant servers and higher .
2007-12-04 20:41:54 +03:00
*/
2016-09-26 21:57:14 +03:00
if ( dev - > subsystem_vendor ! = PCI_VENDOR_ID_HP & &
dev - > subsystem_vendor ! = PCI_VENDOR_ID_HP_3PAR ) {
2007-12-04 20:41:54 +03:00
dev_warn ( & dev - > dev ,
2010-07-28 03:50:57 +04:00
" This server does not have an iLO2+ ASIC. \n " ) ;
2007-12-04 20:41:54 +03:00
return - ENODEV ;
}
2018-12-06 03:42:21 +03:00
if ( pci_match_id ( hpwdt_blacklist , dev ) ) {
dev_dbg ( & dev - > dev , " Not supported on this device \n " ) ;
2013-08-09 20:31:09 +04:00
return - ENODEV ;
2018-12-06 03:42:21 +03:00
}
2013-08-09 20:31:09 +04:00
2007-12-04 20:41:54 +03:00
if ( pci_enable_device ( dev ) ) {
dev_warn ( & dev - > dev ,
" Not possible to enable PCI Device: 0x%x:0x%x. \n " ,
ent - > vendor , ent - > device ) ;
return - ENODEV ;
}
pci_mem_addr = pci_iomap ( dev , 1 , 0x80 ) ;
if ( ! pci_mem_addr ) {
dev_warn ( & dev - > dev ,
2010-07-28 03:50:57 +04:00
" Unable to detect the iLO2+ server memory. \n " ) ;
2007-12-04 20:41:54 +03:00
retval = - ENOMEM ;
goto error_pci_iomap ;
}
2017-10-24 01:46:17 +03:00
hpwdt_nmistat = pci_mem_addr + 0x6e ;
2007-12-04 20:41:54 +03:00
hpwdt_timer_reg = pci_mem_addr + 0x70 ;
hpwdt_timer_con = pci_mem_addr + 0x72 ;
2019-05-17 23:59:40 +03:00
/* Have the core update running timer until user space is ready */
if ( hpwdt_hw_is_running ( ) ) {
dev_info ( & dev - > dev , " timer is running \n " ) ;
set_bit ( WDOG_HW_RUNNING , & hpwdt_dev . status ) ;
}
2012-08-27 22:52:24 +04:00
2010-07-28 22:38:43 +04:00
/* Initialize NMI Decoding functionality */
retval = hpwdt_init_nmi_decoding ( dev ) ;
if ( retval ! = 0 )
goto error_init_nmi_decoding ;
2007-12-04 20:41:54 +03:00
2019-05-17 23:59:38 +03:00
watchdog_stop_on_unregister ( & hpwdt_dev ) ;
2018-02-26 06:22:22 +03:00
watchdog_set_nowayout ( & hpwdt_dev , nowayout ) ;
2019-04-19 21:15:51 +03:00
watchdog_init_timeout ( & hpwdt_dev , soft_margin , NULL ) ;
2018-02-26 06:22:22 +03:00
2020-11-23 05:08:39 +03:00
if ( is_kdump_kernel ( ) ) {
2021-01-20 10:48:10 +03:00
pretimeout = false ;
2020-11-23 05:08:39 +03:00
kdumptimeout = 0 ;
}
2018-09-21 23:50:39 +03:00
if ( pretimeout & & hpwdt_dev . timeout < = PRETIMEOUT_SEC ) {
dev_warn ( & dev - > dev , " timeout <= pretimeout. Setting pretimeout to zero \n " ) ;
2021-01-20 10:48:10 +03:00
pretimeout = false ;
2018-09-21 23:50:39 +03:00
}
2018-08-08 22:13:23 +03:00
hpwdt_dev . pretimeout = pretimeout ? PRETIMEOUT_SEC : 0 ;
2019-05-17 23:59:41 +03:00
kdumptimeout = min ( kdumptimeout , HPWDT_MAX_TIMER ) ;
2018-08-08 22:13:23 +03:00
2018-02-26 06:22:22 +03:00
hpwdt_dev . parent = & dev - > dev ;
retval = watchdog_register_device ( & hpwdt_dev ) ;
2019-05-19 00:27:28 +03:00
if ( retval < 0 )
2018-02-26 06:22:22 +03:00
goto error_wd_register ;
2007-12-04 20:41:54 +03:00
2018-08-08 22:13:25 +03:00
dev_info ( & dev - > dev , " HPE Watchdog Timer Driver: Version: %s \n " ,
HPWDT_VERSION ) ;
dev_info ( & dev - > dev , " timeout: %d seconds (nowayout=%d) \n " ,
hpwdt_dev . timeout , nowayout ) ;
dev_info ( & dev - > dev , " pretimeout: %s. \n " ,
pretimeout ? " on " : " off " ) ;
2019-05-17 23:59:41 +03:00
dev_info ( & dev - > dev , " kdumptimeout: %d. \n " , kdumptimeout ) ;
2018-02-26 06:22:22 +03:00
2018-02-26 06:22:23 +03:00
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_HP_3PAR )
ilo5 = true ;
2007-12-04 20:41:54 +03:00
return 0 ;
2018-02-26 06:22:22 +03:00
error_wd_register :
2010-07-28 22:38:43 +04:00
hpwdt_exit_nmi_decoding ( ) ;
error_init_nmi_decoding :
2007-12-04 20:41:54 +03:00
pci_iounmap ( dev , pci_mem_addr ) ;
error_pci_iomap :
pci_disable_device ( dev ) ;
return retval ;
}
2012-11-19 22:26:24 +04:00
static void hpwdt_exit ( struct pci_dev * dev )
2007-12-04 20:41:54 +03:00
{
2018-02-26 06:22:22 +03:00
watchdog_unregister_device ( & hpwdt_dev ) ;
2010-07-28 22:38:43 +04:00
hpwdt_exit_nmi_decoding ( ) ;
2007-12-04 20:41:54 +03:00
pci_iounmap ( dev , pci_mem_addr ) ;
pci_disable_device ( dev ) ;
}
static struct pci_driver hpwdt_driver = {
. name = " hpwdt " ,
. id_table = hpwdt_devices ,
. probe = hpwdt_init_one ,
2012-11-19 22:21:12 +04:00
. remove = hpwdt_exit ,
2007-12-04 20:41:54 +03:00
} ;
MODULE_AUTHOR ( " Tom Mingarelli " ) ;
2018-02-26 06:22:19 +03:00
MODULE_DESCRIPTION ( " hpe watchdog driver " ) ;
2007-12-04 20:41:54 +03:00
MODULE_LICENSE ( " GPL " ) ;
2009-03-03 03:17:16 +03:00
MODULE_VERSION ( HPWDT_VERSION ) ;
2007-12-04 20:41:54 +03:00
module_param ( soft_margin , int , 0 ) ;
MODULE_PARM_DESC ( soft_margin , " Watchdog timeout in seconds " ) ;
2018-08-08 22:13:26 +03:00
module_param_named ( timeout , soft_margin , int , 0 ) ;
MODULE_PARM_DESC ( timeout , " Alias of soft_margin " ) ;
2012-03-05 19:51:11 +04:00
module_param ( nowayout , bool , 0 ) ;
2007-12-04 20:41:54 +03:00
MODULE_PARM_DESC ( nowayout , " Watchdog cannot be stopped once started (default= "
__MODULE_STRING ( WATCHDOG_NOWAYOUT ) " ) " ) ;
2019-05-17 23:59:41 +03:00
module_param ( kdumptimeout , int , 0444 ) ;
MODULE_PARM_DESC ( kdumptimeout , " Timeout applied for crash kernel transition in seconds " ) ;
2018-02-26 06:22:25 +03:00
# ifdef CONFIG_HPWDT_NMI_DECODING
module_param ( pretimeout , bool , 0 ) ;
MODULE_PARM_DESC ( pretimeout , " Watchdog pretimeout enabled " ) ;
# endif
2012-05-04 16:43:25 +04:00
module_pci_driver ( hpwdt_driver ) ;