Commit a22a9602 authored by Kent Overstreet's avatar Kent Overstreet Committed by Jens Axboe

closures: fix a race on wakeup from closure_sync

The race was when a thread using closure_sync() notices cl->s->done == 1
before the thread calling closure_put() calls wake_up_process(). Then,
it's possible for that thread to return and exit just before
wake_up_process() is called - so we're trying to wake up a process that
no longer exists.

rcu_read_lock() is sufficient to protect against this, as there's an rcu
barrier somewhere in the process teardown path.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@gmail.com>
Acked-by: default avatarColy Li <colyli@suse.de>
Signed-off-by: default avatarJens Axboe <axboe@kernel.dk>
parent d66c9920
...@@ -105,8 +105,14 @@ struct closure_syncer { ...@@ -105,8 +105,14 @@ struct closure_syncer {
static void closure_sync_fn(struct closure *cl) static void closure_sync_fn(struct closure *cl)
{ {
cl->s->done = 1; struct closure_syncer *s = cl->s;
wake_up_process(cl->s->task); struct task_struct *p;
rcu_read_lock();
p = READ_ONCE(s->task);
s->done = 1;
wake_up_process(p);
rcu_read_unlock();
} }
void __sched __closure_sync(struct closure *cl) void __sched __closure_sync(struct closure *cl)
......
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