workqueue: It is likely that WORKER_NOT_RUNNING is true
Running the annotate branch profiler on three boxes, including my main box that runs firefox, evolution, xchat, and is part of the distcc farm, showed this with the likelys in the workqueue code: correct incorrect % Function File Line ------- --------- - -------- ---- ---- 96 996253 99 wq_worker_sleeping workqueue.c 703 96 996247 99 wq_worker_waking_up workqueue.c 677 The likely()s in this case were assuming that WORKER_NOT_RUNNING will most likely be false. But this is not the case. The reason is (and shown by adding trace_printks and testing it) that most of the time WORKER_PREP is set. In worker_thread() we have: worker_clr_flags(worker, WORKER_PREP); [ do work stuff ] worker_set_flags(worker, WORKER_PREP, false); (that 'false' means not to wake up an idle worker) The wq_worker_sleeping() is called from schedule when a worker thread is putting itself to sleep. Which happens most of the time outside of that [ do work stuff ]. The wq_worker_waking_up is called by the wakeup worker code, which is also callod outside that [ do work stuff ]. Thus, the likely and unlikely used by those two functions are actually backwards. Remove the annotation and let gcc figure it out. Acked-by: Tejun Heo <tj@kernel.org> Signed-off-by: Steven Rostedt <rostedt@goodmis.org> Signed-off-by: Tejun Heo <tj@kernel.org>
This commit is contained in:
parent
3e6cd7a4b6
commit
2d64672ed3
|
@ -661,7 +661,7 @@ void wq_worker_waking_up(struct task_struct *task, unsigned int cpu)
|
||||||
{
|
{
|
||||||
struct worker *worker = kthread_data(task);
|
struct worker *worker = kthread_data(task);
|
||||||
|
|
||||||
if (likely(!(worker->flags & WORKER_NOT_RUNNING)))
|
if (!(worker->flags & WORKER_NOT_RUNNING))
|
||||||
atomic_inc(get_gcwq_nr_running(cpu));
|
atomic_inc(get_gcwq_nr_running(cpu));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -687,7 +687,7 @@ struct task_struct *wq_worker_sleeping(struct task_struct *task,
|
||||||
struct global_cwq *gcwq = get_gcwq(cpu);
|
struct global_cwq *gcwq = get_gcwq(cpu);
|
||||||
atomic_t *nr_running = get_gcwq_nr_running(cpu);
|
atomic_t *nr_running = get_gcwq_nr_running(cpu);
|
||||||
|
|
||||||
if (unlikely(worker->flags & WORKER_NOT_RUNNING))
|
if (worker->flags & WORKER_NOT_RUNNING)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
/* this can only happen on the local cpu */
|
/* this can only happen on the local cpu */
|
||||||
|
|
Loading…
Reference in New Issue