|  | // SPDX-License-Identifier: GPL-2.0-or-later | 
|  | /* | 
|  | * Copyright 2020, Jordan Niethe, IBM Corporation. | 
|  | * | 
|  | * This file contains low level CPU setup functions. | 
|  | * Originally written in assembly by Benjamin Herrenschmidt & various other | 
|  | * authors. | 
|  | */ | 
|  |  | 
|  | #include <asm/reg.h> | 
|  | #include <asm/synch.h> | 
|  | #include <linux/bitops.h> | 
|  | #include <asm/cputable.h> | 
|  | #include <asm/cpu_setup.h> | 
|  |  | 
|  | /* Disable CPU_FTR_HVMODE and return false if MSR:HV is not set */ | 
|  | static bool init_hvmode_206(struct cpu_spec *t) | 
|  | { | 
|  | u64 msr; | 
|  |  | 
|  | msr = mfmsr(); | 
|  | if (msr & MSR_HV) | 
|  | return true; | 
|  |  | 
|  | t->cpu_features &= ~(CPU_FTR_HVMODE | CPU_FTR_P9_TM_HV_ASSIST); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | static void init_LPCR_ISA300(u64 lpcr, u64 lpes) | 
|  | { | 
|  | /* POWER9 has no VRMASD */ | 
|  | lpcr |= (lpes << LPCR_LPES_SH) & LPCR_LPES; | 
|  | lpcr |= LPCR_PECE0|LPCR_PECE1|LPCR_PECE2; | 
|  | lpcr |= (4ull << LPCR_DPFD_SH) & LPCR_DPFD; | 
|  | lpcr &= ~LPCR_HDICE;	/* clear HDICE */ | 
|  | lpcr |= (4ull << LPCR_VC_SH); | 
|  | mtspr(SPRN_LPCR, lpcr); | 
|  | isync(); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Setup a sane LPCR: | 
|  | *   Called with initial LPCR and desired LPES 2-bit value | 
|  | * | 
|  | *   LPES = 0b01 (HSRR0/1 used for 0x500) | 
|  | *   PECE = 0b111 | 
|  | *   DPFD = 4 | 
|  | *   HDICE = 0 | 
|  | *   VC = 0b100 (VPM0=1, VPM1=0, ISL=0) | 
|  | *   VRMASD = 0b10000 (L=1, LP=00) | 
|  | * | 
|  | * Other bits untouched for now | 
|  | */ | 
|  | static void init_LPCR_ISA206(u64 lpcr, u64 lpes) | 
|  | { | 
|  | lpcr |= (0x10ull << LPCR_VRMASD_SH) & LPCR_VRMASD; | 
|  | init_LPCR_ISA300(lpcr, lpes); | 
|  | } | 
|  |  | 
|  | static void init_FSCR(void) | 
|  | { | 
|  | u64 fscr; | 
|  |  | 
|  | fscr = mfspr(SPRN_FSCR); | 
|  | fscr |= FSCR_TAR|FSCR_EBB; | 
|  | mtspr(SPRN_FSCR, fscr); | 
|  | } | 
|  |  | 
|  | static void init_FSCR_power9(void) | 
|  | { | 
|  | u64 fscr; | 
|  |  | 
|  | fscr = mfspr(SPRN_FSCR); | 
|  | fscr |= FSCR_SCV; | 
|  | mtspr(SPRN_FSCR, fscr); | 
|  | init_FSCR(); | 
|  | } | 
|  |  | 
|  | static void init_FSCR_power10(void) | 
|  | { | 
|  | u64 fscr; | 
|  |  | 
|  | fscr = mfspr(SPRN_FSCR); | 
|  | fscr |= FSCR_PREFIX; | 
|  | mtspr(SPRN_FSCR, fscr); | 
|  | init_FSCR_power9(); | 
|  | } | 
|  |  | 
|  | static void init_HFSCR(void) | 
|  | { | 
|  | u64 hfscr; | 
|  |  | 
|  | hfscr = mfspr(SPRN_HFSCR); | 
|  | hfscr |= HFSCR_TAR|HFSCR_TM|HFSCR_BHRB|HFSCR_PM|HFSCR_DSCR|\ | 
|  | HFSCR_VECVSX|HFSCR_FP|HFSCR_EBB|HFSCR_MSGP; | 
|  | mtspr(SPRN_HFSCR, hfscr); | 
|  | } | 
|  |  | 
|  | static void init_PMU_HV(void) | 
|  | { | 
|  | mtspr(SPRN_MMCRC, 0); | 
|  | } | 
|  |  | 
|  | static void init_PMU_HV_ISA207(void) | 
|  | { | 
|  | mtspr(SPRN_MMCRH, 0); | 
|  | } | 
|  |  | 
|  | static void init_PMU(void) | 
|  | { | 
|  | mtspr(SPRN_MMCRA, 0); | 
|  | mtspr(SPRN_MMCR0, MMCR0_FC); | 
|  | mtspr(SPRN_MMCR1, 0); | 
|  | mtspr(SPRN_MMCR2, 0); | 
|  | } | 
|  |  | 
|  | static void init_PMU_ISA207(void) | 
|  | { | 
|  | mtspr(SPRN_MMCRS, 0); | 
|  | } | 
|  |  | 
|  | static void init_PMU_ISA31(void) | 
|  | { | 
|  | mtspr(SPRN_MMCR3, 0); | 
|  | mtspr(SPRN_MMCRA, MMCRA_BHRB_DISABLE); | 
|  | mtspr(SPRN_MMCR0, MMCR0_FC | MMCR0_PMCCEXT); | 
|  | } | 
|  |  | 
|  | static void init_DEXCR(void) | 
|  | { | 
|  | mtspr(SPRN_DEXCR, DEXCR_INIT); | 
|  | mtspr(SPRN_HASHKEYR, 0); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Note that we can be called twice of pseudo-PVRs. | 
|  | * The parameter offset is not used. | 
|  | */ | 
|  |  | 
|  | void __setup_cpu_power7(unsigned long offset, struct cpu_spec *t) | 
|  | { | 
|  | if (!init_hvmode_206(t)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA206(mfspr(SPRN_LPCR), LPCR_LPES1 >> LPCR_LPES_SH); | 
|  | } | 
|  |  | 
|  | void __restore_cpu_power7(void) | 
|  | { | 
|  | u64 msr; | 
|  |  | 
|  | msr = mfmsr(); | 
|  | if (!(msr & MSR_HV)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA206(mfspr(SPRN_LPCR), LPCR_LPES1 >> LPCR_LPES_SH); | 
|  | } | 
|  |  | 
|  | void __setup_cpu_power8(unsigned long offset, struct cpu_spec *t) | 
|  | { | 
|  | init_FSCR(); | 
|  | init_PMU(); | 
|  | init_PMU_ISA207(); | 
|  |  | 
|  | if (!init_hvmode_206(t)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA206(mfspr(SPRN_LPCR) | LPCR_PECEDH, 0); /* LPES = 0 */ | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | init_PMU_HV_ISA207(); | 
|  | } | 
|  |  | 
|  | void __restore_cpu_power8(void) | 
|  | { | 
|  | u64 msr; | 
|  |  | 
|  | init_FSCR(); | 
|  | init_PMU(); | 
|  | init_PMU_ISA207(); | 
|  |  | 
|  | msr = mfmsr(); | 
|  | if (!(msr & MSR_HV)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA206(mfspr(SPRN_LPCR) | LPCR_PECEDH, 0); /* LPES = 0 */ | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | init_PMU_HV_ISA207(); | 
|  | } | 
|  |  | 
|  | void __setup_cpu_power9(unsigned long offset, struct cpu_spec *t) | 
|  | { | 
|  | init_FSCR_power9(); | 
|  | init_PMU(); | 
|  |  | 
|  | if (!init_hvmode_206(t)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_PSSCR, 0); | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_PID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA300((mfspr(SPRN_LPCR) | LPCR_PECEDH | LPCR_PECE_HVEE |\ | 
|  | LPCR_HVICE | LPCR_HEIC) & ~(LPCR_UPRT | LPCR_HR), 0); | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | } | 
|  |  | 
|  | void __restore_cpu_power9(void) | 
|  | { | 
|  | u64 msr; | 
|  |  | 
|  | init_FSCR_power9(); | 
|  | init_PMU(); | 
|  |  | 
|  | msr = mfmsr(); | 
|  | if (!(msr & MSR_HV)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_PSSCR, 0); | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_PID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA300((mfspr(SPRN_LPCR) | LPCR_PECEDH | LPCR_PECE_HVEE |\ | 
|  | LPCR_HVICE | LPCR_HEIC) & ~(LPCR_UPRT | LPCR_HR), 0); | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | } | 
|  |  | 
|  | void __setup_cpu_power10(unsigned long offset, struct cpu_spec *t) | 
|  | { | 
|  | init_FSCR_power10(); | 
|  | init_PMU(); | 
|  | init_PMU_ISA31(); | 
|  | init_DEXCR(); | 
|  |  | 
|  | if (!init_hvmode_206(t)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_PSSCR, 0); | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_PID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA300((mfspr(SPRN_LPCR) | LPCR_PECEDH | LPCR_PECE_HVEE |\ | 
|  | LPCR_HVICE | LPCR_HEIC) & ~(LPCR_UPRT | LPCR_HR), 0); | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | } | 
|  |  | 
|  | void __restore_cpu_power10(void) | 
|  | { | 
|  | u64 msr; | 
|  |  | 
|  | init_FSCR_power10(); | 
|  | init_PMU(); | 
|  | init_PMU_ISA31(); | 
|  | init_DEXCR(); | 
|  |  | 
|  | msr = mfmsr(); | 
|  | if (!(msr & MSR_HV)) | 
|  | return; | 
|  |  | 
|  | mtspr(SPRN_PSSCR, 0); | 
|  | mtspr(SPRN_LPID, 0); | 
|  | mtspr(SPRN_PID, 0); | 
|  | mtspr(SPRN_AMOR, ~0); | 
|  | mtspr(SPRN_PCR, PCR_MASK); | 
|  | init_LPCR_ISA300((mfspr(SPRN_LPCR) | LPCR_PECEDH | LPCR_PECE_HVEE |\ | 
|  | LPCR_HVICE | LPCR_HEIC) & ~(LPCR_UPRT | LPCR_HR), 0); | 
|  | init_HFSCR(); | 
|  | init_PMU_HV(); | 
|  | } |