Commit d71ba164 authored by Petr Mladek's avatar Petr Mladek Committed by Linus Torvalds

kthread_worker: fix return value when kthread_mod_delayed_work() races with...

kthread_worker: fix return value when kthread_mod_delayed_work() races with kthread_cancel_delayed_work_sync()

kthread_mod_delayed_work() might race with
kthread_cancel_delayed_work_sync() or another kthread_mod_delayed_work()
call.  The function lets the other operation win when it sees
work->canceling counter set.  And it returns @false.

But it should return @true as it is done by the related workqueue API, see
mod_delayed_work_on().

The reason is that the return value might be used for reference counting.
It has to distinguish the case when the number of queued works has changed
or stayed the same.

The change is safe.  kthread_mod_delayed_work() return value is not
checked anywhere at the moment.

Link: https://lore.kernel.org/r/20210521163526.GA17916@redhat.com
Link: https://lkml.kernel.org/r/20210610133051.15337-4-pmladek@suse.comSigned-off-by: default avatarPetr Mladek <pmladek@suse.com>
Reported-by: default avatarOleg Nesterov <oleg@redhat.com>
Cc: Nathan Chancellor <nathan@kernel.org>
Cc: Nick Desaulniers <ndesaulniers@google.com>
Cc: Tejun Heo <tj@kernel.org>
Cc: Minchan Kim <minchan@google.com>
Cc: <jenhaochen@google.com>
Cc: Martin Liu <liumartin@google.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 20ce0c2d
...@@ -1156,14 +1156,14 @@ static bool __kthread_cancel_work(struct kthread_work *work) ...@@ -1156,14 +1156,14 @@ static bool __kthread_cancel_work(struct kthread_work *work)
* modify @dwork's timer so that it expires after @delay. If @delay is zero, * modify @dwork's timer so that it expires after @delay. If @delay is zero,
* @work is guaranteed to be queued immediately. * @work is guaranteed to be queued immediately.
* *
* Return: %true if @dwork was pending and its timer was modified, * Return: %false if @dwork was idle and queued, %true otherwise.
* %false otherwise.
* *
* A special case is when the work is being canceled in parallel. * A special case is when the work is being canceled in parallel.
* It might be caused either by the real kthread_cancel_delayed_work_sync() * It might be caused either by the real kthread_cancel_delayed_work_sync()
* or yet another kthread_mod_delayed_work() call. We let the other command * or yet another kthread_mod_delayed_work() call. We let the other command
* win and return %false here. The caller is supposed to synchronize these * win and return %true here. The return value can be used for reference
* operations a reasonable way. * counting and the number of queued works stays the same. Anyway, the caller
* is supposed to synchronize these operations a reasonable way.
* *
* This function is safe to call from any context including IRQ handler. * This function is safe to call from any context including IRQ handler.
* See __kthread_cancel_work() and kthread_delayed_work_timer_fn() * See __kthread_cancel_work() and kthread_delayed_work_timer_fn()
...@@ -1175,13 +1175,15 @@ bool kthread_mod_delayed_work(struct kthread_worker *worker, ...@@ -1175,13 +1175,15 @@ bool kthread_mod_delayed_work(struct kthread_worker *worker,
{ {
struct kthread_work *work = &dwork->work; struct kthread_work *work = &dwork->work;
unsigned long flags; unsigned long flags;
int ret = false; int ret;
raw_spin_lock_irqsave(&worker->lock, flags); raw_spin_lock_irqsave(&worker->lock, flags);
/* Do not bother with canceling when never queued. */ /* Do not bother with canceling when never queued. */
if (!work->worker) if (!work->worker) {
ret = false;
goto fast_queue; goto fast_queue;
}
/* Work must not be used with >1 worker, see kthread_queue_work() */ /* Work must not be used with >1 worker, see kthread_queue_work() */
WARN_ON_ONCE(work->worker != worker); WARN_ON_ONCE(work->worker != worker);
...@@ -1199,8 +1201,11 @@ bool kthread_mod_delayed_work(struct kthread_worker *worker, ...@@ -1199,8 +1201,11 @@ bool kthread_mod_delayed_work(struct kthread_worker *worker,
* be used for reference counting. * be used for reference counting.
*/ */
kthread_cancel_delayed_work_timer(work, &flags); kthread_cancel_delayed_work_timer(work, &flags);
if (work->canceling) if (work->canceling) {
/* The number of works in the queue does not change. */
ret = true;
goto out; goto out;
}
ret = __kthread_cancel_work(work); ret = __kthread_cancel_work(work);
fast_queue: fast_queue:
......
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