Commit e5d56efc authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'watchdog-for-linus-v4.11' of...

Merge tag 'watchdog-for-linus-v4.11' of git://git.kernel.org/pub/scm/linux/kernel/git/groeck/linux-staging

Pull watchdog updates from Guenter Roeck:
 "Wim asked me to handle the watchdog pull request this time around.

  Key changes:

   - New drivers: Cortina Gemini, ZTE's zx2967 family, NIC7018

   - Convert to use device managed functions: ebc-c384_wdt, tegra_wdt,
     da9063_wdt, da9062_wdt, da9055_wdt, da9052_wdt, bcm2835_wdt,
     mena21_wdt, wm831x_wdt, digicolor_wdt, intel-mid_wdt, meson_wdt,
     sunxi_wdt, aspeed_wdt, coh901327_wdt, iTCO_wdt

   - Use watchdog core to install restart handler: tangox, dw_wdt,
     bcm2835_wdt, asm9260_wdt, bcm47xx_wdt

   - Convert ts72xx_wdt driver to watchdog core

   - Let core handle heartbeat in ep93xx_wdt driver

   - Enable COMPILE_TEST where possible

   - Various other improvements"

* tag 'watchdog-for-linus-v4.11' of git://git.kernel.org/pub/scm/linux/kernel/git/groeck/linux-staging: (54 commits)
  watchdog: s3c2410: Add prefix to local function
  watchdog: s3c2410: Select MFD_SYSCON on all Exynos platforms
  watchdog: s3c2410: Use dev_dbg instead of pr_info
  watchdog: s3c2410: Fix infinite interrupt in soft mode
  watchdog: s3c2410: Remove confusing CONFIG prefix from local defines
  watchdog: softdog: make pretimeout support a compile option
  watchdog: zx2967: add watchdog controller driver for ZTE's zx2967 family
  dt: bindings: add documentation for zx2967 family watchdog controller
  watchdog: sama5d4: Implement resume hook
  watchdog: sama5d4: Cache MR instead of a partial config
  watchdog: ts72xx_wdt: convert driver to watchdog core
  watchdog: ep93xx_wdt: cleanup and let the core handle the heartbeat
  watchdog: RDC321X_WDT always depends on PCI
  watchdog: add driver for Cortina Gemini watchdog
  watchdog: add DT bindings for Cortina Gemini
  watchdog: constify watchdog_ops structures
  watchdog: Introduce watchdog_stop_on_unregister helper
  watchdog: ebc-c384_wdt: Utilize devm_ functions in driver probe callback
  watchdog: tegra_wdt: Convert to use device managed functions
  watchdog: da9063_wdt: Convert to use device managed functions
  ...
