Nicholas Piggin <npiggin@xxxxxxxxx> writes: > This reverts much of commit c01015091a770 ("KVM: PPC: Book3S HV: Run HPT > guests on POWER9 radix hosts"), which was required to run HPT guests on > RPT hosts on early POWER9 CPUs without support for "mixed mode", which > meant the host could not run with MMU on while guests were running. Would be worth mentioning which CPU versions. Looking at the code it seems like it's P9N < 2.2 and P9C < 1.1. > This code has some corner case bugs, e.g., when the guest hits a machine > check or HMI the primary locks up waiting for secondaries to switch LPCR > to host, which they never do. This could all be fixed in software, but > most CPUs in production have mixed mode support, and those that don't > are believed to be all in installations that don't use this capability. > So simplify things and remove support. Key detail being, AFAICS, you retain enough code to detect that we're in that configuration and cleanly return an error, rather than crashing or anything horrible. Otherwise looks good to me. cheers > diff --git a/arch/powerpc/include/asm/kvm_book3s_asm.h b/arch/powerpc/include/asm/kvm_book3s_asm.h > index 078f4648ea27..b6d31bff5209 100644 > --- a/arch/powerpc/include/asm/kvm_book3s_asm.h > +++ b/arch/powerpc/include/asm/kvm_book3s_asm.h > @@ -74,16 +74,6 @@ struct kvm_split_mode { > u8 do_nap; > u8 napped[MAX_SMT_THREADS]; > struct kvmppc_vcore *vc[MAX_SUBCORES]; > - /* Bits for changing lpcr on P9 */ > - unsigned long lpcr_req; > - unsigned long lpidr_req; > - unsigned long host_lpcr; > - u32 do_set; > - u32 do_restore; > - union { > - u32 allphases; > - u8 phase[4]; > - } lpcr_sync; > }; > > /* > @@ -110,7 +100,6 @@ struct kvmppc_host_state { > u8 hwthread_state; > u8 host_ipi; > u8 ptid; /* thread number within subcore when split */ > - u8 tid; /* thread number within whole core */ > u8 fake_suspend; > struct kvm_vcpu *kvm_vcpu; > struct kvmppc_vcore *kvm_vcore; > diff --git a/arch/powerpc/kernel/asm-offsets.c b/arch/powerpc/kernel/asm-offsets.c > index c2722ff36e98..21496ea09bf1 100644 > --- a/arch/powerpc/kernel/asm-offsets.c > +++ b/arch/powerpc/kernel/asm-offsets.c > @@ -690,7 +690,6 @@ int main(void) > HSTATE_FIELD(HSTATE_SAVED_XIRR, saved_xirr); > HSTATE_FIELD(HSTATE_HOST_IPI, host_ipi); > HSTATE_FIELD(HSTATE_PTID, ptid); > - HSTATE_FIELD(HSTATE_TID, tid); > HSTATE_FIELD(HSTATE_FAKE_SUSPEND, fake_suspend); > HSTATE_FIELD(HSTATE_MMCR0, host_mmcr[0]); > HSTATE_FIELD(HSTATE_MMCR1, host_mmcr[1]); > @@ -720,8 +719,6 @@ int main(void) > OFFSET(KVM_SPLIT_LDBAR, kvm_split_mode, ldbar); > OFFSET(KVM_SPLIT_DO_NAP, kvm_split_mode, do_nap); > OFFSET(KVM_SPLIT_NAPPED, kvm_split_mode, napped); > - OFFSET(KVM_SPLIT_DO_SET, kvm_split_mode, do_set); > - OFFSET(KVM_SPLIT_DO_RESTORE, kvm_split_mode, do_restore); > #endif /* CONFIG_KVM_BOOK3S_HV_POSSIBLE */ > > #ifdef CONFIG_PPC_BOOK3S_64 > diff --git a/arch/powerpc/kvm/book3s_hv.c b/arch/powerpc/kvm/book3s_hv.c > index c94f9595133d..86b78f8e3dde 100644 > --- a/arch/powerpc/kvm/book3s_hv.c > +++ b/arch/powerpc/kvm/book3s_hv.c > @@ -134,7 +134,7 @@ static inline bool nesting_enabled(struct kvm *kvm) > } > > /* If set, the threads on each CPU core have to be in the same MMU mode */ > -static bool no_mixing_hpt_and_radix; > +static bool no_mixing_hpt_and_radix __read_mostly; > > static int kvmppc_hv_setup_htab_rma(struct kvm_vcpu *vcpu); > > @@ -2855,11 +2855,6 @@ static bool can_dynamic_split(struct kvmppc_vcore *vc, struct core_info *cip) > if (one_vm_per_core && vc->kvm != cip->vc[0]->kvm) > return false; > > - /* Some POWER9 chips require all threads to be in the same MMU mode */ > - if (no_mixing_hpt_and_radix && > - kvm_is_radix(vc->kvm) != kvm_is_radix(cip->vc[0]->kvm)) > - return false; > - > if (n_threads < cip->max_subcore_threads) > n_threads = cip->max_subcore_threads; > if (!subcore_config_ok(cip->n_subcores + 1, n_threads)) > @@ -2898,6 +2893,9 @@ static void prepare_threads(struct kvmppc_vcore *vc) > for_each_runnable_thread(i, vcpu, vc) { > if (signal_pending(vcpu->arch.run_task)) > vcpu->arch.ret = -EINTR; > + else if (no_mixing_hpt_and_radix && > + kvm_is_radix(vc->kvm) != radix_enabled()) > + vcpu->arch.ret = -EINVAL; > else if (vcpu->arch.vpa.update_pending || > vcpu->arch.slb_shadow.update_pending || > vcpu->arch.dtl.update_pending) > @@ -3103,7 +3101,6 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > int controlled_threads; > int trap; > bool is_power8; > - bool hpt_on_radix; > > /* > * Remove from the list any threads that have a signal pending > @@ -3136,11 +3133,8 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > * this is a HPT guest on a radix host machine where the > * CPU threads may not be in different MMU modes. > */ > - hpt_on_radix = no_mixing_hpt_and_radix && radix_enabled() && > - !kvm_is_radix(vc->kvm); > - if (((controlled_threads > 1) && > - ((vc->num_threads > threads_per_subcore) || !on_primary_thread())) || > - (hpt_on_radix && vc->kvm->arch.threads_indep)) { > + if ((controlled_threads > 1) && > + ((vc->num_threads > threads_per_subcore) || !on_primary_thread())) { > for_each_runnable_thread(i, vcpu, vc) { > vcpu->arch.ret = -EBUSY; > kvmppc_remove_runnable(vc, vcpu); > @@ -3208,7 +3202,7 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > is_power8 = cpu_has_feature(CPU_FTR_ARCH_207S) > && !cpu_has_feature(CPU_FTR_ARCH_300); > > - if (split > 1 || hpt_on_radix) { > + if (split > 1) { > sip = &split_info; > memset(&split_info, 0, sizeof(split_info)); > for (sub = 0; sub < core_info.n_subcores; ++sub) > @@ -3230,13 +3224,6 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > split_info.subcore_size = subcore_size; > } else { > split_info.subcore_size = 1; > - if (hpt_on_radix) { > - /* Use the split_info for LPCR/LPIDR changes */ > - split_info.lpcr_req = vc->lpcr; > - split_info.lpidr_req = vc->kvm->arch.lpid; > - split_info.host_lpcr = vc->kvm->arch.host_lpcr; > - split_info.do_set = 1; > - } > } > > /* order writes to split_info before kvm_split_mode pointer */ > @@ -3246,7 +3233,6 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > for (thr = 0; thr < controlled_threads; ++thr) { > struct paca_struct *paca = paca_ptrs[pcpu + thr]; > > - paca->kvm_hstate.tid = thr; > paca->kvm_hstate.napping = 0; > paca->kvm_hstate.kvm_split_mode = sip; > } > @@ -3320,10 +3306,8 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > * When doing micro-threading, poke the inactive threads as well. > * This gets them to the nap instruction after kvm_do_nap, > * which reduces the time taken to unsplit later. > - * For POWER9 HPT guest on radix host, we need all the secondary > - * threads woken up so they can do the LPCR/LPIDR change. > */ > - if (cmd_bit || hpt_on_radix) { > + if (cmd_bit) { > split_info.do_nap = 1; /* ask secondaries to nap when done */ > for (thr = 1; thr < threads_per_subcore; ++thr) > if (!(active & (1 << thr))) > @@ -3384,19 +3368,8 @@ static noinline void kvmppc_run_core(struct kvmppc_vcore *vc) > cpu_relax(); > ++loops; > } > - } else if (hpt_on_radix) { > - /* Wait for all threads to have seen final sync */ > - for (thr = 1; thr < controlled_threads; ++thr) { > - struct paca_struct *paca = paca_ptrs[pcpu + thr]; > - > - while (paca->kvm_hstate.kvm_split_mode) { > - HMT_low(); > - barrier(); > - } > - HMT_medium(); > - } > + split_info.do_nap = 0; > } > - split_info.do_nap = 0; > > kvmppc_set_host_core(pcpu); > > @@ -4166,7 +4139,6 @@ int kvmhv_run_single_vcpu(struct kvm_vcpu *vcpu, u64 time_limit, > > kvmppc_clear_host_core(pcpu); > > - local_paca->kvm_hstate.tid = 0; > local_paca->kvm_hstate.napping = 0; > local_paca->kvm_hstate.kvm_split_mode = NULL; > kvmppc_start_thread(vcpu, vc); > @@ -4351,15 +4323,11 @@ static int kvmppc_vcpu_run_hv(struct kvm_vcpu *vcpu) > > do { > /* > - * The early POWER9 chips that can't mix radix and HPT threads > - * on the same core also need the workaround for the problem > - * where the TLB would prefetch entries in the guest exit path > - * for radix guests using the guest PIDR value and LPID 0. > - * The workaround is in the old path (kvmppc_run_vcpu()) > - * but not the new path (kvmhv_run_single_vcpu()). > + * The TLB prefetch bug fixup is only in the kvmppc_run_vcpu > + * path, which also handles hash and dependent threads mode. > */ > if (kvm->arch.threads_indep && kvm_is_radix(kvm) && > - !no_mixing_hpt_and_radix) > + !cpu_has_feature(CPU_FTR_P9_RADIX_PREFETCH_BUG)) > r = kvmhv_run_single_vcpu(vcpu, ~(u64)0, > vcpu->arch.vcore->lpcr); > else > diff --git a/arch/powerpc/kvm/book3s_hv_builtin.c b/arch/powerpc/kvm/book3s_hv_builtin.c > index 8f58dd20b362..f3d3183249fe 100644 > --- a/arch/powerpc/kvm/book3s_hv_builtin.c > +++ b/arch/powerpc/kvm/book3s_hv_builtin.c > @@ -277,8 +277,7 @@ void kvmhv_commence_exit(int trap) > struct kvmppc_vcore *vc = local_paca->kvm_hstate.kvm_vcore; > int ptid = local_paca->kvm_hstate.ptid; > struct kvm_split_mode *sip = local_paca->kvm_hstate.kvm_split_mode; > - int me, ee, i, t; > - int cpu0; > + int me, ee, i; > > /* Set our bit in the threads-exiting-guest map in the 0xff00 > bits of vcore->entry_exit_map */ > @@ -320,22 +319,6 @@ void kvmhv_commence_exit(int trap) > if ((ee >> 8) == 0) > kvmhv_interrupt_vcore(vc, ee); > } > - > - /* > - * On POWER9 when running a HPT guest on a radix host (sip != NULL), > - * we have to interrupt inactive CPU threads to get them to > - * restore the host LPCR value. > - */ > - if (sip->lpcr_req) { > - if (cmpxchg(&sip->do_restore, 0, 1) == 0) { > - vc = local_paca->kvm_hstate.kvm_vcore; > - cpu0 = vc->pcpu + ptid - local_paca->kvm_hstate.tid; > - for (t = 1; t < threads_per_core; ++t) { > - if (sip->napped[t]) > - kvmhv_rm_send_ipi(cpu0 + t); > - } > - } > - } > } > > struct kvmppc_host_rm_ops *kvmppc_host_rm_ops_hv; > @@ -667,86 +650,6 @@ void kvmppc_bad_interrupt(struct pt_regs *regs) > panic("Bad KVM trap"); > } > > -/* > - * Functions used to switch LPCR HR and UPRT bits on all threads > - * when entering and exiting HPT guests on a radix host. > - */ > - > -#define PHASE_REALMODE 1 /* in real mode */ > -#define PHASE_SET_LPCR 2 /* have set LPCR */ > -#define PHASE_OUT_OF_GUEST 4 /* have finished executing in guest */ > -#define PHASE_RESET_LPCR 8 /* have reset LPCR to host value */ > - > -#define ALL(p) (((p) << 24) | ((p) << 16) | ((p) << 8) | (p)) > - > -static void wait_for_sync(struct kvm_split_mode *sip, int phase) > -{ > - int thr = local_paca->kvm_hstate.tid; > - > - sip->lpcr_sync.phase[thr] |= phase; > - phase = ALL(phase); > - while ((sip->lpcr_sync.allphases & phase) != phase) { > - HMT_low(); > - barrier(); > - } > - HMT_medium(); > -} > - > -void kvmhv_p9_set_lpcr(struct kvm_split_mode *sip) > -{ > - unsigned long rb, set; > - > - /* wait for every other thread to get to real mode */ > - wait_for_sync(sip, PHASE_REALMODE); > - > - /* Set LPCR and LPIDR */ > - mtspr(SPRN_LPCR, sip->lpcr_req); > - mtspr(SPRN_LPID, sip->lpidr_req); > - isync(); > - > - /* Invalidate the TLB on thread 0 */ > - if (local_paca->kvm_hstate.tid == 0) { > - sip->do_set = 0; > - asm volatile("ptesync" : : : "memory"); > - for (set = 0; set < POWER9_TLB_SETS_RADIX; ++set) { > - rb = TLBIEL_INVAL_SET_LPID + > - (set << TLBIEL_INVAL_SET_SHIFT); > - asm volatile(PPC_TLBIEL(%0, %1, 0, 0, 0) : : > - "r" (rb), "r" (0)); > - } > - asm volatile("ptesync" : : : "memory"); > - } > - > - /* indicate that we have done so and wait for others */ > - wait_for_sync(sip, PHASE_SET_LPCR); > - /* order read of sip->lpcr_sync.allphases vs. sip->do_set */ > - smp_rmb(); > -} > - > -/* > - * Called when a thread that has been in the guest needs > - * to reload the host LPCR value - but only on POWER9 when > - * running a HPT guest on a radix host. > - */ > -void kvmhv_p9_restore_lpcr(struct kvm_split_mode *sip) > -{ > - /* we're out of the guest... */ > - wait_for_sync(sip, PHASE_OUT_OF_GUEST); > - > - mtspr(SPRN_LPID, 0); > - mtspr(SPRN_LPCR, sip->host_lpcr); > - isync(); > - > - if (local_paca->kvm_hstate.tid == 0) { > - sip->do_restore = 0; > - smp_wmb(); /* order store of do_restore vs. phase */ > - } > - > - wait_for_sync(sip, PHASE_RESET_LPCR); > - smp_mb(); > - local_paca->kvm_hstate.kvm_split_mode = NULL; > -} > - > static void kvmppc_end_cede(struct kvm_vcpu *vcpu) > { > vcpu->arch.ceded = 0; > diff --git a/arch/powerpc/kvm/book3s_hv_rmhandlers.S b/arch/powerpc/kvm/book3s_hv_rmhandlers.S > index cd9995ee8441..d5a9b57ec129 100644 > --- a/arch/powerpc/kvm/book3s_hv_rmhandlers.S > +++ b/arch/powerpc/kvm/book3s_hv_rmhandlers.S > @@ -85,19 +85,6 @@ _GLOBAL_TOC(kvmppc_hv_entry_trampoline) > RFI_TO_KERNEL > > kvmppc_call_hv_entry: > -BEGIN_FTR_SECTION > - /* On P9, do LPCR setting, if necessary */ > - ld r3, HSTATE_SPLIT_MODE(r13) > - cmpdi r3, 0 > - beq 46f > - lwz r4, KVM_SPLIT_DO_SET(r3) > - cmpwi r4, 0 > - beq 46f > - bl kvmhv_p9_set_lpcr > - nop > -46: > -END_FTR_SECTION_IFSET(CPU_FTR_ARCH_300) > - > ld r4, HSTATE_KVM_VCPU(r13) > bl kvmppc_hv_entry > > @@ -361,11 +348,11 @@ kvm_secondary_got_guest: > LOAD_REG_ADDR(r6, decrementer_max) > ld r6, 0(r6) > mtspr SPRN_HDEC, r6 > +BEGIN_FTR_SECTION > /* and set per-LPAR registers, if doing dynamic micro-threading */ > ld r6, HSTATE_SPLIT_MODE(r13) > cmpdi r6, 0 > beq 63f > -BEGIN_FTR_SECTION > ld r0, KVM_SPLIT_RPR(r6) > mtspr SPRN_RPR, r0 > ld r0, KVM_SPLIT_PMMAR(r6) > @@ -373,16 +360,7 @@ BEGIN_FTR_SECTION > ld r0, KVM_SPLIT_LDBAR(r6) > mtspr SPRN_LDBAR, r0 > isync > -FTR_SECTION_ELSE > - /* On P9 we use the split_info for coordinating LPCR changes */ > - lwz r4, KVM_SPLIT_DO_SET(r6) > - cmpwi r4, 0 > - beq 1f > - mr r3, r6 > - bl kvmhv_p9_set_lpcr > - nop > -1: > -ALT_FTR_SECTION_END_IFCLR(CPU_FTR_ARCH_300) > +END_FTR_SECTION_IFCLR(CPU_FTR_ARCH_300) > 63: > /* Order load of vcpu after load of vcore */ > lwsync > @@ -452,19 +430,15 @@ kvm_no_guest: > mtcr r5 > blr > > -53: HMT_LOW > +53: > +BEGIN_FTR_SECTION > + HMT_LOW > ld r5, HSTATE_KVM_VCORE(r13) > cmpdi r5, 0 > bne 60f > ld r3, HSTATE_SPLIT_MODE(r13) > cmpdi r3, 0 > beq kvm_no_guest > - lwz r0, KVM_SPLIT_DO_SET(r3) > - cmpwi r0, 0 > - bne kvmhv_do_set > - lwz r0, KVM_SPLIT_DO_RESTORE(r3) > - cmpwi r0, 0 > - bne kvmhv_do_restore > lbz r0, KVM_SPLIT_DO_NAP(r3) > cmpwi r0, 0 > beq kvm_no_guest > @@ -472,24 +446,19 @@ kvm_no_guest: > b kvm_unsplit_nap > 60: HMT_MEDIUM > b kvm_secondary_got_guest > +FTR_SECTION_ELSE > + HMT_LOW > + ld r5, HSTATE_KVM_VCORE(r13) > + cmpdi r5, 0 > + beq kvm_no_guest > + HMT_MEDIUM > + b kvm_secondary_got_guest > +ALT_FTR_SECTION_END_IFCLR(CPU_FTR_ARCH_300) > > 54: li r0, KVM_HWTHREAD_IN_KVM > stb r0, HSTATE_HWTHREAD_STATE(r13) > b kvm_no_guest > > -kvmhv_do_set: > - /* Set LPCR, LPIDR etc. on P9 */ > - HMT_MEDIUM > - bl kvmhv_p9_set_lpcr > - nop > - b kvm_no_guest > - > -kvmhv_do_restore: > - HMT_MEDIUM > - bl kvmhv_p9_restore_lpcr > - nop > - b kvm_no_guest > - > /* > * Here the primary thread is trying to return the core to > * whole-core mode, so we need to nap. > @@ -527,7 +496,8 @@ END_FTR_SECTION_IFSET(CPU_FTR_ARCH_207S) > /* Set kvm_split_mode.napped[tid] = 1 */ > ld r3, HSTATE_SPLIT_MODE(r13) > li r0, 1 > - lbz r4, HSTATE_TID(r13) > + lhz r4, PACAPACAINDEX(r13) > + clrldi r4, r4, 61 /* micro-threading => P8 => 8 threads/core */ > addi r4, r4, KVM_SPLIT_NAPPED > stbx r0, r3, r4 > /* Check the do_nap flag again after setting napped[] */ > @@ -1938,24 +1908,10 @@ END_FTR_SECTION_IFSET(CPU_FTR_ARCH_207S) > 19: lis r8,0x7fff /* MAX_INT@h */ > mtspr SPRN_HDEC,r8 > > -16: > -BEGIN_FTR_SECTION > - /* On POWER9 with HPT-on-radix we need to wait for all other threads */ > - ld r3, HSTATE_SPLIT_MODE(r13) > - cmpdi r3, 0 > - beq 47f > - lwz r8, KVM_SPLIT_DO_RESTORE(r3) > - cmpwi r8, 0 > - beq 47f > - bl kvmhv_p9_restore_lpcr > - nop > - b 48f > -47: > -END_FTR_SECTION_IFSET(CPU_FTR_ARCH_300) > - ld r8,KVM_HOST_LPCR(r4) > +16: ld r8,KVM_HOST_LPCR(r4) > mtspr SPRN_LPCR,r8 > isync > -48: > + > #ifdef CONFIG_KVM_BOOK3S_HV_EXIT_TIMING > /* Finish timing, if we have a vcpu */ > ld r4, HSTATE_KVM_VCPU(r13) > @@ -2779,8 +2735,10 @@ ALT_FTR_SECTION_END_IFSET(CPU_FTR_ARCH_300) > beq kvm_end_cede > cmpwi r0, NAPPING_NOVCPU > beq kvm_novcpu_wakeup > +BEGIN_FTR_SECTION > cmpwi r0, NAPPING_UNSPLIT > beq kvm_unsplit_wakeup > +END_FTR_SECTION_IFCLR(CPU_FTR_ARCH_300) > twi 31,0,0 /* Nap state must not be zero */ > > 33: mr r4, r3 > -- > 2.23.0