linux-arm-kernel.lists.infradead.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] arm: Use arch_spin_lock for unwind to avoid lockdep deadlock
@ 2011-05-17 14:11 Thomas Gleixner
  2011-05-17 14:20 ` Peter Zijlstra
  2011-05-17 15:26 ` Catalin Marinas
  0 siblings, 2 replies; 5+ messages in thread
From: Thomas Gleixner @ 2011-05-17 14:11 UTC (permalink / raw)
  To: linux-arm-kernel

When lockdep traces a lock taken in a module the arm unwinder takes
unwind_lock which causes a recursive lockdep call deadlocking itself
on the unwind_lock.

Make unwind_lock an arch_spin_lock to avoid the lockdep call and use
the raw_local_irq_save/restore() variants to avoid irq tracing as
well.

Reported-and-tested-by: Torben Hohn <torbenh@linutronix.de>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Cc: stable at kernel.org
---
 arch/arm/kernel/unwind.c |   20 +++++++++++++-------
 1 file changed, 13 insertions(+), 7 deletions(-)

Index: linux-2.6/arch/arm/kernel/unwind.c
===================================================================
--- linux-2.6.orig/arch/arm/kernel/unwind.c
+++ linux-2.6/arch/arm/kernel/unwind.c
@@ -86,7 +86,7 @@ enum regs {
 extern struct unwind_idx __start_unwind_idx[];
 extern struct unwind_idx __stop_unwind_idx[];
 
-static DEFINE_SPINLOCK(unwind_lock);
+static arch_spinlock_t unwind_lock = (arch_spinlock_t)__ARCH_SPIN_LOCK_UNLOCKED;
 static LIST_HEAD(unwind_tables);
 
 /* Convert a prel31 symbol to an absolute address */
@@ -140,7 +140,8 @@ static struct unwind_idx *unwind_find_id
 		/* module unwind tables */
 		struct unwind_table *table;
 
-		spin_lock_irqsave(&unwind_lock, flags);
+		raw_local_irq_save(flags);
+		arch_spin_lock(&unwind_lock);
 		list_for_each_entry(table, &unwind_tables, list) {
 			if (addr >= table->begin_addr &&
 			    addr < table->end_addr) {
@@ -151,7 +152,8 @@ static struct unwind_idx *unwind_find_id
 				break;
 			}
 		}
-		spin_unlock_irqrestore(&unwind_lock, flags);
+		arch_spin_unlock(&unwind_lock);
+		raw_local_irq_restore(flags);
 	}
 
 	pr_debug("%s: idx = %p\n", __func__, idx);
@@ -417,9 +419,11 @@ struct unwind_table *unwind_table_add(un
 	for (idx = tab->start; idx < tab->stop; idx++)
 		idx->addr = prel31_to_addr(&idx->addr);
 
-	spin_lock_irqsave(&unwind_lock, flags);
+	raw_local_irq_save(flags);
+	arch_spin_lock(&unwind_lock);
 	list_add_tail(&tab->list, &unwind_tables);
-	spin_unlock_irqrestore(&unwind_lock, flags);
+	arch_spin_unlock(&unwind_lock);
+	raw_local_irq_restore(flags);
 
 	return tab;
 }
@@ -431,9 +435,11 @@ void unwind_table_del(struct unwind_tabl
 	if (!tab)
 		return;
 
-	spin_lock_irqsave(&unwind_lock, flags);
+	raw_local_irq_save(flags);
+	arch_spin_lock(&unwind_lock);
 	list_del(&tab->list);
-	spin_unlock_irqrestore(&unwind_lock, flags);
+	arch_spin_unlock(&unwind_lock);
+	raw_local_irq_restore(flags);
 
 	kfree(tab);
 }

^ permalink raw reply	[flat|nested] 5+ messages in thread

end of thread, other threads:[~2011-05-17 19:31 UTC | newest]

Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2011-05-17 14:11 [PATCH] arm: Use arch_spin_lock for unwind to avoid lockdep deadlock Thomas Gleixner
2011-05-17 14:20 ` Peter Zijlstra
2011-05-17 15:45   ` Thomas Gleixner
2011-05-17 19:31     ` Thomas Gleixner
2011-05-17 15:26 ` Catalin Marinas

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).