Commit cb444766 authored by Tejun Heo's avatar Tejun Heo

workqueue: use worker_set/clr_flags() only from worker itself

worker_set/clr_flags() assume that if none of NOT_RUNNING flags is set
the worker must be contributing to nr_running which is only true if
the worker is actually running.

As when called from self, it is guaranteed that the worker is running,
those functions can be safely used from the worker itself and they
aren't necessary from other places anyway.  Make the following changes
to fix the bug.

* Make worker_set/clr_flags() whine if not called from self.

* Convert all places which called those functions from other tasks to
  manipulate flags directly.

* Make trustee_thread() directly clear nr_running after setting
  WORKER_ROGUE on all workers.  This is the only place where
  nr_running manipulation is necessary outside of workers themselves.

* While at it, add sanity check for nr_running in worker_enter_idle().
Signed-off-by: default avatarTejun Heo <tj@kernel.org>
parent fb0e7beb
...@@ -601,7 +601,7 @@ struct task_struct *wq_worker_sleeping(struct task_struct *task, ...@@ -601,7 +601,7 @@ struct task_struct *wq_worker_sleeping(struct task_struct *task,
/** /**
* worker_set_flags - set worker flags and adjust nr_running accordingly * worker_set_flags - set worker flags and adjust nr_running accordingly
* @worker: worker to set flags for * @worker: self
* @flags: flags to set * @flags: flags to set
* @wakeup: wakeup an idle worker if necessary * @wakeup: wakeup an idle worker if necessary
* *
...@@ -609,14 +609,16 @@ struct task_struct *wq_worker_sleeping(struct task_struct *task, ...@@ -609,14 +609,16 @@ struct task_struct *wq_worker_sleeping(struct task_struct *task,
* nr_running becomes zero and @wakeup is %true, an idle worker is * nr_running becomes zero and @wakeup is %true, an idle worker is
* woken up. * woken up.
* *
* LOCKING: * CONTEXT:
* spin_lock_irq(gcwq->lock). * spin_lock_irq(gcwq->lock)
*/ */
static inline void worker_set_flags(struct worker *worker, unsigned int flags, static inline void worker_set_flags(struct worker *worker, unsigned int flags,
bool wakeup) bool wakeup)
{ {
struct global_cwq *gcwq = worker->gcwq; struct global_cwq *gcwq = worker->gcwq;
WARN_ON_ONCE(worker->task != current);
/* /*
* If transitioning into NOT_RUNNING, adjust nr_running and * If transitioning into NOT_RUNNING, adjust nr_running and
* wake up an idle worker as necessary if requested by * wake up an idle worker as necessary if requested by
...@@ -639,19 +641,21 @@ static inline void worker_set_flags(struct worker *worker, unsigned int flags, ...@@ -639,19 +641,21 @@ static inline void worker_set_flags(struct worker *worker, unsigned int flags,
/** /**
* worker_clr_flags - clear worker flags and adjust nr_running accordingly * worker_clr_flags - clear worker flags and adjust nr_running accordingly
* @worker: worker to set flags for * @worker: self
* @flags: flags to clear * @flags: flags to clear
* *
* Clear @flags in @worker->flags and adjust nr_running accordingly. * Clear @flags in @worker->flags and adjust nr_running accordingly.
* *
* LOCKING: * CONTEXT:
* spin_lock_irq(gcwq->lock). * spin_lock_irq(gcwq->lock)
*/ */
static inline void worker_clr_flags(struct worker *worker, unsigned int flags) static inline void worker_clr_flags(struct worker *worker, unsigned int flags)
{ {
struct global_cwq *gcwq = worker->gcwq; struct global_cwq *gcwq = worker->gcwq;
unsigned int oflags = worker->flags; unsigned int oflags = worker->flags;
WARN_ON_ONCE(worker->task != current);
worker->flags &= ~flags; worker->flags &= ~flags;
/* if transitioning out of NOT_RUNNING, increment nr_running */ /* if transitioning out of NOT_RUNNING, increment nr_running */
...@@ -1073,7 +1077,8 @@ static void worker_enter_idle(struct worker *worker) ...@@ -1073,7 +1077,8 @@ static void worker_enter_idle(struct worker *worker)
BUG_ON(!list_empty(&worker->entry) && BUG_ON(!list_empty(&worker->entry) &&
(worker->hentry.next || worker->hentry.pprev)); (worker->hentry.next || worker->hentry.pprev));
worker_set_flags(worker, WORKER_IDLE, false); /* can't use worker_set_flags(), also called from start_worker() */
worker->flags |= WORKER_IDLE;
gcwq->nr_idle++; gcwq->nr_idle++;
worker->last_active = jiffies; worker->last_active = jiffies;
...@@ -1086,6 +1091,10 @@ static void worker_enter_idle(struct worker *worker) ...@@ -1086,6 +1091,10 @@ static void worker_enter_idle(struct worker *worker)
jiffies + IDLE_WORKER_TIMEOUT); jiffies + IDLE_WORKER_TIMEOUT);
} else } else
wake_up_all(&gcwq->trustee_wait); wake_up_all(&gcwq->trustee_wait);
/* sanity check nr_running */
WARN_ON_ONCE(gcwq->nr_workers == gcwq->nr_idle &&
atomic_read(get_gcwq_nr_running(gcwq->cpu)));
} }
/** /**
...@@ -1270,7 +1279,7 @@ static struct worker *create_worker(struct global_cwq *gcwq, bool bind) ...@@ -1270,7 +1279,7 @@ static struct worker *create_worker(struct global_cwq *gcwq, bool bind)
*/ */
static void start_worker(struct worker *worker) static void start_worker(struct worker *worker)
{ {
worker_set_flags(worker, WORKER_STARTED, false); worker->flags |= WORKER_STARTED;
worker->gcwq->nr_workers++; worker->gcwq->nr_workers++;
worker_enter_idle(worker); worker_enter_idle(worker);
wake_up_process(worker->task); wake_up_process(worker->task);
...@@ -1300,7 +1309,7 @@ static void destroy_worker(struct worker *worker) ...@@ -1300,7 +1309,7 @@ static void destroy_worker(struct worker *worker)
gcwq->nr_idle--; gcwq->nr_idle--;
list_del_init(&worker->entry); list_del_init(&worker->entry);
worker_set_flags(worker, WORKER_DIE, false); worker->flags |= WORKER_DIE;
spin_unlock_irq(&gcwq->lock); spin_unlock_irq(&gcwq->lock);
...@@ -2979,10 +2988,10 @@ static int __cpuinit trustee_thread(void *__gcwq) ...@@ -2979,10 +2988,10 @@ static int __cpuinit trustee_thread(void *__gcwq)
gcwq->flags |= GCWQ_MANAGING_WORKERS; gcwq->flags |= GCWQ_MANAGING_WORKERS;
list_for_each_entry(worker, &gcwq->idle_list, entry) list_for_each_entry(worker, &gcwq->idle_list, entry)
worker_set_flags(worker, WORKER_ROGUE, false); worker->flags |= WORKER_ROGUE;
for_each_busy_worker(worker, i, pos, gcwq) for_each_busy_worker(worker, i, pos, gcwq)
worker_set_flags(worker, WORKER_ROGUE, false); worker->flags |= WORKER_ROGUE;
/* /*
* Call schedule() so that we cross rq->lock and thus can * Call schedule() so that we cross rq->lock and thus can
...@@ -2995,12 +3004,12 @@ static int __cpuinit trustee_thread(void *__gcwq) ...@@ -2995,12 +3004,12 @@ static int __cpuinit trustee_thread(void *__gcwq)
spin_lock_irq(&gcwq->lock); spin_lock_irq(&gcwq->lock);
/* /*
* Sched callbacks are disabled now. gcwq->nr_running should * Sched callbacks are disabled now. Zap nr_running. After
* be zero and will stay that way, making need_more_worker() * this, nr_running stays zero and need_more_worker() and
* and keep_working() always return true as long as the * keep_working() are always true as long as the worklist is
* worklist is not empty. * not empty.
*/ */
WARN_ON_ONCE(atomic_read(get_gcwq_nr_running(gcwq->cpu)) != 0); atomic_set(get_gcwq_nr_running(gcwq->cpu), 0);
spin_unlock_irq(&gcwq->lock); spin_unlock_irq(&gcwq->lock);
del_timer_sync(&gcwq->idle_timer); del_timer_sync(&gcwq->idle_timer);
...@@ -3046,7 +3055,7 @@ static int __cpuinit trustee_thread(void *__gcwq) ...@@ -3046,7 +3055,7 @@ static int __cpuinit trustee_thread(void *__gcwq)
worker = create_worker(gcwq, false); worker = create_worker(gcwq, false);
spin_lock_irq(&gcwq->lock); spin_lock_irq(&gcwq->lock);
if (worker) { if (worker) {
worker_set_flags(worker, WORKER_ROGUE, false); worker->flags |= WORKER_ROGUE;
start_worker(worker); start_worker(worker);
} }
} }
...@@ -3085,8 +3094,8 @@ static int __cpuinit trustee_thread(void *__gcwq) ...@@ -3085,8 +3094,8 @@ static int __cpuinit trustee_thread(void *__gcwq)
* operations. Use a separate flag to mark that * operations. Use a separate flag to mark that
* rebinding is scheduled. * rebinding is scheduled.
*/ */
worker_set_flags(worker, WORKER_REBIND, false); worker->flags |= WORKER_REBIND;
worker_clr_flags(worker, WORKER_ROGUE); worker->flags &= ~WORKER_ROGUE;
/* queue rebind_work, wq doesn't matter, use the default one */ /* queue rebind_work, wq doesn't matter, use the default one */
if (test_and_set_bit(WORK_STRUCT_PENDING_BIT, if (test_and_set_bit(WORK_STRUCT_PENDING_BIT,
......
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