[PATCH] powerpc/powernv: Rework local TLB flush for boot and MCE on POWER9

Benjamin Herrenschmidt benh at kernel.crashing.org
Wed Jun 28 23:02:56 AEST 2017


On Wed, 2017-06-28 at 08:21 +0530, Aneesh Kumar K.V wrote:
> 
> 
> I am not sure the new location of flushing the tlb is correct/perfect. For ex:
> may be we should do it before htab_initialize() so that we start with
> all everything flushed ? But otherwise

Doesn't matter as long as you do it before you turn on MSR:DR/IR

> Reviewed-by: Aneesh Kumar K.V <aneesh.kumar at linux.vnet.ibm.com>
> 
> 
> > 
> > Signed-off-by: Nicholas Piggin <npiggin at gmail.com>
> > ---
> >  arch/powerpc/include/asm/book3s/64/tlbflush-hash.h |  1 +
> >  .../powerpc/include/asm/book3s/64/tlbflush-radix.h |  3 +
> >  arch/powerpc/include/asm/book3s/64/tlbflush.h      | 34 +++++++++
> >  arch/powerpc/include/asm/cputable.h                | 12 ----
> >  arch/powerpc/kernel/cpu_setup_power.S              | 43 ------------
> >  arch/powerpc/kernel/cputable.c                     | 14 ----
> >  arch/powerpc/kernel/dt_cpu_ftrs.c                  | 42 -----------
> >  arch/powerpc/kernel/mce_power.c                    | 61 +---------------
> >  arch/powerpc/kvm/book3s_hv_ras.c                   |  6 +-
> >  arch/powerpc/mm/hash_native_64.c                   | 82 ++++++++++++++++++++++
> >  arch/powerpc/mm/hash_utils_64.c                    |  4 ++
> >  arch/powerpc/mm/pgtable-radix.c                    |  4 ++
> >  arch/powerpc/mm/tlb-radix.c                        | 57 +++++++++++++++
> >  13 files changed, 189 insertions(+), 174 deletions(-)
> > 
> > diff --git a/arch/powerpc/include/asm/book3s/64/tlbflush-hash.h b/arch/powerpc/include/asm/book3s/64/tlbflush-hash.h
> > index 2f6373144e2c..c02ece27fd7b 100644
> > --- a/arch/powerpc/include/asm/book3s/64/tlbflush-hash.h
> > +++ b/arch/powerpc/include/asm/book3s/64/tlbflush-hash.h
> > @@ -50,6 +50,7 @@ static inline void arch_leave_lazy_mmu_mode(void)
> > 
> >  #define arch_flush_lazy_mmu_mode()      do {} while (0)
> > 
> > +extern void hash__tlbiel_all(unsigned int action);
> > 
> >  extern void flush_hash_page(unsigned long vpn, real_pte_t pte, int psize,
> >  			    int ssize, unsigned long flags);
> > diff --git a/arch/powerpc/include/asm/book3s/64/tlbflush-radix.h b/arch/powerpc/include/asm/book3s/64/tlbflush-radix.h
> > index cc7fbde4f53c..e7b767a3b2fa 100644
> > --- a/arch/powerpc/include/asm/book3s/64/tlbflush-radix.h
> > +++ b/arch/powerpc/include/asm/book3s/64/tlbflush-radix.h
> > @@ -10,6 +10,8 @@ static inline int mmu_get_ap(int psize)
> >  	return mmu_psize_defs[psize].ap;
> >  }
> > 
> > +extern void radix__tlbiel_all(unsigned int action);
> > +
> >  extern void radix__flush_hugetlb_tlb_range(struct vm_area_struct *vma,
> >  					   unsigned long start, unsigned long end);
> >  extern void radix__flush_tlb_range_psize(struct mm_struct *mm, unsigned long start,
> > @@ -44,4 +46,5 @@ extern void radix__flush_tlb_lpid(unsigned long lpid);
> >  extern void radix__flush_tlb_all(void);
> >  extern void radix__flush_tlb_pte_p9_dd1(unsigned long old_pte, struct mm_struct *mm,
> >  					unsigned long address);
> > +
> >  #endif
> > diff --git a/arch/powerpc/include/asm/book3s/64/tlbflush.h b/arch/powerpc/include/asm/book3s/64/tlbflush.h
> > index 72b925f97bab..a6f3a210d4de 100644
> > --- a/arch/powerpc/include/asm/book3s/64/tlbflush.h
> > +++ b/arch/powerpc/include/asm/book3s/64/tlbflush.h
> > @@ -7,6 +7,40 @@
> >  #include <asm/book3s/64/tlbflush-hash.h>
> >  #include <asm/book3s/64/tlbflush-radix.h>
> > 
> > +/* TLB flush actions. Used as argument to tlbiel_all() */
> > +enum {
> > +	TLB_INVAL_SCOPE_GLOBAL = 0,	/* invalidate all TLBs */
> > +	TLB_INVAL_SCOPE_LPID = 1,	/* invalidate TLBs for current LPID */
> > +};
> > +
> > +static inline void tlbiel_all(void)
> > +{
> > +	/*
> > +	 * This is used for host machine check and bootup.
> > +	 *
> > +	 * This could be reimplemented more robustly without using the
> > +	 * radix_is_enabled(), cpu_feature(), etc. calls. However these
> > +	 * should be set up before relocation starts to be used at boot,
> > +	 * so we shouldn't see TLB machine checks before then.
> > +	 */
> > +	if (radix_enabled())
> > +		radix__tlbiel_all(TLB_INVAL_SCOPE_GLOBAL);
> > +	else
> > +		hash__tlbiel_all(TLB_INVAL_SCOPE_GLOBAL);
> > +}
> > +
> > +static inline void tlbiel_all_lpid(bool radix)
> > +{
> > +	/*
> > +	 * This is used for guest machine check.
> > +	 */
> > +	if (radix)
> > +		radix__tlbiel_all(TLB_INVAL_SCOPE_LPID);
> > +	else
> > +		hash__tlbiel_all(TLB_INVAL_SCOPE_LPID);
> > +}
> > +
> > +
> >  #define __HAVE_ARCH_FLUSH_PMD_TLB_RANGE
> >  static inline void flush_pmd_tlb_range(struct vm_area_struct *vma,
> >  				       unsigned long start, unsigned long end)
> > diff --git a/arch/powerpc/include/asm/cputable.h b/arch/powerpc/include/asm/cputable.h
> > index c2d509584a98..808a5aa4bcf2 100644
> > --- a/arch/powerpc/include/asm/cputable.h
> > +++ b/arch/powerpc/include/asm/cputable.h
> > @@ -106,12 +106,6 @@ struct cpu_spec {
> >  	 * called in real mode to handle SLB and TLB errors.
> >  	 */
> >  	long		(*machine_check_early)(struct pt_regs *regs);
> > -
> > -	/*
> > -	 * Processor specific routine to flush tlbs.
> > -	 */
> > -	void		(*flush_tlb)(unsigned int action);
> > -
> >  };
> > 
> >  extern struct cpu_spec		*cur_cpu_spec;
> > @@ -132,12 +126,6 @@ extern void cpu_feature_keys_init(void);
> >  static inline void cpu_feature_keys_init(void) { }
> >  #endif
> > 
> > -/* TLB flush actions. Used as argument to cpu_spec.flush_tlb() hook */
> > -enum {
> > -	TLB_INVAL_SCOPE_GLOBAL = 0,	/* invalidate all TLBs */
> > -	TLB_INVAL_SCOPE_LPID = 1,	/* invalidate TLBs for current LPID */
> > -};
> > -
> >  #endif /* __ASSEMBLY__ */
> > 
> >  /* CPU kernel features */
> > diff --git a/arch/powerpc/kernel/cpu_setup_power.S b/arch/powerpc/kernel/cpu_setup_power.S
> > index 10cb2896b2ae..730ade48329b 100644
> > --- a/arch/powerpc/kernel/cpu_setup_power.S
> > +++ b/arch/powerpc/kernel/cpu_setup_power.S
> > @@ -31,7 +31,6 @@ _GLOBAL(__setup_cpu_power7)
> >  	mfspr	r3,SPRN_LPCR
> >  	li	r4,(LPCR_LPES1 >> LPCR_LPES_SH)
> >  	bl	__init_LPCR_ISA206
> > -	bl	__init_tlb_power7
> >  	mtlr	r11
> >  	blr
> > 
> > @@ -45,7 +44,6 @@ _GLOBAL(__restore_cpu_power7)
> >  	mfspr	r3,SPRN_LPCR
> >  	li	r4,(LPCR_LPES1 >> LPCR_LPES_SH)
> >  	bl	__init_LPCR_ISA206
> > -	bl	__init_tlb_power7
> >  	mtlr	r11
> >  	blr
> > 
> > @@ -64,7 +62,6 @@ _GLOBAL(__setup_cpu_power8)
> >  	li	r4,0 /* LPES = 0 */
> >  	bl	__init_LPCR_ISA206
> >  	bl	__init_HFSCR
> > -	bl	__init_tlb_power8
> >  	bl	__init_PMU_HV
> >  	bl	__init_PMU_HV_ISA207
> >  	mtlr	r11
> > @@ -86,7 +83,6 @@ _GLOBAL(__restore_cpu_power8)
> >  	li	r4,0 /* LPES = 0 */
> >  	bl	__init_LPCR_ISA206
> >  	bl	__init_HFSCR
> > -	bl	__init_tlb_power8
> >  	bl	__init_PMU_HV
> >  	bl	__init_PMU_HV_ISA207
> >  	mtlr	r11
> > @@ -110,7 +106,6 @@ _GLOBAL(__setup_cpu_power9)
> >  	li	r4,0 /* LPES = 0 */
> >  	bl	__init_LPCR_ISA300
> >  	bl	__init_HFSCR
> > -	bl	__init_tlb_power9
> >  	bl	__init_PMU_HV
> >  	mtlr	r11
> >  	blr
> > @@ -134,7 +129,6 @@ _GLOBAL(__restore_cpu_power9)
> >  	li	r4,0 /* LPES = 0 */
> >  	bl	__init_LPCR_ISA300
> >  	bl	__init_HFSCR
> > -	bl	__init_tlb_power9
> >  	bl	__init_PMU_HV
> >  	mtlr	r11
> >  	blr
> > @@ -192,43 +186,6 @@ __init_HFSCR:
> >  	mtspr	SPRN_HFSCR,r3
> >  	blr
> > 
> > -/*
> > - * Clear the TLB using the specified IS form of tlbiel instruction
> > - * (invalidate by congruence class). P7 has 128 CCs., P8 has 512.
> > - */
> > -__init_tlb_power7:
> > -	li	r6,POWER7_TLB_SETS
> > -	mtctr	r6
> > -	li	r7,0xc00	/* IS field = 0b11 */
> > -	ptesync
> > -2:	tlbiel	r7
> > -	addi	r7,r7,0x1000
> > -	bdnz	2b
> > -	ptesync
> > -1:	blr
> > -
> > -__init_tlb_power8:
> > -	li	r6,POWER8_TLB_SETS
> > -	mtctr	r6
> > -	li	r7,0xc00	/* IS field = 0b11 */
> > -	ptesync
> > -2:	tlbiel	r7
> > -	addi	r7,r7,0x1000
> > -	bdnz	2b
> > -	ptesync
> > -1:	blr
> > -
> > -__init_tlb_power9:
> > -	li	r6,POWER9_TLB_SETS_HASH
> > -	mtctr	r6
> > -	li	r7,0xc00	/* IS field = 0b11 */
> > -	ptesync
> > -2:	tlbiel	r7
> > -	addi	r7,r7,0x1000
> > -	bdnz	2b
> > -	ptesync
> > -1:	blr
> > -
> >  __init_PMU_HV:
> >  	li	r5,0
> >  	mtspr	SPRN_MMCRC,r5
> > diff --git a/arch/powerpc/kernel/cputable.c b/arch/powerpc/kernel/cputable.c
> > index 6f849832a669..d0a3eea6365d 100644
> > --- a/arch/powerpc/kernel/cputable.c
> > +++ b/arch/powerpc/kernel/cputable.c
> > @@ -74,9 +74,6 @@ extern void __setup_cpu_power8(unsigned long offset, struct cpu_spec* spec);
> >  extern void __restore_cpu_power8(void);
> >  extern void __setup_cpu_power9(unsigned long offset, struct cpu_spec* spec);
> >  extern void __restore_cpu_power9(void);
> > -extern void __flush_tlb_power7(unsigned int action);
> > -extern void __flush_tlb_power8(unsigned int action);
> > -extern void __flush_tlb_power9(unsigned int action);
> >  extern long __machine_check_early_realmode_p7(struct pt_regs *regs);
> >  extern long __machine_check_early_realmode_p8(struct pt_regs *regs);
> >  extern long __machine_check_early_realmode_p9(struct pt_regs *regs);
> > @@ -368,7 +365,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_cpu_type	= "ppc64/ibm-compat-v1",
> >  		.cpu_setup		= __setup_cpu_power7,
> >  		.cpu_restore		= __restore_cpu_power7,
> > -		.flush_tlb		= __flush_tlb_power7,
> >  		.machine_check_early	= __machine_check_early_realmode_p7,
> >  		.platform		= "power7",
> >  	},
> > @@ -386,7 +382,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_cpu_type	= "ppc64/ibm-compat-v1",
> >  		.cpu_setup		= __setup_cpu_power8,
> >  		.cpu_restore		= __restore_cpu_power8,
> > -		.flush_tlb		= __flush_tlb_power8,
> >  		.machine_check_early	= __machine_check_early_realmode_p8,
> >  		.platform		= "power8",
> >  	},
> > @@ -404,7 +399,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_cpu_type	= "ppc64/ibm-compat-v1",
> >  		.cpu_setup		= __setup_cpu_power9,
> >  		.cpu_restore		= __restore_cpu_power9,
> > -		.flush_tlb		= __flush_tlb_power9,
> >  		.platform		= "power9",
> >  	},
> >  	{	/* Power7 */
> > @@ -423,7 +417,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_POWER4,
> >  		.cpu_setup		= __setup_cpu_power7,
> >  		.cpu_restore		= __restore_cpu_power7,
> > -		.flush_tlb		= __flush_tlb_power7,
> >  		.machine_check_early	= __machine_check_early_realmode_p7,
> >  		.platform		= "power7",
> >  	},
> > @@ -443,7 +436,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_POWER4,
> >  		.cpu_setup		= __setup_cpu_power7,
> >  		.cpu_restore		= __restore_cpu_power7,
> > -		.flush_tlb		= __flush_tlb_power7,
> >  		.machine_check_early	= __machine_check_early_realmode_p7,
> >  		.platform		= "power7+",
> >  	},
> > @@ -463,7 +455,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power8,
> >  		.cpu_restore		= __restore_cpu_power8,
> > -		.flush_tlb		= __flush_tlb_power8,
> >  		.machine_check_early	= __machine_check_early_realmode_p8,
> >  		.platform		= "power8",
> >  	},
> > @@ -483,7 +474,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power8,
> >  		.cpu_restore		= __restore_cpu_power8,
> > -		.flush_tlb		= __flush_tlb_power8,
> >  		.machine_check_early	= __machine_check_early_realmode_p8,
> >  		.platform		= "power8",
> >  	},
> > @@ -503,7 +493,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power8,
> >  		.cpu_restore		= __restore_cpu_power8,
> > -		.flush_tlb		= __flush_tlb_power8,
> >  		.machine_check_early	= __machine_check_early_realmode_p8,
> >  		.platform		= "power8",
> >  	},
> > @@ -523,7 +512,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power8,
> >  		.cpu_restore		= __restore_cpu_power8,
> > -		.flush_tlb		= __flush_tlb_power8,
> >  		.machine_check_early	= __machine_check_early_realmode_p8,
> >  		.platform		= "power8",
> >  	},
> > @@ -543,7 +531,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power9,
> >  		.cpu_restore		= __restore_cpu_power9,
> > -		.flush_tlb		= __flush_tlb_power9,
> >  		.machine_check_early	= __machine_check_early_realmode_p9,
> >  		.platform		= "power9",
> >  	},
> > @@ -563,7 +550,6 @@ static struct cpu_spec __initdata cpu_specs[] = {
> >  		.oprofile_type		= PPC_OPROFILE_INVALID,
> >  		.cpu_setup		= __setup_cpu_power9,
> >  		.cpu_restore		= __restore_cpu_power9,
> > -		.flush_tlb		= __flush_tlb_power9,
> >  		.machine_check_early	= __machine_check_early_realmode_p9,
> >  		.platform		= "power9",
> >  	},
> > diff --git a/arch/powerpc/kernel/dt_cpu_ftrs.c b/arch/powerpc/kernel/dt_cpu_ftrs.c
> > index fcc7588a96d6..030448914a5d 100644
> > --- a/arch/powerpc/kernel/dt_cpu_ftrs.c
> > +++ b/arch/powerpc/kernel/dt_cpu_ftrs.c
> > @@ -76,8 +76,6 @@ struct dt_cpu_feature {
> >   * Set up the base CPU
> >   */
> > 
> > -extern void __flush_tlb_power8(unsigned int action);
> > -extern void __flush_tlb_power9(unsigned int action);
> >  extern long __machine_check_early_realmode_p8(struct pt_regs *regs);
> >  extern long __machine_check_early_realmode_p9(struct pt_regs *regs);
> > 
> > @@ -91,39 +89,6 @@ static struct {
> > 
> >  static void (*init_pmu_registers)(void);
> > 
> > -static void cpufeatures_flush_tlb(void)
> > -{
> > -	unsigned long rb;
> > -	unsigned int i, num_sets;
> > -
> > -	/*
> > -	 * This is a temporary measure to keep equivalent TLB flush as the
> > -	 * cputable based setup code.
> > -	 */
> > -	switch (PVR_VER(mfspr(SPRN_PVR))) {
> > -	case PVR_POWER8:
> > -	case PVR_POWER8E:
> > -	case PVR_POWER8NVL:
> > -		num_sets = POWER8_TLB_SETS;
> > -		break;
> > -	case PVR_POWER9:
> > -		num_sets = POWER9_TLB_SETS_HASH;
> > -		break;
> > -	default:
> > -		num_sets = 1;
> > -		pr_err("unknown CPU version for boot TLB flush\n");
> > -		break;
> > -	}
> > -
> > -	asm volatile("ptesync" : : : "memory");
> > -	rb = TLBIEL_INVAL_SET;
> > -	for (i = 0; i < num_sets; i++) {
> > -		asm volatile("tlbiel %0" : : "r" (rb));
> > -		rb += 1 << TLBIEL_INVAL_SET_SHIFT;
> > -	}
> > -	asm volatile("ptesync" : : : "memory");
> > -}
> > -
> >  static void __restore_cpu_cpufeatures(void)
> >  {
> >  	/*
> > @@ -148,8 +113,6 @@ static void __restore_cpu_cpufeatures(void)
> > 
> >  	if (init_pmu_registers)
> >  		init_pmu_registers();
> > -
> > -	cpufeatures_flush_tlb();
> >  }
> > 
> >  static char dt_cpu_name[64];
> > @@ -168,7 +131,6 @@ static struct cpu_spec __initdata base_cpu_spec = {
> >  	.oprofile_type		= PPC_OPROFILE_INVALID,
> >  	.cpu_setup		= NULL,
> >  	.cpu_restore		= __restore_cpu_cpufeatures,
> > -	.flush_tlb		= NULL,
> >  	.machine_check_early	= NULL,
> >  	.platform		= NULL,
> >  };
> > @@ -423,7 +385,6 @@ static void init_pmu_power8(void)
> >  static int __init feat_enable_mce_power8(struct dt_cpu_feature *f)
> >  {
> >  	cur_cpu_spec->platform = "power8";
> > -	cur_cpu_spec->flush_tlb = __flush_tlb_power8;
> >  	cur_cpu_spec->machine_check_early = __machine_check_early_realmode_p8;
> > 
> >  	return 1;
> > @@ -462,7 +423,6 @@ static void init_pmu_power9(void)
> >  static int __init feat_enable_mce_power9(struct dt_cpu_feature *f)
> >  {
> >  	cur_cpu_spec->platform = "power9";
> > -	cur_cpu_spec->flush_tlb = __flush_tlb_power9;
> >  	cur_cpu_spec->machine_check_early = __machine_check_early_realmode_p9;
> > 
> >  	return 1;
> > @@ -750,8 +710,6 @@ static void __init cpufeatures_setup_finished(void)
> >  	system_registers.hfscr = mfspr(SPRN_HFSCR);
> >  	system_registers.fscr = mfspr(SPRN_FSCR);
> > 
> > -	cpufeatures_flush_tlb();
> > -
> >  	pr_info("final cpu/mmu features = 0x%016lx 0x%08x\n",
> >  		cur_cpu_spec->cpu_features, cur_cpu_spec->mmu_features);
> >  }
> > diff --git a/arch/powerpc/kernel/mce_power.c b/arch/powerpc/kernel/mce_power.c
> > index f913139bb0c2..840f5e0e41f9 100644
> > --- a/arch/powerpc/kernel/mce_power.c
> > +++ b/arch/powerpc/kernel/mce_power.c
> > @@ -28,61 +28,6 @@
> >  #include <asm/mce.h>
> >  #include <asm/machdep.h>
> > 
> > -static void flush_tlb_206(unsigned int num_sets, unsigned int action)
> > -{
> > -	unsigned long rb;
> > -	unsigned int i;
> > -
> > -	switch (action) {
> > -	case TLB_INVAL_SCOPE_GLOBAL:
> > -		rb = TLBIEL_INVAL_SET;
> > -		break;
> > -	case TLB_INVAL_SCOPE_LPID:
> > -		rb = TLBIEL_INVAL_SET_LPID;
> > -		break;
> > -	default:
> > -		BUG();
> > -		break;
> > -	}
> > -
> > -	asm volatile("ptesync" : : : "memory");
> > -	for (i = 0; i < num_sets; i++) {
> > -		asm volatile("tlbiel %0" : : "r" (rb));
> > -		rb += 1 << TLBIEL_INVAL_SET_SHIFT;
> > -	}
> > -	asm volatile("ptesync" : : : "memory");
> > -}
> > -
> > -/*
> > - * Generic routines to flush TLB on POWER processors. These routines
> > - * are used as flush_tlb hook in the cpu_spec.
> > - *
> > - * action => TLB_INVAL_SCOPE_GLOBAL:  Invalidate all TLBs.
> > - *	     TLB_INVAL_SCOPE_LPID: Invalidate TLB for current LPID.
> > - */
> > -void __flush_tlb_power7(unsigned int action)
> > -{
> > -	flush_tlb_206(POWER7_TLB_SETS, action);
> > -}
> > -
> > -void __flush_tlb_power8(unsigned int action)
> > -{
> > -	flush_tlb_206(POWER8_TLB_SETS, action);
> > -}
> > -
> > -void __flush_tlb_power9(unsigned int action)
> > -{
> > -	unsigned int num_sets;
> > -
> > -	if (radix_enabled())
> > -		num_sets = POWER9_TLB_SETS_RADIX;
> > -	else
> > -		num_sets = POWER9_TLB_SETS_HASH;
> > -
> > -	flush_tlb_206(num_sets, action);
> > -}
> > -
> > -
> >  /* flush SLBs and reload */
> >  #ifdef CONFIG_PPC_STD_MMU_64
> >  static void flush_and_reload_slb(void)
> > @@ -142,10 +87,8 @@ static int mce_flush(int what)
> >  		return 1;
> >  	}
> >  	if (what == MCE_FLUSH_TLB) {
> > -		if (cur_cpu_spec && cur_cpu_spec->flush_tlb) {
> > -			cur_cpu_spec->flush_tlb(TLB_INVAL_SCOPE_GLOBAL);
> > -			return 1;
> > -		}
> > +		tlbiel_all();
> > +		return 1;
> >  	}
> > 
> >  	return 0;
> > diff --git a/arch/powerpc/kvm/book3s_hv_ras.c b/arch/powerpc/kvm/book3s_hv_ras.c
> > index 7ef0993214f3..9a8bf0e13064 100644
> > --- a/arch/powerpc/kvm/book3s_hv_ras.c
> > +++ b/arch/powerpc/kvm/book3s_hv_ras.c
> > @@ -87,8 +87,7 @@ static long kvmppc_realmode_mc_power7(struct kvm_vcpu *vcpu)
> >  				   DSISR_MC_SLB_PARITY | DSISR_MC_DERAT_MULTI);
> >  		}
> >  		if (dsisr & DSISR_MC_TLB_MULTI) {
> > -			if (cur_cpu_spec && cur_cpu_spec->flush_tlb)
> > -				cur_cpu_spec->flush_tlb(TLB_INVAL_SCOPE_LPID);
> > +			tlbiel_all_lpid(vcpu->kvm->arch.radix);
> >  			dsisr &= ~DSISR_MC_TLB_MULTI;
> >  		}
> >  		/* Any other errors we don't understand? */
> > @@ -105,8 +104,7 @@ static long kvmppc_realmode_mc_power7(struct kvm_vcpu *vcpu)
> >  		reload_slb(vcpu);
> >  		break;
> >  	case SRR1_MC_IFETCH_TLBMULTI:
> > -		if (cur_cpu_spec && cur_cpu_spec->flush_tlb)
> > -			cur_cpu_spec->flush_tlb(TLB_INVAL_SCOPE_LPID);
> > +		tlbiel_all_lpid(vcpu->kvm->arch.radix);
> >  		break;
> >  	default:
> >  		handled = 0;
> > diff --git a/arch/powerpc/mm/hash_native_64.c b/arch/powerpc/mm/hash_native_64.c
> > index 65bb8f33b399..5e79c04db4fa 100644
> > --- a/arch/powerpc/mm/hash_native_64.c
> > +++ b/arch/powerpc/mm/hash_native_64.c
> > @@ -45,6 +45,88 @@
> > 
> >  DEFINE_RAW_SPINLOCK(native_tlbie_lock);
> > 
> > +static inline void __tlbiel_all_isa206(unsigned int set, unsigned int is)
> > +{
> > +	unsigned long rb;
> > +
> > +	rb = (set << PPC_BITLSHIFT(51)) | (is << PPC_BITLSHIFT(53));
> > +
> > +	asm volatile("tlbiel %0" : : "r" (rb));
> > +}
> > +
> > +static inline void __tlbiel_all_isa300(unsigned int set, unsigned int is,
> > +					unsigned int ric, unsigned int prs)
> > +{
> > +	unsigned int r = 0; /* hash format */
> > +	unsigned long rb;
> > +	unsigned long rs = 0;
> > +
> > +	rb = (set << PPC_BITLSHIFT(51)) | (is << PPC_BITLSHIFT(53));
> > +
> > +	asm volatile(PPC_TLBIEL(%0, %4, %3, %2, %1)
> > +		     : : "r"(rb), "i"(r), "i"(prs), "i"(ric), "r"(rs) : "memory");
> > +}
> > +
> > +static void tlbiel_all_isa206(unsigned int num_sets, unsigned int is)
> > +{
> > +	unsigned int set;
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +
> > +	for (set = 0; set < num_sets; set++)
> > +		__tlbiel_all_isa206(set, is);
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +}
> > +
> > +static void tlbiel_all_isa300(unsigned int num_sets, unsigned int is)
> > +{
> > +	unsigned int set;
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +
> > +	/*
> > +	 * Flush the first set of the TLB, and any caching of partition table
> > +	 * entries. Then flush the remaining sets of the TLB. Hash mode uses
> > +	 * partition scoped TLB translations.
> > +	 */
> > +	__tlbiel_all_isa300(0, is, 2, 0);
> > +	for (set = 1; set < num_sets; set++)
> > +		__tlbiel_all_isa300(set, is, 0, 0);
> > +
> > +	/* Flush process table entries */
> > +	__tlbiel_all_isa300(0, is, 2, 1);
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +}
> > +
> > +void hash__tlbiel_all(unsigned int action)
> > +{
> > +	unsigned int is;
> > +
> > +	switch (action) {
> > +	case TLB_INVAL_SCOPE_GLOBAL:
> > +		is = 3;
> > +		break;
> > +	case TLB_INVAL_SCOPE_LPID:
> > +		is = 2;
> > +		break;
> > +	default:
> > +		BUG();
> > +	}
> > +
> > +	if (cpu_has_feature(CPU_FTR_ARCH_300))
> > +		tlbiel_all_isa300(POWER9_TLB_SETS_HASH, is);
> > +	else if (cpu_has_feature(CPU_FTR_ARCH_207S))
> > +		tlbiel_all_isa206(POWER8_TLB_SETS, is);
> > +	else if (cpu_has_feature(CPU_FTR_ARCH_206))
> > +		tlbiel_all_isa206(POWER7_TLB_SETS, is);
> > +	else
> > +		WARN(1, "%s called on pre-POWER7 CPU\n", __func__);
> > +
> > +	asm volatile(PPC_INVALIDATE_ERAT "; isync" : : :"memory");
> > +}
> > +
> >  static inline void __tlbie(unsigned long vpn, int psize, int apsize, int ssize)
> >  {
> >  	unsigned long va;
> > diff --git a/arch/powerpc/mm/hash_utils_64.c b/arch/powerpc/mm/hash_utils_64.c
> > index f2095ce9d4b0..abe3db5ab554 100644
> > --- a/arch/powerpc/mm/hash_utils_64.c
> > +++ b/arch/powerpc/mm/hash_utils_64.c
> > @@ -1044,6 +1044,8 @@ void __init hash__early_init_mmu(void)
> >  	pr_info("Initializing hash mmu with SLB\n");
> >  	/* Initialize SLB management */
> >  	slb_initialize();
> > +
> > +	tlbiel_all();
> >  }
> > 
> >  #ifdef CONFIG_SMP
> > @@ -1063,6 +1065,8 @@ void hash__early_init_mmu_secondary(void)
> >  	}
> >  	/* Initialize SLB */
> >  	slb_initialize();
> > +
> > +	tlbiel_all();
> >  }
> >  #endif /* CONFIG_SMP */
> > 
> > diff --git a/arch/powerpc/mm/pgtable-radix.c b/arch/powerpc/mm/pgtable-radix.c
> > index c28165d8970b..a326904ca4e2 100644
> > --- a/arch/powerpc/mm/pgtable-radix.c
> > +++ b/arch/powerpc/mm/pgtable-radix.c
> > @@ -426,6 +426,8 @@ void __init radix__early_init_mmu(void)
> > 
> >  	radix_init_iamr();
> >  	radix_init_pgtable();
> > +
> > +	tlbiel_all();
> >  }
> > 
> >  void radix__early_init_mmu_secondary(void)
> > @@ -447,6 +449,8 @@ void radix__early_init_mmu_secondary(void)
> >  		radix_init_amor();
> >  	}
> >  	radix_init_iamr();
> > +
> > +	tlbiel_all();
> >  }
> > 
> >  void radix__mmu_cleanup_all(void)
> > diff --git a/arch/powerpc/mm/tlb-radix.c b/arch/powerpc/mm/tlb-radix.c
> > index 02e71402fdd3..63c12c784e25 100644
> > --- a/arch/powerpc/mm/tlb-radix.c
> > +++ b/arch/powerpc/mm/tlb-radix.c
> > @@ -22,6 +22,63 @@
> >  #define RIC_FLUSH_PWC 1
> >  #define RIC_FLUSH_ALL 2
> > 
> > +static inline void __tlbiel_all_isa300(unsigned int set, unsigned int is,
> > +					unsigned int ric, unsigned int prs)
> > +{
> > +	unsigned int r = 1; /* radix format */
> > +	unsigned long rb;
> > +	unsigned long rs = 0;
> > +
> > +	rb = (set << PPC_BITLSHIFT(51)) | (is << PPC_BITLSHIFT(53));
> > +
> > +	asm volatile(PPC_TLBIEL(%0, %4, %3, %2, %1)
> > +		     : : "r"(rb), "i"(r), "i"(prs), "i"(ric), "r"(rs) : "memory");
> > +}
> > +
> > +static void tlbiel_all_isa300(unsigned int num_sets, unsigned int is)
> > +{
> > +	unsigned int set;
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +
> > +	/*
> > +	 * Flush the first set of the TLB, and the entire Page Walk Cache.
> > +	 * Then flush the remaining sets of the TLB.
> > +	 */
> > +	__tlbiel_all_isa300(0, is, RIC_FLUSH_ALL, 1);
> > +	for (set = 1; set < num_sets; set++)
> > +		__tlbiel_all_isa300(set, is, RIC_FLUSH_TLB, 1);
> > +
> > +	/* Do the same for partitioned scoped entries. */
> > +	__tlbiel_all_isa300(0, is, RIC_FLUSH_ALL, 0);
> > +	for (set = 1; set < num_sets; set++)
> > +		__tlbiel_all_isa300(set, is, RIC_FLUSH_TLB, 0);
> > +
> > +	asm volatile("ptesync": : :"memory");
> > +}
> > +
> > +void radix__tlbiel_all(unsigned int action)
> > +{
> > +	unsigned int is;
> > +
> > +	switch (action) {
> > +	case TLB_INVAL_SCOPE_GLOBAL:
> > +		is = 3;
> > +		break;
> > +	case TLB_INVAL_SCOPE_LPID:
> > +		is = 2;
> > +		break;
> > +	default:
> > +		BUG();
> > +	}
> > +
> > +	if (cpu_has_feature(CPU_FTR_ARCH_300))
> > +		tlbiel_all_isa300(POWER9_TLB_SETS_RADIX, is);
> > +	else
> > +		WARN(1, "%s called on pre-POWER9 CPU\n", __func__);
> > +	asm volatile(PPC_INVALIDATE_ERAT "; isync" : : :"memory");
> > +}
> > +
> >  static inline void __tlbiel_pid(unsigned long pid, int set,
> >  				unsigned long ric)
> >  {
> > -- 
> > 2.11.0


More information about the Linuxppc-dev mailing list