Commit cceb6347 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'timers-urgent-2021-08-08' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip

Pull timer fix from Thomas Gleixner:
 "A single timer fix:

   - Prevent a memory ordering issue in the timer expiry code which
     makes it possible to observe falsely that the callback has been
     executed already while that's not the case, which violates the
     guarantee of del_timer_sync()"

* tag 'timers-urgent-2021-08-08' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip:
  timers: Move clearing of base::timer_running under base:: Lock
parents 713f0f37 bb7262b2
...@@ -1265,8 +1265,10 @@ static inline void timer_base_unlock_expiry(struct timer_base *base) ...@@ -1265,8 +1265,10 @@ static inline void timer_base_unlock_expiry(struct timer_base *base)
static void timer_sync_wait_running(struct timer_base *base) static void timer_sync_wait_running(struct timer_base *base)
{ {
if (atomic_read(&base->timer_waiters)) { if (atomic_read(&base->timer_waiters)) {
raw_spin_unlock_irq(&base->lock);
spin_unlock(&base->expiry_lock); spin_unlock(&base->expiry_lock);
spin_lock(&base->expiry_lock); spin_lock(&base->expiry_lock);
raw_spin_lock_irq(&base->lock);
} }
} }
...@@ -1457,14 +1459,14 @@ static void expire_timers(struct timer_base *base, struct hlist_head *head) ...@@ -1457,14 +1459,14 @@ static void expire_timers(struct timer_base *base, struct hlist_head *head)
if (timer->flags & TIMER_IRQSAFE) { if (timer->flags & TIMER_IRQSAFE) {
raw_spin_unlock(&base->lock); raw_spin_unlock(&base->lock);
call_timer_fn(timer, fn, baseclk); call_timer_fn(timer, fn, baseclk);
base->running_timer = NULL;
raw_spin_lock(&base->lock); raw_spin_lock(&base->lock);
base->running_timer = NULL;
} else { } else {
raw_spin_unlock_irq(&base->lock); raw_spin_unlock_irq(&base->lock);
call_timer_fn(timer, fn, baseclk); call_timer_fn(timer, fn, baseclk);
raw_spin_lock_irq(&base->lock);
base->running_timer = NULL; base->running_timer = NULL;
timer_sync_wait_running(base); timer_sync_wait_running(base);
raw_spin_lock_irq(&base->lock);
} }
} }
} }
......
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