Commit 1f32cab0 authored by Anna-Maria Behnsen's avatar Anna-Maria Behnsen Committed by Thomas Gleixner

timers: Use only bucket expiry for base->next_expiry value

The bucket expiry time is the effective expriy time of timers and is
greater than or equal to the requested timer expiry time. This is due
to the guarantee that timers never expire early and the reduced expiry
granularity in the secondary wheel levels.

When a timer is enqueued, trigger_dyntick_cpu() checks whether the
timer is the new first timer. This check compares next_expiry with
the requested timer expiry value and not with the effective expiry
value of the bucket into which the timer was queued.

Storing the requested timer expiry value in base->next_expiry can lead
to base->clk going backwards if the requested timer expiry value is
smaller than base->clk. Commit 30c66fc3 ("timer: Prevent base->clk
from moving backward") worked around this by preventing the store when
timer->expiry is before base->clk, but did not fix the underlying
problem.

Use the expiry value of the bucket into which the timer is queued to
do the new first timer check. This fixes the base->clk going backward
problem.

The workaround of commit 30c66fc3 ("timer: Prevent base->clk from
moving backward") in trigger_dyntick_cpu() is not longer necessary as the
timers bucket expiry is guaranteed to be greater than or equal base->clk.
Signed-off-by: default avatarAnna-Maria Behnsen <anna-maria@linutronix.de>
Signed-off-by: default avatarFrederic Weisbecker <frederic@kernel.org>
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Link: https://lkml.kernel.org/r/20200717140551.29076-4-frederic@kernel.org
parent 3d2e83a2
...@@ -487,35 +487,39 @@ static inline void timer_set_idx(struct timer_list *timer, unsigned int idx) ...@@ -487,35 +487,39 @@ static inline void timer_set_idx(struct timer_list *timer, unsigned int idx)
* Helper function to calculate the array index for a given expiry * Helper function to calculate the array index for a given expiry
* time. * time.
*/ */
static inline unsigned calc_index(unsigned long expires, unsigned lvl) static inline unsigned calc_index(unsigned long expires, unsigned lvl,
unsigned long *bucket_expiry)
{ {
expires = (expires + LVL_GRAN(lvl)) >> LVL_SHIFT(lvl); expires = (expires + LVL_GRAN(lvl)) >> LVL_SHIFT(lvl);
*bucket_expiry = expires << LVL_SHIFT(lvl);
return LVL_OFFS(lvl) + (expires & LVL_MASK); return LVL_OFFS(lvl) + (expires & LVL_MASK);
} }
static int calc_wheel_index(unsigned long expires, unsigned long clk) static int calc_wheel_index(unsigned long expires, unsigned long clk,
unsigned long *bucket_expiry)
{ {
unsigned long delta = expires - clk; unsigned long delta = expires - clk;
unsigned int idx; unsigned int idx;
if (delta < LVL_START(1)) { if (delta < LVL_START(1)) {
idx = calc_index(expires, 0); idx = calc_index(expires, 0, bucket_expiry);
} else if (delta < LVL_START(2)) { } else if (delta < LVL_START(2)) {
idx = calc_index(expires, 1); idx = calc_index(expires, 1, bucket_expiry);
} else if (delta < LVL_START(3)) { } else if (delta < LVL_START(3)) {
idx = calc_index(expires, 2); idx = calc_index(expires, 2, bucket_expiry);
} else if (delta < LVL_START(4)) { } else if (delta < LVL_START(4)) {
idx = calc_index(expires, 3); idx = calc_index(expires, 3, bucket_expiry);
} else if (delta < LVL_START(5)) { } else if (delta < LVL_START(5)) {
idx = calc_index(expires, 4); idx = calc_index(expires, 4, bucket_expiry);
} else if (delta < LVL_START(6)) { } else if (delta < LVL_START(6)) {
idx = calc_index(expires, 5); idx = calc_index(expires, 5, bucket_expiry);
} else if (delta < LVL_START(7)) { } else if (delta < LVL_START(7)) {
idx = calc_index(expires, 6); idx = calc_index(expires, 6, bucket_expiry);
} else if (LVL_DEPTH > 8 && delta < LVL_START(8)) { } else if (LVL_DEPTH > 8 && delta < LVL_START(8)) {
idx = calc_index(expires, 7); idx = calc_index(expires, 7, bucket_expiry);
} else if ((long) delta < 0) { } else if ((long) delta < 0) {
idx = clk & LVL_MASK; idx = clk & LVL_MASK;
*bucket_expiry = clk;
} else { } else {
/* /*
* Force expire obscene large timeouts to expire at the * Force expire obscene large timeouts to expire at the
...@@ -524,7 +528,7 @@ static int calc_wheel_index(unsigned long expires, unsigned long clk) ...@@ -524,7 +528,7 @@ static int calc_wheel_index(unsigned long expires, unsigned long clk)
if (delta >= WHEEL_TIMEOUT_CUTOFF) if (delta >= WHEEL_TIMEOUT_CUTOFF)
expires = clk + WHEEL_TIMEOUT_MAX; expires = clk + WHEEL_TIMEOUT_MAX;
idx = calc_index(expires, LVL_DEPTH - 1); idx = calc_index(expires, LVL_DEPTH - 1, bucket_expiry);
} }
return idx; return idx;
} }
...@@ -544,16 +548,18 @@ static void enqueue_timer(struct timer_base *base, struct timer_list *timer, ...@@ -544,16 +548,18 @@ static void enqueue_timer(struct timer_base *base, struct timer_list *timer,
} }
static void static void
__internal_add_timer(struct timer_base *base, struct timer_list *timer) __internal_add_timer(struct timer_base *base, struct timer_list *timer,
unsigned long *bucket_expiry)
{ {
unsigned int idx; unsigned int idx;
idx = calc_wheel_index(timer->expires, base->clk); idx = calc_wheel_index(timer->expires, base->clk, bucket_expiry);
enqueue_timer(base, timer, idx); enqueue_timer(base, timer, idx);
} }
static void static void
trigger_dyntick_cpu(struct timer_base *base, struct timer_list *timer) trigger_dyntick_cpu(struct timer_base *base, struct timer_list *timer,
unsigned long bucket_expiry)
{ {
if (!is_timers_nohz_active()) if (!is_timers_nohz_active())
return; return;
...@@ -576,31 +582,29 @@ trigger_dyntick_cpu(struct timer_base *base, struct timer_list *timer) ...@@ -576,31 +582,29 @@ trigger_dyntick_cpu(struct timer_base *base, struct timer_list *timer)
if (!base->is_idle) if (!base->is_idle)
return; return;
/* Check whether this is the new first expiring timer: */ /*
if (time_after_eq(timer->expires, base->next_expiry)) * Check whether this is the new first expiring timer. The
* effective expiry time of the timer is required here
* (bucket_expiry) instead of timer->expires.
*/
if (time_after_eq(bucket_expiry, base->next_expiry))
return; return;
/* /*
* Set the next expiry time and kick the CPU so it can reevaluate the * Set the next expiry time and kick the CPU so it can reevaluate the
* wheel: * wheel:
*/ */
if (time_before(timer->expires, base->clk)) { base->next_expiry = bucket_expiry;
/*
* Prevent from forward_timer_base() moving the base->clk
* backward
*/
base->next_expiry = base->clk;
} else {
base->next_expiry = timer->expires;
}
wake_up_nohz_cpu(base->cpu); wake_up_nohz_cpu(base->cpu);
} }
static void static void
internal_add_timer(struct timer_base *base, struct timer_list *timer) internal_add_timer(struct timer_base *base, struct timer_list *timer)
{ {
__internal_add_timer(base, timer); unsigned long bucket_expiry;
trigger_dyntick_cpu(base, timer);
__internal_add_timer(base, timer, &bucket_expiry);
trigger_dyntick_cpu(base, timer, bucket_expiry);
} }
#ifdef CONFIG_DEBUG_OBJECTS_TIMERS #ifdef CONFIG_DEBUG_OBJECTS_TIMERS
...@@ -959,9 +963,9 @@ static struct timer_base *lock_timer_base(struct timer_list *timer, ...@@ -959,9 +963,9 @@ static struct timer_base *lock_timer_base(struct timer_list *timer,
static inline int static inline int
__mod_timer(struct timer_list *timer, unsigned long expires, unsigned int options) __mod_timer(struct timer_list *timer, unsigned long expires, unsigned int options)
{ {
unsigned long clk = 0, flags, bucket_expiry;
struct timer_base *base, *new_base; struct timer_base *base, *new_base;
unsigned int idx = UINT_MAX; unsigned int idx = UINT_MAX;
unsigned long clk = 0, flags;
int ret = 0; int ret = 0;
BUG_ON(!timer->function); BUG_ON(!timer->function);
...@@ -1000,7 +1004,7 @@ __mod_timer(struct timer_list *timer, unsigned long expires, unsigned int option ...@@ -1000,7 +1004,7 @@ __mod_timer(struct timer_list *timer, unsigned long expires, unsigned int option
} }
clk = base->clk; clk = base->clk;
idx = calc_wheel_index(expires, clk); idx = calc_wheel_index(expires, clk, &bucket_expiry);
/* /*
* Retrieve and compare the array index of the pending * Retrieve and compare the array index of the pending
...@@ -1059,7 +1063,7 @@ __mod_timer(struct timer_list *timer, unsigned long expires, unsigned int option ...@@ -1059,7 +1063,7 @@ __mod_timer(struct timer_list *timer, unsigned long expires, unsigned int option
*/ */
if (idx != UINT_MAX && clk == base->clk) { if (idx != UINT_MAX && clk == base->clk) {
enqueue_timer(base, timer, idx); enqueue_timer(base, timer, idx);
trigger_dyntick_cpu(base, timer); trigger_dyntick_cpu(base, timer, bucket_expiry);
} else { } else {
internal_add_timer(base, timer); internal_add_timer(base, timer);
} }
......
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