Taskq locking optimizations
Testing has shown that tq->tq_lock can be highly contended when a large number of small work items are dispatched. The lock hold time is reduced by the following changes: 1) Use exclusive threads in the work_waitq When a single work item is dispatched we only need to wake a single thread to service it. The current implementation uses non-exclusive threads so all threads are woken when the dispatcher calls wake_up(). If a large number of threads are in the queue this overhead can become non-negligible. 2) Conditionally add/remove threads from work waitq Taskq threads need only add themselves to the work wait queue if there are no pending work items. Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov> Issue #32
This commit is contained in:
parent
0bb43ca282
commit
3c6ed5410b
@ -454,17 +454,17 @@ taskq_thread(void *args)
|
|||||||
|
|
||||||
while (!kthread_should_stop()) {
|
while (!kthread_should_stop()) {
|
||||||
|
|
||||||
add_wait_queue(&tq->tq_work_waitq, &wait);
|
|
||||||
if (list_empty(&tq->tq_pend_list) &&
|
if (list_empty(&tq->tq_pend_list) &&
|
||||||
list_empty(&tq->tq_prio_list)) {
|
list_empty(&tq->tq_prio_list)) {
|
||||||
|
add_wait_queue_exclusive(&tq->tq_work_waitq, &wait);
|
||||||
spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags);
|
spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags);
|
||||||
schedule();
|
schedule();
|
||||||
spin_lock_irqsave(&tq->tq_lock, tq->tq_lock_flags);
|
spin_lock_irqsave(&tq->tq_lock, tq->tq_lock_flags);
|
||||||
|
remove_wait_queue(&tq->tq_work_waitq, &wait);
|
||||||
} else {
|
} else {
|
||||||
__set_current_state(TASK_RUNNING);
|
__set_current_state(TASK_RUNNING);
|
||||||
}
|
}
|
||||||
|
|
||||||
remove_wait_queue(&tq->tq_work_waitq, &wait);
|
|
||||||
|
|
||||||
if (!list_empty(&tq->tq_prio_list))
|
if (!list_empty(&tq->tq_prio_list))
|
||||||
pend_list = &tq->tq_prio_list;
|
pend_list = &tq->tq_prio_list;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user