소스 검색

powerpc/pasemi: Use raw spinlock in SMP TB sync

spin_lock() can hang if called while the timebase is frozen,
so use a raw lock instead, also disable interrupts while
at it.

Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Benjamin Herrenschmidt 16 년 전
부모
커밋
6893ce6c1c
1개의 변경된 파일10개의 추가작업 그리고 5개의 파일을 삭제
  1. 10 5
      arch/powerpc/platforms/pasemi/setup.c

+ 10 - 5
arch/powerpc/platforms/pasemi/setup.c

@@ -71,20 +71,25 @@ static void pas_restart(char *cmd)
 }
 }
 
 
 #ifdef CONFIG_SMP
 #ifdef CONFIG_SMP
-static DEFINE_SPINLOCK(timebase_lock);
+static raw_spinlock_t timebase_lock;
 static unsigned long timebase;
 static unsigned long timebase;
 
 
 static void __devinit pas_give_timebase(void)
 static void __devinit pas_give_timebase(void)
 {
 {
-	spin_lock(&timebase_lock);
+	unsigned long flags;
+
+	local_irq_save(flags);
+	hard_irq_disable();
+	__raw_spin_lock(&timebase_lock);
 	mtspr(SPRN_TBCTL, TBCTL_FREEZE);
 	mtspr(SPRN_TBCTL, TBCTL_FREEZE);
 	isync();
 	isync();
 	timebase = get_tb();
 	timebase = get_tb();
-	spin_unlock(&timebase_lock);
+	__raw_spin_unlock(&timebase_lock);
 
 
 	while (timebase)
 	while (timebase)
 		barrier();
 		barrier();
 	mtspr(SPRN_TBCTL, TBCTL_RESTART);
 	mtspr(SPRN_TBCTL, TBCTL_RESTART);
+	local_irq_restore(flags);
 }
 }
 
 
 static void __devinit pas_take_timebase(void)
 static void __devinit pas_take_timebase(void)
@@ -92,10 +97,10 @@ static void __devinit pas_take_timebase(void)
 	while (!timebase)
 	while (!timebase)
 		smp_rmb();
 		smp_rmb();
 
 
-	spin_lock(&timebase_lock);
+	__raw_spin_lock(&timebase_lock);
 	set_tb(timebase >> 32, timebase & 0xffffffff);
 	set_tb(timebase >> 32, timebase & 0xffffffff);
 	timebase = 0;
 	timebase = 0;
-	spin_unlock(&timebase_lock);
+	__raw_spin_unlock(&timebase_lock);
 }
 }
 
 
 struct smp_ops_t pas_smp_ops = {
 struct smp_ops_t pas_smp_ops = {