Fix taskq dynamic spawning

Currently taskq_dispatch() will spawn new task with a condition that the caller
is also a member of the taskq. However, under this condition, it will still
cause deadlock where a task on tq1 is waiting another thread, who is trying to
dispatch a task on tq1. So this patch removes the check.

For example when you do:
zfs send pp/fs0@001 | zfs recv pp/fs0_copy

This will easily deadlock before this patch.

Also, move the seq_task check from taskq_thread_spawn() to taskq_thread()
because it's not used by the caller from taskq_dispatch().

Signed-off-by: Chunwei Chen <david.chen@osnexus.com>
Signed-off-by: Tim Chase <tim@chase2k.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #496
This commit is contained in:
tuxoko 2015-11-06 15:00:55 -08:00 committed by Brian Behlendorf
parent 3e7e6f34d0
commit f5f2b87df0

View File

@ -538,7 +538,7 @@ taskq_cancel_id(taskq_t *tq, taskqid_t id)
} }
EXPORT_SYMBOL(taskq_cancel_id); EXPORT_SYMBOL(taskq_cancel_id);
static int taskq_thread_spawn(taskq_t *tq, int seq_tasks); static int taskq_thread_spawn(taskq_t *tq);
taskqid_t taskqid_t
taskq_dispatch(taskq_t *tq, task_func_t func, void *arg, uint_t flags) taskq_dispatch(taskq_t *tq, task_func_t func, void *arg, uint_t flags)
@ -587,9 +587,8 @@ taskq_dispatch(taskq_t *tq, task_func_t func, void *arg, uint_t flags)
wake_up(&tq->tq_work_waitq); wake_up(&tq->tq_work_waitq);
out: out:
/* Spawn additional taskq threads if required. */ /* Spawn additional taskq threads if required. */
if (tq->tq_nactive == tq->tq_nthreads && if (tq->tq_nactive == tq->tq_nthreads)
taskq_member_impl(tq, current)) (void) taskq_thread_spawn(tq);
(void) taskq_thread_spawn(tq, spl_taskq_thread_sequential + 1);
spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags); spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags);
return (rc); return (rc);
@ -635,9 +634,8 @@ taskq_dispatch_delay(taskq_t *tq, task_func_t func, void *arg,
spin_unlock(&t->tqent_lock); spin_unlock(&t->tqent_lock);
out: out:
/* Spawn additional taskq threads if required. */ /* Spawn additional taskq threads if required. */
if (tq->tq_nactive == tq->tq_nthreads && if (tq->tq_nactive == tq->tq_nthreads)
taskq_member_impl(tq, current)) (void) taskq_thread_spawn(tq);
(void) taskq_thread_spawn(tq, spl_taskq_thread_sequential + 1);
spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags); spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags);
return (rc); return (rc);
} }
@ -683,9 +681,8 @@ taskq_dispatch_ent(taskq_t *tq, task_func_t func, void *arg, uint_t flags,
wake_up(&tq->tq_work_waitq); wake_up(&tq->tq_work_waitq);
out: out:
/* Spawn additional taskq threads if required. */ /* Spawn additional taskq threads if required. */
if (tq->tq_nactive == tq->tq_nthreads && if (tq->tq_nactive == tq->tq_nthreads)
taskq_member_impl(tq, current)) (void) taskq_thread_spawn(tq);
(void) taskq_thread_spawn(tq, spl_taskq_thread_sequential + 1);
spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags); spin_unlock_irqrestore(&tq->tq_lock, tq->tq_lock_flags);
} }
EXPORT_SYMBOL(taskq_dispatch_ent); EXPORT_SYMBOL(taskq_dispatch_ent);
@ -756,15 +753,14 @@ taskq_thread_spawn_task(void *arg)
* which is also a dynamic taskq cannot be safely used for this. * which is also a dynamic taskq cannot be safely used for this.
*/ */
static int static int
taskq_thread_spawn(taskq_t *tq, int seq_tasks) taskq_thread_spawn(taskq_t *tq)
{ {
int spawning = 0; int spawning = 0;
if (!(tq->tq_flags & TASKQ_DYNAMIC)) if (!(tq->tq_flags & TASKQ_DYNAMIC))
return (0); return (0);
if ((seq_tasks > spl_taskq_thread_sequential) && if ((tq->tq_nthreads + tq->tq_nspawn < tq->tq_maxthreads) &&
(tq->tq_nthreads + tq->tq_nspawn < tq->tq_maxthreads) &&
(tq->tq_flags & TASKQ_ACTIVE)) { (tq->tq_flags & TASKQ_ACTIVE)) {
spawning = (++tq->tq_nspawn); spawning = (++tq->tq_nspawn);
taskq_dispatch(dynamic_taskq, taskq_thread_spawn_task, taskq_dispatch(dynamic_taskq, taskq_thread_spawn_task,
@ -898,7 +894,8 @@ taskq_thread(void *args)
} }
/* Spawn additional taskq threads if required. */ /* Spawn additional taskq threads if required. */
if (taskq_thread_spawn(tq, ++seq_tasks)) if ((++seq_tasks) > spl_taskq_thread_sequential &&
taskq_thread_spawn(tq))
seq_tasks = 0; seq_tasks = 0;
tqt->tqt_id = 0; tqt->tqt_id = 0;