xref: /kernel/linux/linux-6.6/arch/arm64/kvm/nested.c (revision 62306a36)
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 */
26void 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