|  | // SPDX-License-Identifier: GPL-2.0-only | 
|  | /* | 
|  | * drivers/char/watchdog/ixp4xx_wdt.c | 
|  | * | 
|  | * Watchdog driver for Intel IXP4xx network processors | 
|  | * | 
|  | * Author: Deepak Saxena <dsaxena@plexity.net> | 
|  | * Author: Linus Walleij <linus.walleij@linaro.org> | 
|  | * | 
|  | * Copyright 2004 (c) MontaVista, Software, Inc. | 
|  | * Based on sa1100 driver, Copyright (C) 2000 Oleg Drokin <green@crimea.edu> | 
|  | */ | 
|  |  | 
|  | #include <linux/module.h> | 
|  | #include <linux/types.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/watchdog.h> | 
|  | #include <linux/bits.h> | 
|  | #include <linux/platform_device.h> | 
|  | #include <linux/clk.h> | 
|  | #include <linux/soc/ixp4xx/cpu.h> | 
|  |  | 
|  | struct ixp4xx_wdt { | 
|  | struct watchdog_device wdd; | 
|  | void __iomem *base; | 
|  | unsigned long rate; | 
|  | }; | 
|  |  | 
|  | /* Fallback if we do not have a clock for this */ | 
|  | #define IXP4XX_TIMER_FREQ	66666000 | 
|  |  | 
|  | /* Registers after the timer registers */ | 
|  | #define IXP4XX_OSWT_OFFSET	0x14  /* Watchdog Timer */ | 
|  | #define IXP4XX_OSWE_OFFSET	0x18  /* Watchdog Enable */ | 
|  | #define IXP4XX_OSWK_OFFSET	0x1C  /* Watchdog Key */ | 
|  | #define IXP4XX_OSST_OFFSET	0x20  /* Timer Status */ | 
|  |  | 
|  | #define IXP4XX_OSST_TIMER_WDOG_PEND	0x00000008 | 
|  | #define IXP4XX_OSST_TIMER_WARM_RESET	0x00000010 | 
|  | #define IXP4XX_WDT_KEY			0x0000482E | 
|  | #define IXP4XX_WDT_RESET_ENABLE		0x00000001 | 
|  | #define IXP4XX_WDT_IRQ_ENABLE		0x00000002 | 
|  | #define IXP4XX_WDT_COUNT_ENABLE		0x00000004 | 
|  |  | 
|  | static inline | 
|  | struct ixp4xx_wdt *to_ixp4xx_wdt(struct watchdog_device *wdd) | 
|  | { | 
|  | return container_of(wdd, struct ixp4xx_wdt, wdd); | 
|  | } | 
|  |  | 
|  | static int ixp4xx_wdt_start(struct watchdog_device *wdd) | 
|  | { | 
|  | struct ixp4xx_wdt *iwdt = to_ixp4xx_wdt(wdd); | 
|  |  | 
|  | __raw_writel(IXP4XX_WDT_KEY, iwdt->base + IXP4XX_OSWK_OFFSET); | 
|  | __raw_writel(0, iwdt->base + IXP4XX_OSWE_OFFSET); | 
|  | __raw_writel(wdd->timeout * iwdt->rate, | 
|  | iwdt->base + IXP4XX_OSWT_OFFSET); | 
|  | __raw_writel(IXP4XX_WDT_COUNT_ENABLE | IXP4XX_WDT_RESET_ENABLE, | 
|  | iwdt->base + IXP4XX_OSWE_OFFSET); | 
|  | __raw_writel(0, iwdt->base + IXP4XX_OSWK_OFFSET); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ixp4xx_wdt_stop(struct watchdog_device *wdd) | 
|  | { | 
|  | struct ixp4xx_wdt *iwdt = to_ixp4xx_wdt(wdd); | 
|  |  | 
|  | __raw_writel(IXP4XX_WDT_KEY, iwdt->base + IXP4XX_OSWK_OFFSET); | 
|  | __raw_writel(0, iwdt->base + IXP4XX_OSWE_OFFSET); | 
|  | __raw_writel(0, iwdt->base + IXP4XX_OSWK_OFFSET); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ixp4xx_wdt_set_timeout(struct watchdog_device *wdd, | 
|  | unsigned int timeout) | 
|  | { | 
|  | wdd->timeout = timeout; | 
|  | if (watchdog_active(wdd)) | 
|  | ixp4xx_wdt_start(wdd); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ixp4xx_wdt_restart(struct watchdog_device *wdd, | 
|  | unsigned long action, void *data) | 
|  | { | 
|  | struct ixp4xx_wdt *iwdt = to_ixp4xx_wdt(wdd); | 
|  |  | 
|  | __raw_writel(IXP4XX_WDT_KEY, iwdt->base + IXP4XX_OSWK_OFFSET); | 
|  | __raw_writel(0, iwdt->base + IXP4XX_OSWT_OFFSET); | 
|  | __raw_writel(IXP4XX_WDT_COUNT_ENABLE | IXP4XX_WDT_RESET_ENABLE, | 
|  | iwdt->base + IXP4XX_OSWE_OFFSET); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct watchdog_ops ixp4xx_wdt_ops = { | 
|  | .start = ixp4xx_wdt_start, | 
|  | .stop = ixp4xx_wdt_stop, | 
|  | .set_timeout = ixp4xx_wdt_set_timeout, | 
|  | .restart = ixp4xx_wdt_restart, | 
|  | .owner = THIS_MODULE, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * The A0 version of the IXP422 had a bug in the watchdog making | 
|  | * is useless, but we still need to use it to restart the system | 
|  | * as it is the only way, so in this special case we register a | 
|  | * "dummy" watchdog that doesn't really work, but will support | 
|  | * the restart operation. | 
|  | */ | 
|  | static int ixp4xx_wdt_dummy(struct watchdog_device *wdd) | 
|  | { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct watchdog_ops ixp4xx_wdt_restart_only_ops = { | 
|  | .start = ixp4xx_wdt_dummy, | 
|  | .stop = ixp4xx_wdt_dummy, | 
|  | .restart = ixp4xx_wdt_restart, | 
|  | .owner = THIS_MODULE, | 
|  | }; | 
|  |  | 
|  | static const struct watchdog_info ixp4xx_wdt_info = { | 
|  | .options = WDIOF_KEEPALIVEPING | 
|  | | WDIOF_MAGICCLOSE | 
|  | | WDIOF_SETTIMEOUT, | 
|  | .identity = KBUILD_MODNAME, | 
|  | }; | 
|  |  | 
|  | static int ixp4xx_wdt_probe(struct platform_device *pdev) | 
|  | { | 
|  | static const struct watchdog_ops *iwdt_ops; | 
|  | struct device *dev = &pdev->dev; | 
|  | struct ixp4xx_wdt *iwdt; | 
|  | struct clk *clk; | 
|  | int ret; | 
|  |  | 
|  | if (!(read_cpuid_id() & 0xf) && !cpu_is_ixp46x()) { | 
|  | dev_info(dev, "Rev. A0 IXP42x CPU detected - only restart supported\n"); | 
|  | iwdt_ops = &ixp4xx_wdt_restart_only_ops; | 
|  | } else { | 
|  | iwdt_ops = &ixp4xx_wdt_ops; | 
|  | } | 
|  |  | 
|  | iwdt = devm_kzalloc(dev, sizeof(*iwdt), GFP_KERNEL); | 
|  | if (!iwdt) | 
|  | return -ENOMEM; | 
|  | iwdt->base = (void __iomem *)dev->platform_data; | 
|  |  | 
|  | /* | 
|  | * Retrieve rate from a fixed clock from the device tree if | 
|  | * the parent has that, else use the default clock rate. | 
|  | */ | 
|  | clk = devm_clk_get_enabled(dev->parent, NULL); | 
|  | if (!IS_ERR(clk)) | 
|  | iwdt->rate = clk_get_rate(clk); | 
|  |  | 
|  | if (!iwdt->rate) | 
|  | iwdt->rate = IXP4XX_TIMER_FREQ; | 
|  |  | 
|  | iwdt->wdd.info = &ixp4xx_wdt_info; | 
|  | iwdt->wdd.ops = iwdt_ops; | 
|  | iwdt->wdd.min_timeout = 1; | 
|  | iwdt->wdd.max_timeout = U32_MAX / iwdt->rate; | 
|  | iwdt->wdd.parent = dev; | 
|  | /* Default to 60 seconds */ | 
|  | iwdt->wdd.timeout = 60U; | 
|  | watchdog_init_timeout(&iwdt->wdd, 0, dev); | 
|  |  | 
|  | if (__raw_readl(iwdt->base + IXP4XX_OSST_OFFSET) & | 
|  | IXP4XX_OSST_TIMER_WARM_RESET) | 
|  | iwdt->wdd.bootstatus = WDIOF_CARDRESET; | 
|  |  | 
|  | ret = devm_watchdog_register_device(dev, &iwdt->wdd); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | dev_info(dev, "IXP4xx watchdog available\n"); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct platform_driver ixp4xx_wdt_driver = { | 
|  | .probe = ixp4xx_wdt_probe, | 
|  | .driver = { | 
|  | .name   = "ixp4xx-watchdog", | 
|  | }, | 
|  | }; | 
|  | module_platform_driver(ixp4xx_wdt_driver); | 
|  |  | 
|  | MODULE_AUTHOR("Deepak Saxena <dsaxena@plexity.net>"); | 
|  | MODULE_DESCRIPTION("IXP4xx Network Processor Watchdog"); | 
|  | MODULE_LICENSE("GPL"); |