Linux Audio

Check our new training course

Embedded Linux training

Mar 31-Apr 8, 2025
Register
Loading...
Note: File does not exist in v5.4.
  1// SPDX-License-Identifier: GPL-2.0
  2#include <linux/tboot.h>
  3
  4#include <asm/cpu.h>
  5#include <asm/cpufeature.h>
  6#include <asm/msr-index.h>
  7#include <asm/processor.h>
  8#include <asm/vmx.h>
  9
 10#undef pr_fmt
 11#define pr_fmt(fmt)	"x86/cpu: " fmt
 12
 13#ifdef CONFIG_X86_VMX_FEATURE_NAMES
 14enum vmx_feature_leafs {
 15	MISC_FEATURES = 0,
 16	PRIMARY_CTLS,
 17	SECONDARY_CTLS,
 18	TERTIARY_CTLS_LOW,
 19	TERTIARY_CTLS_HIGH,
 20	NR_VMX_FEATURE_WORDS,
 21};
 22
 23#define VMX_F(x) BIT(VMX_FEATURE_##x & 0x1f)
 24
 25static void init_vmx_capabilities(struct cpuinfo_x86 *c)
 26{
 27	u32 supported, funcs, ept, vpid, ign, low, high;
 28
 29	BUILD_BUG_ON(NVMXINTS != NR_VMX_FEATURE_WORDS);
 30
 31	/*
 32	 * The high bits contain the allowed-1 settings, i.e. features that can
 33	 * be turned on.  The low bits contain the allowed-0 settings, i.e.
 34	 * features that can be turned off.  Ignore the allowed-0 settings,
 35	 * if a feature can be turned on then it's supported.
 36	 *
 37	 * Use raw rdmsr() for primary processor controls and pin controls MSRs
 38	 * as they exist on any CPU that supports VMX, i.e. we want the WARN if
 39	 * the RDMSR faults.
 40	 */
 41	rdmsr(MSR_IA32_VMX_PROCBASED_CTLS, ign, supported);
 42	c->vmx_capability[PRIMARY_CTLS] = supported;
 43
 44	rdmsr_safe(MSR_IA32_VMX_PROCBASED_CTLS2, &ign, &supported);
 45	c->vmx_capability[SECONDARY_CTLS] = supported;
 46
 47	/* All 64 bits of tertiary controls MSR are allowed-1 settings. */
 48	rdmsr_safe(MSR_IA32_VMX_PROCBASED_CTLS3, &low, &high);
 49	c->vmx_capability[TERTIARY_CTLS_LOW] = low;
 50	c->vmx_capability[TERTIARY_CTLS_HIGH] = high;
 51
 52	rdmsr(MSR_IA32_VMX_PINBASED_CTLS, ign, supported);
 53	rdmsr_safe(MSR_IA32_VMX_VMFUNC, &ign, &funcs);
 54
 55	/*
 56	 * Except for EPT+VPID, which enumerates support for both in a single
 57	 * MSR, low for EPT, high for VPID.
 58	 */
 59	rdmsr_safe(MSR_IA32_VMX_EPT_VPID_CAP, &ept, &vpid);
 60
 61	/* Pin, EPT, VPID and VM-Func are merged into a single word. */
 62	WARN_ON_ONCE(supported >> 16);
 63	WARN_ON_ONCE(funcs >> 4);
 64	c->vmx_capability[MISC_FEATURES] = (supported & 0xffff) |
 65					   ((vpid & 0x1) << 16) |
 66					   ((funcs & 0xf) << 28);
 67
 68	/* EPT bits are full on scattered and must be manually handled. */
 69	if (ept & VMX_EPT_EXECUTE_ONLY_BIT)
 70		c->vmx_capability[MISC_FEATURES] |= VMX_F(EPT_EXECUTE_ONLY);
 71	if (ept & VMX_EPT_AD_BIT)
 72		c->vmx_capability[MISC_FEATURES] |= VMX_F(EPT_AD);
 73	if (ept & VMX_EPT_1GB_PAGE_BIT)
 74		c->vmx_capability[MISC_FEATURES] |= VMX_F(EPT_1GB);
 75	if (ept & VMX_EPT_PAGE_WALK_5_BIT)
 76		c->vmx_capability[MISC_FEATURES] |= VMX_F(EPT_5LEVEL);
 77
 78	/* Synthetic APIC features that are aggregates of multiple features. */
 79	if ((c->vmx_capability[PRIMARY_CTLS] & VMX_F(VIRTUAL_TPR)) &&
 80	    (c->vmx_capability[SECONDARY_CTLS] & VMX_F(VIRT_APIC_ACCESSES)))
 81		c->vmx_capability[MISC_FEATURES] |= VMX_F(FLEXPRIORITY);
 82
 83	if ((c->vmx_capability[PRIMARY_CTLS] & VMX_F(VIRTUAL_TPR)) &&
 84	    (c->vmx_capability[SECONDARY_CTLS] & VMX_F(APIC_REGISTER_VIRT)) &&
 85	    (c->vmx_capability[SECONDARY_CTLS] & VMX_F(VIRT_INTR_DELIVERY)) &&
 86	    (c->vmx_capability[MISC_FEATURES] & VMX_F(POSTED_INTR)))
 87		c->vmx_capability[MISC_FEATURES] |= VMX_F(APICV);
 88
 89	/* Set the synthetic cpufeatures to preserve /proc/cpuinfo's ABI. */
 90	if (c->vmx_capability[PRIMARY_CTLS] & VMX_F(VIRTUAL_TPR))
 91		set_cpu_cap(c, X86_FEATURE_TPR_SHADOW);
 92	if (c->vmx_capability[MISC_FEATURES] & VMX_F(FLEXPRIORITY))
 93		set_cpu_cap(c, X86_FEATURE_FLEXPRIORITY);
 94	if (c->vmx_capability[MISC_FEATURES] & VMX_F(VIRTUAL_NMIS))
 95		set_cpu_cap(c, X86_FEATURE_VNMI);
 96	if (c->vmx_capability[SECONDARY_CTLS] & VMX_F(EPT))
 97		set_cpu_cap(c, X86_FEATURE_EPT);
 98	if (c->vmx_capability[MISC_FEATURES] & VMX_F(EPT_AD))
 99		set_cpu_cap(c, X86_FEATURE_EPT_AD);
100	if (c->vmx_capability[MISC_FEATURES] & VMX_F(VPID))
101		set_cpu_cap(c, X86_FEATURE_VPID);
102}
103#endif /* CONFIG_X86_VMX_FEATURE_NAMES */
104
105static int __init nosgx(char *str)
106{
107	setup_clear_cpu_cap(X86_FEATURE_SGX);
108
109	return 0;
110}
111
112early_param("nosgx", nosgx);
113
114void init_ia32_feat_ctl(struct cpuinfo_x86 *c)
115{
116	bool enable_sgx_kvm = false, enable_sgx_driver = false;
117	bool tboot = tboot_enabled();
118	bool enable_vmx;
119	u64 msr;
120
121	if (rdmsrl_safe(MSR_IA32_FEAT_CTL, &msr)) {
122		clear_cpu_cap(c, X86_FEATURE_VMX);
123		clear_cpu_cap(c, X86_FEATURE_SGX);
124		return;
125	}
126
127	enable_vmx = cpu_has(c, X86_FEATURE_VMX) &&
128		     IS_ENABLED(CONFIG_KVM_INTEL);
129
130	if (cpu_has(c, X86_FEATURE_SGX) && IS_ENABLED(CONFIG_X86_SGX)) {
131		/*
132		 * Separate out SGX driver enabling from KVM.  This allows KVM
133		 * guests to use SGX even if the kernel SGX driver refuses to
134		 * use it.  This happens if flexible Launch Control is not
135		 * available.
136		 */
137		enable_sgx_driver = cpu_has(c, X86_FEATURE_SGX_LC);
138		enable_sgx_kvm = enable_vmx && IS_ENABLED(CONFIG_X86_SGX_KVM);
139	}
140
141	if (msr & FEAT_CTL_LOCKED)
142		goto update_caps;
143
144	/*
145	 * Ignore whatever value BIOS left in the MSR to avoid enabling random
146	 * features or faulting on the WRMSR.
147	 */
148	msr = FEAT_CTL_LOCKED;
149
150	/*
151	 * Enable VMX if and only if the kernel may do VMXON at some point,
152	 * i.e. KVM is enabled, to avoid unnecessarily adding an attack vector
153	 * for the kernel, e.g. using VMX to hide malicious code.
154	 */
155	if (enable_vmx) {
156		msr |= FEAT_CTL_VMX_ENABLED_OUTSIDE_SMX;
157
158		if (tboot)
159			msr |= FEAT_CTL_VMX_ENABLED_INSIDE_SMX;
160	}
161
162	if (enable_sgx_kvm || enable_sgx_driver) {
163		msr |= FEAT_CTL_SGX_ENABLED;
164		if (enable_sgx_driver)
165			msr |= FEAT_CTL_SGX_LC_ENABLED;
166	}
167
168	wrmsrl(MSR_IA32_FEAT_CTL, msr);
169
170update_caps:
171	set_cpu_cap(c, X86_FEATURE_MSR_IA32_FEAT_CTL);
172
173	if (!cpu_has(c, X86_FEATURE_VMX))
174		goto update_sgx;
175
176	if ( (tboot && !(msr & FEAT_CTL_VMX_ENABLED_INSIDE_SMX)) ||
177	    (!tboot && !(msr & FEAT_CTL_VMX_ENABLED_OUTSIDE_SMX))) {
178		if (IS_ENABLED(CONFIG_KVM_INTEL))
179			pr_err_once("VMX (%s TXT) disabled by BIOS\n",
180				    tboot ? "inside" : "outside");
181		clear_cpu_cap(c, X86_FEATURE_VMX);
182	} else {
183#ifdef CONFIG_X86_VMX_FEATURE_NAMES
184		init_vmx_capabilities(c);
185#endif
186	}
187
188update_sgx:
189	if (!(msr & FEAT_CTL_SGX_ENABLED)) {
190		if (enable_sgx_kvm || enable_sgx_driver)
191			pr_err_once("SGX disabled by BIOS.\n");
192		clear_cpu_cap(c, X86_FEATURE_SGX);
193		return;
194	}
195
196	/*
197	 * VMX feature bit may be cleared due to being disabled in BIOS,
198	 * in which case SGX virtualization cannot be supported either.
199	 */
200	if (!cpu_has(c, X86_FEATURE_VMX) && enable_sgx_kvm) {
201		pr_err_once("SGX virtualization disabled due to lack of VMX.\n");
202		enable_sgx_kvm = 0;
203	}
204
205	if (!(msr & FEAT_CTL_SGX_LC_ENABLED) && enable_sgx_driver) {
206		if (!enable_sgx_kvm) {
207			pr_err_once("SGX Launch Control is locked. Disable SGX.\n");
208			clear_cpu_cap(c, X86_FEATURE_SGX);
209		} else {
210			pr_err_once("SGX Launch Control is locked. Support SGX virtualization only.\n");
211			clear_cpu_cap(c, X86_FEATURE_SGX_LC);
212		}
213	}
214}