Commit 24bce508 authored by Roland Dreier's avatar Roland Dreier

IB/umem: Fix possible hang on process exit

If ib_umem_release() is called after ib_uverbs_close() sets context->closing,
then a process can get stuck in a D state, because the code boils down to

	if (down_write_trylock(&mm->mmap_sem))
		down_write(&mm->mmap_sem);

which is obviously a stupid instant deadlock.  Fix the code so that we
only try to take the lock once.

This bug was introduced in commit f7c6a7b5 ("IB/uverbs: Export
ib_umem_get()/ib_umem_release() to modules") which fortunately never
made it into a release, and was reported by Pete Wyckoff <pw@osc.edu>.
Signed-off-by: default avatarRoland Dreier <rolandd@cisco.com>
parent d025d785
...@@ -225,13 +225,15 @@ void ib_umem_release(struct ib_umem *umem) ...@@ -225,13 +225,15 @@ void ib_umem_release(struct ib_umem *umem)
* up here and not be able to take the mmap_sem. In that case * up here and not be able to take the mmap_sem. In that case
* we defer the vm_locked accounting to the system workqueue. * we defer the vm_locked accounting to the system workqueue.
*/ */
if (context->closing && !down_write_trylock(&mm->mmap_sem)) { if (context->closing) {
INIT_WORK(&umem->work, ib_umem_account); if (!down_write_trylock(&mm->mmap_sem)) {
umem->mm = mm; INIT_WORK(&umem->work, ib_umem_account);
umem->diff = diff; umem->mm = mm;
umem->diff = diff;
schedule_work(&umem->work);
return; schedule_work(&umem->work);
return;
}
} else } else
down_write(&mm->mmap_sem); down_write(&mm->mmap_sem);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment