// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2020 Cortina-Access * */ #include <common.h> #include <dm.h> #include <hang.h> #include <asm/io.h> #include <wdt.h> #include <linux/bitops.h> #define CA_WDT_CTRL 0x00 #define CA_WDT_PS 0x04 #define CA_WDT_DIV 0x08 #define CA_WDT_LD 0x0C #define CA_WDT_LOADE 0x10 #define CA_WDT_CNT 0x14 #define CA_WDT_IE 0x18 #define CA_WDT_INT 0x1C #define CA_WDT_STAT 0x20 /* CA_WDT_CTRL */ #define CTL_WDT_EN BIT(0) #define CTL_WDT_RSTEN BIT(1) #define CTL_WDT_CLK_SEL BIT(2) /* CA_WDT_LOADE */ #define WDT_UPD BIT(0) #define WDT_UPD_PS BIT(1) /* Global config */ #define WDT_RESET_SUB BIT(4) #define WDT_RESET_ALL_BLOCK BIT(6) #define WDT_RESET_REMAP BIT(7) #define WDT_EXT_RESET BIT(8) #define WDT_RESET_DEFAULT (WDT_EXT_RESET | WDT_RESET_REMAP | \ WDT_RESET_ALL_BLOCK | WDT_RESET_SUB) struct ca_wdt_priv { void __iomem *base; void __iomem *global_config; }; static void cortina_wdt_set_timeout(struct udevice *dev, u64 timeout_ms) { struct ca_wdt_priv *priv = dev_get_priv(dev); /* Prescale using millisecond unit */ writel(CORTINA_PER_IO_FREQ / 1000, priv->base + CA_WDT_PS); /* Millisecond */ writel(1, priv->base + CA_WDT_DIV); writel(timeout_ms, priv->base + CA_WDT_LD); writel(WDT_UPD | WDT_UPD_PS, priv->base + CA_WDT_LOADE); } static int cortina_wdt_start(struct udevice *dev, u64 timeout, ulong flags) { struct ca_wdt_priv *priv = dev_get_priv(dev); cortina_wdt_set_timeout(dev, timeout); /* WDT Reset option */ setbits_32(priv->global_config, WDT_RESET_DEFAULT); /* Enable WDT */ setbits_32(priv->base, CTL_WDT_EN | CTL_WDT_RSTEN | CTL_WDT_CLK_SEL); return 0; } static int cortina_wdt_stop(struct udevice *dev) { struct ca_wdt_priv *priv = dev_get_priv(dev); /* Disable WDT */ writel(0, priv->base); return 0; } static int cortina_wdt_reset(struct udevice *dev) { struct ca_wdt_priv *priv = dev_get_priv(dev); /* Reload WDT counter */ writel(WDT_UPD, priv->base + CA_WDT_LOADE); return 0; } static int cortina_wdt_expire_now(struct udevice *dev, ulong flags) { /* Set 1ms timeout to reset system */ cortina_wdt_set_timeout(dev, 1); hang(); return 0; } static int cortina_wdt_probe(struct udevice *dev) { struct ca_wdt_priv *priv = dev_get_priv(dev); priv->base = dev_remap_addr_index(dev, 0); if (!priv->base) return -ENOENT; priv->global_config = dev_remap_addr_index(dev, 1); if (!priv->global_config) return -ENOENT; /* Stop WDT */ cortina_wdt_stop(dev); return 0; } static const struct wdt_ops cortina_wdt_ops = { .start = cortina_wdt_start, .reset = cortina_wdt_reset, .stop = cortina_wdt_stop, .expire_now = cortina_wdt_expire_now, }; static const struct udevice_id cortina_wdt_ids[] = { {.compatible = "cortina,ca-wdt"}, {} }; U_BOOT_DRIVER(cortina_wdt) = { .name = "cortina_wdt", .id = UCLASS_WDT, .probe = cortina_wdt_probe, .of_match = cortina_wdt_ids, .ops = &cortina_wdt_ops, };