Loading...
Note: File does not exist in v4.17.
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 */
26static u64 limit_nv_id_reg(u32 id, u64 val)
27{
28 u64 tmp;
29
30 switch (id) {
31 case SYS_ID_AA64ISAR0_EL1:
32 /* Support everything but TME, O.S. and Range TLBIs */
33 val &= ~(NV_FTR(ISAR0, TLB) |
34 NV_FTR(ISAR0, TME));
35 break;
36
37 case SYS_ID_AA64ISAR1_EL1:
38 /* Support everything but PtrAuth and Spec Invalidation */
39 val &= ~(GENMASK_ULL(63, 56) |
40 NV_FTR(ISAR1, SPECRES) |
41 NV_FTR(ISAR1, GPI) |
42 NV_FTR(ISAR1, GPA) |
43 NV_FTR(ISAR1, API) |
44 NV_FTR(ISAR1, APA));
45 break;
46
47 case SYS_ID_AA64PFR0_EL1:
48 /* No AMU, MPAM, S-EL2, RAS or SVE */
49 val &= ~(GENMASK_ULL(55, 52) |
50 NV_FTR(PFR0, AMU) |
51 NV_FTR(PFR0, MPAM) |
52 NV_FTR(PFR0, SEL2) |
53 NV_FTR(PFR0, RAS) |
54 NV_FTR(PFR0, SVE) |
55 NV_FTR(PFR0, EL3) |
56 NV_FTR(PFR0, EL2) |
57 NV_FTR(PFR0, EL1));
58 /* 64bit EL1/EL2/EL3 only */
59 val |= FIELD_PREP(NV_FTR(PFR0, EL1), 0b0001);
60 val |= FIELD_PREP(NV_FTR(PFR0, EL2), 0b0001);
61 val |= FIELD_PREP(NV_FTR(PFR0, EL3), 0b0001);
62 break;
63
64 case SYS_ID_AA64PFR1_EL1:
65 /* Only support SSBS */
66 val &= NV_FTR(PFR1, SSBS);
67 break;
68
69 case SYS_ID_AA64MMFR0_EL1:
70 /* Hide ECV, ExS, Secure Memory */
71 val &= ~(NV_FTR(MMFR0, ECV) |
72 NV_FTR(MMFR0, EXS) |
73 NV_FTR(MMFR0, TGRAN4_2) |
74 NV_FTR(MMFR0, TGRAN16_2) |
75 NV_FTR(MMFR0, TGRAN64_2) |
76 NV_FTR(MMFR0, SNSMEM));
77
78 /* Disallow unsupported S2 page sizes */
79 switch (PAGE_SIZE) {
80 case SZ_64K:
81 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN16_2), 0b0001);
82 fallthrough;
83 case SZ_16K:
84 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN4_2), 0b0001);
85 fallthrough;
86 case SZ_4K:
87 /* Support everything */
88 break;
89 }
90 /*
91 * Since we can't support a guest S2 page size smaller than
92 * the host's own page size (due to KVM only populating its
93 * own S2 using the kernel's page size), advertise the
94 * limitation using FEAT_GTG.
95 */
96 switch (PAGE_SIZE) {
97 case SZ_4K:
98 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN4_2), 0b0010);
99 fallthrough;
100 case SZ_16K:
101 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN16_2), 0b0010);
102 fallthrough;
103 case SZ_64K:
104 val |= FIELD_PREP(NV_FTR(MMFR0, TGRAN64_2), 0b0010);
105 break;
106 }
107 /* Cap PARange to 48bits */
108 tmp = FIELD_GET(NV_FTR(MMFR0, PARANGE), val);
109 if (tmp > 0b0101) {
110 val &= ~NV_FTR(MMFR0, PARANGE);
111 val |= FIELD_PREP(NV_FTR(MMFR0, PARANGE), 0b0101);
112 }
113 break;
114
115 case SYS_ID_AA64MMFR1_EL1:
116 val &= (NV_FTR(MMFR1, HCX) |
117 NV_FTR(MMFR1, PAN) |
118 NV_FTR(MMFR1, LO) |
119 NV_FTR(MMFR1, HPDS) |
120 NV_FTR(MMFR1, VH) |
121 NV_FTR(MMFR1, VMIDBits));
122 break;
123
124 case SYS_ID_AA64MMFR2_EL1:
125 val &= ~(NV_FTR(MMFR2, BBM) |
126 NV_FTR(MMFR2, TTL) |
127 GENMASK_ULL(47, 44) |
128 NV_FTR(MMFR2, ST) |
129 NV_FTR(MMFR2, CCIDX) |
130 NV_FTR(MMFR2, VARange));
131
132 /* Force TTL support */
133 val |= FIELD_PREP(NV_FTR(MMFR2, TTL), 0b0001);
134 break;
135
136 case SYS_ID_AA64DFR0_EL1:
137 /* Only limited support for PMU, Debug, BPs and WPs */
138 val &= (NV_FTR(DFR0, PMUVer) |
139 NV_FTR(DFR0, WRPs) |
140 NV_FTR(DFR0, BRPs) |
141 NV_FTR(DFR0, DebugVer));
142
143 /* Cap Debug to ARMv8.1 */
144 tmp = FIELD_GET(NV_FTR(DFR0, DebugVer), val);
145 if (tmp > 0b0111) {
146 val &= ~NV_FTR(DFR0, DebugVer);
147 val |= FIELD_PREP(NV_FTR(DFR0, DebugVer), 0b0111);
148 }
149 break;
150
151 default:
152 /* Unknown register, just wipe it clean */
153 val = 0;
154 break;
155 }
156
157 return val;
158}
159int kvm_init_nv_sysregs(struct kvm *kvm)
160{
161 mutex_lock(&kvm->arch.config_lock);
162
163 for (int i = 0; i < KVM_ARM_ID_REG_NUM; i++)
164 kvm->arch.id_regs[i] = limit_nv_id_reg(IDX_IDREG(i),
165 kvm->arch.id_regs[i]);
166
167 mutex_unlock(&kvm->arch.config_lock);
168
169 return 0;
170}