2
0
mirror of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-09-04 20:19:47 +08:00

watchdog: sbsa: Adjust keepalive timeout to avoid MediaTek WS0 race condition

The MediaTek implementation of the sbsa_gwdt watchdog has a race
condition where a write to SBSA_GWDT_WRR is ignored if it occurs while
the hardware is processing a timeout refresh that asserts WS0.

Detect this based on the hardware implementer and adjust
wdd->min_hw_heartbeat_ms to avoid the race by forcing the keepalive ping
to be one second later.

Signed-off-by: Aaron Plattner <aplattner@nvidia.com>
Acked-by: Timur Tabi <ttabi@nvidia.com>
Reviewed-by: Guenter Roeck <linux@roeck-us.net>
Link: https://lore.kernel.org/r/20250721230640.2244915-1-aplattner@nvidia.com
Signed-off-by: Guenter Roeck <linux@roeck-us.net>
Signed-off-by: Wim Van Sebroeck <wim@linux-watchdog.org>
This commit is contained in:
Aaron Plattner 2025-07-21 16:06:39 -07:00 committed by Wim Van Sebroeck
parent ac3dbb91e0
commit 48defdf6b0

View File

@ -75,11 +75,17 @@
#define SBSA_GWDT_VERSION_MASK 0xF
#define SBSA_GWDT_VERSION_SHIFT 16
#define SBSA_GWDT_IMPL_MASK 0x7FF
#define SBSA_GWDT_IMPL_SHIFT 0
#define SBSA_GWDT_IMPL_MEDIATEK 0x426
/**
* struct sbsa_gwdt - Internal representation of the SBSA GWDT
* @wdd: kernel watchdog_device structure
* @clk: store the System Counter clock frequency, in Hz.
* @version: store the architecture version
* @need_ws0_race_workaround:
* indicate whether to adjust wdd->timeout to avoid a race with WS0
* @refresh_base: Virtual address of the watchdog refresh frame
* @control_base: Virtual address of the watchdog control frame
*/
@ -87,6 +93,7 @@ struct sbsa_gwdt {
struct watchdog_device wdd;
u32 clk;
int version;
bool need_ws0_race_workaround;
void __iomem *refresh_base;
void __iomem *control_base;
};
@ -161,6 +168,31 @@ static int sbsa_gwdt_set_timeout(struct watchdog_device *wdd,
*/
sbsa_gwdt_reg_write(((u64)gwdt->clk / 2) * timeout, gwdt);
/*
* Some watchdog hardware has a race condition where it will ignore
* sbsa_gwdt_keepalive() if it is called at the exact moment that a
* timeout occurs and WS0 is being asserted. Unfortunately, the default
* behavior of the watchdog core is very likely to trigger this race
* when action=0 because it programs WOR to be half of the desired
* timeout, and watchdog_next_keepalive() chooses the exact same time to
* send keepalive pings.
*
* This triggers a race where sbsa_gwdt_keepalive() can be called right
* as WS0 is being asserted, and affected hardware will ignore that
* write and continue to assert WS0. After another (timeout / 2)
* seconds, the same race happens again. If the driver wins then the
* explicit refresh will reset WS0 to false but if the hardware wins,
* then WS1 is asserted and the system resets.
*
* Avoid the problem by scheduling keepalive heartbeats one second later
* than the WOR timeout.
*
* This workaround might not be needed in a future revision of the
* hardware.
*/
if (gwdt->need_ws0_race_workaround)
wdd->min_hw_heartbeat_ms = timeout * 500 + 1000;
return 0;
}
@ -202,12 +234,15 @@ static int sbsa_gwdt_keepalive(struct watchdog_device *wdd)
static void sbsa_gwdt_get_version(struct watchdog_device *wdd)
{
struct sbsa_gwdt *gwdt = watchdog_get_drvdata(wdd);
int ver;
int iidr, ver, impl;
ver = readl(gwdt->control_base + SBSA_GWDT_W_IIDR);
ver = (ver >> SBSA_GWDT_VERSION_SHIFT) & SBSA_GWDT_VERSION_MASK;
iidr = readl(gwdt->control_base + SBSA_GWDT_W_IIDR);
ver = (iidr >> SBSA_GWDT_VERSION_SHIFT) & SBSA_GWDT_VERSION_MASK;
impl = (iidr >> SBSA_GWDT_IMPL_SHIFT) & SBSA_GWDT_IMPL_MASK;
gwdt->version = ver;
gwdt->need_ws0_race_workaround =
!action && (impl == SBSA_GWDT_IMPL_MEDIATEK);
}
static int sbsa_gwdt_start(struct watchdog_device *wdd)
@ -299,6 +334,15 @@ static int sbsa_gwdt_probe(struct platform_device *pdev)
else
wdd->max_hw_heartbeat_ms = GENMASK_ULL(47, 0) / gwdt->clk * 1000;
if (gwdt->need_ws0_race_workaround) {
/*
* A timeout of 3 seconds means that WOR will be set to 1.5
* seconds and the heartbeat will be scheduled every 2.5
* seconds.
*/
wdd->min_timeout = 3;
}
status = readl(cf_base + SBSA_GWDT_WCS);
if (status & SBSA_GWDT_WCS_WS1) {
dev_warn(dev, "System reset by WDT.\n");