Linux 6.7-rc7
[linux-modified.git] / arch / arm64 / kvm / nested.c
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 }