diff options
author | Tejun Heo <tj@kernel.org> | 2011-12-25 08:29:14 -0500 |
---|---|---|
committer | Jens Axboe <axboe@kernel.dk> | 2011-12-25 08:29:14 -0500 |
commit | fd63836811d6e5b5f5f608abf865bc9e91762c8c (patch) | |
tree | 9262ccdc0319b331c497701a851c8c5752dedb00 | |
parent | 64c42998f14d5894ea3138625897d620b30c8e4e (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.c | 11 |
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); |