1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * Copyright (C) 2017 - Columbia University and Linaro Ltd. 4 * Author: Jintack Lim <jintack.lim@linaro.org> 5 */ 6 7 #include <linux/kvm.h> 8 #include <linux/kvm_host.h> 9 10 #include <asm/kvm_emulate.h> 11 #include <asm/kvm_nested.h> 12 #include <asm/sysreg.h> 13 14 #include "sys_regs.h" 15 16 /* Protection against the sysreg repainting madness... */ 17 #define NV_FTR(r, f) ID_AA64##r##_EL1_##f 18 19 /* 20 * Our emulated CPU doesn't support all the possible features. For the 21 * sake of simplicity (and probably mental sanity), wipe out a number 22 * of feature bits we don't intend to support for the time being. 23 * This list should get updated as new features get added to the NV 24 * support, and new extension to the architecture. 25 */ 26 void access_nested_id_reg(struct kvm_vcpu *v, struct sys_reg_params *p, 27 const struct sys_reg_desc *r) 28 { 29 u32 id = reg_to_encoding(r); 30 u64 val, tmp; 31 32 val = p->regval; 33 34 switch (id) { 35 case SYS_ID_AA64ISAR0_EL1: 36 /* Support everything but TME, O.S. and Range TLBIs */ 37 val &= ~(NV_FTR(ISAR0, TLB) | 38 NV_FTR(ISAR0, TME)); 39 break; 40 41 case SYS_ID_AA64ISAR1_EL1: 42 /* Support everything but PtrAuth and Spec Invalidation */ 43 val &= ~(GENMASK_ULL(63, 56) | 44 NV_FTR(ISAR1, SPECRES) | 45 NV_FTR(ISAR1, GPI) | 46 NV_FTR(ISAR1, GPA) | 47 NV_FTR(ISAR1, API) | 48 NV_FTR(ISAR1, APA)); 49 break; 50 51 case SYS_ID_AA64PFR0_EL1: 52 /* No AMU, MPAM, S-EL2, RAS or SVE */ 53 val &= ~(GENMASK_ULL(55, 52) | 54 NV_FTR(PFR0, AMU) | 55 NV_FTR(PFR0, MPAM) | 56 NV_FTR(PFR0, SEL2) | 57 NV_FTR(PFR0, RAS) | 58 NV_FTR(PFR0, SVE) | 59 NV_FTR(PFR0, EL3) | 60 NV_FTR(PFR0, EL2) | 61 NV_FTR(PFR0, EL1)); 62 /* 64bit EL1/EL2/EL3 only */ 63 val |= FIELD_PREP(NV_FTR(PFR0, EL1), 0b0001); 64 val |= FIELD_PREP(NV_FTR(PFR0, EL2), 0b0001); 65 val |= FIELD_PREP(NV_FTR(PFR0, EL3), 0b0001); 66 break; 67 68 case SYS_ID_AA64PFR1_EL1: 69 /* Only support SSBS */ 70 val &= NV_FTR(PFR1, SSBS); 71 break; 72 73 case SYS_ID_AA64MMFR0_EL1: 74 /* Hide ECV, ExS, Secure Memory */ 75 val &= ~(NV_FTR(MMFR0, ECV) | 76 NV_FTR(MMFR0, EXS) | 77 NV_FTR(MMFR0, TGRAN4_2) | 78 NV_FTR(MMFR0, TGRAN16_2) | 79 NV_FTR(MMFR0, TGRAN64_2) | 80 NV_FTR(MMFR0, SNSMEM)); 81 82 /* Disallow unsupported S2 page sizes */ 83 switch (PAGE_SIZE) { 84 case SZ_64K: 85 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN16_2), 0b0001); 86 fallthrough; 87 case SZ_16K: 88 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN4_2), 0b0001); 89 fallthrough; 90 case SZ_4K: 91 /* Support everything */ 92 break; 93 } 94 /* 95 * Since we can't support a guest S2 page size smaller than 96 * the host's own page size (due to KVM only populating its 97 * own S2 using the kernel's page size), advertise the 98 * limitation using FEAT_GTG. 99 */ 100 switch (PAGE_SIZE) { 101 case SZ_4K: 102 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN4_2), 0b0010); 103 fallthrough; 104 case SZ_16K: 105 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN16_2), 0b0010); 106 fallthrough; 107 case SZ_64K: 108 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN64_2), 0b0010); 109 break; 110 } 111 /* Cap PARange to 48bits */ 112 tmp = FIELD_GET(NV_FTR(MMFR0, PARANGE), val); 113 if (tmp > 0b0101) { 114 val &= ~NV_FTR(MMFR0, PARANGE); 115 val |= FIELD_PREP(NV_FTR(MMFR0, PARANGE), 0b0101); 116 } 117 break; 118 119 case SYS_ID_AA64MMFR1_EL1: 120 val &= (NV_FTR(MMFR1, HCX) | 121 NV_FTR(MMFR1, PAN) | 122 NV_FTR(MMFR1, LO) | 123 NV_FTR(MMFR1, HPDS) | 124 NV_FTR(MMFR1, VH) | 125 NV_FTR(MMFR1, VMIDBits)); 126 break; 127 128 case SYS_ID_AA64MMFR2_EL1: 129 val &= ~(NV_FTR(MMFR2, BBM) | 130 NV_FTR(MMFR2, TTL) | 131 GENMASK_ULL(47, 44) | 132 NV_FTR(MMFR2, ST) | 133 NV_FTR(MMFR2, CCIDX) | 134 NV_FTR(MMFR2, VARange)); 135 136 /* Force TTL support */ 137 val |= FIELD_PREP(NV_FTR(MMFR2, TTL), 0b0001); 138 break; 139 140 case SYS_ID_AA64DFR0_EL1: 141 /* Only limited support for PMU, Debug, BPs and WPs */ 142 val &= (NV_FTR(DFR0, PMUVer) | 143 NV_FTR(DFR0, WRPs) | 144 NV_FTR(DFR0, BRPs) | 145 NV_FTR(DFR0, DebugVer)); 146 147 /* Cap Debug to ARMv8.1 */ 148 tmp = FIELD_GET(NV_FTR(DFR0, DebugVer), val); 149 if (tmp > 0b0111) { 150 val &= ~NV_FTR(DFR0, DebugVer); 151 val |= FIELD_PREP(NV_FTR(DFR0, DebugVer), 0b0111); 152 } 153 break; 154 155 default: 156 /* Unknown register, just wipe it clean */ 157 val = 0; 158 break; 159 } 160 161 p->regval = val; 162 } 163