parents c4f3f22e e3a60ead
Cortina Systems Gemini SoC Watchdog
Required properties:
- compatible : must be "cortina,gemini-watchdog"
- reg : shall contain base register location and length
- interrupts : shall contain the interrupt for the watchdog
Optional properties:
- timeout-sec : the default watchdog timeout in seconds.
Example:
watchdog@41000000 {
compatible = "cortina,gemini-watchdog";
reg = <0x41000000 0x1000>;
interrupts = <3 IRQ_TYPE_LEVEL_HIGH>;
};
...@@ -6,10 +6,11 @@ occurred. ...@@ -6,10 +6,11 @@ occurred.
Required properties: Required properties:
- compatible : should be one among the following - compatible : should be one among the following
(a) "samsung,s3c2410-wdt" for Exynos4 and previous SoCs - "samsung,s3c2410-wdt" for S3C2410
(b) "samsung,exynos5250-wdt" for Exynos5250 - "samsung,s3c6410-wdt" for S3C6410, S5PV210 and Exynos4
(c) "samsung,exynos5420-wdt" for Exynos5420 - "samsung,exynos5250-wdt" for Exynos5250
(c) "samsung,exynos7-wdt" for Exynos7 - "samsung,exynos5420-wdt" for Exynos5420
- "samsung,exynos7-wdt" for Exynos7
- reg : base physical address of the controller and length of memory mapped - reg : base physical address of the controller and length of memory mapped
region. region.
......
ZTE zx2967 Watchdog timer
Required properties:
- compatible : should be one of the following.
* zte,zx296718-wdt
- reg : Specifies base physical address and size of the registers.
- clocks : Pairs of phandle and specifier referencing the controller's clocks.
- resets : Reference to the reset controller controlling the watchdog
controller.
Optional properties:
- timeout-sec : Contains the watchdog timeout in seconds.
- zte,wdt-reset-sysctrl : Directs how to reset system by the watchdog.
if we don't want to restart system when watchdog been triggered,
it's not required, vice versa.
It should include following fields.
* phandle of aon-sysctrl.
* offset of register that be written, should be 0xb0.
* configure value that be written to aon-sysctrl.
* bit mask, corresponding bits will be affected.
Example:
wdt: watchdog@1465000 {
compatible = "zte,zx296718-wdt";
reg = <0x1465000 0x1000>;
clocks = <&topcrm WDT_WCLK>;
resets = <&toprst 35>;
zte,wdt-reset-sysctrl = <&aon_sysctrl 0xb0 1 0x115>;
};
...@@ -280,6 +280,12 @@ To disable the watchdog on reboot, the user must call the following helper: ...@@ -280,6 +280,12 @@ To disable the watchdog on reboot, the user must call the following helper:
static inline void watchdog_stop_on_reboot(struct watchdog_device *wdd); static inline void watchdog_stop_on_reboot(struct watchdog_device *wdd);
To disable the watchdog when unregistering the watchdog, the user must call
the following helper. Note that this will only stop the watchdog if the
nowayout flag is not set.
static inline void watchdog_stop_on_unregister(struct watchdog_device *wdd);
To change the priority of the restart handler the following helper should be To change the priority of the restart handler the following helper should be
used: used:
......
...@@ -209,6 +209,11 @@ timeout: Initial watchdog timeout in seconds (0<timeout<516, default=60) ...@@ -209,6 +209,11 @@ timeout: Initial watchdog timeout in seconds (0<timeout<516, default=60)
nowayout: Watchdog cannot be stopped once started nowayout: Watchdog cannot be stopped once started
(default=kernel config parameter) (default=kernel config parameter)
------------------------------------------------- -------------------------------------------------
nic7018_wdt:
timeout: Initial watchdog timeout in seconds (0<timeout<464, default=80)
nowayout: Watchdog cannot be stopped once started
(default=kernel config parameter)
-------------------------------------------------
nuc900_wdt: nuc900_wdt:
heartbeat: Watchdog heartbeats in seconds. heartbeat: Watchdog heartbeats in seconds.
(default = 15) (default = 15)
......
...@@ -71,9 +71,17 @@ config SOFT_WATCHDOG ...@@ -71,9 +71,17 @@ config SOFT_WATCHDOG
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called softdog. module will be called softdog.
config SOFT_WATCHDOG_PRETIMEOUT
bool "Software watchdog pretimeout governor support"
depends on SOFT_WATCHDOG && WATCHDOG_PRETIMEOUT_GOV
help
Enable this if you want to use pretimeout governors with the software
watchdog. Be aware that governors might affect the watchdog because it
is purely software, e.g. the panic governor will stall it!
config DA9052_WATCHDOG config DA9052_WATCHDOG
tristate "Dialog DA9052 Watchdog" tristate "Dialog DA9052 Watchdog"
depends on PMIC_DA9052 depends on PMIC_DA9052 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the watchdog in the DA9052 PMIC. Watchdog trigger Support for the watchdog in the DA9052 PMIC. Watchdog trigger
...@@ -85,7 +93,7 @@ config DA9052_WATCHDOG ...@@ -85,7 +93,7 @@ config DA9052_WATCHDOG
config DA9055_WATCHDOG config DA9055_WATCHDOG
tristate "Dialog Semiconductor DA9055 Watchdog" tristate "Dialog Semiconductor DA9055 Watchdog"
depends on MFD_DA9055 depends on MFD_DA9055 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
If you say yes here you get support for watchdog on the Dialog If you say yes here you get support for watchdog on the Dialog
...@@ -96,7 +104,7 @@ config DA9055_WATCHDOG ...@@ -96,7 +104,7 @@ config DA9055_WATCHDOG
config DA9063_WATCHDOG config DA9063_WATCHDOG
tristate "Dialog DA9063 Watchdog" tristate "Dialog DA9063 Watchdog"
depends on MFD_DA9063 depends on MFD_DA9063 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the watchdog in the DA9063 PMIC. Support for the watchdog in the DA9063 PMIC.
...@@ -105,7 +113,7 @@ config DA9063_WATCHDOG ...@@ -105,7 +113,7 @@ config DA9063_WATCHDOG
config DA9062_WATCHDOG config DA9062_WATCHDOG
tristate "Dialog DA9062/61 Watchdog" tristate "Dialog DA9062/61 Watchdog"
depends on MFD_DA9062 depends on MFD_DA9062 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the watchdog in the DA9062 and DA9061 PMICs. Support for the watchdog in the DA9062 and DA9061 PMICs.
...@@ -133,7 +141,7 @@ config GPIO_WATCHDOG_ARCH_INITCALL ...@@ -133,7 +141,7 @@ config GPIO_WATCHDOG_ARCH_INITCALL
config MENF21BMC_WATCHDOG config MENF21BMC_WATCHDOG
tristate "MEN 14F021P00 BMC Watchdog" tristate "MEN 14F021P00 BMC Watchdog"
depends on MFD_MENF21BMC depends on MFD_MENF21BMC || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the MEN 14F021P00 BMC Watchdog. Say Y here to include support for the MEN 14F021P00 BMC Watchdog.
...@@ -168,7 +176,7 @@ config WDAT_WDT ...@@ -168,7 +176,7 @@ config WDAT_WDT
config WM831X_WATCHDOG config WM831X_WATCHDOG
tristate "WM831x watchdog" tristate "WM831x watchdog"
depends on MFD_WM831X depends on MFD_WM831X || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the watchdog in the WM831x AudioPlus PMICs. When Support for the watchdog in the WM831x AudioPlus PMICs. When
...@@ -209,7 +217,7 @@ config ZIIRAVE_WATCHDOG ...@@ -209,7 +217,7 @@ config ZIIRAVE_WATCHDOG
config ARM_SP805_WATCHDOG config ARM_SP805_WATCHDOG
tristate "ARM SP805 Watchdog" tristate "ARM SP805 Watchdog"
depends on (ARM || ARM64) && ARM_AMBA depends on (ARM || ARM64) && (ARM_AMBA || COMPILE_TEST)
select WATCHDOG_CORE select WATCHDOG_CORE
help help
ARM Primecell SP805 Watchdog timer. This will reboot your system when ARM Primecell SP805 Watchdog timer. This will reboot your system when
...@@ -237,7 +245,7 @@ config ARM_SBSA_WATCHDOG ...@@ -237,7 +245,7 @@ config ARM_SBSA_WATCHDOG
config ASM9260_WATCHDOG config ASM9260_WATCHDOG
tristate "Alphascale ASM9260 watchdog" tristate "Alphascale ASM9260 watchdog"
depends on MACH_ASM9260 depends on MACH_ASM9260 || COMPILE_TEST
depends on OF depends on OF
select WATCHDOG_CORE select WATCHDOG_CORE
select RESET_CONTROLLER select RESET_CONTROLLER
...@@ -247,14 +255,14 @@ config ASM9260_WATCHDOG ...@@ -247,14 +255,14 @@ config ASM9260_WATCHDOG
config AT91RM9200_WATCHDOG config AT91RM9200_WATCHDOG
tristate "AT91RM9200 watchdog" tristate "AT91RM9200 watchdog"
depends on SOC_AT91RM9200 && MFD_SYSCON depends on (SOC_AT91RM9200 && MFD_SYSCON) || COMPILE_TEST
help help
Watchdog timer embedded into AT91RM9200 chips. This will reboot your Watchdog timer embedded into AT91RM9200 chips. This will reboot your
system when the timeout is reached. system when the timeout is reached.
config AT91SAM9X_WATCHDOG config AT91SAM9X_WATCHDOG
tristate "AT91SAM9X / AT91CAP9 watchdog" tristate "AT91SAM9X / AT91CAP9 watchdog"
depends on ARCH_AT91 depends on ARCH_AT91 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Watchdog timer embedded into AT91SAM9X and AT91CAP9 chips. This will Watchdog timer embedded into AT91SAM9X and AT91CAP9 chips. This will
...@@ -262,7 +270,7 @@ config AT91SAM9X_WATCHDOG ...@@ -262,7 +270,7 @@ config AT91SAM9X_WATCHDOG
config SAMA5D4_WATCHDOG config SAMA5D4_WATCHDOG
tristate "Atmel SAMA5D4 Watchdog Timer" tristate "Atmel SAMA5D4 Watchdog Timer"
depends on ARCH_AT91 depends on ARCH_AT91 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Atmel SAMA5D4 watchdog timer is embedded into SAMA5D4 chips. Atmel SAMA5D4 watchdog timer is embedded into SAMA5D4 chips.
...@@ -293,7 +301,7 @@ config 21285_WATCHDOG ...@@ -293,7 +301,7 @@ config 21285_WATCHDOG
config 977_WATCHDOG config 977_WATCHDOG
tristate "NetWinder WB83C977 watchdog" tristate "NetWinder WB83C977 watchdog"
depends on FOOTBRIDGE && ARCH_NETWINDER depends on (FOOTBRIDGE && ARCH_NETWINDER) || (ARM && COMPILE_TEST)
help help
Say Y here to include support for the WB977 watchdog included in Say Y here to include support for the WB977 watchdog included in
NetWinder machines. Alternatively say M to compile the driver as NetWinder machines. Alternatively say M to compile the driver as
...@@ -301,6 +309,17 @@ config 977_WATCHDOG ...@@ -301,6 +309,17 @@ config 977_WATCHDOG
Not sure? It's safe to say N. Not sure? It's safe to say N.
config GEMINI_WATCHDOG
tristate "Gemini watchdog"
depends on ARCH_GEMINI
select WATCHDOG_CORE
help
Say Y here if to include support for the watchdog timer
embedded in the Cortina Systems Gemini family of devices.
To compile this driver as a module, choose M here: the
module will be called gemini_wdt.
config IXP4XX_WATCHDOG config IXP4XX_WATCHDOG
tristate "IXP4xx Watchdog" tristate "IXP4xx Watchdog"
depends on ARCH_IXP4XX depends on ARCH_IXP4XX
...@@ -333,9 +352,9 @@ config HAVE_S3C2410_WATCHDOG ...@@ -333,9 +352,9 @@ config HAVE_S3C2410_WATCHDOG
config S3C2410_WATCHDOG config S3C2410_WATCHDOG
tristate "S3C2410 Watchdog" tristate "S3C2410 Watchdog"
depends on HAVE_S3C2410_WATCHDOG depends on HAVE_S3C2410_WATCHDOG || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
select MFD_SYSCON if ARCH_EXYNOS5 select MFD_SYSCON if ARCH_EXYNOS
help help
Watchdog timer block in the Samsung SoCs. This will reboot Watchdog timer block in the Samsung SoCs. This will reboot
the system when the timer expires with the watchdog enabled. the system when the timer expires with the watchdog enabled.
...@@ -372,7 +391,7 @@ config DW_WATCHDOG ...@@ -372,7 +391,7 @@ config DW_WATCHDOG
config EP93XX_WATCHDOG config EP93XX_WATCHDOG
tristate "EP93xx Watchdog" tristate "EP93xx Watchdog"
depends on ARCH_EP93XX depends on ARCH_EP93XX || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here if to include support for the watchdog timer Say Y here if to include support for the watchdog timer
...@@ -383,7 +402,7 @@ config EP93XX_WATCHDOG ...@@ -383,7 +402,7 @@ config EP93XX_WATCHDOG
config OMAP_WATCHDOG config OMAP_WATCHDOG
tristate "OMAP Watchdog" tristate "OMAP Watchdog"
depends on ARCH_OMAP16XX || ARCH_OMAP2PLUS depends on ARCH_OMAP16XX || ARCH_OMAP2PLUS || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for TI OMAP1610/OMAP1710/OMAP2420/OMAP3430/OMAP4430 watchdog. Say 'Y' Support for TI OMAP1610/OMAP1710/OMAP2420/OMAP3430/OMAP4430 watchdog. Say 'Y'
...@@ -419,7 +438,7 @@ config IOP_WATCHDOG ...@@ -419,7 +438,7 @@ config IOP_WATCHDOG
config DAVINCI_WATCHDOG config DAVINCI_WATCHDOG
tristate "DaVinci watchdog" tristate "DaVinci watchdog"
depends on ARCH_DAVINCI || ARCH_KEYSTONE depends on ARCH_DAVINCI || ARCH_KEYSTONE || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here if to include support for the watchdog timer Say Y here if to include support for the watchdog timer
...@@ -432,7 +451,7 @@ config DAVINCI_WATCHDOG ...@@ -432,7 +451,7 @@ config DAVINCI_WATCHDOG
config ORION_WATCHDOG config ORION_WATCHDOG
tristate "Orion watchdog" tristate "Orion watchdog"
depends on ARCH_ORION5X || ARCH_DOVE || MACH_DOVE || ARCH_MVEBU depends on ARCH_ORION5X || ARCH_DOVE || MACH_DOVE || ARCH_MVEBU || COMPILE_TEST
depends on ARM depends on ARM
select WATCHDOG_CORE select WATCHDOG_CORE
help help
...@@ -443,7 +462,7 @@ config ORION_WATCHDOG ...@@ -443,7 +462,7 @@ config ORION_WATCHDOG
config RN5T618_WATCHDOG config RN5T618_WATCHDOG
tristate "Ricoh RN5T618 watchdog" tristate "Ricoh RN5T618 watchdog"
depends on MFD_RN5T618 depends on MFD_RN5T618 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
If you say yes here you get support for watchdog on the Ricoh If you say yes here you get support for watchdog on the Ricoh
...@@ -454,7 +473,7 @@ config RN5T618_WATCHDOG ...@@ -454,7 +473,7 @@ config RN5T618_WATCHDOG
config SUNXI_WATCHDOG config SUNXI_WATCHDOG
tristate "Allwinner SoCs watchdog support" tristate "Allwinner SoCs watchdog support"
depends on ARCH_SUNXI depends on ARCH_SUNXI || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer Say Y here to include support for the watchdog timer
...@@ -464,7 +483,7 @@ config SUNXI_WATCHDOG ...@@ -464,7 +483,7 @@ config SUNXI_WATCHDOG
config COH901327_WATCHDOG config COH901327_WATCHDOG
bool "ST-Ericsson COH 901 327 watchdog" bool "ST-Ericsson COH 901 327 watchdog"
depends on ARCH_U300 depends on ARCH_U300 || (ARM && COMPILE_TEST)
default y if MACH_U300 default y if MACH_U300
select WATCHDOG_CORE select WATCHDOG_CORE
help help
...@@ -483,7 +502,7 @@ config TWL4030_WATCHDOG ...@@ -483,7 +502,7 @@ config TWL4030_WATCHDOG
config STMP3XXX_RTC_WATCHDOG config STMP3XXX_RTC_WATCHDOG
tristate "Freescale STMP3XXX & i.MX23/28 watchdog" tristate "Freescale STMP3XXX & i.MX23/28 watchdog"
depends on RTC_DRV_STMP depends on RTC_DRV_STMP || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer inside Say Y here to include support for the watchdog timer inside
...@@ -493,7 +512,7 @@ config STMP3XXX_RTC_WATCHDOG ...@@ -493,7 +512,7 @@ config STMP3XXX_RTC_WATCHDOG
config NUC900_WATCHDOG config NUC900_WATCHDOG
tristate "Nuvoton NUC900 watchdog" tristate "Nuvoton NUC900 watchdog"
depends on ARCH_W90X900 depends on ARCH_W90X900 || COMPILE_TEST
help help
Say Y here if to include support for the watchdog timer Say Y here if to include support for the watchdog timer
for the Nuvoton NUC900 series SoCs. for the Nuvoton NUC900 series SoCs.
...@@ -513,7 +532,7 @@ config TS4800_WATCHDOG ...@@ -513,7 +532,7 @@ config TS4800_WATCHDOG
config TS72XX_WATCHDOG config TS72XX_WATCHDOG
tristate "TS-72XX SBC Watchdog" tristate "TS-72XX SBC Watchdog"
depends on MACH_TS72XX depends on MACH_TS72XX || COMPILE_TEST
help help
Technologic Systems TS-7200, TS-7250 and TS-7260 boards have Technologic Systems TS-7200, TS-7250 and TS-7260 boards have
watchdog timer implemented in a external CPLD chip. Say Y here watchdog timer implemented in a external CPLD chip. Say Y here
...@@ -531,7 +550,7 @@ config MAX63XX_WATCHDOG ...@@ -531,7 +550,7 @@ config MAX63XX_WATCHDOG
config MAX77620_WATCHDOG config MAX77620_WATCHDOG
tristate "Maxim Max77620 Watchdog Timer" tristate "Maxim Max77620 Watchdog Timer"
depends on MFD_MAX77620 depends on MFD_MAX77620 || COMPILE_TEST
help help
This is the driver for the Max77620 watchdog timer. This is the driver for the Max77620 watchdog timer.
Say 'Y' here to enable the watchdog timer support for Say 'Y' here to enable the watchdog timer support for
...@@ -540,7 +559,7 @@ config MAX77620_WATCHDOG ...@@ -540,7 +559,7 @@ config MAX77620_WATCHDOG
config IMX2_WDT config IMX2_WDT
tristate "IMX2+ Watchdog" tristate "IMX2+ Watchdog"
depends on ARCH_MXC || ARCH_LAYERSCAPE depends on ARCH_MXC || ARCH_LAYERSCAPE || COMPILE_TEST
select REGMAP_MMIO select REGMAP_MMIO
select WATCHDOG_CORE select WATCHDOG_CORE
help help
...@@ -554,7 +573,7 @@ config IMX2_WDT ...@@ -554,7 +573,7 @@ config IMX2_WDT
config UX500_WATCHDOG config UX500_WATCHDOG
tristate "ST-Ericsson Ux500 watchdog" tristate "ST-Ericsson Ux500 watchdog"
depends on MFD_DB8500_PRCMU depends on MFD_DB8500_PRCMU || (ARM && COMPILE_TEST)
select WATCHDOG_CORE select WATCHDOG_CORE
default y default y
help help
...@@ -566,7 +585,7 @@ config UX500_WATCHDOG ...@@ -566,7 +585,7 @@ config UX500_WATCHDOG
config RETU_WATCHDOG config RETU_WATCHDOG
tristate "Retu watchdog" tristate "Retu watchdog"
depends on MFD_RETU depends on MFD_RETU || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Retu watchdog driver for Nokia Internet Tablets (770, N800, Retu watchdog driver for Nokia Internet Tablets (770, N800,
...@@ -578,7 +597,7 @@ config RETU_WATCHDOG ...@@ -578,7 +597,7 @@ config RETU_WATCHDOG
config MOXART_WDT config MOXART_WDT
tristate "MOXART watchdog" tristate "MOXART watchdog"
depends on ARCH_MOXART depends on ARCH_MOXART || COMPILE_TEST
help help
Say Y here to include Watchdog timer support for the watchdog Say Y here to include Watchdog timer support for the watchdog
existing on the MOXA ART SoC series platforms. existing on the MOXA ART SoC series platforms.
...@@ -588,7 +607,7 @@ config MOXART_WDT ...@@ -588,7 +607,7 @@ config MOXART_WDT
config SIRFSOC_WATCHDOG config SIRFSOC_WATCHDOG
tristate "SiRFSOC watchdog" tristate "SiRFSOC watchdog"
depends on ARCH_SIRF depends on ARCH_SIRF || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
default y default y
help help
...@@ -597,7 +616,7 @@ config SIRFSOC_WATCHDOG ...@@ -597,7 +616,7 @@ config SIRFSOC_WATCHDOG
config ST_LPC_WATCHDOG config ST_LPC_WATCHDOG
tristate "STMicroelectronics LPC Watchdog" tristate "STMicroelectronics LPC Watchdog"
depends on ARCH_STI depends on ARCH_STI || COMPILE_TEST
depends on OF depends on OF
select WATCHDOG_CORE select WATCHDOG_CORE
help help
...@@ -621,7 +640,7 @@ config TEGRA_WATCHDOG ...@@ -621,7 +640,7 @@ config TEGRA_WATCHDOG
config QCOM_WDT config QCOM_WDT
tristate "QCOM watchdog" tristate "QCOM watchdog"
depends on HAS_IOMEM depends on HAS_IOMEM
depends on ARCH_QCOM depends on ARCH_QCOM || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include Watchdog timer support for the watchdog found Say Y here to include Watchdog timer support for the watchdog found
...@@ -633,7 +652,7 @@ config QCOM_WDT ...@@ -633,7 +652,7 @@ config QCOM_WDT
config MESON_GXBB_WATCHDOG config MESON_GXBB_WATCHDOG
tristate "Amlogic Meson GXBB SoCs watchdog support" tristate "Amlogic Meson GXBB SoCs watchdog support"
depends on ARCH_MESON depends on ARCH_MESON || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer Say Y here to include support for the watchdog timer
...@@ -643,7 +662,7 @@ config MESON_GXBB_WATCHDOG ...@@ -643,7 +662,7 @@ config MESON_GXBB_WATCHDOG
config MESON_WATCHDOG config MESON_WATCHDOG
tristate "Amlogic Meson SoCs watchdog support" tristate "Amlogic Meson SoCs watchdog support"
depends on ARCH_MESON depends on ARCH_MESON || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer Say Y here to include support for the watchdog timer
...@@ -653,7 +672,7 @@ config MESON_WATCHDOG ...@@ -653,7 +672,7 @@ config MESON_WATCHDOG
config MEDIATEK_WATCHDOG config MEDIATEK_WATCHDOG
tristate "Mediatek SoCs watchdog support" tristate "Mediatek SoCs watchdog support"
depends on ARCH_MEDIATEK depends on ARCH_MEDIATEK || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer Say Y here to include support for the watchdog timer
...@@ -663,7 +682,7 @@ config MEDIATEK_WATCHDOG ...@@ -663,7 +682,7 @@ config MEDIATEK_WATCHDOG
config DIGICOLOR_WATCHDOG config DIGICOLOR_WATCHDOG
tristate "Conexant Digicolor SoCs watchdog support" tristate "Conexant Digicolor SoCs watchdog support"
depends on ARCH_DIGICOLOR depends on ARCH_DIGICOLOR || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Say Y here to include support for the watchdog timer Say Y here to include support for the watchdog timer
...@@ -685,7 +704,7 @@ config LPC18XX_WATCHDOG ...@@ -685,7 +704,7 @@ config LPC18XX_WATCHDOG
config ATLAS7_WATCHDOG config ATLAS7_WATCHDOG
tristate "CSRatlas7 watchdog" tristate "CSRatlas7 watchdog"
depends on ARCH_ATLAS7 depends on ARCH_ATLAS7 || COMPILE_TEST
help help
Say Y here to include Watchdog timer support for the watchdog Say Y here to include Watchdog timer support for the watchdog
existing on the CSRatlas7 series platforms. existing on the CSRatlas7 series platforms.
...@@ -714,11 +733,21 @@ config ASPEED_WATCHDOG ...@@ -714,11 +733,21 @@ config ASPEED_WATCHDOG
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called aspeed_wdt. module will be called aspeed_wdt.
config ZX2967_WATCHDOG
tristate "ZTE zx2967 SoCs watchdog support"
depends on ARCH_ZX
select WATCHDOG_CORE
help
Say Y here to include support for the watchdog timer
in ZTE zx2967 SoCs.
To compile this driver as a module, choose M here: the
module will be called zx2967_wdt.
# AVR32 Architecture # AVR32 Architecture
config AT32AP700X_WDT config AT32AP700X_WDT
tristate "AT32AP700x watchdog" tristate "AT32AP700x watchdog"
depends on CPU_AT32AP700X depends on CPU_AT32AP700X || COMPILE_TEST
help help
Watchdog timer embedded into AT32AP700x devices. This will reboot Watchdog timer embedded into AT32AP700x devices. This will reboot
your system when the timeout is reached. your system when the timeout is reached.
...@@ -822,7 +851,7 @@ config SP5100_TCO ...@@ -822,7 +851,7 @@ config SP5100_TCO
config GEODE_WDT config GEODE_WDT
tristate "AMD Geode CS5535/CS5536 Watchdog" tristate "AMD Geode CS5535/CS5536 Watchdog"
depends on CS5535_MFGPT depends on CS5535_MFGPT || (X86 && COMPILE_TEST)
help help
This driver enables a watchdog capability built into the This driver enables a watchdog capability built into the
CS5535/CS5536 companion chips for the AMD Geode GX and LX CS5535/CS5536 companion chips for the AMD Geode GX and LX
...@@ -835,7 +864,7 @@ config GEODE_WDT ...@@ -835,7 +864,7 @@ config GEODE_WDT
config SC520_WDT config SC520_WDT
tristate "AMD Elan SC520 processor Watchdog" tristate "AMD Elan SC520 processor Watchdog"
depends on MELAN depends on MELAN || COMPILE_TEST
help help
This is the driver for the hardware watchdog built in to the This is the driver for the hardware watchdog built in to the
AMD "Elan" SC520 microcomputer commonly used in embedded systems. AMD "Elan" SC520 microcomputer commonly used in embedded systems.
...@@ -1034,7 +1063,7 @@ config HP_WATCHDOG ...@@ -1034,7 +1063,7 @@ config HP_WATCHDOG
config KEMPLD_WDT config KEMPLD_WDT
tristate "Kontron COM Watchdog Timer" tristate "Kontron COM Watchdog Timer"
depends on MFD_KEMPLD depends on MFD_KEMPLD || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the PLD watchdog on some Kontron ETX and COMexpress Support for the PLD watchdog on some Kontron ETX and COMexpress
...@@ -1108,7 +1137,8 @@ config NV_TCO ...@@ -1108,7 +1137,8 @@ config NV_TCO
config RDC321X_WDT config RDC321X_WDT
tristate "RDC R-321x SoC watchdog" tristate "RDC R-321x SoC watchdog"
depends on X86_RDC321X depends on X86_RDC321X || COMPILE_TEST
depends on PCI
help help
This is the driver for the built in hardware watchdog This is the driver for the built in hardware watchdog
in the RDC R-321x SoC. in the RDC R-321x SoC.
...@@ -1326,6 +1356,16 @@ config NI903X_WDT ...@@ -1326,6 +1356,16 @@ config NI903X_WDT
To compile this driver as a module, choose M here: the module will be To compile this driver as a module, choose M here: the module will be
called ni903x_wdt. called ni903x_wdt.
config NIC7018_WDT
tristate "NIC7018 Watchdog"
depends on X86 && ACPI
select WATCHDOG_CORE
---help---
Support for National Instruments NIC7018 Watchdog.
To compile this driver as a module, choose M here: the module will be
called nic7018_wdt.
# M32R Architecture # M32R Architecture
# M68K Architecture # M68K Architecture
...@@ -1343,14 +1383,14 @@ config M54xx_WATCHDOG ...@@ -1343,14 +1383,14 @@ config M54xx_WATCHDOG
config ATH79_WDT config ATH79_WDT
tristate "Atheros AR71XX/AR724X/AR913X hardware watchdog" tristate "Atheros AR71XX/AR724X/AR913X hardware watchdog"
depends on ATH79 depends on ATH79 || (ARM && COMPILE_TEST)
help help
Hardware driver for the built-in watchdog timer on the Atheros Hardware driver for the built-in watchdog timer on the Atheros
AR71XX/AR724X/AR913X SoCs. AR71XX/AR724X/AR913X SoCs.
config BCM47XX_WDT config BCM47XX_WDT
tristate "Broadcom BCM47xx Watchdog Timer" tristate "Broadcom BCM47xx Watchdog Timer"
depends on BCM47XX || ARCH_BCM_5301X depends on BCM47XX || ARCH_BCM_5301X || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Hardware driver for the Broadcom BCM47xx Watchdog Timer. Hardware driver for the Broadcom BCM47xx Watchdog Timer.
...@@ -1367,7 +1407,7 @@ config RC32434_WDT ...@@ -1367,7 +1407,7 @@ config RC32434_WDT
config INDYDOG config INDYDOG
tristate "Indy/I2 Hardware Watchdog" tristate "Indy/I2 Hardware Watchdog"
depends on SGI_HAS_INDYDOG depends on SGI_HAS_INDYDOG || (MIPS && COMPILE_TEST)
help help
Hardware driver for the Indy's/I2's watchdog. This is a Hardware driver for the Indy's/I2's watchdog. This is a
watchdog timer that will reboot the machine after a 60 second watchdog timer that will reboot the machine after a 60 second
...@@ -1383,7 +1423,7 @@ config JZ4740_WDT ...@@ -1383,7 +1423,7 @@ config JZ4740_WDT
config WDT_MTX1 config WDT_MTX1
tristate "MTX-1 Hardware Watchdog" tristate "MTX-1 Hardware Watchdog"
depends on MIPS_MTX1 depends on MIPS_MTX1 || (MIPS && COMPILE_TEST)
help help
Hardware driver for the MTX-1 boards. This is a watchdog timer that Hardware driver for the MTX-1 boards. This is a watchdog timer that
will reboot the machine after a 100 seconds timer expired. will reboot the machine after a 100 seconds timer expired.
...@@ -1391,6 +1431,7 @@ config WDT_MTX1 ...@@ -1391,6 +1431,7 @@ config WDT_MTX1
config PNX833X_WDT config PNX833X_WDT
tristate "PNX833x Hardware Watchdog" tristate "PNX833x Hardware Watchdog"
depends on SOC_PNX8335 depends on SOC_PNX8335
depends on BROKEN
help help
Hardware driver for the PNX833x's watchdog. This is a Hardware driver for the PNX833x's watchdog. This is a
watchdog timer that will reboot the machine after a programmable watchdog timer that will reboot the machine after a programmable
...@@ -1399,7 +1440,7 @@ config PNX833X_WDT ...@@ -1399,7 +1440,7 @@ config PNX833X_WDT
config SIBYTE_WDOG config SIBYTE_WDOG
tristate "Sibyte SoC hardware watchdog" tristate "Sibyte SoC hardware watchdog"
depends on CPU_SB1 depends on CPU_SB1 || (MIPS && COMPILE_TEST)
help help
Watchdog driver for the built in watchdog hardware in Sibyte Watchdog driver for the built in watchdog hardware in Sibyte
SoC processors. There are apparently two watchdog timers SoC processors. There are apparently two watchdog timers
...@@ -1412,13 +1453,13 @@ config SIBYTE_WDOG ...@@ -1412,13 +1453,13 @@ config SIBYTE_WDOG
config AR7_WDT config AR7_WDT
tristate "TI AR7 Watchdog Timer" tristate "TI AR7 Watchdog Timer"
depends on AR7 depends on AR7 || (MIPS && COMPILE_TEST)
help help
Hardware driver for the TI AR7 Watchdog Timer. Hardware driver for the TI AR7 Watchdog Timer.
config TXX9_WDT config TXX9_WDT
tristate "Toshiba TXx9 Watchdog Timer" tristate "Toshiba TXx9 Watchdog Timer"
depends on CPU_TX39XX || CPU_TX49XX depends on CPU_TX39XX || CPU_TX49XX || (MIPS && COMPILE_TEST)
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Hardware driver for the built-in watchdog timer on TXx9 MIPS SoCs. Hardware driver for the built-in watchdog timer on TXx9 MIPS SoCs.
...@@ -1454,7 +1495,7 @@ config BCM63XX_WDT ...@@ -1454,7 +1495,7 @@ config BCM63XX_WDT
config BCM2835_WDT config BCM2835_WDT
tristate "Broadcom BCM2835 hardware watchdog" tristate "Broadcom BCM2835 hardware watchdog"
depends on ARCH_BCM2835 depends on ARCH_BCM2835 || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Watchdog driver for the built in watchdog hardware in Broadcom Watchdog driver for the built in watchdog hardware in Broadcom
...@@ -1465,7 +1506,7 @@ config BCM2835_WDT ...@@ -1465,7 +1506,7 @@ config BCM2835_WDT
config BCM_KONA_WDT config BCM_KONA_WDT
tristate "BCM Kona Watchdog" tristate "BCM Kona Watchdog"
depends on ARCH_BCM_MOBILE depends on ARCH_BCM_MOBILE || COMPILE_TEST
select WATCHDOG_CORE select WATCHDOG_CORE
help help
Support for the watchdog timer on the following Broadcom BCM281xx Support for the watchdog timer on the following Broadcom BCM281xx
...@@ -1477,7 +1518,7 @@ config BCM_KONA_WDT ...@@ -1477,7 +1518,7 @@ config BCM_KONA_WDT
config BCM_KONA_WDT_DEBUG config BCM_KONA_WDT_DEBUG
bool "DEBUGFS support for BCM Kona Watchdog" bool "DEBUGFS support for BCM Kona Watchdog"
depends on BCM_KONA_WDT depends on BCM_KONA_WDT || COMPILE_TEST
help help
If enabled, adds /sys/kernel/debug/bcm_kona_wdt/info which provides If enabled, adds /sys/kernel/debug/bcm_kona_wdt/info which provides
access to the driver's internal data structures as well as watchdog access to the driver's internal data structures as well as watchdog
...@@ -1538,7 +1579,7 @@ config MT7621_WDT ...@@ -1538,7 +1579,7 @@ config MT7621_WDT
config PIC32_WDT config PIC32_WDT
tristate "Microchip PIC32 hardware watchdog" tristate "Microchip PIC32 hardware watchdog"
select WATCHDOG_CORE select WATCHDOG_CORE
depends on MACH_PIC32 depends on MACH_PIC32 || (MIPS && COMPILE_TEST)
help help
Watchdog driver for the built in watchdog hardware in a PIC32. Watchdog driver for the built in watchdog hardware in a PIC32.
...@@ -1551,7 +1592,7 @@ config PIC32_WDT ...@@ -1551,7 +1592,7 @@ config PIC32_WDT
config PIC32_DMT config PIC32_DMT
tristate "Microchip PIC32 Deadman Timer" tristate "Microchip PIC32 Deadman Timer"
select WATCHDOG_CORE select WATCHDOG_CORE
depends on MACH_PIC32 depends on MACH_PIC32 || (MIPS && COMPILE_TEST)
help help
Watchdog driver for PIC32 instruction fetch counting timer. This specific Watchdog driver for PIC32 instruction fetch counting timer. This specific
timer is typically be used in misson critical and safety critical timer is typically be used in misson critical and safety critical
...@@ -1573,7 +1614,7 @@ config GEF_WDT ...@@ -1573,7 +1614,7 @@ config GEF_WDT
config MPC5200_WDT config MPC5200_WDT
bool "MPC52xx Watchdog Timer" bool "MPC52xx Watchdog Timer"
depends on PPC_MPC52xx depends on PPC_MPC52xx || COMPILE_TEST
help help
Use General Purpose Timer (GPT) 0 on the MPC5200 as Watchdog. Use General Purpose Timer (GPT) 0 on the MPC5200 as Watchdog.
...@@ -1592,11 +1633,11 @@ config 8xxx_WDT ...@@ -1592,11 +1633,11 @@ config 8xxx_WDT
config MV64X60_WDT config MV64X60_WDT
tristate "MV64X60 (Marvell Discovery) Watchdog Timer" tristate "MV64X60 (Marvell Discovery) Watchdog Timer"
depends on MV64X60 depends on MV64X60 || COMPILE_TEST
config PIKA_WDT config PIKA_WDT
tristate "PIKA FPGA Watchdog" tristate "PIKA FPGA Watchdog"
depends on WARP depends on WARP || (PPC64 && COMPILE_TEST)
default y default y
help help
This enables the watchdog in the PIKA FPGA. Currently used on This enables the watchdog in the PIKA FPGA. Currently used on
...@@ -1646,7 +1687,7 @@ config MEN_A21_WDT ...@@ -1646,7 +1687,7 @@ config MEN_A21_WDT
config WATCHDOG_RTAS config WATCHDOG_RTAS
tristate "RTAS watchdog" tristate "RTAS watchdog"
depends on PPC_RTAS depends on PPC_RTAS || (PPC64 && COMPILE_TEST)
help help
This driver adds watchdog support for the RTAS watchdog. This driver adds watchdog support for the RTAS watchdog.
...@@ -1674,7 +1715,7 @@ config DIAG288_WATCHDOG ...@@ -1674,7 +1715,7 @@ config DIAG288_WATCHDOG
config SH_WDT config SH_WDT
tristate "SuperH Watchdog" tristate "SuperH Watchdog"
depends on SUPERH && (CPU_SH3 || CPU_SH4) depends on SUPERH && (CPU_SH3 || CPU_SH4 || COMPILE_TEST)
select WATCHDOG_CORE select WATCHDOG_CORE
help help
This driver adds watchdog support for the integrated watchdog in the This driver adds watchdog support for the integrated watchdog in the
...@@ -1741,7 +1782,7 @@ config XEN_WDT ...@@ -1741,7 +1782,7 @@ config XEN_WDT
config UML_WATCHDOG config UML_WATCHDOG
tristate "UML watchdog" tristate "UML watchdog"
depends on UML depends on UML || COMPILE_TEST
# #
# ISA-based Watchdog Cards # ISA-based Watchdog Cards
......
...@@ -45,6 +45,7 @@ obj-$(CONFIG_OMAP_WATCHDOG) += omap_wdt.o ...@@ -45,6 +45,7 @@ obj-$(CONFIG_OMAP_WATCHDOG) += omap_wdt.o
obj-$(CONFIG_TWL4030_WATCHDOG) += twl4030_wdt.o obj-$(CONFIG_TWL4030_WATCHDOG) += twl4030_wdt.o
obj-$(CONFIG_21285_WATCHDOG) += wdt285.o obj-$(CONFIG_21285_WATCHDOG) += wdt285.o
obj-$(CONFIG_977_WATCHDOG) += wdt977.o obj-$(CONFIG_977_WATCHDOG) += wdt977.o
obj-$(CONFIG_GEMINI_WATCHDOG) += gemini_wdt.o
obj-$(CONFIG_IXP4XX_WATCHDOG) += ixp4xx_wdt.o obj-$(CONFIG_IXP4XX_WATCHDOG) += ixp4xx_wdt.o
obj-$(CONFIG_KS8695_WATCHDOG) += ks8695_wdt.o obj-$(CONFIG_KS8695_WATCHDOG) += ks8695_wdt.o
obj-$(CONFIG_S3C2410_WATCHDOG) += s3c2410_wdt.o obj-$(CONFIG_S3C2410_WATCHDOG) += s3c2410_wdt.o
...@@ -82,6 +83,7 @@ obj-$(CONFIG_BCM7038_WDT) += bcm7038_wdt.o ...@@ -82,6 +83,7 @@ obj-$(CONFIG_BCM7038_WDT) += bcm7038_wdt.o
obj-$(CONFIG_ATLAS7_WATCHDOG) += atlas7_wdt.o obj-$(CONFIG_ATLAS7_WATCHDOG) += atlas7_wdt.o
obj-$(CONFIG_RENESAS_WDT) += renesas_wdt.o obj-$(CONFIG_RENESAS_WDT) += renesas_wdt.o
obj-$(CONFIG_ASPEED_WATCHDOG) += aspeed_wdt.o obj-$(CONFIG_ASPEED_WATCHDOG) += aspeed_wdt.o
obj-$(CONFIG_ZX2967_WATCHDOG) += zx2967_wdt.o
# AVR32 Architecture # AVR32 Architecture
obj-$(CONFIG_AT32AP700X_WDT) += at32ap700x_wdt.o obj-$(CONFIG_AT32AP700X_WDT) += at32ap700x_wdt.o
...@@ -139,6 +141,7 @@ obj-$(CONFIG_INTEL_SCU_WATCHDOG) += intel_scu_watchdog.o ...@@ -139,6 +141,7 @@ obj-$(CONFIG_INTEL_SCU_WATCHDOG) += intel_scu_watchdog.o
obj-$(CONFIG_INTEL_MID_WATCHDOG) += intel-mid_wdt.o obj-$(CONFIG_INTEL_MID_WATCHDOG) += intel-mid_wdt.o
obj-$(CONFIG_INTEL_MEI_WDT) += mei_wdt.o obj-$(CONFIG_INTEL_MEI_WDT) += mei_wdt.o
obj-$(CONFIG_NI903X_WDT) += ni903x_wdt.o obj-$(CONFIG_NI903X_WDT) += ni903x_wdt.o
obj-$(CONFIG_NIC7018_WDT) += nic7018_wdt.o
# M32R Architecture # M32R Architecture
......
...@@ -14,7 +14,6 @@ ...@@ -14,7 +14,6 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/reboot.h>
#include <linux/reset.h> #include <linux/reset.h>
#include <linux/watchdog.h> #include <linux/watchdog.h>
...@@ -59,7 +58,6 @@ struct asm9260_wdt_priv { ...@@ -59,7 +58,6 @@ struct asm9260_wdt_priv {
struct clk *clk; struct clk *clk;
struct clk *clk_ahb; struct clk *clk_ahb;
struct reset_control *rst; struct reset_control *rst;
struct notifier_block restart_handler;
void __iomem *iobase; void __iomem *iobase;
int irq; int irq;
...@@ -172,15 +170,14 @@ static irqreturn_t asm9260_wdt_irq(int irq, void *devid) ...@@ -172,15 +170,14 @@ static irqreturn_t asm9260_wdt_irq(int irq, void *devid)
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static int asm9260_restart_handler(struct notifier_block *this, static int asm9260_restart(struct watchdog_device *wdd, unsigned long action,
unsigned long mode, void *cmd) void *data)
{ {
struct asm9260_wdt_priv *priv = struct asm9260_wdt_priv *priv = watchdog_get_drvdata(wdd);
container_of(this, struct asm9260_wdt_priv, restart_handler);
asm9260_wdt_sys_reset(priv); asm9260_wdt_sys_reset(priv);
return NOTIFY_DONE; return 0;
} }
static const struct watchdog_info asm9260_wdt_ident = { static const struct watchdog_info asm9260_wdt_ident = {
...@@ -189,13 +186,14 @@ static const struct watchdog_info asm9260_wdt_ident = { ...@@ -189,13 +186,14 @@ static const struct watchdog_info asm9260_wdt_ident = {
.identity = "Alphascale asm9260 Watchdog", .identity = "Alphascale asm9260 Watchdog",
}; };
static struct watchdog_ops asm9260_wdt_ops = { static const struct watchdog_ops asm9260_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = asm9260_wdt_enable, .start = asm9260_wdt_enable,
.stop = asm9260_wdt_disable, .stop = asm9260_wdt_disable,
.get_timeleft = asm9260_wdt_gettimeleft, .get_timeleft = asm9260_wdt_gettimeleft,
.ping = asm9260_wdt_feed, .ping = asm9260_wdt_feed,
.set_timeout = asm9260_wdt_settimeout, .set_timeout = asm9260_wdt_settimeout,
.restart = asm9260_restart,
}; };
static int asm9260_wdt_get_dt_clks(struct asm9260_wdt_priv *priv) static int asm9260_wdt_get_dt_clks(struct asm9260_wdt_priv *priv)
...@@ -335,18 +333,14 @@ static int asm9260_wdt_probe(struct platform_device *pdev) ...@@ -335,18 +333,14 @@ static int asm9260_wdt_probe(struct platform_device *pdev)
dev_warn(&pdev->dev, "failed to request IRQ\n"); dev_warn(&pdev->dev, "failed to request IRQ\n");
} }
watchdog_set_restart_priority(wdd, 128);
ret = watchdog_register_device(wdd); ret = watchdog_register_device(wdd);
if (ret) if (ret)
goto clk_off; goto clk_off;
platform_set_drvdata(pdev, priv); platform_set_drvdata(pdev, priv);
priv->restart_handler.notifier_call = asm9260_restart_handler;
priv->restart_handler.priority = 128;
ret = register_restart_handler(&priv->restart_handler);
if (ret)
dev_warn(&pdev->dev, "cannot register restart handler\n");
dev_info(&pdev->dev, "Watchdog enabled (timeout: %d sec, mode: %s)\n", dev_info(&pdev->dev, "Watchdog enabled (timeout: %d sec, mode: %s)\n",
wdd->timeout, mode_name[priv->mode]); wdd->timeout, mode_name[priv->mode]);
return 0; return 0;
...@@ -370,8 +364,6 @@ static int asm9260_wdt_remove(struct platform_device *pdev) ...@@ -370,8 +364,6 @@ static int asm9260_wdt_remove(struct platform_device *pdev)
asm9260_wdt_disable(&priv->wdd); asm9260_wdt_disable(&priv->wdd);
unregister_restart_handler(&priv->restart_handler);
watchdog_unregister_device(&priv->wdd); watchdog_unregister_device(&priv->wdd);
clk_disable_unprepare(priv->clk); clk_disable_unprepare(priv->clk);
......
...@@ -136,15 +136,6 @@ static const struct watchdog_info aspeed_wdt_info = { ...@@ -136,15 +136,6 @@ static const struct watchdog_info aspeed_wdt_info = {
.identity = KBUILD_MODNAME, .identity = KBUILD_MODNAME,
}; };
static int aspeed_wdt_remove(struct platform_device *pdev)
{
struct aspeed_wdt *wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&wdt->wdd);
return 0;
}
static int aspeed_wdt_probe(struct platform_device *pdev) static int aspeed_wdt_probe(struct platform_device *pdev)
{ {
struct aspeed_wdt *wdt; struct aspeed_wdt *wdt;
...@@ -187,20 +178,17 @@ static int aspeed_wdt_probe(struct platform_device *pdev) ...@@ -187,20 +178,17 @@ static int aspeed_wdt_probe(struct platform_device *pdev)
set_bit(WDOG_HW_RUNNING, &wdt->wdd.status); set_bit(WDOG_HW_RUNNING, &wdt->wdd.status);
} }
ret = watchdog_register_device(&wdt->wdd); ret = devm_watchdog_register_device(&pdev->dev, &wdt->wdd);
if (ret) { if (ret) {
dev_err(&pdev->dev, "failed to register\n"); dev_err(&pdev->dev, "failed to register\n");
return ret; return ret;
} }
platform_set_drvdata(pdev, wdt);
return 0; return 0;
} }
static struct platform_driver aspeed_watchdog_driver = { static struct platform_driver aspeed_watchdog_driver = {
.probe = aspeed_wdt_probe, .probe = aspeed_wdt_probe,
.remove = aspeed_wdt_remove,
.driver = { .driver = {
.name = KBUILD_MODNAME, .name = KBUILD_MODNAME,
.of_match_table = of_match_ptr(aspeed_wdt_of_table), .of_match_table = of_match_ptr(aspeed_wdt_of_table),
......
...@@ -105,7 +105,7 @@ static const struct watchdog_info atlas7_wdt_ident = { ...@@ -105,7 +105,7 @@ static const struct watchdog_info atlas7_wdt_ident = {
.identity = "atlas7 Watchdog", .identity = "atlas7 Watchdog",
}; };
static struct watchdog_ops atlas7_wdt_ops = { static const struct watchdog_ops atlas7_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = atlas7_wdt_enable, .start = atlas7_wdt_enable,
.stop = atlas7_wdt_disable, .stop = atlas7_wdt_disable,
......
...@@ -14,7 +14,6 @@ ...@@ -14,7 +14,6 @@
*/ */
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/reboot.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/io.h> #include <linux/io.h>
...@@ -49,7 +48,6 @@ ...@@ -49,7 +48,6 @@
struct bcm2835_wdt { struct bcm2835_wdt {
void __iomem *base; void __iomem *base;
spinlock_t lock; spinlock_t lock;
struct notifier_block restart_handler;
}; };
static unsigned int heartbeat; static unsigned int heartbeat;
...@@ -99,11 +97,37 @@ static unsigned int bcm2835_wdt_get_timeleft(struct watchdog_device *wdog) ...@@ -99,11 +97,37 @@ static unsigned int bcm2835_wdt_get_timeleft(struct watchdog_device *wdog)
return WDOG_TICKS_TO_SECS(ret & PM_WDOG_TIME_SET); return WDOG_TICKS_TO_SECS(ret & PM_WDOG_TIME_SET);
} }
static void __bcm2835_restart(struct bcm2835_wdt *wdt)
{
u32 val;
/* use a timeout of 10 ticks (~150us) */
writel_relaxed(10 | PM_PASSWORD, wdt->base + PM_WDOG);
val = readl_relaxed(wdt->base + PM_RSTC);
val &= PM_RSTC_WRCFG_CLR;
val |= PM_PASSWORD | PM_RSTC_WRCFG_FULL_RESET;
writel_relaxed(val, wdt->base + PM_RSTC);
/* No sleeping, possibly atomic. */
mdelay(1);
}
static int bcm2835_restart(struct watchdog_device *wdog,
unsigned long action, void *data)
{
struct bcm2835_wdt *wdt = watchdog_get_drvdata(wdog);
__bcm2835_restart(wdt);
return 0;
}
static const struct watchdog_ops bcm2835_wdt_ops = { static const struct watchdog_ops bcm2835_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = bcm2835_wdt_start, .start = bcm2835_wdt_start,
.stop = bcm2835_wdt_stop, .stop = bcm2835_wdt_stop,
.get_timeleft = bcm2835_wdt_get_timeleft, .get_timeleft = bcm2835_wdt_get_timeleft,
.restart = bcm2835_restart,
}; };
static const struct watchdog_info bcm2835_wdt_info = { static const struct watchdog_info bcm2835_wdt_info = {
...@@ -120,26 +144,6 @@ static struct watchdog_device bcm2835_wdt_wdd = { ...@@ -120,26 +144,6 @@ static struct watchdog_device bcm2835_wdt_wdd = {
.timeout = WDOG_TICKS_TO_SECS(PM_WDOG_TIME_SET), .timeout = WDOG_TICKS_TO_SECS(PM_WDOG_TIME_SET),
}; };
static int
bcm2835_restart(struct notifier_block *this, unsigned long mode, void *cmd)
{
struct bcm2835_wdt *wdt = container_of(this, struct bcm2835_wdt,
restart_handler);
u32 val;
/* use a timeout of 10 ticks (~150us) */
writel_relaxed(10 | PM_PASSWORD, wdt->base + PM_WDOG);
val = readl_relaxed(wdt->base + PM_RSTC);
val &= PM_RSTC_WRCFG_CLR;
val |= PM_PASSWORD | PM_RSTC_WRCFG_FULL_RESET;
writel_relaxed(val, wdt->base + PM_RSTC);
/* No sleeping, possibly atomic. */
mdelay(1);
return 0;
}
/* /*
* We can't really power off, but if we do the normal reset scheme, and * We can't really power off, but if we do the normal reset scheme, and
* indicate to bootcode.bin not to reboot, then most of the chip will be * indicate to bootcode.bin not to reboot, then most of the chip will be
...@@ -163,13 +167,13 @@ static void bcm2835_power_off(void) ...@@ -163,13 +167,13 @@ static void bcm2835_power_off(void)
writel_relaxed(val, wdt->base + PM_RSTS); writel_relaxed(val, wdt->base + PM_RSTS);
/* Continue with normal reset mechanism */ /* Continue with normal reset mechanism */
bcm2835_restart(&wdt->restart_handler, REBOOT_HARD, NULL); __bcm2835_restart(wdt);
} }
static int bcm2835_wdt_probe(struct platform_device *pdev) static int bcm2835_wdt_probe(struct platform_device *pdev)
{ {
struct resource *res;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node;
struct bcm2835_wdt *wdt; struct bcm2835_wdt *wdt;
int err; int err;
...@@ -180,16 +184,15 @@ static int bcm2835_wdt_probe(struct platform_device *pdev) ...@@ -180,16 +184,15 @@ static int bcm2835_wdt_probe(struct platform_device *pdev)
spin_lock_init(&wdt->lock); spin_lock_init(&wdt->lock);
wdt->base = of_iomap(np, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!wdt->base) { wdt->base = devm_ioremap_resource(dev, res);
dev_err(dev, "Failed to remap watchdog regs"); if (IS_ERR(wdt->base))
return -ENODEV; return PTR_ERR(wdt->base);
}
watchdog_set_drvdata(&bcm2835_wdt_wdd, wdt); watchdog_set_drvdata(&bcm2835_wdt_wdd, wdt);
watchdog_init_timeout(&bcm2835_wdt_wdd, heartbeat, dev); watchdog_init_timeout(&bcm2835_wdt_wdd, heartbeat, dev);
watchdog_set_nowayout(&bcm2835_wdt_wdd, nowayout); watchdog_set_nowayout(&bcm2835_wdt_wdd, nowayout);
bcm2835_wdt_wdd.parent = &pdev->dev; bcm2835_wdt_wdd.parent = dev;
if (bcm2835_wdt_is_running(wdt)) { if (bcm2835_wdt_is_running(wdt)) {
/* /*
* The currently active timeout value (set by the * The currently active timeout value (set by the
...@@ -201,16 +204,16 @@ static int bcm2835_wdt_probe(struct platform_device *pdev) ...@@ -201,16 +204,16 @@ static int bcm2835_wdt_probe(struct platform_device *pdev)
*/ */
set_bit(WDOG_HW_RUNNING, &bcm2835_wdt_wdd.status); set_bit(WDOG_HW_RUNNING, &bcm2835_wdt_wdd.status);
} }
err = watchdog_register_device(&bcm2835_wdt_wdd);
watchdog_set_restart_priority(&bcm2835_wdt_wdd, 128);
watchdog_stop_on_reboot(&bcm2835_wdt_wdd);
err = devm_watchdog_register_device(dev, &bcm2835_wdt_wdd);
if (err) { if (err) {
dev_err(dev, "Failed to register watchdog device"); dev_err(dev, "Failed to register watchdog device");
iounmap(wdt->base);
return err; return err;
} }
wdt->restart_handler.notifier_call = bcm2835_restart;
wdt->restart_handler.priority = 128;
register_restart_handler(&wdt->restart_handler);
if (pm_power_off == NULL) if (pm_power_off == NULL)
pm_power_off = bcm2835_power_off; pm_power_off = bcm2835_power_off;
...@@ -220,22 +223,12 @@ static int bcm2835_wdt_probe(struct platform_device *pdev) ...@@ -220,22 +223,12 @@ static int bcm2835_wdt_probe(struct platform_device *pdev)
static int bcm2835_wdt_remove(struct platform_device *pdev) static int bcm2835_wdt_remove(struct platform_device *pdev)
{ {
struct bcm2835_wdt *wdt = platform_get_drvdata(pdev);
unregister_restart_handler(&wdt->restart_handler);
if (pm_power_off == bcm2835_power_off) if (pm_power_off == bcm2835_power_off)
pm_power_off = NULL; pm_power_off = NULL;
watchdog_unregister_device(&bcm2835_wdt_wdd);
iounmap(wdt->base);
return 0; return 0;
} }
static void bcm2835_wdt_shutdown(struct platform_device *pdev)
{
bcm2835_wdt_stop(&bcm2835_wdt_wdd);
}
static const struct of_device_id bcm2835_wdt_of_match[] = { static const struct of_device_id bcm2835_wdt_of_match[] = {
{ .compatible = "brcm,bcm2835-pm-wdt", }, { .compatible = "brcm,bcm2835-pm-wdt", },
{}, {},
...@@ -245,7 +238,6 @@ MODULE_DEVICE_TABLE(of, bcm2835_wdt_of_match); ...@@ -245,7 +238,6 @@ MODULE_DEVICE_TABLE(of, bcm2835_wdt_of_match);
static struct platform_driver bcm2835_wdt_driver = { static struct platform_driver bcm2835_wdt_driver = {
.probe = bcm2835_wdt_probe, .probe = bcm2835_wdt_probe,
.remove = bcm2835_wdt_remove, .remove = bcm2835_wdt_remove,
.shutdown = bcm2835_wdt_shutdown,
.driver = { .driver = {
.name = "bcm2835-wdt", .name = "bcm2835-wdt",
.of_match_table = bcm2835_wdt_of_match, .of_match_table = bcm2835_wdt_of_match,
......
...@@ -226,9 +226,6 @@ static int bcm47xx_wdt_remove(struct platform_device *pdev) ...@@ -226,9 +226,6 @@ static int bcm47xx_wdt_remove(struct platform_device *pdev)
{ {
struct bcm47xx_wdt *wdt = dev_get_platdata(&pdev->dev); struct bcm47xx_wdt *wdt = dev_get_platdata(&pdev->dev);
if (!wdt)
return -ENXIO;
watchdog_unregister_device(&wdt->wdd); watchdog_unregister_device(&wdt->wdd);
return 0; return 0;
......
...@@ -101,7 +101,7 @@ static unsigned int bcm7038_wdt_get_timeleft(struct watchdog_device *wdog) ...@@ -101,7 +101,7 @@ static unsigned int bcm7038_wdt_get_timeleft(struct watchdog_device *wdog)
return time_left / wdt->rate; return time_left / wdt->rate;
} }
static struct watchdog_info bcm7038_wdt_info = { static const struct watchdog_info bcm7038_wdt_info = {
.identity = "Broadcom BCM7038 Watchdog Timer", .identity = "Broadcom BCM7038 Watchdog Timer",
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING |
WDIOF_MAGICCLOSE WDIOF_MAGICCLOSE
......
...@@ -266,7 +266,7 @@ static int bcm_kona_wdt_stop(struct watchdog_device *wdog) ...@@ -266,7 +266,7 @@ static int bcm_kona_wdt_stop(struct watchdog_device *wdog)
SECWDOG_SRSTEN_MASK, 0); SECWDOG_SRSTEN_MASK, 0);
} }
static struct watchdog_ops bcm_kona_wdt_ops = { static const struct watchdog_ops bcm_kona_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = bcm_kona_wdt_start, .start = bcm_kona_wdt_start,
.stop = bcm_kona_wdt_stop, .stop = bcm_kona_wdt_stop,
...@@ -274,7 +274,7 @@ static struct watchdog_ops bcm_kona_wdt_ops = { ...@@ -274,7 +274,7 @@ static struct watchdog_ops bcm_kona_wdt_ops = {
.get_timeleft = bcm_kona_wdt_get_timeleft, .get_timeleft = bcm_kona_wdt_get_timeleft,
}; };
static struct watchdog_info bcm_kona_wdt_info = { static const struct watchdog_info bcm_kona_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE | .options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE |
WDIOF_KEEPALIVEPING, WDIOF_KEEPALIVEPING,
.identity = "Broadcom Kona Watchdog Timer", .identity = "Broadcom Kona Watchdog Timer",
......
...@@ -192,12 +192,12 @@ static int booke_wdt_set_timeout(struct watchdog_device *wdt_dev, ...@@ -192,12 +192,12 @@ static int booke_wdt_set_timeout(struct watchdog_device *wdt_dev,
return 0; return 0;
} }
static struct watchdog_info booke_wdt_info = { static struct watchdog_info booke_wdt_info __ro_after_init = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING,
.identity = "PowerPC Book-E Watchdog", .identity = "PowerPC Book-E Watchdog",
}; };
static struct watchdog_ops booke_wdt_ops = { static const struct watchdog_ops booke_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = booke_wdt_start, .start = booke_wdt_start,
.stop = booke_wdt_stop, .stop = booke_wdt_stop,
......
...@@ -262,7 +262,7 @@ static irqreturn_t cdns_wdt_irq_handler(int irq, void *dev_id) ...@@ -262,7 +262,7 @@ static irqreturn_t cdns_wdt_irq_handler(int irq, void *dev_id)
* Info structure used to indicate the features supported by the device * Info structure used to indicate the features supported by the device
* to the upper layers. This is defined in watchdog.h header file. * to the upper layers. This is defined in watchdog.h header file.
*/ */
static struct watchdog_info cdns_wdt_info = { static const struct watchdog_info cdns_wdt_info = {
.identity = "cdns_wdt watchdog", .identity = "cdns_wdt watchdog",
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING |
WDIOF_MAGICCLOSE, WDIOF_MAGICCLOSE,
......
...@@ -68,17 +68,10 @@ ...@@ -68,17 +68,10 @@
/* Default timeout in seconds = 1 minute */ /* Default timeout in seconds = 1 minute */
static unsigned int margin = 60; static unsigned int margin = 60;
static resource_size_t phybase;
static resource_size_t physize;
static int irq; static int irq;
static void __iomem *virtbase; static void __iomem *virtbase;
static struct device *parent; static struct device *parent;
/*
* The watchdog block is of course always clocked, the
* clk_enable()/clk_disable() calls are mainly for performing reference
* counting higher up in the clock hierarchy.
*/
static struct clk *clk; static struct clk *clk;
/* /*
...@@ -90,7 +83,6 @@ static void coh901327_enable(u16 timeout) ...@@ -90,7 +83,6 @@ static void coh901327_enable(u16 timeout)
unsigned long freq; unsigned long freq;
unsigned long delay_ns; unsigned long delay_ns;
clk_enable(clk);
/* Restart timer if it is disabled */ /* Restart timer if it is disabled */
val = readw(virtbase + U300_WDOG_D2R); val = readw(virtbase + U300_WDOG_D2R);
if (val == U300_WDOG_D2R_DISABLE_STATUS_DISABLED) if (val == U300_WDOG_D2R_DISABLE_STATUS_DISABLED)
...@@ -118,7 +110,6 @@ static void coh901327_enable(u16 timeout) ...@@ -118,7 +110,6 @@ static void coh901327_enable(u16 timeout)
*/ */
(void) readw(virtbase + U300_WDOG_CR); (void) readw(virtbase + U300_WDOG_CR);
val = readw(virtbase + U300_WDOG_D2R); val = readw(virtbase + U300_WDOG_D2R);
clk_disable(clk);
if (val != U300_WDOG_D2R_DISABLE_STATUS_ENABLED) if (val != U300_WDOG_D2R_DISABLE_STATUS_ENABLED)
dev_err(parent, dev_err(parent,
"%s(): watchdog not enabled! D2R value %04x\n", "%s(): watchdog not enabled! D2R value %04x\n",
...@@ -129,7 +120,6 @@ static void coh901327_disable(void) ...@@ -129,7 +120,6 @@ static void coh901327_disable(void)
{ {
u16 val; u16 val;
clk_enable(clk);
/* Disable the watchdog interrupt if it is active */ /* Disable the watchdog interrupt if it is active */
writew(0x0000U, virtbase + U300_WDOG_IMR); writew(0x0000U, virtbase + U300_WDOG_IMR);
/* If the watchdog is currently enabled, attempt to disable it */ /* If the watchdog is currently enabled, attempt to disable it */
...@@ -144,7 +134,6 @@ static void coh901327_disable(void) ...@@ -144,7 +134,6 @@ static void coh901327_disable(void)
virtbase + U300_WDOG_D2R); virtbase + U300_WDOG_D2R);
} }
val = readw(virtbase + U300_WDOG_D2R); val = readw(virtbase + U300_WDOG_D2R);
clk_disable(clk);
if (val != U300_WDOG_D2R_DISABLE_STATUS_DISABLED) if (val != U300_WDOG_D2R_DISABLE_STATUS_DISABLED)
dev_err(parent, dev_err(parent,
"%s(): watchdog not disabled! D2R value %04x\n", "%s(): watchdog not disabled! D2R value %04x\n",
...@@ -165,11 +154,9 @@ static int coh901327_stop(struct watchdog_device *wdt_dev) ...@@ -165,11 +154,9 @@ static int coh901327_stop(struct watchdog_device *wdt_dev)
static int coh901327_ping(struct watchdog_device *wdd) static int coh901327_ping(struct watchdog_device *wdd)
{ {
clk_enable(clk);
/* Feed the watchdog */ /* Feed the watchdog */
writew(U300_WDOG_FR_FEED_RESTART_TIMER, writew(U300_WDOG_FR_FEED_RESTART_TIMER,
virtbase + U300_WDOG_FR); virtbase + U300_WDOG_FR);
clk_disable(clk);
return 0; return 0;
} }
...@@ -177,13 +164,11 @@ static int coh901327_settimeout(struct watchdog_device *wdt_dev, ...@@ -177,13 +164,11 @@ static int coh901327_settimeout(struct watchdog_device *wdt_dev,
unsigned int time) unsigned int time)
{ {
wdt_dev->timeout = time; wdt_dev->timeout = time;
clk_enable(clk);
/* Set new timeout value */ /* Set new timeout value */
writew(time * 100, virtbase + U300_WDOG_TR); writew(time * 100, virtbase + U300_WDOG_TR);
/* Feed the dog */ /* Feed the dog */
writew(U300_WDOG_FR_FEED_RESTART_TIMER, writew(U300_WDOG_FR_FEED_RESTART_TIMER,
virtbase + U300_WDOG_FR); virtbase + U300_WDOG_FR);
clk_disable(clk);
return 0; return 0;
} }
...@@ -191,13 +176,11 @@ static unsigned int coh901327_gettimeleft(struct watchdog_device *wdt_dev) ...@@ -191,13 +176,11 @@ static unsigned int coh901327_gettimeleft(struct watchdog_device *wdt_dev)
{ {
u16 val; u16 val;
clk_enable(clk);
/* Read repeatedly until the value is stable! */ /* Read repeatedly until the value is stable! */
val = readw(virtbase + U300_WDOG_CR); val = readw(virtbase + U300_WDOG_CR);
while (val & U300_WDOG_CR_VALID_IND) while (val & U300_WDOG_CR_VALID_IND)
val = readw(virtbase + U300_WDOG_CR); val = readw(virtbase + U300_WDOG_CR);
val &= U300_WDOG_CR_COUNT_VALUE_MASK; val &= U300_WDOG_CR_COUNT_VALUE_MASK;
clk_disable(clk);
if (val != 0) if (val != 0)
val /= 100; val /= 100;
...@@ -221,13 +204,11 @@ static irqreturn_t coh901327_interrupt(int irq, void *data) ...@@ -221,13 +204,11 @@ static irqreturn_t coh901327_interrupt(int irq, void *data)
* to prevent a watchdog reset by feeding the watchdog at this * to prevent a watchdog reset by feeding the watchdog at this
* point. * point.
*/ */
clk_enable(clk);
val = readw(virtbase + U300_WDOG_IER); val = readw(virtbase + U300_WDOG_IER);
if (val == U300_WDOG_IER_WILL_BARK_IRQ_EVENT_IND) if (val == U300_WDOG_IER_WILL_BARK_IRQ_EVENT_IND)
writew(U300_WDOG_IER_WILL_BARK_IRQ_ACK_ENABLE, writew(U300_WDOG_IER_WILL_BARK_IRQ_ACK_ENABLE,
virtbase + U300_WDOG_IER); virtbase + U300_WDOG_IER);
writew(0x0000U, virtbase + U300_WDOG_IMR); writew(0x0000U, virtbase + U300_WDOG_IMR);
clk_disable(clk);
dev_crit(parent, "watchdog is barking!\n"); dev_crit(parent, "watchdog is barking!\n");
return IRQ_HANDLED; return IRQ_HANDLED;
} }
...@@ -263,81 +244,63 @@ static int __exit coh901327_remove(struct platform_device *pdev) ...@@ -263,81 +244,63 @@ static int __exit coh901327_remove(struct platform_device *pdev)
watchdog_unregister_device(&coh901327_wdt); watchdog_unregister_device(&coh901327_wdt);
coh901327_disable(); coh901327_disable();
free_irq(irq, pdev); free_irq(irq, pdev);
clk_unprepare(clk); clk_disable_unprepare(clk);
clk_put(clk); clk_put(clk);
iounmap(virtbase);
release_mem_region(phybase, physize);
return 0; return 0;
} }
static int __init coh901327_probe(struct platform_device *pdev) static int __init coh901327_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev;
int ret; int ret;
u16 val; u16 val;
struct resource *res; struct resource *res;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); parent = dev;
if (!res)
return -ENOENT;
parent = &pdev->dev;
physize = resource_size(res);
phybase = res->start;
if (request_mem_region(phybase, physize, DRV_NAME) == NULL) { res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
ret = -EBUSY; virtbase = devm_ioremap_resource(dev, res);
goto out; if (IS_ERR(virtbase))
} return PTR_ERR(virtbase);
virtbase = ioremap(phybase, physize);
if (!virtbase) {
ret = -ENOMEM;
goto out_no_remap;
}
clk = clk_get(&pdev->dev, NULL); clk = clk_get(dev, NULL);
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
ret = PTR_ERR(clk); ret = PTR_ERR(clk);
dev_err(&pdev->dev, "could not get clock\n"); dev_err(dev, "could not get clock\n");
goto out_no_clk; return ret;
} }
ret = clk_prepare_enable(clk); ret = clk_prepare_enable(clk);
if (ret) { if (ret) {
dev_err(&pdev->dev, "could not prepare and enable clock\n"); dev_err(dev, "could not prepare and enable clock\n");
goto out_no_clk_enable; goto out_no_clk_enable;
} }
val = readw(virtbase + U300_WDOG_SR); val = readw(virtbase + U300_WDOG_SR);
switch (val) { switch (val) {
case U300_WDOG_SR_STATUS_TIMED_OUT: case U300_WDOG_SR_STATUS_TIMED_OUT:
dev_info(&pdev->dev, dev_info(dev, "watchdog timed out since last chip reset!\n");
"watchdog timed out since last chip reset!\n");
coh901327_wdt.bootstatus |= WDIOF_CARDRESET; coh901327_wdt.bootstatus |= WDIOF_CARDRESET;
/* Status will be cleared below */ /* Status will be cleared below */
break; break;
case U300_WDOG_SR_STATUS_NORMAL: case U300_WDOG_SR_STATUS_NORMAL:
dev_info(&pdev->dev, dev_info(dev, "in normal status, no timeouts have occurred.\n");
"in normal status, no timeouts have occurred.\n");
break; break;
default: default:
dev_info(&pdev->dev, dev_info(dev, "contains an illegal status code (%08x)\n", val);
"contains an illegal status code (%08x)\n", val);
break; break;
} }
val = readw(virtbase + U300_WDOG_D2R); val = readw(virtbase + U300_WDOG_D2R);
switch (val) { switch (val) {
case U300_WDOG_D2R_DISABLE_STATUS_DISABLED: case U300_WDOG_D2R_DISABLE_STATUS_DISABLED:
dev_info(&pdev->dev, "currently disabled.\n"); dev_info(dev, "currently disabled.\n");
break; break;
case U300_WDOG_D2R_DISABLE_STATUS_ENABLED: case U300_WDOG_D2R_DISABLE_STATUS_ENABLED:
dev_info(&pdev->dev, dev_info(dev, "currently enabled! (disabling it now)\n");
"currently enabled! (disabling it now)\n");
coh901327_disable(); coh901327_disable();
break; break;
default: default:
dev_err(&pdev->dev, dev_err(dev, "contains an illegal enable/disable code (%08x)\n",
"contains an illegal enable/disable code (%08x)\n",
val); val);
break; break;
} }
...@@ -352,20 +315,16 @@ static int __init coh901327_probe(struct platform_device *pdev) ...@@ -352,20 +315,16 @@ static int __init coh901327_probe(struct platform_device *pdev)
goto out_no_irq; goto out_no_irq;
} }
clk_disable(clk); ret = watchdog_init_timeout(&coh901327_wdt, margin, dev);
ret = watchdog_init_timeout(&coh901327_wdt, margin, &pdev->dev);
if (ret < 0) if (ret < 0)
coh901327_wdt.timeout = 60; coh901327_wdt.timeout = 60;
coh901327_wdt.parent = &pdev->dev; coh901327_wdt.parent = dev;
ret = watchdog_register_device(&coh901327_wdt); ret = watchdog_register_device(&coh901327_wdt);
if (ret == 0) if (ret)
dev_info(&pdev->dev,
"initialized. timer margin=%d sec\n", margin);
else
goto out_no_wdog; goto out_no_wdog;
dev_info(dev, "initialized. timer margin=%d sec\n", margin);
return 0; return 0;
out_no_wdog: out_no_wdog:
...@@ -374,11 +333,6 @@ static int __init coh901327_probe(struct platform_device *pdev) ...@@ -374,11 +333,6 @@ static int __init coh901327_probe(struct platform_device *pdev)
clk_disable_unprepare(clk); clk_disable_unprepare(clk);
out_no_clk_enable: out_no_clk_enable:
clk_put(clk); clk_put(clk);
out_no_clk:
iounmap(virtbase);
out_no_remap:
release_mem_region(phybase, SZ_4K);
out:
return ret; return ret;
} }
......
...@@ -128,19 +128,17 @@ static int da9052_wdt_ping(struct watchdog_device *wdt_dev) ...@@ -128,19 +128,17 @@ static int da9052_wdt_ping(struct watchdog_device *wdt_dev)
ret = da9052_reg_update(da9052, DA9052_CONTROL_D_REG, ret = da9052_reg_update(da9052, DA9052_CONTROL_D_REG,
DA9052_CONTROLD_WATCHDOG, 1 << 7); DA9052_CONTROLD_WATCHDOG, 1 << 7);
if (ret < 0) if (ret < 0)
goto err_strobe; return ret;
/* /*
* FIXME: Reset the watchdog core, in general PMIC * FIXME: Reset the watchdog core, in general PMIC
* is supposed to do this * is supposed to do this
*/ */
ret = da9052_reg_update(da9052, DA9052_CONTROL_D_REG, return da9052_reg_update(da9052, DA9052_CONTROL_D_REG,
DA9052_CONTROLD_WATCHDOG, 0 << 7); DA9052_CONTROLD_WATCHDOG, 0 << 7);
err_strobe:
return ret;
} }
static struct watchdog_info da9052_wdt_info = { static const struct watchdog_info da9052_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING,
.identity = "DA9052 Watchdog", .identity = "DA9052 Watchdog",
}; };
...@@ -163,10 +161,8 @@ static int da9052_wdt_probe(struct platform_device *pdev) ...@@ -163,10 +161,8 @@ static int da9052_wdt_probe(struct platform_device *pdev)
driver_data = devm_kzalloc(&pdev->dev, sizeof(*driver_data), driver_data = devm_kzalloc(&pdev->dev, sizeof(*driver_data),
GFP_KERNEL); GFP_KERNEL);
if (!driver_data) { if (!driver_data)
ret = -ENOMEM; return -ENOMEM;
goto err;
}
driver_data->da9052 = da9052; driver_data->da9052 = da9052;
da9052_wdt = &driver_data->wdt; da9052_wdt = &driver_data->wdt;
...@@ -182,33 +178,21 @@ static int da9052_wdt_probe(struct platform_device *pdev) ...@@ -182,33 +178,21 @@ static int da9052_wdt_probe(struct platform_device *pdev)
if (ret < 0) { if (ret < 0) {
dev_err(&pdev->dev, "Failed to disable watchdog bits, %d\n", dev_err(&pdev->dev, "Failed to disable watchdog bits, %d\n",
ret); ret);
goto err; return ret;
} }
ret = watchdog_register_device(&driver_data->wdt); ret = devm_watchdog_register_device(&pdev->dev, &driver_data->wdt);
if (ret != 0) { if (ret != 0) {
dev_err(da9052->dev, "watchdog_register_device() failed: %d\n", dev_err(da9052->dev, "watchdog_register_device() failed: %d\n",
ret); ret);
goto err; return ret;
} }
platform_set_drvdata(pdev, driver_data);
err:
return ret; return ret;
} }
static int da9052_wdt_remove(struct platform_device *pdev)
{
struct da9052_wdt_data *driver_data = platform_get_drvdata(pdev);
watchdog_unregister_device(&driver_data->wdt);
return 0;
}
static struct platform_driver da9052_wdt_driver = { static struct platform_driver da9052_wdt_driver = {
.probe = da9052_wdt_probe, .probe = da9052_wdt_probe,
.remove = da9052_wdt_remove,
.driver = { .driver = {
.name = "da9052-watchdog", .name = "da9052-watchdog",
}, },
......
...@@ -108,7 +108,7 @@ static int da9055_wdt_stop(struct watchdog_device *wdt_dev) ...@@ -108,7 +108,7 @@ static int da9055_wdt_stop(struct watchdog_device *wdt_dev)
return da9055_wdt_set_timeout(wdt_dev, 0); return da9055_wdt_set_timeout(wdt_dev, 0);
} }
static struct watchdog_info da9055_wdt_info = { static const struct watchdog_info da9055_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING,
.identity = "DA9055 Watchdog", .identity = "DA9055 Watchdog",
}; };
...@@ -147,32 +147,19 @@ static int da9055_wdt_probe(struct platform_device *pdev) ...@@ -147,32 +147,19 @@ static int da9055_wdt_probe(struct platform_device *pdev)
ret = da9055_wdt_stop(da9055_wdt); ret = da9055_wdt_stop(da9055_wdt);
if (ret < 0) { if (ret < 0) {
dev_err(&pdev->dev, "Failed to stop watchdog, %d\n", ret); dev_err(&pdev->dev, "Failed to stop watchdog, %d\n", ret);
goto err; return ret;
} }
platform_set_drvdata(pdev, driver_data); ret = devm_watchdog_register_device(&pdev->dev, &driver_data->wdt);
ret = watchdog_register_device(&driver_data->wdt);
if (ret != 0) if (ret != 0)
dev_err(da9055->dev, "watchdog_register_device() failed: %d\n", dev_err(da9055->dev, "watchdog_register_device() failed: %d\n",
ret); ret);
err:
return ret; return ret;
} }
static int da9055_wdt_remove(struct platform_device *pdev)
{
struct da9055_wdt_data *driver_data = platform_get_drvdata(pdev);
watchdog_unregister_device(&driver_data->wdt);
return 0;
}
static struct platform_driver da9055_wdt_driver = { static struct platform_driver da9055_wdt_driver = {
.probe = da9055_wdt_probe, .probe = da9055_wdt_probe,
.remove = da9055_wdt_remove,
.driver = { .driver = {
.name = "da9055-watchdog", .name = "da9055-watchdog",
}, },
......
...@@ -220,9 +220,8 @@ static int da9062_wdt_probe(struct platform_device *pdev) ...@@ -220,9 +220,8 @@ static int da9062_wdt_probe(struct platform_device *pdev)
wdt->wdtdev.parent = &pdev->dev; wdt->wdtdev.parent = &pdev->dev;
watchdog_set_drvdata(&wdt->wdtdev, wdt); watchdog_set_drvdata(&wdt->wdtdev, wdt);
dev_set_drvdata(&pdev->dev, wdt);
ret = watchdog_register_device(&wdt->wdtdev); ret = devm_watchdog_register_device(&pdev->dev, &wdt->wdtdev);
if (ret < 0) { if (ret < 0) {
dev_err(wdt->hw->dev, dev_err(wdt->hw->dev,
"watchdog registration failed (%d)\n", ret); "watchdog registration failed (%d)\n", ret);
...@@ -231,24 +230,11 @@ static int da9062_wdt_probe(struct platform_device *pdev) ...@@ -231,24 +230,11 @@ static int da9062_wdt_probe(struct platform_device *pdev)
da9062_set_window_start(wdt); da9062_set_window_start(wdt);
ret = da9062_wdt_ping(&wdt->wdtdev); return da9062_wdt_ping(&wdt->wdtdev);
if (ret < 0)
watchdog_unregister_device(&wdt->wdtdev);
return ret;
}
static int da9062_wdt_remove(struct platform_device *pdev)
{
struct da9062_watchdog *wdt = dev_get_drvdata(&pdev->dev);
watchdog_unregister_device(&wdt->wdtdev);
return 0;
} }
static struct platform_driver da9062_wdt_driver = { static struct platform_driver da9062_wdt_driver = {
.probe = da9062_wdt_probe, .probe = da9062_wdt_probe,
.remove = da9062_wdt_remove,
.driver = { .driver = {
.name = "da9062-watchdog", .name = "da9062-watchdog",
.of_match_table = da9062_compatible_id_table, .of_match_table = da9062_compatible_id_table,
......
...@@ -151,7 +151,6 @@ static const struct watchdog_ops da9063_watchdog_ops = { ...@@ -151,7 +151,6 @@ static const struct watchdog_ops da9063_watchdog_ops = {
static int da9063_wdt_probe(struct platform_device *pdev) static int da9063_wdt_probe(struct platform_device *pdev)
{ {
int ret;
struct da9063 *da9063; struct da9063 *da9063;
struct da9063_watchdog *wdt; struct da9063_watchdog *wdt;
...@@ -181,27 +180,12 @@ static int da9063_wdt_probe(struct platform_device *pdev) ...@@ -181,27 +180,12 @@ static int da9063_wdt_probe(struct platform_device *pdev)
watchdog_set_restart_priority(&wdt->wdtdev, 128); watchdog_set_restart_priority(&wdt->wdtdev, 128);
watchdog_set_drvdata(&wdt->wdtdev, wdt); watchdog_set_drvdata(&wdt->wdtdev, wdt);
dev_set_drvdata(&pdev->dev, wdt);
ret = watchdog_register_device(&wdt->wdtdev);
if (ret)
return ret;
return 0;
}
static int da9063_wdt_remove(struct platform_device *pdev)
{
struct da9063_watchdog *wdt = dev_get_drvdata(&pdev->dev);
watchdog_unregister_device(&wdt->wdtdev);
return 0; return devm_watchdog_register_device(&pdev->dev, &wdt->wdtdev);
} }
static struct platform_driver da9063_wdt_driver = { static struct platform_driver da9063_wdt_driver = {
.probe = da9063_wdt_probe, .probe = da9063_wdt_probe,
.remove = da9063_wdt_remove,
.driver = { .driver = {
.name = DA9063_DRVNAME_WATCHDOG, .name = DA9063_DRVNAME_WATCHDOG,
}, },
......
...@@ -205,7 +205,7 @@ static int wdt_set_timeout(struct watchdog_device * dev, unsigned int new_to) ...@@ -205,7 +205,7 @@ static int wdt_set_timeout(struct watchdog_device * dev, unsigned int new_to)
return wdt_ping(dev); return wdt_ping(dev);
} }
static struct watchdog_ops wdt_ops = { static const struct watchdog_ops wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = wdt_start, .start = wdt_start,
.stop = wdt_stop, .stop = wdt_stop,
......
...@@ -96,7 +96,7 @@ static unsigned int dc_wdt_get_timeleft(struct watchdog_device *wdog) ...@@ -96,7 +96,7 @@ static unsigned int dc_wdt_get_timeleft(struct watchdog_device *wdog)
return count / clk_get_rate(wdt->clk); return count / clk_get_rate(wdt->clk);
} }
static struct watchdog_ops dc_wdt_ops = { static const struct watchdog_ops dc_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = dc_wdt_start, .start = dc_wdt_start,
.stop = dc_wdt_stop, .stop = dc_wdt_stop,
...@@ -105,7 +105,7 @@ static struct watchdog_ops dc_wdt_ops = { ...@@ -105,7 +105,7 @@ static struct watchdog_ops dc_wdt_ops = {
.restart = dc_wdt_restart, .restart = dc_wdt_restart,
}; };
static struct watchdog_info dc_wdt_info = { static const struct watchdog_info dc_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE .options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE
| WDIOF_KEEPALIVEPING, | WDIOF_KEEPALIVEPING,
.identity = "Conexant Digicolor Watchdog", .identity = "Conexant Digicolor Watchdog",
...@@ -119,62 +119,40 @@ static struct watchdog_device dc_wdt_wdd = { ...@@ -119,62 +119,40 @@ static struct watchdog_device dc_wdt_wdd = {
static int dc_wdt_probe(struct platform_device *pdev) static int dc_wdt_probe(struct platform_device *pdev)
{ {
struct resource *res;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node;
struct dc_wdt *wdt; struct dc_wdt *wdt;
int ret; int ret;
wdt = devm_kzalloc(dev, sizeof(struct dc_wdt), GFP_KERNEL); wdt = devm_kzalloc(dev, sizeof(struct dc_wdt), GFP_KERNEL);
if (!wdt) if (!wdt)
return -ENOMEM; return -ENOMEM;
platform_set_drvdata(pdev, wdt);
wdt->base = of_iomap(np, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!wdt->base) { wdt->base = devm_ioremap_resource(dev, res);
dev_err(dev, "Failed to remap watchdog regs"); if (IS_ERR(wdt->base))
return -ENODEV; return PTR_ERR(wdt->base);
}
wdt->clk = devm_clk_get(&pdev->dev, NULL); wdt->clk = devm_clk_get(dev, NULL);
if (IS_ERR(wdt->clk)) { if (IS_ERR(wdt->clk))
ret = PTR_ERR(wdt->clk); return PTR_ERR(wdt->clk);
goto err_iounmap;
}
dc_wdt_wdd.max_timeout = U32_MAX / clk_get_rate(wdt->clk); dc_wdt_wdd.max_timeout = U32_MAX / clk_get_rate(wdt->clk);
dc_wdt_wdd.timeout = dc_wdt_wdd.max_timeout; dc_wdt_wdd.timeout = dc_wdt_wdd.max_timeout;
dc_wdt_wdd.parent = &pdev->dev; dc_wdt_wdd.parent = dev;
spin_lock_init(&wdt->lock); spin_lock_init(&wdt->lock);
watchdog_set_drvdata(&dc_wdt_wdd, wdt); watchdog_set_drvdata(&dc_wdt_wdd, wdt);
watchdog_set_restart_priority(&dc_wdt_wdd, 128); watchdog_set_restart_priority(&dc_wdt_wdd, 128);
watchdog_init_timeout(&dc_wdt_wdd, timeout, dev); watchdog_init_timeout(&dc_wdt_wdd, timeout, dev);
ret = watchdog_register_device(&dc_wdt_wdd); watchdog_stop_on_reboot(&dc_wdt_wdd);
ret = devm_watchdog_register_device(dev, &dc_wdt_wdd);
if (ret) { if (ret) {
dev_err(dev, "Failed to register watchdog device"); dev_err(dev, "Failed to register watchdog device");
goto err_iounmap; return ret;
} }
return 0; return 0;
err_iounmap:
iounmap(wdt->base);
return ret;
}
static int dc_wdt_remove(struct platform_device *pdev)
{
struct dc_wdt *wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&dc_wdt_wdd);
iounmap(wdt->base);
return 0;
}
static void dc_wdt_shutdown(struct platform_device *pdev)
{
dc_wdt_stop(&dc_wdt_wdd);
} }
static const struct of_device_id dc_wdt_of_match[] = { static const struct of_device_id dc_wdt_of_match[] = {
...@@ -185,8 +163,6 @@ MODULE_DEVICE_TABLE(of, dc_wdt_of_match); ...@@ -185,8 +163,6 @@ MODULE_DEVICE_TABLE(of, dc_wdt_of_match);
static struct platform_driver dc_wdt_driver = { static struct platform_driver dc_wdt_driver = {
.probe = dc_wdt_probe, .probe = dc_wdt_probe,
.remove = dc_wdt_remove,
.shutdown = dc_wdt_shutdown,
.driver = { .driver = {
.name = "digicolor-wdt", .name = "digicolor-wdt",
.of_match_table = dc_wdt_of_match, .of_match_table = dc_wdt_of_match,
......
...@@ -26,11 +26,9 @@ ...@@ -26,11 +26,9 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/notifier.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/pm.h> #include <linux/pm.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/reboot.h>
#include <linux/watchdog.h> #include <linux/watchdog.h>
#define WDOG_CONTROL_REG_OFFSET 0x00 #define WDOG_CONTROL_REG_OFFSET 0x00
...@@ -55,7 +53,6 @@ struct dw_wdt { ...@@ -55,7 +53,6 @@ struct dw_wdt {
void __iomem *regs; void __iomem *regs;
struct clk *clk; struct clk *clk;
unsigned long rate; unsigned long rate;
struct notifier_block restart_handler;
struct watchdog_device wdd; struct watchdog_device wdd;
}; };
...@@ -136,14 +133,12 @@ static int dw_wdt_start(struct watchdog_device *wdd) ...@@ -136,14 +133,12 @@ static int dw_wdt_start(struct watchdog_device *wdd)
return 0; return 0;
} }
static int dw_wdt_restart_handle(struct notifier_block *this, static int dw_wdt_restart(struct watchdog_device *wdd,
unsigned long mode, void *cmd) unsigned long action, void *data)
{ {
struct dw_wdt *dw_wdt; struct dw_wdt *dw_wdt = to_dw_wdt(wdd);
u32 val; u32 val;
dw_wdt = container_of(this, struct dw_wdt, restart_handler);
writel(0, dw_wdt->regs + WDOG_TIMEOUT_RANGE_REG_OFFSET); writel(0, dw_wdt->regs + WDOG_TIMEOUT_RANGE_REG_OFFSET);
val = readl(dw_wdt->regs + WDOG_CONTROL_REG_OFFSET); val = readl(dw_wdt->regs + WDOG_CONTROL_REG_OFFSET);
if (val & WDOG_CONTROL_REG_WDT_EN_MASK) if (val & WDOG_CONTROL_REG_WDT_EN_MASK)
...@@ -156,7 +151,7 @@ static int dw_wdt_restart_handle(struct notifier_block *this, ...@@ -156,7 +151,7 @@ static int dw_wdt_restart_handle(struct notifier_block *this,
/* wait for reset to assert... */ /* wait for reset to assert... */
mdelay(500); mdelay(500);
return NOTIFY_DONE; return 0;
} }
static unsigned int dw_wdt_get_timeleft(struct watchdog_device *wdd) static unsigned int dw_wdt_get_timeleft(struct watchdog_device *wdd)
...@@ -179,6 +174,7 @@ static const struct watchdog_ops dw_wdt_ops = { ...@@ -179,6 +174,7 @@ static const struct watchdog_ops dw_wdt_ops = {
.ping = dw_wdt_ping, .ping = dw_wdt_ping,
.set_timeout = dw_wdt_set_timeout, .set_timeout = dw_wdt_set_timeout,
.get_timeleft = dw_wdt_get_timeleft, .get_timeleft = dw_wdt_get_timeleft,
.restart = dw_wdt_restart,
}; };
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
...@@ -265,16 +261,12 @@ static int dw_wdt_drv_probe(struct platform_device *pdev) ...@@ -265,16 +261,12 @@ static int dw_wdt_drv_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, dw_wdt); platform_set_drvdata(pdev, dw_wdt);
watchdog_set_restart_priority(wdd, 128);
ret = watchdog_register_device(wdd); ret = watchdog_register_device(wdd);
if (ret) if (ret)
goto out_disable_clk; goto out_disable_clk;
dw_wdt->restart_handler.notifier_call = dw_wdt_restart_handle;
dw_wdt->restart_handler.priority = 128;
ret = register_restart_handler(&dw_wdt->restart_handler);
if (ret)
pr_warn("cannot register restart handler\n");
return 0; return 0;
out_disable_clk: out_disable_clk:
...@@ -286,7 +278,6 @@ static int dw_wdt_drv_remove(struct platform_device *pdev) ...@@ -286,7 +278,6 @@ static int dw_wdt_drv_remove(struct platform_device *pdev)
{ {
struct dw_wdt *dw_wdt = platform_get_drvdata(pdev); struct dw_wdt *dw_wdt = platform_get_drvdata(pdev);
unregister_restart_handler(&dw_wdt->restart_handler);
watchdog_unregister_device(&dw_wdt->wdd); watchdog_unregister_device(&dw_wdt->wdd);
clk_disable_unprepare(dw_wdt->clk); clk_disable_unprepare(dw_wdt->clk);
......
...@@ -121,18 +121,7 @@ static int ebc_c384_wdt_probe(struct device *dev, unsigned int id) ...@@ -121,18 +121,7 @@ static int ebc_c384_wdt_probe(struct device *dev, unsigned int id)
dev_warn(dev, "Invalid timeout (%u seconds), using default (%u seconds)\n", dev_warn(dev, "Invalid timeout (%u seconds), using default (%u seconds)\n",
timeout, WATCHDOG_TIMEOUT); timeout, WATCHDOG_TIMEOUT);
dev_set_drvdata(dev, wdd); return devm_watchdog_register_device(dev, wdd);
return watchdog_register_device(wdd);
}
static int ebc_c384_wdt_remove(struct device *dev, unsigned int id)
{
struct watchdog_device *wdd = dev_get_drvdata(dev);
watchdog_unregister_device(wdd);
return 0;
} }
static struct isa_driver ebc_c384_wdt_driver = { static struct isa_driver ebc_c384_wdt_driver = {
...@@ -140,7 +129,6 @@ static struct isa_driver ebc_c384_wdt_driver = { ...@@ -140,7 +129,6 @@ static struct isa_driver ebc_c384_wdt_driver = {
.driver = { .driver = {
.name = MODULE_NAME .name = MODULE_NAME
}, },
.remove = ebc_c384_wdt_remove
}; };
static int __init ebc_c384_wdt_init(void) static int __init ebc_c384_wdt_init(void)
......
...@@ -19,21 +19,13 @@ ...@@ -19,21 +19,13 @@
* for us to rely on the user space daemon alone. So we ping the * for us to rely on the user space daemon alone. So we ping the
* wdt each ~200msec and eventually stop doing it if the user space * wdt each ~200msec and eventually stop doing it if the user space
* daemon dies. * daemon dies.
*
* TODO:
*
* - Test last reset from watchdog status
* - Add a few missing ioctls
*/ */
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/watchdog.h> #include <linux/watchdog.h>
#include <linux/timer.h>
#include <linux/io.h> #include <linux/io.h>
#define WDT_VERSION "0.4"
/* default timeout (secs) */ /* default timeout (secs) */
#define WDT_TIMEOUT 30 #define WDT_TIMEOUT 30
...@@ -41,117 +33,101 @@ static bool nowayout = WATCHDOG_NOWAYOUT; ...@@ -41,117 +33,101 @@ static bool nowayout = WATCHDOG_NOWAYOUT;
module_param(nowayout, bool, 0); module_param(nowayout, bool, 0);
MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started"); MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started");
static unsigned int timeout = WDT_TIMEOUT; static unsigned int timeout;
module_param(timeout, uint, 0); module_param(timeout, uint, 0);
MODULE_PARM_DESC(timeout, MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds.");
"Watchdog timeout in seconds. (1<=timeout<=3600, default="
__MODULE_STRING(WDT_TIMEOUT) ")");
static void __iomem *mmio_base;
static struct timer_list timer;
static unsigned long next_heartbeat;
#define EP93XX_WATCHDOG 0x00 #define EP93XX_WATCHDOG 0x00
#define EP93XX_WDSTATUS 0x04 #define EP93XX_WDSTATUS 0x04
/* reset the wdt every ~200ms - the heartbeat of the device is 0.250 seconds*/ struct ep93xx_wdt_priv {
#define WDT_INTERVAL (HZ/5) void __iomem *mmio;
struct watchdog_device wdd;
static void ep93xx_wdt_timer_ping(unsigned long data) };
{
if (time_before(jiffies, next_heartbeat))
writel(0x5555, mmio_base + EP93XX_WATCHDOG);
/* Re-set the timer interval */
mod_timer(&timer, jiffies + WDT_INTERVAL);
}
static int ep93xx_wdt_start(struct watchdog_device *wdd) static int ep93xx_wdt_start(struct watchdog_device *wdd)
{ {
next_heartbeat = jiffies + (timeout * HZ); struct ep93xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
writel(0xaaaa, mmio_base + EP93XX_WATCHDOG); writel(0xaaaa, priv->mmio + EP93XX_WATCHDOG);
mod_timer(&timer, jiffies + WDT_INTERVAL);
return 0; return 0;
} }
static int ep93xx_wdt_stop(struct watchdog_device *wdd) static int ep93xx_wdt_stop(struct watchdog_device *wdd)
{ {
del_timer_sync(&timer); struct ep93xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
writel(0xaa55, mmio_base + EP93XX_WATCHDOG);
writel(0xaa55, priv->mmio + EP93XX_WATCHDOG);
return 0; return 0;
} }
static int ep93xx_wdt_keepalive(struct watchdog_device *wdd) static int ep93xx_wdt_ping(struct watchdog_device *wdd)
{ {
/* user land ping */ struct ep93xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
next_heartbeat = jiffies + (timeout * HZ);
writel(0x5555, priv->mmio + EP93XX_WATCHDOG);
return 0; return 0;
} }
static const struct watchdog_info ep93xx_wdt_ident = { static const struct watchdog_info ep93xx_wdt_ident = {
.options = WDIOF_CARDRESET | .options = WDIOF_CARDRESET |
WDIOF_SETTIMEOUT |
WDIOF_MAGICCLOSE | WDIOF_MAGICCLOSE |
WDIOF_KEEPALIVEPING, WDIOF_KEEPALIVEPING,
.identity = "EP93xx Watchdog", .identity = "EP93xx Watchdog",
}; };
static struct watchdog_ops ep93xx_wdt_ops = { static const struct watchdog_ops ep93xx_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = ep93xx_wdt_start, .start = ep93xx_wdt_start,
.stop = ep93xx_wdt_stop, .stop = ep93xx_wdt_stop,
.ping = ep93xx_wdt_keepalive, .ping = ep93xx_wdt_ping,
};
static struct watchdog_device ep93xx_wdt_wdd = {
.info = &ep93xx_wdt_ident,
.ops = &ep93xx_wdt_ops,
}; };
static int ep93xx_wdt_probe(struct platform_device *pdev) static int ep93xx_wdt_probe(struct platform_device *pdev)
{ {
struct ep93xx_wdt_priv *priv;
struct watchdog_device *wdd;
struct resource *res; struct resource *res;
unsigned long val; unsigned long val;
int err; int ret;
priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
if (!priv)
return -ENOMEM;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
mmio_base = devm_ioremap_resource(&pdev->dev, res); priv->mmio = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(mmio_base)) if (IS_ERR(priv->mmio))
return PTR_ERR(mmio_base); return PTR_ERR(priv->mmio);
if (timeout < 1 || timeout > 3600) { val = readl(priv->mmio + EP93XX_WATCHDOG);
timeout = WDT_TIMEOUT;
dev_warn(&pdev->dev,
"timeout value must be 1<=x<=3600, using %d\n",
timeout);
}
val = readl(mmio_base + EP93XX_WATCHDOG); wdd = &priv->wdd;
ep93xx_wdt_wdd.bootstatus = (val & 0x01) ? WDIOF_CARDRESET : 0; wdd->bootstatus = (val & 0x01) ? WDIOF_CARDRESET : 0;
ep93xx_wdt_wdd.timeout = timeout; wdd->info = &ep93xx_wdt_ident;
ep93xx_wdt_wdd.parent = &pdev->dev; wdd->ops = &ep93xx_wdt_ops;
wdd->min_timeout = 1;
wdd->max_hw_heartbeat_ms = 200;
wdd->parent = &pdev->dev;
watchdog_set_nowayout(&ep93xx_wdt_wdd, nowayout); watchdog_set_nowayout(wdd, nowayout);
setup_timer(&timer, ep93xx_wdt_timer_ping, 1); wdd->timeout = WDT_TIMEOUT;
watchdog_init_timeout(wdd, timeout, &pdev->dev);
err = watchdog_register_device(&ep93xx_wdt_wdd); watchdog_set_drvdata(wdd, priv);
if (err)
return err;
dev_info(&pdev->dev, ret = devm_watchdog_register_device(&pdev->dev, wdd);
"EP93XX watchdog, driver version " WDT_VERSION "%s\n", if (ret)
(val & 0x08) ? " (nCS1 disable detected)" : ""); return ret;
return 0; dev_info(&pdev->dev, "EP93XX watchdog driver %s\n",
} (val & 0x08) ? " (nCS1 disable detected)" : "");
static int ep93xx_wdt_remove(struct platform_device *pdev)
{
watchdog_unregister_device(&ep93xx_wdt_wdd);
return 0; return 0;
} }
...@@ -160,7 +136,6 @@ static struct platform_driver ep93xx_wdt_driver = { ...@@ -160,7 +136,6 @@ static struct platform_driver ep93xx_wdt_driver = {
.name = "ep93xx-wdt", .name = "ep93xx-wdt",
}, },
.probe = ep93xx_wdt_probe, .probe = ep93xx_wdt_probe,
.remove = ep93xx_wdt_remove,
}; };
module_platform_driver(ep93xx_wdt_driver); module_platform_driver(ep93xx_wdt_driver);
...@@ -170,4 +145,3 @@ MODULE_AUTHOR("Alessandro Zummo <a.zummo@towertech.it>"); ...@@ -170,4 +145,3 @@ MODULE_AUTHOR("Alessandro Zummo <a.zummo@towertech.it>");
MODULE_AUTHOR("H Hartley Sweeten <hsweeten@visionengravers.com>"); MODULE_AUTHOR("H Hartley Sweeten <hsweeten@visionengravers.com>");
MODULE_DESCRIPTION("EP93xx Watchdog"); MODULE_DESCRIPTION("EP93xx Watchdog");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_VERSION(WDT_VERSION);
/*
* Watchdog driver for Cortina Systems Gemini SoC
*
* Copyright (C) 2017 Linus Walleij <linus.walleij@linaro.org>
*
* Inspired by the out-of-tree drivers from OpenWRT:
* Copyright (C) 2009 Paulius Zaleckas <paulius.zaleckas@teltonika.lt>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#include <linux/bitops.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
#include <linux/watchdog.h>
#define GEMINI_WDCOUNTER 0x0
#define GEMINI_WDLOAD 0x4
#define GEMINI_WDRESTART 0x8
#define GEMINI_WDCR 0xC
#define WDRESTART_MAGIC 0x5AB9
#define WDCR_CLOCK_5MHZ BIT(4)
#define WDCR_SYS_RST BIT(1)
#define WDCR_ENABLE BIT(0)
#define WDT_CLOCK 5000000 /* 5 MHz */
struct gemini_wdt {
struct watchdog_device wdd;
struct device *dev;
void __iomem *base;
};
static inline
struct gemini_wdt *to_gemini_wdt(struct watchdog_device *wdd)
{
return container_of(wdd, struct gemini_wdt, wdd);
}
static int gemini_wdt_start(struct watchdog_device *wdd)
{
struct gemini_wdt *gwdt = to_gemini_wdt(wdd);
writel(wdd->timeout * WDT_CLOCK, gwdt->base + GEMINI_WDLOAD);
writel(WDRESTART_MAGIC, gwdt->base + GEMINI_WDRESTART);
/* set clock before enabling */
writel(WDCR_CLOCK_5MHZ | WDCR_SYS_RST,
gwdt->base + GEMINI_WDCR);
writel(WDCR_CLOCK_5MHZ | WDCR_SYS_RST | WDCR_ENABLE,
gwdt->base + GEMINI_WDCR);
return 0;
}
static int gemini_wdt_stop(struct watchdog_device *wdd)
{
struct gemini_wdt *gwdt = to_gemini_wdt(wdd);
writel(0, gwdt->base + GEMINI_WDCR);
return 0;
}
static int gemini_wdt_ping(struct watchdog_device *wdd)
{
struct gemini_wdt *gwdt = to_gemini_wdt(wdd);
writel(WDRESTART_MAGIC, gwdt->base + GEMINI_WDRESTART);
return 0;
}
static int gemini_wdt_set_timeout(struct watchdog_device *wdd,
unsigned int timeout)
{
wdd->timeout = timeout;
if (watchdog_active(wdd))
gemini_wdt_start(wdd);
return 0;
}
static irqreturn_t gemini_wdt_interrupt(int irq, void *data)
{
struct gemini_wdt *gwdt = data;
watchdog_notify_pretimeout(&gwdt->wdd);
return IRQ_HANDLED;
}
static const struct watchdog_ops gemini_wdt_ops = {
.start = gemini_wdt_start,
.stop = gemini_wdt_stop,
.ping = gemini_wdt_ping,
.set_timeout = gemini_wdt_set_timeout,
.owner = THIS_MODULE,
};
static const struct watchdog_info gemini_wdt_info = {
.options = WDIOF_KEEPALIVEPING
| WDIOF_MAGICCLOSE
| WDIOF_SETTIMEOUT,
.identity = KBUILD_MODNAME,
};
static int gemini_wdt_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct resource *res;
struct gemini_wdt *gwdt;
unsigned int reg;
int irq;
int ret;
gwdt = devm_kzalloc(dev, sizeof(*gwdt), GFP_KERNEL);
if (!gwdt)
return -ENOMEM;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
gwdt->base = devm_ioremap_resource(dev, res);
if (IS_ERR(gwdt->base))
return PTR_ERR(gwdt->base);
irq = platform_get_irq(pdev, 0);
if (!irq)
return -EINVAL;
gwdt->dev = dev;
gwdt->wdd.info = &gemini_wdt_info;
gwdt->wdd.ops = &gemini_wdt_ops;
gwdt->wdd.min_timeout = 1;
gwdt->wdd.max_timeout = 0xFFFFFFFF / WDT_CLOCK;
gwdt->wdd.parent = dev;
/*
* If 'timeout-sec' unspecified in devicetree, assume a 13 second
* default.
*/
gwdt->wdd.timeout = 13U;
watchdog_init_timeout(&gwdt->wdd, 0, dev);
reg = readw(gwdt->base + GEMINI_WDCR);
if (reg & WDCR_ENABLE) {
/* Watchdog was enabled by the bootloader, disable it. */
reg &= ~WDCR_ENABLE;
writel(reg, gwdt->base + GEMINI_WDCR);
}
ret = devm_request_irq(dev, irq, gemini_wdt_interrupt, 0,
"watchdog bark", gwdt);
if (ret)
return ret;
ret = devm_watchdog_register_device(dev, &gwdt->wdd);
if (ret) {
dev_err(&pdev->dev, "failed to register watchdog\n");
return ret;
}
/* Set up platform driver data */
platform_set_drvdata(pdev, gwdt);
dev_info(dev, "Gemini watchdog driver enabled\n");
return 0;
}
static int __maybe_unused gemini_wdt_suspend(struct device *dev)
{
struct gemini_wdt *gwdt = dev_get_drvdata(dev);
unsigned int reg;
reg = readw(gwdt->base + GEMINI_WDCR);
reg &= ~WDCR_ENABLE;
writel(reg, gwdt->base + GEMINI_WDCR);
return 0;
}
static int __maybe_unused gemini_wdt_resume(struct device *dev)
{
struct gemini_wdt *gwdt = dev_get_drvdata(dev);
unsigned int reg;
if (watchdog_active(&gwdt->wdd)) {
reg = readw(gwdt->base + GEMINI_WDCR);
reg |= WDCR_ENABLE;
writel(reg, gwdt->base + GEMINI_WDCR);
}
return 0;
}
static const struct dev_pm_ops gemini_wdt_dev_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS(gemini_wdt_suspend,
gemini_wdt_resume)
};
#ifdef CONFIG_OF
static const struct of_device_id gemini_wdt_match[] = {
{ .compatible = "cortina,gemini-watchdog" },
{},
};
MODULE_DEVICE_TABLE(of, gemini_wdt_match);
#endif
static struct platform_driver gemini_wdt_driver = {
.probe = gemini_wdt_probe,
.driver = {
.name = "gemini-wdt",
.of_match_table = of_match_ptr(gemini_wdt_match),
.pm = &gemini_wdt_dev_pm_ops,
},
};
module_platform_driver(gemini_wdt_driver);
MODULE_AUTHOR("Linus Walleij");
MODULE_DESCRIPTION("Watchdog driver for Gemini");
MODULE_LICENSE("GPL");
...@@ -72,22 +72,24 @@ ...@@ -72,22 +72,24 @@
/* Address definitions for the TCO */ /* Address definitions for the TCO */
/* TCO base address */ /* TCO base address */
#define TCOBASE (iTCO_wdt_private.tco_res->start) #define TCOBASE(p) ((p)->tco_res->start)
/* SMI Control and Enable Register */ /* SMI Control and Enable Register */
#define SMI_EN (iTCO_wdt_private.smi_res->start) #define SMI_EN(p) ((p)->smi_res->start)
#define TCO_RLD (TCOBASE + 0x00) /* TCO Timer Reload and Curr. Value */ #define TCO_RLD(p) (TCOBASE(p) + 0x00) /* TCO Timer Reload/Curr. Value */
#define TCOv1_TMR (TCOBASE + 0x01) /* TCOv1 Timer Initial Value */ #define TCOv1_TMR(p) (TCOBASE(p) + 0x01) /* TCOv1 Timer Initial Value*/
#define TCO_DAT_IN (TCOBASE + 0x02) /* TCO Data In Register */ #define TCO_DAT_IN(p) (TCOBASE(p) + 0x02) /* TCO Data In Register */
#define TCO_DAT_OUT (TCOBASE + 0x03) /* TCO Data Out Register */ #define TCO_DAT_OUT(p) (TCOBASE(p) + 0x03) /* TCO Data Out Register */
#define TCO1_STS (TCOBASE + 0x04) /* TCO1 Status Register */ #define TCO1_STS(p) (TCOBASE(p) + 0x04) /* TCO1 Status Register */
#define TCO2_STS (TCOBASE + 0x06) /* TCO2 Status Register */ #define TCO2_STS(p) (TCOBASE(p) + 0x06) /* TCO2 Status Register */
#define TCO1_CNT (TCOBASE + 0x08) /* TCO1 Control Register */ #define TCO1_CNT(p) (TCOBASE(p) + 0x08) /* TCO1 Control Register */
#define TCO2_CNT (TCOBASE + 0x0a) /* TCO2 Control Register */ #define TCO2_CNT(p) (TCOBASE(p) + 0x0a) /* TCO2 Control Register */
#define TCOv2_TMR (TCOBASE + 0x12) /* TCOv2 Timer Initial Value */ #define TCOv2_TMR(p) (TCOBASE(p) + 0x12) /* TCOv2 Timer Initial Value*/
/* internal variables */ /* internal variables */
static struct { /* this is private data for the iTCO_wdt device */ struct iTCO_wdt_private {
struct watchdog_device wddev;
/* TCO version/generation */ /* TCO version/generation */
unsigned int iTCO_version; unsigned int iTCO_version;
struct resource *tco_res; struct resource *tco_res;
...@@ -100,12 +102,11 @@ static struct { /* this is private data for the iTCO_wdt device */ ...@@ -100,12 +102,11 @@ static struct { /* this is private data for the iTCO_wdt device */
unsigned long __iomem *gcs_pmc; unsigned long __iomem *gcs_pmc;
/* the lock for io operations */ /* the lock for io operations */
spinlock_t io_lock; spinlock_t io_lock;
struct platform_device *dev;
/* the PCI-device */ /* the PCI-device */
struct pci_dev *pdev; struct pci_dev *pci_dev;
/* whether or not the watchdog has been suspended */ /* whether or not the watchdog has been suspended */
bool suspended; bool suspended;
} iTCO_wdt_private; };
/* module parameters */ /* module parameters */
#define WATCHDOG_TIMEOUT 30 /* 30 sec default heartbeat */ #define WATCHDOG_TIMEOUT 30 /* 30 sec default heartbeat */
...@@ -135,21 +136,23 @@ MODULE_PARM_DESC(turn_SMI_watchdog_clear_off, ...@@ -135,21 +136,23 @@ MODULE_PARM_DESC(turn_SMI_watchdog_clear_off,
* every 0.6 seconds. v3's internal timer is stored as seconds (some * every 0.6 seconds. v3's internal timer is stored as seconds (some
* datasheets incorrectly state 0.6 seconds). * datasheets incorrectly state 0.6 seconds).
*/ */
static inline unsigned int seconds_to_ticks(int secs) static inline unsigned int seconds_to_ticks(struct iTCO_wdt_private *p,
int secs)
{ {
return iTCO_wdt_private.iTCO_version == 3 ? secs : (secs * 10) / 6; return p->iTCO_version == 3 ? secs : (secs * 10) / 6;
} }
static inline unsigned int ticks_to_seconds(int ticks) static inline unsigned int ticks_to_seconds(struct iTCO_wdt_private *p,
int ticks)
{ {
return iTCO_wdt_private.iTCO_version == 3 ? ticks : (ticks * 6) / 10; return p->iTCO_version == 3 ? ticks : (ticks * 6) / 10;
} }
static inline u32 no_reboot_bit(void) static inline u32 no_reboot_bit(struct iTCO_wdt_private *p)
{ {
u32 enable_bit; u32 enable_bit;
switch (iTCO_wdt_private.iTCO_version) { switch (p->iTCO_version) {
case 5: case 5:
case 3: case 3:
enable_bit = 0x00000010; enable_bit = 0x00000010;
...@@ -167,40 +170,40 @@ static inline u32 no_reboot_bit(void) ...@@ -167,40 +170,40 @@ static inline u32 no_reboot_bit(void)
return enable_bit; return enable_bit;
} }
static void iTCO_wdt_set_NO_REBOOT_bit(void) static void iTCO_wdt_set_NO_REBOOT_bit(struct iTCO_wdt_private *p)
{ {
u32 val32; u32 val32;
/* Set the NO_REBOOT bit: this disables reboots */ /* Set the NO_REBOOT bit: this disables reboots */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
val32 = readl(iTCO_wdt_private.gcs_pmc); val32 = readl(p->gcs_pmc);
val32 |= no_reboot_bit(); val32 |= no_reboot_bit(p);
writel(val32, iTCO_wdt_private.gcs_pmc); writel(val32, p->gcs_pmc);
} else if (iTCO_wdt_private.iTCO_version == 1) { } else if (p->iTCO_version == 1) {
pci_read_config_dword(iTCO_wdt_private.pdev, 0xd4, &val32); pci_read_config_dword(p->pci_dev, 0xd4, &val32);
val32 |= no_reboot_bit(); val32 |= no_reboot_bit(p);
pci_write_config_dword(iTCO_wdt_private.pdev, 0xd4, val32); pci_write_config_dword(p->pci_dev, 0xd4, val32);
} }
} }
static int iTCO_wdt_unset_NO_REBOOT_bit(void) static int iTCO_wdt_unset_NO_REBOOT_bit(struct iTCO_wdt_private *p)
{ {
u32 enable_bit = no_reboot_bit(); u32 enable_bit = no_reboot_bit(p);
u32 val32 = 0; u32 val32 = 0;
/* Unset the NO_REBOOT bit: this enables reboots */ /* Unset the NO_REBOOT bit: this enables reboots */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
val32 = readl(iTCO_wdt_private.gcs_pmc); val32 = readl(p->gcs_pmc);
val32 &= ~enable_bit; val32 &= ~enable_bit;
writel(val32, iTCO_wdt_private.gcs_pmc); writel(val32, p->gcs_pmc);
val32 = readl(iTCO_wdt_private.gcs_pmc); val32 = readl(p->gcs_pmc);
} else if (iTCO_wdt_private.iTCO_version == 1) { } else if (p->iTCO_version == 1) {
pci_read_config_dword(iTCO_wdt_private.pdev, 0xd4, &val32); pci_read_config_dword(p->pci_dev, 0xd4, &val32);
val32 &= ~enable_bit; val32 &= ~enable_bit;
pci_write_config_dword(iTCO_wdt_private.pdev, 0xd4, val32); pci_write_config_dword(p->pci_dev, 0xd4, val32);
pci_read_config_dword(iTCO_wdt_private.pdev, 0xd4, &val32); pci_read_config_dword(p->pci_dev, 0xd4, &val32);
} }
if (val32 & enable_bit) if (val32 & enable_bit)
...@@ -211,32 +214,33 @@ static int iTCO_wdt_unset_NO_REBOOT_bit(void) ...@@ -211,32 +214,33 @@ static int iTCO_wdt_unset_NO_REBOOT_bit(void)
static int iTCO_wdt_start(struct watchdog_device *wd_dev) static int iTCO_wdt_start(struct watchdog_device *wd_dev)
{ {
struct iTCO_wdt_private *p = watchdog_get_drvdata(wd_dev);
unsigned int val; unsigned int val;
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
iTCO_vendor_pre_start(iTCO_wdt_private.smi_res, wd_dev->timeout); iTCO_vendor_pre_start(p->smi_res, wd_dev->timeout);
/* disable chipset's NO_REBOOT bit */ /* disable chipset's NO_REBOOT bit */
if (iTCO_wdt_unset_NO_REBOOT_bit()) { if (iTCO_wdt_unset_NO_REBOOT_bit(p)) {
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
pr_err("failed to reset NO_REBOOT flag, reboot disabled by hardware/BIOS\n"); pr_err("failed to reset NO_REBOOT flag, reboot disabled by hardware/BIOS\n");
return -EIO; return -EIO;
} }
/* Force the timer to its reload value by writing to the TCO_RLD /* Force the timer to its reload value by writing to the TCO_RLD
register */ register */
if (iTCO_wdt_private.iTCO_version >= 2) if (p->iTCO_version >= 2)
outw(0x01, TCO_RLD); outw(0x01, TCO_RLD(p));
else if (iTCO_wdt_private.iTCO_version == 1) else if (p->iTCO_version == 1)
outb(0x01, TCO_RLD); outb(0x01, TCO_RLD(p));
/* Bit 11: TCO Timer Halt -> 0 = The TCO timer is enabled to count */ /* Bit 11: TCO Timer Halt -> 0 = The TCO timer is enabled to count */
val = inw(TCO1_CNT); val = inw(TCO1_CNT(p));
val &= 0xf7ff; val &= 0xf7ff;
outw(val, TCO1_CNT); outw(val, TCO1_CNT(p));
val = inw(TCO1_CNT); val = inw(TCO1_CNT(p));
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
if (val & 0x0800) if (val & 0x0800)
return -1; return -1;
...@@ -245,22 +249,23 @@ static int iTCO_wdt_start(struct watchdog_device *wd_dev) ...@@ -245,22 +249,23 @@ static int iTCO_wdt_start(struct watchdog_device *wd_dev)
static int iTCO_wdt_stop(struct watchdog_device *wd_dev) static int iTCO_wdt_stop(struct watchdog_device *wd_dev)
{ {
struct iTCO_wdt_private *p = watchdog_get_drvdata(wd_dev);
unsigned int val; unsigned int val;
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
iTCO_vendor_pre_stop(iTCO_wdt_private.smi_res); iTCO_vendor_pre_stop(p->smi_res);
/* Bit 11: TCO Timer Halt -> 1 = The TCO timer is disabled */ /* Bit 11: TCO Timer Halt -> 1 = The TCO timer is disabled */
val = inw(TCO1_CNT); val = inw(TCO1_CNT(p));
val |= 0x0800; val |= 0x0800;
outw(val, TCO1_CNT); outw(val, TCO1_CNT(p));
val = inw(TCO1_CNT); val = inw(TCO1_CNT(p));
/* Set the NO_REBOOT bit to prevent later reboots, just for sure */ /* Set the NO_REBOOT bit to prevent later reboots, just for sure */
iTCO_wdt_set_NO_REBOOT_bit(); iTCO_wdt_set_NO_REBOOT_bit(p);
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
if ((val & 0x0800) == 0) if ((val & 0x0800) == 0)
return -1; return -1;
...@@ -269,67 +274,70 @@ static int iTCO_wdt_stop(struct watchdog_device *wd_dev) ...@@ -269,67 +274,70 @@ static int iTCO_wdt_stop(struct watchdog_device *wd_dev)
static int iTCO_wdt_ping(struct watchdog_device *wd_dev) static int iTCO_wdt_ping(struct watchdog_device *wd_dev)
{ {
spin_lock(&iTCO_wdt_private.io_lock); struct iTCO_wdt_private *p = watchdog_get_drvdata(wd_dev);
iTCO_vendor_pre_keepalive(iTCO_wdt_private.smi_res, wd_dev->timeout); spin_lock(&p->io_lock);
iTCO_vendor_pre_keepalive(p->smi_res, wd_dev->timeout);
/* Reload the timer by writing to the TCO Timer Counter register */ /* Reload the timer by writing to the TCO Timer Counter register */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
outw(0x01, TCO_RLD); outw(0x01, TCO_RLD(p));
} else if (iTCO_wdt_private.iTCO_version == 1) { } else if (p->iTCO_version == 1) {
/* Reset the timeout status bit so that the timer /* Reset the timeout status bit so that the timer
* needs to count down twice again before rebooting */ * needs to count down twice again before rebooting */
outw(0x0008, TCO1_STS); /* write 1 to clear bit */ outw(0x0008, TCO1_STS(p)); /* write 1 to clear bit */
outb(0x01, TCO_RLD); outb(0x01, TCO_RLD(p));
} }
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
return 0; return 0;
} }
static int iTCO_wdt_set_timeout(struct watchdog_device *wd_dev, unsigned int t) static int iTCO_wdt_set_timeout(struct watchdog_device *wd_dev, unsigned int t)
{ {
struct iTCO_wdt_private *p = watchdog_get_drvdata(wd_dev);
unsigned int val16; unsigned int val16;
unsigned char val8; unsigned char val8;
unsigned int tmrval; unsigned int tmrval;
tmrval = seconds_to_ticks(t); tmrval = seconds_to_ticks(p, t);
/* For TCO v1 the timer counts down twice before rebooting */ /* For TCO v1 the timer counts down twice before rebooting */
if (iTCO_wdt_private.iTCO_version == 1) if (p->iTCO_version == 1)
tmrval /= 2; tmrval /= 2;
/* from the specs: */ /* from the specs: */
/* "Values of 0h-3h are ignored and should not be attempted" */ /* "Values of 0h-3h are ignored and should not be attempted" */
if (tmrval < 0x04) if (tmrval < 0x04)
return -EINVAL; return -EINVAL;
if (((iTCO_wdt_private.iTCO_version >= 2) && (tmrval > 0x3ff)) || if ((p->iTCO_version >= 2 && tmrval > 0x3ff) ||
((iTCO_wdt_private.iTCO_version == 1) && (tmrval > 0x03f))) (p->iTCO_version == 1 && tmrval > 0x03f))
return -EINVAL; return -EINVAL;
iTCO_vendor_pre_set_heartbeat(tmrval); iTCO_vendor_pre_set_heartbeat(tmrval);
/* Write new heartbeat to watchdog */ /* Write new heartbeat to watchdog */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
val16 = inw(TCOv2_TMR); val16 = inw(TCOv2_TMR(p));
val16 &= 0xfc00; val16 &= 0xfc00;
val16 |= tmrval; val16 |= tmrval;
outw(val16, TCOv2_TMR); outw(val16, TCOv2_TMR(p));
val16 = inw(TCOv2_TMR); val16 = inw(TCOv2_TMR(p));
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
if ((val16 & 0x3ff) != tmrval) if ((val16 & 0x3ff) != tmrval)
return -EINVAL; return -EINVAL;
} else if (iTCO_wdt_private.iTCO_version == 1) { } else if (p->iTCO_version == 1) {
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
val8 = inb(TCOv1_TMR); val8 = inb(TCOv1_TMR(p));
val8 &= 0xc0; val8 &= 0xc0;
val8 |= (tmrval & 0xff); val8 |= (tmrval & 0xff);
outb(val8, TCOv1_TMR); outb(val8, TCOv1_TMR(p));
val8 = inb(TCOv1_TMR); val8 = inb(TCOv1_TMR(p));
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
if ((val8 & 0x3f) != tmrval) if ((val8 & 0x3f) != tmrval)
return -EINVAL; return -EINVAL;
...@@ -341,27 +349,28 @@ static int iTCO_wdt_set_timeout(struct watchdog_device *wd_dev, unsigned int t) ...@@ -341,27 +349,28 @@ static int iTCO_wdt_set_timeout(struct watchdog_device *wd_dev, unsigned int t)
static unsigned int iTCO_wdt_get_timeleft(struct watchdog_device *wd_dev) static unsigned int iTCO_wdt_get_timeleft(struct watchdog_device *wd_dev)
{ {
struct iTCO_wdt_private *p = watchdog_get_drvdata(wd_dev);
unsigned int val16; unsigned int val16;
unsigned char val8; unsigned char val8;
unsigned int time_left = 0; unsigned int time_left = 0;
/* read the TCO Timer */ /* read the TCO Timer */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
val16 = inw(TCO_RLD); val16 = inw(TCO_RLD(p));
val16 &= 0x3ff; val16 &= 0x3ff;
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
time_left = ticks_to_seconds(val16); time_left = ticks_to_seconds(p, val16);
} else if (iTCO_wdt_private.iTCO_version == 1) { } else if (p->iTCO_version == 1) {
spin_lock(&iTCO_wdt_private.io_lock); spin_lock(&p->io_lock);
val8 = inb(TCO_RLD); val8 = inb(TCO_RLD(p));
val8 &= 0x3f; val8 &= 0x3f;
if (!(inw(TCO1_STS) & 0x0008)) if (!(inw(TCO1_STS(p)) & 0x0008))
val8 += (inb(TCOv1_TMR) & 0x3f); val8 += (inb(TCOv1_TMR(p)) & 0x3f);
spin_unlock(&iTCO_wdt_private.io_lock); spin_unlock(&p->io_lock);
time_left = ticks_to_seconds(val8); time_left = ticks_to_seconds(p, val8);
} }
return time_left; return time_left;
} }
...@@ -387,209 +396,152 @@ static const struct watchdog_ops iTCO_wdt_ops = { ...@@ -387,209 +396,152 @@ static const struct watchdog_ops iTCO_wdt_ops = {
.get_timeleft = iTCO_wdt_get_timeleft, .get_timeleft = iTCO_wdt_get_timeleft,
}; };
static struct watchdog_device iTCO_wdt_watchdog_dev = {
.info = &ident,
.ops = &iTCO_wdt_ops,
};
/* /*
* Init & exit routines * Init & exit routines
*/ */
static void iTCO_wdt_cleanup(void) static int iTCO_wdt_probe(struct platform_device *pdev)
{
/* Stop the timer before we leave */
if (!nowayout)
iTCO_wdt_stop(&iTCO_wdt_watchdog_dev);
/* Deregister */
watchdog_unregister_device(&iTCO_wdt_watchdog_dev);
/* release resources */
release_region(iTCO_wdt_private.tco_res->start,
resource_size(iTCO_wdt_private.tco_res));
release_region(iTCO_wdt_private.smi_res->start,
resource_size(iTCO_wdt_private.smi_res));
if (iTCO_wdt_private.iTCO_version >= 2) {
iounmap(iTCO_wdt_private.gcs_pmc);
release_mem_region(iTCO_wdt_private.gcs_pmc_res->start,
resource_size(iTCO_wdt_private.gcs_pmc_res));
}
iTCO_wdt_private.tco_res = NULL;
iTCO_wdt_private.smi_res = NULL;
iTCO_wdt_private.gcs_pmc_res = NULL;
iTCO_wdt_private.gcs_pmc = NULL;
}
static int iTCO_wdt_probe(struct platform_device *dev)
{ {
int ret = -ENODEV; struct device *dev = &pdev->dev;
struct itco_wdt_platform_data *pdata = dev_get_platdata(dev);
struct iTCO_wdt_private *p;
unsigned long val32; unsigned long val32;
struct itco_wdt_platform_data *pdata = dev_get_platdata(&dev->dev); int ret;
if (!pdata) if (!pdata)
goto out; return -ENODEV;
p = devm_kzalloc(dev, sizeof(*p), GFP_KERNEL);
if (!p)
return -ENOMEM;
spin_lock_init(&iTCO_wdt_private.io_lock); spin_lock_init(&p->io_lock);
iTCO_wdt_private.tco_res = p->tco_res = platform_get_resource(pdev, IORESOURCE_IO, ICH_RES_IO_TCO);
platform_get_resource(dev, IORESOURCE_IO, ICH_RES_IO_TCO); if (!p->tco_res)
if (!iTCO_wdt_private.tco_res) return -ENODEV;
goto out;
iTCO_wdt_private.smi_res = p->smi_res = platform_get_resource(pdev, IORESOURCE_IO, ICH_RES_IO_SMI);
platform_get_resource(dev, IORESOURCE_IO, ICH_RES_IO_SMI); if (!p->smi_res)
if (!iTCO_wdt_private.smi_res) return -ENODEV;
goto out;
iTCO_wdt_private.iTCO_version = pdata->version; p->iTCO_version = pdata->version;
iTCO_wdt_private.dev = dev; p->pci_dev = to_pci_dev(dev->parent);
iTCO_wdt_private.pdev = to_pci_dev(dev->dev.parent);
/* /*
* Get the Memory-Mapped GCS or PMC register, we need it for the * Get the Memory-Mapped GCS or PMC register, we need it for the
* NO_REBOOT flag (TCO v2 and v3). * NO_REBOOT flag (TCO v2 and v3).
*/ */
if (iTCO_wdt_private.iTCO_version >= 2) { if (p->iTCO_version >= 2) {
iTCO_wdt_private.gcs_pmc_res = platform_get_resource(dev, p->gcs_pmc_res = platform_get_resource(pdev,
IORESOURCE_MEM, IORESOURCE_MEM,
ICH_RES_MEM_GCS_PMC); ICH_RES_MEM_GCS_PMC);
p->gcs_pmc = devm_ioremap_resource(dev, p->gcs_pmc_res);
if (!iTCO_wdt_private.gcs_pmc_res) if (IS_ERR(p->gcs_pmc))
goto out; return PTR_ERR(p->gcs_pmc);
if (!request_mem_region(iTCO_wdt_private.gcs_pmc_res->start,
resource_size(iTCO_wdt_private.gcs_pmc_res), dev->name)) {
ret = -EBUSY;
goto out;
}
iTCO_wdt_private.gcs_pmc = ioremap(iTCO_wdt_private.gcs_pmc_res->start,
resource_size(iTCO_wdt_private.gcs_pmc_res));
if (!iTCO_wdt_private.gcs_pmc) {
ret = -EIO;
goto unreg_gcs_pmc;
}
} }
/* Check chipset's NO_REBOOT bit */ /* Check chipset's NO_REBOOT bit */
if (iTCO_wdt_unset_NO_REBOOT_bit() && iTCO_vendor_check_noreboot_on()) { if (iTCO_wdt_unset_NO_REBOOT_bit(p) &&
iTCO_vendor_check_noreboot_on()) {
pr_info("unable to reset NO_REBOOT flag, device disabled by hardware/BIOS\n"); pr_info("unable to reset NO_REBOOT flag, device disabled by hardware/BIOS\n");
ret = -ENODEV; /* Cannot reset NO_REBOOT bit */ return -ENODEV; /* Cannot reset NO_REBOOT bit */
goto unmap_gcs_pmc;
} }
/* Set the NO_REBOOT bit to prevent later reboots, just for sure */ /* Set the NO_REBOOT bit to prevent later reboots, just for sure */
iTCO_wdt_set_NO_REBOOT_bit(); iTCO_wdt_set_NO_REBOOT_bit(p);
/* The TCO logic uses the TCO_EN bit in the SMI_EN register */ /* The TCO logic uses the TCO_EN bit in the SMI_EN register */
if (!request_region(iTCO_wdt_private.smi_res->start, if (!devm_request_region(dev, p->smi_res->start,
resource_size(iTCO_wdt_private.smi_res), dev->name)) { resource_size(p->smi_res),
pdev->name)) {
pr_err("I/O address 0x%04llx already in use, device disabled\n", pr_err("I/O address 0x%04llx already in use, device disabled\n",
(u64)SMI_EN); (u64)SMI_EN(p));
ret = -EBUSY; return -EBUSY;
goto unmap_gcs_pmc;
} }
if (turn_SMI_watchdog_clear_off >= iTCO_wdt_private.iTCO_version) { if (turn_SMI_watchdog_clear_off >= p->iTCO_version) {
/* /*
* Bit 13: TCO_EN -> 0 * Bit 13: TCO_EN -> 0
* Disables TCO logic generating an SMI# * Disables TCO logic generating an SMI#
*/ */
val32 = inl(SMI_EN); val32 = inl(SMI_EN(p));
val32 &= 0xffffdfff; /* Turn off SMI clearing watchdog */ val32 &= 0xffffdfff; /* Turn off SMI clearing watchdog */
outl(val32, SMI_EN); outl(val32, SMI_EN(p));
} }
if (!request_region(iTCO_wdt_private.tco_res->start, if (!devm_request_region(dev, p->tco_res->start,
resource_size(iTCO_wdt_private.tco_res), dev->name)) { resource_size(p->tco_res),
pdev->name)) {
pr_err("I/O address 0x%04llx already in use, device disabled\n", pr_err("I/O address 0x%04llx already in use, device disabled\n",
(u64)TCOBASE); (u64)TCOBASE(p));
ret = -EBUSY; return -EBUSY;
goto unreg_smi;
} }
pr_info("Found a %s TCO device (Version=%d, TCOBASE=0x%04llx)\n", pr_info("Found a %s TCO device (Version=%d, TCOBASE=0x%04llx)\n",
pdata->name, pdata->version, (u64)TCOBASE); pdata->name, pdata->version, (u64)TCOBASE(p));
/* Clear out the (probably old) status */ /* Clear out the (probably old) status */
switch (iTCO_wdt_private.iTCO_version) { switch (p->iTCO_version) {
case 5: case 5:
case 4: case 4:
outw(0x0008, TCO1_STS); /* Clear the Time Out Status bit */ outw(0x0008, TCO1_STS(p)); /* Clear the Time Out Status bit */
outw(0x0002, TCO2_STS); /* Clear SECOND_TO_STS bit */ outw(0x0002, TCO2_STS(p)); /* Clear SECOND_TO_STS bit */
break; break;
case 3: case 3:
outl(0x20008, TCO1_STS); outl(0x20008, TCO1_STS(p));
break; break;
case 2: case 2:
case 1: case 1:
default: default:
outw(0x0008, TCO1_STS); /* Clear the Time Out Status bit */ outw(0x0008, TCO1_STS(p)); /* Clear the Time Out Status bit */
outw(0x0002, TCO2_STS); /* Clear SECOND_TO_STS bit */ outw(0x0002, TCO2_STS(p)); /* Clear SECOND_TO_STS bit */
outw(0x0004, TCO2_STS); /* Clear BOOT_STS bit */ outw(0x0004, TCO2_STS(p)); /* Clear BOOT_STS bit */
break; break;
} }
iTCO_wdt_watchdog_dev.bootstatus = 0; p->wddev.info = &ident,
iTCO_wdt_watchdog_dev.timeout = WATCHDOG_TIMEOUT; p->wddev.ops = &iTCO_wdt_ops,
watchdog_set_nowayout(&iTCO_wdt_watchdog_dev, nowayout); p->wddev.bootstatus = 0;
iTCO_wdt_watchdog_dev.parent = &dev->dev; p->wddev.timeout = WATCHDOG_TIMEOUT;
watchdog_set_nowayout(&p->wddev, nowayout);
p->wddev.parent = dev;
watchdog_set_drvdata(&p->wddev, p);
platform_set_drvdata(pdev, p);
/* Make sure the watchdog is not running */ /* Make sure the watchdog is not running */
iTCO_wdt_stop(&iTCO_wdt_watchdog_dev); iTCO_wdt_stop(&p->wddev);
/* Check that the heartbeat value is within it's range; /* Check that the heartbeat value is within it's range;
if not reset to the default */ if not reset to the default */
if (iTCO_wdt_set_timeout(&iTCO_wdt_watchdog_dev, heartbeat)) { if (iTCO_wdt_set_timeout(&p->wddev, heartbeat)) {
iTCO_wdt_set_timeout(&iTCO_wdt_watchdog_dev, WATCHDOG_TIMEOUT); iTCO_wdt_set_timeout(&p->wddev, WATCHDOG_TIMEOUT);
pr_info("timeout value out of range, using %d\n", pr_info("timeout value out of range, using %d\n",
WATCHDOG_TIMEOUT); WATCHDOG_TIMEOUT);
} }
ret = watchdog_register_device(&iTCO_wdt_watchdog_dev); watchdog_stop_on_reboot(&p->wddev);
ret = devm_watchdog_register_device(dev, &p->wddev);
if (ret != 0) { if (ret != 0) {
pr_err("cannot register watchdog device (err=%d)\n", ret); pr_err("cannot register watchdog device (err=%d)\n", ret);
goto unreg_tco; return ret;
} }
pr_info("initialized. heartbeat=%d sec (nowayout=%d)\n", pr_info("initialized. heartbeat=%d sec (nowayout=%d)\n",
heartbeat, nowayout); heartbeat, nowayout);
return 0; return 0;
unreg_tco:
release_region(iTCO_wdt_private.tco_res->start,
resource_size(iTCO_wdt_private.tco_res));
unreg_smi:
release_region(iTCO_wdt_private.smi_res->start,
resource_size(iTCO_wdt_private.smi_res));
unmap_gcs_pmc:
if (iTCO_wdt_private.iTCO_version >= 2)
iounmap(iTCO_wdt_private.gcs_pmc);
unreg_gcs_pmc:
if (iTCO_wdt_private.iTCO_version >= 2)
release_mem_region(iTCO_wdt_private.gcs_pmc_res->start,
resource_size(iTCO_wdt_private.gcs_pmc_res));
out:
iTCO_wdt_private.tco_res = NULL;
iTCO_wdt_private.smi_res = NULL;
iTCO_wdt_private.gcs_pmc_res = NULL;
iTCO_wdt_private.gcs_pmc = NULL;
return ret;
} }
static int iTCO_wdt_remove(struct platform_device *dev) static int iTCO_wdt_remove(struct platform_device *pdev)
{ {
if (iTCO_wdt_private.tco_res || iTCO_wdt_private.smi_res) struct iTCO_wdt_private *p = platform_get_drvdata(pdev);
iTCO_wdt_cleanup();
return 0; /* Stop the timer before we leave */
} if (!nowayout)
iTCO_wdt_stop(&p->wddev);
static void iTCO_wdt_shutdown(struct platform_device *dev) return 0;
{
iTCO_wdt_stop(NULL);
} }
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
...@@ -610,21 +562,24 @@ static inline bool need_suspend(void) { return true; } ...@@ -610,21 +562,24 @@ static inline bool need_suspend(void) { return true; }
static int iTCO_wdt_suspend_noirq(struct device *dev) static int iTCO_wdt_suspend_noirq(struct device *dev)
{ {
struct iTCO_wdt_private *p = dev_get_drvdata(dev);
int ret = 0; int ret = 0;
iTCO_wdt_private.suspended = false; p->suspended = false;
if (watchdog_active(&iTCO_wdt_watchdog_dev) && need_suspend()) { if (watchdog_active(&p->wddev) && need_suspend()) {
ret = iTCO_wdt_stop(&iTCO_wdt_watchdog_dev); ret = iTCO_wdt_stop(&p->wddev);
if (!ret) if (!ret)
iTCO_wdt_private.suspended = true; p->suspended = true;
} }
return ret; return ret;
} }
static int iTCO_wdt_resume_noirq(struct device *dev) static int iTCO_wdt_resume_noirq(struct device *dev)
{ {
if (iTCO_wdt_private.suspended) struct iTCO_wdt_private *p = dev_get_drvdata(dev);
iTCO_wdt_start(&iTCO_wdt_watchdog_dev);
if (p->suspended)
iTCO_wdt_start(&p->wddev);
return 0; return 0;
} }
...@@ -642,7 +597,6 @@ static const struct dev_pm_ops iTCO_wdt_pm = { ...@@ -642,7 +597,6 @@ static const struct dev_pm_ops iTCO_wdt_pm = {
static struct platform_driver iTCO_wdt_driver = { static struct platform_driver iTCO_wdt_driver = {
.probe = iTCO_wdt_probe, .probe = iTCO_wdt_probe,
.remove = iTCO_wdt_remove, .remove = iTCO_wdt_remove,
.shutdown = iTCO_wdt_shutdown,
.driver = { .driver = {
.name = DRV_NAME, .name = DRV_NAME,
.pm = ITCO_WDT_PM_OPS, .pm = ITCO_WDT_PM_OPS,
...@@ -651,15 +605,9 @@ static struct platform_driver iTCO_wdt_driver = { ...@@ -651,15 +605,9 @@ static struct platform_driver iTCO_wdt_driver = {
static int __init iTCO_wdt_init_module(void) static int __init iTCO_wdt_init_module(void)
{ {
int err;
pr_info("Intel TCO WatchDog Timer Driver v%s\n", DRV_VERSION); pr_info("Intel TCO WatchDog Timer Driver v%s\n", DRV_VERSION);
err = platform_driver_register(&iTCO_wdt_driver); return platform_driver_register(&iTCO_wdt_driver);
if (err)
return err;
return 0;
} }
static void __exit iTCO_wdt_cleanup_module(void) static void __exit iTCO_wdt_cleanup_module(void)
......
...@@ -161,7 +161,7 @@ static int pdc_wdt_restart(struct watchdog_device *wdt_dev, ...@@ -161,7 +161,7 @@ static int pdc_wdt_restart(struct watchdog_device *wdt_dev,
return 0; return 0;
} }
static struct watchdog_info pdc_wdt_info = { static const struct watchdog_info pdc_wdt_info = {
.identity = "IMG PDC Watchdog", .identity = "IMG PDC Watchdog",
.options = WDIOF_SETTIMEOUT | .options = WDIOF_SETTIMEOUT |
WDIOF_KEEPALIVEPING | WDIOF_KEEPALIVEPING |
......
...@@ -137,7 +137,6 @@ static int mid_wdt_probe(struct platform_device *pdev) ...@@ -137,7 +137,6 @@ static int mid_wdt_probe(struct platform_device *pdev)
wdt_dev->parent = &pdev->dev; wdt_dev->parent = &pdev->dev;
watchdog_set_drvdata(wdt_dev, &pdev->dev); watchdog_set_drvdata(wdt_dev, &pdev->dev);
platform_set_drvdata(pdev, wdt_dev);
ret = devm_request_irq(&pdev->dev, pdata->irq, mid_wdt_irq, ret = devm_request_irq(&pdev->dev, pdata->irq, mid_wdt_irq,
IRQF_SHARED | IRQF_NO_SUSPEND, "watchdog", IRQF_SHARED | IRQF_NO_SUSPEND, "watchdog",
...@@ -151,7 +150,7 @@ static int mid_wdt_probe(struct platform_device *pdev) ...@@ -151,7 +150,7 @@ static int mid_wdt_probe(struct platform_device *pdev)
/* Make sure the watchdog is not running */ /* Make sure the watchdog is not running */
wdt_stop(wdt_dev); wdt_stop(wdt_dev);
ret = watchdog_register_device(wdt_dev); ret = devm_watchdog_register_device(&pdev->dev, wdt_dev);
if (ret) { if (ret) {
dev_err(&pdev->dev, "error registering watchdog device\n"); dev_err(&pdev->dev, "error registering watchdog device\n");
return ret; return ret;
...@@ -162,16 +161,8 @@ static int mid_wdt_probe(struct platform_device *pdev) ...@@ -162,16 +161,8 @@ static int mid_wdt_probe(struct platform_device *pdev)
return 0; return 0;
} }
static int mid_wdt_remove(struct platform_device *pdev)
{
struct watchdog_device *wd = platform_get_drvdata(pdev);
watchdog_unregister_device(wd);
return 0;
}
static struct platform_driver mid_wdt_driver = { static struct platform_driver mid_wdt_driver = {
.probe = mid_wdt_probe, .probe = mid_wdt_probe,
.remove = mid_wdt_remove,
.driver = { .driver = {
.name = "intel_mid_wdt", .name = "intel_mid_wdt",
}, },
......
...@@ -422,7 +422,7 @@ static int kempld_wdt_probe_stages(struct watchdog_device *wdd) ...@@ -422,7 +422,7 @@ static int kempld_wdt_probe_stages(struct watchdog_device *wdd)
return 0; return 0;
} }
static struct watchdog_info kempld_wdt_info = { static const struct watchdog_info kempld_wdt_info = {
.identity = "KEMPLD Watchdog", .identity = "KEMPLD Watchdog",
.options = WDIOF_SETTIMEOUT | .options = WDIOF_SETTIMEOUT |
WDIOF_KEEPALIVEPING | WDIOF_KEEPALIVEPING |
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. * by the Free Software Foundation.
* *
* Copyright (C) 2010 John Crispin <blogic@openwrt.org> * Copyright (C) 2010 John Crispin <john@phrozen.org>
* Based on EP93xx wdt driver * Based on EP93xx wdt driver
*/ */
...@@ -240,6 +240,6 @@ module_platform_driver(ltq_wdt_driver); ...@@ -240,6 +240,6 @@ module_platform_driver(ltq_wdt_driver);
module_param(nowayout, bool, 0); module_param(nowayout, bool, 0);
MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started"); MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started");
MODULE_AUTHOR("John Crispin <blogic@openwrt.org>"); MODULE_AUTHOR("John Crispin <john@phrozen.org>");
MODULE_DESCRIPTION("Lantiq SoC Watchdog"); MODULE_DESCRIPTION("Lantiq SoC Watchdog");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -181,7 +181,7 @@ static int lpc18xx_wdt_restart(struct watchdog_device *wdt_dev, ...@@ -181,7 +181,7 @@ static int lpc18xx_wdt_restart(struct watchdog_device *wdt_dev,
return 0; return 0;
} }
static struct watchdog_info lpc18xx_wdt_info = { static const struct watchdog_info lpc18xx_wdt_info = {
.identity = "NXP LPC18xx Watchdog", .identity = "NXP LPC18xx Watchdog",
.options = WDIOF_SETTIMEOUT | .options = WDIOF_SETTIMEOUT |
WDIOF_KEEPALIVEPING | WDIOF_KEEPALIVEPING |
......
...@@ -212,33 +212,14 @@ static int a21_wdt_probe(struct platform_device *pdev) ...@@ -212,33 +212,14 @@ static int a21_wdt_probe(struct platform_device *pdev)
drv->wdt = a21_wdt; drv->wdt = a21_wdt;
dev_set_drvdata(&pdev->dev, drv); dev_set_drvdata(&pdev->dev, drv);
ret = watchdog_register_device(&a21_wdt); ret = devm_watchdog_register_device(&pdev->dev, &a21_wdt);
if (ret) { if (ret) {
dev_err(&pdev->dev, "Cannot register watchdog device\n"); dev_err(&pdev->dev, "Cannot register watchdog device\n");
goto err_register_wd; return ret;
} }
dev_info(&pdev->dev, "MEN A21 watchdog timer driver enabled\n"); dev_info(&pdev->dev, "MEN A21 watchdog timer driver enabled\n");
return 0;
err_register_wd:
mutex_destroy(&drv->lock);
return ret;
}
static int a21_wdt_remove(struct platform_device *pdev)
{
struct a21_wdt_drv *drv = dev_get_drvdata(&pdev->dev);
dev_warn(&pdev->dev,
"Unregistering A21 watchdog driver, board may reboot\n");
watchdog_unregister_device(&drv->wdt);
mutex_destroy(&drv->lock);
return 0; return 0;
} }
...@@ -257,7 +238,6 @@ MODULE_DEVICE_TABLE(of, a21_wdt_ids); ...@@ -257,7 +238,6 @@ MODULE_DEVICE_TABLE(of, a21_wdt_ids);
static struct platform_driver a21_wdt_driver = { static struct platform_driver a21_wdt_driver = {
.probe = a21_wdt_probe, .probe = a21_wdt_probe,
.remove = a21_wdt_remove,
.shutdown = a21_wdt_shutdown, .shutdown = a21_wdt_shutdown,
.driver = { .driver = {
.name = "a21-watchdog", .name = "a21-watchdog",
......
...@@ -201,38 +201,19 @@ static int meson_wdt_probe(struct platform_device *pdev) ...@@ -201,38 +201,19 @@ static int meson_wdt_probe(struct platform_device *pdev)
meson_wdt_stop(&meson_wdt->wdt_dev); meson_wdt_stop(&meson_wdt->wdt_dev);
err = watchdog_register_device(&meson_wdt->wdt_dev); watchdog_stop_on_reboot(&meson_wdt->wdt_dev);
err = devm_watchdog_register_device(&pdev->dev, &meson_wdt->wdt_dev);
if (err) if (err)
return err; return err;
platform_set_drvdata(pdev, meson_wdt);
dev_info(&pdev->dev, "Watchdog enabled (timeout=%d sec, nowayout=%d)", dev_info(&pdev->dev, "Watchdog enabled (timeout=%d sec, nowayout=%d)",
meson_wdt->wdt_dev.timeout, nowayout); meson_wdt->wdt_dev.timeout, nowayout);
return 0; return 0;
} }
static int meson_wdt_remove(struct platform_device *pdev)
{
struct meson_wdt_dev *meson_wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&meson_wdt->wdt_dev);
return 0;
}
static void meson_wdt_shutdown(struct platform_device *pdev)
{
struct meson_wdt_dev *meson_wdt = platform_get_drvdata(pdev);
meson_wdt_stop(&meson_wdt->wdt_dev);
}
static struct platform_driver meson_wdt_driver = { static struct platform_driver meson_wdt_driver = {
.probe = meson_wdt_probe, .probe = meson_wdt_probe,
.remove = meson_wdt_remove,
.shutdown = meson_wdt_shutdown,
.driver = { .driver = {
.name = DRV_NAME, .name = DRV_NAME,
.of_match_table = meson_wdt_dt_ids, .of_match_table = meson_wdt_dt_ids,
......
/* /*
* Ralink MT7621/MT7628 built-in hardware watchdog timer * Ralink MT7621/MT7628 built-in hardware watchdog timer
* *
* Copyright (C) 2014 John Crispin <blogic@openwrt.org> * Copyright (C) 2014 John Crispin <john@phrozen.org>
* *
* This driver was based on: drivers/watchdog/rt2880_wdt.c * This driver was based on: drivers/watchdog/rt2880_wdt.c
* *
...@@ -110,7 +110,7 @@ static struct watchdog_info mt7621_wdt_info = { ...@@ -110,7 +110,7 @@ static struct watchdog_info mt7621_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE,
}; };
static struct watchdog_ops mt7621_wdt_ops = { static const struct watchdog_ops mt7621_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = mt7621_wdt_start, .start = mt7621_wdt_start,
.stop = mt7621_wdt_stop, .stop = mt7621_wdt_stop,
...@@ -181,5 +181,5 @@ static struct platform_driver mt7621_wdt_driver = { ...@@ -181,5 +181,5 @@ static struct platform_driver mt7621_wdt_driver = {
module_platform_driver(mt7621_wdt_driver); module_platform_driver(mt7621_wdt_driver);
MODULE_DESCRIPTION("MediaTek MT762x hardware watchdog driver"); MODULE_DESCRIPTION("MediaTek MT762x hardware watchdog driver");
MODULE_AUTHOR("John Crispin <blogic@openwrt.org"); MODULE_AUTHOR("John Crispin <john@phrozen.org");
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
/*
* Copyright (C) 2016 National Instruments Corp.
*
* 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.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*/
#include <linux/acpi.h>
#include <linux/bitops.h>
#include <linux/device.h>
#include <linux/io.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/watchdog.h>
#define LOCK 0xA5
#define UNLOCK 0x5A
#define WDT_CTRL_RESET_EN BIT(7)
#define WDT_RELOAD_PORT_EN BIT(7)
#define WDT_CTRL 1
#define WDT_RELOAD_CTRL 2
#define WDT_PRESET_PRESCALE 4
#define WDT_REG_LOCK 5
#define WDT_COUNT 6
#define WDT_RELOAD_PORT 7
#define WDT_MIN_TIMEOUT 1
#define WDT_MAX_TIMEOUT 464
#define WDT_DEFAULT_TIMEOUT 80
#define WDT_MAX_COUNTER 15
static unsigned int timeout;
module_param(timeout, uint, 0);
MODULE_PARM_DESC(timeout,
"Watchdog timeout in seconds. (default="
__MODULE_STRING(WDT_DEFAULT_TIMEOUT) ")");
static bool nowayout = WATCHDOG_NOWAYOUT;
module_param(nowayout, bool, 0);
MODULE_PARM_DESC(nowayout,
"Watchdog cannot be stopped once started. (default="
__MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
struct nic7018_wdt {
u16 io_base;
u32 period;
struct watchdog_device wdd;
};
struct nic7018_config {
u32 period;
u8 divider;
};
static const struct nic7018_config nic7018_configs[] = {
{ 2, 4 },
{ 32, 5 },
};
static inline u32 nic7018_timeout(u32 period, u8 counter)
{
return period * counter - period / 2;
}
static const struct nic7018_config *nic7018_get_config(u32 timeout,
u8 *counter)
{
const struct nic7018_config *config;
u8 count;
if (timeout < 30 && timeout != 16) {
config = &nic7018_configs[0];
count = timeout / 2 + 1;
} else {
config = &nic7018_configs[1];
count = DIV_ROUND_UP(timeout + 16, 32);
if (count > WDT_MAX_COUNTER)
count = WDT_MAX_COUNTER;
}
*counter = count;
return config;
}
static int nic7018_set_timeout(struct watchdog_device *wdd,
unsigned int timeout)
{
struct nic7018_wdt *wdt = watchdog_get_drvdata(wdd);
const struct nic7018_config *config;
u8 counter;
config = nic7018_get_config(timeout, &counter);
outb(counter << 4 | config->divider,
wdt->io_base + WDT_PRESET_PRESCALE);
wdd->timeout = nic7018_timeout(config->period, counter);
wdt->period = config->period;
return 0;
}
static int nic7018_start(struct watchdog_device *wdd)
{
struct nic7018_wdt *wdt = watchdog_get_drvdata(wdd);
u8 control;
nic7018_set_timeout(wdd, wdd->timeout);
control = inb(wdt->io_base + WDT_RELOAD_CTRL);
outb(control | WDT_RELOAD_PORT_EN, wdt->io_base + WDT_RELOAD_CTRL);
outb(1, wdt->io_base + WDT_RELOAD_PORT);
control = inb(wdt->io_base + WDT_CTRL);
outb(control | WDT_CTRL_RESET_EN, wdt->io_base + WDT_CTRL);
return 0;
}
static int nic7018_stop(struct watchdog_device *wdd)
{
struct nic7018_wdt *wdt = watchdog_get_drvdata(wdd);
outb(0, wdt->io_base + WDT_CTRL);
outb(0, wdt->io_base + WDT_RELOAD_CTRL);
outb(0xF0, wdt->io_base + WDT_PRESET_PRESCALE);
return 0;
}
static int nic7018_ping(struct watchdog_device *wdd)
{
struct nic7018_wdt *wdt = watchdog_get_drvdata(wdd);
outb(1, wdt->io_base + WDT_RELOAD_PORT);
return 0;
}
static unsigned int nic7018_get_timeleft(struct watchdog_device *wdd)
{
struct nic7018_wdt *wdt = watchdog_get_drvdata(wdd);
u8 count;
count = inb(wdt->io_base + WDT_COUNT) & 0xF;
if (!count)
return 0;
return nic7018_timeout(wdt->period, count);
}
static const struct watchdog_info nic7018_wdd_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE,
.identity = "NIC7018 Watchdog",
};
static const struct watchdog_ops nic7018_wdd_ops = {
.owner = THIS_MODULE,
.start = nic7018_start,
.stop = nic7018_stop,
.ping = nic7018_ping,
.set_timeout = nic7018_set_timeout,
.get_timeleft = nic7018_get_timeleft,
};
static int nic7018_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct watchdog_device *wdd;
struct nic7018_wdt *wdt;
struct resource *io_rc;
int ret;
wdt = devm_kzalloc(dev, sizeof(*wdt), GFP_KERNEL);
if (!wdt)
return -ENOMEM;
platform_set_drvdata(pdev, wdt);
io_rc = platform_get_resource(pdev, IORESOURCE_IO, 0);
if (!io_rc) {
dev_err(dev, "missing IO resources\n");
return -EINVAL;
}
if (!devm_request_region(dev, io_rc->start, resource_size(io_rc),
KBUILD_MODNAME)) {
dev_err(dev, "failed to get IO region\n");
return -EBUSY;
}
wdt->io_base = io_rc->start;
wdd = &wdt->wdd;
wdd->info = &nic7018_wdd_info;
wdd->ops = &nic7018_wdd_ops;
wdd->min_timeout = WDT_MIN_TIMEOUT;
wdd->max_timeout = WDT_MAX_TIMEOUT;
wdd->timeout = WDT_DEFAULT_TIMEOUT;
wdd->parent = dev;
watchdog_set_drvdata(wdd, wdt);
watchdog_set_nowayout(wdd, nowayout);
ret = watchdog_init_timeout(wdd, timeout, dev);
if (ret)
dev_warn(dev, "unable to set timeout value, using default\n");
/* Unlock WDT register */
outb(UNLOCK, wdt->io_base + WDT_REG_LOCK);
ret = watchdog_register_device(wdd);
if (ret) {
outb(LOCK, wdt->io_base + WDT_REG_LOCK);
dev_err(dev, "failed to register watchdog\n");
return ret;
}
dev_dbg(dev, "io_base=0x%04X, timeout=%d, nowayout=%d\n",
wdt->io_base, timeout, nowayout);
return 0;
}
static int nic7018_remove(struct platform_device *pdev)
{
struct nic7018_wdt *wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&wdt->wdd);
/* Lock WDT register */
outb(LOCK, wdt->io_base + WDT_REG_LOCK);
return 0;
}
static const struct acpi_device_id nic7018_device_ids[] = {
{"NIC7018", 0},
{"", 0},
};
MODULE_DEVICE_TABLE(acpi, nic7018_device_ids);
static struct platform_driver watchdog_driver = {
.probe = nic7018_probe,
.remove = nic7018_remove,
.driver = {
.name = KBUILD_MODNAME,
.acpi_match_table = ACPI_PTR(nic7018_device_ids),
},
};
module_platform_driver(watchdog_driver);
MODULE_DESCRIPTION("National Instruments NIC7018 Watchdog driver");
MODULE_AUTHOR("Hui Chun Ong <hui.chun.ong@ni.com>");
MODULE_LICENSE("GPL");
...@@ -395,7 +395,7 @@ static void __iomem *orion_wdt_ioremap_rstout(struct platform_device *pdev, ...@@ -395,7 +395,7 @@ static void __iomem *orion_wdt_ioremap_rstout(struct platform_device *pdev,
rstout = internal_regs + ORION_RSTOUT_MASK_OFFSET; rstout = internal_regs + ORION_RSTOUT_MASK_OFFSET;
WARN(1, FW_BUG "falling back to harcoded RSTOUT reg %pa\n", &rstout); WARN(1, FW_BUG "falling back to hardcoded RSTOUT reg %pa\n", &rstout);
return devm_ioremap(&pdev->dev, rstout, 0x4); return devm_ioremap(&pdev->dev, rstout, 0x4);
} }
......
...@@ -54,7 +54,7 @@ static struct { ...@@ -54,7 +54,7 @@ static struct {
struct timer_list timer; /* The timer that pings the watchdog */ struct timer_list timer; /* The timer that pings the watchdog */
} pikawdt_private; } pikawdt_private;
static struct watchdog_info ident = { static struct watchdog_info ident __ro_after_init = {
.identity = DRV_NAME, .identity = DRV_NAME,
.options = WDIOF_CARDRESET | .options = WDIOF_CARDRESET |
WDIOF_SETTIMEOUT | WDIOF_SETTIMEOUT |
......
...@@ -130,7 +130,7 @@ static int rn5t618_wdt_ping(struct watchdog_device *wdt_dev) ...@@ -130,7 +130,7 @@ static int rn5t618_wdt_ping(struct watchdog_device *wdt_dev)
RN5T618_PWRIRQ_IR_WDOG, 0); RN5T618_PWRIRQ_IR_WDOG, 0);
} }
static struct watchdog_info rn5t618_wdt_info = { static const struct watchdog_info rn5t618_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE | .options = WDIOF_SETTIMEOUT | WDIOF_MAGICCLOSE |
WDIOF_KEEPALIVEPING, WDIOF_KEEPALIVEPING,
.identity = DRIVER_NAME, .identity = DRIVER_NAME,
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* Ralink RT288x/RT3xxx/MT76xx built-in hardware watchdog timer * Ralink RT288x/RT3xxx/MT76xx built-in hardware watchdog timer
* *
* Copyright (C) 2011 Gabor Juhos <juhosg@openwrt.org> * Copyright (C) 2011 Gabor Juhos <juhosg@openwrt.org>
* Copyright (C) 2013 John Crispin <blogic@openwrt.org> * Copyright (C) 2013 John Crispin <john@phrozen.org>
* *
* This driver was based on: drivers/watchdog/softdog.c * This driver was based on: drivers/watchdog/softdog.c
* *
...@@ -124,7 +124,7 @@ static struct watchdog_info rt288x_wdt_info = { ...@@ -124,7 +124,7 @@ static struct watchdog_info rt288x_wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE,
}; };
static struct watchdog_ops rt288x_wdt_ops = { static const struct watchdog_ops rt288x_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = rt288x_wdt_start, .start = rt288x_wdt_start,
.stop = rt288x_wdt_stop, .stop = rt288x_wdt_stop,
......
...@@ -23,8 +23,6 @@ ...@@ -23,8 +23,6 @@
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/types.h> #include <linux/types.h>
...@@ -46,6 +44,7 @@ ...@@ -46,6 +44,7 @@
#define S3C2410_WTCON 0x00 #define S3C2410_WTCON 0x00
#define S3C2410_WTDAT 0x04 #define S3C2410_WTDAT 0x04
#define S3C2410_WTCNT 0x08 #define S3C2410_WTCNT 0x08
#define S3C2410_WTCLRINT 0x0c
#define S3C2410_WTCNT_MAXCNT 0xffff #define S3C2410_WTCNT_MAXCNT 0xffff
...@@ -64,14 +63,15 @@ ...@@ -64,14 +63,15 @@
#define S3C2410_WTCON_PRESCALE_MASK (0xff << 8) #define S3C2410_WTCON_PRESCALE_MASK (0xff << 8)
#define S3C2410_WTCON_PRESCALE_MAX 0xff #define S3C2410_WTCON_PRESCALE_MAX 0xff
#define CONFIG_S3C2410_WATCHDOG_ATBOOT (0) #define S3C2410_WATCHDOG_ATBOOT (0)
#define CONFIG_S3C2410_WATCHDOG_DEFAULT_TIME (15) #define S3C2410_WATCHDOG_DEFAULT_TIME (15)
#define EXYNOS5_RST_STAT_REG_OFFSET 0x0404 #define EXYNOS5_RST_STAT_REG_OFFSET 0x0404
#define EXYNOS5_WDT_DISABLE_REG_OFFSET 0x0408 #define EXYNOS5_WDT_DISABLE_REG_OFFSET 0x0408
#define EXYNOS5_WDT_MASK_RESET_REG_OFFSET 0x040c #define EXYNOS5_WDT_MASK_RESET_REG_OFFSET 0x040c
#define QUIRK_HAS_PMU_CONFIG (1 << 0) #define QUIRK_HAS_PMU_CONFIG (1 << 0)
#define QUIRK_HAS_RST_STAT (1 << 1) #define QUIRK_HAS_RST_STAT (1 << 1)
#define QUIRK_HAS_WTCLRINT_REG (1 << 2)
/* These quirks require that we have a PMU register map */ /* These quirks require that we have a PMU register map */
#define QUIRKS_HAVE_PMUREG (QUIRK_HAS_PMU_CONFIG | \ #define QUIRKS_HAVE_PMUREG (QUIRK_HAS_PMU_CONFIG | \
...@@ -79,26 +79,23 @@ ...@@ -79,26 +79,23 @@
static bool nowayout = WATCHDOG_NOWAYOUT; static bool nowayout = WATCHDOG_NOWAYOUT;
static int tmr_margin; static int tmr_margin;
static int tmr_atboot = CONFIG_S3C2410_WATCHDOG_ATBOOT; static int tmr_atboot = S3C2410_WATCHDOG_ATBOOT;
static int soft_noboot; static int soft_noboot;
static int debug;
module_param(tmr_margin, int, 0); module_param(tmr_margin, int, 0);
module_param(tmr_atboot, int, 0); module_param(tmr_atboot, int, 0);
module_param(nowayout, bool, 0); module_param(nowayout, bool, 0);
module_param(soft_noboot, int, 0); module_param(soft_noboot, int, 0);
module_param(debug, int, 0);
MODULE_PARM_DESC(tmr_margin, "Watchdog tmr_margin in seconds. (default=" MODULE_PARM_DESC(tmr_margin, "Watchdog tmr_margin in seconds. (default="
__MODULE_STRING(CONFIG_S3C2410_WATCHDOG_DEFAULT_TIME) ")"); __MODULE_STRING(S3C2410_WATCHDOG_DEFAULT_TIME) ")");
MODULE_PARM_DESC(tmr_atboot, MODULE_PARM_DESC(tmr_atboot,
"Watchdog is started at boot time if set to 1, default=" "Watchdog is started at boot time if set to 1, default="
__MODULE_STRING(CONFIG_S3C2410_WATCHDOG_ATBOOT)); __MODULE_STRING(S3C2410_WATCHDOG_ATBOOT));
MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=" MODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default="
__MODULE_STRING(WATCHDOG_NOWAYOUT) ")"); __MODULE_STRING(WATCHDOG_NOWAYOUT) ")");
MODULE_PARM_DESC(soft_noboot, "Watchdog action, set to 1 to ignore reboots, " MODULE_PARM_DESC(soft_noboot, "Watchdog action, set to 1 to ignore reboots, "
"0 to reboot (default 0)"); "0 to reboot (default 0)");
MODULE_PARM_DESC(debug, "Watchdog debug, set to >1 for debug (default 0)");
/** /**
* struct s3c2410_wdt_variant - Per-variant config data * struct s3c2410_wdt_variant - Per-variant config data
...@@ -143,13 +140,18 @@ static const struct s3c2410_wdt_variant drv_data_s3c2410 = { ...@@ -143,13 +140,18 @@ static const struct s3c2410_wdt_variant drv_data_s3c2410 = {
}; };
#ifdef CONFIG_OF #ifdef CONFIG_OF
static const struct s3c2410_wdt_variant drv_data_s3c6410 = {
.quirks = QUIRK_HAS_WTCLRINT_REG,
};
static const struct s3c2410_wdt_variant drv_data_exynos5250 = { static const struct s3c2410_wdt_variant drv_data_exynos5250 = {
.disable_reg = EXYNOS5_WDT_DISABLE_REG_OFFSET, .disable_reg = EXYNOS5_WDT_DISABLE_REG_OFFSET,
.mask_reset_reg = EXYNOS5_WDT_MASK_RESET_REG_OFFSET, .mask_reset_reg = EXYNOS5_WDT_MASK_RESET_REG_OFFSET,
.mask_bit = 20, .mask_bit = 20,
.rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET, .rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET,
.rst_stat_bit = 20, .rst_stat_bit = 20,
.quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT, .quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT \
| QUIRK_HAS_WTCLRINT_REG,
}; };
static const struct s3c2410_wdt_variant drv_data_exynos5420 = { static const struct s3c2410_wdt_variant drv_data_exynos5420 = {
...@@ -158,7 +160,8 @@ static const struct s3c2410_wdt_variant drv_data_exynos5420 = { ...@@ -158,7 +160,8 @@ static const struct s3c2410_wdt_variant drv_data_exynos5420 = {
.mask_bit = 0, .mask_bit = 0,
.rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET, .rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET,
.rst_stat_bit = 9, .rst_stat_bit = 9,
.quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT, .quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT \
| QUIRK_HAS_WTCLRINT_REG,
}; };
static const struct s3c2410_wdt_variant drv_data_exynos7 = { static const struct s3c2410_wdt_variant drv_data_exynos7 = {
...@@ -167,12 +170,15 @@ static const struct s3c2410_wdt_variant drv_data_exynos7 = { ...@@ -167,12 +170,15 @@ static const struct s3c2410_wdt_variant drv_data_exynos7 = {
.mask_bit = 23, .mask_bit = 23,
.rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET, .rst_stat_reg = EXYNOS5_RST_STAT_REG_OFFSET,
.rst_stat_bit = 23, /* A57 WDTRESET */ .rst_stat_bit = 23, /* A57 WDTRESET */
.quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT, .quirks = QUIRK_HAS_PMU_CONFIG | QUIRK_HAS_RST_STAT \
| QUIRK_HAS_WTCLRINT_REG,
}; };
static const struct of_device_id s3c2410_wdt_match[] = { static const struct of_device_id s3c2410_wdt_match[] = {
{ .compatible = "samsung,s3c2410-wdt", { .compatible = "samsung,s3c2410-wdt",
.data = &drv_data_s3c2410 }, .data = &drv_data_s3c2410 },
{ .compatible = "samsung,s3c6410-wdt",
.data = &drv_data_s3c6410 },
{ .compatible = "samsung,exynos5250-wdt", { .compatible = "samsung,exynos5250-wdt",
.data = &drv_data_exynos5250 }, .data = &drv_data_exynos5250 },
{ .compatible = "samsung,exynos5420-wdt", { .compatible = "samsung,exynos5420-wdt",
...@@ -193,14 +199,6 @@ static const struct platform_device_id s3c2410_wdt_ids[] = { ...@@ -193,14 +199,6 @@ static const struct platform_device_id s3c2410_wdt_ids[] = {
}; };
MODULE_DEVICE_TABLE(platform, s3c2410_wdt_ids); MODULE_DEVICE_TABLE(platform, s3c2410_wdt_ids);
/* watchdog control routines */
#define DBG(fmt, ...) \
do { \
if (debug) \
pr_info(fmt, ##__VA_ARGS__); \
} while (0)
/* functions */ /* functions */
static inline unsigned int s3c2410wdt_max_timeout(struct clk *clock) static inline unsigned int s3c2410wdt_max_timeout(struct clk *clock)
...@@ -296,8 +294,8 @@ static int s3c2410wdt_start(struct watchdog_device *wdd) ...@@ -296,8 +294,8 @@ static int s3c2410wdt_start(struct watchdog_device *wdd)
wtcon |= S3C2410_WTCON_RSTEN; wtcon |= S3C2410_WTCON_RSTEN;
} }
DBG("%s: count=0x%08x, wtcon=%08lx\n", dev_dbg(wdt->dev, "Starting watchdog: count=0x%08x, wtcon=%08lx\n",
__func__, wdt->count, wtcon); wdt->count, wtcon);
writel(wdt->count, wdt->reg_base + S3C2410_WTDAT); writel(wdt->count, wdt->reg_base + S3C2410_WTDAT);
writel(wdt->count, wdt->reg_base + S3C2410_WTCNT); writel(wdt->count, wdt->reg_base + S3C2410_WTCNT);
...@@ -326,8 +324,8 @@ static int s3c2410wdt_set_heartbeat(struct watchdog_device *wdd, unsigned timeou ...@@ -326,8 +324,8 @@ static int s3c2410wdt_set_heartbeat(struct watchdog_device *wdd, unsigned timeou
freq = DIV_ROUND_UP(freq, 128); freq = DIV_ROUND_UP(freq, 128);
count = timeout * freq; count = timeout * freq;
DBG("%s: count=%d, timeout=%d, freq=%lu\n", dev_dbg(wdt->dev, "Heartbeat: count=%d, timeout=%d, freq=%lu\n",
__func__, count, timeout, freq); count, timeout, freq);
/* if the count is bigger than the watchdog register, /* if the count is bigger than the watchdog register,
then work out what we need to do (and if) we can then work out what we need to do (and if) we can
...@@ -343,8 +341,8 @@ static int s3c2410wdt_set_heartbeat(struct watchdog_device *wdd, unsigned timeou ...@@ -343,8 +341,8 @@ static int s3c2410wdt_set_heartbeat(struct watchdog_device *wdd, unsigned timeou
} }
} }
DBG("%s: timeout=%d, divisor=%d, count=%d (%08x)\n", dev_dbg(wdt->dev, "Heartbeat: timeout=%d, divisor=%d, count=%d (%08x)\n",
__func__, timeout, divisor, count, DIV_ROUND_UP(count, divisor)); timeout, divisor, count, DIV_ROUND_UP(count, divisor));
count = DIV_ROUND_UP(count, divisor); count = DIV_ROUND_UP(count, divisor);
wdt->count = count; wdt->count = count;
...@@ -394,7 +392,7 @@ static const struct watchdog_info s3c2410_wdt_ident = { ...@@ -394,7 +392,7 @@ static const struct watchdog_info s3c2410_wdt_ident = {
.identity = "S3C2410 Watchdog", .identity = "S3C2410 Watchdog",
}; };
static struct watchdog_ops s3c2410wdt_ops = { static const struct watchdog_ops s3c2410wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = s3c2410wdt_start, .start = s3c2410wdt_start,
.stop = s3c2410wdt_stop, .stop = s3c2410wdt_stop,
...@@ -406,7 +404,7 @@ static struct watchdog_ops s3c2410wdt_ops = { ...@@ -406,7 +404,7 @@ static struct watchdog_ops s3c2410wdt_ops = {
static struct watchdog_device s3c2410_wdd = { static struct watchdog_device s3c2410_wdd = {
.info = &s3c2410_wdt_ident, .info = &s3c2410_wdt_ident,
.ops = &s3c2410wdt_ops, .ops = &s3c2410wdt_ops,
.timeout = CONFIG_S3C2410_WATCHDOG_DEFAULT_TIME, .timeout = S3C2410_WATCHDOG_DEFAULT_TIME,
}; };
/* interrupt handler code */ /* interrupt handler code */
...@@ -418,6 +416,10 @@ static irqreturn_t s3c2410wdt_irq(int irqno, void *param) ...@@ -418,6 +416,10 @@ static irqreturn_t s3c2410wdt_irq(int irqno, void *param)
dev_info(wdt->dev, "watchdog timer expired (irq)\n"); dev_info(wdt->dev, "watchdog timer expired (irq)\n");
s3c2410wdt_keepalive(&wdt->wdt_device); s3c2410wdt_keepalive(&wdt->wdt_device);
if (wdt->drv_data->quirks & QUIRK_HAS_WTCLRINT_REG)
writel(0x1, wdt->reg_base + S3C2410_WTCLRINT);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
...@@ -505,9 +507,8 @@ static inline unsigned int s3c2410wdt_get_bootstatus(struct s3c2410_wdt *wdt) ...@@ -505,9 +507,8 @@ static inline unsigned int s3c2410wdt_get_bootstatus(struct s3c2410_wdt *wdt)
return 0; return 0;
} }
/* s3c2410_get_wdt_driver_data */
static inline struct s3c2410_wdt_variant * static inline struct s3c2410_wdt_variant *
get_wdt_drv_data(struct platform_device *pdev) s3c2410_get_wdt_drv_data(struct platform_device *pdev)
{ {
if (pdev->dev.of_node) { if (pdev->dev.of_node) {
const struct of_device_id *match; const struct of_device_id *match;
...@@ -529,8 +530,6 @@ static int s3c2410wdt_probe(struct platform_device *pdev) ...@@ -529,8 +530,6 @@ static int s3c2410wdt_probe(struct platform_device *pdev)
int started = 0; int started = 0;
int ret; int ret;
DBG("%s: probe=%p\n", __func__, pdev);
dev = &pdev->dev; dev = &pdev->dev;
wdt = devm_kzalloc(dev, sizeof(*wdt), GFP_KERNEL); wdt = devm_kzalloc(dev, sizeof(*wdt), GFP_KERNEL);
...@@ -541,7 +540,7 @@ static int s3c2410wdt_probe(struct platform_device *pdev) ...@@ -541,7 +540,7 @@ static int s3c2410wdt_probe(struct platform_device *pdev)
spin_lock_init(&wdt->lock); spin_lock_init(&wdt->lock);
wdt->wdt_device = s3c2410_wdd; wdt->wdt_device = s3c2410_wdd;
wdt->drv_data = get_wdt_drv_data(pdev); wdt->drv_data = s3c2410_get_wdt_drv_data(pdev);
if (wdt->drv_data->quirks & QUIRKS_HAVE_PMUREG) { if (wdt->drv_data->quirks & QUIRKS_HAVE_PMUREG) {
wdt->pmureg = syscon_regmap_lookup_by_phandle(dev->of_node, wdt->pmureg = syscon_regmap_lookup_by_phandle(dev->of_node,
"samsung,syscon-phandle"); "samsung,syscon-phandle");
...@@ -566,8 +565,6 @@ static int s3c2410wdt_probe(struct platform_device *pdev) ...@@ -566,8 +565,6 @@ static int s3c2410wdt_probe(struct platform_device *pdev)
goto err; goto err;
} }
DBG("probe: mapped reg_base=%p\n", wdt->reg_base);
wdt->clock = devm_clk_get(dev, "watchdog"); wdt->clock = devm_clk_get(dev, "watchdog");
if (IS_ERR(wdt->clock)) { if (IS_ERR(wdt->clock)) {
dev_err(dev, "failed to find watchdog clock source\n"); dev_err(dev, "failed to find watchdog clock source\n");
...@@ -600,12 +597,12 @@ static int s3c2410wdt_probe(struct platform_device *pdev) ...@@ -600,12 +597,12 @@ static int s3c2410wdt_probe(struct platform_device *pdev)
wdt->wdt_device.timeout); wdt->wdt_device.timeout);
if (ret) { if (ret) {
started = s3c2410wdt_set_heartbeat(&wdt->wdt_device, started = s3c2410wdt_set_heartbeat(&wdt->wdt_device,
CONFIG_S3C2410_WATCHDOG_DEFAULT_TIME); S3C2410_WATCHDOG_DEFAULT_TIME);
if (started == 0) if (started == 0)
dev_info(dev, dev_info(dev,
"tmr_margin value out of range, default %d used\n", "tmr_margin value out of range, default %d used\n",
CONFIG_S3C2410_WATCHDOG_DEFAULT_TIME); S3C2410_WATCHDOG_DEFAULT_TIME);
else else
dev_info(dev, "default timer value is out of range, " dev_info(dev, "default timer value is out of range, "
"cannot start\n"); "cannot start\n");
......
...@@ -188,12 +188,14 @@ static int __init sa1100dog_init(void) ...@@ -188,12 +188,14 @@ static int __init sa1100dog_init(void)
pre_margin = oscr_freq * margin; pre_margin = oscr_freq * margin;
ret = misc_register(&sa1100dog_miscdev); ret = misc_register(&sa1100dog_miscdev);
if (ret == 0) if (ret == 0) {
pr_info("SA1100/PXA2xx Watchdog Timer: timer margin %d sec\n", pr_info("SA1100/PXA2xx Watchdog Timer: timer margin %d sec\n",
margin); margin);
return ret; return 0;
err: }
clk_disable_unprepare(clk); clk_disable_unprepare(clk);
err:
clk_put(clk); clk_put(clk);
return ret; return ret;
} }
......
...@@ -28,7 +28,7 @@ ...@@ -28,7 +28,7 @@
struct sama5d4_wdt { struct sama5d4_wdt {
struct watchdog_device wdd; struct watchdog_device wdd;
void __iomem *reg_base; void __iomem *reg_base;
u32 config; u32 mr;
}; };
static int wdt_timeout = WDT_DEFAULT_TIMEOUT; static int wdt_timeout = WDT_DEFAULT_TIMEOUT;
...@@ -53,11 +53,9 @@ MODULE_PARM_DESC(nowayout, ...@@ -53,11 +53,9 @@ MODULE_PARM_DESC(nowayout,
static int sama5d4_wdt_start(struct watchdog_device *wdd) static int sama5d4_wdt_start(struct watchdog_device *wdd)
{ {
struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd); struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd);
u32 reg;
reg = wdt_read(wdt, AT91_WDT_MR); wdt->mr &= ~AT91_WDT_WDDIS;
reg &= ~AT91_WDT_WDDIS; wdt_write(wdt, AT91_WDT_MR, wdt->mr);
wdt_write(wdt, AT91_WDT_MR, reg);
return 0; return 0;
} }
...@@ -65,11 +63,9 @@ static int sama5d4_wdt_start(struct watchdog_device *wdd) ...@@ -65,11 +63,9 @@ static int sama5d4_wdt_start(struct watchdog_device *wdd)
static int sama5d4_wdt_stop(struct watchdog_device *wdd) static int sama5d4_wdt_stop(struct watchdog_device *wdd)
{ {
struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd); struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd);
u32 reg;
reg = wdt_read(wdt, AT91_WDT_MR); wdt->mr |= AT91_WDT_WDDIS;
reg |= AT91_WDT_WDDIS; wdt_write(wdt, AT91_WDT_MR, wdt->mr);
wdt_write(wdt, AT91_WDT_MR, reg);
return 0; return 0;
} }
...@@ -88,14 +84,12 @@ static int sama5d4_wdt_set_timeout(struct watchdog_device *wdd, ...@@ -88,14 +84,12 @@ static int sama5d4_wdt_set_timeout(struct watchdog_device *wdd,
{ {
struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd); struct sama5d4_wdt *wdt = watchdog_get_drvdata(wdd);
u32 value = WDT_SEC2TICKS(timeout); u32 value = WDT_SEC2TICKS(timeout);
u32 reg;
reg = wdt_read(wdt, AT91_WDT_MR); wdt->mr &= ~AT91_WDT_WDV;
reg &= ~AT91_WDT_WDV; wdt->mr &= ~AT91_WDT_WDD;
reg &= ~AT91_WDT_WDD; wdt->mr |= AT91_WDT_SET_WDV(value);
reg |= AT91_WDT_SET_WDV(value); wdt->mr |= AT91_WDT_SET_WDD(value);
reg |= AT91_WDT_SET_WDD(value); wdt_write(wdt, AT91_WDT_MR, wdt->mr);
wdt_write(wdt, AT91_WDT_MR, reg);
wdd->timeout = timeout; wdd->timeout = timeout;
...@@ -107,7 +101,7 @@ static const struct watchdog_info sama5d4_wdt_info = { ...@@ -107,7 +101,7 @@ static const struct watchdog_info sama5d4_wdt_info = {
.identity = "Atmel SAMA5D4 Watchdog", .identity = "Atmel SAMA5D4 Watchdog",
}; };
static struct watchdog_ops sama5d4_wdt_ops = { static const struct watchdog_ops sama5d4_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = sama5d4_wdt_start, .start = sama5d4_wdt_start,
.stop = sama5d4_wdt_stop, .stop = sama5d4_wdt_stop,
...@@ -132,19 +126,19 @@ static int of_sama5d4_wdt_init(struct device_node *np, struct sama5d4_wdt *wdt) ...@@ -132,19 +126,19 @@ static int of_sama5d4_wdt_init(struct device_node *np, struct sama5d4_wdt *wdt)
{ {
const char *tmp; const char *tmp;
wdt->config = AT91_WDT_WDDIS; wdt->mr = AT91_WDT_WDDIS;
if (!of_property_read_string(np, "atmel,watchdog-type", &tmp) && if (!of_property_read_string(np, "atmel,watchdog-type", &tmp) &&
!strcmp(tmp, "software")) !strcmp(tmp, "software"))
wdt->config |= AT91_WDT_WDFIEN; wdt->mr |= AT91_WDT_WDFIEN;
else else
wdt->config |= AT91_WDT_WDRSTEN; wdt->mr |= AT91_WDT_WDRSTEN;
if (of_property_read_bool(np, "atmel,idle-halt")) if (of_property_read_bool(np, "atmel,idle-halt"))
wdt->config |= AT91_WDT_WDIDLEHLT; wdt->mr |= AT91_WDT_WDIDLEHLT;
if (of_property_read_bool(np, "atmel,dbg-halt")) if (of_property_read_bool(np, "atmel,dbg-halt"))
wdt->config |= AT91_WDT_WDDBGHLT; wdt->mr |= AT91_WDT_WDDBGHLT;
return 0; return 0;
} }
...@@ -163,11 +157,10 @@ static int sama5d4_wdt_init(struct sama5d4_wdt *wdt) ...@@ -163,11 +157,10 @@ static int sama5d4_wdt_init(struct sama5d4_wdt *wdt)
reg &= ~AT91_WDT_WDDIS; reg &= ~AT91_WDT_WDDIS;
wdt_write(wdt, AT91_WDT_MR, reg); wdt_write(wdt, AT91_WDT_MR, reg);
reg = wdt->config; wdt->mr |= AT91_WDT_SET_WDD(value);
reg |= AT91_WDT_SET_WDD(value); wdt->mr |= AT91_WDT_SET_WDV(value);
reg |= AT91_WDT_SET_WDV(value);
wdt_write(wdt, AT91_WDT_MR, reg); wdt_write(wdt, AT91_WDT_MR, wdt->mr);
return 0; return 0;
} }
...@@ -211,7 +204,7 @@ static int sama5d4_wdt_probe(struct platform_device *pdev) ...@@ -211,7 +204,7 @@ static int sama5d4_wdt_probe(struct platform_device *pdev)
return ret; return ret;
} }
if ((wdt->config & AT91_WDT_WDFIEN) && irq) { if ((wdt->mr & AT91_WDT_WDFIEN) && irq) {
ret = devm_request_irq(&pdev->dev, irq, sama5d4_wdt_irq_handler, ret = devm_request_irq(&pdev->dev, irq, sama5d4_wdt_irq_handler,
IRQF_SHARED | IRQF_IRQPOLL | IRQF_SHARED | IRQF_IRQPOLL |
IRQF_NO_SUSPEND, pdev->name, pdev); IRQF_NO_SUSPEND, pdev->name, pdev);
...@@ -265,11 +258,28 @@ static const struct of_device_id sama5d4_wdt_of_match[] = { ...@@ -265,11 +258,28 @@ static const struct of_device_id sama5d4_wdt_of_match[] = {
}; };
MODULE_DEVICE_TABLE(of, sama5d4_wdt_of_match); MODULE_DEVICE_TABLE(of, sama5d4_wdt_of_match);
#ifdef CONFIG_PM_SLEEP
static int sama5d4_wdt_resume(struct device *dev)
{
struct sama5d4_wdt *wdt = dev_get_drvdata(dev);
wdt_write(wdt, AT91_WDT_MR, wdt->mr & ~AT91_WDT_WDDIS);
if (wdt->mr & AT91_WDT_WDDIS)
wdt_write(wdt, AT91_WDT_MR, wdt->mr);
return 0;
}
#endif
static SIMPLE_DEV_PM_OPS(sama5d4_wdt_pm_ops, NULL,
sama5d4_wdt_resume);
static struct platform_driver sama5d4_wdt_driver = { static struct platform_driver sama5d4_wdt_driver = {
.probe = sama5d4_wdt_probe, .probe = sama5d4_wdt_probe,
.remove = sama5d4_wdt_remove, .remove = sama5d4_wdt_remove,
.driver = { .driver = {
.name = "sama5d4_wdt", .name = "sama5d4_wdt",
.pm = &sama5d4_wdt_pm_ops,
.of_match_table = sama5d4_wdt_of_match, .of_match_table = sama5d4_wdt_of_match,
} }
}; };
......
...@@ -207,7 +207,7 @@ static irqreturn_t sbsa_gwdt_interrupt(int irq, void *dev_id) ...@@ -207,7 +207,7 @@ static irqreturn_t sbsa_gwdt_interrupt(int irq, void *dev_id)
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static struct watchdog_info sbsa_gwdt_info = { static const struct watchdog_info sbsa_gwdt_info = {
.identity = WATCHDOG_NAME, .identity = WATCHDOG_NAME,
.options = WDIOF_SETTIMEOUT | .options = WDIOF_SETTIMEOUT |
WDIOF_KEEPALIVEPING | WDIOF_KEEPALIVEPING |
...@@ -215,7 +215,7 @@ static struct watchdog_info sbsa_gwdt_info = { ...@@ -215,7 +215,7 @@ static struct watchdog_info sbsa_gwdt_info = {
WDIOF_CARDRESET, WDIOF_CARDRESET,
}; };
static struct watchdog_ops sbsa_gwdt_ops = { static const struct watchdog_ops sbsa_gwdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = sbsa_gwdt_start, .start = sbsa_gwdt_start,
.stop = sbsa_gwdt_stop, .stop = sbsa_gwdt_stop,
......
...@@ -127,7 +127,7 @@ static const struct watchdog_info sirfsoc_wdt_ident = { ...@@ -127,7 +127,7 @@ static const struct watchdog_info sirfsoc_wdt_ident = {
.identity = "SiRFSOC Watchdog", .identity = "SiRFSOC Watchdog",
}; };
static struct watchdog_ops sirfsoc_wdt_ops = { static const struct watchdog_ops sirfsoc_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = sirfsoc_wdt_enable, .start = sirfsoc_wdt_enable,
.stop = sirfsoc_wdt_disable, .stop = sirfsoc_wdt_disable,
......
...@@ -87,11 +87,13 @@ static int softdog_ping(struct watchdog_device *w) ...@@ -87,11 +87,13 @@ static int softdog_ping(struct watchdog_device *w)
if (!mod_timer(&softdog_ticktock, jiffies + (w->timeout * HZ))) if (!mod_timer(&softdog_ticktock, jiffies + (w->timeout * HZ)))
__module_get(THIS_MODULE); __module_get(THIS_MODULE);
if (w->pretimeout) if (IS_ENABLED(CONFIG_SOFT_WATCHDOG_PRETIMEOUT)) {
mod_timer(&softdog_preticktock, jiffies + if (w->pretimeout)
(w->timeout - w->pretimeout) * HZ); mod_timer(&softdog_preticktock, jiffies +
else (w->timeout - w->pretimeout) * HZ);
del_timer(&softdog_preticktock); else
del_timer(&softdog_preticktock);
}
return 0; return 0;
} }
...@@ -101,15 +103,15 @@ static int softdog_stop(struct watchdog_device *w) ...@@ -101,15 +103,15 @@ static int softdog_stop(struct watchdog_device *w)
if (del_timer(&softdog_ticktock)) if (del_timer(&softdog_ticktock))
module_put(THIS_MODULE); module_put(THIS_MODULE);
del_timer(&softdog_preticktock); if (IS_ENABLED(CONFIG_SOFT_WATCHDOG_PRETIMEOUT))
del_timer(&softdog_preticktock);
return 0; return 0;
} }
static struct watchdog_info softdog_info = { static struct watchdog_info softdog_info = {
.identity = "Software Watchdog", .identity = "Software Watchdog",
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE | .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE,
WDIOF_PRETIMEOUT,
}; };
static const struct watchdog_ops softdog_ops = { static const struct watchdog_ops softdog_ops = {
...@@ -134,6 +136,9 @@ static int __init softdog_init(void) ...@@ -134,6 +136,9 @@ static int __init softdog_init(void)
watchdog_set_nowayout(&softdog_dev, nowayout); watchdog_set_nowayout(&softdog_dev, nowayout);
watchdog_stop_on_reboot(&softdog_dev); watchdog_stop_on_reboot(&softdog_dev);
if (IS_ENABLED(CONFIG_SOFT_WATCHDOG_PRETIMEOUT))
softdog_info.options |= WDIOF_PRETIMEOUT;
ret = watchdog_register_device(&softdog_dev); ret = watchdog_register_device(&softdog_dev);
if (ret) if (ret)
return ret; return ret;
......
...@@ -77,7 +77,7 @@ static const struct watchdog_info sun4v_wdt_ident = { ...@@ -77,7 +77,7 @@ static const struct watchdog_info sun4v_wdt_ident = {
.firmware_version = 0, .firmware_version = 0,
}; };
static struct watchdog_ops sun4v_wdt_ops = { static const struct watchdog_ops sun4v_wdt_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.start = sun4v_wdt_ping, .start = sun4v_wdt_ping,
.stop = sun4v_wdt_stop, .stop = sun4v_wdt_stop,
......
...@@ -242,8 +242,6 @@ static int sunxi_wdt_probe(struct platform_device *pdev) ...@@ -242,8 +242,6 @@ static int sunxi_wdt_probe(struct platform_device *pdev)
if (!sunxi_wdt) if (!sunxi_wdt)
return -EINVAL; return -EINVAL;
platform_set_drvdata(pdev, sunxi_wdt);
device = of_match_device(sunxi_wdt_dt_ids, &pdev->dev); device = of_match_device(sunxi_wdt_dt_ids, &pdev->dev);
if (!device) if (!device)
return -ENODEV; return -ENODEV;
...@@ -270,7 +268,8 @@ static int sunxi_wdt_probe(struct platform_device *pdev) ...@@ -270,7 +268,8 @@ static int sunxi_wdt_probe(struct platform_device *pdev)
sunxi_wdt_stop(&sunxi_wdt->wdt_dev); sunxi_wdt_stop(&sunxi_wdt->wdt_dev);
err = watchdog_register_device(&sunxi_wdt->wdt_dev); watchdog_stop_on_reboot(&sunxi_wdt->wdt_dev);
err = devm_watchdog_register_device(&pdev->dev, &sunxi_wdt->wdt_dev);
if (unlikely(err)) if (unlikely(err))
return err; return err;
...@@ -280,27 +279,8 @@ static int sunxi_wdt_probe(struct platform_device *pdev) ...@@ -280,27 +279,8 @@ static int sunxi_wdt_probe(struct platform_device *pdev)
return 0; return 0;
} }
static int sunxi_wdt_remove(struct platform_device *pdev)
{
struct sunxi_wdt_dev *sunxi_wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&sunxi_wdt->wdt_dev);
watchdog_set_drvdata(&sunxi_wdt->wdt_dev, NULL);
return 0;
}
static void sunxi_wdt_shutdown(struct platform_device *pdev)
{
struct sunxi_wdt_dev *sunxi_wdt = platform_get_drvdata(pdev);
sunxi_wdt_stop(&sunxi_wdt->wdt_dev);
}
static struct platform_driver sunxi_wdt_driver = { static struct platform_driver sunxi_wdt_driver = {
.probe = sunxi_wdt_probe, .probe = sunxi_wdt_probe,
.remove = sunxi_wdt_remove,
.shutdown = sunxi_wdt_shutdown,
.driver = { .driver = {
.name = DRV_NAME, .name = DRV_NAME,
.of_match_table = sunxi_wdt_dt_ids, .of_match_table = sunxi_wdt_dt_ids,
......
...@@ -15,9 +15,7 @@ ...@@ -15,9 +15,7 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/notifier.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/reboot.h>
#include <linux/watchdog.h> #include <linux/watchdog.h>
#define DEFAULT_TIMEOUT 30 #define DEFAULT_TIMEOUT 30
...@@ -47,7 +45,6 @@ struct tangox_wdt_device { ...@@ -47,7 +45,6 @@ struct tangox_wdt_device {
void __iomem *base; void __iomem *base;
unsigned long clk_rate; unsigned long clk_rate;
struct clk *clk; struct clk *clk;
struct notifier_block restart;
}; };
static int tangox_wdt_set_timeout(struct watchdog_device *wdt, static int tangox_wdt_set_timeout(struct watchdog_device *wdt,
...@@ -96,24 +93,24 @@ static const struct watchdog_info tangox_wdt_info = { ...@@ -96,24 +93,24 @@ static const struct watchdog_info tangox_wdt_info = {
.identity = "tangox watchdog", .identity = "tangox watchdog",
}; };
static int tangox_wdt_restart(struct watchdog_device *wdt,
unsigned long action, void *data)
{
struct tangox_wdt_device *dev = watchdog_get_drvdata(wdt);
writel(1, dev->base + WD_COUNTER);
return 0;
}
static const struct watchdog_ops tangox_wdt_ops = { static const struct watchdog_ops tangox_wdt_ops = {
.start = tangox_wdt_start, .start = tangox_wdt_start,
.stop = tangox_wdt_stop, .stop = tangox_wdt_stop,
.set_timeout = tangox_wdt_set_timeout, .set_timeout = tangox_wdt_set_timeout,
.get_timeleft = tangox_wdt_get_timeleft, .get_timeleft = tangox_wdt_get_timeleft,
.restart = tangox_wdt_restart,
}; };
static int tangox_wdt_restart(struct notifier_block *nb, unsigned long action,
void *data)
{
struct tangox_wdt_device *dev =
container_of(nb, struct tangox_wdt_device, restart);
writel(1, dev->base + WD_COUNTER);
return NOTIFY_DONE;
}
static int tangox_wdt_probe(struct platform_device *pdev) static int tangox_wdt_probe(struct platform_device *pdev)
{ {
struct tangox_wdt_device *dev; struct tangox_wdt_device *dev;
...@@ -174,18 +171,14 @@ static int tangox_wdt_probe(struct platform_device *pdev) ...@@ -174,18 +171,14 @@ static int tangox_wdt_probe(struct platform_device *pdev)
tangox_wdt_start(&dev->wdt); tangox_wdt_start(&dev->wdt);
} }
watchdog_set_restart_priority(&dev->wdt, 128);
err = watchdog_register_device(&dev->wdt); err = watchdog_register_device(&dev->wdt);
if (err) if (err)
goto err; goto err;
platform_set_drvdata(pdev, dev); platform_set_drvdata(pdev, dev);
dev->restart.notifier_call = tangox_wdt_restart;
dev->restart.priority = 128;
err = register_restart_handler(&dev->restart);
if (err)
dev_warn(&pdev->dev, "failed to register restart handler\n");
dev_info(&pdev->dev, "SMP86xx/SMP87xx watchdog registered\n"); dev_info(&pdev->dev, "SMP86xx/SMP87xx watchdog registered\n");
return 0; return 0;
...@@ -202,7 +195,6 @@ static int tangox_wdt_remove(struct platform_device *pdev) ...@@ -202,7 +195,6 @@ static int tangox_wdt_remove(struct platform_device *pdev)
tangox_wdt_stop(&dev->wdt); tangox_wdt_stop(&dev->wdt);
clk_disable_unprepare(dev->clk); clk_disable_unprepare(dev->clk);
unregister_restart_handler(&dev->restart);
watchdog_unregister_device(&dev->wdt); watchdog_unregister_device(&dev->wdt);
return 0; return 0;
......
...@@ -226,7 +226,7 @@ static int tegra_wdt_probe(struct platform_device *pdev) ...@@ -226,7 +226,7 @@ static int tegra_wdt_probe(struct platform_device *pdev)
watchdog_set_nowayout(wdd, nowayout); watchdog_set_nowayout(wdd, nowayout);
ret = watchdog_register_device(wdd); ret = devm_watchdog_register_device(&pdev->dev, wdd);
if (ret) { if (ret) {
dev_err(&pdev->dev, dev_err(&pdev->dev,
"failed to register watchdog device\n"); "failed to register watchdog device\n");
...@@ -248,8 +248,6 @@ static int tegra_wdt_remove(struct platform_device *pdev) ...@@ -248,8 +248,6 @@ static int tegra_wdt_remove(struct platform_device *pdev)
tegra_wdt_stop(&wdt->wdd); tegra_wdt_stop(&wdt->wdd);
watchdog_unregister_device(&wdt->wdd);
dev_info(&pdev->dev, "removed wdt\n"); dev_info(&pdev->dev, "removed wdt\n");
return 0; return 0;
......
...@@ -13,428 +13,159 @@ ...@@ -13,428 +13,159 @@
* warranty of any kind, whether express or implied. * warranty of any kind, whether express or implied.
*/ */
#include <linux/fs.h>
#include <linux/io.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/miscdevice.h>
#include <linux/mutex.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/slab.h> #include <linux/module.h>
#include <linux/watchdog.h> #include <linux/watchdog.h>
#include <linux/uaccess.h> #include <linux/io.h>
#define TS72XX_WDT_FEED_VAL 0x05 #define TS72XX_WDT_DEFAULT_TIMEOUT 30
#define TS72XX_WDT_DEFAULT_TIMEOUT 8
static int timeout = TS72XX_WDT_DEFAULT_TIMEOUT; static int timeout;
module_param(timeout, int, 0); module_param(timeout, int, 0);
MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds. " MODULE_PARM_DESC(timeout, "Watchdog timeout in seconds.");
"(1 <= timeout <= 8, default="
__MODULE_STRING(TS72XX_WDT_DEFAULT_TIMEOUT)
")");
static bool nowayout = WATCHDOG_NOWAYOUT; static bool nowayout = WATCHDOG_NOWAYOUT;
module_param(nowayout, bool, 0); module_param(nowayout, bool, 0);
MODULE_PARM_DESC(nowayout, "Disable watchdog shutdown on close"); MODULE_PARM_DESC(nowayout, "Disable watchdog shutdown on close");
/** /* priv->control_reg */
* struct ts72xx_wdt - watchdog control structure #define TS72XX_WDT_CTRL_DISABLE 0x00
* @lock: lock that protects this structure #define TS72XX_WDT_CTRL_250MS 0x01
* @regval: watchdog timeout value suitable for control register #define TS72XX_WDT_CTRL_500MS 0x02
* @flags: flags controlling watchdog device state #define TS72XX_WDT_CTRL_1SEC 0x03
* @control_reg: watchdog control register #define TS72XX_WDT_CTRL_RESERVED 0x04
* @feed_reg: watchdog feed register #define TS72XX_WDT_CTRL_2SEC 0x05
* @pdev: back pointer to platform dev #define TS72XX_WDT_CTRL_4SEC 0x06
*/ #define TS72XX_WDT_CTRL_8SEC 0x07
struct ts72xx_wdt {
struct mutex lock; /* priv->feed_reg */
int regval; #define TS72XX_WDT_FEED_VAL 0x05
#define TS72XX_WDT_BUSY_FLAG 1
#define TS72XX_WDT_EXPECT_CLOSE_FLAG 2
int flags;
struct ts72xx_wdt_priv {
void __iomem *control_reg; void __iomem *control_reg;
void __iomem *feed_reg; void __iomem *feed_reg;
struct watchdog_device wdd;
struct platform_device *pdev; unsigned char regval;
}; };
static struct platform_device *ts72xx_wdt_pdev; static int ts72xx_wdt_start(struct watchdog_device *wdd)
/*
* TS-72xx Watchdog supports following timeouts (value written
* to control register):
* value description
* -------------------------
* 0x00 watchdog disabled
* 0x01 250ms
* 0x02 500ms
* 0x03 1s
* 0x04 reserved
* 0x05 2s
* 0x06 4s
* 0x07 8s
*
* Timeouts below 1s are not very usable so we don't
* allow them at all.
*
* We provide two functions that convert between these:
* timeout_to_regval() and regval_to_timeout().
*/
static const struct {
int timeout;
int regval;
} ts72xx_wdt_map[] = {
{ 1, 3 },
{ 2, 5 },
{ 4, 6 },
{ 8, 7 },
};
/**
* timeout_to_regval() - converts given timeout to control register value
* @new_timeout: timeout in seconds to be converted
*
* Function converts given @new_timeout into valid value that can
* be programmed into watchdog control register. When conversion is
* not possible, function returns %-EINVAL.
*/
static int timeout_to_regval(int new_timeout)
{
int i;
/* first limit it to 1 - 8 seconds */
new_timeout = clamp_val(new_timeout, 1, 8);
for (i = 0; i < ARRAY_SIZE(ts72xx_wdt_map); i++) {
if (ts72xx_wdt_map[i].timeout >= new_timeout)
return ts72xx_wdt_map[i].regval;
}
return -EINVAL;
}
/**
* regval_to_timeout() - converts control register value to timeout
* @regval: control register value to be converted
*
* Function converts given @regval to timeout in seconds (1, 2, 4 or 8).
* If @regval cannot be converted, function returns %-EINVAL.
*/
static int regval_to_timeout(int regval)
{ {
int i; struct ts72xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
for (i = 0; i < ARRAY_SIZE(ts72xx_wdt_map); i++) { writeb(TS72XX_WDT_FEED_VAL, priv->feed_reg);
if (ts72xx_wdt_map[i].regval == regval) writeb(priv->regval, priv->control_reg);
return ts72xx_wdt_map[i].timeout;
}
return -EINVAL; return 0;
} }
/** static int ts72xx_wdt_stop(struct watchdog_device *wdd)
* ts72xx_wdt_kick() - kick the watchdog
* @wdt: watchdog to be kicked
*
* Called with @wdt->lock held.
*/
static inline void ts72xx_wdt_kick(struct ts72xx_wdt *wdt)
{ {
__raw_writeb(TS72XX_WDT_FEED_VAL, wdt->feed_reg); struct ts72xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
}
/** writeb(TS72XX_WDT_FEED_VAL, priv->feed_reg);
* ts72xx_wdt_start() - starts the watchdog timer writeb(TS72XX_WDT_CTRL_DISABLE, priv->control_reg);
* @wdt: watchdog to be started
*
* This function programs timeout to watchdog timer
* and starts it.
*
* Called with @wdt->lock held.
*/
static void ts72xx_wdt_start(struct ts72xx_wdt *wdt)
{
/*
* To program the wdt, it first must be "fed" and
* only after that (within 30 usecs) the configuration
* can be changed.
*/
ts72xx_wdt_kick(wdt);
__raw_writeb((u8)wdt->regval, wdt->control_reg);
}
/** return 0;
* ts72xx_wdt_stop() - stops the watchdog timer
* @wdt: watchdog to be stopped
*
* Called with @wdt->lock held.
*/
static void ts72xx_wdt_stop(struct ts72xx_wdt *wdt)
{
ts72xx_wdt_kick(wdt);
__raw_writeb(0, wdt->control_reg);
} }
static int ts72xx_wdt_open(struct inode *inode, struct file *file) static int ts72xx_wdt_ping(struct watchdog_device *wdd)
{ {
struct ts72xx_wdt *wdt = platform_get_drvdata(ts72xx_wdt_pdev); struct ts72xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
int regval;
/*
* Try to convert default timeout to valid register
* value first.
*/
regval = timeout_to_regval(timeout);
if (regval < 0) {
dev_err(&wdt->pdev->dev,
"failed to convert timeout (%d) to register value\n",
timeout);
return regval;
}
if (mutex_lock_interruptible(&wdt->lock))
return -ERESTARTSYS;
if ((wdt->flags & TS72XX_WDT_BUSY_FLAG) != 0) { writeb(TS72XX_WDT_FEED_VAL, priv->feed_reg);
mutex_unlock(&wdt->lock);
return -EBUSY;
}
wdt->flags = TS72XX_WDT_BUSY_FLAG;
wdt->regval = regval;
file->private_data = wdt;
ts72xx_wdt_start(wdt);
mutex_unlock(&wdt->lock); return 0;
return nonseekable_open(inode, file);
} }
static int ts72xx_wdt_release(struct inode *inode, struct file *file) static int ts72xx_wdt_settimeout(struct watchdog_device *wdd, unsigned int to)
{ {
struct ts72xx_wdt *wdt = file->private_data; struct ts72xx_wdt_priv *priv = watchdog_get_drvdata(wdd);
if (mutex_lock_interruptible(&wdt->lock)) if (to == 1) {
return -ERESTARTSYS; priv->regval = TS72XX_WDT_CTRL_1SEC;
} else if (to == 2) {
if ((wdt->flags & TS72XX_WDT_EXPECT_CLOSE_FLAG) != 0) { priv->regval = TS72XX_WDT_CTRL_2SEC;
ts72xx_wdt_stop(wdt); } else if (to <= 4) {
priv->regval = TS72XX_WDT_CTRL_4SEC;
to = 4;
} else { } else {
dev_warn(&wdt->pdev->dev, priv->regval = TS72XX_WDT_CTRL_8SEC;
"TS-72XX WDT device closed unexpectly. " if (to <= 8)
"Watchdog timer will not stop!\n"); to = 8;
/*
* Kick it one more time, to give userland some time
* to recover (for example, respawning the kicker
* daemon).
*/
ts72xx_wdt_kick(wdt);
} }
wdt->flags = 0; wdd->timeout = to;
mutex_unlock(&wdt->lock); if (watchdog_active(wdd)) {
return 0; ts72xx_wdt_stop(wdd);
} ts72xx_wdt_start(wdd);
static ssize_t ts72xx_wdt_write(struct file *file,
const char __user *data,
size_t len,
loff_t *ppos)
{
struct ts72xx_wdt *wdt = file->private_data;
if (!len)
return 0;
if (mutex_lock_interruptible(&wdt->lock))
return -ERESTARTSYS;
ts72xx_wdt_kick(wdt);
/*
* Support for magic character closing. User process
* writes 'V' into the device, just before it is closed.
* This means that we know that the wdt timer can be
* stopped after user closes the device.
*/
if (!nowayout) {
int i;
for (i = 0; i < len; i++) {
char c;
/* In case it was set long ago */
wdt->flags &= ~TS72XX_WDT_EXPECT_CLOSE_FLAG;
if (get_user(c, data + i)) {
mutex_unlock(&wdt->lock);
return -EFAULT;
}
if (c == 'V') {
wdt->flags |= TS72XX_WDT_EXPECT_CLOSE_FLAG;
break;
}
}
} }
mutex_unlock(&wdt->lock); return 0;
return len;
} }
static const struct watchdog_info winfo = { static const struct watchdog_info ts72xx_wdt_ident = {
.options = WDIOF_KEEPALIVEPING | WDIOF_SETTIMEOUT | .options = WDIOF_KEEPALIVEPING |
WDIOF_SETTIMEOUT |
WDIOF_MAGICCLOSE, WDIOF_MAGICCLOSE,
.firmware_version = 1, .firmware_version = 1,
.identity = "TS-72XX WDT", .identity = "TS-72XX WDT",
}; };
static long ts72xx_wdt_ioctl(struct file *file, unsigned int cmd, static struct watchdog_ops ts72xx_wdt_ops = {
unsigned long arg)
{
struct ts72xx_wdt *wdt = file->private_data;
void __user *argp = (void __user *)arg;
int __user *p = (int __user *)argp;
int error = 0;
if (mutex_lock_interruptible(&wdt->lock))
return -ERESTARTSYS;
switch (cmd) {
case WDIOC_GETSUPPORT:
if (copy_to_user(argp, &winfo, sizeof(winfo)))
error = -EFAULT;
break;
case WDIOC_GETSTATUS:
case WDIOC_GETBOOTSTATUS:
error = put_user(0, p);
break;
case WDIOC_KEEPALIVE:
ts72xx_wdt_kick(wdt);
break;
case WDIOC_SETOPTIONS: {
int options;
error = get_user(options, p);
if (error)
break;
error = -EINVAL;
if ((options & WDIOS_DISABLECARD) != 0) {
ts72xx_wdt_stop(wdt);
error = 0;
}
if ((options & WDIOS_ENABLECARD) != 0) {
ts72xx_wdt_start(wdt);
error = 0;
}
break;
}
case WDIOC_SETTIMEOUT: {
int new_timeout;
int regval;
error = get_user(new_timeout, p);
if (error)
break;
regval = timeout_to_regval(new_timeout);
if (regval < 0) {
error = regval;
break;
}
ts72xx_wdt_stop(wdt);
wdt->regval = regval;
ts72xx_wdt_start(wdt);
/*FALLTHROUGH*/
}
case WDIOC_GETTIMEOUT:
error = put_user(regval_to_timeout(wdt->regval), p);
break;
default:
error = -ENOTTY;
break;
}
mutex_unlock(&wdt->lock);
return error;
}
static const struct file_operations ts72xx_wdt_fops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.llseek = no_llseek, .start = ts72xx_wdt_start,
.open = ts72xx_wdt_open, .stop = ts72xx_wdt_stop,
.release = ts72xx_wdt_release, .ping = ts72xx_wdt_ping,
.write = ts72xx_wdt_write, .set_timeout = ts72xx_wdt_settimeout,
.unlocked_ioctl = ts72xx_wdt_ioctl,
};
static struct miscdevice ts72xx_wdt_miscdev = {
.minor = WATCHDOG_MINOR,
.name = "watchdog",
.fops = &ts72xx_wdt_fops,
}; };
static int ts72xx_wdt_probe(struct platform_device *pdev) static int ts72xx_wdt_probe(struct platform_device *pdev)
{ {
struct ts72xx_wdt *wdt; struct ts72xx_wdt_priv *priv;
struct resource *r1, *r2; struct watchdog_device *wdd;
int error = 0; struct resource *res;
int ret;
wdt = devm_kzalloc(&pdev->dev, sizeof(struct ts72xx_wdt), GFP_KERNEL); priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
if (!wdt) if (!priv)
return -ENOMEM; return -ENOMEM;
r1 = platform_get_resource(pdev, IORESOURCE_MEM, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
wdt->control_reg = devm_ioremap_resource(&pdev->dev, r1); priv->control_reg = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(wdt->control_reg)) if (IS_ERR(priv->control_reg))
return PTR_ERR(wdt->control_reg); return PTR_ERR(priv->control_reg);
r2 = platform_get_resource(pdev, IORESOURCE_MEM, 1); res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
wdt->feed_reg = devm_ioremap_resource(&pdev->dev, r2); priv->feed_reg = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(wdt->feed_reg)) if (IS_ERR(priv->feed_reg))
return PTR_ERR(wdt->feed_reg); return PTR_ERR(priv->feed_reg);
platform_set_drvdata(pdev, wdt); wdd = &priv->wdd;
ts72xx_wdt_pdev = pdev; wdd->info = &ts72xx_wdt_ident;
wdt->pdev = pdev; wdd->ops = &ts72xx_wdt_ops;
mutex_init(&wdt->lock); wdd->min_timeout = 1;
wdd->max_hw_heartbeat_ms = 8000;
wdd->parent = &pdev->dev;
/* make sure that the watchdog is disabled */ watchdog_set_nowayout(wdd, nowayout);
ts72xx_wdt_stop(wdt);
error = misc_register(&ts72xx_wdt_miscdev); wdd->timeout = TS72XX_WDT_DEFAULT_TIMEOUT;
if (error) { watchdog_init_timeout(wdd, timeout, &pdev->dev);
dev_err(&pdev->dev, "failed to register miscdev\n");
return error;
}
dev_info(&pdev->dev, "TS-72xx Watchdog driver\n"); watchdog_set_drvdata(wdd, priv);
return 0; ret = devm_watchdog_register_device(&pdev->dev, wdd);
} if (ret)
return ret;
dev_info(&pdev->dev, "TS-72xx Watchdog driver\n");
static int ts72xx_wdt_remove(struct platform_device *pdev)
{
misc_deregister(&ts72xx_wdt_miscdev);
return 0; return 0;
} }
static struct platform_driver ts72xx_wdt_driver = { static struct platform_driver ts72xx_wdt_driver = {
.probe = ts72xx_wdt_probe, .probe = ts72xx_wdt_probe,
.remove = ts72xx_wdt_remove,
.driver = { .driver = {
.name = "ts72xx-wdt", .name = "ts72xx-wdt",
}, },
......
...@@ -297,7 +297,7 @@ static unsigned int wdt_get_time(struct watchdog_device *wdog) ...@@ -297,7 +297,7 @@ static unsigned int wdt_get_time(struct watchdog_device *wdog)
* Kernel Interfaces * Kernel Interfaces
*/ */
static struct watchdog_info wdt_info = { static const struct watchdog_info wdt_info = {
.options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE, .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE,
.identity = "W83627HF Watchdog", .identity = "W83627HF Watchdog",
}; };
......
...@@ -987,6 +987,11 @@ static void watchdog_cdev_unregister(struct watchdog_device *wdd) ...@@ -987,6 +987,11 @@ static void watchdog_cdev_unregister(struct watchdog_device *wdd)
wdd->wd_data = NULL; wdd->wd_data = NULL;
mutex_unlock(&wd_data->lock); mutex_unlock(&wd_data->lock);
if (watchdog_active(wdd) &&
test_bit(WDOG_STOP_ON_UNREGISTER, &wdd->status)) {
watchdog_stop(wdd);
}
cancel_delayed_work_sync(&wd_data->work); cancel_delayed_work_sync(&wd_data->work);
kref_put(&wd_data->kref, watchdog_core_data_release); kref_put(&wd_data->kref, watchdog_core_data_release);
......
...@@ -194,7 +194,7 @@ static int wm831x_wdt_probe(struct platform_device *pdev) ...@@ -194,7 +194,7 @@ static int wm831x_wdt_probe(struct platform_device *pdev)
if (ret < 0) { if (ret < 0) {
dev_err(wm831x->dev, "Failed to read watchdog status: %d\n", dev_err(wm831x->dev, "Failed to read watchdog status: %d\n",
ret); ret);
goto err; return ret;
} }
reg = ret; reg = ret;
...@@ -203,10 +203,8 @@ static int wm831x_wdt_probe(struct platform_device *pdev) ...@@ -203,10 +203,8 @@ static int wm831x_wdt_probe(struct platform_device *pdev)
driver_data = devm_kzalloc(&pdev->dev, sizeof(*driver_data), driver_data = devm_kzalloc(&pdev->dev, sizeof(*driver_data),
GFP_KERNEL); GFP_KERNEL);
if (!driver_data) { if (!driver_data)
ret = -ENOMEM; return -ENOMEM;
goto err;
}
mutex_init(&driver_data->lock); mutex_init(&driver_data->lock);
driver_data->wm831x = wm831x; driver_data->wm831x = wm831x;
...@@ -253,7 +251,7 @@ static int wm831x_wdt_probe(struct platform_device *pdev) ...@@ -253,7 +251,7 @@ static int wm831x_wdt_probe(struct platform_device *pdev)
dev_err(wm831x->dev, dev_err(wm831x->dev,
"Failed to request update GPIO: %d\n", "Failed to request update GPIO: %d\n",
ret); ret);
goto err; return ret;
} }
driver_data->update_gpio = pdata->update_gpio; driver_data->update_gpio = pdata->update_gpio;
...@@ -269,37 +267,22 @@ static int wm831x_wdt_probe(struct platform_device *pdev) ...@@ -269,37 +267,22 @@ static int wm831x_wdt_probe(struct platform_device *pdev)
} else { } else {
dev_err(wm831x->dev, dev_err(wm831x->dev,
"Failed to unlock security key: %d\n", ret); "Failed to unlock security key: %d\n", ret);
goto err; return ret;
} }
} }
ret = watchdog_register_device(&driver_data->wdt); ret = devm_watchdog_register_device(&pdev->dev, &driver_data->wdt);
if (ret != 0) { if (ret != 0) {
dev_err(wm831x->dev, "watchdog_register_device() failed: %d\n", dev_err(wm831x->dev, "watchdog_register_device() failed: %d\n",
ret); ret);
goto err; return ret;
} }
platform_set_drvdata(pdev, driver_data);
return 0;
err:
return ret;
}
static int wm831x_wdt_remove(struct platform_device *pdev)
{
struct wm831x_wdt_drvdata *driver_data = platform_get_drvdata(pdev);
watchdog_unregister_device(&driver_data->wdt);
return 0; return 0;
} }
static struct platform_driver wm831x_wdt_driver = { static struct platform_driver wm831x_wdt_driver = {
.probe = wm831x_wdt_probe, .probe = wm831x_wdt_probe,
.remove = wm831x_wdt_remove,
.driver = { .driver = {
.name = "wm831x-watchdog", .name = "wm831x-watchdog",
}, },
......
/*
* watchdog driver for ZTE's zx2967 family
*
* Copyright (C) 2017 ZTE Ltd.
*
* Author: Baoyou Xie <baoyou.xie@linaro.org>
*
* License terms: GNU General Public License (GPL) version 2
*/
#include <linux/clk.h>
#include <linux/io.h>
#include <linux/mfd/syscon.h>
#include <linux/module.h>
#include <linux/of_address.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/reset.h>
#include <linux/watchdog.h>
#define ZX2967_WDT_CFG_REG 0x4
#define ZX2967_WDT_LOAD_REG 0x8
#define ZX2967_WDT_REFRESH_REG 0x18
#define ZX2967_WDT_START_REG 0x1c
#define ZX2967_WDT_REFRESH_MASK GENMASK(5, 0)
#define ZX2967_WDT_CFG_DIV(n) ((((n) & 0xff) - 1) << 8)
#define ZX2967_WDT_START_EN 0x1
/*
* Hardware magic number.
* When watchdog reg is written, the lowest 16 bits are valid, but
* the highest 16 bits should be always this number.
*/
#define ZX2967_WDT_WRITEKEY (0x1234 << 16)
#define ZX2967_WDT_VAL_MASK GENMASK(15, 0)
#define ZX2967_WDT_DIV_DEFAULT 16
#define ZX2967_WDT_DEFAULT_TIMEOUT 32
#define ZX2967_WDT_MIN_TIMEOUT 1
#define ZX2967_WDT_MAX_TIMEOUT 524
#define ZX2967_WDT_MAX_COUNT 0xffff
#define ZX2967_WDT_CLK_FREQ 0x8000
#define ZX2967_WDT_FLAG_REBOOT_MON BIT(0)
struct zx2967_wdt {
struct watchdog_device wdt_device;
void __iomem *reg_base;
struct clk *clock;
};
static inline u32 zx2967_wdt_readl(struct zx2967_wdt *wdt, u16 reg)
{
return readl_relaxed(wdt->reg_base + reg);
}
static inline void zx2967_wdt_writel(struct zx2967_wdt *wdt, u16 reg, u32 val)
{
writel_relaxed(val | ZX2967_WDT_WRITEKEY, wdt->reg_base + reg);
}
static void zx2967_wdt_refresh(struct zx2967_wdt *wdt)
{
u32 val;
val = zx2967_wdt_readl(wdt, ZX2967_WDT_REFRESH_REG);
/*
* Bit 4-5, 1 and 2: refresh config info
* Bit 2-3, 1 and 2: refresh counter
* Bit 0-1, 1 and 2: refresh int-value
* we shift each group value between 1 and 2 to refresh all data.
*/
val ^= ZX2967_WDT_REFRESH_MASK;
zx2967_wdt_writel(wdt, ZX2967_WDT_REFRESH_REG,
val & ZX2967_WDT_VAL_MASK);
}
static int
zx2967_wdt_set_timeout(struct watchdog_device *wdd, unsigned int timeout)
{
struct zx2967_wdt *wdt = watchdog_get_drvdata(wdd);
unsigned int divisor = ZX2967_WDT_DIV_DEFAULT;
u32 count;
count = timeout * ZX2967_WDT_CLK_FREQ;
if (count > divisor * ZX2967_WDT_MAX_COUNT)
divisor = DIV_ROUND_UP(count, ZX2967_WDT_MAX_COUNT);
count = DIV_ROUND_UP(count, divisor);
zx2967_wdt_writel(wdt, ZX2967_WDT_CFG_REG,
ZX2967_WDT_CFG_DIV(divisor) & ZX2967_WDT_VAL_MASK);
zx2967_wdt_writel(wdt, ZX2967_WDT_LOAD_REG,
count & ZX2967_WDT_VAL_MASK);
zx2967_wdt_refresh(wdt);
wdd->timeout = (count * divisor) / ZX2967_WDT_CLK_FREQ;
return 0;
}
static void __zx2967_wdt_start(struct zx2967_wdt *wdt)
{
u32 val;
val = zx2967_wdt_readl(wdt, ZX2967_WDT_START_REG);
val |= ZX2967_WDT_START_EN;
zx2967_wdt_writel(wdt, ZX2967_WDT_START_REG,
val & ZX2967_WDT_VAL_MASK);
}
static void __zx2967_wdt_stop(struct zx2967_wdt *wdt)
{
u32 val;
val = zx2967_wdt_readl(wdt, ZX2967_WDT_START_REG);
val &= ~ZX2967_WDT_START_EN;
zx2967_wdt_writel(wdt, ZX2967_WDT_START_REG,
val & ZX2967_WDT_VAL_MASK);
}
static int zx2967_wdt_start(struct watchdog_device *wdd)
{
struct zx2967_wdt *wdt = watchdog_get_drvdata(wdd);
zx2967_wdt_set_timeout(wdd, wdd->timeout);
__zx2967_wdt_start(wdt);
return 0;
}
static int zx2967_wdt_stop(struct watchdog_device *wdd)
{
struct zx2967_wdt *wdt = watchdog_get_drvdata(wdd);
__zx2967_wdt_stop(wdt);
return 0;
}
static int zx2967_wdt_keepalive(struct watchdog_device *wdd)
{
struct zx2967_wdt *wdt = watchdog_get_drvdata(wdd);
zx2967_wdt_refresh(wdt);
return 0;
}
#define ZX2967_WDT_OPTIONS \
(WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | WDIOF_MAGICCLOSE)
static const struct watchdog_info zx2967_wdt_ident = {
.options = ZX2967_WDT_OPTIONS,
.identity = "zx2967 watchdog",
};
static struct watchdog_ops zx2967_wdt_ops = {
.owner = THIS_MODULE,
.start = zx2967_wdt_start,
.stop = zx2967_wdt_stop,
.ping = zx2967_wdt_keepalive,
.set_timeout = zx2967_wdt_set_timeout,
};
static void zx2967_wdt_reset_sysctrl(struct device *dev)
{
int ret;
void __iomem *regmap;
unsigned int offset, mask, config;
struct of_phandle_args out_args;
ret = of_parse_phandle_with_fixed_args(dev->of_node,
"zte,wdt-reset-sysctrl", 3, 0, &out_args);
if (ret)
return;
offset = out_args.args[0];
config = out_args.args[1];
mask = out_args.args[2];
regmap = syscon_node_to_regmap(out_args.np);
if (IS_ERR(regmap)) {
of_node_put(out_args.np);
return;
}
regmap_update_bits(regmap, offset, mask, config);
of_node_put(out_args.np);
}
static int zx2967_wdt_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct zx2967_wdt *wdt;
struct resource *base;
int ret;
struct reset_control *rstc;
wdt = devm_kzalloc(dev, sizeof(*wdt), GFP_KERNEL);
if (!wdt)
return -ENOMEM;
platform_set_drvdata(pdev, wdt);
wdt->wdt_device.info = &zx2967_wdt_ident;
wdt->wdt_device.ops = &zx2967_wdt_ops;
wdt->wdt_device.timeout = ZX2967_WDT_DEFAULT_TIMEOUT;
wdt->wdt_device.max_timeout = ZX2967_WDT_MAX_TIMEOUT;
wdt->wdt_device.min_timeout = ZX2967_WDT_MIN_TIMEOUT;
wdt->wdt_device.parent = &pdev->dev;
base = platform_get_resource(pdev, IORESOURCE_MEM, 0);
wdt->reg_base = devm_ioremap_resource(dev, base);
if (IS_ERR(wdt->reg_base)) {
dev_err(dev, "ioremap failed\n");
return PTR_ERR(wdt->reg_base);
}
zx2967_wdt_reset_sysctrl(dev);
wdt->clock = devm_clk_get(dev, NULL);
if (IS_ERR(wdt->clock)) {
dev_err(dev, "failed to find watchdog clock source\n");
return PTR_ERR(wdt->clock);
}
ret = clk_prepare_enable(wdt->clock);
if (ret < 0) {
dev_err(dev, "failed to enable clock\n");
return ret;
}
clk_set_rate(wdt->clock, ZX2967_WDT_CLK_FREQ);
rstc = devm_reset_control_get(dev, NULL);
if (IS_ERR(rstc)) {
dev_err(dev, "failed to get rstc");
ret = PTR_ERR(rstc);
goto err;
}
reset_control_assert(rstc);
reset_control_deassert(rstc);
watchdog_set_drvdata(&wdt->wdt_device, wdt);
watchdog_init_timeout(&wdt->wdt_device,
ZX2967_WDT_DEFAULT_TIMEOUT, dev);
watchdog_set_nowayout(&wdt->wdt_device, WATCHDOG_NOWAYOUT);
ret = watchdog_register_device(&wdt->wdt_device);
if (ret)
goto err;
dev_info(dev, "watchdog enabled (timeout=%d sec, nowayout=%d)",
wdt->wdt_device.timeout, WATCHDOG_NOWAYOUT);
return 0;
err:
clk_disable_unprepare(wdt->clock);
return ret;
}
static int zx2967_wdt_remove(struct platform_device *pdev)
{
struct zx2967_wdt *wdt = platform_get_drvdata(pdev);
watchdog_unregister_device(&wdt->wdt_device);
clk_disable_unprepare(wdt->clock);
return 0;
}
static const struct of_device_id zx2967_wdt_match[] = {
{ .compatible = "zte,zx296718-wdt", },
{}
};
MODULE_DEVICE_TABLE(of, zx2967_wdt_match);
static struct platform_driver zx2967_wdt_driver = {
.probe = zx2967_wdt_probe,
.remove = zx2967_wdt_remove,
.driver = {
.name = "zx2967-wdt",
.of_match_table = of_match_ptr(zx2967_wdt_match),
},
};
module_platform_driver(zx2967_wdt_driver);
MODULE_AUTHOR("Baoyou Xie <baoyou.xie@linaro.org>");
MODULE_DESCRIPTION("ZTE zx2967 Watchdog Device Driver");
MODULE_LICENSE("GPL v2");
...@@ -117,6 +117,7 @@ struct watchdog_device { ...@@ -117,6 +117,7 @@ struct watchdog_device {
#define WDOG_NO_WAY_OUT 1 /* Is 'nowayout' feature set ? */ #define WDOG_NO_WAY_OUT 1 /* Is 'nowayout' feature set ? */
#define WDOG_STOP_ON_REBOOT 2 /* Should be stopped on reboot */ #define WDOG_STOP_ON_REBOOT 2 /* Should be stopped on reboot */
#define WDOG_HW_RUNNING 3 /* True if HW watchdog running */ #define WDOG_HW_RUNNING 3 /* True if HW watchdog running */
#define WDOG_STOP_ON_UNREGISTER 4 /* Should be stopped on unregister */
struct list_head deferred; struct list_head deferred;
}; };
...@@ -151,6 +152,12 @@ static inline void watchdog_stop_on_reboot(struct watchdog_device *wdd) ...@@ -151,6 +152,12 @@ static inline void watchdog_stop_on_reboot(struct watchdog_device *wdd)
set_bit(WDOG_STOP_ON_REBOOT, &wdd->status); set_bit(WDOG_STOP_ON_REBOOT, &wdd->status);
} }
/* Use the following function to stop the watchdog when unregistering it */
static inline void watchdog_stop_on_unregister(struct watchdog_device *wdd)
{
set_bit(WDOG_STOP_ON_UNREGISTER, &wdd->status);
}
/* Use the following function to check if a timeout value is invalid */ /* Use the following function to check if a timeout value is invalid */
static inline bool watchdog_timeout_invalid(struct watchdog_device *wdd, unsigned int t) static inline bool watchdog_timeout_invalid(struct watchdog_device *wdd, unsigned int t)
{ {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment