aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2011-12-25 08:29:14 -0500
committerJens Axboe <axboe@kernel.dk>2011-12-25 08:29:14 -0500
commitfd63836811d6e5b5f5f608abf865bc9e91762c8c (patch)
tree9262ccdc0319b331c497701a851c8c5752dedb00
parent64c42998f14d5894ea3138625897d620b30c8e4e (diff)
block: an exiting task should be allowed to create io_context
While fixing io_context creation / task exit race condition, 6e736be7f2 "block: make ioc get/put interface more conventional and fix race on alloction" also prevented an exiting (%PF_EXITING) task from creating its own io_context. This is incorrect as exit path may issue IOs, e.g. from exit_files(), and if those IOs are the first ones issued by the task, io_context needs to be created to process the IOs. Combined with the existing problem of io_context / io_cq creation failure having the possibility of stalling IO, this problem results in deterministic full IO lockup with certain workloads. Fix it by allowing io_context creation regardless of %PF_EXITING for %current. Signed-off-by: Tejun Heo <tj@kernel.org> Reported-by: Andrew Morton <akpm@linux-foundation.org> Reported-by: Hugh Dickins <hughd@google.com> Signed-off-by: Jens Axboe <axboe@kernel.dk>
-rw-r--r--block/blk-ioc.c11
1 files changed, 9 insertions, 2 deletions
diff --git a/block/blk-ioc.c b/block/blk-ioc.c
index ce9b35a9468e..33fae7df16a0 100644
--- a/block/blk-ioc.c
+++ b/block/blk-ioc.c
@@ -281,9 +281,16 @@ void create_io_context_slowpath(struct task_struct *task, gfp_t gfp_flags,
281 INIT_HLIST_HEAD(&ioc->icq_list); 281 INIT_HLIST_HEAD(&ioc->icq_list);
282 INIT_WORK(&ioc->release_work, ioc_release_fn); 282 INIT_WORK(&ioc->release_work, ioc_release_fn);
283 283
284 /* try to install, somebody might already have beaten us to it */ 284 /*
285 * Try to install. ioc shouldn't be installed if someone else
286 * already did or @task, which isn't %current, is exiting. Note
287 * that we need to allow ioc creation on exiting %current as exit
288 * path may issue IOs from e.g. exit_files(). The exit path is
289 * responsible for not issuing IO after exit_io_context().
290 */
285 task_lock(task); 291 task_lock(task);
286 if (!task->io_context && !(task->flags & PF_EXITING)) 292 if (!task->io_context &&
293 (task == current || !(task->flags & PF_EXITING)))
287 task->io_context = ioc; 294 task->io_context = ioc;
288 else 295 else
289 kmem_cache_free(iocontext_cachep, ioc); 296 kmem_cache_free(iocontext_cachep, ioc);