diff options
| author | Sage Weil <sage@inktank.com> | 2013-08-15 11:11:45 -0700 | 
|---|---|---|
| committer | Sage Weil <sage@inktank.com> | 2013-08-15 11:11:45 -0700 | 
| commit | ee3e542fec6e69bc9fb668698889a37d93950ddf (patch) | |
| tree | e74ee766a4764769ef1d3d45d266b4dea64101d3 /arch/arm/include/asm/mmu_context.h | |
| parent | fe2a801b50c0bb8039d627e5ae1fec249d10ff39 (diff) | |
| parent | f1d6e17f540af37bb1891480143669ba7636c4cf (diff) | |
Merge remote-tracking branch 'linus/master' into testing
Diffstat (limited to 'arch/arm/include/asm/mmu_context.h')
| -rw-r--r-- | arch/arm/include/asm/mmu_context.h | 43 | 
1 files changed, 34 insertions, 9 deletions
| diff --git a/arch/arm/include/asm/mmu_context.h b/arch/arm/include/asm/mmu_context.h index a7b85e0d0cc..9b32f76bb0d 100644 --- a/arch/arm/include/asm/mmu_context.h +++ b/arch/arm/include/asm/mmu_context.h @@ -18,6 +18,7 @@  #include <asm/cacheflush.h>  #include <asm/cachetype.h>  #include <asm/proc-fns.h> +#include <asm/smp_plat.h>  #include <asm-generic/mm_hooks.h>  void __check_vmalloc_seq(struct mm_struct *mm); @@ -27,7 +28,15 @@ void __check_vmalloc_seq(struct mm_struct *mm);  void check_and_switch_context(struct mm_struct *mm, struct task_struct *tsk);  #define init_new_context(tsk,mm)	({ atomic64_set(&mm->context.id, 0); 0; }) -DECLARE_PER_CPU(atomic64_t, active_asids); +#ifdef CONFIG_ARM_ERRATA_798181 +void a15_erratum_get_cpumask(int this_cpu, struct mm_struct *mm, +			     cpumask_t *mask); +#else  /* !CONFIG_ARM_ERRATA_798181 */ +static inline void a15_erratum_get_cpumask(int this_cpu, struct mm_struct *mm, +					   cpumask_t *mask) +{ +} +#endif /* CONFIG_ARM_ERRATA_798181 */  #else	/* !CONFIG_CPU_HAS_ASID */ @@ -47,7 +56,7 @@ static inline void check_and_switch_context(struct mm_struct *mm,  		 * on non-ASID CPUs, the old mm will remain valid until the  		 * finish_arch_post_lock_switch() call.  		 */ -		set_ti_thread_flag(task_thread_info(tsk), TIF_SWITCH_MM); +		mm->context.switch_pending = 1;  	else  		cpu_switch_mm(mm->pgd, mm);  } @@ -56,9 +65,21 @@ static inline void check_and_switch_context(struct mm_struct *mm,  	finish_arch_post_lock_switch  static inline void finish_arch_post_lock_switch(void)  { -	if (test_and_clear_thread_flag(TIF_SWITCH_MM)) { -		struct mm_struct *mm = current->mm; -		cpu_switch_mm(mm->pgd, mm); +	struct mm_struct *mm = current->mm; + +	if (mm && mm->context.switch_pending) { +		/* +		 * Preemption must be disabled during cpu_switch_mm() as we +		 * have some stateful cache flush implementations. Check +		 * switch_pending again in case we were preempted and the +		 * switch to this mm was already done. +		 */ +		preempt_disable(); +		if (mm->context.switch_pending) { +			mm->context.switch_pending = 0; +			cpu_switch_mm(mm->pgd, mm); +		} +		preempt_enable_no_resched();  	}  } @@ -98,12 +119,16 @@ switch_mm(struct mm_struct *prev, struct mm_struct *next,  #ifdef CONFIG_MMU  	unsigned int cpu = smp_processor_id(); -#ifdef CONFIG_SMP -	/* check for possible thread migration */ -	if (!cpumask_empty(mm_cpumask(next)) && +	/* +	 * __sync_icache_dcache doesn't broadcast the I-cache invalidation, +	 * so check for possible thread migration and invalidate the I-cache +	 * if we're new to this CPU. +	 */ +	if (cache_ops_need_broadcast() && +	    !cpumask_empty(mm_cpumask(next)) &&  	    !cpumask_test_cpu(cpu, mm_cpumask(next)))  		__flush_icache_all(); -#endif +  	if (!cpumask_test_and_set_cpu(cpu, mm_cpumask(next)) || prev != next) {  		check_and_switch_context(next, tsk);  		if (cache_is_vivt()) | 
