提交 36aa9dfc 编写于 作者: O Oleg Nesterov 提交者: Linus Torvalds

workqueue: don't clear cwq->thread until it exits

Pointed out by Srivatsa Vaddagiri.

cleanup_workqueue_thread() sets cwq->thread = NULL and does kthread_stop().
This breaks the "if (cwq->thread == current)" logic in flush_cpu_workqueue()
and leads to deadlock.

Kill the thead first, then clear cwq->thread. workqueue_mutex protects us
from create_workqueue_thread() so we don't need cwq->lock.
Signed-off-by: NOleg Nesterov <oleg@tv-sign.ru>
Cc: Srivatsa Vaddagiri <vatsa@in.ibm.com>
Cc: "Pallipadi, Venkatesh" <venkatesh.pallipadi@intel.com>
Cc: Gautham shenoy <ego@in.ibm.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 d721304d
...@@ -625,17 +625,12 @@ EXPORT_SYMBOL_GPL(__create_workqueue); ...@@ -625,17 +625,12 @@ EXPORT_SYMBOL_GPL(__create_workqueue);
static void cleanup_workqueue_thread(struct workqueue_struct *wq, int cpu) static void cleanup_workqueue_thread(struct workqueue_struct *wq, int cpu)
{ {
struct cpu_workqueue_struct *cwq; struct cpu_workqueue_struct *cwq = per_cpu_ptr(wq->cpu_wq, cpu);
unsigned long flags;
struct task_struct *p;
cwq = per_cpu_ptr(wq->cpu_wq, cpu); if (cwq->thread) {
spin_lock_irqsave(&cwq->lock, flags); kthread_stop(cwq->thread);
p = cwq->thread; cwq->thread = NULL;
cwq->thread = NULL; }
spin_unlock_irqrestore(&cwq->lock, flags);
if (p)
kthread_stop(p);
} }
/** /**
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册