[PATCH 1/1] sparc64: Fix tick/stick comparator equal-compare hazard

From: Tony Rodriguez

Date: Mon May 18 2026 - 19:59:04 EST


On SPARC64 the timer interrupt is generated only when the counter
equals the comparator (tick == tick_cmpr). There is a latency between
writing the comparator register and the comparator becoming active.
If the counter reaches the comparator value before the comparator is
armed, the equality moment can be missed and the hardware may not
generate the interrupt.

Previously the helper treated equality as success:

return ((long)(new_tick - (orig_tick + adj))) > 0L;

Here `exp` is the expected comparator value (orig_tick + adj). If
new_tick == exp the code considered the event scheduled, but the
hardware may already have missed the equality window. Treat equality
as failure so the caller will retry with a later expiration:

return ((long)(new_tick - (orig_tick + adj))) >= 0L;

This prevents missed timer interrupts and stalled hrtimers on affected
SPARC64 systems.

Signed-off-by: Tony Rodriguez <unixpro1970@xxxxxxxxx>
---
arch/sparc/kernel/time_64.c | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/arch/sparc/kernel/time_64.c b/arch/sparc/kernel/time_64.c
index 87b267043ccd..783b60e547c4 100644
--- a/arch/sparc/kernel/time_64.c
+++ b/arch/sparc/kernel/time_64.c
@@ -146,7 +146,7 @@ static int tick_add_compare(unsigned long adj)
: "=r" (new_tick));
new_tick &= ~TICKCMP_IRQ_BIT;

- return ((long)(new_tick - (orig_tick+adj))) > 0L;
+ return ((long)(new_tick - (orig_tick+adj))) >= 0L;
}

static unsigned long tick_add_tick(unsigned long adj)
@@ -277,7 +277,7 @@ static int stick_add_compare(unsigned long adj)
: "=r" (new_tick));
new_tick &= ~TICKCMP_IRQ_BIT;

- return ((long)(new_tick - (orig_tick+adj))) > 0L;
+ return ((long)(new_tick - (orig_tick+adj))) >= 0L;
}

static unsigned long stick_get_frequency(void)
@@ -411,7 +411,7 @@ static int hbtick_add_compare(unsigned long adj)

val2 = __hbird_read_stick() & ~TICKCMP_IRQ_BIT;

- return ((long)(val2 - val)) > 0L;
+ return ((long)(val2 - val)) >= 0L;
}

static unsigned long hbtick_get_frequency(void)
--
2.53.0