|
|
1830f97 |
From patchwork Wed Sep 14 19:43:15 2011
|
|
|
1830f97 |
Content-Type: text/plain; charset="utf-8"
|
|
|
1830f97 |
MIME-Version: 1.0
|
|
|
1830f97 |
Content-Transfer-Encoding: 7bit
|
|
|
1830f97 |
Subject: powerpc: Fix deadlock in icswx code
|
|
|
1830f97 |
Date: Wed, 14 Sep 2011 09:43:15 -0000
|
|
|
1830f97 |
From: Anton Blanchard <anton@samba.org>
|
|
|
1830f97 |
X-Patchwork-Id: 114701
|
|
|
1830f97 |
Message-Id: <20110915054315.5e5ae062@kryten>
|
|
|
1830f97 |
To: benh@kernel.crashing.org, paulus@samba.org
|
|
|
1830f97 |
Cc: linuxppc-dev@lists.ozlabs.org
|
|
|
1830f97 |
|
|
|
1830f97 |
The icswx code introduced an A-B B-A deadlock:
|
|
|
1830f97 |
|
|
|
1830f97 |
CPU0 CPU1
|
|
|
1830f97 |
---- ----
|
|
|
1830f97 |
lock(&anon_vma->mutex);
|
|
|
1830f97 |
lock(&mm->mmap_sem);
|
|
|
1830f97 |
lock(&anon_vma->mutex);
|
|
|
1830f97 |
lock(&mm->mmap_sem);
|
|
|
1830f97 |
|
|
|
1830f97 |
Instead of using the mmap_sem to keep mm_users constant, take the
|
|
|
1830f97 |
page table spinlock.
|
|
|
1830f97 |
|
|
|
1830f97 |
Signed-off-by: Anton Blanchard <anton@samba.org>
|
|
|
1830f97 |
Cc: <stable@kernel.org>
|
|
|
1830f97 |
|
|
|
1830f97 |
---
|
|
|
1830f97 |
|
|
|
1830f97 |
|
|
|
1830f97 |
diff --git a/arch/powerpc/mm/mmu_context_hash64.c b/arch/powerpc/mm/mmu_context_hash64.c
|
|
|
1830f97 |
index 3bafc3d..4ff587e 100644
|
|
|
1830f97 |
--- a/arch/powerpc/mm/mmu_context_hash64.c
|
|
|
1830f97 |
+++ b/arch/powerpc/mm/mmu_context_hash64.c
|
|
|
1830f97 |
@@ -136,8 +136,8 @@ int use_cop(unsigned long acop, struct mm_struct *mm)
|
|
|
1830f97 |
if (!mm || !acop)
|
|
|
1830f97 |
return -EINVAL;
|
|
|
1830f97 |
|
|
|
1830f97 |
- /* We need to make sure mm_users doesn't change */
|
|
|
1830f97 |
- down_read(&mm->mmap_sem);
|
|
|
1830f97 |
+ /* The page_table_lock ensures mm_users won't change under us */
|
|
|
1830f97 |
+ spin_lock(&mm->page_table_lock);
|
|
|
1830f97 |
spin_lock(mm->context.cop_lockp);
|
|
|
1830f97 |
|
|
|
1830f97 |
if (mm->context.cop_pid == COP_PID_NONE) {
|
|
|
1830f97 |
@@ -164,7 +164,7 @@ int use_cop(unsigned long acop, struct mm_struct *mm)
|
|
|
1830f97 |
|
|
|
1830f97 |
out:
|
|
|
1830f97 |
spin_unlock(mm->context.cop_lockp);
|
|
|
1830f97 |
- up_read(&mm->mmap_sem);
|
|
|
1830f97 |
+ spin_unlock(&mm->page_table_lock);
|
|
|
1830f97 |
|
|
|
1830f97 |
return ret;
|
|
|
1830f97 |
}
|
|
|
1830f97 |
@@ -185,8 +185,8 @@ void drop_cop(unsigned long acop, struct mm_struct *mm)
|
|
|
1830f97 |
if (WARN_ON_ONCE(!mm))
|
|
|
1830f97 |
return;
|
|
|
1830f97 |
|
|
|
1830f97 |
- /* We need to make sure mm_users doesn't change */
|
|
|
1830f97 |
- down_read(&mm->mmap_sem);
|
|
|
1830f97 |
+ /* The page_table_lock ensures mm_users won't change under us */
|
|
|
1830f97 |
+ spin_lock(&mm->page_table_lock);
|
|
|
1830f97 |
spin_lock(mm->context.cop_lockp);
|
|
|
1830f97 |
|
|
|
1830f97 |
mm->context.acop &= ~acop;
|
|
|
1830f97 |
@@ -213,7 +213,7 @@ void drop_cop(unsigned long acop, struct mm_struct *mm)
|
|
|
1830f97 |
}
|
|
|
1830f97 |
|
|
|
1830f97 |
spin_unlock(mm->context.cop_lockp);
|
|
|
1830f97 |
- up_read(&mm->mmap_sem);
|
|
|
1830f97 |
+ spin_unlock(&mm->page_table_lock);
|
|
|
1830f97 |
}
|
|
|
1830f97 |
EXPORT_SYMBOL_GPL(drop_cop);
|
|
|
1830f97 |
|