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