diff options
author | Andreas Larsson <andreas@gaisler.com> | 2014-12-18 07:23:23 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-12-18 12:47:54 -0500 |
commit | 66d0f7ec9f1038452178b1993fc07fd96d30fd38 (patch) | |
tree | 8894bb4925a29d29301a851d4bb384b68bd6a24d /arch/sparc | |
parent | 44e8967d591686463e84a88b46b03beba3ab49fb (diff) |
sparc32: destroy_context() and switch_mm() needs to disable interrupts.
Load balancing can be triggered in the critical sections protected by
srmmu_context_spinlock in destroy_context() and switch_mm() and can hang
the cpu waiting for the rq lock of another cpu that in turn has called
switch_mm hangning on srmmu_context_spinlock leading to deadlock.
So, disable interrupt while taking srmmu_context_spinlock in
destroy_context() and switch_mm() so we don't deadlock.
See also commit 77b838fa1ef0 ("[SPARC64]: destroy_context() needs to disable
interrupts.")
Signed-off-by: Andreas Larsson <andreas@gaisler.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'arch/sparc')
-rw-r--r-- | arch/sparc/mm/srmmu.c | 11 |
1 files changed, 7 insertions, 4 deletions
diff --git a/arch/sparc/mm/srmmu.c b/arch/sparc/mm/srmmu.c index be65f035d18a..5cbc96d801ff 100644 --- a/arch/sparc/mm/srmmu.c +++ b/arch/sparc/mm/srmmu.c | |||
@@ -460,10 +460,12 @@ static void __init sparc_context_init(int numctx) | |||
460 | void switch_mm(struct mm_struct *old_mm, struct mm_struct *mm, | 460 | void switch_mm(struct mm_struct *old_mm, struct mm_struct *mm, |
461 | struct task_struct *tsk) | 461 | struct task_struct *tsk) |
462 | { | 462 | { |
463 | unsigned long flags; | ||
464 | |||
463 | if (mm->context == NO_CONTEXT) { | 465 | if (mm->context == NO_CONTEXT) { |
464 | spin_lock(&srmmu_context_spinlock); | 466 | spin_lock_irqsave(&srmmu_context_spinlock, flags); |
465 | alloc_context(old_mm, mm); | 467 | alloc_context(old_mm, mm); |
466 | spin_unlock(&srmmu_context_spinlock); | 468 | spin_unlock_irqrestore(&srmmu_context_spinlock, flags); |
467 | srmmu_ctxd_set(&srmmu_context_table[mm->context], mm->pgd); | 469 | srmmu_ctxd_set(&srmmu_context_table[mm->context], mm->pgd); |
468 | } | 470 | } |
469 | 471 | ||
@@ -986,14 +988,15 @@ int init_new_context(struct task_struct *tsk, struct mm_struct *mm) | |||
986 | 988 | ||
987 | void destroy_context(struct mm_struct *mm) | 989 | void destroy_context(struct mm_struct *mm) |
988 | { | 990 | { |
991 | unsigned long flags; | ||
989 | 992 | ||
990 | if (mm->context != NO_CONTEXT) { | 993 | if (mm->context != NO_CONTEXT) { |
991 | flush_cache_mm(mm); | 994 | flush_cache_mm(mm); |
992 | srmmu_ctxd_set(&srmmu_context_table[mm->context], srmmu_swapper_pg_dir); | 995 | srmmu_ctxd_set(&srmmu_context_table[mm->context], srmmu_swapper_pg_dir); |
993 | flush_tlb_mm(mm); | 996 | flush_tlb_mm(mm); |
994 | spin_lock(&srmmu_context_spinlock); | 997 | spin_lock_irqsave(&srmmu_context_spinlock, flags); |
995 | free_context(mm->context); | 998 | free_context(mm->context); |
996 | spin_unlock(&srmmu_context_spinlock); | 999 | spin_unlock_irqrestore(&srmmu_context_spinlock, flags); |
997 | mm->context = NO_CONTEXT; | 1000 | mm->context = NO_CONTEXT; |
998 | } | 1001 | } |
999 | } | 1002 | } |