The patch handles the case, when someone waits on parked completion but kthread exits meanwhile. To avoid infinite wait the waiter has to spin once more in a loop and simply try to get an alive kthread. If the kthread has been died, put_kthread_cb() wakes up possible waiter when kthread->vfork_done is already NULL, so next attempt to grab alive kthread pointer will fail.
Signed-off-by: Roman Pen <roman.peny...@profitbricks.com> Cc: Andy Lutomirski <l...@kernel.org> Cc: Oleg Nesterov <o...@redhat.com> Cc: Peter Zijlstra <pet...@infradead.org> Cc: Thomas Gleixner <t...@linutronix.de> Cc: Ingo Molnar <mi...@redhat.com> Cc: Tejun Heo <t...@kernel.org> Cc: linux-kernel@vger.kernel.org --- kernel/kthread.c | 41 ++++++++++++++++++++++++++++++++++++----- 1 file changed, 36 insertions(+), 5 deletions(-) diff --git a/kernel/kthread.c b/kernel/kthread.c index e8adc10556e0..a001c1ec489b 100644 --- a/kernel/kthread.c +++ b/kernel/kthread.c @@ -77,6 +77,12 @@ static void put_kthread_cb(struct callback_head *work) struct kthread *kthread; kthread = container_of(work, struct kthread, put_work); + /* + * Kick out possible waiter on a parked completion before + * ref put. That will force them to spin in a loop once + * more and eventually get the NULL kthread pointer. + */ + complete(&kthread->parked); put_kthread(kthread); } @@ -449,6 +455,11 @@ static void __kthread_unpark(struct task_struct *k, struct kthread *kthread) } } +static bool __kthread_isparked(struct kthread *kthread) +{ + return test_bit(KTHREAD_IS_PARKED, &kthread->flags); +} + /** * kthread_unpark - unpark a thread created by kthread_create(). * @k: thread created by kthread_create(). @@ -479,23 +490,43 @@ EXPORT_SYMBOL_GPL(kthread_unpark); * * Returns 0 if the thread is parked, -ENOSYS if the thread exited. * If called by the kthread itself just the park bit is set. + * + * BEWARE: The caller is responsible for ensuring the validity of @k when + * calling this function. + * + * BEWARE: Only one simultaneous caller is possible. Others will hang + * forever. You have been warned. */ int kthread_park(struct task_struct *k) { - struct kthread *kthread = to_live_kthread_and_get(k); + struct kthread *kthread; int ret = -ENOSYS; - if (kthread) { - if (!test_bit(KTHREAD_IS_PARKED, &kthread->flags)) { + /* + * Here we try to be careful and handle the case, when kthread + * is going to die and will never park. In that particular case + * put_kthread_cb() is called when kthread->vfork_done is already + * NULL. put_kthread_cb() does the last completion on kthread->parked, + * thus we will spin once more and next attempt to get an alive + * kthread will fail. + */ + do { + kthread = to_live_kthread_and_get(k); + if (!kthread) + break; + if (!__kthread_isparked(kthread)) { set_bit(KTHREAD_SHOULD_PARK, &kthread->flags); if (k != current) { wake_up_process(k); wait_for_completion(&kthread->parked); } } + if (k == current || __kthread_isparked(kthread)) + /* The way out */ + ret = 0; put_kthread(kthread); - ret = 0; - } + } while (ret); + return ret; } EXPORT_SYMBOL_GPL(kthread_park); -- 2.9.3