2018-12-04 04:53:18 +07:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
#include <linux/frame.h>
|
|
|
|
#include <linux/percpu.h>
|
|
|
|
|
|
|
|
#include <asm/debugreg.h>
|
|
|
|
#include <asm/mmu_context.h>
|
|
|
|
|
|
|
|
#include "cpuid.h"
|
|
|
|
#include "hyperv.h"
|
|
|
|
#include "mmu.h"
|
|
|
|
#include "nested.h"
|
2019-11-14 07:17:16 +07:00
|
|
|
#include "pmu.h"
|
2018-12-04 04:53:18 +07:00
|
|
|
#include "trace.h"
|
|
|
|
#include "x86.h"
|
|
|
|
|
|
|
|
static bool __read_mostly enable_shadow_vmcs = 1;
|
|
|
|
module_param_named(enable_shadow_vmcs, enable_shadow_vmcs, bool, S_IRUGO);
|
|
|
|
|
|
|
|
static bool __read_mostly nested_early_check = 0;
|
|
|
|
module_param(nested_early_check, bool, S_IRUGO);
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
#define CC(consistency_check) \
|
|
|
|
({ \
|
|
|
|
bool failed = (consistency_check); \
|
|
|
|
if (failed) \
|
2019-07-11 22:58:30 +07:00
|
|
|
trace_kvm_nested_vmenter_failed(#consistency_check, 0); \
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
failed; \
|
|
|
|
})
|
|
|
|
|
2019-11-14 07:17:18 +07:00
|
|
|
#define SET_MSR_OR_WARN(vcpu, idx, data) \
|
|
|
|
({ \
|
|
|
|
bool failed = kvm_set_msr(vcpu, idx, data); \
|
|
|
|
if (failed) \
|
|
|
|
pr_warn_ratelimited( \
|
|
|
|
"%s cannot write MSR (0x%x, 0x%llx)\n", \
|
|
|
|
__func__, idx, data); \
|
|
|
|
failed; \
|
|
|
|
})
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Hyper-V requires all of these, so mark them as supported even though
|
|
|
|
* they are just treated the same as all-context.
|
|
|
|
*/
|
|
|
|
#define VMX_VPID_EXTENT_SUPPORTED_MASK \
|
|
|
|
(VMX_VPID_EXTENT_INDIVIDUAL_ADDR_BIT | \
|
|
|
|
VMX_VPID_EXTENT_SINGLE_CONTEXT_BIT | \
|
|
|
|
VMX_VPID_EXTENT_GLOBAL_CONTEXT_BIT | \
|
|
|
|
VMX_VPID_EXTENT_SINGLE_NON_GLOBAL_BIT)
|
|
|
|
|
|
|
|
#define VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE 5
|
|
|
|
|
|
|
|
enum {
|
|
|
|
VMX_VMREAD_BITMAP,
|
|
|
|
VMX_VMWRITE_BITMAP,
|
|
|
|
VMX_BITMAP_NR
|
|
|
|
};
|
|
|
|
static unsigned long *vmx_bitmap[VMX_BITMAP_NR];
|
|
|
|
|
|
|
|
#define vmx_vmread_bitmap (vmx_bitmap[VMX_VMREAD_BITMAP])
|
|
|
|
#define vmx_vmwrite_bitmap (vmx_bitmap[VMX_VMWRITE_BITMAP])
|
|
|
|
|
2019-05-07 22:36:25 +07:00
|
|
|
struct shadow_vmcs_field {
|
|
|
|
u16 encoding;
|
|
|
|
u16 offset;
|
|
|
|
};
|
|
|
|
static struct shadow_vmcs_field shadow_read_only_fields[] = {
|
|
|
|
#define SHADOW_FIELD_RO(x, y) { x, offsetof(struct vmcs12, y) },
|
2018-12-04 04:53:18 +07:00
|
|
|
#include "vmcs_shadow_fields.h"
|
|
|
|
};
|
|
|
|
static int max_shadow_read_only_fields =
|
|
|
|
ARRAY_SIZE(shadow_read_only_fields);
|
|
|
|
|
2019-05-07 22:36:25 +07:00
|
|
|
static struct shadow_vmcs_field shadow_read_write_fields[] = {
|
|
|
|
#define SHADOW_FIELD_RW(x, y) { x, offsetof(struct vmcs12, y) },
|
2018-12-04 04:53:18 +07:00
|
|
|
#include "vmcs_shadow_fields.h"
|
|
|
|
};
|
|
|
|
static int max_shadow_read_write_fields =
|
|
|
|
ARRAY_SIZE(shadow_read_write_fields);
|
|
|
|
|
2019-01-21 14:27:05 +07:00
|
|
|
static void init_vmcs_shadow_fields(void)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
memset(vmx_vmread_bitmap, 0xff, PAGE_SIZE);
|
|
|
|
memset(vmx_vmwrite_bitmap, 0xff, PAGE_SIZE);
|
|
|
|
|
|
|
|
for (i = j = 0; i < max_shadow_read_only_fields; i++) {
|
2019-05-07 22:36:25 +07:00
|
|
|
struct shadow_vmcs_field entry = shadow_read_only_fields[i];
|
|
|
|
u16 field = entry.encoding;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vmcs_field_width(field) == VMCS_FIELD_WIDTH_U64 &&
|
|
|
|
(i + 1 == max_shadow_read_only_fields ||
|
2019-05-07 22:36:25 +07:00
|
|
|
shadow_read_only_fields[i + 1].encoding != field + 1))
|
2018-12-04 04:53:18 +07:00
|
|
|
pr_err("Missing field from shadow_read_only_field %x\n",
|
|
|
|
field + 1);
|
|
|
|
|
|
|
|
clear_bit(field, vmx_vmread_bitmap);
|
|
|
|
if (field & 1)
|
2019-05-07 22:36:25 +07:00
|
|
|
#ifdef CONFIG_X86_64
|
2018-12-04 04:53:18 +07:00
|
|
|
continue;
|
2019-05-07 22:36:25 +07:00
|
|
|
#else
|
|
|
|
entry.offset += sizeof(u32);
|
2018-12-04 04:53:18 +07:00
|
|
|
#endif
|
2019-05-07 22:36:25 +07:00
|
|
|
shadow_read_only_fields[j++] = entry;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
max_shadow_read_only_fields = j;
|
|
|
|
|
|
|
|
for (i = j = 0; i < max_shadow_read_write_fields; i++) {
|
2019-05-07 22:36:25 +07:00
|
|
|
struct shadow_vmcs_field entry = shadow_read_write_fields[i];
|
|
|
|
u16 field = entry.encoding;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vmcs_field_width(field) == VMCS_FIELD_WIDTH_U64 &&
|
|
|
|
(i + 1 == max_shadow_read_write_fields ||
|
2019-05-07 22:36:25 +07:00
|
|
|
shadow_read_write_fields[i + 1].encoding != field + 1))
|
2018-12-04 04:53:18 +07:00
|
|
|
pr_err("Missing field from shadow_read_write_field %x\n",
|
|
|
|
field + 1);
|
|
|
|
|
KVM: nVMX: Intercept VMWRITEs to GUEST_{CS,SS}_AR_BYTES
VMMs frequently read the guest's CS and SS AR bytes to detect 64-bit
mode and CPL respectively, but effectively never write said fields once
the VM is initialized. Intercepting VMWRITEs for the two fields saves
~55 cycles in copy_shadow_to_vmcs12().
Because some Intel CPUs, e.g. Haswell, drop the reserved bits of the
guest access rights fields on VMWRITE, exposing the fields to L1 for
VMREAD but not VMWRITE leads to inconsistent behavior between L1 and L2.
On hardware that drops the bits, L1 will see the stripped down value due
to reading the value from hardware, while L2 will see the full original
value as stored by KVM. To avoid such an inconsistency, emulate the
behavior on all CPUS, but only for intercepted VMWRITEs so as to avoid
introducing pointless latency into copy_shadow_to_vmcs12(), e.g. if the
emulation were added to vmcs12_write_any().
Since the AR_BYTES emulation is done only for intercepted VMWRITE, if a
future patch (re)exposed AR_BYTES for both VMWRITE and VMREAD, then KVM
would end up with incosistent behavior on pre-Haswell hardware, e.g. KVM
would drop the reserved bits on intercepted VMWRITE, but direct VMWRITE
to the shadow VMCS would not drop the bits. Add a WARN in the shadow
field initialization to detect any attempt to expose an AR_BYTES field
without updating vmcs12_write_any().
Note, emulation of the AR_BYTES reserved bit behavior is based on a
patch[1] from Jim Mattson that applied the emulation to all writes to
vmcs12 so that live migration across different generations of hardware
would not introduce divergent behavior. But given that live migration
of nested state has already been enabled, that ship has sailed (not to
mention that no sane VMM will be affected by this behavior).
[1] https://patchwork.kernel.org/patch/10483321/
Cc: Jim Mattson <jmattson@google.com>
Cc: Liran Alon <liran.alon@oracle.com>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:24 +07:00
|
|
|
WARN_ONCE(field >= GUEST_ES_AR_BYTES &&
|
|
|
|
field <= GUEST_TR_AR_BYTES,
|
2019-05-07 22:36:25 +07:00
|
|
|
"Update vmcs12_write_any() to drop reserved bits from AR_BYTES");
|
KVM: nVMX: Intercept VMWRITEs to GUEST_{CS,SS}_AR_BYTES
VMMs frequently read the guest's CS and SS AR bytes to detect 64-bit
mode and CPL respectively, but effectively never write said fields once
the VM is initialized. Intercepting VMWRITEs for the two fields saves
~55 cycles in copy_shadow_to_vmcs12().
Because some Intel CPUs, e.g. Haswell, drop the reserved bits of the
guest access rights fields on VMWRITE, exposing the fields to L1 for
VMREAD but not VMWRITE leads to inconsistent behavior between L1 and L2.
On hardware that drops the bits, L1 will see the stripped down value due
to reading the value from hardware, while L2 will see the full original
value as stored by KVM. To avoid such an inconsistency, emulate the
behavior on all CPUS, but only for intercepted VMWRITEs so as to avoid
introducing pointless latency into copy_shadow_to_vmcs12(), e.g. if the
emulation were added to vmcs12_write_any().
Since the AR_BYTES emulation is done only for intercepted VMWRITE, if a
future patch (re)exposed AR_BYTES for both VMWRITE and VMREAD, then KVM
would end up with incosistent behavior on pre-Haswell hardware, e.g. KVM
would drop the reserved bits on intercepted VMWRITE, but direct VMWRITE
to the shadow VMCS would not drop the bits. Add a WARN in the shadow
field initialization to detect any attempt to expose an AR_BYTES field
without updating vmcs12_write_any().
Note, emulation of the AR_BYTES reserved bit behavior is based on a
patch[1] from Jim Mattson that applied the emulation to all writes to
vmcs12 so that live migration across different generations of hardware
would not introduce divergent behavior. But given that live migration
of nested state has already been enabled, that ship has sailed (not to
mention that no sane VMM will be affected by this behavior).
[1] https://patchwork.kernel.org/patch/10483321/
Cc: Jim Mattson <jmattson@google.com>
Cc: Liran Alon <liran.alon@oracle.com>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:24 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* PML and the preemption timer can be emulated, but the
|
|
|
|
* processor cannot vmwrite to fields that don't exist
|
|
|
|
* on bare metal.
|
|
|
|
*/
|
|
|
|
switch (field) {
|
|
|
|
case GUEST_PML_INDEX:
|
|
|
|
if (!cpu_has_vmx_pml())
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
case VMX_PREEMPTION_TIMER_VALUE:
|
|
|
|
if (!cpu_has_vmx_preemption_timer())
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
case GUEST_INTR_STATUS:
|
|
|
|
if (!cpu_has_vmx_apicv())
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
clear_bit(field, vmx_vmwrite_bitmap);
|
|
|
|
clear_bit(field, vmx_vmread_bitmap);
|
|
|
|
if (field & 1)
|
2019-05-07 22:36:25 +07:00
|
|
|
#ifdef CONFIG_X86_64
|
2018-12-04 04:53:18 +07:00
|
|
|
continue;
|
2019-05-07 22:36:25 +07:00
|
|
|
#else
|
|
|
|
entry.offset += sizeof(u32);
|
2018-12-04 04:53:18 +07:00
|
|
|
#endif
|
2019-05-07 22:36:25 +07:00
|
|
|
shadow_read_write_fields[j++] = entry;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
max_shadow_read_write_fields = j;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The following 3 functions, nested_vmx_succeed()/failValid()/failInvalid(),
|
|
|
|
* set the success or error code of an emulated VMX instruction (as specified
|
|
|
|
* by Vol 2B, VMX Instruction Reference, "Conventions"), and skip the emulated
|
|
|
|
* instruction.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_succeed(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
vmx_set_rflags(vcpu, vmx_get_rflags(vcpu)
|
|
|
|
& ~(X86_EFLAGS_CF | X86_EFLAGS_PF | X86_EFLAGS_AF |
|
|
|
|
X86_EFLAGS_ZF | X86_EFLAGS_SF | X86_EFLAGS_OF));
|
|
|
|
return kvm_skip_emulated_instruction(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_failInvalid(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
vmx_set_rflags(vcpu, (vmx_get_rflags(vcpu)
|
|
|
|
& ~(X86_EFLAGS_PF | X86_EFLAGS_AF | X86_EFLAGS_ZF |
|
|
|
|
X86_EFLAGS_SF | X86_EFLAGS_OF))
|
|
|
|
| X86_EFLAGS_CF);
|
|
|
|
return kvm_skip_emulated_instruction(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_failValid(struct kvm_vcpu *vcpu,
|
|
|
|
u32 vm_instruction_error)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* failValid writes the error number to the current VMCS, which
|
|
|
|
* can't be done if there isn't a current VMCS.
|
|
|
|
*/
|
|
|
|
if (vmx->nested.current_vmptr == -1ull && !vmx->nested.hv_evmcs)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
vmx_set_rflags(vcpu, (vmx_get_rflags(vcpu)
|
|
|
|
& ~(X86_EFLAGS_CF | X86_EFLAGS_PF | X86_EFLAGS_AF |
|
|
|
|
X86_EFLAGS_SF | X86_EFLAGS_OF))
|
|
|
|
| X86_EFLAGS_ZF);
|
|
|
|
get_vmcs12(vcpu)->vm_instruction_error = vm_instruction_error;
|
|
|
|
/*
|
|
|
|
* We don't need to force a shadow sync because
|
|
|
|
* VM_INSTRUCTION_ERROR is not shadowed
|
|
|
|
*/
|
|
|
|
return kvm_skip_emulated_instruction(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_vmx_abort(struct kvm_vcpu *vcpu, u32 indicator)
|
|
|
|
{
|
|
|
|
/* TODO: not to reset guest simply here. */
|
|
|
|
kvm_make_request(KVM_REQ_TRIPLE_FAULT, vcpu);
|
|
|
|
pr_debug_ratelimited("kvm: nested vmx abort, indicator %d\n", indicator);
|
|
|
|
}
|
|
|
|
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
static inline bool vmx_control_verify(u32 control, u32 low, u32 high)
|
|
|
|
{
|
|
|
|
return fixed_bits_valid(control, low, high);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u64 vmx_control_msr(u32 low, u32 high)
|
|
|
|
{
|
|
|
|
return low | ((u64)high << 32);
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static void vmx_disable_shadow_vmcs(struct vcpu_vmx *vmx)
|
|
|
|
{
|
2019-05-08 02:17:57 +07:00
|
|
|
secondary_exec_controls_clearbit(vmx, SECONDARY_EXEC_SHADOW_VMCS);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write64(VMCS_LINK_POINTER, -1ull);
|
2019-07-19 23:41:10 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = false;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void nested_release_evmcs(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
if (!vmx->nested.hv_evmcs)
|
|
|
|
return;
|
|
|
|
|
2019-02-01 03:24:42 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &vmx->nested.hv_evmcs_map, true);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.hv_evmcs_vmptr = -1ull;
|
|
|
|
vmx->nested.hv_evmcs = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Free whatever needs to be freed from vmx->nested when L1 goes down, or
|
|
|
|
* just stops using VMX.
|
|
|
|
*/
|
|
|
|
static void free_nested(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
if (!vmx->nested.vmxon && !vmx->nested.smm.vmxon)
|
|
|
|
return;
|
|
|
|
|
2019-07-21 18:52:18 +07:00
|
|
|
kvm_clear_request(KVM_REQ_GET_VMCS12_PAGES, vcpu);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.vmxon = false;
|
|
|
|
vmx->nested.smm.vmxon = false;
|
|
|
|
free_vpid(vmx->nested.vpid02);
|
|
|
|
vmx->nested.posted_intr_nv = -1;
|
|
|
|
vmx->nested.current_vmptr = -1ull;
|
|
|
|
if (enable_shadow_vmcs) {
|
|
|
|
vmx_disable_shadow_vmcs(vmx);
|
|
|
|
vmcs_clear(vmx->vmcs01.shadow_vmcs);
|
|
|
|
free_vmcs(vmx->vmcs01.shadow_vmcs);
|
|
|
|
vmx->vmcs01.shadow_vmcs = NULL;
|
|
|
|
}
|
|
|
|
kfree(vmx->nested.cached_vmcs12);
|
2019-07-21 21:01:36 +07:00
|
|
|
vmx->nested.cached_vmcs12 = NULL;
|
2018-12-04 04:53:18 +07:00
|
|
|
kfree(vmx->nested.cached_shadow_vmcs12);
|
2019-07-21 21:01:36 +07:00
|
|
|
vmx->nested.cached_shadow_vmcs12 = NULL;
|
2018-12-04 04:53:18 +07:00
|
|
|
/* Unpin physical memory we referred to in the vmcs02 */
|
|
|
|
if (vmx->nested.apic_access_page) {
|
2019-11-21 05:31:47 +07:00
|
|
|
kvm_release_page_clean(vmx->nested.apic_access_page);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.apic_access_page = NULL;
|
|
|
|
}
|
2019-02-01 03:24:37 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &vmx->nested.virtual_apic_map, true);
|
2019-02-01 03:24:38 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &vmx->nested.pi_desc_map, true);
|
|
|
|
vmx->nested.pi_desc = NULL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
kvm_mmu_free_roots(vcpu, &vcpu->arch.guest_mmu, KVM_MMU_ROOTS_ALL);
|
|
|
|
|
|
|
|
nested_release_evmcs(vcpu);
|
|
|
|
|
|
|
|
free_loaded_vmcs(&vmx->nested.vmcs02);
|
|
|
|
}
|
|
|
|
|
2019-05-07 23:06:31 +07:00
|
|
|
static void vmx_sync_vmcs_host_state(struct vcpu_vmx *vmx,
|
|
|
|
struct loaded_vmcs *prev)
|
|
|
|
{
|
|
|
|
struct vmcs_host_state *dest, *src;
|
|
|
|
|
|
|
|
if (unlikely(!vmx->guest_state_loaded))
|
|
|
|
return;
|
|
|
|
|
|
|
|
src = &prev->host_state;
|
|
|
|
dest = &vmx->loaded_vmcs->host_state;
|
|
|
|
|
|
|
|
vmx_set_host_fs_gs(dest, src->fs_sel, src->gs_sel, src->fs_base, src->gs_base);
|
|
|
|
dest->ldt_sel = src->ldt_sel;
|
|
|
|
#ifdef CONFIG_X86_64
|
|
|
|
dest->ds_sel = src->ds_sel;
|
|
|
|
dest->es_sel = src->es_sel;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static void vmx_switch_vmcs(struct kvm_vcpu *vcpu, struct loaded_vmcs *vmcs)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
2019-05-07 23:06:31 +07:00
|
|
|
struct loaded_vmcs *prev;
|
2018-12-04 04:53:18 +07:00
|
|
|
int cpu;
|
|
|
|
|
|
|
|
if (vmx->loaded_vmcs == vmcs)
|
|
|
|
return;
|
|
|
|
|
|
|
|
cpu = get_cpu();
|
2019-05-07 23:06:31 +07:00
|
|
|
prev = vmx->loaded_vmcs;
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->loaded_vmcs = vmcs;
|
2019-05-07 23:06:32 +07:00
|
|
|
vmx_vcpu_load_vmcs(vcpu, cpu);
|
2019-05-07 23:06:31 +07:00
|
|
|
vmx_sync_vmcs_host_state(vmx, prev);
|
2018-12-04 04:53:18 +07:00
|
|
|
put_cpu();
|
|
|
|
|
|
|
|
vmx_segment_cache_clear(vmx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Ensure that the current vmcs of the logical processor is the
|
|
|
|
* vmcs01 of the vcpu before calling free_nested().
|
|
|
|
*/
|
|
|
|
void nested_vmx_free_vcpu(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
vcpu_load(vcpu);
|
2019-01-30 01:12:35 +07:00
|
|
|
vmx_leave_nested(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx_switch_vmcs(vcpu, &to_vmx(vcpu)->vmcs01);
|
|
|
|
free_nested(vcpu);
|
|
|
|
vcpu_put(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_ept_inject_page_fault(struct kvm_vcpu *vcpu,
|
|
|
|
struct x86_exception *fault)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u32 exit_reason;
|
|
|
|
unsigned long exit_qualification = vcpu->arch.exit_qualification;
|
|
|
|
|
|
|
|
if (vmx->nested.pml_full) {
|
|
|
|
exit_reason = EXIT_REASON_PML_FULL;
|
|
|
|
vmx->nested.pml_full = false;
|
|
|
|
exit_qualification &= INTR_INFO_UNBLOCK_NMI;
|
|
|
|
} else if (fault->error_code & PFERR_RSVD_MASK)
|
|
|
|
exit_reason = EXIT_REASON_EPT_MISCONFIG;
|
|
|
|
else
|
|
|
|
exit_reason = EXIT_REASON_EPT_VIOLATION;
|
|
|
|
|
|
|
|
nested_vmx_vmexit(vcpu, exit_reason, 0, exit_qualification);
|
|
|
|
vmcs12->guest_physical_address = fault->address;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_ept_init_mmu_context(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
WARN_ON(mmu_is_nested(vcpu));
|
|
|
|
|
|
|
|
vcpu->arch.mmu = &vcpu->arch.guest_mmu;
|
|
|
|
kvm_init_shadow_ept_mmu(vcpu,
|
|
|
|
to_vmx(vcpu)->nested.msrs.ept_caps &
|
|
|
|
VMX_EPT_EXECUTE_ONLY_BIT,
|
|
|
|
nested_ept_ad_enabled(vcpu),
|
|
|
|
nested_ept_get_cr3(vcpu));
|
|
|
|
vcpu->arch.mmu->set_cr3 = vmx_set_cr3;
|
|
|
|
vcpu->arch.mmu->get_cr3 = nested_ept_get_cr3;
|
|
|
|
vcpu->arch.mmu->inject_page_fault = nested_ept_inject_page_fault;
|
|
|
|
vcpu->arch.mmu->get_pdptr = kvm_pdptr_read;
|
|
|
|
|
|
|
|
vcpu->arch.walk_mmu = &vcpu->arch.nested_mmu;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_ept_uninit_mmu_context(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
vcpu->arch.mmu = &vcpu->arch.root_mmu;
|
|
|
|
vcpu->arch.walk_mmu = &vcpu->arch.root_mmu;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool nested_vmx_is_page_fault_vmexit(struct vmcs12 *vmcs12,
|
|
|
|
u16 error_code)
|
|
|
|
{
|
|
|
|
bool inequality, bit;
|
|
|
|
|
|
|
|
bit = (vmcs12->exception_bitmap & (1u << PF_VECTOR)) != 0;
|
|
|
|
inequality =
|
|
|
|
(error_code & vmcs12->page_fault_error_code_mask) !=
|
|
|
|
vmcs12->page_fault_error_code_match;
|
|
|
|
return inequality ^ bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* KVM wants to inject page-faults which it got to the guest. This function
|
|
|
|
* checks whether in a nested guest, we need to inject them to L1 or L2.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_check_exception(struct kvm_vcpu *vcpu, unsigned long *exit_qual)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
unsigned int nr = vcpu->arch.exception.nr;
|
|
|
|
bool has_payload = vcpu->arch.exception.has_payload;
|
|
|
|
unsigned long payload = vcpu->arch.exception.payload;
|
|
|
|
|
|
|
|
if (nr == PF_VECTOR) {
|
|
|
|
if (vcpu->arch.exception.nested_apf) {
|
|
|
|
*exit_qual = vcpu->arch.apf.nested_apf_token;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (nested_vmx_is_page_fault_vmexit(vmcs12,
|
|
|
|
vcpu->arch.exception.error_code)) {
|
|
|
|
*exit_qual = has_payload ? payload : vcpu->arch.cr2;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else if (vmcs12->exception_bitmap & (1u << nr)) {
|
|
|
|
if (nr == DB_VECTOR) {
|
|
|
|
if (!has_payload) {
|
|
|
|
payload = vcpu->arch.dr6;
|
|
|
|
payload &= ~(DR6_FIXED_1 | DR6_BT);
|
|
|
|
payload ^= DR6_RTM;
|
|
|
|
}
|
|
|
|
*exit_qual = payload;
|
|
|
|
} else
|
|
|
|
*exit_qual = 0;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void vmx_inject_page_fault_nested(struct kvm_vcpu *vcpu,
|
|
|
|
struct x86_exception *fault)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
|
|
|
WARN_ON(!is_guest_mode(vcpu));
|
|
|
|
|
|
|
|
if (nested_vmx_is_page_fault_vmexit(vmcs12, fault->error_code) &&
|
|
|
|
!to_vmx(vcpu)->nested.nested_run_pending) {
|
|
|
|
vmcs12->vm_exit_intr_error_code = fault->error_code;
|
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_EXCEPTION_NMI,
|
|
|
|
PF_VECTOR | INTR_TYPE_HARD_EXCEPTION |
|
|
|
|
INTR_INFO_DELIVER_CODE_MASK | INTR_INFO_VALID_MASK,
|
|
|
|
fault->address);
|
|
|
|
} else {
|
|
|
|
kvm_inject_page_fault(vcpu, fault);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool page_address_valid(struct kvm_vcpu *vcpu, gpa_t gpa)
|
|
|
|
{
|
|
|
|
return PAGE_ALIGNED(gpa) && !(gpa >> cpuid_maxphyaddr(vcpu));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_io_bitmap_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has(vmcs12, CPU_BASED_USE_IO_BITMAPS))
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!page_address_valid(vcpu, vmcs12->io_bitmap_a)) ||
|
|
|
|
CC(!page_address_valid(vcpu, vmcs12->io_bitmap_b)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_msr_bitmap_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has(vmcs12, CPU_BASED_USE_MSR_BITMAPS))
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!page_address_valid(vcpu, vmcs12->msr_bitmap)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_tpr_shadow_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has(vmcs12, CPU_BASED_TPR_SHADOW))
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!page_address_valid(vcpu, vmcs12->virtual_apic_page_addr)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if MSR is intercepted for L01 MSR bitmap.
|
|
|
|
*/
|
|
|
|
static bool msr_write_intercepted_l01(struct kvm_vcpu *vcpu, u32 msr)
|
|
|
|
{
|
|
|
|
unsigned long *msr_bitmap;
|
|
|
|
int f = sizeof(unsigned long);
|
|
|
|
|
|
|
|
if (!cpu_has_vmx_msr_bitmap())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
msr_bitmap = to_vmx(vcpu)->vmcs01.msr_bitmap;
|
|
|
|
|
|
|
|
if (msr <= 0x1fff) {
|
|
|
|
return !!test_bit(msr, msr_bitmap + 0x800 / f);
|
|
|
|
} else if ((msr >= 0xc0000000) && (msr <= 0xc0001fff)) {
|
|
|
|
msr &= 0x1fff;
|
|
|
|
return !!test_bit(msr, msr_bitmap + 0xc00 / f);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If a msr is allowed by L0, we should check whether it is allowed by L1.
|
|
|
|
* The corresponding bit will be cleared unless both of L0 and L1 allow it.
|
|
|
|
*/
|
|
|
|
static void nested_vmx_disable_intercept_for_msr(unsigned long *msr_bitmap_l1,
|
|
|
|
unsigned long *msr_bitmap_nested,
|
|
|
|
u32 msr, int type)
|
|
|
|
{
|
|
|
|
int f = sizeof(unsigned long);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See Intel PRM Vol. 3, 20.6.9 (MSR-Bitmap Address). Early manuals
|
|
|
|
* have the write-low and read-high bitmap offsets the wrong way round.
|
|
|
|
* We can control MSRs 0x00000000-0x00001fff and 0xc0000000-0xc0001fff.
|
|
|
|
*/
|
|
|
|
if (msr <= 0x1fff) {
|
|
|
|
if (type & MSR_TYPE_R &&
|
|
|
|
!test_bit(msr, msr_bitmap_l1 + 0x000 / f))
|
|
|
|
/* read-low */
|
|
|
|
__clear_bit(msr, msr_bitmap_nested + 0x000 / f);
|
|
|
|
|
|
|
|
if (type & MSR_TYPE_W &&
|
|
|
|
!test_bit(msr, msr_bitmap_l1 + 0x800 / f))
|
|
|
|
/* write-low */
|
|
|
|
__clear_bit(msr, msr_bitmap_nested + 0x800 / f);
|
|
|
|
|
|
|
|
} else if ((msr >= 0xc0000000) && (msr <= 0xc0001fff)) {
|
|
|
|
msr &= 0x1fff;
|
|
|
|
if (type & MSR_TYPE_R &&
|
|
|
|
!test_bit(msr, msr_bitmap_l1 + 0x400 / f))
|
|
|
|
/* read-high */
|
|
|
|
__clear_bit(msr, msr_bitmap_nested + 0x400 / f);
|
|
|
|
|
|
|
|
if (type & MSR_TYPE_W &&
|
|
|
|
!test_bit(msr, msr_bitmap_l1 + 0xc00 / f))
|
|
|
|
/* write-high */
|
|
|
|
__clear_bit(msr, msr_bitmap_nested + 0xc00 / f);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-02 13:55:59 +07:00
|
|
|
static inline void enable_x2apic_msr_intercepts(unsigned long *msr_bitmap) {
|
|
|
|
int msr;
|
|
|
|
|
|
|
|
for (msr = 0x800; msr <= 0x8ff; msr += BITS_PER_LONG) {
|
|
|
|
unsigned word = msr / BITS_PER_LONG;
|
|
|
|
|
|
|
|
msr_bitmap[word] = ~0;
|
|
|
|
msr_bitmap[word + (0x800 / sizeof(long))] = ~0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Merge L0's and L1's MSR bitmap, return false to indicate that
|
|
|
|
* we do not use the hardware.
|
|
|
|
*/
|
|
|
|
static inline bool nested_vmx_prepare_msr_bitmap(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
int msr;
|
|
|
|
unsigned long *msr_bitmap_l1;
|
|
|
|
unsigned long *msr_bitmap_l0 = to_vmx(vcpu)->nested.vmcs02.msr_bitmap;
|
2019-02-01 03:24:36 +07:00
|
|
|
struct kvm_host_map *map = &to_vmx(vcpu)->nested.msr_bitmap_map;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Nothing to do if the MSR bitmap is not in use. */
|
|
|
|
if (!cpu_has_vmx_msr_bitmap() ||
|
|
|
|
!nested_cpu_has(vmcs12, CPU_BASED_USE_MSR_BITMAPS))
|
|
|
|
return false;
|
|
|
|
|
2019-02-01 03:24:36 +07:00
|
|
|
if (kvm_vcpu_map(vcpu, gpa_to_gfn(vmcs12->msr_bitmap), map))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
|
2019-02-01 03:24:36 +07:00
|
|
|
msr_bitmap_l1 = (unsigned long *)map->hva;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-04-02 13:55:59 +07:00
|
|
|
/*
|
|
|
|
* To keep the control flow simple, pay eight 8-byte writes (sixteen
|
|
|
|
* 4-byte writes on 32-bit systems) up front to enable intercepts for
|
|
|
|
* the x2APIC MSR range and selectively disable them below.
|
|
|
|
*/
|
|
|
|
enable_x2apic_msr_intercepts(msr_bitmap_l0);
|
|
|
|
|
|
|
|
if (nested_cpu_has_virt_x2apic_mode(vmcs12)) {
|
|
|
|
if (nested_cpu_has_apic_reg_virt(vmcs12)) {
|
|
|
|
/*
|
|
|
|
* L0 need not intercept reads for MSRs between 0x800
|
|
|
|
* and 0x8ff, it just lets the processor take the value
|
|
|
|
* from the virtual-APIC page; take those 256 bits
|
|
|
|
* directly from the L1 bitmap.
|
|
|
|
*/
|
|
|
|
for (msr = 0x800; msr <= 0x8ff; msr += BITS_PER_LONG) {
|
|
|
|
unsigned word = msr / BITS_PER_LONG;
|
|
|
|
|
|
|
|
msr_bitmap_l0[word] = msr_bitmap_l1[word];
|
|
|
|
}
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
nested_vmx_disable_intercept_for_msr(
|
|
|
|
msr_bitmap_l1, msr_bitmap_l0,
|
2019-04-02 13:55:59 +07:00
|
|
|
X2APIC_MSR(APIC_TASKPRI),
|
2019-04-02 13:56:00 +07:00
|
|
|
MSR_TYPE_R | MSR_TYPE_W);
|
2019-04-02 13:55:59 +07:00
|
|
|
|
|
|
|
if (nested_cpu_has_vid(vmcs12)) {
|
|
|
|
nested_vmx_disable_intercept_for_msr(
|
|
|
|
msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
X2APIC_MSR(APIC_EOI),
|
|
|
|
MSR_TYPE_W);
|
|
|
|
nested_vmx_disable_intercept_for_msr(
|
|
|
|
msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
X2APIC_MSR(APIC_SELF_IPI),
|
|
|
|
MSR_TYPE_W);
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
2019-05-08 21:32:15 +07:00
|
|
|
/* KVM unconditionally exposes the FS/GS base MSRs to L1. */
|
|
|
|
nested_vmx_disable_intercept_for_msr(msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
MSR_FS_BASE, MSR_TYPE_RW);
|
|
|
|
|
|
|
|
nested_vmx_disable_intercept_for_msr(msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
MSR_GS_BASE, MSR_TYPE_RW);
|
|
|
|
|
|
|
|
nested_vmx_disable_intercept_for_msr(msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
MSR_KERNEL_GS_BASE, MSR_TYPE_RW);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checking the L0->L1 bitmap is trying to verify two things:
|
|
|
|
*
|
|
|
|
* 1. L0 gave a permission to L1 to actually passthrough the MSR. This
|
|
|
|
* ensures that we do not accidentally generate an L02 MSR bitmap
|
|
|
|
* from the L12 MSR bitmap that is too permissive.
|
|
|
|
* 2. That L1 or L2s have actually used the MSR. This avoids
|
|
|
|
* unnecessarily merging of the bitmap if the MSR is unused. This
|
|
|
|
* works properly because we only update the L01 MSR bitmap lazily.
|
|
|
|
* So even if L0 should pass L1 these MSRs, the L01 bitmap is only
|
|
|
|
* updated to reflect this when L1 (or its L2s) actually write to
|
|
|
|
* the MSR.
|
|
|
|
*/
|
|
|
|
if (!msr_write_intercepted_l01(vcpu, MSR_IA32_SPEC_CTRL))
|
2018-12-04 04:53:18 +07:00
|
|
|
nested_vmx_disable_intercept_for_msr(
|
|
|
|
msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
MSR_IA32_SPEC_CTRL,
|
|
|
|
MSR_TYPE_R | MSR_TYPE_W);
|
|
|
|
|
2019-05-08 21:32:15 +07:00
|
|
|
if (!msr_write_intercepted_l01(vcpu, MSR_IA32_PRED_CMD))
|
2018-12-04 04:53:18 +07:00
|
|
|
nested_vmx_disable_intercept_for_msr(
|
|
|
|
msr_bitmap_l1, msr_bitmap_l0,
|
|
|
|
MSR_IA32_PRED_CMD,
|
|
|
|
MSR_TYPE_W);
|
|
|
|
|
2019-02-01 03:24:36 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &to_vmx(vcpu)->nested.msr_bitmap_map, false);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_cache_shadow_vmcs12(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
2019-02-01 03:24:41 +07:00
|
|
|
struct kvm_host_map map;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vmcs12 *shadow;
|
|
|
|
|
|
|
|
if (!nested_cpu_has_shadow_vmcs(vmcs12) ||
|
|
|
|
vmcs12->vmcs_link_pointer == -1ull)
|
|
|
|
return;
|
|
|
|
|
|
|
|
shadow = get_shadow_vmcs12(vcpu);
|
|
|
|
|
2019-02-01 03:24:41 +07:00
|
|
|
if (kvm_vcpu_map(vcpu, gpa_to_gfn(vmcs12->vmcs_link_pointer), &map))
|
|
|
|
return;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-02-01 03:24:41 +07:00
|
|
|
memcpy(shadow, map.hva, VMCS12_SIZE);
|
|
|
|
kvm_vcpu_unmap(vcpu, &map, false);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_flush_cached_shadow_vmcs12(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
if (!nested_cpu_has_shadow_vmcs(vmcs12) ||
|
|
|
|
vmcs12->vmcs_link_pointer == -1ull)
|
|
|
|
return;
|
|
|
|
|
|
|
|
kvm_write_guest(vmx->vcpu.kvm, vmcs12->vmcs_link_pointer,
|
|
|
|
get_shadow_vmcs12(vcpu), VMCS12_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In nested virtualization, check if L1 has set
|
|
|
|
* VM_EXIT_ACK_INTR_ON_EXIT
|
|
|
|
*/
|
|
|
|
static bool nested_exit_intr_ack_set(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
return get_vmcs12(vcpu)->vm_exit_controls &
|
|
|
|
VM_EXIT_ACK_INTR_ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool nested_exit_on_nmi(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
return nested_cpu_has_nmi_exiting(get_vmcs12(vcpu));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_apic_access_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (nested_cpu_has2(vmcs12, SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(!page_address_valid(vcpu, vmcs12->apic_access_addr)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_apicv_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has_virt_x2apic_mode(vmcs12) &&
|
|
|
|
!nested_cpu_has_apic_reg_virt(vmcs12) &&
|
|
|
|
!nested_cpu_has_vid(vmcs12) &&
|
|
|
|
!nested_cpu_has_posted_intr(vmcs12))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If virtualize x2apic mode is enabled,
|
|
|
|
* virtualize apic access must be disabled.
|
|
|
|
*/
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_cpu_has_virt_x2apic_mode(vmcs12) &&
|
|
|
|
nested_cpu_has2(vmcs12, SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If virtual interrupt delivery is enabled,
|
|
|
|
* we must exit on external interrupts.
|
|
|
|
*/
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_cpu_has_vid(vmcs12) && !nested_exit_on_intr(vcpu)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bits 15:8 should be zero in posted_intr_nv,
|
|
|
|
* the descriptor address has been already checked
|
|
|
|
* in nested_get_vmcs12_pages.
|
|
|
|
*
|
|
|
|
* bits 5:0 of posted_intr_desc_addr should be zero.
|
|
|
|
*/
|
|
|
|
if (nested_cpu_has_posted_intr(vmcs12) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
(CC(!nested_cpu_has_vid(vmcs12)) ||
|
|
|
|
CC(!nested_exit_intr_ack_set(vcpu)) ||
|
|
|
|
CC((vmcs12->posted_intr_nv & 0xff00)) ||
|
|
|
|
CC((vmcs12->posted_intr_desc_addr & 0x3f)) ||
|
|
|
|
CC((vmcs12->posted_intr_desc_addr >> cpuid_maxphyaddr(vcpu)))))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* tpr shadow is needed by all apicv features. */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cpu_has(vmcs12, CPU_BASED_TPR_SHADOW)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_msr_switch(struct kvm_vcpu *vcpu,
|
2018-12-13 01:30:08 +07:00
|
|
|
u32 count, u64 addr)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
int maxphyaddr;
|
|
|
|
|
|
|
|
if (count == 0)
|
|
|
|
return 0;
|
|
|
|
maxphyaddr = cpuid_maxphyaddr(vcpu);
|
|
|
|
if (!IS_ALIGNED(addr, 16) || addr >> maxphyaddr ||
|
2018-12-13 01:30:08 +07:00
|
|
|
(addr + count * sizeof(struct vmx_msr_entry) - 1) >> maxphyaddr)
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
2018-12-13 01:30:08 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:09 +07:00
|
|
|
static int nested_vmx_check_exit_msr_switch_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_vmx_check_msr_switch(vcpu,
|
|
|
|
vmcs12->vm_exit_msr_load_count,
|
|
|
|
vmcs12->vm_exit_msr_load_addr)) ||
|
|
|
|
CC(nested_vmx_check_msr_switch(vcpu,
|
|
|
|
vmcs12->vm_exit_msr_store_count,
|
|
|
|
vmcs12->vm_exit_msr_store_addr)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
2018-12-13 01:30:08 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:10 +07:00
|
|
|
static int nested_vmx_check_entry_msr_switch_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
2018-12-13 01:30:09 +07:00
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_vmx_check_msr_switch(vcpu,
|
|
|
|
vmcs12->vm_entry_msr_load_count,
|
|
|
|
vmcs12->vm_entry_msr_load_addr)))
|
2018-12-13 01:30:09 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static int nested_vmx_check_pml_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has_pml(vmcs12))
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cpu_has_ept(vmcs12)) ||
|
|
|
|
CC(!page_address_valid(vcpu, vmcs12->pml_address)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_unrestricted_guest_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_cpu_has2(vmcs12, SECONDARY_EXEC_UNRESTRICTED_GUEST) &&
|
|
|
|
!nested_cpu_has_ept(vmcs12)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_mode_based_ept_exec_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(nested_cpu_has2(vmcs12, SECONDARY_EXEC_MODE_BASED_EPT_EXEC) &&
|
|
|
|
!nested_cpu_has_ept(vmcs12)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_shadow_vmcs_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (!nested_cpu_has_shadow_vmcs(vmcs12))
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!page_address_valid(vcpu, vmcs12->vmread_bitmap)) ||
|
|
|
|
CC(!page_address_valid(vcpu, vmcs12->vmwrite_bitmap)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_msr_check_common(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmx_msr_entry *e)
|
|
|
|
{
|
|
|
|
/* x2APIC MSR accesses are not allowed */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vcpu->arch.apic_base & X2APIC_ENABLE && e->index >> 8 == 0x8))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(e->index == MSR_IA32_UCODE_WRITE) || /* SDM Table 35-2 */
|
|
|
|
CC(e->index == MSR_IA32_UCODE_REV))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(e->reserved != 0))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_load_msr_check(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmx_msr_entry *e)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(e->index == MSR_FS_BASE) ||
|
|
|
|
CC(e->index == MSR_GS_BASE) ||
|
|
|
|
CC(e->index == MSR_IA32_SMM_MONITOR_CTL) || /* SMM is not supported */
|
2018-12-04 04:53:18 +07:00
|
|
|
nested_vmx_msr_check_common(vcpu, e))
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_store_msr_check(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmx_msr_entry *e)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(e->index == MSR_IA32_SMBASE) || /* SMM is not supported */
|
2018-12-04 04:53:18 +07:00
|
|
|
nested_vmx_msr_check_common(vcpu, e))
|
|
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
static u32 nested_vmx_max_atomic_switch_msrs(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u64 vmx_misc = vmx_control_msr(vmx->nested.msrs.misc_low,
|
|
|
|
vmx->nested.msrs.misc_high);
|
|
|
|
|
|
|
|
return (vmx_misc_max_msr(vmx_misc) + 1) * VMX_MISC_MSR_LIST_MULTIPLIER;
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Load guest's/host's msr at nested entry/exit.
|
|
|
|
* return 0 for success, entry index for failure.
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
*
|
|
|
|
* One of the failure modes for MSR load/store is when a list exceeds the
|
|
|
|
* virtual hardware's capacity. To maintain compatibility with hardware inasmuch
|
|
|
|
* as possible, process all valid entries before failing rather than precheck
|
|
|
|
* for a capacity violation.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
|
|
|
static u32 nested_vmx_load_msr(struct kvm_vcpu *vcpu, u64 gpa, u32 count)
|
|
|
|
{
|
|
|
|
u32 i;
|
|
|
|
struct vmx_msr_entry e;
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
u32 max_msr_list_size = nested_vmx_max_atomic_switch_msrs(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
if (unlikely(i >= max_msr_list_size))
|
|
|
|
goto fail;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (kvm_vcpu_read_guest(vcpu, gpa + i * sizeof(e),
|
|
|
|
&e, sizeof(e))) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s cannot read MSR entry (%u, 0x%08llx)\n",
|
|
|
|
__func__, i, gpa + i * sizeof(e));
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if (nested_vmx_load_msr_check(vcpu, &e)) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s check failed (%u, 0x%x, 0x%x)\n",
|
|
|
|
__func__, i, e.index, e.reserved);
|
|
|
|
goto fail;
|
|
|
|
}
|
2019-09-06 04:22:54 +07:00
|
|
|
if (kvm_set_msr(vcpu, e.index, e.value)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s cannot write MSR (%u, 0x%x, 0x%llx)\n",
|
|
|
|
__func__, i, e.index, e.value);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
fail:
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
|
2019-11-08 12:14:39 +07:00
|
|
|
static bool nested_vmx_get_vmexit_msr_value(struct kvm_vcpu *vcpu,
|
|
|
|
u32 msr_index,
|
|
|
|
u64 *data)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the L0 hypervisor stored a more accurate value for the TSC that
|
|
|
|
* does not include the time taken for emulation of the L2->L1
|
|
|
|
* VM-exit in L0, use the more accurate value.
|
|
|
|
*/
|
|
|
|
if (msr_index == MSR_IA32_TSC) {
|
|
|
|
int index = vmx_find_msr_index(&vmx->msr_autostore.guest,
|
|
|
|
MSR_IA32_TSC);
|
|
|
|
|
|
|
|
if (index >= 0) {
|
|
|
|
u64 val = vmx->msr_autostore.guest.val[index].value;
|
|
|
|
|
|
|
|
*data = kvm_read_l1_tsc(vcpu, val);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (kvm_get_msr(vcpu, msr_index, data)) {
|
|
|
|
pr_debug_ratelimited("%s cannot read MSR (0x%x)\n", __func__,
|
|
|
|
msr_index);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-11-08 12:14:36 +07:00
|
|
|
static bool read_and_check_msr_entry(struct kvm_vcpu *vcpu, u64 gpa, int i,
|
|
|
|
struct vmx_msr_entry *e)
|
|
|
|
{
|
|
|
|
if (kvm_vcpu_read_guest(vcpu,
|
|
|
|
gpa + i * sizeof(*e),
|
|
|
|
e, 2 * sizeof(u32))) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s cannot read MSR entry (%u, 0x%08llx)\n",
|
|
|
|
__func__, i, gpa + i * sizeof(*e));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (nested_vmx_store_msr_check(vcpu, e)) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s check failed (%u, 0x%x, 0x%x)\n",
|
|
|
|
__func__, i, e->index, e->reserved);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static int nested_vmx_store_msr(struct kvm_vcpu *vcpu, u64 gpa, u32 count)
|
|
|
|
{
|
2019-09-06 04:22:54 +07:00
|
|
|
u64 data;
|
2018-12-04 04:53:18 +07:00
|
|
|
u32 i;
|
|
|
|
struct vmx_msr_entry e;
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
u32 max_msr_list_size = nested_vmx_max_atomic_switch_msrs(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
kvm: nvmx: limit atomic switch MSRs
Allowing an unlimited number of MSRs to be specified via the VMX
load/store MSR lists (e.g., vm-entry MSR load list) is bad for two
reasons. First, a guest can specify an unreasonable number of MSRs,
forcing KVM to process all of them in software. Second, the SDM bounds
the number of MSRs allowed to be packed into the atomic switch MSR lists.
Quoting the "Miscellaneous Data" section in the "VMX Capability
Reporting Facility" appendix:
"Bits 27:25 is used to compute the recommended maximum number of MSRs
that should appear in the VM-exit MSR-store list, the VM-exit MSR-load
list, or the VM-entry MSR-load list. Specifically, if the value bits
27:25 of IA32_VMX_MISC is N, then 512 * (N + 1) is the recommended
maximum number of MSRs to be included in each list. If the limit is
exceeded, undefined processor behavior may result (including a machine
check during the VMX transition)."
Because KVM needs to protect itself and can't model "undefined processor
behavior", arbitrarily force a VM-entry to fail due to MSR loading when
the MSR load list is too large. Similarly, trigger an abort during a VM
exit that encounters an MSR load list or MSR store list that is too large.
The MSR list size is intentionally not pre-checked so as to maintain
compatibility with hardware inasmuch as possible.
Test these new checks with the kvm-unit-test "x86: nvmx: test max atomic
switch MSRs".
Suggested-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Signed-off-by: Marc Orr <marcorr@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-09-18 01:50:57 +07:00
|
|
|
if (unlikely(i >= max_msr_list_size))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-11-08 12:14:36 +07:00
|
|
|
if (!read_and_check_msr_entry(vcpu, gpa, i, &e))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
2019-11-08 12:14:36 +07:00
|
|
|
|
2019-11-08 12:14:39 +07:00
|
|
|
if (!nested_vmx_get_vmexit_msr_value(vcpu, e.index, &data))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
2019-11-08 12:14:39 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (kvm_vcpu_write_guest(vcpu,
|
|
|
|
gpa + i * sizeof(e) +
|
|
|
|
offsetof(struct vmx_msr_entry, value),
|
2019-09-06 04:22:54 +07:00
|
|
|
&data, sizeof(data))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s cannot write MSR (%u, 0x%x, 0x%llx)\n",
|
2019-09-06 04:22:54 +07:00
|
|
|
__func__, i, e.index, data);
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-11-08 12:14:39 +07:00
|
|
|
static bool nested_msr_store_list_has_msr(struct kvm_vcpu *vcpu, u32 msr_index)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
u32 count = vmcs12->vm_exit_msr_store_count;
|
|
|
|
u64 gpa = vmcs12->vm_exit_msr_store_addr;
|
|
|
|
struct vmx_msr_entry e;
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
if (!read_and_check_msr_entry(vcpu, gpa, i, &e))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (e.index == msr_index)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void prepare_vmx_msr_autostore_list(struct kvm_vcpu *vcpu,
|
|
|
|
u32 msr_index)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmx_msrs *autostore = &vmx->msr_autostore.guest;
|
|
|
|
bool in_vmcs12_store_list;
|
|
|
|
int msr_autostore_index;
|
|
|
|
bool in_autostore_list;
|
|
|
|
int last;
|
|
|
|
|
|
|
|
msr_autostore_index = vmx_find_msr_index(autostore, msr_index);
|
|
|
|
in_autostore_list = msr_autostore_index >= 0;
|
|
|
|
in_vmcs12_store_list = nested_msr_store_list_has_msr(vcpu, msr_index);
|
|
|
|
|
|
|
|
if (in_vmcs12_store_list && !in_autostore_list) {
|
|
|
|
if (autostore->nr == NR_LOADSTORE_MSRS) {
|
|
|
|
/*
|
|
|
|
* Emulated VMEntry does not fail here. Instead a less
|
|
|
|
* accurate value will be returned by
|
|
|
|
* nested_vmx_get_vmexit_msr_value() using kvm_get_msr()
|
|
|
|
* instead of reading the value from the vmcs02 VMExit
|
|
|
|
* MSR-store area.
|
|
|
|
*/
|
|
|
|
pr_warn_ratelimited(
|
|
|
|
"Not enough msr entries in msr_autostore. Can't add msr %x\n",
|
|
|
|
msr_index);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
last = autostore->nr++;
|
|
|
|
autostore->val[last].index = msr_index;
|
|
|
|
} else if (!in_vmcs12_store_list && in_autostore_list) {
|
|
|
|
last = --autostore->nr;
|
|
|
|
autostore->val[msr_autostore_index] = autostore->val[last];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static bool nested_cr3_valid(struct kvm_vcpu *vcpu, unsigned long val)
|
|
|
|
{
|
|
|
|
unsigned long invalid_mask;
|
|
|
|
|
|
|
|
invalid_mask = (~0ULL) << cpuid_maxphyaddr(vcpu);
|
|
|
|
return (val & invalid_mask) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load guest's/host's cr3 at nested entry/exit. nested_ept is true if we are
|
|
|
|
* emulating VM entry into a guest with EPT enabled.
|
|
|
|
* Returns 0 on success, 1 on failure. Invalid state exit qualification code
|
|
|
|
* is assigned to entry_failure_code on failure.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_load_cr3(struct kvm_vcpu *vcpu, unsigned long cr3, bool nested_ept,
|
|
|
|
u32 *entry_failure_code)
|
|
|
|
{
|
|
|
|
if (cr3 != kvm_read_cr3(vcpu) || (!nested_ept && pdptrs_changed(vcpu))) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cr3_valid(vcpu, cr3))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
*entry_failure_code = ENTRY_FAIL_DEFAULT;
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If PAE paging and EPT are both on, CR3 is not used by the CPU and
|
|
|
|
* must not be dereferenced.
|
|
|
|
*/
|
2019-06-06 23:52:44 +07:00
|
|
|
if (is_pae_paging(vcpu) && !nested_ept) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!load_pdptrs(vcpu, vcpu->arch.walk_mmu, cr3))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
*entry_failure_code = ENTRY_FAIL_PDPTE;
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nested_ept)
|
|
|
|
kvm_mmu_new_cr3(vcpu, cr3, false);
|
|
|
|
|
|
|
|
vcpu->arch.cr3 = cr3;
|
2019-09-28 04:45:22 +07:00
|
|
|
kvm_register_mark_available(vcpu, VCPU_EXREG_CR3);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
kvm_init_mmu(vcpu, false);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns if KVM is able to config CPU to tag TLB entries
|
|
|
|
* populated by L2 differently than TLB entries populated
|
|
|
|
* by L1.
|
|
|
|
*
|
2019-11-20 19:24:52 +07:00
|
|
|
* If L0 uses EPT, L1 and L2 run with different EPTP because
|
|
|
|
* guest_mode is part of kvm_mmu_page_role. Thus, TLB entries
|
|
|
|
* are tagged with different EPTP.
|
2018-12-04 04:53:18 +07:00
|
|
|
*
|
|
|
|
* If L1 uses VPID and we allocated a vpid02, TLB entries are tagged
|
|
|
|
* with different VPID (L1 entries are tagged with vmx->vpid
|
|
|
|
* while L2 entries are tagged with vmx->nested.vpid02).
|
|
|
|
*/
|
|
|
|
static bool nested_has_guest_tlb_tag(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
2019-11-20 19:24:52 +07:00
|
|
|
return enable_ept ||
|
2018-12-04 04:53:18 +07:00
|
|
|
(nested_cpu_has_vpid(vmcs12) && to_vmx(vcpu)->nested.vpid02);
|
|
|
|
}
|
|
|
|
|
|
|
|
static u16 nested_get_vpid02(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
return vmx->nested.vpid02 ? vmx->nested.vpid02 : vmx->vpid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_bitwise_subset(u64 superset, u64 subset, u64 mask)
|
|
|
|
{
|
|
|
|
superset &= mask;
|
|
|
|
subset &= mask;
|
|
|
|
|
|
|
|
return (superset | subset) == superset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_restore_vmx_basic(struct vcpu_vmx *vmx, u64 data)
|
|
|
|
{
|
|
|
|
const u64 feature_and_reserved =
|
|
|
|
/* feature (except bit 48; see below) */
|
|
|
|
BIT_ULL(49) | BIT_ULL(54) | BIT_ULL(55) |
|
|
|
|
/* reserved */
|
|
|
|
BIT_ULL(31) | GENMASK_ULL(47, 45) | GENMASK_ULL(63, 56);
|
|
|
|
u64 vmx_basic = vmx->nested.msrs.basic;
|
|
|
|
|
|
|
|
if (!is_bitwise_subset(vmx_basic, data, feature_and_reserved))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* KVM does not emulate a version of VMX that constrains physical
|
|
|
|
* addresses of VMX structures (e.g. VMCS) to 32-bits.
|
|
|
|
*/
|
|
|
|
if (data & BIT_ULL(48))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (vmx_basic_vmcs_revision_id(vmx_basic) !=
|
|
|
|
vmx_basic_vmcs_revision_id(data))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (vmx_basic_vmcs_size(vmx_basic) > vmx_basic_vmcs_size(data))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
vmx->nested.msrs.basic = data;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vmx_restore_control_msr(struct vcpu_vmx *vmx, u32 msr_index, u64 data)
|
|
|
|
{
|
|
|
|
u64 supported;
|
|
|
|
u32 *lowp, *highp;
|
|
|
|
|
|
|
|
switch (msr_index) {
|
|
|
|
case MSR_IA32_VMX_TRUE_PINBASED_CTLS:
|
|
|
|
lowp = &vmx->nested.msrs.pinbased_ctls_low;
|
|
|
|
highp = &vmx->nested.msrs.pinbased_ctls_high;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_PROCBASED_CTLS:
|
|
|
|
lowp = &vmx->nested.msrs.procbased_ctls_low;
|
|
|
|
highp = &vmx->nested.msrs.procbased_ctls_high;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_EXIT_CTLS:
|
|
|
|
lowp = &vmx->nested.msrs.exit_ctls_low;
|
|
|
|
highp = &vmx->nested.msrs.exit_ctls_high;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_ENTRY_CTLS:
|
|
|
|
lowp = &vmx->nested.msrs.entry_ctls_low;
|
|
|
|
highp = &vmx->nested.msrs.entry_ctls_high;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_PROCBASED_CTLS2:
|
|
|
|
lowp = &vmx->nested.msrs.secondary_ctls_low;
|
|
|
|
highp = &vmx->nested.msrs.secondary_ctls_high;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
|
|
|
|
supported = vmx_control_msr(*lowp, *highp);
|
|
|
|
|
|
|
|
/* Check must-be-1 bits are still 1. */
|
|
|
|
if (!is_bitwise_subset(data, supported, GENMASK_ULL(31, 0)))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Check must-be-0 bits are still 0. */
|
|
|
|
if (!is_bitwise_subset(supported, data, GENMASK_ULL(63, 32)))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*lowp = data;
|
|
|
|
*highp = data >> 32;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_restore_vmx_misc(struct vcpu_vmx *vmx, u64 data)
|
|
|
|
{
|
|
|
|
const u64 feature_and_reserved_bits =
|
|
|
|
/* feature */
|
|
|
|
BIT_ULL(5) | GENMASK_ULL(8, 6) | BIT_ULL(14) | BIT_ULL(15) |
|
|
|
|
BIT_ULL(28) | BIT_ULL(29) | BIT_ULL(30) |
|
|
|
|
/* reserved */
|
|
|
|
GENMASK_ULL(13, 9) | BIT_ULL(31);
|
|
|
|
u64 vmx_misc;
|
|
|
|
|
|
|
|
vmx_misc = vmx_control_msr(vmx->nested.msrs.misc_low,
|
|
|
|
vmx->nested.msrs.misc_high);
|
|
|
|
|
|
|
|
if (!is_bitwise_subset(vmx_misc, data, feature_and_reserved_bits))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if ((vmx->nested.msrs.pinbased_ctls_high &
|
|
|
|
PIN_BASED_VMX_PREEMPTION_TIMER) &&
|
|
|
|
vmx_misc_preemption_timer_rate(data) !=
|
|
|
|
vmx_misc_preemption_timer_rate(vmx_misc))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (vmx_misc_cr3_count(data) > vmx_misc_cr3_count(vmx_misc))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (vmx_misc_max_msr(data) > vmx_misc_max_msr(vmx_misc))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (vmx_misc_mseg_revid(data) != vmx_misc_mseg_revid(vmx_misc))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
vmx->nested.msrs.misc_low = data;
|
|
|
|
vmx->nested.msrs.misc_high = data >> 32;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_restore_vmx_ept_vpid_cap(struct vcpu_vmx *vmx, u64 data)
|
|
|
|
{
|
|
|
|
u64 vmx_ept_vpid_cap;
|
|
|
|
|
|
|
|
vmx_ept_vpid_cap = vmx_control_msr(vmx->nested.msrs.ept_caps,
|
|
|
|
vmx->nested.msrs.vpid_caps);
|
|
|
|
|
|
|
|
/* Every bit is either reserved or a feature bit. */
|
|
|
|
if (!is_bitwise_subset(vmx_ept_vpid_cap, data, -1ULL))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
vmx->nested.msrs.ept_caps = data;
|
|
|
|
vmx->nested.msrs.vpid_caps = data >> 32;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_restore_fixed0_msr(struct vcpu_vmx *vmx, u32 msr_index, u64 data)
|
|
|
|
{
|
|
|
|
u64 *msr;
|
|
|
|
|
|
|
|
switch (msr_index) {
|
|
|
|
case MSR_IA32_VMX_CR0_FIXED0:
|
|
|
|
msr = &vmx->nested.msrs.cr0_fixed0;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_CR4_FIXED0:
|
|
|
|
msr = &vmx->nested.msrs.cr4_fixed0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 1 bits (which indicates bits which "must-be-1" during VMX operation)
|
|
|
|
* must be 1 in the restored value.
|
|
|
|
*/
|
|
|
|
if (!is_bitwise_subset(data, *msr, -1ULL))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*msr = data;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Called when userspace is restoring VMX MSRs.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, non-0 otherwise.
|
|
|
|
*/
|
|
|
|
int vmx_set_vmx_msr(struct kvm_vcpu *vcpu, u32 msr_index, u64 data)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Don't allow changes to the VMX capability MSRs while the vCPU
|
|
|
|
* is in VMX operation.
|
|
|
|
*/
|
|
|
|
if (vmx->nested.vmxon)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
switch (msr_index) {
|
|
|
|
case MSR_IA32_VMX_BASIC:
|
|
|
|
return vmx_restore_vmx_basic(vmx, data);
|
|
|
|
case MSR_IA32_VMX_PINBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_PROCBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_EXIT_CTLS:
|
|
|
|
case MSR_IA32_VMX_ENTRY_CTLS:
|
|
|
|
/*
|
|
|
|
* The "non-true" VMX capability MSRs are generated from the
|
|
|
|
* "true" MSRs, so we do not support restoring them directly.
|
|
|
|
*
|
|
|
|
* If userspace wants to emulate VMX_BASIC[55]=0, userspace
|
|
|
|
* should restore the "true" MSRs with the must-be-1 bits
|
|
|
|
* set according to the SDM Vol 3. A.2 "RESERVED CONTROLS AND
|
|
|
|
* DEFAULT SETTINGS".
|
|
|
|
*/
|
|
|
|
return -EINVAL;
|
|
|
|
case MSR_IA32_VMX_TRUE_PINBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_TRUE_PROCBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_TRUE_EXIT_CTLS:
|
|
|
|
case MSR_IA32_VMX_TRUE_ENTRY_CTLS:
|
|
|
|
case MSR_IA32_VMX_PROCBASED_CTLS2:
|
|
|
|
return vmx_restore_control_msr(vmx, msr_index, data);
|
|
|
|
case MSR_IA32_VMX_MISC:
|
|
|
|
return vmx_restore_vmx_misc(vmx, data);
|
|
|
|
case MSR_IA32_VMX_CR0_FIXED0:
|
|
|
|
case MSR_IA32_VMX_CR4_FIXED0:
|
|
|
|
return vmx_restore_fixed0_msr(vmx, msr_index, data);
|
|
|
|
case MSR_IA32_VMX_CR0_FIXED1:
|
|
|
|
case MSR_IA32_VMX_CR4_FIXED1:
|
|
|
|
/*
|
|
|
|
* These MSRs are generated based on the vCPU's CPUID, so we
|
|
|
|
* do not support restoring them directly.
|
|
|
|
*/
|
|
|
|
return -EINVAL;
|
|
|
|
case MSR_IA32_VMX_EPT_VPID_CAP:
|
|
|
|
return vmx_restore_vmx_ept_vpid_cap(vmx, data);
|
|
|
|
case MSR_IA32_VMX_VMCS_ENUM:
|
|
|
|
vmx->nested.msrs.vmcs_enum = data;
|
|
|
|
return 0;
|
2019-07-02 19:40:40 +07:00
|
|
|
case MSR_IA32_VMX_VMFUNC:
|
|
|
|
if (data & ~vmx->nested.msrs.vmfunc_controls)
|
|
|
|
return -EINVAL;
|
|
|
|
vmx->nested.msrs.vmfunc_controls = data;
|
|
|
|
return 0;
|
2018-12-04 04:53:18 +07:00
|
|
|
default:
|
|
|
|
/*
|
|
|
|
* The rest of the VMX capability MSRs do not support restore.
|
|
|
|
*/
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns 0 on success, non-0 otherwise. */
|
|
|
|
int vmx_get_vmx_msr(struct nested_vmx_msrs *msrs, u32 msr_index, u64 *pdata)
|
|
|
|
{
|
|
|
|
switch (msr_index) {
|
|
|
|
case MSR_IA32_VMX_BASIC:
|
|
|
|
*pdata = msrs->basic;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_PINBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_PINBASED_CTLS:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->pinbased_ctls_low,
|
|
|
|
msrs->pinbased_ctls_high);
|
|
|
|
if (msr_index == MSR_IA32_VMX_PINBASED_CTLS)
|
|
|
|
*pdata |= PIN_BASED_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_PROCBASED_CTLS:
|
|
|
|
case MSR_IA32_VMX_PROCBASED_CTLS:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->procbased_ctls_low,
|
|
|
|
msrs->procbased_ctls_high);
|
|
|
|
if (msr_index == MSR_IA32_VMX_PROCBASED_CTLS)
|
|
|
|
*pdata |= CPU_BASED_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_EXIT_CTLS:
|
|
|
|
case MSR_IA32_VMX_EXIT_CTLS:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->exit_ctls_low,
|
|
|
|
msrs->exit_ctls_high);
|
|
|
|
if (msr_index == MSR_IA32_VMX_EXIT_CTLS)
|
|
|
|
*pdata |= VM_EXIT_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_TRUE_ENTRY_CTLS:
|
|
|
|
case MSR_IA32_VMX_ENTRY_CTLS:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->entry_ctls_low,
|
|
|
|
msrs->entry_ctls_high);
|
|
|
|
if (msr_index == MSR_IA32_VMX_ENTRY_CTLS)
|
|
|
|
*pdata |= VM_ENTRY_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_MISC:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->misc_low,
|
|
|
|
msrs->misc_high);
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_CR0_FIXED0:
|
|
|
|
*pdata = msrs->cr0_fixed0;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_CR0_FIXED1:
|
|
|
|
*pdata = msrs->cr0_fixed1;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_CR4_FIXED0:
|
|
|
|
*pdata = msrs->cr4_fixed0;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_CR4_FIXED1:
|
|
|
|
*pdata = msrs->cr4_fixed1;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_VMCS_ENUM:
|
|
|
|
*pdata = msrs->vmcs_enum;
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_PROCBASED_CTLS2:
|
|
|
|
*pdata = vmx_control_msr(
|
|
|
|
msrs->secondary_ctls_low,
|
|
|
|
msrs->secondary_ctls_high);
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_EPT_VPID_CAP:
|
|
|
|
*pdata = msrs->ept_caps |
|
|
|
|
((u64)msrs->vpid_caps << 32);
|
|
|
|
break;
|
|
|
|
case MSR_IA32_VMX_VMFUNC:
|
|
|
|
*pdata = msrs->vmfunc_controls;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
* Copy the writable VMCS shadow fields back to the VMCS12, in case they have
|
|
|
|
* been modified by the L1 guest. Note, "writable" in this context means
|
|
|
|
* "writable by the guest", i.e. tagged SHADOW_FIELD_RW; the set of
|
|
|
|
* fields tagged SHADOW_FIELD_RO may or may not align with the "read-only"
|
|
|
|
* VM-exit information fields (which are actually writable if the vCPU is
|
|
|
|
* configured to support "VMWRITE to any supported field in the VMCS").
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
|
|
|
static void copy_shadow_to_vmcs12(struct vcpu_vmx *vmx)
|
|
|
|
{
|
|
|
|
struct vmcs *shadow_vmcs = vmx->vmcs01.shadow_vmcs;
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(&vmx->vcpu);
|
2019-05-07 22:36:25 +07:00
|
|
|
struct shadow_vmcs_field field;
|
|
|
|
unsigned long val;
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
int i;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-07-19 23:41:10 +07:00
|
|
|
if (WARN_ON(!shadow_vmcs))
|
|
|
|
return;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
preempt_disable();
|
|
|
|
|
|
|
|
vmcs_load(shadow_vmcs);
|
|
|
|
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
for (i = 0; i < max_shadow_read_write_fields; i++) {
|
|
|
|
field = shadow_read_write_fields[i];
|
2019-05-07 22:36:25 +07:00
|
|
|
val = __vmcs_readl(field.encoding);
|
|
|
|
vmcs12_write_any(vmcs12, field.encoding, field.offset, val);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
vmcs_clear(shadow_vmcs);
|
|
|
|
vmcs_load(vmx->loaded_vmcs->vmcs);
|
|
|
|
|
|
|
|
preempt_enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void copy_vmcs12_to_shadow(struct vcpu_vmx *vmx)
|
|
|
|
{
|
2019-05-07 22:36:25 +07:00
|
|
|
const struct shadow_vmcs_field *fields[] = {
|
2018-12-04 04:53:18 +07:00
|
|
|
shadow_read_write_fields,
|
|
|
|
shadow_read_only_fields
|
|
|
|
};
|
|
|
|
const int max_fields[] = {
|
|
|
|
max_shadow_read_write_fields,
|
|
|
|
max_shadow_read_only_fields
|
|
|
|
};
|
|
|
|
struct vmcs *shadow_vmcs = vmx->vmcs01.shadow_vmcs;
|
2019-05-07 22:36:25 +07:00
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(&vmx->vcpu);
|
|
|
|
struct shadow_vmcs_field field;
|
|
|
|
unsigned long val;
|
|
|
|
int i, q;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-07-19 23:41:10 +07:00
|
|
|
if (WARN_ON(!shadow_vmcs))
|
|
|
|
return;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_load(shadow_vmcs);
|
|
|
|
|
|
|
|
for (q = 0; q < ARRAY_SIZE(fields); q++) {
|
|
|
|
for (i = 0; i < max_fields[q]; i++) {
|
|
|
|
field = fields[q][i];
|
2019-05-07 22:36:25 +07:00
|
|
|
val = vmcs12_read_any(vmcs12, field.encoding,
|
|
|
|
field.offset);
|
|
|
|
__vmcs_writel(field.encoding, val);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs_clear(shadow_vmcs);
|
|
|
|
vmcs_load(vmx->loaded_vmcs->vmcs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int copy_enlightened_to_vmcs12(struct vcpu_vmx *vmx)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = vmx->nested.cached_vmcs12;
|
|
|
|
struct hv_enlightened_vmcs *evmcs = vmx->nested.hv_evmcs;
|
|
|
|
|
|
|
|
/* HV_VMX_ENLIGHTENED_CLEAN_FIELD_NONE */
|
|
|
|
vmcs12->tpr_threshold = evmcs->tpr_threshold;
|
|
|
|
vmcs12->guest_rip = evmcs->guest_rip;
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_BASIC))) {
|
|
|
|
vmcs12->guest_rsp = evmcs->guest_rsp;
|
|
|
|
vmcs12->guest_rflags = evmcs->guest_rflags;
|
|
|
|
vmcs12->guest_interruptibility_info =
|
|
|
|
evmcs->guest_interruptibility_info;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_PROC))) {
|
|
|
|
vmcs12->cpu_based_vm_exec_control =
|
|
|
|
evmcs->cpu_based_vm_exec_control;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
2019-06-13 18:35:02 +07:00
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_EXCPN))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs12->exception_bitmap = evmcs->exception_bitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_ENTRY))) {
|
|
|
|
vmcs12->vm_entry_controls = evmcs->vm_entry_controls;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_EVENT))) {
|
|
|
|
vmcs12->vm_entry_intr_info_field =
|
|
|
|
evmcs->vm_entry_intr_info_field;
|
|
|
|
vmcs12->vm_entry_exception_error_code =
|
|
|
|
evmcs->vm_entry_exception_error_code;
|
|
|
|
vmcs12->vm_entry_instruction_len =
|
|
|
|
evmcs->vm_entry_instruction_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_HOST_GRP1))) {
|
|
|
|
vmcs12->host_ia32_pat = evmcs->host_ia32_pat;
|
|
|
|
vmcs12->host_ia32_efer = evmcs->host_ia32_efer;
|
|
|
|
vmcs12->host_cr0 = evmcs->host_cr0;
|
|
|
|
vmcs12->host_cr3 = evmcs->host_cr3;
|
|
|
|
vmcs12->host_cr4 = evmcs->host_cr4;
|
|
|
|
vmcs12->host_ia32_sysenter_esp = evmcs->host_ia32_sysenter_esp;
|
|
|
|
vmcs12->host_ia32_sysenter_eip = evmcs->host_ia32_sysenter_eip;
|
|
|
|
vmcs12->host_rip = evmcs->host_rip;
|
|
|
|
vmcs12->host_ia32_sysenter_cs = evmcs->host_ia32_sysenter_cs;
|
|
|
|
vmcs12->host_es_selector = evmcs->host_es_selector;
|
|
|
|
vmcs12->host_cs_selector = evmcs->host_cs_selector;
|
|
|
|
vmcs12->host_ss_selector = evmcs->host_ss_selector;
|
|
|
|
vmcs12->host_ds_selector = evmcs->host_ds_selector;
|
|
|
|
vmcs12->host_fs_selector = evmcs->host_fs_selector;
|
|
|
|
vmcs12->host_gs_selector = evmcs->host_gs_selector;
|
|
|
|
vmcs12->host_tr_selector = evmcs->host_tr_selector;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
2019-06-13 18:35:02 +07:00
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_GRP1))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs12->pin_based_vm_exec_control =
|
|
|
|
evmcs->pin_based_vm_exec_control;
|
|
|
|
vmcs12->vm_exit_controls = evmcs->vm_exit_controls;
|
|
|
|
vmcs12->secondary_vm_exec_control =
|
|
|
|
evmcs->secondary_vm_exec_control;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_IO_BITMAP))) {
|
|
|
|
vmcs12->io_bitmap_a = evmcs->io_bitmap_a;
|
|
|
|
vmcs12->io_bitmap_b = evmcs->io_bitmap_b;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_MSR_BITMAP))) {
|
|
|
|
vmcs12->msr_bitmap = evmcs->msr_bitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_GRP2))) {
|
|
|
|
vmcs12->guest_es_base = evmcs->guest_es_base;
|
|
|
|
vmcs12->guest_cs_base = evmcs->guest_cs_base;
|
|
|
|
vmcs12->guest_ss_base = evmcs->guest_ss_base;
|
|
|
|
vmcs12->guest_ds_base = evmcs->guest_ds_base;
|
|
|
|
vmcs12->guest_fs_base = evmcs->guest_fs_base;
|
|
|
|
vmcs12->guest_gs_base = evmcs->guest_gs_base;
|
|
|
|
vmcs12->guest_ldtr_base = evmcs->guest_ldtr_base;
|
|
|
|
vmcs12->guest_tr_base = evmcs->guest_tr_base;
|
|
|
|
vmcs12->guest_gdtr_base = evmcs->guest_gdtr_base;
|
|
|
|
vmcs12->guest_idtr_base = evmcs->guest_idtr_base;
|
|
|
|
vmcs12->guest_es_limit = evmcs->guest_es_limit;
|
|
|
|
vmcs12->guest_cs_limit = evmcs->guest_cs_limit;
|
|
|
|
vmcs12->guest_ss_limit = evmcs->guest_ss_limit;
|
|
|
|
vmcs12->guest_ds_limit = evmcs->guest_ds_limit;
|
|
|
|
vmcs12->guest_fs_limit = evmcs->guest_fs_limit;
|
|
|
|
vmcs12->guest_gs_limit = evmcs->guest_gs_limit;
|
|
|
|
vmcs12->guest_ldtr_limit = evmcs->guest_ldtr_limit;
|
|
|
|
vmcs12->guest_tr_limit = evmcs->guest_tr_limit;
|
|
|
|
vmcs12->guest_gdtr_limit = evmcs->guest_gdtr_limit;
|
|
|
|
vmcs12->guest_idtr_limit = evmcs->guest_idtr_limit;
|
|
|
|
vmcs12->guest_es_ar_bytes = evmcs->guest_es_ar_bytes;
|
|
|
|
vmcs12->guest_cs_ar_bytes = evmcs->guest_cs_ar_bytes;
|
|
|
|
vmcs12->guest_ss_ar_bytes = evmcs->guest_ss_ar_bytes;
|
|
|
|
vmcs12->guest_ds_ar_bytes = evmcs->guest_ds_ar_bytes;
|
|
|
|
vmcs12->guest_fs_ar_bytes = evmcs->guest_fs_ar_bytes;
|
|
|
|
vmcs12->guest_gs_ar_bytes = evmcs->guest_gs_ar_bytes;
|
|
|
|
vmcs12->guest_ldtr_ar_bytes = evmcs->guest_ldtr_ar_bytes;
|
|
|
|
vmcs12->guest_tr_ar_bytes = evmcs->guest_tr_ar_bytes;
|
|
|
|
vmcs12->guest_es_selector = evmcs->guest_es_selector;
|
|
|
|
vmcs12->guest_cs_selector = evmcs->guest_cs_selector;
|
|
|
|
vmcs12->guest_ss_selector = evmcs->guest_ss_selector;
|
|
|
|
vmcs12->guest_ds_selector = evmcs->guest_ds_selector;
|
|
|
|
vmcs12->guest_fs_selector = evmcs->guest_fs_selector;
|
|
|
|
vmcs12->guest_gs_selector = evmcs->guest_gs_selector;
|
|
|
|
vmcs12->guest_ldtr_selector = evmcs->guest_ldtr_selector;
|
|
|
|
vmcs12->guest_tr_selector = evmcs->guest_tr_selector;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_GRP2))) {
|
|
|
|
vmcs12->tsc_offset = evmcs->tsc_offset;
|
|
|
|
vmcs12->virtual_apic_page_addr = evmcs->virtual_apic_page_addr;
|
|
|
|
vmcs12->xss_exit_bitmap = evmcs->xss_exit_bitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CRDR))) {
|
|
|
|
vmcs12->cr0_guest_host_mask = evmcs->cr0_guest_host_mask;
|
|
|
|
vmcs12->cr4_guest_host_mask = evmcs->cr4_guest_host_mask;
|
|
|
|
vmcs12->cr0_read_shadow = evmcs->cr0_read_shadow;
|
|
|
|
vmcs12->cr4_read_shadow = evmcs->cr4_read_shadow;
|
|
|
|
vmcs12->guest_cr0 = evmcs->guest_cr0;
|
|
|
|
vmcs12->guest_cr3 = evmcs->guest_cr3;
|
|
|
|
vmcs12->guest_cr4 = evmcs->guest_cr4;
|
|
|
|
vmcs12->guest_dr7 = evmcs->guest_dr7;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_HOST_POINTER))) {
|
|
|
|
vmcs12->host_fs_base = evmcs->host_fs_base;
|
|
|
|
vmcs12->host_gs_base = evmcs->host_gs_base;
|
|
|
|
vmcs12->host_tr_base = evmcs->host_tr_base;
|
|
|
|
vmcs12->host_gdtr_base = evmcs->host_gdtr_base;
|
|
|
|
vmcs12->host_idtr_base = evmcs->host_idtr_base;
|
|
|
|
vmcs12->host_rsp = evmcs->host_rsp;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_CONTROL_XLAT))) {
|
|
|
|
vmcs12->ept_pointer = evmcs->ept_pointer;
|
|
|
|
vmcs12->virtual_processor_id = evmcs->virtual_processor_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!(evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_GRP1))) {
|
|
|
|
vmcs12->vmcs_link_pointer = evmcs->vmcs_link_pointer;
|
|
|
|
vmcs12->guest_ia32_debugctl = evmcs->guest_ia32_debugctl;
|
|
|
|
vmcs12->guest_ia32_pat = evmcs->guest_ia32_pat;
|
|
|
|
vmcs12->guest_ia32_efer = evmcs->guest_ia32_efer;
|
|
|
|
vmcs12->guest_pdptr0 = evmcs->guest_pdptr0;
|
|
|
|
vmcs12->guest_pdptr1 = evmcs->guest_pdptr1;
|
|
|
|
vmcs12->guest_pdptr2 = evmcs->guest_pdptr2;
|
|
|
|
vmcs12->guest_pdptr3 = evmcs->guest_pdptr3;
|
|
|
|
vmcs12->guest_pending_dbg_exceptions =
|
|
|
|
evmcs->guest_pending_dbg_exceptions;
|
|
|
|
vmcs12->guest_sysenter_esp = evmcs->guest_sysenter_esp;
|
|
|
|
vmcs12->guest_sysenter_eip = evmcs->guest_sysenter_eip;
|
|
|
|
vmcs12->guest_bndcfgs = evmcs->guest_bndcfgs;
|
|
|
|
vmcs12->guest_activity_state = evmcs->guest_activity_state;
|
|
|
|
vmcs12->guest_sysenter_cs = evmcs->guest_sysenter_cs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Not used?
|
|
|
|
* vmcs12->vm_exit_msr_store_addr = evmcs->vm_exit_msr_store_addr;
|
|
|
|
* vmcs12->vm_exit_msr_load_addr = evmcs->vm_exit_msr_load_addr;
|
|
|
|
* vmcs12->vm_entry_msr_load_addr = evmcs->vm_entry_msr_load_addr;
|
|
|
|
* vmcs12->cr3_target_value0 = evmcs->cr3_target_value0;
|
|
|
|
* vmcs12->cr3_target_value1 = evmcs->cr3_target_value1;
|
|
|
|
* vmcs12->cr3_target_value2 = evmcs->cr3_target_value2;
|
|
|
|
* vmcs12->cr3_target_value3 = evmcs->cr3_target_value3;
|
|
|
|
* vmcs12->page_fault_error_code_mask =
|
|
|
|
* evmcs->page_fault_error_code_mask;
|
|
|
|
* vmcs12->page_fault_error_code_match =
|
|
|
|
* evmcs->page_fault_error_code_match;
|
|
|
|
* vmcs12->cr3_target_count = evmcs->cr3_target_count;
|
|
|
|
* vmcs12->vm_exit_msr_store_count = evmcs->vm_exit_msr_store_count;
|
|
|
|
* vmcs12->vm_exit_msr_load_count = evmcs->vm_exit_msr_load_count;
|
|
|
|
* vmcs12->vm_entry_msr_load_count = evmcs->vm_entry_msr_load_count;
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read only fields:
|
|
|
|
* vmcs12->guest_physical_address = evmcs->guest_physical_address;
|
|
|
|
* vmcs12->vm_instruction_error = evmcs->vm_instruction_error;
|
|
|
|
* vmcs12->vm_exit_reason = evmcs->vm_exit_reason;
|
|
|
|
* vmcs12->vm_exit_intr_info = evmcs->vm_exit_intr_info;
|
|
|
|
* vmcs12->vm_exit_intr_error_code = evmcs->vm_exit_intr_error_code;
|
|
|
|
* vmcs12->idt_vectoring_info_field = evmcs->idt_vectoring_info_field;
|
|
|
|
* vmcs12->idt_vectoring_error_code = evmcs->idt_vectoring_error_code;
|
|
|
|
* vmcs12->vm_exit_instruction_len = evmcs->vm_exit_instruction_len;
|
|
|
|
* vmcs12->vmx_instruction_info = evmcs->vmx_instruction_info;
|
|
|
|
* vmcs12->exit_qualification = evmcs->exit_qualification;
|
|
|
|
* vmcs12->guest_linear_address = evmcs->guest_linear_address;
|
|
|
|
*
|
|
|
|
* Not present in struct vmcs12:
|
|
|
|
* vmcs12->exit_io_instruction_ecx = evmcs->exit_io_instruction_ecx;
|
|
|
|
* vmcs12->exit_io_instruction_esi = evmcs->exit_io_instruction_esi;
|
|
|
|
* vmcs12->exit_io_instruction_edi = evmcs->exit_io_instruction_edi;
|
|
|
|
* vmcs12->exit_io_instruction_eip = evmcs->exit_io_instruction_eip;
|
|
|
|
*/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int copy_vmcs12_to_enlightened(struct vcpu_vmx *vmx)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = vmx->nested.cached_vmcs12;
|
|
|
|
struct hv_enlightened_vmcs *evmcs = vmx->nested.hv_evmcs;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Should not be changed by KVM:
|
|
|
|
*
|
|
|
|
* evmcs->host_es_selector = vmcs12->host_es_selector;
|
|
|
|
* evmcs->host_cs_selector = vmcs12->host_cs_selector;
|
|
|
|
* evmcs->host_ss_selector = vmcs12->host_ss_selector;
|
|
|
|
* evmcs->host_ds_selector = vmcs12->host_ds_selector;
|
|
|
|
* evmcs->host_fs_selector = vmcs12->host_fs_selector;
|
|
|
|
* evmcs->host_gs_selector = vmcs12->host_gs_selector;
|
|
|
|
* evmcs->host_tr_selector = vmcs12->host_tr_selector;
|
|
|
|
* evmcs->host_ia32_pat = vmcs12->host_ia32_pat;
|
|
|
|
* evmcs->host_ia32_efer = vmcs12->host_ia32_efer;
|
|
|
|
* evmcs->host_cr0 = vmcs12->host_cr0;
|
|
|
|
* evmcs->host_cr3 = vmcs12->host_cr3;
|
|
|
|
* evmcs->host_cr4 = vmcs12->host_cr4;
|
|
|
|
* evmcs->host_ia32_sysenter_esp = vmcs12->host_ia32_sysenter_esp;
|
|
|
|
* evmcs->host_ia32_sysenter_eip = vmcs12->host_ia32_sysenter_eip;
|
|
|
|
* evmcs->host_rip = vmcs12->host_rip;
|
|
|
|
* evmcs->host_ia32_sysenter_cs = vmcs12->host_ia32_sysenter_cs;
|
|
|
|
* evmcs->host_fs_base = vmcs12->host_fs_base;
|
|
|
|
* evmcs->host_gs_base = vmcs12->host_gs_base;
|
|
|
|
* evmcs->host_tr_base = vmcs12->host_tr_base;
|
|
|
|
* evmcs->host_gdtr_base = vmcs12->host_gdtr_base;
|
|
|
|
* evmcs->host_idtr_base = vmcs12->host_idtr_base;
|
|
|
|
* evmcs->host_rsp = vmcs12->host_rsp;
|
2019-05-07 22:36:27 +07:00
|
|
|
* sync_vmcs02_to_vmcs12() doesn't read these:
|
2018-12-04 04:53:18 +07:00
|
|
|
* evmcs->io_bitmap_a = vmcs12->io_bitmap_a;
|
|
|
|
* evmcs->io_bitmap_b = vmcs12->io_bitmap_b;
|
|
|
|
* evmcs->msr_bitmap = vmcs12->msr_bitmap;
|
|
|
|
* evmcs->ept_pointer = vmcs12->ept_pointer;
|
|
|
|
* evmcs->xss_exit_bitmap = vmcs12->xss_exit_bitmap;
|
|
|
|
* evmcs->vm_exit_msr_store_addr = vmcs12->vm_exit_msr_store_addr;
|
|
|
|
* evmcs->vm_exit_msr_load_addr = vmcs12->vm_exit_msr_load_addr;
|
|
|
|
* evmcs->vm_entry_msr_load_addr = vmcs12->vm_entry_msr_load_addr;
|
|
|
|
* evmcs->cr3_target_value0 = vmcs12->cr3_target_value0;
|
|
|
|
* evmcs->cr3_target_value1 = vmcs12->cr3_target_value1;
|
|
|
|
* evmcs->cr3_target_value2 = vmcs12->cr3_target_value2;
|
|
|
|
* evmcs->cr3_target_value3 = vmcs12->cr3_target_value3;
|
|
|
|
* evmcs->tpr_threshold = vmcs12->tpr_threshold;
|
|
|
|
* evmcs->virtual_processor_id = vmcs12->virtual_processor_id;
|
|
|
|
* evmcs->exception_bitmap = vmcs12->exception_bitmap;
|
|
|
|
* evmcs->vmcs_link_pointer = vmcs12->vmcs_link_pointer;
|
|
|
|
* evmcs->pin_based_vm_exec_control = vmcs12->pin_based_vm_exec_control;
|
|
|
|
* evmcs->vm_exit_controls = vmcs12->vm_exit_controls;
|
|
|
|
* evmcs->secondary_vm_exec_control = vmcs12->secondary_vm_exec_control;
|
|
|
|
* evmcs->page_fault_error_code_mask =
|
|
|
|
* vmcs12->page_fault_error_code_mask;
|
|
|
|
* evmcs->page_fault_error_code_match =
|
|
|
|
* vmcs12->page_fault_error_code_match;
|
|
|
|
* evmcs->cr3_target_count = vmcs12->cr3_target_count;
|
|
|
|
* evmcs->virtual_apic_page_addr = vmcs12->virtual_apic_page_addr;
|
|
|
|
* evmcs->tsc_offset = vmcs12->tsc_offset;
|
|
|
|
* evmcs->guest_ia32_debugctl = vmcs12->guest_ia32_debugctl;
|
|
|
|
* evmcs->cr0_guest_host_mask = vmcs12->cr0_guest_host_mask;
|
|
|
|
* evmcs->cr4_guest_host_mask = vmcs12->cr4_guest_host_mask;
|
|
|
|
* evmcs->cr0_read_shadow = vmcs12->cr0_read_shadow;
|
|
|
|
* evmcs->cr4_read_shadow = vmcs12->cr4_read_shadow;
|
|
|
|
* evmcs->vm_exit_msr_store_count = vmcs12->vm_exit_msr_store_count;
|
|
|
|
* evmcs->vm_exit_msr_load_count = vmcs12->vm_exit_msr_load_count;
|
|
|
|
* evmcs->vm_entry_msr_load_count = vmcs12->vm_entry_msr_load_count;
|
|
|
|
*
|
|
|
|
* Not present in struct vmcs12:
|
|
|
|
* evmcs->exit_io_instruction_ecx = vmcs12->exit_io_instruction_ecx;
|
|
|
|
* evmcs->exit_io_instruction_esi = vmcs12->exit_io_instruction_esi;
|
|
|
|
* evmcs->exit_io_instruction_edi = vmcs12->exit_io_instruction_edi;
|
|
|
|
* evmcs->exit_io_instruction_eip = vmcs12->exit_io_instruction_eip;
|
|
|
|
*/
|
|
|
|
|
|
|
|
evmcs->guest_es_selector = vmcs12->guest_es_selector;
|
|
|
|
evmcs->guest_cs_selector = vmcs12->guest_cs_selector;
|
|
|
|
evmcs->guest_ss_selector = vmcs12->guest_ss_selector;
|
|
|
|
evmcs->guest_ds_selector = vmcs12->guest_ds_selector;
|
|
|
|
evmcs->guest_fs_selector = vmcs12->guest_fs_selector;
|
|
|
|
evmcs->guest_gs_selector = vmcs12->guest_gs_selector;
|
|
|
|
evmcs->guest_ldtr_selector = vmcs12->guest_ldtr_selector;
|
|
|
|
evmcs->guest_tr_selector = vmcs12->guest_tr_selector;
|
|
|
|
|
|
|
|
evmcs->guest_es_limit = vmcs12->guest_es_limit;
|
|
|
|
evmcs->guest_cs_limit = vmcs12->guest_cs_limit;
|
|
|
|
evmcs->guest_ss_limit = vmcs12->guest_ss_limit;
|
|
|
|
evmcs->guest_ds_limit = vmcs12->guest_ds_limit;
|
|
|
|
evmcs->guest_fs_limit = vmcs12->guest_fs_limit;
|
|
|
|
evmcs->guest_gs_limit = vmcs12->guest_gs_limit;
|
|
|
|
evmcs->guest_ldtr_limit = vmcs12->guest_ldtr_limit;
|
|
|
|
evmcs->guest_tr_limit = vmcs12->guest_tr_limit;
|
|
|
|
evmcs->guest_gdtr_limit = vmcs12->guest_gdtr_limit;
|
|
|
|
evmcs->guest_idtr_limit = vmcs12->guest_idtr_limit;
|
|
|
|
|
|
|
|
evmcs->guest_es_ar_bytes = vmcs12->guest_es_ar_bytes;
|
|
|
|
evmcs->guest_cs_ar_bytes = vmcs12->guest_cs_ar_bytes;
|
|
|
|
evmcs->guest_ss_ar_bytes = vmcs12->guest_ss_ar_bytes;
|
|
|
|
evmcs->guest_ds_ar_bytes = vmcs12->guest_ds_ar_bytes;
|
|
|
|
evmcs->guest_fs_ar_bytes = vmcs12->guest_fs_ar_bytes;
|
|
|
|
evmcs->guest_gs_ar_bytes = vmcs12->guest_gs_ar_bytes;
|
|
|
|
evmcs->guest_ldtr_ar_bytes = vmcs12->guest_ldtr_ar_bytes;
|
|
|
|
evmcs->guest_tr_ar_bytes = vmcs12->guest_tr_ar_bytes;
|
|
|
|
|
|
|
|
evmcs->guest_es_base = vmcs12->guest_es_base;
|
|
|
|
evmcs->guest_cs_base = vmcs12->guest_cs_base;
|
|
|
|
evmcs->guest_ss_base = vmcs12->guest_ss_base;
|
|
|
|
evmcs->guest_ds_base = vmcs12->guest_ds_base;
|
|
|
|
evmcs->guest_fs_base = vmcs12->guest_fs_base;
|
|
|
|
evmcs->guest_gs_base = vmcs12->guest_gs_base;
|
|
|
|
evmcs->guest_ldtr_base = vmcs12->guest_ldtr_base;
|
|
|
|
evmcs->guest_tr_base = vmcs12->guest_tr_base;
|
|
|
|
evmcs->guest_gdtr_base = vmcs12->guest_gdtr_base;
|
|
|
|
evmcs->guest_idtr_base = vmcs12->guest_idtr_base;
|
|
|
|
|
|
|
|
evmcs->guest_ia32_pat = vmcs12->guest_ia32_pat;
|
|
|
|
evmcs->guest_ia32_efer = vmcs12->guest_ia32_efer;
|
|
|
|
|
|
|
|
evmcs->guest_pdptr0 = vmcs12->guest_pdptr0;
|
|
|
|
evmcs->guest_pdptr1 = vmcs12->guest_pdptr1;
|
|
|
|
evmcs->guest_pdptr2 = vmcs12->guest_pdptr2;
|
|
|
|
evmcs->guest_pdptr3 = vmcs12->guest_pdptr3;
|
|
|
|
|
|
|
|
evmcs->guest_pending_dbg_exceptions =
|
|
|
|
vmcs12->guest_pending_dbg_exceptions;
|
|
|
|
evmcs->guest_sysenter_esp = vmcs12->guest_sysenter_esp;
|
|
|
|
evmcs->guest_sysenter_eip = vmcs12->guest_sysenter_eip;
|
|
|
|
|
|
|
|
evmcs->guest_activity_state = vmcs12->guest_activity_state;
|
|
|
|
evmcs->guest_sysenter_cs = vmcs12->guest_sysenter_cs;
|
|
|
|
|
|
|
|
evmcs->guest_cr0 = vmcs12->guest_cr0;
|
|
|
|
evmcs->guest_cr3 = vmcs12->guest_cr3;
|
|
|
|
evmcs->guest_cr4 = vmcs12->guest_cr4;
|
|
|
|
evmcs->guest_dr7 = vmcs12->guest_dr7;
|
|
|
|
|
|
|
|
evmcs->guest_physical_address = vmcs12->guest_physical_address;
|
|
|
|
|
|
|
|
evmcs->vm_instruction_error = vmcs12->vm_instruction_error;
|
|
|
|
evmcs->vm_exit_reason = vmcs12->vm_exit_reason;
|
|
|
|
evmcs->vm_exit_intr_info = vmcs12->vm_exit_intr_info;
|
|
|
|
evmcs->vm_exit_intr_error_code = vmcs12->vm_exit_intr_error_code;
|
|
|
|
evmcs->idt_vectoring_info_field = vmcs12->idt_vectoring_info_field;
|
|
|
|
evmcs->idt_vectoring_error_code = vmcs12->idt_vectoring_error_code;
|
|
|
|
evmcs->vm_exit_instruction_len = vmcs12->vm_exit_instruction_len;
|
|
|
|
evmcs->vmx_instruction_info = vmcs12->vmx_instruction_info;
|
|
|
|
|
|
|
|
evmcs->exit_qualification = vmcs12->exit_qualification;
|
|
|
|
|
|
|
|
evmcs->guest_linear_address = vmcs12->guest_linear_address;
|
|
|
|
evmcs->guest_rsp = vmcs12->guest_rsp;
|
|
|
|
evmcs->guest_rflags = vmcs12->guest_rflags;
|
|
|
|
|
|
|
|
evmcs->guest_interruptibility_info =
|
|
|
|
vmcs12->guest_interruptibility_info;
|
|
|
|
evmcs->cpu_based_vm_exec_control = vmcs12->cpu_based_vm_exec_control;
|
|
|
|
evmcs->vm_entry_controls = vmcs12->vm_entry_controls;
|
|
|
|
evmcs->vm_entry_intr_info_field = vmcs12->vm_entry_intr_info_field;
|
|
|
|
evmcs->vm_entry_exception_error_code =
|
|
|
|
vmcs12->vm_entry_exception_error_code;
|
|
|
|
evmcs->vm_entry_instruction_len = vmcs12->vm_entry_instruction_len;
|
|
|
|
|
|
|
|
evmcs->guest_rip = vmcs12->guest_rip;
|
|
|
|
|
|
|
|
evmcs->guest_bndcfgs = vmcs12->guest_bndcfgs;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is an equivalent of the nested hypervisor executing the vmptrld
|
|
|
|
* instruction.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_handle_enlightened_vmptrld(struct kvm_vcpu *vcpu,
|
|
|
|
bool from_launch)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
2019-06-28 18:23:32 +07:00
|
|
|
bool evmcs_gpa_changed = false;
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
u64 evmcs_gpa;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (likely(!vmx->nested.enlightened_vmcs_enabled))
|
|
|
|
return 1;
|
|
|
|
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
if (!nested_enlightened_vmentry(vcpu, &evmcs_gpa))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
if (unlikely(evmcs_gpa != vmx->nested.hv_evmcs_vmptr)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
if (!vmx->nested.hv_evmcs)
|
|
|
|
vmx->nested.current_vmptr = -1ull;
|
|
|
|
|
|
|
|
nested_release_evmcs(vcpu);
|
|
|
|
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
if (kvm_vcpu_map(vcpu, gpa_to_gfn(evmcs_gpa),
|
2019-02-01 03:24:42 +07:00
|
|
|
&vmx->nested.hv_evmcs_map))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
|
2019-02-01 03:24:42 +07:00
|
|
|
vmx->nested.hv_evmcs = vmx->nested.hv_evmcs_map.hva;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Currently, KVM only supports eVMCS version 1
|
|
|
|
* (== KVM_EVMCS_VERSION) and thus we expect guest to set this
|
|
|
|
* value to first u32 field of eVMCS which should specify eVMCS
|
|
|
|
* VersionNumber.
|
|
|
|
*
|
|
|
|
* Guest should be aware of supported eVMCS versions by host by
|
|
|
|
* examining CPUID.0x4000000A.EAX[0:15]. Host userspace VMM is
|
|
|
|
* expected to set this CPUID leaf according to the value
|
|
|
|
* returned in vmcs_version from nested_enable_evmcs().
|
|
|
|
*
|
|
|
|
* However, it turns out that Microsoft Hyper-V fails to comply
|
|
|
|
* to their own invented interface: When Hyper-V use eVMCS, it
|
|
|
|
* just sets first u32 field of eVMCS to revision_id specified
|
|
|
|
* in MSR_IA32_VMX_BASIC. Instead of used eVMCS version number
|
|
|
|
* which is one of the supported versions specified in
|
|
|
|
* CPUID.0x4000000A.EAX[0:15].
|
|
|
|
*
|
|
|
|
* To overcome Hyper-V bug, we accept here either a supported
|
|
|
|
* eVMCS version or VMCS12 revision_id as valid values for first
|
|
|
|
* u32 field of eVMCS.
|
|
|
|
*/
|
|
|
|
if ((vmx->nested.hv_evmcs->revision_id != KVM_EVMCS_VERSION) &&
|
|
|
|
(vmx->nested.hv_evmcs->revision_id != VMCS12_REVISION)) {
|
|
|
|
nested_release_evmcs(vcpu);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmx->nested.dirty_vmcs12 = true;
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
vmx->nested.hv_evmcs_vmptr = evmcs_gpa;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-06-28 18:23:32 +07:00
|
|
|
evmcs_gpa_changed = true;
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Unlike normal vmcs12, enlightened vmcs12 is not fully
|
|
|
|
* reloaded from guest's memory (read only fields, fields not
|
|
|
|
* present in struct hv_enlightened_vmcs, ...). Make sure there
|
|
|
|
* are no leftovers.
|
|
|
|
*/
|
|
|
|
if (from_launch) {
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
memset(vmcs12, 0, sizeof(*vmcs12));
|
|
|
|
vmcs12->hdr.revision_id = VMCS12_REVISION;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2019-06-28 18:23:32 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Clean fields data can't de used on VMLAUNCH and when we switch
|
|
|
|
* between different L2 guests as KVM keeps a single VMCS12 per L1.
|
|
|
|
*/
|
|
|
|
if (from_launch || evmcs_gpa_changed)
|
|
|
|
vmx->nested.hv_evmcs->hv_clean_fields &=
|
|
|
|
~HV_VMX_ENLIGHTENED_CLEAN_FIELD_ALL;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-05-07 22:36:27 +07:00
|
|
|
void nested_sync_vmcs12_to_shadow(struct kvm_vcpu *vcpu)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* hv_evmcs may end up being not mapped after migration (when
|
|
|
|
* L2 was running), map it here to make sure vmcs12 changes are
|
|
|
|
* properly reflected.
|
|
|
|
*/
|
|
|
|
if (vmx->nested.enlightened_vmcs_enabled && !vmx->nested.hv_evmcs)
|
|
|
|
nested_vmx_handle_enlightened_vmptrld(vcpu, false);
|
|
|
|
|
|
|
|
if (vmx->nested.hv_evmcs) {
|
|
|
|
copy_vmcs12_to_enlightened(vmx);
|
|
|
|
/* All fields are clean */
|
|
|
|
vmx->nested.hv_evmcs->hv_clean_fields |=
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_ALL;
|
|
|
|
} else {
|
|
|
|
copy_vmcs12_to_shadow(vmx);
|
|
|
|
}
|
|
|
|
|
2019-05-07 22:36:27 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = false;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static enum hrtimer_restart vmx_preemption_timer_fn(struct hrtimer *timer)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx =
|
|
|
|
container_of(timer, struct vcpu_vmx, nested.preemption_timer);
|
|
|
|
|
|
|
|
vmx->nested.preemption_timer_expired = true;
|
|
|
|
kvm_make_request(KVM_REQ_EVENT, &vmx->vcpu);
|
|
|
|
kvm_vcpu_kick(&vmx->vcpu);
|
|
|
|
|
|
|
|
return HRTIMER_NORESTART;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vmx_start_preemption_timer(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
u64 preemption_timeout = get_vmcs12(vcpu)->vmx_preemption_timer_value;
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A timer value of zero is architecturally guaranteed to cause
|
|
|
|
* a VMExit prior to executing any instructions in the guest.
|
|
|
|
*/
|
|
|
|
if (preemption_timeout == 0) {
|
|
|
|
vmx_preemption_timer_fn(&vmx->nested.preemption_timer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vcpu->arch.virtual_tsc_khz == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
preemption_timeout <<= VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE;
|
|
|
|
preemption_timeout *= 1000000;
|
|
|
|
do_div(preemption_timeout, vcpu->arch.virtual_tsc_khz);
|
|
|
|
hrtimer_start(&vmx->nested.preemption_timer,
|
|
|
|
ns_to_ktime(preemption_timeout), HRTIMER_MODE_REL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static u64 nested_vmx_calc_efer(struct vcpu_vmx *vmx, struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (vmx->nested.nested_run_pending &&
|
|
|
|
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_EFER))
|
|
|
|
return vmcs12->guest_ia32_efer;
|
|
|
|
else if (vmcs12->vm_entry_controls & VM_ENTRY_IA32E_MODE)
|
|
|
|
return vmx->vcpu.arch.efer | (EFER_LMA | EFER_LME);
|
|
|
|
else
|
|
|
|
return vmx->vcpu.arch.efer & ~(EFER_LMA | EFER_LME);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void prepare_vmcs02_constant_state(struct vcpu_vmx *vmx)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If vmcs02 hasn't been initialized, set the constant vmcs02 state
|
|
|
|
* according to L0's settings (vmcs12 is irrelevant here). Host
|
|
|
|
* fields that come from L0 and are not constant, e.g. HOST_CR3,
|
|
|
|
* will be set as needed prior to VMLAUNCH/VMRESUME.
|
|
|
|
*/
|
|
|
|
if (vmx->nested.vmcs02_initialized)
|
|
|
|
return;
|
|
|
|
vmx->nested.vmcs02_initialized = true;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We don't care what the EPTP value is we just need to guarantee
|
|
|
|
* it's valid so we don't get a false positive when doing early
|
|
|
|
* consistency checks.
|
|
|
|
*/
|
|
|
|
if (enable_ept && nested_early_check)
|
|
|
|
vmcs_write64(EPT_POINTER, construct_eptp(&vmx->vcpu, 0));
|
|
|
|
|
|
|
|
/* All VMFUNCs are currently emulated through L0 vmexits. */
|
|
|
|
if (cpu_has_vmx_vmfunc())
|
|
|
|
vmcs_write64(VM_FUNCTION_CONTROL, 0);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_posted_intr())
|
|
|
|
vmcs_write16(POSTED_INTR_NV, POSTED_INTR_NESTED_VECTOR);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_msr_bitmap())
|
|
|
|
vmcs_write64(MSR_BITMAP, __pa(vmx->nested.vmcs02.msr_bitmap));
|
|
|
|
|
2019-05-07 23:06:30 +07:00
|
|
|
/*
|
|
|
|
* The PML address never changes, so it is constant in vmcs02.
|
|
|
|
* Conceptually we want to copy the PML index from vmcs01 here,
|
|
|
|
* and then back to vmcs01 on nested vmexit. But since we flush
|
|
|
|
* the log and reset GUEST_PML_INDEX on each vmexit, the PML
|
|
|
|
* index is also effectively constant in vmcs02.
|
|
|
|
*/
|
|
|
|
if (enable_pml) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write64(PML_ADDRESS, page_to_phys(vmx->pml_pg));
|
2019-05-07 23:06:30 +07:00
|
|
|
vmcs_write16(GUEST_PML_INDEX, PML_ENTITY_NUM - 1);
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-05-07 23:06:29 +07:00
|
|
|
if (cpu_has_vmx_encls_vmexit())
|
|
|
|
vmcs_write64(ENCLS_EXITING_BITMAP, -1ull);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the MSR load/store lists to match L0's settings. Only the
|
|
|
|
* addresses are constant (for vmcs02), the counts can change based
|
|
|
|
* on L2's behavior, e.g. switching to/from long mode.
|
|
|
|
*/
|
2019-11-08 12:14:39 +07:00
|
|
|
vmcs_write64(VM_EXIT_MSR_STORE_ADDR, __pa(vmx->msr_autostore.guest.val));
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write64(VM_EXIT_MSR_LOAD_ADDR, __pa(vmx->msr_autoload.host.val));
|
|
|
|
vmcs_write64(VM_ENTRY_MSR_LOAD_ADDR, __pa(vmx->msr_autoload.guest.val));
|
|
|
|
|
|
|
|
vmx_set_constant_host_state(vmx);
|
|
|
|
}
|
|
|
|
|
2019-06-06 22:24:00 +07:00
|
|
|
static void prepare_vmcs02_early_rare(struct vcpu_vmx *vmx,
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
prepare_vmcs02_constant_state(vmx);
|
|
|
|
|
|
|
|
vmcs_write64(VMCS_LINK_POINTER, -1ull);
|
|
|
|
|
|
|
|
if (enable_vpid) {
|
|
|
|
if (nested_cpu_has_vpid(vmcs12) && vmx->nested.vpid02)
|
|
|
|
vmcs_write16(VIRTUAL_PROCESSOR_ID, vmx->nested.vpid02);
|
|
|
|
else
|
|
|
|
vmcs_write16(VIRTUAL_PROCESSOR_ID, vmx->vpid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void prepare_vmcs02_early(struct vcpu_vmx *vmx, struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
u32 exec_control, vmcs12_exec_ctrl;
|
|
|
|
u64 guest_efer = nested_vmx_calc_efer(vmx, vmcs12);
|
|
|
|
|
|
|
|
if (vmx->nested.dirty_vmcs12 || vmx->nested.hv_evmcs)
|
2019-06-06 22:24:00 +07:00
|
|
|
prepare_vmcs02_early_rare(vmx, vmcs12);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* PIN CONTROLS
|
|
|
|
*/
|
2019-05-08 02:17:53 +07:00
|
|
|
exec_control = vmx_pin_based_exec_ctrl(vmx);
|
KVM: VMX: Leave preemption timer running when it's disabled
VMWRITEs to the major VMCS controls, pin controls included, are
deceptively expensive. CPUs with VMCS caching (Westmere and later) also
optimize away consistency checks on VM-Entry, i.e. skip consistency
checks if the relevant fields have not changed since the last successful
VM-Entry (of the cached VMCS). Because uops are a precious commodity,
uCode's dirty VMCS field tracking isn't as precise as software would
prefer. Notably, writing any of the major VMCS fields effectively marks
the entire VMCS dirty, i.e. causes the next VM-Entry to perform all
consistency checks, which consumes several hundred cycles.
As it pertains to KVM, toggling PIN_BASED_VMX_PREEMPTION_TIMER more than
doubles the latency of the next VM-Entry (and again when/if the flag is
toggled back). In a non-nested scenario, running a "standard" guest
with the preemption timer enabled, toggling the timer flag is uncommon
but not rare, e.g. roughly 1 in 10 entries. Disabling the preemption
timer can change these numbers due to its use for "immediate exits",
even when explicitly disabled by userspace.
Nested virtualization in particular is painful, as the timer flag is set
for the majority of VM-Enters, but prepare_vmcs02() initializes vmcs02's
pin controls to *clear* the flag since its the timer's final state isn't
known until vmx_vcpu_run(). I.e. the majority of nested VM-Enters end
up unnecessarily writing pin controls *twice*.
Rather than toggle the timer flag in pin controls, set the timer value
itself to the largest allowed value to put it into a "soft disabled"
state, and ignore any spurious preemption timer exits.
Sadly, the timer is a 32-bit value and so theoretically it can fire
before the head death of the universe, i.e. spurious exits are possible.
But because KVM does *not* save the timer value on VM-Exit and because
the timer runs at a slower rate than the TSC, the maximuma timer value
is still sufficiently large for KVM's purposes. E.g. on a modern CPU
with a timer that runs at 1/32 the frequency of a 2.4ghz constant-rate
TSC, the timer will fire after ~55 seconds of *uninterrupted* guest
execution. In other words, spurious VM-Exits are effectively only
possible if the host is completely tickless on the logical CPU, the
guest is not using the preemption timer, and the guest is not generating
VM-Exits for any other reason.
To be safe from bad/weird hardware, disable the preemption timer if its
maximum delay is less than ten seconds. Ten seconds is mostly arbitrary
and was selected in no small part because it's a nice round number.
For simplicity and paranoia, fall back to __kvm_request_immediate_exit()
if the preemption timer is disabled by KVM or userspace. Previously
KVM continued to use the preemption timer to force immediate exits even
when the timer was disabled by userspace. Now that KVM leaves the timer
running instead of truly disabling it, allow userspace to kill it
entirely in the unlikely event the timer (or KVM) malfunctions.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-08 02:18:05 +07:00
|
|
|
exec_control |= (vmcs12->pin_based_vm_exec_control &
|
|
|
|
~PIN_BASED_VMX_PREEMPTION_TIMER);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Posted interrupts setting is only taken from vmcs12. */
|
|
|
|
if (nested_cpu_has_posted_intr(vmcs12)) {
|
|
|
|
vmx->nested.posted_intr_nv = vmcs12->posted_intr_nv;
|
|
|
|
vmx->nested.pi_pending = false;
|
|
|
|
} else {
|
|
|
|
exec_control &= ~PIN_BASED_POSTED_INTR;
|
|
|
|
}
|
2019-05-08 02:18:00 +07:00
|
|
|
pin_controls_set(vmx, exec_control);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* EXEC CONTROLS
|
|
|
|
*/
|
|
|
|
exec_control = vmx_exec_control(vmx); /* L0's desires */
|
|
|
|
exec_control &= ~CPU_BASED_VIRTUAL_INTR_PENDING;
|
|
|
|
exec_control &= ~CPU_BASED_VIRTUAL_NMI_PENDING;
|
|
|
|
exec_control &= ~CPU_BASED_TPR_SHADOW;
|
|
|
|
exec_control |= vmcs12->cpu_based_vm_exec_control;
|
|
|
|
|
2019-11-11 19:30:55 +07:00
|
|
|
vmx->nested.l1_tpr_threshold = -1;
|
2019-05-07 23:06:33 +07:00
|
|
|
if (exec_control & CPU_BASED_TPR_SHADOW)
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write32(TPR_THRESHOLD, vmcs12->tpr_threshold);
|
|
|
|
#ifdef CONFIG_X86_64
|
2019-05-07 23:06:33 +07:00
|
|
|
else
|
2018-12-04 04:53:18 +07:00
|
|
|
exec_control |= CPU_BASED_CR8_LOAD_EXITING |
|
|
|
|
CPU_BASED_CR8_STORE_EXITING;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A vmexit (to either L1 hypervisor or L0 userspace) is always needed
|
|
|
|
* for I/O port accesses.
|
|
|
|
*/
|
|
|
|
exec_control |= CPU_BASED_UNCOND_IO_EXITING;
|
2019-05-08 02:18:01 +07:00
|
|
|
exec_control &= ~CPU_BASED_USE_IO_BITMAPS;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This bit will be computed in nested_get_vmcs12_pages, because
|
|
|
|
* we do not have access to L1's MSR bitmap yet. For now, keep
|
|
|
|
* the same bit as before, hoping to avoid multiple VMWRITEs that
|
|
|
|
* only set/clear this bit.
|
|
|
|
*/
|
|
|
|
exec_control &= ~CPU_BASED_USE_MSR_BITMAPS;
|
|
|
|
exec_control |= exec_controls_get(vmx) & CPU_BASED_USE_MSR_BITMAPS;
|
|
|
|
|
2019-05-08 02:18:00 +07:00
|
|
|
exec_controls_set(vmx, exec_control);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* SECONDARY EXEC CONTROLS
|
|
|
|
*/
|
|
|
|
if (cpu_has_secondary_exec_ctrls()) {
|
|
|
|
exec_control = vmx->secondary_exec_control;
|
|
|
|
|
|
|
|
/* Take the following fields only from vmcs12 */
|
|
|
|
exec_control &= ~(SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES |
|
|
|
|
SECONDARY_EXEC_ENABLE_INVPCID |
|
|
|
|
SECONDARY_EXEC_RDTSCP |
|
|
|
|
SECONDARY_EXEC_XSAVES |
|
KVM: x86: Add support for user wait instructions
UMONITOR, UMWAIT and TPAUSE are a set of user wait instructions.
This patch adds support for user wait instructions in KVM. Availability
of the user wait instructions is indicated by the presence of the CPUID
feature flag WAITPKG CPUID.0x07.0x0:ECX[5]. User wait instructions may
be executed at any privilege level, and use 32bit IA32_UMWAIT_CONTROL MSR
to set the maximum time.
The behavior of user wait instructions in VMX non-root operation is
determined first by the setting of the "enable user wait and pause"
secondary processor-based VM-execution control bit 26.
If the VM-execution control is 0, UMONITOR/UMWAIT/TPAUSE cause
an invalid-opcode exception (#UD).
If the VM-execution control is 1, treatment is based on the
setting of the “RDTSC exiting†VM-execution control. Because KVM never
enables RDTSC exiting, if the instruction causes a delay, the amount of
time delayed is called here the physical delay. The physical delay is
first computed by determining the virtual delay. If
IA32_UMWAIT_CONTROL[31:2] is zero, the virtual delay is the value in
EDX:EAX minus the value that RDTSC would return; if
IA32_UMWAIT_CONTROL[31:2] is not zero, the virtual delay is the minimum
of that difference and AND(IA32_UMWAIT_CONTROL,FFFFFFFCH).
Because umwait and tpause can put a (psysical) CPU into a power saving
state, by default we dont't expose it to kvm and enable it only when
guest CPUID has it.
Detailed information about user wait instructions can be found in the
latest Intel 64 and IA-32 Architectures Software Developer's Manual.
Co-developed-by: Jingqi Liu <jingqi.liu@intel.com>
Signed-off-by: Jingqi Liu <jingqi.liu@intel.com>
Signed-off-by: Tao Xu <tao3.xu@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-16 13:55:49 +07:00
|
|
|
SECONDARY_EXEC_ENABLE_USR_WAIT_PAUSE |
|
2018-12-04 04:53:18 +07:00
|
|
|
SECONDARY_EXEC_VIRTUAL_INTR_DELIVERY |
|
|
|
|
SECONDARY_EXEC_APIC_REGISTER_VIRT |
|
|
|
|
SECONDARY_EXEC_ENABLE_VMFUNC);
|
|
|
|
if (nested_cpu_has(vmcs12,
|
|
|
|
CPU_BASED_ACTIVATE_SECONDARY_CONTROLS)) {
|
|
|
|
vmcs12_exec_ctrl = vmcs12->secondary_vm_exec_control &
|
|
|
|
~SECONDARY_EXEC_ENABLE_PML;
|
|
|
|
exec_control |= vmcs12_exec_ctrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VMCS shadowing for L2 is emulated for now */
|
|
|
|
exec_control &= ~SECONDARY_EXEC_SHADOW_VMCS;
|
|
|
|
|
|
|
|
/*
|
2019-05-08 02:18:02 +07:00
|
|
|
* Preset *DT exiting when emulating UMIP, so that vmx_set_cr4()
|
|
|
|
* will not have to rewrite the controls just for this bit.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
2019-05-08 02:18:02 +07:00
|
|
|
if (!boot_cpu_has(X86_FEATURE_UMIP) && vmx_umip_emulated() &&
|
|
|
|
(vmcs12->guest_cr4 & X86_CR4_UMIP))
|
|
|
|
exec_control |= SECONDARY_EXEC_DESC;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (exec_control & SECONDARY_EXEC_VIRTUAL_INTR_DELIVERY)
|
|
|
|
vmcs_write16(GUEST_INTR_STATUS,
|
|
|
|
vmcs12->guest_intr_status);
|
|
|
|
|
2019-05-08 02:18:00 +07:00
|
|
|
secondary_exec_controls_set(vmx, exec_control);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ENTRY CONTROLS
|
|
|
|
*
|
|
|
|
* vmcs12's VM_{ENTRY,EXIT}_LOAD_IA32_EFER and VM_ENTRY_IA32E_MODE
|
|
|
|
* are emulated by vmx_set_efer() in prepare_vmcs02(), but speculate
|
|
|
|
* on the related bits (if supported by the CPU) in the hope that
|
|
|
|
* we can avoid VMWrites during vmx_set_efer().
|
|
|
|
*/
|
|
|
|
exec_control = (vmcs12->vm_entry_controls | vmx_vmentry_ctrl()) &
|
|
|
|
~VM_ENTRY_IA32E_MODE & ~VM_ENTRY_LOAD_IA32_EFER;
|
|
|
|
if (cpu_has_load_ia32_efer()) {
|
|
|
|
if (guest_efer & EFER_LMA)
|
|
|
|
exec_control |= VM_ENTRY_IA32E_MODE;
|
|
|
|
if (guest_efer != host_efer)
|
|
|
|
exec_control |= VM_ENTRY_LOAD_IA32_EFER;
|
|
|
|
}
|
2019-05-08 02:18:00 +07:00
|
|
|
vm_entry_controls_set(vmx, exec_control);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* EXIT CONTROLS
|
|
|
|
*
|
|
|
|
* L2->L1 exit controls are emulated - the hardware exit is to L0 so
|
|
|
|
* we should use its exit controls. Note that VM_EXIT_LOAD_IA32_EFER
|
|
|
|
* bits may be modified by vmx_set_efer() in prepare_vmcs02().
|
|
|
|
*/
|
|
|
|
exec_control = vmx_vmexit_ctrl();
|
|
|
|
if (cpu_has_load_ia32_efer() && guest_efer != host_efer)
|
|
|
|
exec_control |= VM_EXIT_LOAD_IA32_EFER;
|
2019-05-08 02:18:00 +07:00
|
|
|
vm_exit_controls_set(vmx, exec_control);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Interrupt/Exception Fields
|
|
|
|
*/
|
|
|
|
if (vmx->nested.nested_run_pending) {
|
|
|
|
vmcs_write32(VM_ENTRY_INTR_INFO_FIELD,
|
|
|
|
vmcs12->vm_entry_intr_info_field);
|
|
|
|
vmcs_write32(VM_ENTRY_EXCEPTION_ERROR_CODE,
|
|
|
|
vmcs12->vm_entry_exception_error_code);
|
|
|
|
vmcs_write32(VM_ENTRY_INSTRUCTION_LEN,
|
|
|
|
vmcs12->vm_entry_instruction_len);
|
|
|
|
vmcs_write32(GUEST_INTERRUPTIBILITY_INFO,
|
|
|
|
vmcs12->guest_interruptibility_info);
|
|
|
|
vmx->loaded_vmcs->nmi_known_unmasked =
|
|
|
|
!(vmcs12->guest_interruptibility_info & GUEST_INTR_STATE_NMI);
|
|
|
|
} else {
|
|
|
|
vmcs_write32(VM_ENTRY_INTR_INFO_FIELD, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-06 22:24:00 +07:00
|
|
|
static void prepare_vmcs02_rare(struct vcpu_vmx *vmx, struct vmcs12 *vmcs12)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct hv_enlightened_vmcs *hv_evmcs = vmx->nested.hv_evmcs;
|
|
|
|
|
|
|
|
if (!hv_evmcs || !(hv_evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_GRP2)) {
|
|
|
|
vmcs_write16(GUEST_ES_SELECTOR, vmcs12->guest_es_selector);
|
|
|
|
vmcs_write16(GUEST_CS_SELECTOR, vmcs12->guest_cs_selector);
|
|
|
|
vmcs_write16(GUEST_SS_SELECTOR, vmcs12->guest_ss_selector);
|
|
|
|
vmcs_write16(GUEST_DS_SELECTOR, vmcs12->guest_ds_selector);
|
|
|
|
vmcs_write16(GUEST_FS_SELECTOR, vmcs12->guest_fs_selector);
|
|
|
|
vmcs_write16(GUEST_GS_SELECTOR, vmcs12->guest_gs_selector);
|
|
|
|
vmcs_write16(GUEST_LDTR_SELECTOR, vmcs12->guest_ldtr_selector);
|
|
|
|
vmcs_write16(GUEST_TR_SELECTOR, vmcs12->guest_tr_selector);
|
|
|
|
vmcs_write32(GUEST_ES_LIMIT, vmcs12->guest_es_limit);
|
|
|
|
vmcs_write32(GUEST_CS_LIMIT, vmcs12->guest_cs_limit);
|
|
|
|
vmcs_write32(GUEST_SS_LIMIT, vmcs12->guest_ss_limit);
|
|
|
|
vmcs_write32(GUEST_DS_LIMIT, vmcs12->guest_ds_limit);
|
|
|
|
vmcs_write32(GUEST_FS_LIMIT, vmcs12->guest_fs_limit);
|
|
|
|
vmcs_write32(GUEST_GS_LIMIT, vmcs12->guest_gs_limit);
|
|
|
|
vmcs_write32(GUEST_LDTR_LIMIT, vmcs12->guest_ldtr_limit);
|
|
|
|
vmcs_write32(GUEST_TR_LIMIT, vmcs12->guest_tr_limit);
|
|
|
|
vmcs_write32(GUEST_GDTR_LIMIT, vmcs12->guest_gdtr_limit);
|
|
|
|
vmcs_write32(GUEST_IDTR_LIMIT, vmcs12->guest_idtr_limit);
|
2019-05-07 22:36:25 +07:00
|
|
|
vmcs_write32(GUEST_CS_AR_BYTES, vmcs12->guest_cs_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_SS_AR_BYTES, vmcs12->guest_ss_ar_bytes);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write32(GUEST_ES_AR_BYTES, vmcs12->guest_es_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_DS_AR_BYTES, vmcs12->guest_ds_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_FS_AR_BYTES, vmcs12->guest_fs_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_GS_AR_BYTES, vmcs12->guest_gs_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_LDTR_AR_BYTES, vmcs12->guest_ldtr_ar_bytes);
|
|
|
|
vmcs_write32(GUEST_TR_AR_BYTES, vmcs12->guest_tr_ar_bytes);
|
|
|
|
vmcs_writel(GUEST_ES_BASE, vmcs12->guest_es_base);
|
|
|
|
vmcs_writel(GUEST_CS_BASE, vmcs12->guest_cs_base);
|
|
|
|
vmcs_writel(GUEST_SS_BASE, vmcs12->guest_ss_base);
|
|
|
|
vmcs_writel(GUEST_DS_BASE, vmcs12->guest_ds_base);
|
|
|
|
vmcs_writel(GUEST_FS_BASE, vmcs12->guest_fs_base);
|
|
|
|
vmcs_writel(GUEST_GS_BASE, vmcs12->guest_gs_base);
|
|
|
|
vmcs_writel(GUEST_LDTR_BASE, vmcs12->guest_ldtr_base);
|
|
|
|
vmcs_writel(GUEST_TR_BASE, vmcs12->guest_tr_base);
|
|
|
|
vmcs_writel(GUEST_GDTR_BASE, vmcs12->guest_gdtr_base);
|
|
|
|
vmcs_writel(GUEST_IDTR_BASE, vmcs12->guest_idtr_base);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hv_evmcs || !(hv_evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_GRP1)) {
|
|
|
|
vmcs_write32(GUEST_SYSENTER_CS, vmcs12->guest_sysenter_cs);
|
|
|
|
vmcs_writel(GUEST_PENDING_DBG_EXCEPTIONS,
|
|
|
|
vmcs12->guest_pending_dbg_exceptions);
|
|
|
|
vmcs_writel(GUEST_SYSENTER_ESP, vmcs12->guest_sysenter_esp);
|
|
|
|
vmcs_writel(GUEST_SYSENTER_EIP, vmcs12->guest_sysenter_eip);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* L1 may access the L2's PDPTR, so save them to construct
|
|
|
|
* vmcs12
|
|
|
|
*/
|
|
|
|
if (enable_ept) {
|
|
|
|
vmcs_write64(GUEST_PDPTR0, vmcs12->guest_pdptr0);
|
|
|
|
vmcs_write64(GUEST_PDPTR1, vmcs12->guest_pdptr1);
|
|
|
|
vmcs_write64(GUEST_PDPTR2, vmcs12->guest_pdptr2);
|
|
|
|
vmcs_write64(GUEST_PDPTR3, vmcs12->guest_pdptr3);
|
|
|
|
}
|
2019-05-07 23:06:39 +07:00
|
|
|
|
|
|
|
if (kvm_mpx_supported() && vmx->nested.nested_run_pending &&
|
|
|
|
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_BNDCFGS))
|
|
|
|
vmcs_write64(GUEST_BNDCFGS, vmcs12->guest_bndcfgs);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has_xsaves(vmcs12))
|
|
|
|
vmcs_write64(XSS_EXIT_BITMAP, vmcs12->xss_exit_bitmap);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Whether page-faults are trapped is determined by a combination of
|
|
|
|
* 3 settings: PFEC_MASK, PFEC_MATCH and EXCEPTION_BITMAP.PF.
|
|
|
|
* If enable_ept, L0 doesn't care about page faults and we should
|
|
|
|
* set all of these to L1's desires. However, if !enable_ept, L0 does
|
|
|
|
* care about (at least some) page faults, and because it is not easy
|
|
|
|
* (if at all possible?) to merge L0 and L1's desires, we simply ask
|
|
|
|
* to exit on each and every L2 page fault. This is done by setting
|
|
|
|
* MASK=MATCH=0 and (see below) EB.PF=1.
|
|
|
|
* Note that below we don't need special code to set EB.PF beyond the
|
|
|
|
* "or"ing of the EB of vmcs01 and vmcs12, because when enable_ept,
|
|
|
|
* vmcs01's EB.PF is 0 so the "or" will take vmcs12's value, and when
|
|
|
|
* !enable_ept, EB.PF is 1, so the "or" will always be 1.
|
|
|
|
*/
|
|
|
|
vmcs_write32(PAGE_FAULT_ERROR_CODE_MASK,
|
|
|
|
enable_ept ? vmcs12->page_fault_error_code_mask : 0);
|
|
|
|
vmcs_write32(PAGE_FAULT_ERROR_CODE_MATCH,
|
|
|
|
enable_ept ? vmcs12->page_fault_error_code_match : 0);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_apicv()) {
|
|
|
|
vmcs_write64(EOI_EXIT_BITMAP0, vmcs12->eoi_exit_bitmap0);
|
|
|
|
vmcs_write64(EOI_EXIT_BITMAP1, vmcs12->eoi_exit_bitmap1);
|
|
|
|
vmcs_write64(EOI_EXIT_BITMAP2, vmcs12->eoi_exit_bitmap2);
|
|
|
|
vmcs_write64(EOI_EXIT_BITMAP3, vmcs12->eoi_exit_bitmap3);
|
|
|
|
}
|
|
|
|
|
2019-11-08 12:14:39 +07:00
|
|
|
/*
|
|
|
|
* Make sure the msr_autostore list is up to date before we set the
|
|
|
|
* count in the vmcs02.
|
|
|
|
*/
|
|
|
|
prepare_vmx_msr_autostore_list(&vmx->vcpu, MSR_IA32_TSC);
|
|
|
|
|
|
|
|
vmcs_write32(VM_EXIT_MSR_STORE_COUNT, vmx->msr_autostore.guest.nr);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, vmx->msr_autoload.host.nr);
|
|
|
|
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, vmx->msr_autoload.guest.nr);
|
|
|
|
|
|
|
|
set_cr4_guest_host_mask(vmx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* prepare_vmcs02 is called when the L1 guest hypervisor runs its nested
|
|
|
|
* L2 guest. L1 has a vmcs for L2 (vmcs12), and this function "merges" it
|
|
|
|
* with L0's requirements for its guest (a.k.a. vmcs01), so we can run the L2
|
|
|
|
* guest in a way that will both be appropriate to L1's requests, and our
|
|
|
|
* needs. In addition to modifying the active vmcs (which is vmcs02), this
|
|
|
|
* function also has additional necessary side-effects, like setting various
|
|
|
|
* vcpu->arch fields.
|
|
|
|
* Returns 0 on success, 1 on failure. Invalid state exit qualification code
|
|
|
|
* is assigned to entry_failure_code on failure.
|
|
|
|
*/
|
|
|
|
static int prepare_vmcs02(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12,
|
|
|
|
u32 *entry_failure_code)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct hv_enlightened_vmcs *hv_evmcs = vmx->nested.hv_evmcs;
|
2019-05-07 23:06:40 +07:00
|
|
|
bool load_guest_pdptrs_vmcs12 = false;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-05-07 23:06:40 +07:00
|
|
|
if (vmx->nested.dirty_vmcs12 || hv_evmcs) {
|
2019-06-06 22:24:00 +07:00
|
|
|
prepare_vmcs02_rare(vmx, vmcs12);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.dirty_vmcs12 = false;
|
|
|
|
|
2019-05-07 23:06:40 +07:00
|
|
|
load_guest_pdptrs_vmcs12 = !hv_evmcs ||
|
|
|
|
!(hv_evmcs->hv_clean_fields &
|
|
|
|
HV_VMX_ENLIGHTENED_CLEAN_FIELD_GUEST_GRP1);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vmx->nested.nested_run_pending &&
|
|
|
|
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_DEBUG_CONTROLS)) {
|
|
|
|
kvm_set_dr(vcpu, 7, vmcs12->guest_dr7);
|
|
|
|
vmcs_write64(GUEST_IA32_DEBUGCTL, vmcs12->guest_ia32_debugctl);
|
|
|
|
} else {
|
|
|
|
kvm_set_dr(vcpu, 7, vcpu->arch.dr7);
|
|
|
|
vmcs_write64(GUEST_IA32_DEBUGCTL, vmx->nested.vmcs01_debugctl);
|
|
|
|
}
|
2019-05-07 23:06:28 +07:00
|
|
|
if (kvm_mpx_supported() && (!vmx->nested.nested_run_pending ||
|
|
|
|
!(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_BNDCFGS)))
|
|
|
|
vmcs_write64(GUEST_BNDCFGS, vmx->nested.vmcs01_guest_bndcfgs);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx_set_rflags(vcpu, vmcs12->guest_rflags);
|
|
|
|
|
|
|
|
/* EXCEPTION_BITMAP and CR0_GUEST_HOST_MASK should basically be the
|
|
|
|
* bitwise-or of what L1 wants to trap for L2, and what we want to
|
|
|
|
* trap. Note that CR0.TS also needs updating - we do this later.
|
|
|
|
*/
|
|
|
|
update_exception_bitmap(vcpu);
|
|
|
|
vcpu->arch.cr0_guest_owned_bits &= ~vmcs12->cr0_guest_host_mask;
|
|
|
|
vmcs_writel(CR0_GUEST_HOST_MASK, ~vcpu->arch.cr0_guest_owned_bits);
|
|
|
|
|
|
|
|
if (vmx->nested.nested_run_pending &&
|
|
|
|
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_PAT)) {
|
|
|
|
vmcs_write64(GUEST_IA32_PAT, vmcs12->guest_ia32_pat);
|
|
|
|
vcpu->arch.pat = vmcs12->guest_ia32_pat;
|
|
|
|
} else if (vmcs_config.vmentry_ctrl & VM_ENTRY_LOAD_IA32_PAT) {
|
|
|
|
vmcs_write64(GUEST_IA32_PAT, vmx->vcpu.arch.pat);
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset);
|
|
|
|
|
|
|
|
if (kvm_has_tsc_control)
|
|
|
|
decache_tsc_multiplier(vmx);
|
|
|
|
|
|
|
|
if (enable_vpid) {
|
|
|
|
/*
|
|
|
|
* There is no direct mapping between vpid02 and vpid12, the
|
|
|
|
* vpid02 is per-vCPU for L0 and reused while the value of
|
|
|
|
* vpid12 is changed w/ one invvpid during nested vmentry.
|
|
|
|
* The vpid12 is allocated by L1 for L2, so it will not
|
|
|
|
* influence global bitmap(for vpid01 and vpid02 allocation)
|
|
|
|
* even if spawn a lot of nested vCPUs.
|
|
|
|
*/
|
|
|
|
if (nested_cpu_has_vpid(vmcs12) && nested_has_guest_tlb_tag(vcpu)) {
|
|
|
|
if (vmcs12->virtual_processor_id != vmx->nested.last_vpid) {
|
|
|
|
vmx->nested.last_vpid = vmcs12->virtual_processor_id;
|
|
|
|
__vmx_flush_tlb(vcpu, nested_get_vpid02(vcpu), false);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* If L1 use EPT, then L0 needs to execute INVEPT on
|
|
|
|
* EPTP02 instead of EPTP01. Therefore, delay TLB
|
|
|
|
* flush until vmcs02->eptp is fully updated by
|
|
|
|
* KVM_REQ_LOAD_CR3. Note that this assumes
|
|
|
|
* KVM_REQ_TLB_FLUSH is evaluated after
|
|
|
|
* KVM_REQ_LOAD_CR3 in vcpu_enter_guest().
|
|
|
|
*/
|
|
|
|
kvm_make_request(KVM_REQ_TLB_FLUSH, vcpu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has_ept(vmcs12))
|
|
|
|
nested_ept_init_mmu_context(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This sets GUEST_CR0 to vmcs12->guest_cr0, possibly modifying those
|
|
|
|
* bits which we consider mandatory enabled.
|
|
|
|
* The CR0_READ_SHADOW is what L2 should have expected to read given
|
|
|
|
* the specifications by L1; It's not enough to take
|
|
|
|
* vmcs12->cr0_read_shadow because on our cr0_guest_host_mask we we
|
|
|
|
* have more bits than L1 expected.
|
|
|
|
*/
|
|
|
|
vmx_set_cr0(vcpu, vmcs12->guest_cr0);
|
|
|
|
vmcs_writel(CR0_READ_SHADOW, nested_read_cr0(vmcs12));
|
|
|
|
|
|
|
|
vmx_set_cr4(vcpu, vmcs12->guest_cr4);
|
|
|
|
vmcs_writel(CR4_READ_SHADOW, nested_read_cr4(vmcs12));
|
|
|
|
|
|
|
|
vcpu->arch.efer = nested_vmx_calc_efer(vmx, vmcs12);
|
|
|
|
/* Note: may modify VM_ENTRY/EXIT_CONTROLS and GUEST/HOST_IA32_EFER */
|
|
|
|
vmx_set_efer(vcpu, vcpu->arch.efer);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Guest state is invalid and unrestricted guest is disabled,
|
|
|
|
* which means L1 attempted VMEntry to L2 with invalid state.
|
|
|
|
* Fail the VMEntry.
|
|
|
|
*/
|
|
|
|
if (vmx->emulation_required) {
|
|
|
|
*entry_failure_code = ENTRY_FAIL_DEFAULT;
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Shadow page tables on either EPT or shadow page tables. */
|
|
|
|
if (nested_vmx_load_cr3(vcpu, vmcs12->guest_cr3, nested_cpu_has_ept(vmcs12),
|
|
|
|
entry_failure_code))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-09-28 04:45:16 +07:00
|
|
|
/*
|
|
|
|
* Immediately write vmcs02.GUEST_CR3. It will be propagated to vmcs12
|
|
|
|
* on nested VM-Exit, which can occur without actually running L2 and
|
|
|
|
* thus without hitting vmx_set_cr3(), e.g. if L1 is entering L2 with
|
|
|
|
* vmcs12.GUEST_ACTIVITYSTATE=HLT, in which case KVM will intercept the
|
|
|
|
* transition to HLT instead of running L2.
|
|
|
|
*/
|
|
|
|
if (enable_ept)
|
|
|
|
vmcs_writel(GUEST_CR3, vmcs12->guest_cr3);
|
|
|
|
|
2019-05-07 23:06:40 +07:00
|
|
|
/* Late preparation of GUEST_PDPTRs now that EFER and CRs are set. */
|
|
|
|
if (load_guest_pdptrs_vmcs12 && nested_cpu_has_ept(vmcs12) &&
|
|
|
|
is_pae_paging(vcpu)) {
|
|
|
|
vmcs_write64(GUEST_PDPTR0, vmcs12->guest_pdptr0);
|
|
|
|
vmcs_write64(GUEST_PDPTR1, vmcs12->guest_pdptr1);
|
|
|
|
vmcs_write64(GUEST_PDPTR2, vmcs12->guest_pdptr2);
|
|
|
|
vmcs_write64(GUEST_PDPTR3, vmcs12->guest_pdptr3);
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (!enable_ept)
|
|
|
|
vcpu->arch.walk_mmu->inject_page_fault = vmx_inject_page_fault_nested;
|
|
|
|
|
2019-11-14 07:17:19 +07:00
|
|
|
if ((vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_PERF_GLOBAL_CTRL) &&
|
|
|
|
SET_MSR_OR_WARN(vcpu, MSR_CORE_PERF_GLOBAL_CTRL,
|
|
|
|
vmcs12->guest_ia32_perf_global_ctrl))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-05-01 03:07:26 +07:00
|
|
|
kvm_rsp_write(vcpu, vmcs12->guest_rsp);
|
|
|
|
kvm_rip_write(vcpu, vmcs12->guest_rip);
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_nmi_controls(struct vmcs12 *vmcs12)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cpu_has_nmi_exiting(vmcs12) &&
|
|
|
|
nested_cpu_has_virtual_nmis(vmcs12)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cpu_has_virtual_nmis(vmcs12) &&
|
|
|
|
nested_cpu_has(vmcs12, CPU_BASED_VIRTUAL_NMI_PENDING)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool valid_ept_address(struct kvm_vcpu *vcpu, u64 address)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
int maxphyaddr = cpuid_maxphyaddr(vcpu);
|
|
|
|
|
|
|
|
/* Check for memory type validity */
|
|
|
|
switch (address & VMX_EPTP_MT_MASK) {
|
|
|
|
case VMX_EPTP_MT_UC:
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!(vmx->nested.msrs.ept_caps & VMX_EPTP_UC_BIT)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
case VMX_EPTP_MT_WB:
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!(vmx->nested.msrs.ept_caps & VMX_EPTP_WB_BIT)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* only 4 levels page-walk length are valid */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC((address & VMX_EPTP_PWL_MASK) != VMX_EPTP_PWL_4))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Reserved bits should not be set */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(address >> maxphyaddr || ((address >> 7) & 0x1f)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* AD, if set, should be supported */
|
|
|
|
if (address & VMX_EPTP_AD_ENABLE_BIT) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!(vmx->nested.msrs.ept_caps & VMX_EPT_AD_BIT)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:07 +07:00
|
|
|
/*
|
|
|
|
* Checks related to VM-Execution Control Fields
|
|
|
|
*/
|
|
|
|
static int nested_check_vm_execution_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!vmx_control_verify(vmcs12->pin_based_vm_exec_control,
|
|
|
|
vmx->nested.msrs.pinbased_ctls_low,
|
|
|
|
vmx->nested.msrs.pinbased_ctls_high)) ||
|
|
|
|
CC(!vmx_control_verify(vmcs12->cpu_based_vm_exec_control,
|
|
|
|
vmx->nested.msrs.procbased_ctls_low,
|
|
|
|
vmx->nested.msrs.procbased_ctls_high)))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2018-12-13 01:30:07 +07:00
|
|
|
if (nested_cpu_has(vmcs12, CPU_BASED_ACTIVATE_SECONDARY_CONTROLS) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(!vmx_control_verify(vmcs12->secondary_vm_exec_control,
|
|
|
|
vmx->nested.msrs.secondary_ctls_low,
|
|
|
|
vmx->nested.msrs.secondary_ctls_high)))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vmcs12->cr3_target_count > nested_cpu_vmx_misc_cr3_count(vcpu)) ||
|
2018-12-13 01:30:07 +07:00
|
|
|
nested_vmx_check_io_bitmap_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_msr_bitmap_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_tpr_shadow_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_apic_access_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_apicv_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_nmi_controls(vmcs12) ||
|
|
|
|
nested_vmx_check_pml_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_unrestricted_guest_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_mode_based_ept_exec_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_shadow_vmcs_controls(vcpu, vmcs12) ||
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(nested_cpu_has_vpid(vmcs12) && !vmcs12->virtual_processor_id))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-02-13 07:42:23 +07:00
|
|
|
if (!nested_cpu_has_preemption_timer(vmcs12) &&
|
|
|
|
nested_cpu_has_save_preemption_timer(vmcs12))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2018-12-13 01:30:07 +07:00
|
|
|
if (nested_cpu_has_ept(vmcs12) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(!valid_ept_address(vcpu, vmcs12->ept_pointer)))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (nested_cpu_has_vmfunc(vmcs12)) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vmcs12->vm_function_control &
|
|
|
|
~vmx->nested.msrs.vmfunc_controls))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (nested_cpu_has_eptp_switching(vmcs12)) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_cpu_has_ept(vmcs12)) ||
|
|
|
|
CC(!page_address_valid(vcpu, vmcs12->eptp_list_address)))
|
2018-12-13 01:30:07 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:07 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:09 +07:00
|
|
|
/*
|
|
|
|
* Checks related to VM-Exit Control Fields
|
|
|
|
*/
|
|
|
|
static int nested_check_vm_exit_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!vmx_control_verify(vmcs12->vm_exit_controls,
|
|
|
|
vmx->nested.msrs.exit_ctls_low,
|
|
|
|
vmx->nested.msrs.exit_ctls_high)) ||
|
|
|
|
CC(nested_vmx_check_exit_msr_switch_controls(vcpu, vmcs12)))
|
2018-12-13 01:30:09 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:10 +07:00
|
|
|
/*
|
|
|
|
* Checks related to VM-Entry Control Fields
|
|
|
|
*/
|
|
|
|
static int nested_check_vm_entry_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
2018-12-13 01:30:07 +07:00
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!vmx_control_verify(vmcs12->vm_entry_controls,
|
|
|
|
vmx->nested.msrs.entry_ctls_low,
|
|
|
|
vmx->nested.msrs.entry_ctls_high)))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* From the Intel SDM, volume 3:
|
|
|
|
* Fields relevant to VM-entry event injection must be set properly.
|
|
|
|
* These fields are the VM-entry interruption-information field, the
|
|
|
|
* VM-entry exception error code, and the VM-entry instruction length.
|
|
|
|
*/
|
|
|
|
if (vmcs12->vm_entry_intr_info_field & INTR_INFO_VALID_MASK) {
|
|
|
|
u32 intr_info = vmcs12->vm_entry_intr_info_field;
|
|
|
|
u8 vector = intr_info & INTR_INFO_VECTOR_MASK;
|
|
|
|
u32 intr_type = intr_info & INTR_INFO_INTR_TYPE_MASK;
|
|
|
|
bool has_error_code = intr_info & INTR_INFO_DELIVER_CODE_MASK;
|
|
|
|
bool should_have_error_code;
|
|
|
|
bool urg = nested_cpu_has2(vmcs12,
|
|
|
|
SECONDARY_EXEC_UNRESTRICTED_GUEST);
|
|
|
|
bool prot_mode = !urg || vmcs12->guest_cr0 & X86_CR0_PE;
|
|
|
|
|
|
|
|
/* VM-entry interruption-info field: interruption type */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(intr_type == INTR_TYPE_RESERVED) ||
|
|
|
|
CC(intr_type == INTR_TYPE_OTHER_EVENT &&
|
|
|
|
!nested_cpu_supports_monitor_trap_flag(vcpu)))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* VM-entry interruption-info field: vector */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(intr_type == INTR_TYPE_NMI_INTR && vector != NMI_VECTOR) ||
|
|
|
|
CC(intr_type == INTR_TYPE_HARD_EXCEPTION && vector > 31) ||
|
|
|
|
CC(intr_type == INTR_TYPE_OTHER_EVENT && vector != 0))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* VM-entry interruption-info field: deliver error code */
|
|
|
|
should_have_error_code =
|
|
|
|
intr_type == INTR_TYPE_HARD_EXCEPTION && prot_mode &&
|
|
|
|
x86_exception_has_error_code(vector);
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(has_error_code != should_have_error_code))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* VM-entry exception error code */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(has_error_code &&
|
2019-10-01 23:21:23 +07:00
|
|
|
vmcs12->vm_entry_exception_error_code & GENMASK(31, 16)))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* VM-entry interruption-info field: reserved bits */
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(intr_info & INTR_INFO_RESVD_BITS_MASK))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* VM-entry instruction length */
|
|
|
|
switch (intr_type) {
|
|
|
|
case INTR_TYPE_SOFT_EXCEPTION:
|
|
|
|
case INTR_TYPE_SOFT_INTR:
|
|
|
|
case INTR_TYPE_PRIV_SW_EXCEPTION:
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vmcs12->vm_entry_instruction_len > 15) ||
|
|
|
|
CC(vmcs12->vm_entry_instruction_len == 0 &&
|
|
|
|
CC(!nested_cpu_has_zero_length_injection(vcpu))))
|
2018-12-13 01:30:10 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-13 01:30:10 +07:00
|
|
|
if (nested_vmx_check_entry_msr_switch_controls(vcpu, vmcs12))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-12 02:18:06 +07:00
|
|
|
static int nested_vmx_check_controls(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
if (nested_check_vm_execution_controls(vcpu, vmcs12) ||
|
|
|
|
nested_check_vm_exit_controls(vcpu, vmcs12) ||
|
|
|
|
nested_check_vm_entry_controls(vcpu, vmcs12))
|
2019-04-12 15:19:57 +07:00
|
|
|
return -EINVAL;
|
2019-04-12 02:18:06 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-12 15:19:57 +07:00
|
|
|
static int nested_vmx_check_host_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
2018-12-13 01:30:10 +07:00
|
|
|
{
|
|
|
|
bool ia32e;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_host_cr0_valid(vcpu, vmcs12->host_cr0)) ||
|
|
|
|
CC(!nested_host_cr4_valid(vcpu, vmcs12->host_cr4)) ||
|
|
|
|
CC(!nested_cr3_valid(vcpu, vmcs12->host_cr3)))
|
2018-12-13 01:30:11 +07:00
|
|
|
return -EINVAL;
|
2019-02-08 02:05:30 +07:00
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(is_noncanonical_address(vmcs12->host_ia32_sysenter_esp, vcpu)) ||
|
|
|
|
CC(is_noncanonical_address(vmcs12->host_ia32_sysenter_eip, vcpu)))
|
2019-02-08 02:05:30 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-04-09 04:35:11 +07:00
|
|
|
if ((vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_PAT) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(!kvm_pat_valid(vmcs12->host_ia32_pat)))
|
2019-04-09 04:35:11 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-11-14 07:17:17 +07:00
|
|
|
if ((vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_PERF_GLOBAL_CTRL) &&
|
|
|
|
CC(!kvm_valid_perf_global_ctrl(vcpu_to_pmu(vcpu),
|
|
|
|
vmcs12->host_ia32_perf_global_ctrl)))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-09-25 23:33:53 +07:00
|
|
|
#ifdef CONFIG_X86_64
|
|
|
|
ia32e = !!(vcpu->arch.efer & EFER_LMA);
|
|
|
|
#else
|
|
|
|
ia32e = false;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (ia32e) {
|
|
|
|
if (CC(!(vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE)) ||
|
|
|
|
CC(!(vmcs12->host_cr4 & X86_CR4_PAE)))
|
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
|
|
|
if (CC(vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE) ||
|
|
|
|
CC(vmcs12->vm_entry_controls & VM_ENTRY_IA32E_MODE) ||
|
|
|
|
CC(vmcs12->host_cr4 & X86_CR4_PCIDE) ||
|
|
|
|
CC((vmcs12->host_rip) >> 32))
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2019-07-04 06:54:35 +07:00
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vmcs12->host_cs_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_ss_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_ds_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_es_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_fs_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_gs_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_tr_selector & (SEGMENT_RPL_MASK | SEGMENT_TI_MASK)) ||
|
|
|
|
CC(vmcs12->host_cs_selector == 0) ||
|
|
|
|
CC(vmcs12->host_tr_selector == 0) ||
|
|
|
|
CC(vmcs12->host_ss_selector == 0 && !ia32e))
|
2019-07-04 06:54:35 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
#ifdef CONFIG_X86_64
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(is_noncanonical_address(vmcs12->host_fs_base, vcpu)) ||
|
|
|
|
CC(is_noncanonical_address(vmcs12->host_gs_base, vcpu)) ||
|
|
|
|
CC(is_noncanonical_address(vmcs12->host_gdtr_base, vcpu)) ||
|
|
|
|
CC(is_noncanonical_address(vmcs12->host_idtr_base, vcpu)) ||
|
2019-09-25 23:33:53 +07:00
|
|
|
CC(is_noncanonical_address(vmcs12->host_tr_base, vcpu)) ||
|
|
|
|
CC(is_noncanonical_address(vmcs12->host_rip, vcpu)))
|
2019-08-10 02:26:19 +07:00
|
|
|
return -EINVAL;
|
2019-07-04 06:54:35 +07:00
|
|
|
#endif
|
|
|
|
|
2018-12-13 01:30:10 +07:00
|
|
|
/*
|
|
|
|
* If the load IA32_EFER VM-exit control is 1, bits reserved in the
|
|
|
|
* IA32_EFER MSR must be 0 in the field for that register. In addition,
|
|
|
|
* the values of the LMA and LME bits in the field must each be that of
|
|
|
|
* the host address-space size VM-exit control.
|
|
|
|
*/
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_EFER) {
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!kvm_valid_efer(vcpu, vmcs12->host_ia32_efer)) ||
|
|
|
|
CC(ia32e != !!(vmcs12->host_ia32_efer & EFER_LMA)) ||
|
|
|
|
CC(ia32e != !!(vmcs12->host_ia32_efer & EFER_LME)))
|
2018-12-13 01:30:11 +07:00
|
|
|
return -EINVAL;
|
2018-12-13 01:30:10 +07:00
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_check_vmcs_link_ptr(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
2019-02-01 03:24:41 +07:00
|
|
|
int r = 0;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vmcs12 *shadow;
|
2019-02-01 03:24:41 +07:00
|
|
|
struct kvm_host_map map;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vmcs12->vmcs_link_pointer == -1ull)
|
|
|
|
return 0;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!page_address_valid(vcpu, vmcs12->vmcs_link_pointer)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(kvm_vcpu_map(vcpu, gpa_to_gfn(vmcs12->vmcs_link_pointer), &map)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-02-01 03:24:41 +07:00
|
|
|
shadow = map.hva;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(shadow->hdr.revision_id != VMCS12_REVISION) ||
|
|
|
|
CC(shadow->hdr.shadow_vmcs != nested_cpu_has_shadow_vmcs(vmcs12)))
|
2018-12-04 04:53:18 +07:00
|
|
|
r = -EINVAL;
|
2019-02-01 03:24:41 +07:00
|
|
|
|
|
|
|
kvm_vcpu_unmap(vcpu, &map, false);
|
2018-12-04 04:53:18 +07:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2019-04-12 02:18:05 +07:00
|
|
|
/*
|
|
|
|
* Checks related to Guest Non-register State
|
|
|
|
*/
|
|
|
|
static int nested_check_guest_non_reg_state(struct vmcs12 *vmcs12)
|
|
|
|
{
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(vmcs12->guest_activity_state != GUEST_ACTIVITY_ACTIVE &&
|
|
|
|
vmcs12->guest_activity_state != GUEST_ACTIVITY_HLT))
|
2019-04-12 02:18:05 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-12 02:18:06 +07:00
|
|
|
static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12,
|
|
|
|
u32 *exit_qual)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
bool ia32e;
|
|
|
|
|
|
|
|
*exit_qual = ENTRY_FAIL_DEFAULT;
|
|
|
|
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!nested_guest_cr0_valid(vcpu, vmcs12->guest_cr0)) ||
|
|
|
|
CC(!nested_guest_cr4_valid(vcpu, vmcs12->guest_cr4)))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-04-09 04:35:12 +07:00
|
|
|
if ((vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_PAT) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
CC(!kvm_pat_valid(vmcs12->guest_ia32_pat)))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (nested_vmx_check_vmcs_link_ptr(vcpu, vmcs12)) {
|
|
|
|
*exit_qual = ENTRY_FAIL_VMCS_LINK_PTR;
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
2019-11-14 07:17:16 +07:00
|
|
|
if ((vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_PERF_GLOBAL_CTRL) &&
|
|
|
|
CC(!kvm_valid_perf_global_ctrl(vcpu_to_pmu(vcpu),
|
|
|
|
vmcs12->guest_ia32_perf_global_ctrl)))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* If the load IA32_EFER VM-entry control is 1, the following checks
|
|
|
|
* are performed on the field for the IA32_EFER MSR:
|
|
|
|
* - Bits reserved in the IA32_EFER MSR must be 0.
|
|
|
|
* - Bit 10 (corresponding to IA32_EFER.LMA) must equal the value of
|
|
|
|
* the IA-32e mode guest VM-exit control. It must also be identical
|
|
|
|
* to bit 8 (LME) if bit 31 in the CR0 field (corresponding to
|
|
|
|
* CR0.PG) is 1.
|
|
|
|
*/
|
|
|
|
if (to_vmx(vcpu)->nested.nested_run_pending &&
|
|
|
|
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_EFER)) {
|
|
|
|
ia32e = (vmcs12->vm_entry_controls & VM_ENTRY_IA32E_MODE) != 0;
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
if (CC(!kvm_valid_efer(vcpu, vmcs12->guest_ia32_efer)) ||
|
|
|
|
CC(ia32e != !!(vmcs12->guest_ia32_efer & EFER_LMA)) ||
|
|
|
|
CC(((vmcs12->guest_cr0 & X86_CR0_PG) &&
|
|
|
|
ia32e != !!(vmcs12->guest_ia32_efer & EFER_LME))))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((vmcs12->vm_entry_controls & VM_ENTRY_LOAD_BNDCFGS) &&
|
KVM: nVMX: add tracepoint for failed nested VM-Enter
Debugging a failed VM-Enter is often like searching for a needle in a
haystack, e.g. there are over 80 consistency checks that funnel into
the "invalid control field" error code. One way to expedite debug is
to run the buggy code as an L1 guest under KVM (and pray that the
failing check is detected by KVM). However, extracting useful debug
information out of L0 KVM requires attaching a debugger to KVM and/or
modifying the source, e.g. to log which check is failing.
Make life a little less painful for VMM developers and add a tracepoint
for failed VM-Enter consistency checks. Ideally the tracepoint would
capture both what check failed and precisely why it failed, but logging
why a checked failed is difficult to do in a generic tracepoint without
resorting to invasive techniques, e.g. generating a custom string on
failure. That being said, for the vast majority of VM-Enter failures
the most difficult step is figuring out exactly what to look at, e.g.
figuring out which bit was incorrectly set in a control field is usually
not too painful once the guilty field as been identified.
To reach a happy medium between precision and ease of use, simply log
the code that detected a failed check, using a macro to execute the
check and log the trace event on failure. This approach enables tracing
arbitrary code, e.g. it's not limited to function calls or specific
formats of checks, and the changes to the existing code are minimally
invasive. A macro with a two-character name is desirable as usage of
the macro doesn't result in overly long lines or confusing alignment,
while still retaining some amount of readability. I.e. a one-character
name is a little too terse, and a three-character name results in the
contents being passed to the macro aligning with an indented line when
the macro is used an in if-statement, e.g.:
if (VCC(nested_vmx_check_long_line_one(...) &&
nested_vmx_check_long_line_two(...)))
return -EINVAL;
And that is the story of how the CC(), a.k.a. Consistency Check, macro
got its name.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-07-11 22:58:29 +07:00
|
|
|
(CC(is_noncanonical_address(vmcs12->guest_bndcfgs & PAGE_MASK, vcpu)) ||
|
|
|
|
CC((vmcs12->guest_bndcfgs & MSR_IA32_BNDCFGS_RSVD))))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-04-12 02:18:05 +07:00
|
|
|
if (nested_check_guest_non_reg_state(vmcs12))
|
2019-04-12 02:18:09 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
static int nested_vmx_check_vmentry_hw(struct kvm_vcpu *vcpu)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
unsigned long cr3, cr4;
|
2019-01-25 22:40:58 +07:00
|
|
|
bool vm_fail;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (!nested_early_check)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (vmx->msr_autoload.host.nr)
|
|
|
|
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, 0);
|
|
|
|
if (vmx->msr_autoload.guest.nr)
|
|
|
|
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, 0);
|
|
|
|
|
|
|
|
preempt_disable();
|
|
|
|
|
|
|
|
vmx_prepare_switch_to_guest(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Induce a consistency check VMExit by clearing bit 1 in GUEST_RFLAGS,
|
|
|
|
* which is reserved to '1' by hardware. GUEST_RFLAGS is guaranteed to
|
|
|
|
* be written (by preparve_vmcs02()) before the "real" VMEnter, i.e.
|
|
|
|
* there is no need to preserve other bits or save/restore the field.
|
|
|
|
*/
|
|
|
|
vmcs_writel(GUEST_RFLAGS, 0);
|
|
|
|
|
|
|
|
cr3 = __get_current_cr3_fast();
|
|
|
|
if (unlikely(cr3 != vmx->loaded_vmcs->host_state.cr3)) {
|
|
|
|
vmcs_writel(HOST_CR3, cr3);
|
|
|
|
vmx->loaded_vmcs->host_state.cr3 = cr3;
|
|
|
|
}
|
|
|
|
|
|
|
|
cr4 = cr4_read_shadow();
|
|
|
|
if (unlikely(cr4 != vmx->loaded_vmcs->host_state.cr4)) {
|
|
|
|
vmcs_writel(HOST_CR4, cr4);
|
|
|
|
vmx->loaded_vmcs->host_state.cr4 = cr4;
|
|
|
|
}
|
|
|
|
|
|
|
|
asm(
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
"sub $%c[wordsize], %%" _ASM_SP "\n\t" /* temporarily adjust RSP for CALL */
|
2019-01-25 22:41:02 +07:00
|
|
|
"cmp %%" _ASM_SP ", %c[host_state_rsp](%[loaded_vmcs]) \n\t"
|
|
|
|
"je 1f \n\t"
|
2019-01-25 22:41:01 +07:00
|
|
|
__ex("vmwrite %%" _ASM_SP ", %[HOST_RSP]") "\n\t"
|
2019-01-25 22:41:02 +07:00
|
|
|
"mov %%" _ASM_SP ", %c[host_state_rsp](%[loaded_vmcs]) \n\t"
|
|
|
|
"1: \n\t"
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
"add $%c[wordsize], %%" _ASM_SP "\n\t" /* un-adjust RSP */
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Check if vmlaunch or vmresume is needed */
|
KVM: nVMX: Reference vmx->loaded_vmcs->launched directly
Temporarily propagating vmx->loaded_vmcs->launched to vmx->__launched
is not functionally necessary, but rather was done historically to
avoid passing both 'vmx' and 'loaded_vmcs' to the vCPU-run asm blob.
Nested early checks inherited this behavior by virtue of copy+paste.
A future patch will move HOST_RSP caching to be per-VMCS, i.e. store
'host_rsp' in loaded VMCS. Now that the reference to 'vmx->fail' is
also gone from nested early checks, referencing 'loaded_vmcs' directly
means we can drop the 'vmx' reference when introducing per-VMCS RSP
caching. And it means __launched can be dropped from struct vcpu_vmx
if/when vCPU-run receives similar treatment.
Note the use of a named register constraint for 'loaded_vmcs'. Using
RCX to hold 'vmx' was inherited from vCPU-run. In the vCPU-run case,
the scratch register needs to be explicitly defined as it is crushed
when loading guest state, i.e. deferring to the compiler would corrupt
the pointer. Since nested early checks never loads guests state, it's
a-ok to let the compiler pick any register. Naming the constraint
avoids the fragility of referencing constraints via %1, %2, etc.., which
breaks horribly when modifying constraints, and generally makes the asm
blob more readable.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-01-25 22:41:00 +07:00
|
|
|
"cmpb $0, %c[launched](%[loaded_vmcs])\n\t"
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
|
2019-01-25 22:40:58 +07:00
|
|
|
/*
|
|
|
|
* VMLAUNCH and VMRESUME clear RFLAGS.{CF,ZF} on VM-Exit, set
|
|
|
|
* RFLAGS.CF on VM-Fail Invalid and set RFLAGS.ZF on VM-Fail
|
|
|
|
* Valid. vmx_vmenter() directly "returns" RFLAGS, and so the
|
2019-01-25 22:40:59 +07:00
|
|
|
* results of VM-Enter is captured via CC_{SET,OUT} to vm_fail.
|
2019-01-25 22:40:58 +07:00
|
|
|
*/
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
"call vmx_vmenter\n\t"
|
|
|
|
|
2019-01-25 22:40:59 +07:00
|
|
|
CC_SET(be)
|
|
|
|
: ASM_CALL_CONSTRAINT, CC_OUT(be) (vm_fail)
|
2019-01-25 22:41:02 +07:00
|
|
|
: [HOST_RSP]"r"((unsigned long)HOST_RSP),
|
KVM: nVMX: Reference vmx->loaded_vmcs->launched directly
Temporarily propagating vmx->loaded_vmcs->launched to vmx->__launched
is not functionally necessary, but rather was done historically to
avoid passing both 'vmx' and 'loaded_vmcs' to the vCPU-run asm blob.
Nested early checks inherited this behavior by virtue of copy+paste.
A future patch will move HOST_RSP caching to be per-VMCS, i.e. store
'host_rsp' in loaded VMCS. Now that the reference to 'vmx->fail' is
also gone from nested early checks, referencing 'loaded_vmcs' directly
means we can drop the 'vmx' reference when introducing per-VMCS RSP
caching. And it means __launched can be dropped from struct vcpu_vmx
if/when vCPU-run receives similar treatment.
Note the use of a named register constraint for 'loaded_vmcs'. Using
RCX to hold 'vmx' was inherited from vCPU-run. In the vCPU-run case,
the scratch register needs to be explicitly defined as it is crushed
when loading guest state, i.e. deferring to the compiler would corrupt
the pointer. Since nested early checks never loads guests state, it's
a-ok to let the compiler pick any register. Naming the constraint
avoids the fragility of referencing constraints via %1, %2, etc.., which
breaks horribly when modifying constraints, and generally makes the asm
blob more readable.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Jim Mattson <jmattson@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-01-25 22:41:00 +07:00
|
|
|
[loaded_vmcs]"r"(vmx->loaded_vmcs),
|
|
|
|
[launched]"i"(offsetof(struct loaded_vmcs, launched)),
|
2019-01-25 22:41:02 +07:00
|
|
|
[host_state_rsp]"i"(offsetof(struct loaded_vmcs, host_state.rsp)),
|
KVM: VMX: Move VM-Enter + VM-Exit handling to non-inline sub-routines
Transitioning to/from a VMX guest requires KVM to manually save/load
the bulk of CPU state that the guest is allowed to direclty access,
e.g. XSAVE state, CR2, GPRs, etc... For obvious reasons, loading the
guest's GPR snapshot prior to VM-Enter and saving the snapshot after
VM-Exit is done via handcoded assembly. The assembly blob is written
as inline asm so that it can easily access KVM-defined structs that
are used to hold guest state, e.g. moving the blob to a standalone
assembly file would require generating defines for struct offsets.
The other relevant aspect of VMX transitions in KVM is the handling of
VM-Exits. KVM doesn't employ a separate VM-Exit handler per se, but
rather treats the VMX transition as a mega instruction (with many side
effects), i.e. sets the VMCS.HOST_RIP to a label immediately following
VMLAUNCH/VMRESUME. The label is then exposed to C code via a global
variable definition in the inline assembly.
Because of the global variable, KVM takes steps to (attempt to) ensure
only a single instance of the owning C function, e.g. vmx_vcpu_run, is
generated by the compiler. The earliest approach placed the inline
assembly in a separate noinline function[1]. Later, the assembly was
folded back into vmx_vcpu_run() and tagged with __noclone[2][3], which
is still used today.
After moving to __noclone, an edge case was encountered where GCC's
-ftracer optimization resulted in the inline assembly blob being
duplicated. This was "fixed" by explicitly disabling -ftracer in the
__noclone definition[4].
Recently, it was found that disabling -ftracer causes build warnings
for unsuspecting users of __noclone[5], and more importantly for KVM,
prevents the compiler for properly optimizing vmx_vcpu_run()[6]. And
perhaps most importantly of all, it was pointed out that there is no
way to prevent duplication of a function with 100% reliability[7],
i.e. more edge cases may be encountered in the future.
So to summarize, the only way to prevent the compiler from duplicating
the global variable definition is to move the variable out of inline
assembly, which has been suggested several times over[1][7][8].
Resolve the aforementioned issues by moving the VMLAUNCH+VRESUME and
VM-Exit "handler" to standalone assembly sub-routines. Moving only
the core VMX transition codes allows the struct indexing to remain as
inline assembly and also allows the sub-routines to be used by
nested_vmx_check_vmentry_hw(). Reusing the sub-routines has a happy
side-effect of eliminating two VMWRITEs in the nested_early_check path
as there is no longer a need to dynamically change VMCS.HOST_RIP.
Note that callers to vmx_vmenter() must account for the CALL modifying
RSP, e.g. must subtract op-size from RSP when synchronizing RSP with
VMCS.HOST_RSP and "restore" RSP prior to the CALL. There are no great
alternatives to fudging RSP. Saving RSP in vmx_enter() is difficult
because doing so requires a second register (VMWRITE does not provide
an immediate encoding for the VMCS field and KVM supports Hyper-V's
memory-based eVMCS ABI). The other more drastic alternative would be
to use eschew VMCS.HOST_RSP and manually save/load RSP using a per-cpu
variable (which can be encoded as e.g. gs:[imm]). But because a valid
stack is needed at the time of VM-Exit (NMIs aren't blocked and a user
could theoretically insert INT3/INT1ICEBRK at the VM-Exit handler), a
dedicated per-cpu VM-Exit stack would be required. A dedicated stack
isn't difficult to implement, but it would require at least one page
per CPU and knowledge of the stack in the dumpstack routines. And in
most cases there is essentially zero overhead in dynamically updating
VMCS.HOST_RSP, e.g. the VMWRITE can be avoided for all but the first
VMLAUNCH unless nested_early_check=1, which is not a fast path. In
other words, avoiding the VMCS.HOST_RSP by using a dedicated stack
would only make the code marginally less ugly while requiring at least
one page per CPU and forcing the kernel to be aware (and approve) of
the VM-Exit stack shenanigans.
[1] cea15c24ca39 ("KVM: Move KVM context switch into own function")
[2] a3b5ba49a8c5 ("KVM: VMX: add the __noclone attribute to vmx_vcpu_run")
[3] 104f226bfd0a ("KVM: VMX: Fold __vmx_vcpu_run() into vmx_vcpu_run()")
[4] 95272c29378e ("compiler-gcc: disable -ftracer for __noclone functions")
[5] https://lkml.kernel.org/r/20181218140105.ajuiglkpvstt3qxs@treble
[6] https://patchwork.kernel.org/patch/8707981/#21817015
[7] https://lkml.kernel.org/r/ri6y38lo23g.fsf@suse.cz
[8] https://lkml.kernel.org/r/20181218212042.GE25620@tassilo.jf.intel.com
Suggested-by: Andi Kleen <ak@linux.intel.com>
Suggested-by: Martin Jambor <mjambor@suse.cz>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: Nadav Amit <namit@vmware.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Josh Poimboeuf <jpoimboe@redhat.com>
Cc: Martin Jambor <mjambor@suse.cz>
Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Miroslav Benes <mbenes@suse.cz>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-21 03:25:17 +07:00
|
|
|
[wordsize]"i"(sizeof(ulong))
|
2019-05-27 15:45:44 +07:00
|
|
|
: "memory"
|
2018-12-04 04:53:18 +07:00
|
|
|
);
|
|
|
|
|
|
|
|
if (vmx->msr_autoload.host.nr)
|
|
|
|
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, vmx->msr_autoload.host.nr);
|
|
|
|
if (vmx->msr_autoload.guest.nr)
|
|
|
|
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, vmx->msr_autoload.guest.nr);
|
|
|
|
|
2019-01-25 22:40:58 +07:00
|
|
|
if (vm_fail) {
|
2019-07-11 22:58:30 +07:00
|
|
|
u32 error = vmcs_read32(VM_INSTRUCTION_ERROR);
|
|
|
|
|
2019-05-17 15:49:50 +07:00
|
|
|
preempt_enable();
|
2019-07-11 22:58:30 +07:00
|
|
|
|
|
|
|
trace_kvm_nested_vmenter_failed(
|
|
|
|
"early hardware check VM-instruction error: ", error);
|
|
|
|
WARN_ON_ONCE(error != VMXERR_ENTRY_INVALID_CONTROL_FIELD);
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* VMExit clears RFLAGS.IF and DR7, even on a consistency check.
|
|
|
|
*/
|
|
|
|
local_irq_enable();
|
|
|
|
if (hw_breakpoint_active())
|
|
|
|
set_debugreg(__this_cpu_read(cpu_dr7), 7);
|
2019-05-17 15:49:50 +07:00
|
|
|
preempt_enable();
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A non-failing VMEntry means we somehow entered guest mode with
|
|
|
|
* an illegal RIP, and that's just the tip of the iceberg. There
|
|
|
|
* is no telling what memory has been modified or what state has
|
|
|
|
* been exposed to unknown code. Hitting this all but guarantees
|
|
|
|
* a (very critical) hardware issue.
|
|
|
|
*/
|
|
|
|
WARN_ON(!(vmcs_read32(VM_EXIT_REASON) &
|
|
|
|
VMX_EXIT_REASONS_FAILED_VMENTRY));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool nested_vmx_prepare_msr_bitmap(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12);
|
|
|
|
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
static bool nested_get_vmcs12_pages(struct kvm_vcpu *vcpu)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
2019-02-01 03:24:37 +07:00
|
|
|
struct kvm_host_map *map;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct page *page;
|
|
|
|
u64 hpa;
|
|
|
|
|
|
|
|
if (nested_cpu_has2(vmcs12, SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES)) {
|
|
|
|
/*
|
|
|
|
* Translate L1 physical address to host physical
|
|
|
|
* address for vmcs02. Keep the page pinned, so this
|
|
|
|
* physical address remains valid. We keep a reference
|
|
|
|
* to it so we can release it later.
|
|
|
|
*/
|
|
|
|
if (vmx->nested.apic_access_page) { /* shouldn't happen */
|
2019-11-21 05:31:47 +07:00
|
|
|
kvm_release_page_clean(vmx->nested.apic_access_page);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.apic_access_page = NULL;
|
|
|
|
}
|
|
|
|
page = kvm_vcpu_gpa_to_page(vcpu, vmcs12->apic_access_addr);
|
|
|
|
if (!is_error_page(page)) {
|
|
|
|
vmx->nested.apic_access_page = page;
|
|
|
|
hpa = page_to_phys(vmx->nested.apic_access_page);
|
|
|
|
vmcs_write64(APIC_ACCESS_ADDR, hpa);
|
|
|
|
} else {
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
pr_debug_ratelimited("%s: no backing 'struct page' for APIC-access address in vmcs12\n",
|
|
|
|
__func__);
|
|
|
|
vcpu->run->exit_reason = KVM_EXIT_INTERNAL_ERROR;
|
|
|
|
vcpu->run->internal.suberror =
|
|
|
|
KVM_INTERNAL_ERROR_EMULATION;
|
|
|
|
vcpu->run->internal.ndata = 0;
|
|
|
|
return false;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has(vmcs12, CPU_BASED_TPR_SHADOW)) {
|
2019-02-01 03:24:37 +07:00
|
|
|
map = &vmx->nested.virtual_apic_map;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-02-01 03:24:37 +07:00
|
|
|
if (!kvm_vcpu_map(vcpu, gpa_to_gfn(vmcs12->virtual_apic_page_addr), map)) {
|
|
|
|
vmcs_write64(VIRTUAL_APIC_PAGE_ADDR, pfn_to_hpa(map->pfn));
|
2019-04-15 20:16:17 +07:00
|
|
|
} else if (nested_cpu_has(vmcs12, CPU_BASED_CR8_LOAD_EXITING) &&
|
|
|
|
nested_cpu_has(vmcs12, CPU_BASED_CR8_STORE_EXITING) &&
|
|
|
|
!nested_cpu_has2(vmcs12, SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES)) {
|
|
|
|
/*
|
|
|
|
* The processor will never use the TPR shadow, simply
|
|
|
|
* clear the bit from the execution control. Such a
|
|
|
|
* configuration is useless, but it happens in tests.
|
|
|
|
* For any other configuration, failing the vm entry is
|
|
|
|
* _not_ what the processor does but it's basically the
|
|
|
|
* only possibility we have.
|
|
|
|
*/
|
2019-05-08 02:17:56 +07:00
|
|
|
exec_controls_clearbit(vmx, CPU_BASED_TPR_SHADOW);
|
2019-04-15 20:16:17 +07:00
|
|
|
} else {
|
2019-05-07 23:06:33 +07:00
|
|
|
/*
|
|
|
|
* Write an illegal value to VIRTUAL_APIC_PAGE_ADDR to
|
|
|
|
* force VM-Entry to fail.
|
|
|
|
*/
|
|
|
|
vmcs_write64(VIRTUAL_APIC_PAGE_ADDR, -1ull);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has_posted_intr(vmcs12)) {
|
2019-02-01 03:24:38 +07:00
|
|
|
map = &vmx->nested.pi_desc_map;
|
|
|
|
|
|
|
|
if (!kvm_vcpu_map(vcpu, gpa_to_gfn(vmcs12->posted_intr_desc_addr), map)) {
|
|
|
|
vmx->nested.pi_desc =
|
|
|
|
(struct pi_desc *)(((void *)map->hva) +
|
|
|
|
offset_in_page(vmcs12->posted_intr_desc_addr));
|
|
|
|
vmcs_write64(POSTED_INTR_DESC_ADDR,
|
|
|
|
pfn_to_hpa(map->pfn) + offset_in_page(vmcs12->posted_intr_desc_addr));
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nested_vmx_prepare_msr_bitmap(vcpu, vmcs12))
|
2019-05-08 02:17:56 +07:00
|
|
|
exec_controls_setbit(vmx, CPU_BASED_USE_MSR_BITMAPS);
|
2018-12-04 04:53:18 +07:00
|
|
|
else
|
2019-05-08 02:17:56 +07:00
|
|
|
exec_controls_clearbit(vmx, CPU_BASED_USE_MSR_BITMAPS);
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
return true;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Intel's VMX Instruction Reference specifies a common set of prerequisites
|
|
|
|
* for running VMX instructions (except VMXON, whose prerequisites are
|
|
|
|
* slightly different). It also specifies what exception to inject otherwise.
|
|
|
|
* Note that many of these exceptions have priority over VM exits, so they
|
|
|
|
* don't have to be checked again here.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_check_permission(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
if (!to_vmx(vcpu)->nested.vmxon) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vmx_get_cpl(vcpu)) {
|
|
|
|
kvm_inject_gp(vcpu, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 vmx_has_apicv_interrupt(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
u8 rvi = vmx_get_rvi();
|
|
|
|
u8 vppr = kvm_lapic_get_reg(vcpu->arch.apic, APIC_PROCPRI);
|
|
|
|
|
|
|
|
return ((rvi & 0xf0) > (vppr & 0xf0));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void load_vmcs12_host_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If from_vmentry is false, this is being called from state restore (either RSM
|
|
|
|
* or KVM_SET_NESTED_STATE). Otherwise it's called from vmlaunch/vmresume.
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* NVMX_ENTRY_SUCCESS: Entered VMX non-root mode
|
|
|
|
* NVMX_ENTRY_VMFAIL: Consistency check VMFail
|
|
|
|
* NVMX_ENTRY_VMEXIT: Consistency check VMExit
|
|
|
|
* NVMX_ENTRY_KVM_INTERNAL_ERROR: KVM internal error
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
enum nvmx_vmentry_status nested_vmx_enter_non_root_mode(struct kvm_vcpu *vcpu,
|
|
|
|
bool from_vmentry)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
bool evaluate_pending_interrupts;
|
|
|
|
u32 exit_reason = EXIT_REASON_INVALID_STATE;
|
|
|
|
u32 exit_qual;
|
|
|
|
|
2019-05-08 02:17:56 +07:00
|
|
|
evaluate_pending_interrupts = exec_controls_get(vmx) &
|
2018-12-04 04:53:18 +07:00
|
|
|
(CPU_BASED_VIRTUAL_INTR_PENDING | CPU_BASED_VIRTUAL_NMI_PENDING);
|
|
|
|
if (likely(!evaluate_pending_interrupts) && kvm_vcpu_apicv_active(vcpu))
|
|
|
|
evaluate_pending_interrupts |= vmx_has_apicv_interrupt(vcpu);
|
|
|
|
|
|
|
|
if (!(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_DEBUG_CONTROLS))
|
|
|
|
vmx->nested.vmcs01_debugctl = vmcs_read64(GUEST_IA32_DEBUGCTL);
|
|
|
|
if (kvm_mpx_supported() &&
|
|
|
|
!(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_BNDCFGS))
|
|
|
|
vmx->nested.vmcs01_guest_bndcfgs = vmcs_read64(GUEST_BNDCFGS);
|
|
|
|
|
KVM: nVMX: Stash L1's CR3 in vmcs01.GUEST_CR3 on nested entry w/o EPT
KVM does not have 100% coverage of VMX consistency checks, i.e. some
checks that cause VM-Fail may only be detected by hardware during a
nested VM-Entry. In such a case, KVM must restore L1's state to the
pre-VM-Enter state as L2's state has already been loaded into KVM's
software model.
L1's CR3 and PDPTRs in particular are loaded from vmcs01.GUEST_*. But
when EPT is disabled, the associated fields hold KVM's shadow values,
not L1's "real" values. Fortunately, when EPT is disabled the PDPTRs
come from memory, i.e. are not cached in the VMCS. Which leaves CR3
as the sole anomaly.
A previously applied workaround to handle CR3 was to force nested early
checks if EPT is disabled:
commit 2b27924bb1d48 ("KVM: nVMX: always use early vmcs check when EPT
is disabled")
Forcing nested early checks is undesirable as doing so adds hundreds of
cycles to every nested VM-Entry. Rather than take this performance hit,
handle CR3 by overwriting vmcs01.GUEST_CR3 with L1's CR3 during nested
VM-Entry when EPT is disabled *and* nested early checks are disabled.
By stuffing vmcs01.GUEST_CR3, nested_vmx_restore_host_state() will
naturally restore the correct vcpu->arch.cr3 from vmcs01.GUEST_CR3.
These shenanigans work because nested_vmx_restore_host_state() does a
full kvm_mmu_reset_context(), i.e. unloads the current MMU, which
guarantees vmcs01.GUEST_CR3 will be rewritten with a new shadow CR3
prior to re-entering L1.
vcpu->arch.root_mmu.root_hpa is set to INVALID_PAGE via:
nested_vmx_restore_host_state() ->
kvm_mmu_reset_context() ->
kvm_mmu_unload() ->
kvm_mmu_free_roots()
kvm_mmu_unload() has WARN_ON(root_hpa != INVALID_PAGE), i.e. we can bank
on 'root_hpa == INVALID_PAGE' unless the implementation of
kvm_mmu_reset_context() is changed.
On the way into L1, VMCS.GUEST_CR3 is guaranteed to be written (on a
successful entry) via:
vcpu_enter_guest() ->
kvm_mmu_reload() ->
kvm_mmu_load() ->
kvm_mmu_load_cr3() ->
vmx_set_cr3()
Stuff vmcs01.GUEST_CR3 if and only if nested early checks are disabled
as a "late" VM-Fail should never happen win that case (KVM WARNs), and
the conditional write avoids the need to restore the correct GUEST_CR3
when nested_vmx_check_vmentry_hw() fails.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Message-Id: <20190607185534.24368-1-sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-08 01:55:34 +07:00
|
|
|
/*
|
|
|
|
* Overwrite vmcs01.GUEST_CR3 with L1's CR3 if EPT is disabled *and*
|
|
|
|
* nested early checks are disabled. In the event of a "late" VM-Fail,
|
|
|
|
* i.e. a VM-Fail detected by hardware but not KVM, KVM must unwind its
|
|
|
|
* software model to the pre-VMEntry host state. When EPT is disabled,
|
|
|
|
* GUEST_CR3 holds KVM's shadow CR3, not L1's "real" CR3, which causes
|
|
|
|
* nested_vmx_restore_host_state() to corrupt vcpu->arch.cr3. Stuffing
|
|
|
|
* vmcs01.GUEST_CR3 results in the unwind naturally setting arch.cr3 to
|
|
|
|
* the correct value. Smashing vmcs01.GUEST_CR3 is safe because nested
|
|
|
|
* VM-Exits, and the unwind, reset KVM's MMU, i.e. vmcs01.GUEST_CR3 is
|
|
|
|
* guaranteed to be overwritten with a shadow CR3 prior to re-entering
|
|
|
|
* L1. Don't stuff vmcs01.GUEST_CR3 when using nested early checks as
|
|
|
|
* KVM modifies vcpu->arch.cr3 if and only if the early hardware checks
|
|
|
|
* pass, and early VM-Fails do not reset KVM's MMU, i.e. the VM-Fail
|
|
|
|
* path would need to manually save/restore vmcs01.GUEST_CR3.
|
|
|
|
*/
|
|
|
|
if (!enable_ept && !nested_early_check)
|
|
|
|
vmcs_writel(GUEST_CR3, vcpu->arch.cr3);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx_switch_vmcs(vcpu, &vmx->nested.vmcs02);
|
|
|
|
|
|
|
|
prepare_vmcs02_early(vmx, vmcs12);
|
|
|
|
|
|
|
|
if (from_vmentry) {
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
if (unlikely(!nested_get_vmcs12_pages(vcpu)))
|
|
|
|
return NVMX_VMENTRY_KVM_INTERNAL_ERROR;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (nested_vmx_check_vmentry_hw(vcpu)) {
|
|
|
|
vmx_switch_vmcs(vcpu, &vmx->vmcs01);
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
return NVMX_VMENTRY_VMFAIL;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
2019-04-12 02:18:06 +07:00
|
|
|
if (nested_vmx_check_guest_state(vcpu, vmcs12, &exit_qual))
|
2018-12-04 04:53:18 +07:00
|
|
|
goto vmentry_fail_vmexit;
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_guest_mode(vcpu);
|
|
|
|
if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING)
|
|
|
|
vcpu->arch.tsc_offset += vmcs12->tsc_offset;
|
|
|
|
|
|
|
|
if (prepare_vmcs02(vcpu, vmcs12, &exit_qual))
|
|
|
|
goto vmentry_fail_vmexit_guest_mode;
|
|
|
|
|
|
|
|
if (from_vmentry) {
|
|
|
|
exit_reason = EXIT_REASON_MSR_LOAD_FAIL;
|
|
|
|
exit_qual = nested_vmx_load_msr(vcpu,
|
|
|
|
vmcs12->vm_entry_msr_load_addr,
|
|
|
|
vmcs12->vm_entry_msr_load_count);
|
|
|
|
if (exit_qual)
|
|
|
|
goto vmentry_fail_vmexit_guest_mode;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* The MMU is not initialized to point at the right entities yet and
|
|
|
|
* "get pages" would need to read data from the guest (i.e. we will
|
|
|
|
* need to perform gpa to hpa translation). Request a call
|
|
|
|
* to nested_get_vmcs12_pages before the next VM-entry. The MSRs
|
|
|
|
* have already been set at vmentry time and should not be reset.
|
|
|
|
*/
|
|
|
|
kvm_make_request(KVM_REQ_GET_VMCS12_PAGES, vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If L1 had a pending IRQ/NMI until it executed
|
|
|
|
* VMLAUNCH/VMRESUME which wasn't delivered because it was
|
|
|
|
* disallowed (e.g. interrupts disabled), L0 needs to
|
|
|
|
* evaluate if this pending event should cause an exit from L2
|
|
|
|
* to L1 or delivered directly to L2 (e.g. In case L1 don't
|
|
|
|
* intercept EXTERNAL_INTERRUPT).
|
|
|
|
*
|
|
|
|
* Usually this would be handled by the processor noticing an
|
|
|
|
* IRQ/NMI window request, or checking RVI during evaluation of
|
|
|
|
* pending virtual interrupts. However, this setting was done
|
|
|
|
* on VMCS01 and now VMCS02 is active instead. Thus, we force L0
|
|
|
|
* to perform pending event evaluation by requesting a KVM_REQ_EVENT.
|
|
|
|
*/
|
|
|
|
if (unlikely(evaluate_pending_interrupts))
|
|
|
|
kvm_make_request(KVM_REQ_EVENT, vcpu);
|
|
|
|
|
2019-01-30 01:14:46 +07:00
|
|
|
/*
|
|
|
|
* Do not start the preemption timer hrtimer until after we know
|
|
|
|
* we are successful, so that only nested_vmx_vmexit needs to cancel
|
|
|
|
* the timer.
|
|
|
|
*/
|
|
|
|
vmx->nested.preemption_timer_expired = false;
|
|
|
|
if (nested_cpu_has_preemption_timer(vmcs12))
|
|
|
|
vmx_start_preemption_timer(vcpu);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Note no nested_vmx_succeed or nested_vmx_fail here. At this point
|
|
|
|
* we are no longer running L1, and VMLAUNCH/VMRESUME has not yet
|
|
|
|
* returned as far as L1 is concerned. It will only return (and set
|
|
|
|
* the success flag) when L2 exits (see nested_vmx_vmexit()).
|
|
|
|
*/
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
return NVMX_VMENTRY_SUCCESS;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A failed consistency check that leads to a VMExit during L1's
|
|
|
|
* VMEnter to L2 is a variation of a normal VMexit, as explained in
|
|
|
|
* 26.7 "VM-entry failures during or after loading guest state".
|
|
|
|
*/
|
|
|
|
vmentry_fail_vmexit_guest_mode:
|
|
|
|
if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING)
|
|
|
|
vcpu->arch.tsc_offset -= vmcs12->tsc_offset;
|
|
|
|
leave_guest_mode(vcpu);
|
|
|
|
|
|
|
|
vmentry_fail_vmexit:
|
|
|
|
vmx_switch_vmcs(vcpu, &vmx->vmcs01);
|
|
|
|
|
|
|
|
if (!from_vmentry)
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
return NVMX_VMENTRY_VMEXIT;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
load_vmcs12_host_state(vcpu, vmcs12);
|
|
|
|
vmcs12->vm_exit_reason = exit_reason | VMX_EXIT_REASONS_FAILED_VMENTRY;
|
|
|
|
vmcs12->exit_qualification = exit_qual;
|
|
|
|
if (enable_shadow_vmcs || vmx->nested.hv_evmcs)
|
2019-05-07 22:36:27 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = true;
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
return NVMX_VMENTRY_VMEXIT;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* nested_vmx_run() handles a nested entry, i.e., a VMLAUNCH or VMRESUME on L1
|
|
|
|
* for running an L2 nested guest.
|
|
|
|
*/
|
|
|
|
static int nested_vmx_run(struct kvm_vcpu *vcpu, bool launch)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12;
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
enum nvmx_vmentry_status status;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u32 interrupt_shadow = vmx_get_interrupt_shadow(vcpu);
|
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
2019-06-28 18:23:32 +07:00
|
|
|
if (!nested_vmx_handle_enlightened_vmptrld(vcpu, launch))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (!vmx->nested.hv_evmcs && vmx->nested.current_vmptr == -1ull)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Can't VMLAUNCH or VMRESUME a shadow VMCS. Despite the fact
|
|
|
|
* that there *is* a valid VMCS pointer, RFLAGS.CF is set
|
|
|
|
* rather than RFLAGS.ZF, and no error number is stored to the
|
|
|
|
* VM-instruction error field.
|
|
|
|
*/
|
|
|
|
if (vmcs12->hdr.shadow_vmcs)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
if (vmx->nested.hv_evmcs) {
|
|
|
|
copy_enlightened_to_vmcs12(vmx);
|
|
|
|
/* Enlightened VMCS doesn't have launch state */
|
|
|
|
vmcs12->launch_state = !launch;
|
|
|
|
} else if (enable_shadow_vmcs) {
|
|
|
|
copy_shadow_to_vmcs12(vmx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The nested entry process starts with enforcing various prerequisites
|
|
|
|
* on vmcs12 as required by the Intel SDM, and act appropriately when
|
|
|
|
* they fail: As the SDM explains, some conditions should cause the
|
|
|
|
* instruction to fail, while others will cause the instruction to seem
|
|
|
|
* to succeed, but return an EXIT_REASON_INVALID_STATE.
|
|
|
|
* To speed up the normal (success) code path, we should avoid checking
|
|
|
|
* for misconfigurations which will anyway be caught by the processor
|
|
|
|
* when using the merged vmcs02.
|
|
|
|
*/
|
|
|
|
if (interrupt_shadow & KVM_X86_SHADOW_INT_MOV_SS)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_ENTRY_EVENTS_BLOCKED_BY_MOV_SS);
|
|
|
|
|
|
|
|
if (vmcs12->launch_state == launch)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
launch ? VMXERR_VMLAUNCH_NONCLEAR_VMCS
|
|
|
|
: VMXERR_VMRESUME_NONLAUNCHED_VMCS);
|
|
|
|
|
2019-04-12 15:19:57 +07:00
|
|
|
if (nested_vmx_check_controls(vcpu, vmcs12))
|
|
|
|
return nested_vmx_failValid(vcpu, VMXERR_ENTRY_INVALID_CONTROL_FIELD);
|
2019-04-12 02:18:06 +07:00
|
|
|
|
2019-04-12 15:19:57 +07:00
|
|
|
if (nested_vmx_check_host_state(vcpu, vmcs12))
|
|
|
|
return nested_vmx_failValid(vcpu, VMXERR_ENTRY_INVALID_HOST_STATE_FIELD);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We're finally done with prerequisite checking, and can start with
|
|
|
|
* the nested entry.
|
|
|
|
*/
|
|
|
|
vmx->nested.nested_run_pending = 1;
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
status = nested_vmx_enter_non_root_mode(vcpu, true);
|
|
|
|
if (unlikely(status != NVMX_VMENTRY_SUCCESS))
|
|
|
|
goto vmentry_failed;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Hide L1D cache contents from the nested guest. */
|
|
|
|
vmx->vcpu.arch.l1tf_flush_l1d = true;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Must happen outside of nested_vmx_enter_non_root_mode() as it will
|
|
|
|
* also be used as part of restoring nVMX state for
|
|
|
|
* snapshot restore (migration).
|
|
|
|
*
|
|
|
|
* In this flow, it is assumed that vmcs12 cache was
|
|
|
|
* trasferred as part of captured nVMX state and should
|
|
|
|
* therefore not be read from guest memory (which may not
|
|
|
|
* exist on destination host yet).
|
|
|
|
*/
|
|
|
|
nested_cache_shadow_vmcs12(vcpu, vmcs12);
|
|
|
|
|
|
|
|
/*
|
2018-11-27 02:22:32 +07:00
|
|
|
* If we're entering a halted L2 vcpu and the L2 vcpu won't be
|
|
|
|
* awakened by event injection or by an NMI-window VM-exit or
|
|
|
|
* by an interrupt-window VM-exit, halt the vcpu.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
|
|
|
if ((vmcs12->guest_activity_state == GUEST_ACTIVITY_HLT) &&
|
2018-11-27 02:22:32 +07:00
|
|
|
!(vmcs12->vm_entry_intr_info_field & INTR_INFO_VALID_MASK) &&
|
|
|
|
!(vmcs12->cpu_based_vm_exec_control & CPU_BASED_VIRTUAL_NMI_PENDING) &&
|
|
|
|
!((vmcs12->cpu_based_vm_exec_control & CPU_BASED_VIRTUAL_INTR_PENDING) &&
|
|
|
|
(vmcs12->guest_rflags & X86_EFLAGS_IF))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.nested_run_pending = 0;
|
|
|
|
return kvm_vcpu_halt(vcpu);
|
|
|
|
}
|
|
|
|
return 1;
|
KVM: nVMX: Don't leak L1 MMIO regions to L2
If the "virtualize APIC accesses" VM-execution control is set in the
VMCS, the APIC virtualization hardware is triggered when a page walk
in VMX non-root mode terminates at a PTE wherein the address of the 4k
page frame matches the APIC-access address specified in the VMCS. On
hardware, the APIC-access address may be any valid 4k-aligned physical
address.
KVM's nVMX implementation enforces the additional constraint that the
APIC-access address specified in the vmcs12 must be backed by
a "struct page" in L1. If not, L0 will simply clear the "virtualize
APIC accesses" VM-execution control in the vmcs02.
The problem with this approach is that the L1 guest has arranged the
vmcs12 EPT tables--or shadow page tables, if the "enable EPT"
VM-execution control is clear in the vmcs12--so that the L2 guest
physical address(es)--or L2 guest linear address(es)--that reference
the L2 APIC map to the APIC-access address specified in the
vmcs12. Without the "virtualize APIC accesses" VM-execution control in
the vmcs02, the APIC accesses in the L2 guest will directly access the
APIC-access page in L1.
When there is no mapping whatsoever for the APIC-access address in L1,
the L2 VM just loses the intended APIC virtualization. However, when
the APIC-access address is mapped to an MMIO region in L1, the L2
guest gets direct access to the L1 MMIO device. For example, if the
APIC-access address specified in the vmcs12 is 0xfee00000, then L2
gets direct access to L1's APIC.
Since this vmcs12 configuration is something that KVM cannot
faithfully emulate, the appropriate response is to exit to userspace
with KVM_INTERNAL_ERROR_EMULATION.
Fixes: fe3ef05c7572 ("KVM: nVMX: Prepare vmcs02 from vmcs01 and vmcs12")
Reported-by: Dan Cross <dcross@google.com>
Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Peter Shier <pshier@google.com>
Reviewed-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-10-16 00:44:05 +07:00
|
|
|
|
|
|
|
vmentry_failed:
|
|
|
|
vmx->nested.nested_run_pending = 0;
|
|
|
|
if (status == NVMX_VMENTRY_KVM_INTERNAL_ERROR)
|
|
|
|
return 0;
|
|
|
|
if (status == NVMX_VMENTRY_VMEXIT)
|
|
|
|
return 1;
|
|
|
|
WARN_ON_ONCE(status != NVMX_VMENTRY_VMFAIL);
|
|
|
|
return nested_vmx_failValid(vcpu, VMXERR_ENTRY_INVALID_CONTROL_FIELD);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* On a nested exit from L2 to L1, vmcs12.guest_cr0 might not be up-to-date
|
|
|
|
* because L2 may have changed some cr0 bits directly (CRO_GUEST_HOST_MASK).
|
|
|
|
* This function returns the new value we should put in vmcs12.guest_cr0.
|
|
|
|
* It's not enough to just return the vmcs02 GUEST_CR0. Rather,
|
|
|
|
* 1. Bits that neither L0 nor L1 trapped, were set directly by L2 and are now
|
|
|
|
* available in vmcs02 GUEST_CR0. (Note: It's enough to check that L0
|
|
|
|
* didn't trap the bit, because if L1 did, so would L0).
|
|
|
|
* 2. Bits that L1 asked to trap (and therefore L0 also did) could not have
|
|
|
|
* been modified by L2, and L1 knows it. So just leave the old value of
|
|
|
|
* the bit from vmcs12.guest_cr0. Note that the bit from vmcs02 GUEST_CR0
|
|
|
|
* isn't relevant, because if L0 traps this bit it can set it to anything.
|
|
|
|
* 3. Bits that L1 didn't trap, but L0 did. L1 believes the guest could have
|
|
|
|
* changed these bits, and therefore they need to be updated, but L0
|
|
|
|
* didn't necessarily allow them to be changed in GUEST_CR0 - and rather
|
|
|
|
* put them in vmcs02 CR0_READ_SHADOW. So take these bits from there.
|
|
|
|
*/
|
|
|
|
static inline unsigned long
|
|
|
|
vmcs12_guest_cr0(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
/*1*/ (vmcs_readl(GUEST_CR0) & vcpu->arch.cr0_guest_owned_bits) |
|
|
|
|
/*2*/ (vmcs12->guest_cr0 & vmcs12->cr0_guest_host_mask) |
|
|
|
|
/*3*/ (vmcs_readl(CR0_READ_SHADOW) & ~(vmcs12->cr0_guest_host_mask |
|
|
|
|
vcpu->arch.cr0_guest_owned_bits));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline unsigned long
|
|
|
|
vmcs12_guest_cr4(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
/*1*/ (vmcs_readl(GUEST_CR4) & vcpu->arch.cr4_guest_owned_bits) |
|
|
|
|
/*2*/ (vmcs12->guest_cr4 & vmcs12->cr4_guest_host_mask) |
|
|
|
|
/*3*/ (vmcs_readl(CR4_READ_SHADOW) & ~(vmcs12->cr4_guest_host_mask |
|
|
|
|
vcpu->arch.cr4_guest_owned_bits));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vmcs12_save_pending_event(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
u32 idt_vectoring;
|
|
|
|
unsigned int nr;
|
|
|
|
|
|
|
|
if (vcpu->arch.exception.injected) {
|
|
|
|
nr = vcpu->arch.exception.nr;
|
|
|
|
idt_vectoring = nr | VECTORING_INFO_VALID_MASK;
|
|
|
|
|
|
|
|
if (kvm_exception_is_soft(nr)) {
|
|
|
|
vmcs12->vm_exit_instruction_len =
|
|
|
|
vcpu->arch.event_exit_inst_len;
|
|
|
|
idt_vectoring |= INTR_TYPE_SOFT_EXCEPTION;
|
|
|
|
} else
|
|
|
|
idt_vectoring |= INTR_TYPE_HARD_EXCEPTION;
|
|
|
|
|
|
|
|
if (vcpu->arch.exception.has_error_code) {
|
|
|
|
idt_vectoring |= VECTORING_INFO_DELIVER_CODE_MASK;
|
|
|
|
vmcs12->idt_vectoring_error_code =
|
|
|
|
vcpu->arch.exception.error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs12->idt_vectoring_info_field = idt_vectoring;
|
|
|
|
} else if (vcpu->arch.nmi_injected) {
|
|
|
|
vmcs12->idt_vectoring_info_field =
|
|
|
|
INTR_TYPE_NMI_INTR | INTR_INFO_VALID_MASK | NMI_VECTOR;
|
|
|
|
} else if (vcpu->arch.interrupt.injected) {
|
|
|
|
nr = vcpu->arch.interrupt.nr;
|
|
|
|
idt_vectoring = nr | VECTORING_INFO_VALID_MASK;
|
|
|
|
|
|
|
|
if (vcpu->arch.interrupt.soft) {
|
|
|
|
idt_vectoring |= INTR_TYPE_SOFT_INTR;
|
|
|
|
vmcs12->vm_entry_instruction_len =
|
|
|
|
vcpu->arch.event_exit_inst_len;
|
|
|
|
} else
|
|
|
|
idt_vectoring |= INTR_TYPE_EXT_INTR;
|
|
|
|
|
|
|
|
vmcs12->idt_vectoring_info_field = idt_vectoring;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void nested_mark_vmcs12_pages_dirty(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
gfn_t gfn;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Don't need to mark the APIC access page dirty; it is never
|
|
|
|
* written to by the CPU during APIC virtualization.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (nested_cpu_has(vmcs12, CPU_BASED_TPR_SHADOW)) {
|
|
|
|
gfn = vmcs12->virtual_apic_page_addr >> PAGE_SHIFT;
|
|
|
|
kvm_vcpu_mark_page_dirty(vcpu, gfn);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has_posted_intr(vmcs12)) {
|
|
|
|
gfn = vmcs12->posted_intr_desc_addr >> PAGE_SHIFT;
|
|
|
|
kvm_vcpu_mark_page_dirty(vcpu, gfn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vmx_complete_nested_posted_interrupt(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
int max_irr;
|
|
|
|
void *vapic_page;
|
|
|
|
u16 status;
|
|
|
|
|
|
|
|
if (!vmx->nested.pi_desc || !vmx->nested.pi_pending)
|
|
|
|
return;
|
|
|
|
|
|
|
|
vmx->nested.pi_pending = false;
|
|
|
|
if (!pi_test_and_clear_on(vmx->nested.pi_desc))
|
|
|
|
return;
|
|
|
|
|
|
|
|
max_irr = find_last_bit((unsigned long *)vmx->nested.pi_desc->pir, 256);
|
|
|
|
if (max_irr != 256) {
|
2019-02-01 03:24:37 +07:00
|
|
|
vapic_page = vmx->nested.virtual_apic_map.hva;
|
|
|
|
if (!vapic_page)
|
|
|
|
return;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
__kvm_apic_update_irr(vmx->nested.pi_desc->pir,
|
|
|
|
vapic_page, &max_irr);
|
|
|
|
status = vmcs_read16(GUEST_INTR_STATUS);
|
|
|
|
if ((u8)max_irr > ((u8)status & 0xff)) {
|
|
|
|
status &= ~0xff;
|
|
|
|
status |= (u8)max_irr;
|
|
|
|
vmcs_write16(GUEST_INTR_STATUS, status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nested_mark_vmcs12_pages_dirty(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_vmx_inject_exception_vmexit(struct kvm_vcpu *vcpu,
|
|
|
|
unsigned long exit_qual)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
unsigned int nr = vcpu->arch.exception.nr;
|
|
|
|
u32 intr_info = nr | INTR_INFO_VALID_MASK;
|
|
|
|
|
|
|
|
if (vcpu->arch.exception.has_error_code) {
|
|
|
|
vmcs12->vm_exit_intr_error_code = vcpu->arch.exception.error_code;
|
|
|
|
intr_info |= INTR_INFO_DELIVER_CODE_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (kvm_exception_is_soft(nr))
|
|
|
|
intr_info |= INTR_TYPE_SOFT_EXCEPTION;
|
|
|
|
else
|
|
|
|
intr_info |= INTR_TYPE_HARD_EXCEPTION;
|
|
|
|
|
|
|
|
if (!(vmcs12->idt_vectoring_info_field & VECTORING_INFO_VALID_MASK) &&
|
|
|
|
vmx_get_nmi_mask(vcpu))
|
|
|
|
intr_info |= INTR_INFO_UNBLOCK_NMI;
|
|
|
|
|
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_EXCEPTION_NMI, intr_info, exit_qual);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_check_nested_events(struct kvm_vcpu *vcpu, bool external_intr)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
unsigned long exit_qual;
|
|
|
|
bool block_nested_events =
|
|
|
|
vmx->nested.nested_run_pending || kvm_event_needs_reinjection(vcpu);
|
2019-08-26 17:24:49 +07:00
|
|
|
struct kvm_lapic *apic = vcpu->arch.apic;
|
|
|
|
|
|
|
|
if (lapic_in_kernel(vcpu) &&
|
|
|
|
test_bit(KVM_APIC_INIT, &apic->pending_events)) {
|
|
|
|
if (block_nested_events)
|
|
|
|
return -EBUSY;
|
2019-11-11 19:16:05 +07:00
|
|
|
clear_bit(KVM_APIC_INIT, &apic->pending_events);
|
2019-08-26 17:24:49 +07:00
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_INIT_SIGNAL, 0, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vcpu->arch.exception.pending &&
|
|
|
|
nested_vmx_check_exception(vcpu, &exit_qual)) {
|
|
|
|
if (block_nested_events)
|
|
|
|
return -EBUSY;
|
|
|
|
nested_vmx_inject_exception_vmexit(vcpu, exit_qual);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_cpu_has_preemption_timer(get_vmcs12(vcpu)) &&
|
|
|
|
vmx->nested.preemption_timer_expired) {
|
|
|
|
if (block_nested_events)
|
|
|
|
return -EBUSY;
|
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_PREEMPTION_TIMER, 0, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vcpu->arch.nmi_pending && nested_exit_on_nmi(vcpu)) {
|
|
|
|
if (block_nested_events)
|
|
|
|
return -EBUSY;
|
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_EXCEPTION_NMI,
|
|
|
|
NMI_VECTOR | INTR_TYPE_NMI_INTR |
|
|
|
|
INTR_INFO_VALID_MASK, 0);
|
|
|
|
/*
|
|
|
|
* The NMI-triggered VM exit counts as injection:
|
|
|
|
* clear this one and block further NMIs.
|
|
|
|
*/
|
|
|
|
vcpu->arch.nmi_pending = 0;
|
|
|
|
vmx_set_nmi_mask(vcpu, true);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((kvm_cpu_has_interrupt(vcpu) || external_intr) &&
|
|
|
|
nested_exit_on_intr(vcpu)) {
|
|
|
|
if (block_nested_events)
|
|
|
|
return -EBUSY;
|
|
|
|
nested_vmx_vmexit(vcpu, EXIT_REASON_EXTERNAL_INTERRUPT, 0, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmx_complete_nested_posted_interrupt(vcpu);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32 vmx_get_preemption_timer_value(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
ktime_t remaining =
|
|
|
|
hrtimer_get_remaining(&to_vmx(vcpu)->nested.preemption_timer);
|
|
|
|
u64 value;
|
|
|
|
|
|
|
|
if (ktime_to_ns(remaining) <= 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
value = ktime_to_ns(remaining) * vcpu->arch.virtual_tsc_khz;
|
|
|
|
do_div(value, 1000000);
|
|
|
|
return value >> VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE;
|
|
|
|
}
|
|
|
|
|
2019-05-07 22:36:29 +07:00
|
|
|
static bool is_vmcs12_ext_field(unsigned long field)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
2019-05-07 22:36:29 +07:00
|
|
|
switch (field) {
|
|
|
|
case GUEST_ES_SELECTOR:
|
|
|
|
case GUEST_CS_SELECTOR:
|
|
|
|
case GUEST_SS_SELECTOR:
|
|
|
|
case GUEST_DS_SELECTOR:
|
|
|
|
case GUEST_FS_SELECTOR:
|
|
|
|
case GUEST_GS_SELECTOR:
|
|
|
|
case GUEST_LDTR_SELECTOR:
|
|
|
|
case GUEST_TR_SELECTOR:
|
|
|
|
case GUEST_ES_LIMIT:
|
|
|
|
case GUEST_CS_LIMIT:
|
|
|
|
case GUEST_SS_LIMIT:
|
|
|
|
case GUEST_DS_LIMIT:
|
|
|
|
case GUEST_FS_LIMIT:
|
|
|
|
case GUEST_GS_LIMIT:
|
|
|
|
case GUEST_LDTR_LIMIT:
|
|
|
|
case GUEST_TR_LIMIT:
|
|
|
|
case GUEST_GDTR_LIMIT:
|
|
|
|
case GUEST_IDTR_LIMIT:
|
|
|
|
case GUEST_ES_AR_BYTES:
|
|
|
|
case GUEST_DS_AR_BYTES:
|
|
|
|
case GUEST_FS_AR_BYTES:
|
|
|
|
case GUEST_GS_AR_BYTES:
|
|
|
|
case GUEST_LDTR_AR_BYTES:
|
|
|
|
case GUEST_TR_AR_BYTES:
|
|
|
|
case GUEST_ES_BASE:
|
|
|
|
case GUEST_CS_BASE:
|
|
|
|
case GUEST_SS_BASE:
|
|
|
|
case GUEST_DS_BASE:
|
|
|
|
case GUEST_FS_BASE:
|
|
|
|
case GUEST_GS_BASE:
|
|
|
|
case GUEST_LDTR_BASE:
|
|
|
|
case GUEST_TR_BASE:
|
|
|
|
case GUEST_GDTR_BASE:
|
|
|
|
case GUEST_IDTR_BASE:
|
|
|
|
case GUEST_PENDING_DBG_EXCEPTIONS:
|
|
|
|
case GUEST_BNDCFGS:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-05-07 22:36:29 +07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sync_vmcs02_to_vmcs12_rare(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
vmcs12->guest_es_selector = vmcs_read16(GUEST_ES_SELECTOR);
|
|
|
|
vmcs12->guest_cs_selector = vmcs_read16(GUEST_CS_SELECTOR);
|
|
|
|
vmcs12->guest_ss_selector = vmcs_read16(GUEST_SS_SELECTOR);
|
|
|
|
vmcs12->guest_ds_selector = vmcs_read16(GUEST_DS_SELECTOR);
|
|
|
|
vmcs12->guest_fs_selector = vmcs_read16(GUEST_FS_SELECTOR);
|
|
|
|
vmcs12->guest_gs_selector = vmcs_read16(GUEST_GS_SELECTOR);
|
|
|
|
vmcs12->guest_ldtr_selector = vmcs_read16(GUEST_LDTR_SELECTOR);
|
|
|
|
vmcs12->guest_tr_selector = vmcs_read16(GUEST_TR_SELECTOR);
|
|
|
|
vmcs12->guest_es_limit = vmcs_read32(GUEST_ES_LIMIT);
|
|
|
|
vmcs12->guest_cs_limit = vmcs_read32(GUEST_CS_LIMIT);
|
|
|
|
vmcs12->guest_ss_limit = vmcs_read32(GUEST_SS_LIMIT);
|
|
|
|
vmcs12->guest_ds_limit = vmcs_read32(GUEST_DS_LIMIT);
|
|
|
|
vmcs12->guest_fs_limit = vmcs_read32(GUEST_FS_LIMIT);
|
|
|
|
vmcs12->guest_gs_limit = vmcs_read32(GUEST_GS_LIMIT);
|
|
|
|
vmcs12->guest_ldtr_limit = vmcs_read32(GUEST_LDTR_LIMIT);
|
|
|
|
vmcs12->guest_tr_limit = vmcs_read32(GUEST_TR_LIMIT);
|
|
|
|
vmcs12->guest_gdtr_limit = vmcs_read32(GUEST_GDTR_LIMIT);
|
|
|
|
vmcs12->guest_idtr_limit = vmcs_read32(GUEST_IDTR_LIMIT);
|
|
|
|
vmcs12->guest_es_ar_bytes = vmcs_read32(GUEST_ES_AR_BYTES);
|
|
|
|
vmcs12->guest_ds_ar_bytes = vmcs_read32(GUEST_DS_AR_BYTES);
|
|
|
|
vmcs12->guest_fs_ar_bytes = vmcs_read32(GUEST_FS_AR_BYTES);
|
|
|
|
vmcs12->guest_gs_ar_bytes = vmcs_read32(GUEST_GS_AR_BYTES);
|
|
|
|
vmcs12->guest_ldtr_ar_bytes = vmcs_read32(GUEST_LDTR_AR_BYTES);
|
|
|
|
vmcs12->guest_tr_ar_bytes = vmcs_read32(GUEST_TR_AR_BYTES);
|
|
|
|
vmcs12->guest_es_base = vmcs_readl(GUEST_ES_BASE);
|
|
|
|
vmcs12->guest_cs_base = vmcs_readl(GUEST_CS_BASE);
|
|
|
|
vmcs12->guest_ss_base = vmcs_readl(GUEST_SS_BASE);
|
|
|
|
vmcs12->guest_ds_base = vmcs_readl(GUEST_DS_BASE);
|
|
|
|
vmcs12->guest_fs_base = vmcs_readl(GUEST_FS_BASE);
|
|
|
|
vmcs12->guest_gs_base = vmcs_readl(GUEST_GS_BASE);
|
|
|
|
vmcs12->guest_ldtr_base = vmcs_readl(GUEST_LDTR_BASE);
|
|
|
|
vmcs12->guest_tr_base = vmcs_readl(GUEST_TR_BASE);
|
|
|
|
vmcs12->guest_gdtr_base = vmcs_readl(GUEST_GDTR_BASE);
|
|
|
|
vmcs12->guest_idtr_base = vmcs_readl(GUEST_IDTR_BASE);
|
2019-05-07 22:36:29 +07:00
|
|
|
vmcs12->guest_pending_dbg_exceptions =
|
|
|
|
vmcs_readl(GUEST_PENDING_DBG_EXCEPTIONS);
|
|
|
|
if (kvm_mpx_supported())
|
|
|
|
vmcs12->guest_bndcfgs = vmcs_read64(GUEST_BNDCFGS);
|
|
|
|
|
|
|
|
vmx->nested.need_sync_vmcs02_to_vmcs12_rare = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void copy_vmcs02_to_vmcs12_rare(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
int cpu;
|
|
|
|
|
|
|
|
if (!vmx->nested.need_sync_vmcs02_to_vmcs12_rare)
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
WARN_ON_ONCE(vmx->loaded_vmcs != &vmx->vmcs01);
|
|
|
|
|
|
|
|
cpu = get_cpu();
|
|
|
|
vmx->loaded_vmcs = &vmx->nested.vmcs02;
|
|
|
|
vmx_vcpu_load(&vmx->vcpu, cpu);
|
|
|
|
|
|
|
|
sync_vmcs02_to_vmcs12_rare(vcpu, vmcs12);
|
|
|
|
|
|
|
|
vmx->loaded_vmcs = &vmx->vmcs01;
|
|
|
|
vmx_vcpu_load(&vmx->vcpu, cpu);
|
|
|
|
put_cpu();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update the guest state fields of vmcs12 to reflect changes that
|
|
|
|
* occurred while L2 was running. (The "IA-32e mode guest" bit of the
|
|
|
|
* VM-entry controls is also updated, since this is really a guest
|
|
|
|
* state bit.)
|
|
|
|
*/
|
|
|
|
static void sync_vmcs02_to_vmcs12(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
if (vmx->nested.hv_evmcs)
|
|
|
|
sync_vmcs02_to_vmcs12_rare(vcpu, vmcs12);
|
|
|
|
|
|
|
|
vmx->nested.need_sync_vmcs02_to_vmcs12_rare = !vmx->nested.hv_evmcs;
|
|
|
|
|
|
|
|
vmcs12->guest_cr0 = vmcs12_guest_cr0(vcpu, vmcs12);
|
|
|
|
vmcs12->guest_cr4 = vmcs12_guest_cr4(vcpu, vmcs12);
|
|
|
|
|
|
|
|
vmcs12->guest_rsp = kvm_rsp_read(vcpu);
|
|
|
|
vmcs12->guest_rip = kvm_rip_read(vcpu);
|
|
|
|
vmcs12->guest_rflags = vmcs_readl(GUEST_RFLAGS);
|
|
|
|
|
|
|
|
vmcs12->guest_cs_ar_bytes = vmcs_read32(GUEST_CS_AR_BYTES);
|
|
|
|
vmcs12->guest_ss_ar_bytes = vmcs_read32(GUEST_SS_AR_BYTES);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-05-07 23:06:36 +07:00
|
|
|
vmcs12->guest_sysenter_cs = vmcs_read32(GUEST_SYSENTER_CS);
|
|
|
|
vmcs12->guest_sysenter_esp = vmcs_readl(GUEST_SYSENTER_ESP);
|
|
|
|
vmcs12->guest_sysenter_eip = vmcs_readl(GUEST_SYSENTER_EIP);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
vmcs12->guest_interruptibility_info =
|
|
|
|
vmcs_read32(GUEST_INTERRUPTIBILITY_INFO);
|
2019-05-07 22:36:29 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (vcpu->arch.mp_state == KVM_MP_STATE_HALTED)
|
|
|
|
vmcs12->guest_activity_state = GUEST_ACTIVITY_HLT;
|
|
|
|
else
|
|
|
|
vmcs12->guest_activity_state = GUEST_ACTIVITY_ACTIVE;
|
|
|
|
|
2019-01-30 01:12:35 +07:00
|
|
|
if (nested_cpu_has_preemption_timer(vmcs12) &&
|
|
|
|
vmcs12->vm_exit_controls & VM_EXIT_SAVE_VMX_PREEMPTION_TIMER)
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs12->vmx_preemption_timer_value =
|
|
|
|
vmx_get_preemption_timer_value(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In some cases (usually, nested EPT), L2 is allowed to change its
|
|
|
|
* own CR3 without exiting. If it has changed it, we must keep it.
|
|
|
|
* Of course, if L0 is using shadow page tables, GUEST_CR3 was defined
|
|
|
|
* by L0, not L1 or L2, so we mustn't unconditionally copy it to vmcs12.
|
|
|
|
*
|
|
|
|
* Additionally, restore L2's PDPTR to vmcs12.
|
|
|
|
*/
|
|
|
|
if (enable_ept) {
|
|
|
|
vmcs12->guest_cr3 = vmcs_readl(GUEST_CR3);
|
2019-05-07 23:06:40 +07:00
|
|
|
if (nested_cpu_has_ept(vmcs12) && is_pae_paging(vcpu)) {
|
|
|
|
vmcs12->guest_pdptr0 = vmcs_read64(GUEST_PDPTR0);
|
|
|
|
vmcs12->guest_pdptr1 = vmcs_read64(GUEST_PDPTR1);
|
|
|
|
vmcs12->guest_pdptr2 = vmcs_read64(GUEST_PDPTR2);
|
|
|
|
vmcs12->guest_pdptr3 = vmcs_read64(GUEST_PDPTR3);
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
vmcs12->guest_linear_address = vmcs_readl(GUEST_LINEAR_ADDRESS);
|
|
|
|
|
|
|
|
if (nested_cpu_has_vid(vmcs12))
|
|
|
|
vmcs12->guest_intr_status = vmcs_read16(GUEST_INTR_STATUS);
|
|
|
|
|
|
|
|
vmcs12->vm_entry_controls =
|
|
|
|
(vmcs12->vm_entry_controls & ~VM_ENTRY_IA32E_MODE) |
|
|
|
|
(vm_entry_controls_get(to_vmx(vcpu)) & VM_ENTRY_IA32E_MODE);
|
|
|
|
|
2019-05-07 23:06:37 +07:00
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_SAVE_DEBUG_CONTROLS)
|
2018-12-04 04:53:18 +07:00
|
|
|
kvm_get_dr(vcpu, 7, (unsigned long *)&vmcs12->guest_dr7);
|
|
|
|
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_SAVE_IA32_EFER)
|
|
|
|
vmcs12->guest_ia32_efer = vcpu->arch.efer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* prepare_vmcs12 is part of what we need to do when the nested L2 guest exits
|
|
|
|
* and we want to prepare to run its L1 parent. L1 keeps a vmcs for L2 (vmcs12),
|
|
|
|
* and this function updates it to reflect the changes to the guest state while
|
|
|
|
* L2 was running (and perhaps made some exits which were handled directly by L0
|
|
|
|
* without going back to L1), and to reflect the exit reason.
|
|
|
|
* Note that we do not have to copy here all VMCS fields, just those that
|
|
|
|
* could have changed by the L2 guest or the exit - i.e., the guest-state and
|
|
|
|
* exit-information fields only. Other fields are modified by L1 with VMWRITE,
|
|
|
|
* which already writes to vmcs12 directly.
|
|
|
|
*/
|
|
|
|
static void prepare_vmcs12(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12,
|
|
|
|
u32 exit_reason, u32 exit_intr_info,
|
|
|
|
unsigned long exit_qualification)
|
|
|
|
{
|
|
|
|
/* update exit information fields: */
|
|
|
|
vmcs12->vm_exit_reason = exit_reason;
|
|
|
|
vmcs12->exit_qualification = exit_qualification;
|
|
|
|
vmcs12->vm_exit_intr_info = exit_intr_info;
|
|
|
|
|
|
|
|
vmcs12->idt_vectoring_info_field = 0;
|
|
|
|
vmcs12->vm_exit_instruction_len = vmcs_read32(VM_EXIT_INSTRUCTION_LEN);
|
|
|
|
vmcs12->vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
|
|
|
|
if (!(vmcs12->vm_exit_reason & VMX_EXIT_REASONS_FAILED_VMENTRY)) {
|
|
|
|
vmcs12->launch_state = 1;
|
|
|
|
|
|
|
|
/* vm_entry_intr_info_field is cleared on exit. Emulate this
|
|
|
|
* instead of reading the real value. */
|
|
|
|
vmcs12->vm_entry_intr_info_field &= ~INTR_INFO_VALID_MASK;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Transfer the event that L0 or L1 may wanted to inject into
|
|
|
|
* L2 to IDT_VECTORING_INFO_FIELD.
|
|
|
|
*/
|
|
|
|
vmcs12_save_pending_event(vcpu, vmcs12);
|
2018-12-05 07:00:13 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* According to spec, there's no need to store the guest's
|
|
|
|
* MSRs if the exit is due to a VM-entry failure that occurs
|
|
|
|
* during or after loading the guest state. Since this exit
|
|
|
|
* does not fall in that category, we need to save the MSRs.
|
|
|
|
*/
|
|
|
|
if (nested_vmx_store_msr(vcpu,
|
|
|
|
vmcs12->vm_exit_msr_store_addr,
|
|
|
|
vmcs12->vm_exit_msr_store_count))
|
|
|
|
nested_vmx_abort(vcpu,
|
|
|
|
VMX_ABORT_SAVE_GUEST_MSR_FAIL);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Drop what we picked up for L2 via vmx_complete_interrupts. It is
|
|
|
|
* preserved above and would only end up incorrectly in L1.
|
|
|
|
*/
|
|
|
|
vcpu->arch.nmi_injected = false;
|
|
|
|
kvm_clear_exception_queue(vcpu);
|
|
|
|
kvm_clear_interrupt_queue(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A part of what we need to when the nested L2 guest exits and we want to
|
|
|
|
* run its L1 parent, is to reset L1's guest state to the host state specified
|
|
|
|
* in vmcs12.
|
|
|
|
* This function is to be called not only on normal nested exit, but also on
|
|
|
|
* a nested entry failure, as explained in Intel's spec, 3B.23.7 ("VM-Entry
|
|
|
|
* Failures During or After Loading Guest State").
|
|
|
|
* This function should be called when the active VMCS is L1's (vmcs01).
|
|
|
|
*/
|
|
|
|
static void load_vmcs12_host_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
struct kvm_segment seg;
|
|
|
|
u32 entry_failure_code;
|
|
|
|
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_EFER)
|
|
|
|
vcpu->arch.efer = vmcs12->host_ia32_efer;
|
|
|
|
else if (vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE)
|
|
|
|
vcpu->arch.efer |= (EFER_LMA | EFER_LME);
|
|
|
|
else
|
|
|
|
vcpu->arch.efer &= ~(EFER_LMA | EFER_LME);
|
|
|
|
vmx_set_efer(vcpu, vcpu->arch.efer);
|
|
|
|
|
2019-05-01 03:07:26 +07:00
|
|
|
kvm_rsp_write(vcpu, vmcs12->host_rsp);
|
|
|
|
kvm_rip_write(vcpu, vmcs12->host_rip);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx_set_rflags(vcpu, X86_EFLAGS_FIXED);
|
|
|
|
vmx_set_interrupt_shadow(vcpu, 0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note that calling vmx_set_cr0 is important, even if cr0 hasn't
|
|
|
|
* actually changed, because vmx_set_cr0 refers to efer set above.
|
|
|
|
*
|
|
|
|
* CR0_GUEST_HOST_MASK is already set in the original vmcs01
|
|
|
|
* (KVM doesn't change it);
|
|
|
|
*/
|
|
|
|
vcpu->arch.cr0_guest_owned_bits = X86_CR0_TS;
|
|
|
|
vmx_set_cr0(vcpu, vmcs12->host_cr0);
|
|
|
|
|
|
|
|
/* Same as above - no reason to call set_cr4_guest_host_mask(). */
|
|
|
|
vcpu->arch.cr4_guest_owned_bits = ~vmcs_readl(CR4_GUEST_HOST_MASK);
|
|
|
|
vmx_set_cr4(vcpu, vmcs12->host_cr4);
|
|
|
|
|
|
|
|
nested_ept_uninit_mmu_context(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Only PDPTE load can fail as the value of cr3 was checked on entry and
|
|
|
|
* couldn't have changed.
|
|
|
|
*/
|
|
|
|
if (nested_vmx_load_cr3(vcpu, vmcs12->host_cr3, false, &entry_failure_code))
|
|
|
|
nested_vmx_abort(vcpu, VMX_ABORT_LOAD_HOST_PDPTE_FAIL);
|
|
|
|
|
|
|
|
if (!enable_ept)
|
|
|
|
vcpu->arch.walk_mmu->inject_page_fault = kvm_inject_page_fault;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If vmcs01 doesn't use VPID, CPU flushes TLB on every
|
|
|
|
* VMEntry/VMExit. Thus, no need to flush TLB.
|
|
|
|
*
|
|
|
|
* If vmcs12 doesn't use VPID, L1 expects TLB to be
|
|
|
|
* flushed on every VMEntry/VMExit.
|
|
|
|
*
|
|
|
|
* Otherwise, we can preserve TLB entries as long as we are
|
|
|
|
* able to tag L1 TLB entries differently than L2 TLB entries.
|
|
|
|
*
|
|
|
|
* If vmcs12 uses EPT, we need to execute this flush on EPTP01
|
|
|
|
* and therefore we request the TLB flush to happen only after VMCS EPTP
|
|
|
|
* has been set by KVM_REQ_LOAD_CR3.
|
|
|
|
*/
|
|
|
|
if (enable_vpid &&
|
|
|
|
(!nested_cpu_has_vpid(vmcs12) || !nested_has_guest_tlb_tag(vcpu))) {
|
|
|
|
kvm_make_request(KVM_REQ_TLB_FLUSH, vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs_write32(GUEST_SYSENTER_CS, vmcs12->host_ia32_sysenter_cs);
|
|
|
|
vmcs_writel(GUEST_SYSENTER_ESP, vmcs12->host_ia32_sysenter_esp);
|
|
|
|
vmcs_writel(GUEST_SYSENTER_EIP, vmcs12->host_ia32_sysenter_eip);
|
|
|
|
vmcs_writel(GUEST_IDTR_BASE, vmcs12->host_idtr_base);
|
|
|
|
vmcs_writel(GUEST_GDTR_BASE, vmcs12->host_gdtr_base);
|
|
|
|
vmcs_write32(GUEST_IDTR_LIMIT, 0xFFFF);
|
|
|
|
vmcs_write32(GUEST_GDTR_LIMIT, 0xFFFF);
|
|
|
|
|
|
|
|
/* If not VM_EXIT_CLEAR_BNDCFGS, the L2 value propagates to L1. */
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_CLEAR_BNDCFGS)
|
|
|
|
vmcs_write64(GUEST_BNDCFGS, 0);
|
|
|
|
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_PAT) {
|
|
|
|
vmcs_write64(GUEST_IA32_PAT, vmcs12->host_ia32_pat);
|
|
|
|
vcpu->arch.pat = vmcs12->host_ia32_pat;
|
|
|
|
}
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_LOAD_IA32_PERF_GLOBAL_CTRL)
|
2019-11-14 07:17:18 +07:00
|
|
|
SET_MSR_OR_WARN(vcpu, MSR_CORE_PERF_GLOBAL_CTRL,
|
|
|
|
vmcs12->host_ia32_perf_global_ctrl);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Set L1 segment info according to Intel SDM
|
|
|
|
27.5.2 Loading Host Segment and Descriptor-Table Registers */
|
|
|
|
seg = (struct kvm_segment) {
|
|
|
|
.base = 0,
|
|
|
|
.limit = 0xFFFFFFFF,
|
|
|
|
.selector = vmcs12->host_cs_selector,
|
|
|
|
.type = 11,
|
|
|
|
.present = 1,
|
|
|
|
.s = 1,
|
|
|
|
.g = 1
|
|
|
|
};
|
|
|
|
if (vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE)
|
|
|
|
seg.l = 1;
|
|
|
|
else
|
|
|
|
seg.db = 1;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_CS);
|
|
|
|
seg = (struct kvm_segment) {
|
|
|
|
.base = 0,
|
|
|
|
.limit = 0xFFFFFFFF,
|
|
|
|
.type = 3,
|
|
|
|
.present = 1,
|
|
|
|
.s = 1,
|
|
|
|
.db = 1,
|
|
|
|
.g = 1
|
|
|
|
};
|
|
|
|
seg.selector = vmcs12->host_ds_selector;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_DS);
|
|
|
|
seg.selector = vmcs12->host_es_selector;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_ES);
|
|
|
|
seg.selector = vmcs12->host_ss_selector;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_SS);
|
|
|
|
seg.selector = vmcs12->host_fs_selector;
|
|
|
|
seg.base = vmcs12->host_fs_base;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_FS);
|
|
|
|
seg.selector = vmcs12->host_gs_selector;
|
|
|
|
seg.base = vmcs12->host_gs_base;
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_GS);
|
|
|
|
seg = (struct kvm_segment) {
|
|
|
|
.base = vmcs12->host_tr_base,
|
|
|
|
.limit = 0x67,
|
|
|
|
.selector = vmcs12->host_tr_selector,
|
|
|
|
.type = 11,
|
|
|
|
.present = 1
|
|
|
|
};
|
|
|
|
vmx_set_segment(vcpu, &seg, VCPU_SREG_TR);
|
|
|
|
|
|
|
|
kvm_set_dr(vcpu, 7, 0x400);
|
|
|
|
vmcs_write64(GUEST_IA32_DEBUGCTL, 0);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_msr_bitmap())
|
|
|
|
vmx_update_msr_bitmap(vcpu);
|
|
|
|
|
|
|
|
if (nested_vmx_load_msr(vcpu, vmcs12->vm_exit_msr_load_addr,
|
|
|
|
vmcs12->vm_exit_msr_load_count))
|
|
|
|
nested_vmx_abort(vcpu, VMX_ABORT_LOAD_HOST_MSR_FAIL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u64 nested_vmx_get_vmcs01_guest_efer(struct vcpu_vmx *vmx)
|
|
|
|
{
|
|
|
|
struct shared_msr_entry *efer_msr;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
if (vm_entry_controls_get(vmx) & VM_ENTRY_LOAD_IA32_EFER)
|
|
|
|
return vmcs_read64(GUEST_IA32_EFER);
|
|
|
|
|
|
|
|
if (cpu_has_load_ia32_efer())
|
|
|
|
return host_efer;
|
|
|
|
|
|
|
|
for (i = 0; i < vmx->msr_autoload.guest.nr; ++i) {
|
|
|
|
if (vmx->msr_autoload.guest.val[i].index == MSR_EFER)
|
|
|
|
return vmx->msr_autoload.guest.val[i].value;
|
|
|
|
}
|
|
|
|
|
|
|
|
efer_msr = find_msr_entry(vmx, MSR_EFER);
|
|
|
|
if (efer_msr)
|
|
|
|
return efer_msr->data;
|
|
|
|
|
|
|
|
return host_efer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nested_vmx_restore_host_state(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmx_msr_entry g, h;
|
|
|
|
gpa_t gpa;
|
|
|
|
u32 i, j;
|
|
|
|
|
|
|
|
vcpu->arch.pat = vmcs_read64(GUEST_IA32_PAT);
|
|
|
|
|
|
|
|
if (vmcs12->vm_entry_controls & VM_ENTRY_LOAD_DEBUG_CONTROLS) {
|
|
|
|
/*
|
|
|
|
* L1's host DR7 is lost if KVM_GUESTDBG_USE_HW_BP is set
|
|
|
|
* as vmcs01.GUEST_DR7 contains a userspace defined value
|
|
|
|
* and vcpu->arch.dr7 is not squirreled away before the
|
|
|
|
* nested VMENTER (not worth adding a variable in nested_vmx).
|
|
|
|
*/
|
|
|
|
if (vcpu->guest_debug & KVM_GUESTDBG_USE_HW_BP)
|
|
|
|
kvm_set_dr(vcpu, 7, DR7_FIXED_1);
|
|
|
|
else
|
|
|
|
WARN_ON(kvm_set_dr(vcpu, 7, vmcs_readl(GUEST_DR7)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note that calling vmx_set_{efer,cr0,cr4} is important as they
|
|
|
|
* handle a variety of side effects to KVM's software model.
|
|
|
|
*/
|
|
|
|
vmx_set_efer(vcpu, nested_vmx_get_vmcs01_guest_efer(vmx));
|
|
|
|
|
|
|
|
vcpu->arch.cr0_guest_owned_bits = X86_CR0_TS;
|
|
|
|
vmx_set_cr0(vcpu, vmcs_readl(CR0_READ_SHADOW));
|
|
|
|
|
|
|
|
vcpu->arch.cr4_guest_owned_bits = ~vmcs_readl(CR4_GUEST_HOST_MASK);
|
|
|
|
vmx_set_cr4(vcpu, vmcs_readl(CR4_READ_SHADOW));
|
|
|
|
|
|
|
|
nested_ept_uninit_mmu_context(vcpu);
|
KVM: nVMX: Stash L1's CR3 in vmcs01.GUEST_CR3 on nested entry w/o EPT
KVM does not have 100% coverage of VMX consistency checks, i.e. some
checks that cause VM-Fail may only be detected by hardware during a
nested VM-Entry. In such a case, KVM must restore L1's state to the
pre-VM-Enter state as L2's state has already been loaded into KVM's
software model.
L1's CR3 and PDPTRs in particular are loaded from vmcs01.GUEST_*. But
when EPT is disabled, the associated fields hold KVM's shadow values,
not L1's "real" values. Fortunately, when EPT is disabled the PDPTRs
come from memory, i.e. are not cached in the VMCS. Which leaves CR3
as the sole anomaly.
A previously applied workaround to handle CR3 was to force nested early
checks if EPT is disabled:
commit 2b27924bb1d48 ("KVM: nVMX: always use early vmcs check when EPT
is disabled")
Forcing nested early checks is undesirable as doing so adds hundreds of
cycles to every nested VM-Entry. Rather than take this performance hit,
handle CR3 by overwriting vmcs01.GUEST_CR3 with L1's CR3 during nested
VM-Entry when EPT is disabled *and* nested early checks are disabled.
By stuffing vmcs01.GUEST_CR3, nested_vmx_restore_host_state() will
naturally restore the correct vcpu->arch.cr3 from vmcs01.GUEST_CR3.
These shenanigans work because nested_vmx_restore_host_state() does a
full kvm_mmu_reset_context(), i.e. unloads the current MMU, which
guarantees vmcs01.GUEST_CR3 will be rewritten with a new shadow CR3
prior to re-entering L1.
vcpu->arch.root_mmu.root_hpa is set to INVALID_PAGE via:
nested_vmx_restore_host_state() ->
kvm_mmu_reset_context() ->
kvm_mmu_unload() ->
kvm_mmu_free_roots()
kvm_mmu_unload() has WARN_ON(root_hpa != INVALID_PAGE), i.e. we can bank
on 'root_hpa == INVALID_PAGE' unless the implementation of
kvm_mmu_reset_context() is changed.
On the way into L1, VMCS.GUEST_CR3 is guaranteed to be written (on a
successful entry) via:
vcpu_enter_guest() ->
kvm_mmu_reload() ->
kvm_mmu_load() ->
kvm_mmu_load_cr3() ->
vmx_set_cr3()
Stuff vmcs01.GUEST_CR3 if and only if nested early checks are disabled
as a "late" VM-Fail should never happen win that case (KVM WARNs), and
the conditional write avoids the need to restore the correct GUEST_CR3
when nested_vmx_check_vmentry_hw() fails.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Message-Id: <20190607185534.24368-1-sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-08 01:55:34 +07:00
|
|
|
vcpu->arch.cr3 = vmcs_readl(GUEST_CR3);
|
2019-09-28 04:45:22 +07:00
|
|
|
kvm_register_mark_available(vcpu, VCPU_EXREG_CR3);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Use ept_save_pdptrs(vcpu) to load the MMU's cached PDPTRs
|
|
|
|
* from vmcs01 (if necessary). The PDPTRs are not loaded on
|
|
|
|
* VMFail, like everything else we just need to ensure our
|
|
|
|
* software model is up-to-date.
|
|
|
|
*/
|
KVM: nVMX: Stash L1's CR3 in vmcs01.GUEST_CR3 on nested entry w/o EPT
KVM does not have 100% coverage of VMX consistency checks, i.e. some
checks that cause VM-Fail may only be detected by hardware during a
nested VM-Entry. In such a case, KVM must restore L1's state to the
pre-VM-Enter state as L2's state has already been loaded into KVM's
software model.
L1's CR3 and PDPTRs in particular are loaded from vmcs01.GUEST_*. But
when EPT is disabled, the associated fields hold KVM's shadow values,
not L1's "real" values. Fortunately, when EPT is disabled the PDPTRs
come from memory, i.e. are not cached in the VMCS. Which leaves CR3
as the sole anomaly.
A previously applied workaround to handle CR3 was to force nested early
checks if EPT is disabled:
commit 2b27924bb1d48 ("KVM: nVMX: always use early vmcs check when EPT
is disabled")
Forcing nested early checks is undesirable as doing so adds hundreds of
cycles to every nested VM-Entry. Rather than take this performance hit,
handle CR3 by overwriting vmcs01.GUEST_CR3 with L1's CR3 during nested
VM-Entry when EPT is disabled *and* nested early checks are disabled.
By stuffing vmcs01.GUEST_CR3, nested_vmx_restore_host_state() will
naturally restore the correct vcpu->arch.cr3 from vmcs01.GUEST_CR3.
These shenanigans work because nested_vmx_restore_host_state() does a
full kvm_mmu_reset_context(), i.e. unloads the current MMU, which
guarantees vmcs01.GUEST_CR3 will be rewritten with a new shadow CR3
prior to re-entering L1.
vcpu->arch.root_mmu.root_hpa is set to INVALID_PAGE via:
nested_vmx_restore_host_state() ->
kvm_mmu_reset_context() ->
kvm_mmu_unload() ->
kvm_mmu_free_roots()
kvm_mmu_unload() has WARN_ON(root_hpa != INVALID_PAGE), i.e. we can bank
on 'root_hpa == INVALID_PAGE' unless the implementation of
kvm_mmu_reset_context() is changed.
On the way into L1, VMCS.GUEST_CR3 is guaranteed to be written (on a
successful entry) via:
vcpu_enter_guest() ->
kvm_mmu_reload() ->
kvm_mmu_load() ->
kvm_mmu_load_cr3() ->
vmx_set_cr3()
Stuff vmcs01.GUEST_CR3 if and only if nested early checks are disabled
as a "late" VM-Fail should never happen win that case (KVM WARNs), and
the conditional write avoids the need to restore the correct GUEST_CR3
when nested_vmx_check_vmentry_hw() fails.
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Message-Id: <20190607185534.24368-1-sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-08 01:55:34 +07:00
|
|
|
if (enable_ept)
|
|
|
|
ept_save_pdptrs(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
kvm_mmu_reset_context(vcpu);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_msr_bitmap())
|
|
|
|
vmx_update_msr_bitmap(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This nasty bit of open coding is a compromise between blindly
|
|
|
|
* loading L1's MSRs using the exit load lists (incorrect emulation
|
|
|
|
* of VMFail), leaving the nested VM's MSRs in the software model
|
|
|
|
* (incorrect behavior) and snapshotting the modified MSRs (too
|
|
|
|
* expensive since the lists are unbound by hardware). For each
|
|
|
|
* MSR that was (prematurely) loaded from the nested VMEntry load
|
|
|
|
* list, reload it from the exit load list if it exists and differs
|
|
|
|
* from the guest value. The intent is to stuff host state as
|
|
|
|
* silently as possible, not to fully process the exit load list.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < vmcs12->vm_entry_msr_load_count; i++) {
|
|
|
|
gpa = vmcs12->vm_entry_msr_load_addr + (i * sizeof(g));
|
|
|
|
if (kvm_vcpu_read_guest(vcpu, gpa, &g, sizeof(g))) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s read MSR index failed (%u, 0x%08llx)\n",
|
|
|
|
__func__, i, gpa);
|
|
|
|
goto vmabort;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (j = 0; j < vmcs12->vm_exit_msr_load_count; j++) {
|
|
|
|
gpa = vmcs12->vm_exit_msr_load_addr + (j * sizeof(h));
|
|
|
|
if (kvm_vcpu_read_guest(vcpu, gpa, &h, sizeof(h))) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s read MSR failed (%u, 0x%08llx)\n",
|
|
|
|
__func__, j, gpa);
|
|
|
|
goto vmabort;
|
|
|
|
}
|
|
|
|
if (h.index != g.index)
|
|
|
|
continue;
|
|
|
|
if (h.value == g.value)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (nested_vmx_load_msr_check(vcpu, &h)) {
|
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s check failed (%u, 0x%x, 0x%x)\n",
|
|
|
|
__func__, j, h.index, h.reserved);
|
|
|
|
goto vmabort;
|
|
|
|
}
|
|
|
|
|
2019-09-06 04:22:54 +07:00
|
|
|
if (kvm_set_msr(vcpu, h.index, h.value)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
pr_debug_ratelimited(
|
|
|
|
"%s WRMSR failed (%u, 0x%x, 0x%llx)\n",
|
|
|
|
__func__, j, h.index, h.value);
|
|
|
|
goto vmabort;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
vmabort:
|
|
|
|
nested_vmx_abort(vcpu, VMX_ABORT_LOAD_HOST_MSR_FAIL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Emulate an exit from nested guest (L2) to L1, i.e., prepare to run L1
|
|
|
|
* and modify vmcs12 to make it see what it would expect to see there if
|
|
|
|
* L2 was its real guest. Must only be called when in L2 (is_guest_mode())
|
|
|
|
*/
|
|
|
|
void nested_vmx_vmexit(struct kvm_vcpu *vcpu, u32 exit_reason,
|
|
|
|
u32 exit_intr_info, unsigned long exit_qualification)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
|
|
|
/* trying to cancel vmlaunch/vmresume is a bug */
|
|
|
|
WARN_ON_ONCE(vmx->nested.nested_run_pending);
|
|
|
|
|
|
|
|
leave_guest_mode(vcpu);
|
|
|
|
|
2019-01-30 01:12:35 +07:00
|
|
|
if (nested_cpu_has_preemption_timer(vmcs12))
|
|
|
|
hrtimer_cancel(&to_vmx(vcpu)->nested.preemption_timer);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING)
|
|
|
|
vcpu->arch.tsc_offset -= vmcs12->tsc_offset;
|
|
|
|
|
|
|
|
if (likely(!vmx->fail)) {
|
2019-05-07 22:36:27 +07:00
|
|
|
sync_vmcs02_to_vmcs12(vcpu, vmcs12);
|
2019-05-07 22:36:26 +07:00
|
|
|
|
|
|
|
if (exit_reason != -1)
|
2018-12-04 04:53:18 +07:00
|
|
|
prepare_vmcs12(vcpu, vmcs12, exit_reason, exit_intr_info,
|
|
|
|
exit_qualification);
|
|
|
|
|
|
|
|
/*
|
2019-05-07 22:36:27 +07:00
|
|
|
* Must happen outside of sync_vmcs02_to_vmcs12() as it will
|
2018-12-04 04:53:18 +07:00
|
|
|
* also be used to capture vmcs12 cache as part of
|
|
|
|
* capturing nVMX state for snapshot (migration).
|
|
|
|
*
|
|
|
|
* Otherwise, this flush will dirty guest memory at a
|
|
|
|
* point it is already assumed by user-space to be
|
|
|
|
* immutable.
|
|
|
|
*/
|
|
|
|
nested_flush_cached_shadow_vmcs12(vcpu, vmcs12);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* The only expected VM-instruction error is "VM entry with
|
|
|
|
* invalid control field(s)." Anything else indicates a
|
|
|
|
* problem with L0. And we should never get here with a
|
|
|
|
* VMFail of any type if early consistency checks are enabled.
|
|
|
|
*/
|
|
|
|
WARN_ON_ONCE(vmcs_read32(VM_INSTRUCTION_ERROR) !=
|
|
|
|
VMXERR_ENTRY_INVALID_CONTROL_FIELD);
|
|
|
|
WARN_ON_ONCE(nested_early_check);
|
|
|
|
}
|
|
|
|
|
|
|
|
vmx_switch_vmcs(vcpu, &vmx->vmcs01);
|
|
|
|
|
|
|
|
/* Update any VMCS fields that might have changed while L2 ran */
|
|
|
|
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, vmx->msr_autoload.host.nr);
|
|
|
|
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, vmx->msr_autoload.guest.nr);
|
|
|
|
vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset);
|
2019-11-11 19:30:55 +07:00
|
|
|
if (vmx->nested.l1_tpr_threshold != -1)
|
|
|
|
vmcs_write32(TPR_THRESHOLD, vmx->nested.l1_tpr_threshold);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (kvm_has_tsc_control)
|
|
|
|
decache_tsc_multiplier(vmx);
|
|
|
|
|
|
|
|
if (vmx->nested.change_vmcs01_virtual_apic_mode) {
|
|
|
|
vmx->nested.change_vmcs01_virtual_apic_mode = false;
|
|
|
|
vmx_set_virtual_apic_mode(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unpin physical memory we referred to in vmcs02 */
|
|
|
|
if (vmx->nested.apic_access_page) {
|
2019-11-21 05:31:47 +07:00
|
|
|
kvm_release_page_clean(vmx->nested.apic_access_page);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.apic_access_page = NULL;
|
|
|
|
}
|
2019-02-01 03:24:37 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &vmx->nested.virtual_apic_map, true);
|
2019-02-01 03:24:38 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &vmx->nested.pi_desc_map, true);
|
|
|
|
vmx->nested.pi_desc = NULL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We are now running in L2, mmu_notifier will force to reload the
|
|
|
|
* page's hpa for L2 vmcs. Need to reload it for L1 before entering L1.
|
|
|
|
*/
|
|
|
|
kvm_make_request(KVM_REQ_APIC_PAGE_RELOAD, vcpu);
|
|
|
|
|
|
|
|
if ((exit_reason != -1) && (enable_shadow_vmcs || vmx->nested.hv_evmcs))
|
2019-05-07 22:36:27 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = true;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* in case we halted in L2 */
|
|
|
|
vcpu->arch.mp_state = KVM_MP_STATE_RUNNABLE;
|
|
|
|
|
|
|
|
if (likely(!vmx->fail)) {
|
|
|
|
/*
|
|
|
|
* TODO: SDM says that with acknowledge interrupt on
|
|
|
|
* exit, bit 31 of the VM-exit interrupt information
|
|
|
|
* (valid interrupt) is always set to 1 on
|
|
|
|
* EXIT_REASON_EXTERNAL_INTERRUPT, so we shouldn't
|
|
|
|
* need kvm_cpu_has_interrupt(). See the commit
|
|
|
|
* message for details.
|
|
|
|
*/
|
|
|
|
if (nested_exit_intr_ack_set(vcpu) &&
|
|
|
|
exit_reason == EXIT_REASON_EXTERNAL_INTERRUPT &&
|
|
|
|
kvm_cpu_has_interrupt(vcpu)) {
|
|
|
|
int irq = kvm_cpu_get_interrupt(vcpu);
|
|
|
|
WARN_ON(irq < 0);
|
|
|
|
vmcs12->vm_exit_intr_info = irq |
|
|
|
|
INTR_INFO_VALID_MASK | INTR_TYPE_EXT_INTR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exit_reason != -1)
|
|
|
|
trace_kvm_nested_vmexit_inject(vmcs12->vm_exit_reason,
|
|
|
|
vmcs12->exit_qualification,
|
|
|
|
vmcs12->idt_vectoring_info_field,
|
|
|
|
vmcs12->vm_exit_intr_info,
|
|
|
|
vmcs12->vm_exit_intr_error_code,
|
|
|
|
KVM_ISA_VMX);
|
|
|
|
|
|
|
|
load_vmcs12_host_state(vcpu, vmcs12);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* After an early L2 VM-entry failure, we're now back
|
|
|
|
* in L1 which thinks it just finished a VMLAUNCH or
|
|
|
|
* VMRESUME instruction, so we need to set the failure
|
|
|
|
* flag and the VM-instruction error field of the VMCS
|
|
|
|
* accordingly, and skip the emulated instruction.
|
|
|
|
*/
|
|
|
|
(void)nested_vmx_failValid(vcpu, VMXERR_ENTRY_INVALID_CONTROL_FIELD);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Restore L1's host state to KVM's software model. We're here
|
|
|
|
* because a consistency check was caught by hardware, which
|
|
|
|
* means some amount of guest state has been propagated to KVM's
|
|
|
|
* model and needs to be unwound to the host's state.
|
|
|
|
*/
|
|
|
|
nested_vmx_restore_host_state(vcpu);
|
|
|
|
|
|
|
|
vmx->fail = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Decode the memory-address operand of a vmx instruction, as recorded on an
|
|
|
|
* exit caused by such an instruction (run by a guest hypervisor).
|
|
|
|
* On success, returns 0. When the operand is invalid, returns 1 and throws
|
|
|
|
* #UD or #GP.
|
|
|
|
*/
|
|
|
|
int get_vmx_mem_address(struct kvm_vcpu *vcpu, unsigned long exit_qualification,
|
2019-06-06 04:19:16 +07:00
|
|
|
u32 vmx_instruction_info, bool wr, int len, gva_t *ret)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
gva_t off;
|
|
|
|
bool exn;
|
|
|
|
struct kvm_segment s;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* According to Vol. 3B, "Information for VM Exits Due to Instruction
|
|
|
|
* Execution", on an exit, vmx_instruction_info holds most of the
|
|
|
|
* addressing components of the operand. Only the displacement part
|
|
|
|
* is put in exit_qualification (see 3B, "Basic VM-Exit Information").
|
|
|
|
* For how an actual address is calculated from all these components,
|
|
|
|
* refer to Vol. 1, "Operand Addressing".
|
|
|
|
*/
|
|
|
|
int scaling = vmx_instruction_info & 3;
|
|
|
|
int addr_size = (vmx_instruction_info >> 7) & 7;
|
|
|
|
bool is_reg = vmx_instruction_info & (1u << 10);
|
|
|
|
int seg_reg = (vmx_instruction_info >> 15) & 7;
|
|
|
|
int index_reg = (vmx_instruction_info >> 18) & 0xf;
|
|
|
|
bool index_is_valid = !(vmx_instruction_info & (1u << 22));
|
|
|
|
int base_reg = (vmx_instruction_info >> 23) & 0xf;
|
|
|
|
bool base_is_valid = !(vmx_instruction_info & (1u << 27));
|
|
|
|
|
|
|
|
if (is_reg) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Addr = segment_base + offset */
|
|
|
|
/* offset = base + [index * scale] + displacement */
|
|
|
|
off = exit_qualification; /* holds the displacement */
|
KVM: nVMX: Sign extend displacements of VMX instr's mem operands
The VMCS.EXIT_QUALIFCATION field reports the displacements of memory
operands for various instructions, including VMX instructions, as a
naturally sized unsigned value, but masks the value by the addr size,
e.g. given a ModRM encoded as -0x28(%ebp), the -0x28 displacement is
reported as 0xffffffd8 for a 32-bit address size. Despite some weird
wording regarding sign extension, the SDM explicitly states that bits
beyond the instructions address size are undefined:
In all cases, bits of this field beyond the instruction’s address
size are undefined.
Failure to sign extend the displacement results in KVM incorrectly
treating a negative displacement as a large positive displacement when
the address size of the VMX instruction is smaller than KVM's native
size, e.g. a 32-bit address size on a 64-bit KVM.
The very original decoding, added by commit 064aea774768 ("KVM: nVMX:
Decoding memory operands of VMX instructions"), sort of modeled sign
extension by truncating the final virtual/linear address for a 32-bit
address size. I.e. it messed up the effective address but made it work
by adjusting the final address.
When segmentation checks were added, the truncation logic was kept
as-is and no sign extension logic was introduced. In other words, it
kept calculating the wrong effective address while mostly generating
the correct virtual/linear address. As the effective address is what's
used in the segment limit checks, this results in KVM incorreclty
injecting #GP/#SS faults due to non-existent segment violations when
a nested VMM uses negative displacements with an address size smaller
than KVM's native address size.
Using the -0x28(%ebp) example, an EBP value of 0x1000 will result in
KVM using 0x100000fd8 as the effective address when checking for a
segment limit violation. This causes a 100% failure rate when running
a 32-bit KVM build as L1 on top of a 64-bit KVM L0.
Fixes: f9eb4af67c9d ("KVM: nVMX: VMX instructions: add checks for #GP/#SS exceptions")
Cc: stable@vger.kernel.org
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-01-24 05:39:23 +07:00
|
|
|
if (addr_size == 1)
|
|
|
|
off = (gva_t)sign_extend64(off, 31);
|
|
|
|
else if (addr_size == 0)
|
|
|
|
off = (gva_t)sign_extend64(off, 15);
|
2018-12-04 04:53:18 +07:00
|
|
|
if (base_is_valid)
|
|
|
|
off += kvm_register_read(vcpu, base_reg);
|
|
|
|
if (index_is_valid)
|
|
|
|
off += kvm_register_read(vcpu, index_reg)<<scaling;
|
|
|
|
vmx_get_segment(vcpu, &s, seg_reg);
|
|
|
|
|
2019-01-24 05:39:24 +07:00
|
|
|
/*
|
|
|
|
* The effective address, i.e. @off, of a memory operand is truncated
|
|
|
|
* based on the address size of the instruction. Note that this is
|
|
|
|
* the *effective address*, i.e. the address prior to accounting for
|
|
|
|
* the segment's base.
|
|
|
|
*/
|
2018-12-04 04:53:18 +07:00
|
|
|
if (addr_size == 1) /* 32 bit */
|
2019-01-24 05:39:24 +07:00
|
|
|
off &= 0xffffffff;
|
|
|
|
else if (addr_size == 0) /* 16 bit */
|
|
|
|
off &= 0xffff;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/* Checks for #GP/#SS exceptions. */
|
|
|
|
exn = false;
|
|
|
|
if (is_long_mode(vcpu)) {
|
2019-01-24 05:39:24 +07:00
|
|
|
/*
|
|
|
|
* The virtual/linear address is never truncated in 64-bit
|
|
|
|
* mode, e.g. a 32-bit address size can yield a 64-bit virtual
|
|
|
|
* address when using FS/GS with a non-zero base.
|
|
|
|
*/
|
2019-07-15 22:47:44 +07:00
|
|
|
if (seg_reg == VCPU_SREG_FS || seg_reg == VCPU_SREG_GS)
|
|
|
|
*ret = s.base + off;
|
|
|
|
else
|
|
|
|
*ret = off;
|
2019-01-24 05:39:24 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/* Long mode: #GP(0)/#SS(0) if the memory address is in a
|
|
|
|
* non-canonical form. This is the only check on the memory
|
|
|
|
* destination for long mode!
|
|
|
|
*/
|
|
|
|
exn = is_noncanonical_address(*ret, vcpu);
|
2019-01-30 23:25:38 +07:00
|
|
|
} else {
|
2019-01-24 05:39:24 +07:00
|
|
|
/*
|
|
|
|
* When not in long mode, the virtual/linear address is
|
|
|
|
* unconditionally truncated to 32 bits regardless of the
|
|
|
|
* address size.
|
|
|
|
*/
|
|
|
|
*ret = (s.base + off) & 0xffffffff;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/* Protected mode: apply checks for segment validity in the
|
|
|
|
* following order:
|
|
|
|
* - segment type check (#GP(0) may be thrown)
|
|
|
|
* - usability check (#GP(0)/#SS(0))
|
|
|
|
* - limit check (#GP(0)/#SS(0))
|
|
|
|
*/
|
|
|
|
if (wr)
|
|
|
|
/* #GP(0) if the destination operand is located in a
|
|
|
|
* read-only data segment or any code segment.
|
|
|
|
*/
|
|
|
|
exn = ((s.type & 0xa) == 0 || (s.type & 8));
|
|
|
|
else
|
|
|
|
/* #GP(0) if the source operand is located in an
|
|
|
|
* execute-only code segment
|
|
|
|
*/
|
|
|
|
exn = ((s.type & 0xa) == 8);
|
|
|
|
if (exn) {
|
|
|
|
kvm_queue_exception_e(vcpu, GP_VECTOR, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
/* Protected mode: #GP(0)/#SS(0) if the segment is unusable.
|
|
|
|
*/
|
|
|
|
exn = (s.unusable != 0);
|
2019-01-24 05:39:25 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Protected mode: #GP(0)/#SS(0) if the memory operand is
|
|
|
|
* outside the segment limit. All CPUs that support VMX ignore
|
|
|
|
* limit checks for flat segments, i.e. segments with base==0,
|
|
|
|
* limit==0xffffffff and of type expand-up data or code.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
2019-01-24 05:39:25 +07:00
|
|
|
if (!(s.base == 0 && s.limit == 0xffffffff &&
|
|
|
|
((s.type & 8) || !(s.type & 4))))
|
2019-06-06 04:19:16 +07:00
|
|
|
exn = exn || ((u64)off + len - 1 > s.limit);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
if (exn) {
|
|
|
|
kvm_queue_exception_e(vcpu,
|
|
|
|
seg_reg == VCPU_SREG_SS ?
|
|
|
|
SS_VECTOR : GP_VECTOR,
|
|
|
|
0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-11-14 07:17:20 +07:00
|
|
|
void nested_vmx_pmu_entry_exit_ctls_update(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx;
|
|
|
|
|
|
|
|
if (!nested_vmx_allowed(vcpu))
|
|
|
|
return;
|
|
|
|
|
|
|
|
vmx = to_vmx(vcpu);
|
|
|
|
if (kvm_x86_ops->pmu_ops->is_valid_msr(vcpu, MSR_CORE_PERF_GLOBAL_CTRL)) {
|
|
|
|
vmx->nested.msrs.entry_ctls_high |=
|
|
|
|
VM_ENTRY_LOAD_IA32_PERF_GLOBAL_CTRL;
|
|
|
|
vmx->nested.msrs.exit_ctls_high |=
|
|
|
|
VM_EXIT_LOAD_IA32_PERF_GLOBAL_CTRL;
|
|
|
|
} else {
|
|
|
|
vmx->nested.msrs.entry_ctls_high &=
|
|
|
|
~VM_ENTRY_LOAD_IA32_PERF_GLOBAL_CTRL;
|
|
|
|
vmx->nested.msrs.exit_ctls_high &=
|
|
|
|
~VM_ENTRY_LOAD_IA32_PERF_GLOBAL_CTRL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
static int nested_vmx_get_vmptr(struct kvm_vcpu *vcpu, gpa_t *vmpointer)
|
|
|
|
{
|
|
|
|
gva_t gva;
|
|
|
|
struct x86_exception e;
|
|
|
|
|
|
|
|
if (get_vmx_mem_address(vcpu, vmcs_readl(EXIT_QUALIFICATION),
|
2019-06-06 04:19:16 +07:00
|
|
|
vmcs_read32(VMX_INSTRUCTION_INFO), false,
|
|
|
|
sizeof(*vmpointer), &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (kvm_read_guest_virt(vcpu, gva, vmpointer, sizeof(*vmpointer), &e)) {
|
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a shadow VMCS and associate it with the currently loaded
|
|
|
|
* VMCS, unless such a shadow VMCS already exists. The newly allocated
|
|
|
|
* VMCS is also VMCLEARed, so that it is ready for use.
|
|
|
|
*/
|
|
|
|
static struct vmcs *alloc_shadow_vmcs(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct loaded_vmcs *loaded_vmcs = vmx->loaded_vmcs;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We should allocate a shadow vmcs for vmcs01 only when L1
|
|
|
|
* executes VMXON and free it when L1 executes VMXOFF.
|
|
|
|
* As it is invalid to execute VMXON twice, we shouldn't reach
|
|
|
|
* here when vmcs01 already have an allocated shadow vmcs.
|
|
|
|
*/
|
|
|
|
WARN_ON(loaded_vmcs == &vmx->vmcs01 && loaded_vmcs->shadow_vmcs);
|
|
|
|
|
|
|
|
if (!loaded_vmcs->shadow_vmcs) {
|
|
|
|
loaded_vmcs->shadow_vmcs = alloc_vmcs(true);
|
|
|
|
if (loaded_vmcs->shadow_vmcs)
|
|
|
|
vmcs_clear(loaded_vmcs->shadow_vmcs);
|
|
|
|
}
|
|
|
|
return loaded_vmcs->shadow_vmcs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int enter_vmx_operation(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = alloc_loaded_vmcs(&vmx->nested.vmcs02);
|
|
|
|
if (r < 0)
|
|
|
|
goto out_vmcs02;
|
|
|
|
|
2019-02-12 02:02:52 +07:00
|
|
|
vmx->nested.cached_vmcs12 = kzalloc(VMCS12_SIZE, GFP_KERNEL_ACCOUNT);
|
2018-12-04 04:53:18 +07:00
|
|
|
if (!vmx->nested.cached_vmcs12)
|
|
|
|
goto out_cached_vmcs12;
|
|
|
|
|
2019-02-12 02:02:52 +07:00
|
|
|
vmx->nested.cached_shadow_vmcs12 = kzalloc(VMCS12_SIZE, GFP_KERNEL_ACCOUNT);
|
2018-12-04 04:53:18 +07:00
|
|
|
if (!vmx->nested.cached_shadow_vmcs12)
|
|
|
|
goto out_cached_shadow_vmcs12;
|
|
|
|
|
|
|
|
if (enable_shadow_vmcs && !alloc_shadow_vmcs(vcpu))
|
|
|
|
goto out_shadow_vmcs;
|
|
|
|
|
|
|
|
hrtimer_init(&vmx->nested.preemption_timer, CLOCK_MONOTONIC,
|
|
|
|
HRTIMER_MODE_REL_PINNED);
|
|
|
|
vmx->nested.preemption_timer.function = vmx_preemption_timer_fn;
|
|
|
|
|
|
|
|
vmx->nested.vpid02 = allocate_vpid();
|
|
|
|
|
|
|
|
vmx->nested.vmcs02_initialized = false;
|
|
|
|
vmx->nested.vmxon = true;
|
2018-10-24 15:05:16 +07:00
|
|
|
|
|
|
|
if (pt_mode == PT_MODE_HOST_GUEST) {
|
|
|
|
vmx->pt_desc.guest.ctl = 0;
|
|
|
|
pt_update_intercept_for_msr(vmx);
|
|
|
|
}
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_shadow_vmcs:
|
|
|
|
kfree(vmx->nested.cached_shadow_vmcs12);
|
|
|
|
|
|
|
|
out_cached_shadow_vmcs12:
|
|
|
|
kfree(vmx->nested.cached_vmcs12);
|
|
|
|
|
|
|
|
out_cached_vmcs12:
|
|
|
|
free_loaded_vmcs(&vmx->nested.vmcs02);
|
|
|
|
|
|
|
|
out_vmcs02:
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Emulate the VMXON instruction.
|
|
|
|
* Currently, we just remember that VMX is active, and do not save or even
|
|
|
|
* inspect the argument to VMXON (the so-called "VMXON pointer") because we
|
|
|
|
* do not currently need to store anything in that guest-allocated memory
|
|
|
|
* region. Consequently, VMCLEAR and VMPTRLD also do not verify that the their
|
|
|
|
* argument is different from the VMXON pointer (which the spec says they do).
|
|
|
|
*/
|
|
|
|
static int handle_vmon(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
gpa_t vmptr;
|
2019-02-01 03:24:31 +07:00
|
|
|
uint32_t revision;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
const u64 VMXON_NEEDED_FEATURES = FEATURE_CONTROL_LOCKED
|
|
|
|
| FEATURE_CONTROL_VMXON_ENABLED_OUTSIDE_SMX;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The Intel VMX Instruction Reference lists a bunch of bits that are
|
|
|
|
* prerequisite to running VMXON, most notably cr4.VMXE must be set to
|
|
|
|
* 1 (see vmx_set_cr4() for when we allow the guest to set this).
|
|
|
|
* Otherwise, we should fail with #UD. But most faulting conditions
|
|
|
|
* have already been checked by hardware, prior to the VM-exit for
|
|
|
|
* VMXON. We do test guest cr4.VMXE because processor CR4 always has
|
|
|
|
* that bit set to 1 in non-root mode.
|
|
|
|
*/
|
|
|
|
if (!kvm_read_cr4_bits(vcpu, X86_CR4_VMXE)) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* CPL=0 must be checked manually. */
|
|
|
|
if (vmx_get_cpl(vcpu)) {
|
|
|
|
kvm_inject_gp(vcpu, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vmx->nested.vmxon)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMXON_IN_VMX_ROOT_OPERATION);
|
|
|
|
|
|
|
|
if ((vmx->msr_ia32_feature_control & VMXON_NEEDED_FEATURES)
|
|
|
|
!= VMXON_NEEDED_FEATURES) {
|
|
|
|
kvm_inject_gp(vcpu, 0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nested_vmx_get_vmptr(vcpu, &vmptr))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SDM 3: 24.11.5
|
|
|
|
* The first 4 bytes of VMXON region contain the supported
|
|
|
|
* VMCS revision identifier
|
|
|
|
*
|
|
|
|
* Note - IA32_VMX_BASIC[48] will never be 1 for the nested case;
|
|
|
|
* which replaces physical address width with 32
|
|
|
|
*/
|
2019-02-01 03:24:43 +07:00
|
|
|
if (!page_address_valid(vcpu, vmptr))
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
2019-02-01 03:24:31 +07:00
|
|
|
if (kvm_read_guest(vcpu->kvm, vmptr, &revision, sizeof(revision)) ||
|
|
|
|
revision != VMCS12_REVISION)
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
vmx->nested.vmxon_ptr = vmptr;
|
|
|
|
ret = enter_vmx_operation(vcpu);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void nested_release_vmcs12(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
|
|
|
|
if (vmx->nested.current_vmptr == -1ull)
|
|
|
|
return;
|
|
|
|
|
2019-05-07 22:36:29 +07:00
|
|
|
copy_vmcs02_to_vmcs12_rare(vcpu, get_vmcs12(vcpu));
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (enable_shadow_vmcs) {
|
|
|
|
/* copy to memory all shadowed fields in case
|
|
|
|
they were modified */
|
|
|
|
copy_shadow_to_vmcs12(vmx);
|
|
|
|
vmx_disable_shadow_vmcs(vmx);
|
|
|
|
}
|
|
|
|
vmx->nested.posted_intr_nv = -1;
|
|
|
|
|
|
|
|
/* Flush VMCS12 to guest memory */
|
|
|
|
kvm_vcpu_write_guest_page(vcpu,
|
|
|
|
vmx->nested.current_vmptr >> PAGE_SHIFT,
|
|
|
|
vmx->nested.cached_vmcs12, 0, VMCS12_SIZE);
|
|
|
|
|
|
|
|
kvm_mmu_free_roots(vcpu, &vcpu->arch.guest_mmu, KVM_MMU_ROOTS_ALL);
|
|
|
|
|
|
|
|
vmx->nested.current_vmptr = -1ull;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the VMXOFF instruction */
|
|
|
|
static int handle_vmoff(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
2019-08-26 17:24:49 +07:00
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
free_nested(vcpu);
|
2019-08-26 17:24:49 +07:00
|
|
|
|
|
|
|
/* Process a latched INIT during time CPU was in VMX operation */
|
|
|
|
kvm_make_request(KVM_REQ_EVENT, vcpu);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the VMCLEAR instruction */
|
|
|
|
static int handle_vmclear(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u32 zero = 0;
|
|
|
|
gpa_t vmptr;
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
u64 evmcs_gpa;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (nested_vmx_get_vmptr(vcpu, &vmptr))
|
|
|
|
return 1;
|
|
|
|
|
2019-02-01 03:24:43 +07:00
|
|
|
if (!page_address_valid(vcpu, vmptr))
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMCLEAR_INVALID_ADDRESS);
|
|
|
|
|
|
|
|
if (vmptr == vmx->nested.vmxon_ptr)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMCLEAR_VMXON_POINTER);
|
|
|
|
|
x86/kvm/nVMX: fix VMCLEAR when Enlightened VMCS is in use
When Enlightened VMCS is in use, it is valid to do VMCLEAR and,
according to TLFS, this should "transition an enlightened VMCS from the
active to the non-active state". It is, however, wrong to assume that
it is only valid to do VMCLEAR for the eVMCS which is currently active
on the vCPU performing VMCLEAR.
Currently, the logic in handle_vmclear() is broken: in case, there is no
active eVMCS on the vCPU doing VMCLEAR we treat the argument as a 'normal'
VMCS and kvm_vcpu_write_guest() to the 'launch_state' field irreversibly
corrupts the memory area.
So, in case the VMCLEAR argument is not the current active eVMCS on the
vCPU, how can we know if the area it is pointing to is a normal or an
enlightened VMCS?
Thanks to the bug in Hyper-V (see commit 72aeb60c52bf7 ("KVM: nVMX: Verify
eVMCS revision id match supported eVMCS version on eVMCS VMPTRLD")) we can
not, the revision can't be used to distinguish between them. So let's
assume it is always enlightened in case enlightened vmentry is enabled in
the assist page. Also, check if vmx->nested.enlightened_vmcs_enabled to
minimize the impact for 'unenlightened' workloads.
Fixes: b8bbab928fb1 ("KVM: nVMX: implement enlightened VMPTRLD and VMCLEAR")
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-06-28 18:23:33 +07:00
|
|
|
/*
|
|
|
|
* When Enlightened VMEntry is enabled on the calling CPU we treat
|
|
|
|
* memory area pointer by vmptr as Enlightened VMCS (as there's no good
|
|
|
|
* way to distinguish it from VMCS12) and we must not corrupt it by
|
|
|
|
* writing to the non-existent 'launch_state' field. The area doesn't
|
|
|
|
* have to be the currently active EVMCS on the calling CPU and there's
|
|
|
|
* nothing KVM has to do to transition it from 'active' to 'non-active'
|
|
|
|
* state. It is possible that the area will stay mapped as
|
|
|
|
* vmx->nested.hv_evmcs but this shouldn't be a problem.
|
|
|
|
*/
|
|
|
|
if (likely(!vmx->nested.enlightened_vmcs_enabled ||
|
|
|
|
!nested_enlightened_vmentry(vcpu, &evmcs_gpa))) {
|
2018-12-04 04:53:18 +07:00
|
|
|
if (vmptr == vmx->nested.current_vmptr)
|
|
|
|
nested_release_vmcs12(vcpu);
|
|
|
|
|
|
|
|
kvm_vcpu_write_guest(vcpu,
|
|
|
|
vmptr + offsetof(struct vmcs12,
|
|
|
|
launch_state),
|
|
|
|
&zero, sizeof(zero));
|
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_run(struct kvm_vcpu *vcpu, bool launch);
|
|
|
|
|
|
|
|
/* Emulate the VMLAUNCH instruction */
|
|
|
|
static int handle_vmlaunch(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
return nested_vmx_run(vcpu, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the VMRESUME instruction */
|
|
|
|
static int handle_vmresume(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
|
|
|
|
return nested_vmx_run(vcpu, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_vmread(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
unsigned long field;
|
|
|
|
u64 field_value;
|
|
|
|
unsigned long exit_qualification = vmcs_readl(EXIT_QUALIFICATION);
|
|
|
|
u32 vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
2019-06-06 04:19:16 +07:00
|
|
|
int len;
|
2018-12-04 04:53:18 +07:00
|
|
|
gva_t gva = 0;
|
|
|
|
struct vmcs12 *vmcs12;
|
2019-09-14 05:26:27 +07:00
|
|
|
struct x86_exception e;
|
2019-05-07 22:36:25 +07:00
|
|
|
short offset;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (to_vmx(vcpu)->nested.current_vmptr == -1ull)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
if (!is_guest_mode(vcpu))
|
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
|
|
|
else {
|
|
|
|
/*
|
|
|
|
* When vmcs->vmcs_link_pointer is -1ull, any VMREAD
|
|
|
|
* to shadowed-field sets the ALU flags for VMfailInvalid.
|
|
|
|
*/
|
|
|
|
if (get_vmcs12(vcpu)->vmcs_link_pointer == -1ull)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
vmcs12 = get_shadow_vmcs12(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Decode instruction info and find the field to read */
|
|
|
|
field = kvm_register_readl(vcpu, (((vmx_instruction_info) >> 28) & 0xf));
|
2019-05-07 22:36:25 +07:00
|
|
|
|
|
|
|
offset = vmcs_field_to_offset(field);
|
|
|
|
if (offset < 0)
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_UNSUPPORTED_VMCS_COMPONENT);
|
|
|
|
|
2019-05-07 22:36:29 +07:00
|
|
|
if (!is_guest_mode(vcpu) && is_vmcs12_ext_field(field))
|
|
|
|
copy_vmcs02_to_vmcs12_rare(vcpu, vmcs12);
|
|
|
|
|
2019-05-07 22:36:25 +07:00
|
|
|
/* Read the field, zero-extended to a u64 field_value */
|
|
|
|
field_value = vmcs12_read_any(vmcs12, field, offset);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Now copy part of this value to register or memory, as requested.
|
|
|
|
* Note that the number of bits actually copied is 32 or 64 depending
|
|
|
|
* on the guest's mode (32 or 64 bit), not on the given field's length.
|
|
|
|
*/
|
|
|
|
if (vmx_instruction_info & (1u << 10)) {
|
|
|
|
kvm_register_writel(vcpu, (((vmx_instruction_info) >> 3) & 0xf),
|
|
|
|
field_value);
|
|
|
|
} else {
|
2019-06-06 04:19:16 +07:00
|
|
|
len = is_64_bit_mode(vcpu) ? 8 : 4;
|
2018-12-04 04:53:18 +07:00
|
|
|
if (get_vmx_mem_address(vcpu, exit_qualification,
|
2019-06-06 04:19:16 +07:00
|
|
|
vmx_instruction_info, true, len, &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
/* _system ok, nested_vmx_check_permission has verified cpl=0 */
|
2019-09-14 05:26:27 +07:00
|
|
|
if (kvm_write_guest_virt_system(vcpu, gva, &field_value, len, &e))
|
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
2019-05-07 22:36:28 +07:00
|
|
|
static bool is_shadow_field_rw(unsigned long field)
|
|
|
|
{
|
|
|
|
switch (field) {
|
|
|
|
#define SHADOW_FIELD_RW(x, y) case x:
|
|
|
|
#include "vmcs_shadow_fields.h"
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_shadow_field_ro(unsigned long field)
|
|
|
|
{
|
|
|
|
switch (field) {
|
|
|
|
#define SHADOW_FIELD_RO(x, y) case x:
|
|
|
|
#include "vmcs_shadow_fields.h"
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
static int handle_vmwrite(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
unsigned long field;
|
2019-06-06 04:19:16 +07:00
|
|
|
int len;
|
2018-12-04 04:53:18 +07:00
|
|
|
gva_t gva;
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
unsigned long exit_qualification = vmcs_readl(EXIT_QUALIFICATION);
|
|
|
|
u32 vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
|
|
|
|
/* The value to write might be 32 or 64 bits, depending on L1's long
|
|
|
|
* mode, and eventually we need to write that into a field of several
|
|
|
|
* possible lengths. The code below first zero-extends the value to 64
|
|
|
|
* bit (field_value), and then copies only the appropriate number of
|
|
|
|
* bits into the vmcs12 field.
|
|
|
|
*/
|
|
|
|
u64 field_value = 0;
|
|
|
|
struct x86_exception e;
|
|
|
|
struct vmcs12 *vmcs12;
|
2019-05-07 22:36:25 +07:00
|
|
|
short offset;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (vmx->nested.current_vmptr == -1ull)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
|
|
|
|
if (vmx_instruction_info & (1u << 10))
|
|
|
|
field_value = kvm_register_readl(vcpu,
|
|
|
|
(((vmx_instruction_info) >> 3) & 0xf));
|
|
|
|
else {
|
2019-06-06 04:19:16 +07:00
|
|
|
len = is_64_bit_mode(vcpu) ? 8 : 4;
|
2018-12-04 04:53:18 +07:00
|
|
|
if (get_vmx_mem_address(vcpu, exit_qualification,
|
2019-06-06 04:19:16 +07:00
|
|
|
vmx_instruction_info, false, len, &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
2019-06-06 04:19:16 +07:00
|
|
|
if (kvm_read_guest_virt(vcpu, gva, &field_value, len, &e)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
field = kvm_register_readl(vcpu, (((vmx_instruction_info) >> 28) & 0xf));
|
|
|
|
/*
|
|
|
|
* If the vCPU supports "VMWRITE to any supported field in the
|
|
|
|
* VMCS," then the "read-only" fields are actually read/write.
|
|
|
|
*/
|
|
|
|
if (vmcs_field_readonly(field) &&
|
|
|
|
!nested_cpu_has_vmwrite_any_field(vcpu))
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMWRITE_READ_ONLY_VMCS_COMPONENT);
|
|
|
|
|
2019-05-07 22:36:29 +07:00
|
|
|
if (!is_guest_mode(vcpu)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
2019-05-07 22:36:29 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Ensure vmcs12 is up-to-date before any VMWRITE that dirties
|
|
|
|
* vmcs12, else we may crush a field or consume a stale value.
|
|
|
|
*/
|
|
|
|
if (!is_shadow_field_rw(field))
|
|
|
|
copy_vmcs02_to_vmcs12_rare(vcpu, vmcs12);
|
|
|
|
} else {
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* When vmcs->vmcs_link_pointer is -1ull, any VMWRITE
|
|
|
|
* to shadowed-field sets the ALU flags for VMfailInvalid.
|
|
|
|
*/
|
|
|
|
if (get_vmcs12(vcpu)->vmcs_link_pointer == -1ull)
|
|
|
|
return nested_vmx_failInvalid(vcpu);
|
|
|
|
vmcs12 = get_shadow_vmcs12(vcpu);
|
|
|
|
}
|
|
|
|
|
2019-05-07 22:36:25 +07:00
|
|
|
offset = vmcs_field_to_offset(field);
|
|
|
|
if (offset < 0)
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_UNSUPPORTED_VMCS_COMPONENT);
|
|
|
|
|
|
|
|
/*
|
KVM: nVMX: Intercept VMWRITEs to GUEST_{CS,SS}_AR_BYTES
VMMs frequently read the guest's CS and SS AR bytes to detect 64-bit
mode and CPL respectively, but effectively never write said fields once
the VM is initialized. Intercepting VMWRITEs for the two fields saves
~55 cycles in copy_shadow_to_vmcs12().
Because some Intel CPUs, e.g. Haswell, drop the reserved bits of the
guest access rights fields on VMWRITE, exposing the fields to L1 for
VMREAD but not VMWRITE leads to inconsistent behavior between L1 and L2.
On hardware that drops the bits, L1 will see the stripped down value due
to reading the value from hardware, while L2 will see the full original
value as stored by KVM. To avoid such an inconsistency, emulate the
behavior on all CPUS, but only for intercepted VMWRITEs so as to avoid
introducing pointless latency into copy_shadow_to_vmcs12(), e.g. if the
emulation were added to vmcs12_write_any().
Since the AR_BYTES emulation is done only for intercepted VMWRITE, if a
future patch (re)exposed AR_BYTES for both VMWRITE and VMREAD, then KVM
would end up with incosistent behavior on pre-Haswell hardware, e.g. KVM
would drop the reserved bits on intercepted VMWRITE, but direct VMWRITE
to the shadow VMCS would not drop the bits. Add a WARN in the shadow
field initialization to detect any attempt to expose an AR_BYTES field
without updating vmcs12_write_any().
Note, emulation of the AR_BYTES reserved bit behavior is based on a
patch[1] from Jim Mattson that applied the emulation to all writes to
vmcs12 so that live migration across different generations of hardware
would not introduce divergent behavior. But given that live migration
of nested state has already been enabled, that ship has sailed (not to
mention that no sane VMM will be affected by this behavior).
[1] https://patchwork.kernel.org/patch/10483321/
Cc: Jim Mattson <jmattson@google.com>
Cc: Liran Alon <liran.alon@oracle.com>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:24 +07:00
|
|
|
* Some Intel CPUs intentionally drop the reserved bits of the AR byte
|
|
|
|
* fields on VMWRITE. Emulate this behavior to ensure consistent KVM
|
|
|
|
* behavior regardless of the underlying hardware, e.g. if an AR_BYTE
|
|
|
|
* field is intercepted for VMWRITE but not VMREAD (in L1), then VMREAD
|
|
|
|
* from L1 will return a different value than VMREAD from L2 (L1 sees
|
|
|
|
* the stripped down value, L2 sees the full value as stored by KVM).
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
KVM: nVMX: Intercept VMWRITEs to GUEST_{CS,SS}_AR_BYTES
VMMs frequently read the guest's CS and SS AR bytes to detect 64-bit
mode and CPL respectively, but effectively never write said fields once
the VM is initialized. Intercepting VMWRITEs for the two fields saves
~55 cycles in copy_shadow_to_vmcs12().
Because some Intel CPUs, e.g. Haswell, drop the reserved bits of the
guest access rights fields on VMWRITE, exposing the fields to L1 for
VMREAD but not VMWRITE leads to inconsistent behavior between L1 and L2.
On hardware that drops the bits, L1 will see the stripped down value due
to reading the value from hardware, while L2 will see the full original
value as stored by KVM. To avoid such an inconsistency, emulate the
behavior on all CPUS, but only for intercepted VMWRITEs so as to avoid
introducing pointless latency into copy_shadow_to_vmcs12(), e.g. if the
emulation were added to vmcs12_write_any().
Since the AR_BYTES emulation is done only for intercepted VMWRITE, if a
future patch (re)exposed AR_BYTES for both VMWRITE and VMREAD, then KVM
would end up with incosistent behavior on pre-Haswell hardware, e.g. KVM
would drop the reserved bits on intercepted VMWRITE, but direct VMWRITE
to the shadow VMCS would not drop the bits. Add a WARN in the shadow
field initialization to detect any attempt to expose an AR_BYTES field
without updating vmcs12_write_any().
Note, emulation of the AR_BYTES reserved bit behavior is based on a
patch[1] from Jim Mattson that applied the emulation to all writes to
vmcs12 so that live migration across different generations of hardware
would not introduce divergent behavior. But given that live migration
of nested state has already been enabled, that ship has sailed (not to
mention that no sane VMM will be affected by this behavior).
[1] https://patchwork.kernel.org/patch/10483321/
Cc: Jim Mattson <jmattson@google.com>
Cc: Liran Alon <liran.alon@oracle.com>
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:24 +07:00
|
|
|
if (field >= GUEST_ES_AR_BYTES && field <= GUEST_TR_AR_BYTES)
|
|
|
|
field_value &= 0x1f0ff;
|
|
|
|
|
2019-05-07 22:36:25 +07:00
|
|
|
vmcs12_write_any(vmcs12, field, offset, field_value);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
2019-05-07 22:36:28 +07:00
|
|
|
* Do not track vmcs12 dirty-state if in guest-mode as we actually
|
|
|
|
* dirty shadow vmcs12 instead of vmcs12. Fields that can be updated
|
|
|
|
* by L1 without a vmexit are always updated in the vmcs02, i.e. don't
|
|
|
|
* "dirty" vmcs12, all others go down the prepare_vmcs02() slow path.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
2019-05-07 22:36:28 +07:00
|
|
|
if (!is_guest_mode(vcpu) && !is_shadow_field_rw(field)) {
|
|
|
|
/*
|
|
|
|
* L1 can read these fields without exiting, ensure the
|
|
|
|
* shadow VMCS is up-to-date.
|
|
|
|
*/
|
|
|
|
if (enable_shadow_vmcs && is_shadow_field_ro(field)) {
|
|
|
|
preempt_disable();
|
|
|
|
vmcs_load(vmx->vmcs01.shadow_vmcs);
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
|
2019-05-07 22:36:28 +07:00
|
|
|
__vmcs_writel(field, field_value);
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
|
2019-05-07 22:36:28 +07:00
|
|
|
vmcs_clear(vmx->vmcs01.shadow_vmcs);
|
|
|
|
vmcs_load(vmx->loaded_vmcs->vmcs);
|
|
|
|
preempt_enable();
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
2019-05-07 22:36:28 +07:00
|
|
|
vmx->nested.dirty_vmcs12 = true;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void set_current_vmptr(struct vcpu_vmx *vmx, gpa_t vmptr)
|
|
|
|
{
|
|
|
|
vmx->nested.current_vmptr = vmptr;
|
|
|
|
if (enable_shadow_vmcs) {
|
2019-05-08 02:17:57 +07:00
|
|
|
secondary_exec_controls_setbit(vmx, SECONDARY_EXEC_SHADOW_VMCS);
|
2018-12-04 04:53:18 +07:00
|
|
|
vmcs_write64(VMCS_LINK_POINTER,
|
|
|
|
__pa(vmx->vmcs01.shadow_vmcs));
|
2019-05-07 22:36:27 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = true;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
vmx->nested.dirty_vmcs12 = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the VMPTRLD instruction */
|
|
|
|
static int handle_vmptrld(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
gpa_t vmptr;
|
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (nested_vmx_get_vmptr(vcpu, &vmptr))
|
|
|
|
return 1;
|
|
|
|
|
2019-02-01 03:24:43 +07:00
|
|
|
if (!page_address_valid(vcpu, vmptr))
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMPTRLD_INVALID_ADDRESS);
|
|
|
|
|
|
|
|
if (vmptr == vmx->nested.vmxon_ptr)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMPTRLD_VMXON_POINTER);
|
|
|
|
|
|
|
|
/* Forbid normal VMPTRLD if Enlightened version was used */
|
|
|
|
if (vmx->nested.hv_evmcs)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (vmx->nested.current_vmptr != vmptr) {
|
2019-02-01 03:24:35 +07:00
|
|
|
struct kvm_host_map map;
|
2018-12-04 04:53:18 +07:00
|
|
|
struct vmcs12 *new_vmcs12;
|
|
|
|
|
2019-02-01 03:24:35 +07:00
|
|
|
if (kvm_vcpu_map(vcpu, gpa_to_gfn(vmptr), &map)) {
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
|
|
|
* Reads from an unbacked page return all 1s,
|
|
|
|
* which means that the 32 bits located at the
|
|
|
|
* given physical address won't match the required
|
|
|
|
* VMCS12_REVISION identifier.
|
|
|
|
*/
|
2019-01-10 00:22:56 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
2018-12-04 04:53:18 +07:00
|
|
|
VMXERR_VMPTRLD_INCORRECT_VMCS_REVISION_ID);
|
|
|
|
}
|
2019-02-01 03:24:35 +07:00
|
|
|
|
|
|
|
new_vmcs12 = map.hva;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (new_vmcs12->hdr.revision_id != VMCS12_REVISION ||
|
|
|
|
(new_vmcs12->hdr.shadow_vmcs &&
|
|
|
|
!nested_cpu_has_vmx_shadow_vmcs(vcpu))) {
|
2019-02-01 03:24:35 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &map, false);
|
2018-12-04 04:53:18 +07:00
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_VMPTRLD_INCORRECT_VMCS_REVISION_ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
nested_release_vmcs12(vcpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load VMCS12 from guest memory since it is not already
|
|
|
|
* cached.
|
|
|
|
*/
|
|
|
|
memcpy(vmx->nested.cached_vmcs12, new_vmcs12, VMCS12_SIZE);
|
2019-02-01 03:24:35 +07:00
|
|
|
kvm_vcpu_unmap(vcpu, &map, false);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
set_current_vmptr(vmx, vmptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the VMPTRST instruction */
|
|
|
|
static int handle_vmptrst(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
unsigned long exit_qual = vmcs_readl(EXIT_QUALIFICATION);
|
|
|
|
u32 instr_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
gpa_t current_vmptr = to_vmx(vcpu)->nested.current_vmptr;
|
|
|
|
struct x86_exception e;
|
|
|
|
gva_t gva;
|
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (unlikely(to_vmx(vcpu)->nested.hv_evmcs))
|
|
|
|
return 1;
|
|
|
|
|
2019-06-06 04:19:16 +07:00
|
|
|
if (get_vmx_mem_address(vcpu, exit_qual, instr_info,
|
|
|
|
true, sizeof(gpa_t), &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
/* *_system ok, nested_vmx_check_permission has verified cpl=0 */
|
|
|
|
if (kvm_write_guest_virt_system(vcpu, gva, (void *)¤t_vmptr,
|
|
|
|
sizeof(gpa_t), &e)) {
|
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Emulate the INVEPT instruction */
|
|
|
|
static int handle_invept(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u32 vmx_instruction_info, types;
|
|
|
|
unsigned long type;
|
|
|
|
gva_t gva;
|
|
|
|
struct x86_exception e;
|
|
|
|
struct {
|
|
|
|
u64 eptp, gpa;
|
|
|
|
} operand;
|
|
|
|
|
|
|
|
if (!(vmx->nested.msrs.secondary_ctls_high &
|
|
|
|
SECONDARY_EXEC_ENABLE_EPT) ||
|
|
|
|
!(vmx->nested.msrs.ept_caps & VMX_EPT_INVEPT_BIT)) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
type = kvm_register_readl(vcpu, (vmx_instruction_info >> 28) & 0xf);
|
|
|
|
|
|
|
|
types = (vmx->nested.msrs.ept_caps >> VMX_EPT_EXTENT_SHIFT) & 6;
|
|
|
|
|
|
|
|
if (type >= 32 || !(types & (1 << type)))
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_INVALID_OPERAND_TO_INVEPT_INVVPID);
|
|
|
|
|
|
|
|
/* According to the Intel VMX instruction reference, the memory
|
|
|
|
* operand is read even if it isn't needed (e.g., for type==global)
|
|
|
|
*/
|
|
|
|
if (get_vmx_mem_address(vcpu, vmcs_readl(EXIT_QUALIFICATION),
|
2019-06-06 04:19:16 +07:00
|
|
|
vmx_instruction_info, false, sizeof(operand), &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
if (kvm_read_guest_virt(vcpu, gva, &operand, sizeof(operand), &e)) {
|
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case VMX_EPT_EXTENT_GLOBAL:
|
2019-06-13 23:16:08 +07:00
|
|
|
case VMX_EPT_EXTENT_CONTEXT:
|
2018-12-04 04:53:18 +07:00
|
|
|
/*
|
2019-06-13 23:16:08 +07:00
|
|
|
* TODO: Sync the necessary shadow EPT roots here, rather than
|
|
|
|
* at the next emulated VM-entry.
|
2018-12-04 04:53:18 +07:00
|
|
|
*/
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG_ON(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_invvpid(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
u32 vmx_instruction_info;
|
|
|
|
unsigned long type, types;
|
|
|
|
gva_t gva;
|
|
|
|
struct x86_exception e;
|
|
|
|
struct {
|
|
|
|
u64 vpid;
|
|
|
|
u64 gla;
|
|
|
|
} operand;
|
|
|
|
u16 vpid02;
|
|
|
|
|
|
|
|
if (!(vmx->nested.msrs.secondary_ctls_high &
|
|
|
|
SECONDARY_EXEC_ENABLE_VPID) ||
|
|
|
|
!(vmx->nested.msrs.vpid_caps & VMX_VPID_INVVPID_BIT)) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nested_vmx_check_permission(vcpu))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
type = kvm_register_readl(vcpu, (vmx_instruction_info >> 28) & 0xf);
|
|
|
|
|
|
|
|
types = (vmx->nested.msrs.vpid_caps &
|
|
|
|
VMX_VPID_EXTENT_SUPPORTED_MASK) >> 8;
|
|
|
|
|
|
|
|
if (type >= 32 || !(types & (1 << type)))
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_INVALID_OPERAND_TO_INVEPT_INVVPID);
|
|
|
|
|
|
|
|
/* according to the intel vmx instruction reference, the memory
|
|
|
|
* operand is read even if it isn't needed (e.g., for type==global)
|
|
|
|
*/
|
|
|
|
if (get_vmx_mem_address(vcpu, vmcs_readl(EXIT_QUALIFICATION),
|
2019-06-06 04:19:16 +07:00
|
|
|
vmx_instruction_info, false, sizeof(operand), &gva))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 1;
|
|
|
|
if (kvm_read_guest_virt(vcpu, gva, &operand, sizeof(operand), &e)) {
|
|
|
|
kvm_inject_page_fault(vcpu, &e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (operand.vpid >> 16)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_INVALID_OPERAND_TO_INVEPT_INVVPID);
|
|
|
|
|
|
|
|
vpid02 = nested_get_vpid02(vcpu);
|
|
|
|
switch (type) {
|
|
|
|
case VMX_VPID_EXTENT_INDIVIDUAL_ADDR:
|
|
|
|
if (!operand.vpid ||
|
|
|
|
is_noncanonical_address(operand.gla, vcpu))
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_INVALID_OPERAND_TO_INVEPT_INVVPID);
|
|
|
|
if (cpu_has_vmx_invvpid_individual_addr()) {
|
|
|
|
__invvpid(VMX_VPID_EXTENT_INDIVIDUAL_ADDR,
|
|
|
|
vpid02, operand.gla);
|
|
|
|
} else
|
|
|
|
__vmx_flush_tlb(vcpu, vpid02, false);
|
|
|
|
break;
|
|
|
|
case VMX_VPID_EXTENT_SINGLE_CONTEXT:
|
|
|
|
case VMX_VPID_EXTENT_SINGLE_NON_GLOBAL:
|
|
|
|
if (!operand.vpid)
|
|
|
|
return nested_vmx_failValid(vcpu,
|
|
|
|
VMXERR_INVALID_OPERAND_TO_INVEPT_INVVPID);
|
|
|
|
__vmx_flush_tlb(vcpu, vpid02, false);
|
|
|
|
break;
|
|
|
|
case VMX_VPID_EXTENT_ALL_CONTEXT:
|
|
|
|
__vmx_flush_tlb(vcpu, vpid02, false);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
WARN_ON_ONCE(1);
|
|
|
|
return kvm_skip_emulated_instruction(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nested_vmx_succeed(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int nested_vmx_eptp_switching(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
2019-05-01 00:36:19 +07:00
|
|
|
u32 index = kvm_rcx_read(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
u64 address;
|
|
|
|
bool accessed_dirty;
|
|
|
|
struct kvm_mmu *mmu = vcpu->arch.walk_mmu;
|
|
|
|
|
|
|
|
if (!nested_cpu_has_eptp_switching(vmcs12) ||
|
|
|
|
!nested_cpu_has_ept(vmcs12))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (index >= VMFUNC_EPTP_ENTRIES)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
|
|
|
|
if (kvm_vcpu_read_guest_page(vcpu, vmcs12->eptp_list_address >> PAGE_SHIFT,
|
|
|
|
&address, index * 8, 8))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
accessed_dirty = !!(address & VMX_EPTP_AD_ENABLE_BIT);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the (L2) guest does a vmfunc to the currently
|
|
|
|
* active ept pointer, we don't have to do anything else
|
|
|
|
*/
|
|
|
|
if (vmcs12->ept_pointer != address) {
|
|
|
|
if (!valid_ept_address(vcpu, address))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
kvm_mmu_unload(vcpu);
|
|
|
|
mmu->ept_ad = accessed_dirty;
|
|
|
|
mmu->mmu_role.base.ad_disabled = !accessed_dirty;
|
|
|
|
vmcs12->ept_pointer = address;
|
|
|
|
/*
|
|
|
|
* TODO: Check what's the correct approach in case
|
|
|
|
* mmu reload fails. Currently, we just let the next
|
|
|
|
* reload potentially fail
|
|
|
|
*/
|
|
|
|
kvm_mmu_reload(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_vmfunc(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmcs12 *vmcs12;
|
2019-05-01 00:36:19 +07:00
|
|
|
u32 function = kvm_rax_read(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* VMFUNC is only supported for nested guests, but we always enable the
|
|
|
|
* secondary control for simplicity; for non-nested mode, fake that we
|
|
|
|
* didn't by injecting #UD.
|
|
|
|
*/
|
|
|
|
if (!is_guest_mode(vcpu)) {
|
|
|
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
|
|
|
if ((vmcs12->vm_function_control & (1 << function)) == 0)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
switch (function) {
|
|
|
|
case 0:
|
|
|
|
if (nested_vmx_eptp_switching(vcpu, vmcs12))
|
|
|
|
goto fail;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
return kvm_skip_emulated_instruction(vcpu);
|
|
|
|
|
|
|
|
fail:
|
|
|
|
nested_vmx_vmexit(vcpu, vmx->exit_reason,
|
|
|
|
vmcs_read32(VM_EXIT_INTR_INFO),
|
|
|
|
vmcs_readl(EXIT_QUALIFICATION));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool nested_vmx_exit_handled_io(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
unsigned long exit_qualification;
|
|
|
|
gpa_t bitmap, last_bitmap;
|
|
|
|
unsigned int port;
|
|
|
|
int size;
|
|
|
|
u8 b;
|
|
|
|
|
|
|
|
if (!nested_cpu_has(vmcs12, CPU_BASED_USE_IO_BITMAPS))
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_UNCOND_IO_EXITING);
|
|
|
|
|
|
|
|
exit_qualification = vmcs_readl(EXIT_QUALIFICATION);
|
|
|
|
|
|
|
|
port = exit_qualification >> 16;
|
|
|
|
size = (exit_qualification & 7) + 1;
|
|
|
|
|
|
|
|
last_bitmap = (gpa_t)-1;
|
|
|
|
b = -1;
|
|
|
|
|
|
|
|
while (size > 0) {
|
|
|
|
if (port < 0x8000)
|
|
|
|
bitmap = vmcs12->io_bitmap_a;
|
|
|
|
else if (port < 0x10000)
|
|
|
|
bitmap = vmcs12->io_bitmap_b;
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
bitmap += (port & 0x7fff) / 8;
|
|
|
|
|
|
|
|
if (last_bitmap != bitmap)
|
|
|
|
if (kvm_vcpu_read_guest(vcpu, bitmap, &b, 1))
|
|
|
|
return true;
|
|
|
|
if (b & (1 << (port & 7)))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
port++;
|
|
|
|
size--;
|
|
|
|
last_bitmap = bitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return 1 if we should exit from L2 to L1 to handle an MSR access access,
|
|
|
|
* rather than handle it ourselves in L0. I.e., check whether L1 expressed
|
|
|
|
* disinterest in the current event (read or write a specific MSR) by using an
|
|
|
|
* MSR bitmap. This may be the case even when L0 doesn't use MSR bitmaps.
|
|
|
|
*/
|
|
|
|
static bool nested_vmx_exit_handled_msr(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12, u32 exit_reason)
|
|
|
|
{
|
2019-05-01 00:36:19 +07:00
|
|
|
u32 msr_index = kvm_rcx_read(vcpu);
|
2018-12-04 04:53:18 +07:00
|
|
|
gpa_t bitmap;
|
|
|
|
|
|
|
|
if (!nested_cpu_has(vmcs12, CPU_BASED_USE_MSR_BITMAPS))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The MSR_BITMAP page is divided into four 1024-byte bitmaps,
|
|
|
|
* for the four combinations of read/write and low/high MSR numbers.
|
|
|
|
* First we need to figure out which of the four to use:
|
|
|
|
*/
|
|
|
|
bitmap = vmcs12->msr_bitmap;
|
|
|
|
if (exit_reason == EXIT_REASON_MSR_WRITE)
|
|
|
|
bitmap += 2048;
|
|
|
|
if (msr_index >= 0xc0000000) {
|
|
|
|
msr_index -= 0xc0000000;
|
|
|
|
bitmap += 1024;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Then read the msr_index'th bit from this bitmap: */
|
|
|
|
if (msr_index < 1024*8) {
|
|
|
|
unsigned char b;
|
|
|
|
if (kvm_vcpu_read_guest(vcpu, bitmap + msr_index/8, &b, 1))
|
|
|
|
return true;
|
|
|
|
return 1 & (b >> (msr_index & 7));
|
|
|
|
} else
|
|
|
|
return true; /* let L1 handle the wrong parameter */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return 1 if we should exit from L2 to L1 to handle a CR access exit,
|
|
|
|
* rather than handle it ourselves in L0. I.e., check if L1 wanted to
|
|
|
|
* intercept (via guest_host_mask etc.) the current event.
|
|
|
|
*/
|
|
|
|
static bool nested_vmx_exit_handled_cr(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12)
|
|
|
|
{
|
|
|
|
unsigned long exit_qualification = vmcs_readl(EXIT_QUALIFICATION);
|
|
|
|
int cr = exit_qualification & 15;
|
|
|
|
int reg;
|
|
|
|
unsigned long val;
|
|
|
|
|
|
|
|
switch ((exit_qualification >> 4) & 3) {
|
|
|
|
case 0: /* mov to cr */
|
|
|
|
reg = (exit_qualification >> 8) & 15;
|
|
|
|
val = kvm_register_readl(vcpu, reg);
|
|
|
|
switch (cr) {
|
|
|
|
case 0:
|
|
|
|
if (vmcs12->cr0_guest_host_mask &
|
|
|
|
(val ^ vmcs12->cr0_read_shadow))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
if ((vmcs12->cr3_target_count >= 1 &&
|
|
|
|
vmcs12->cr3_target_value0 == val) ||
|
|
|
|
(vmcs12->cr3_target_count >= 2 &&
|
|
|
|
vmcs12->cr3_target_value1 == val) ||
|
|
|
|
(vmcs12->cr3_target_count >= 3 &&
|
|
|
|
vmcs12->cr3_target_value2 == val) ||
|
|
|
|
(vmcs12->cr3_target_count >= 4 &&
|
|
|
|
vmcs12->cr3_target_value3 == val))
|
|
|
|
return false;
|
|
|
|
if (nested_cpu_has(vmcs12, CPU_BASED_CR3_LOAD_EXITING))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
if (vmcs12->cr4_guest_host_mask &
|
|
|
|
(vmcs12->cr4_read_shadow ^ val))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
if (nested_cpu_has(vmcs12, CPU_BASED_CR8_LOAD_EXITING))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2: /* clts */
|
|
|
|
if ((vmcs12->cr0_guest_host_mask & X86_CR0_TS) &&
|
|
|
|
(vmcs12->cr0_read_shadow & X86_CR0_TS))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case 1: /* mov from cr */
|
|
|
|
switch (cr) {
|
|
|
|
case 3:
|
|
|
|
if (vmcs12->cpu_based_vm_exec_control &
|
|
|
|
CPU_BASED_CR3_STORE_EXITING)
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
if (vmcs12->cpu_based_vm_exec_control &
|
|
|
|
CPU_BASED_CR8_STORE_EXITING)
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 3: /* lmsw */
|
|
|
|
/*
|
|
|
|
* lmsw can change bits 1..3 of cr0, and only set bit 0 of
|
|
|
|
* cr0. Other attempted changes are ignored, with no exit.
|
|
|
|
*/
|
|
|
|
val = (exit_qualification >> LMSW_SOURCE_DATA_SHIFT) & 0x0f;
|
|
|
|
if (vmcs12->cr0_guest_host_mask & 0xe &
|
|
|
|
(val ^ vmcs12->cr0_read_shadow))
|
|
|
|
return true;
|
|
|
|
if ((vmcs12->cr0_guest_host_mask & 0x1) &&
|
|
|
|
!(vmcs12->cr0_read_shadow & 0x1) &&
|
|
|
|
(val & 0x1))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool nested_vmx_exit_handled_vmcs_access(struct kvm_vcpu *vcpu,
|
|
|
|
struct vmcs12 *vmcs12, gpa_t bitmap)
|
|
|
|
{
|
|
|
|
u32 vmx_instruction_info;
|
|
|
|
unsigned long field;
|
|
|
|
u8 b;
|
|
|
|
|
|
|
|
if (!nested_cpu_has_shadow_vmcs(vmcs12))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
/* Decode instruction info and find the field to access */
|
|
|
|
vmx_instruction_info = vmcs_read32(VMX_INSTRUCTION_INFO);
|
|
|
|
field = kvm_register_read(vcpu, (((vmx_instruction_info) >> 28) & 0xf));
|
|
|
|
|
|
|
|
/* Out-of-range fields always cause a VM exit from L2 to L1 */
|
|
|
|
if (field >> 15)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (kvm_vcpu_read_guest(vcpu, bitmap + field/8, &b, 1))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return 1 & (b >> (field & 7));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return 1 if we should exit from L2 to L1 to handle an exit, or 0 if we
|
|
|
|
* should handle it ourselves in L0 (and then continue L2). Only call this
|
|
|
|
* when in is_guest_mode (L2).
|
|
|
|
*/
|
|
|
|
bool nested_vmx_exit_reflected(struct kvm_vcpu *vcpu, u32 exit_reason)
|
|
|
|
{
|
|
|
|
u32 intr_info = vmcs_read32(VM_EXIT_INTR_INFO);
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmcs12 *vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
|
|
|
if (vmx->nested.nested_run_pending)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (unlikely(vmx->fail)) {
|
2019-07-11 22:58:30 +07:00
|
|
|
trace_kvm_nested_vmenter_failed(
|
|
|
|
"hardware VM-instruction error: ",
|
|
|
|
vmcs_read32(VM_INSTRUCTION_ERROR));
|
2018-12-04 04:53:18 +07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The host physical addresses of some pages of guest memory
|
|
|
|
* are loaded into the vmcs02 (e.g. vmcs12's Virtual APIC
|
|
|
|
* Page). The CPU may write to these pages via their host
|
|
|
|
* physical address while L2 is running, bypassing any
|
|
|
|
* address-translation-based dirty tracking (e.g. EPT write
|
|
|
|
* protection).
|
|
|
|
*
|
|
|
|
* Mark them dirty on every exit from L2 to prevent them from
|
|
|
|
* getting out of sync with dirty tracking.
|
|
|
|
*/
|
|
|
|
nested_mark_vmcs12_pages_dirty(vcpu);
|
|
|
|
|
|
|
|
trace_kvm_nested_vmexit(kvm_rip_read(vcpu), exit_reason,
|
|
|
|
vmcs_readl(EXIT_QUALIFICATION),
|
|
|
|
vmx->idt_vectoring_info,
|
|
|
|
intr_info,
|
|
|
|
vmcs_read32(VM_EXIT_INTR_ERROR_CODE),
|
|
|
|
KVM_ISA_VMX);
|
|
|
|
|
|
|
|
switch (exit_reason) {
|
|
|
|
case EXIT_REASON_EXCEPTION_NMI:
|
|
|
|
if (is_nmi(intr_info))
|
|
|
|
return false;
|
|
|
|
else if (is_page_fault(intr_info))
|
|
|
|
return !vmx->vcpu.arch.apf.host_apf_reason && enable_ept;
|
|
|
|
else if (is_debug(intr_info) &&
|
|
|
|
vcpu->guest_debug &
|
|
|
|
(KVM_GUESTDBG_SINGLESTEP | KVM_GUESTDBG_USE_HW_BP))
|
|
|
|
return false;
|
|
|
|
else if (is_breakpoint(intr_info) &&
|
|
|
|
vcpu->guest_debug & KVM_GUESTDBG_USE_SW_BP)
|
|
|
|
return false;
|
|
|
|
return vmcs12->exception_bitmap &
|
|
|
|
(1u << (intr_info & INTR_INFO_VECTOR_MASK));
|
|
|
|
case EXIT_REASON_EXTERNAL_INTERRUPT:
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_TRIPLE_FAULT:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_PENDING_INTERRUPT:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_VIRTUAL_INTR_PENDING);
|
|
|
|
case EXIT_REASON_NMI_WINDOW:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_VIRTUAL_NMI_PENDING);
|
|
|
|
case EXIT_REASON_TASK_SWITCH:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_CPUID:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_HLT:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_HLT_EXITING);
|
|
|
|
case EXIT_REASON_INVD:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_INVLPG:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_INVLPG_EXITING);
|
|
|
|
case EXIT_REASON_RDPMC:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_RDPMC_EXITING);
|
|
|
|
case EXIT_REASON_RDRAND:
|
|
|
|
return nested_cpu_has2(vmcs12, SECONDARY_EXEC_RDRAND_EXITING);
|
|
|
|
case EXIT_REASON_RDSEED:
|
|
|
|
return nested_cpu_has2(vmcs12, SECONDARY_EXEC_RDSEED_EXITING);
|
|
|
|
case EXIT_REASON_RDTSC: case EXIT_REASON_RDTSCP:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_RDTSC_EXITING);
|
|
|
|
case EXIT_REASON_VMREAD:
|
|
|
|
return nested_vmx_exit_handled_vmcs_access(vcpu, vmcs12,
|
|
|
|
vmcs12->vmread_bitmap);
|
|
|
|
case EXIT_REASON_VMWRITE:
|
|
|
|
return nested_vmx_exit_handled_vmcs_access(vcpu, vmcs12,
|
|
|
|
vmcs12->vmwrite_bitmap);
|
|
|
|
case EXIT_REASON_VMCALL: case EXIT_REASON_VMCLEAR:
|
|
|
|
case EXIT_REASON_VMLAUNCH: case EXIT_REASON_VMPTRLD:
|
|
|
|
case EXIT_REASON_VMPTRST: case EXIT_REASON_VMRESUME:
|
|
|
|
case EXIT_REASON_VMOFF: case EXIT_REASON_VMON:
|
|
|
|
case EXIT_REASON_INVEPT: case EXIT_REASON_INVVPID:
|
|
|
|
/*
|
|
|
|
* VMX instructions trap unconditionally. This allows L1 to
|
|
|
|
* emulate them for its L2 guest, i.e., allows 3-level nesting!
|
|
|
|
*/
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_CR_ACCESS:
|
|
|
|
return nested_vmx_exit_handled_cr(vcpu, vmcs12);
|
|
|
|
case EXIT_REASON_DR_ACCESS:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_MOV_DR_EXITING);
|
|
|
|
case EXIT_REASON_IO_INSTRUCTION:
|
|
|
|
return nested_vmx_exit_handled_io(vcpu, vmcs12);
|
|
|
|
case EXIT_REASON_GDTR_IDTR: case EXIT_REASON_LDTR_TR:
|
|
|
|
return nested_cpu_has2(vmcs12, SECONDARY_EXEC_DESC);
|
|
|
|
case EXIT_REASON_MSR_READ:
|
|
|
|
case EXIT_REASON_MSR_WRITE:
|
|
|
|
return nested_vmx_exit_handled_msr(vcpu, vmcs12, exit_reason);
|
|
|
|
case EXIT_REASON_INVALID_STATE:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_MWAIT_INSTRUCTION:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_MWAIT_EXITING);
|
|
|
|
case EXIT_REASON_MONITOR_TRAP_FLAG:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_MONITOR_TRAP_FLAG);
|
|
|
|
case EXIT_REASON_MONITOR_INSTRUCTION:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_MONITOR_EXITING);
|
|
|
|
case EXIT_REASON_PAUSE_INSTRUCTION:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_PAUSE_EXITING) ||
|
|
|
|
nested_cpu_has2(vmcs12,
|
|
|
|
SECONDARY_EXEC_PAUSE_LOOP_EXITING);
|
|
|
|
case EXIT_REASON_MCE_DURING_VMENTRY:
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_TPR_BELOW_THRESHOLD:
|
|
|
|
return nested_cpu_has(vmcs12, CPU_BASED_TPR_SHADOW);
|
|
|
|
case EXIT_REASON_APIC_ACCESS:
|
|
|
|
case EXIT_REASON_APIC_WRITE:
|
|
|
|
case EXIT_REASON_EOI_INDUCED:
|
|
|
|
/*
|
|
|
|
* The controls for "virtualize APIC accesses," "APIC-
|
|
|
|
* register virtualization," and "virtual-interrupt
|
|
|
|
* delivery" only come from vmcs12.
|
|
|
|
*/
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_EPT_VIOLATION:
|
|
|
|
/*
|
|
|
|
* L0 always deals with the EPT violation. If nested EPT is
|
|
|
|
* used, and the nested mmu code discovers that the address is
|
|
|
|
* missing in the guest EPT table (EPT12), the EPT violation
|
|
|
|
* will be injected with nested_ept_inject_page_fault()
|
|
|
|
*/
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_EPT_MISCONFIG:
|
|
|
|
/*
|
|
|
|
* L2 never uses directly L1's EPT, but rather L0's own EPT
|
|
|
|
* table (shadow on EPT) or a merged EPT table that L0 built
|
|
|
|
* (EPT on EPT). So any problems with the structure of the
|
|
|
|
* table is L0's fault.
|
|
|
|
*/
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_INVPCID:
|
|
|
|
return
|
|
|
|
nested_cpu_has2(vmcs12, SECONDARY_EXEC_ENABLE_INVPCID) &&
|
|
|
|
nested_cpu_has(vmcs12, CPU_BASED_INVLPG_EXITING);
|
|
|
|
case EXIT_REASON_WBINVD:
|
|
|
|
return nested_cpu_has2(vmcs12, SECONDARY_EXEC_WBINVD_EXITING);
|
|
|
|
case EXIT_REASON_XSETBV:
|
|
|
|
return true;
|
|
|
|
case EXIT_REASON_XSAVES: case EXIT_REASON_XRSTORS:
|
|
|
|
/*
|
|
|
|
* This should never happen, since it is not possible to
|
|
|
|
* set XSS to a non-zero value---neither in L1 nor in L2.
|
|
|
|
* If if it were, XSS would have to be checked against
|
|
|
|
* the XSS exit bitmap in vmcs12.
|
|
|
|
*/
|
|
|
|
return nested_cpu_has2(vmcs12, SECONDARY_EXEC_XSAVES);
|
|
|
|
case EXIT_REASON_PREEMPTION_TIMER:
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_PML_FULL:
|
|
|
|
/* We emulate PML support to L1. */
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_VMFUNC:
|
|
|
|
/* VM functions are emulated through L2->L0 vmexits. */
|
|
|
|
return false;
|
|
|
|
case EXIT_REASON_ENCLS:
|
|
|
|
/* SGX is never exposed to L1 */
|
|
|
|
return false;
|
2019-07-16 13:55:51 +07:00
|
|
|
case EXIT_REASON_UMWAIT:
|
|
|
|
case EXIT_REASON_TPAUSE:
|
|
|
|
return nested_cpu_has2(vmcs12,
|
|
|
|
SECONDARY_EXEC_ENABLE_USR_WAIT_PAUSE);
|
2018-12-04 04:53:18 +07:00
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int vmx_get_nested_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct kvm_nested_state __user *user_kvm_nested_state,
|
|
|
|
u32 user_data_size)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx;
|
|
|
|
struct vmcs12 *vmcs12;
|
|
|
|
struct kvm_nested_state kvm_state = {
|
|
|
|
.flags = 0,
|
2019-06-16 19:03:10 +07:00
|
|
|
.format = KVM_STATE_NESTED_FORMAT_VMX,
|
2018-12-04 04:53:18 +07:00
|
|
|
.size = sizeof(kvm_state),
|
2019-06-16 19:03:10 +07:00
|
|
|
.hdr.vmx.vmxon_pa = -1ull,
|
|
|
|
.hdr.vmx.vmcs12_pa = -1ull,
|
2018-12-04 04:53:18 +07:00
|
|
|
};
|
2019-06-16 19:03:10 +07:00
|
|
|
struct kvm_vmx_nested_state_data __user *user_vmx_nested_state =
|
|
|
|
&user_kvm_nested_state->data.vmx[0];
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (!vcpu)
|
2019-06-16 19:03:10 +07:00
|
|
|
return kvm_state.size + sizeof(*user_vmx_nested_state);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
vmx = to_vmx(vcpu);
|
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
|
|
|
|
|
|
|
if (nested_vmx_allowed(vcpu) &&
|
|
|
|
(vmx->nested.vmxon || vmx->nested.smm.vmxon)) {
|
2019-06-16 19:03:10 +07:00
|
|
|
kvm_state.hdr.vmx.vmxon_pa = vmx->nested.vmxon_ptr;
|
|
|
|
kvm_state.hdr.vmx.vmcs12_pa = vmx->nested.current_vmptr;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vmx_has_valid_vmcs12(vcpu)) {
|
2019-06-16 19:03:10 +07:00
|
|
|
kvm_state.size += sizeof(user_vmx_nested_state->vmcs12);
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-06-26 20:09:27 +07:00
|
|
|
if (vmx->nested.hv_evmcs)
|
|
|
|
kvm_state.flags |= KVM_STATE_NESTED_EVMCS;
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
if (is_guest_mode(vcpu) &&
|
|
|
|
nested_cpu_has_shadow_vmcs(vmcs12) &&
|
|
|
|
vmcs12->vmcs_link_pointer != -1ull)
|
2019-06-16 19:03:10 +07:00
|
|
|
kvm_state.size += sizeof(user_vmx_nested_state->shadow_vmcs12);
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vmx->nested.smm.vmxon)
|
2019-06-16 19:03:10 +07:00
|
|
|
kvm_state.hdr.vmx.smm.flags |= KVM_STATE_NESTED_SMM_VMXON;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (vmx->nested.smm.guest_mode)
|
2019-06-16 19:03:10 +07:00
|
|
|
kvm_state.hdr.vmx.smm.flags |= KVM_STATE_NESTED_SMM_GUEST_MODE;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (is_guest_mode(vcpu)) {
|
|
|
|
kvm_state.flags |= KVM_STATE_NESTED_GUEST_MODE;
|
|
|
|
|
|
|
|
if (vmx->nested.nested_run_pending)
|
|
|
|
kvm_state.flags |= KVM_STATE_NESTED_RUN_PENDING;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (user_data_size < kvm_state.size)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (copy_to_user(user_kvm_nested_state, &kvm_state, sizeof(kvm_state)))
|
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
if (!vmx_has_valid_vmcs12(vcpu))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When running L2, the authoritative vmcs12 state is in the
|
|
|
|
* vmcs02. When running L1, the authoritative vmcs12 state is
|
|
|
|
* in the shadow or enlightened vmcs linked to vmcs01, unless
|
2019-05-07 22:36:27 +07:00
|
|
|
* need_vmcs12_to_shadow_sync is set, in which case, the authoritative
|
2018-12-04 04:53:18 +07:00
|
|
|
* vmcs12 state is in the vmcs12 already.
|
|
|
|
*/
|
|
|
|
if (is_guest_mode(vcpu)) {
|
2019-05-07 22:36:27 +07:00
|
|
|
sync_vmcs02_to_vmcs12(vcpu, vmcs12);
|
2019-05-07 22:36:29 +07:00
|
|
|
sync_vmcs02_to_vmcs12_rare(vcpu, vmcs12);
|
2019-05-07 22:36:27 +07:00
|
|
|
} else if (!vmx->nested.need_vmcs12_to_shadow_sync) {
|
2018-12-04 04:53:18 +07:00
|
|
|
if (vmx->nested.hv_evmcs)
|
|
|
|
copy_enlightened_to_vmcs12(vmx);
|
|
|
|
else if (enable_shadow_vmcs)
|
|
|
|
copy_shadow_to_vmcs12(vmx);
|
|
|
|
}
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
BUILD_BUG_ON(sizeof(user_vmx_nested_state->vmcs12) < VMCS12_SIZE);
|
|
|
|
BUILD_BUG_ON(sizeof(user_vmx_nested_state->shadow_vmcs12) < VMCS12_SIZE);
|
|
|
|
|
2019-01-25 04:48:20 +07:00
|
|
|
/*
|
|
|
|
* Copy over the full allocated size of vmcs12 rather than just the size
|
|
|
|
* of the struct.
|
|
|
|
*/
|
2019-06-16 19:03:10 +07:00
|
|
|
if (copy_to_user(user_vmx_nested_state->vmcs12, vmcs12, VMCS12_SIZE))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
if (nested_cpu_has_shadow_vmcs(vmcs12) &&
|
|
|
|
vmcs12->vmcs_link_pointer != -1ull) {
|
2019-06-16 19:03:10 +07:00
|
|
|
if (copy_to_user(user_vmx_nested_state->shadow_vmcs12,
|
2019-01-25 04:48:20 +07:00
|
|
|
get_shadow_vmcs12(vcpu), VMCS12_SIZE))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return kvm_state.size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Forcibly leave nested mode in order to be able to reset the VCPU later on.
|
|
|
|
*/
|
|
|
|
void vmx_leave_nested(struct kvm_vcpu *vcpu)
|
|
|
|
{
|
|
|
|
if (is_guest_mode(vcpu)) {
|
|
|
|
to_vmx(vcpu)->nested.nested_run_pending = 0;
|
|
|
|
nested_vmx_vmexit(vcpu, -1, 0, 0);
|
|
|
|
}
|
|
|
|
free_nested(vcpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmx_set_nested_state(struct kvm_vcpu *vcpu,
|
|
|
|
struct kvm_nested_state __user *user_kvm_nested_state,
|
|
|
|
struct kvm_nested_state *kvm_state)
|
|
|
|
{
|
|
|
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
|
|
|
struct vmcs12 *vmcs12;
|
|
|
|
u32 exit_qual;
|
2019-06-16 19:03:10 +07:00
|
|
|
struct kvm_vmx_nested_state_data __user *user_vmx_nested_state =
|
|
|
|
&user_kvm_nested_state->data.vmx[0];
|
2018-12-04 04:53:18 +07:00
|
|
|
int ret;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->format != KVM_STATE_NESTED_FORMAT_VMX)
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.vmxon_pa == -1ull) {
|
|
|
|
if (kvm_state->hdr.vmx.smm.flags)
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.vmcs12_pa != -1ull)
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-26 20:09:27 +07:00
|
|
|
/*
|
|
|
|
* KVM_STATE_NESTED_EVMCS used to signal that KVM should
|
|
|
|
* enable eVMCS capability on vCPU. However, since then
|
|
|
|
* code was changed such that flag signals vmcs12 should
|
|
|
|
* be copied into eVMCS in guest memory.
|
|
|
|
*
|
|
|
|
* To preserve backwards compatability, allow user
|
|
|
|
* to set this flag even when there is no VMXON region.
|
|
|
|
*/
|
2019-06-19 21:52:27 +07:00
|
|
|
if (kvm_state->flags & ~KVM_STATE_NESTED_EVMCS)
|
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
|
|
|
if (!nested_vmx_allowed(vcpu))
|
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-06-19 21:52:27 +07:00
|
|
|
if (!page_address_valid(vcpu, kvm_state->hdr.vmx.vmxon_pa))
|
|
|
|
return -EINVAL;
|
2019-06-26 20:09:27 +07:00
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if ((kvm_state->hdr.vmx.smm.flags & KVM_STATE_NESTED_SMM_GUEST_MODE) &&
|
2018-12-04 04:53:18 +07:00
|
|
|
(kvm_state->flags & KVM_STATE_NESTED_GUEST_MODE))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.smm.flags &
|
2018-12-04 04:53:18 +07:00
|
|
|
~(KVM_STATE_NESTED_SMM_GUEST_MODE | KVM_STATE_NESTED_SMM_VMXON))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SMM temporarily disables VMX, so we cannot be in guest mode,
|
|
|
|
* nor can VMLAUNCH/VMRESUME be pending. Outside SMM, SMM flags
|
|
|
|
* must be zero.
|
|
|
|
*/
|
2019-06-25 18:26:42 +07:00
|
|
|
if (is_smm(vcpu) ?
|
|
|
|
(kvm_state->flags &
|
|
|
|
(KVM_STATE_NESTED_GUEST_MODE | KVM_STATE_NESTED_RUN_PENDING))
|
|
|
|
: kvm_state->hdr.vmx.smm.flags)
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if ((kvm_state->hdr.vmx.smm.flags & KVM_STATE_NESTED_SMM_GUEST_MODE) &&
|
|
|
|
!(kvm_state->hdr.vmx.smm.flags & KVM_STATE_NESTED_SMM_VMXON))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-26 20:09:27 +07:00
|
|
|
if ((kvm_state->flags & KVM_STATE_NESTED_EVMCS) &&
|
|
|
|
(!nested_vmx_allowed(vcpu) || !vmx->nested.enlightened_vmcs_enabled))
|
2019-06-19 21:52:27 +07:00
|
|
|
return -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
2019-06-26 20:09:27 +07:00
|
|
|
vmx_leave_nested(vcpu);
|
2019-06-19 21:52:27 +07:00
|
|
|
|
|
|
|
if (kvm_state->hdr.vmx.vmxon_pa == -1ull)
|
|
|
|
return 0;
|
2019-05-03 01:31:33 +07:00
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
vmx->nested.vmxon_ptr = kvm_state->hdr.vmx.vmxon_pa;
|
2018-12-04 04:53:18 +07:00
|
|
|
ret = enter_vmx_operation(vcpu);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* Empty 'VMXON' state is permitted */
|
2019-01-18 02:55:58 +07:00
|
|
|
if (kvm_state->size < sizeof(*kvm_state) + sizeof(*vmcs12))
|
2018-12-04 04:53:18 +07:00
|
|
|
return 0;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.vmcs12_pa != -1ull) {
|
|
|
|
if (kvm_state->hdr.vmx.vmcs12_pa == kvm_state->hdr.vmx.vmxon_pa ||
|
|
|
|
!page_address_valid(vcpu, kvm_state->hdr.vmx.vmcs12_pa))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
set_current_vmptr(vmx, kvm_state->hdr.vmx.vmcs12_pa);
|
2018-12-04 04:53:18 +07:00
|
|
|
} else if (kvm_state->flags & KVM_STATE_NESTED_EVMCS) {
|
|
|
|
/*
|
|
|
|
* Sync eVMCS upon entry as we may not have
|
|
|
|
* HV_X64_MSR_VP_ASSIST_PAGE set up yet.
|
|
|
|
*/
|
2019-05-07 22:36:27 +07:00
|
|
|
vmx->nested.need_vmcs12_to_shadow_sync = true;
|
2018-12-04 04:53:18 +07:00
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.smm.flags & KVM_STATE_NESTED_SMM_VMXON) {
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.smm.vmxon = true;
|
|
|
|
vmx->nested.vmxon = false;
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->hdr.vmx.smm.flags & KVM_STATE_NESTED_SMM_GUEST_MODE)
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx->nested.smm.guest_mode = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
vmcs12 = get_vmcs12(vcpu);
|
2019-06-16 19:03:10 +07:00
|
|
|
if (copy_from_user(vmcs12, user_vmx_nested_state->vmcs12, sizeof(*vmcs12)))
|
2018-12-04 04:53:18 +07:00
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
if (vmcs12->hdr.revision_id != VMCS12_REVISION)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (!(kvm_state->flags & KVM_STATE_NESTED_GUEST_MODE))
|
|
|
|
return 0;
|
|
|
|
|
2019-05-09 01:04:32 +07:00
|
|
|
vmx->nested.nested_run_pending =
|
|
|
|
!!(kvm_state->flags & KVM_STATE_NESTED_RUN_PENDING);
|
|
|
|
|
|
|
|
ret = -EINVAL;
|
2018-12-04 04:53:18 +07:00
|
|
|
if (nested_cpu_has_shadow_vmcs(vmcs12) &&
|
|
|
|
vmcs12->vmcs_link_pointer != -1ull) {
|
|
|
|
struct vmcs12 *shadow_vmcs12 = get_shadow_vmcs12(vcpu);
|
|
|
|
|
2019-06-16 19:03:10 +07:00
|
|
|
if (kvm_state->size <
|
|
|
|
sizeof(*kvm_state) +
|
|
|
|
sizeof(user_vmx_nested_state->vmcs12) + sizeof(*shadow_vmcs12))
|
2019-05-09 01:04:32 +07:00
|
|
|
goto error_guest_mode;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (copy_from_user(shadow_vmcs12,
|
2019-06-16 19:03:10 +07:00
|
|
|
user_vmx_nested_state->shadow_vmcs12,
|
|
|
|
sizeof(*shadow_vmcs12))) {
|
2019-05-09 01:04:32 +07:00
|
|
|
ret = -EFAULT;
|
|
|
|
goto error_guest_mode;
|
|
|
|
}
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
if (shadow_vmcs12->hdr.revision_id != VMCS12_REVISION ||
|
|
|
|
!shadow_vmcs12->hdr.shadow_vmcs)
|
2019-05-09 01:04:32 +07:00
|
|
|
goto error_guest_mode;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
2019-04-12 02:18:06 +07:00
|
|
|
if (nested_vmx_check_controls(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_host_state(vcpu, vmcs12) ||
|
|
|
|
nested_vmx_check_guest_state(vcpu, vmcs12, &exit_qual))
|
2019-05-09 01:04:32 +07:00
|
|
|
goto error_guest_mode;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
vmx->nested.dirty_vmcs12 = true;
|
|
|
|
ret = nested_vmx_enter_non_root_mode(vcpu, false);
|
2019-05-09 01:04:32 +07:00
|
|
|
if (ret)
|
|
|
|
goto error_guest_mode;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
return 0;
|
2019-05-09 01:04:32 +07:00
|
|
|
|
|
|
|
error_guest_mode:
|
|
|
|
vmx->nested.nested_run_pending = 0;
|
|
|
|
return ret;
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
|
2019-10-20 16:11:01 +07:00
|
|
|
void nested_vmx_set_vmcs_shadowing_bitmap(void)
|
2018-12-04 04:53:18 +07:00
|
|
|
{
|
|
|
|
if (enable_shadow_vmcs) {
|
|
|
|
vmcs_write64(VMREAD_BITMAP, __pa(vmx_vmread_bitmap));
|
KVM: nVMX: Intercept VMWRITEs to read-only shadow VMCS fields
Allowing L1 to VMWRITE read-only fields is only beneficial in a double
nesting scenario, e.g. no sane VMM will VMWRITE VM_EXIT_REASON in normal
non-nested operation. Intercepting RO fields means KVM doesn't need to
sync them from the shadow VMCS to vmcs12 when running L2. The obvious
downside is that L1 will VM-Exit more often when running L3, but it's
likely safe to assume most folks would happily sacrifice a bit of L3
performance, which may not even be noticeable in the grande scheme, to
improve L2 performance across the board.
Not intercepting fields tagged read-only also allows for additional
optimizations, e.g. marking GUEST_{CS,SS}_AR_BYTES as SHADOW_FIELD_RO
since those fields are rarely written by a VMMs, but read frequently.
When utilizing a shadow VMCS with asymmetric R/W and R/O bitmaps, fields
that cause VM-Exit on VMWRITE but not VMREAD need to be propagated to
the shadow VMCS during VMWRITE emulation, otherwise a subsequence VMREAD
from L1 will consume a stale value.
Note, KVM currently utilizes asymmetric bitmaps when "VMWRITE any field"
is not exposed to L1, but only so that it can reject the VMWRITE, i.e.
propagating the VMWRITE to the shadow VMCS is a new requirement, not a
bug fix.
Eliminating the copying of RO fields reduces the latency of nested
VM-Entry (copy_shadow_to_vmcs12()) by ~100 cycles (plus 40-50 cycles
if/when the AR_BYTES fields are exposed RO).
Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2019-05-07 22:36:23 +07:00
|
|
|
vmcs_write64(VMWRITE_BITMAP, __pa(vmx_vmwrite_bitmap));
|
2018-12-04 04:53:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* nested_vmx_setup_ctls_msrs() sets up variables containing the values to be
|
|
|
|
* returned for the various VMX controls MSRs when nested VMX is enabled.
|
|
|
|
* The same values should also be used to verify that vmcs12 control fields are
|
|
|
|
* valid during nested entry from L1 to L2.
|
|
|
|
* Each of these control msrs has a low and high 32-bit half: A low bit is on
|
|
|
|
* if the corresponding bit in the (32-bit) control field *must* be on, and a
|
|
|
|
* bit in the high half is on if the corresponding bit in the control field
|
|
|
|
* may be on. See also vmx_control_verify().
|
|
|
|
*/
|
|
|
|
void nested_vmx_setup_ctls_msrs(struct nested_vmx_msrs *msrs, u32 ept_caps,
|
|
|
|
bool apicv)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Note that as a general rule, the high half of the MSRs (bits in
|
|
|
|
* the control fields which may be 1) should be initialized by the
|
|
|
|
* intersection of the underlying hardware's MSR (i.e., features which
|
|
|
|
* can be supported) and the list of features we want to expose -
|
|
|
|
* because they are known to be properly supported in our code.
|
|
|
|
* Also, usually, the low half of the MSRs (bits which must be 1) can
|
|
|
|
* be set to 0, meaning that L1 may turn off any of these bits. The
|
|
|
|
* reason is that if one of these bits is necessary, it will appear
|
|
|
|
* in vmcs01 and prepare_vmcs02, when it bitwise-or's the control
|
|
|
|
* fields of vmcs01 and vmcs02, will turn these bits off - and
|
|
|
|
* nested_vmx_exit_reflected() will not pass related exits to L1.
|
|
|
|
* These rules have exceptions below.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* pin-based controls */
|
|
|
|
rdmsr(MSR_IA32_VMX_PINBASED_CTLS,
|
|
|
|
msrs->pinbased_ctls_low,
|
|
|
|
msrs->pinbased_ctls_high);
|
|
|
|
msrs->pinbased_ctls_low |=
|
|
|
|
PIN_BASED_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
msrs->pinbased_ctls_high &=
|
|
|
|
PIN_BASED_EXT_INTR_MASK |
|
|
|
|
PIN_BASED_NMI_EXITING |
|
|
|
|
PIN_BASED_VIRTUAL_NMIS |
|
|
|
|
(apicv ? PIN_BASED_POSTED_INTR : 0);
|
|
|
|
msrs->pinbased_ctls_high |=
|
|
|
|
PIN_BASED_ALWAYSON_WITHOUT_TRUE_MSR |
|
|
|
|
PIN_BASED_VMX_PREEMPTION_TIMER;
|
|
|
|
|
|
|
|
/* exit controls */
|
|
|
|
rdmsr(MSR_IA32_VMX_EXIT_CTLS,
|
|
|
|
msrs->exit_ctls_low,
|
|
|
|
msrs->exit_ctls_high);
|
|
|
|
msrs->exit_ctls_low =
|
|
|
|
VM_EXIT_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
|
|
|
|
msrs->exit_ctls_high &=
|
|
|
|
#ifdef CONFIG_X86_64
|
|
|
|
VM_EXIT_HOST_ADDR_SPACE_SIZE |
|
|
|
|
#endif
|
|
|
|
VM_EXIT_LOAD_IA32_PAT | VM_EXIT_SAVE_IA32_PAT;
|
|
|
|
msrs->exit_ctls_high |=
|
|
|
|
VM_EXIT_ALWAYSON_WITHOUT_TRUE_MSR |
|
|
|
|
VM_EXIT_LOAD_IA32_EFER | VM_EXIT_SAVE_IA32_EFER |
|
|
|
|
VM_EXIT_SAVE_VMX_PREEMPTION_TIMER | VM_EXIT_ACK_INTR_ON_EXIT;
|
|
|
|
|
|
|
|
/* We support free control of debug control saving. */
|
|
|
|
msrs->exit_ctls_low &= ~VM_EXIT_SAVE_DEBUG_CONTROLS;
|
|
|
|
|
|
|
|
/* entry controls */
|
|
|
|
rdmsr(MSR_IA32_VMX_ENTRY_CTLS,
|
|
|
|
msrs->entry_ctls_low,
|
|
|
|
msrs->entry_ctls_high);
|
|
|
|
msrs->entry_ctls_low =
|
|
|
|
VM_ENTRY_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
msrs->entry_ctls_high &=
|
|
|
|
#ifdef CONFIG_X86_64
|
|
|
|
VM_ENTRY_IA32E_MODE |
|
|
|
|
#endif
|
|
|
|
VM_ENTRY_LOAD_IA32_PAT;
|
|
|
|
msrs->entry_ctls_high |=
|
|
|
|
(VM_ENTRY_ALWAYSON_WITHOUT_TRUE_MSR | VM_ENTRY_LOAD_IA32_EFER);
|
|
|
|
|
|
|
|
/* We support free control of debug control loading. */
|
|
|
|
msrs->entry_ctls_low &= ~VM_ENTRY_LOAD_DEBUG_CONTROLS;
|
|
|
|
|
|
|
|
/* cpu-based controls */
|
|
|
|
rdmsr(MSR_IA32_VMX_PROCBASED_CTLS,
|
|
|
|
msrs->procbased_ctls_low,
|
|
|
|
msrs->procbased_ctls_high);
|
|
|
|
msrs->procbased_ctls_low =
|
|
|
|
CPU_BASED_ALWAYSON_WITHOUT_TRUE_MSR;
|
|
|
|
msrs->procbased_ctls_high &=
|
|
|
|
CPU_BASED_VIRTUAL_INTR_PENDING |
|
|
|
|
CPU_BASED_VIRTUAL_NMI_PENDING | CPU_BASED_USE_TSC_OFFSETING |
|
|
|
|
CPU_BASED_HLT_EXITING | CPU_BASED_INVLPG_EXITING |
|
|
|
|
CPU_BASED_MWAIT_EXITING | CPU_BASED_CR3_LOAD_EXITING |
|
|
|
|
CPU_BASED_CR3_STORE_EXITING |
|
|
|
|
#ifdef CONFIG_X86_64
|
|
|
|
CPU_BASED_CR8_LOAD_EXITING | CPU_BASED_CR8_STORE_EXITING |
|
|
|
|
#endif
|
|
|
|
CPU_BASED_MOV_DR_EXITING | CPU_BASED_UNCOND_IO_EXITING |
|
|
|
|
CPU_BASED_USE_IO_BITMAPS | CPU_BASED_MONITOR_TRAP_FLAG |
|
|
|
|
CPU_BASED_MONITOR_EXITING | CPU_BASED_RDPMC_EXITING |
|
|
|
|
CPU_BASED_RDTSC_EXITING | CPU_BASED_PAUSE_EXITING |
|
|
|
|
CPU_BASED_TPR_SHADOW | CPU_BASED_ACTIVATE_SECONDARY_CONTROLS;
|
|
|
|
/*
|
|
|
|
* We can allow some features even when not supported by the
|
|
|
|
* hardware. For example, L1 can specify an MSR bitmap - and we
|
|
|
|
* can use it to avoid exits to L1 - even when L0 runs L2
|
|
|
|
* without MSR bitmaps.
|
|
|
|
*/
|
|
|
|
msrs->procbased_ctls_high |=
|
|
|
|
CPU_BASED_ALWAYSON_WITHOUT_TRUE_MSR |
|
|
|
|
CPU_BASED_USE_MSR_BITMAPS;
|
|
|
|
|
|
|
|
/* We support free control of CR3 access interception. */
|
|
|
|
msrs->procbased_ctls_low &=
|
|
|
|
~(CPU_BASED_CR3_LOAD_EXITING | CPU_BASED_CR3_STORE_EXITING);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* secondary cpu-based controls. Do not include those that
|
|
|
|
* depend on CPUID bits, they are added later by vmx_cpuid_update.
|
|
|
|
*/
|
2019-02-07 17:42:14 +07:00
|
|
|
if (msrs->procbased_ctls_high & CPU_BASED_ACTIVATE_SECONDARY_CONTROLS)
|
|
|
|
rdmsr(MSR_IA32_VMX_PROCBASED_CTLS2,
|
|
|
|
msrs->secondary_ctls_low,
|
|
|
|
msrs->secondary_ctls_high);
|
|
|
|
|
2018-12-04 04:53:18 +07:00
|
|
|
msrs->secondary_ctls_low = 0;
|
|
|
|
msrs->secondary_ctls_high &=
|
|
|
|
SECONDARY_EXEC_DESC |
|
2019-07-02 19:39:29 +07:00
|
|
|
SECONDARY_EXEC_RDTSCP |
|
2018-12-04 04:53:18 +07:00
|
|
|
SECONDARY_EXEC_VIRTUALIZE_X2APIC_MODE |
|
2019-07-02 19:39:29 +07:00
|
|
|
SECONDARY_EXEC_WBINVD_EXITING |
|
2018-12-04 04:53:18 +07:00
|
|
|
SECONDARY_EXEC_APIC_REGISTER_VIRT |
|
|
|
|
SECONDARY_EXEC_VIRTUAL_INTR_DELIVERY |
|
2019-07-02 19:39:29 +07:00
|
|
|
SECONDARY_EXEC_RDRAND_EXITING |
|
|
|
|
SECONDARY_EXEC_ENABLE_INVPCID |
|
|
|
|
SECONDARY_EXEC_RDSEED_EXITING |
|
|
|
|
SECONDARY_EXEC_XSAVES;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We can emulate "VMCS shadowing," even if the hardware
|
|
|
|
* doesn't support it.
|
|
|
|
*/
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_SHADOW_VMCS;
|
|
|
|
|
|
|
|
if (enable_ept) {
|
|
|
|
/* nested EPT: emulate EPT also to L1 */
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_ENABLE_EPT;
|
|
|
|
msrs->ept_caps = VMX_EPT_PAGE_WALK_4_BIT |
|
|
|
|
VMX_EPTP_WB_BIT | VMX_EPT_INVEPT_BIT;
|
|
|
|
if (cpu_has_vmx_ept_execute_only())
|
|
|
|
msrs->ept_caps |=
|
|
|
|
VMX_EPT_EXECUTE_ONLY_BIT;
|
|
|
|
msrs->ept_caps &= ept_caps;
|
|
|
|
msrs->ept_caps |= VMX_EPT_EXTENT_GLOBAL_BIT |
|
|
|
|
VMX_EPT_EXTENT_CONTEXT_BIT | VMX_EPT_2MB_PAGE_BIT |
|
|
|
|
VMX_EPT_1GB_PAGE_BIT;
|
|
|
|
if (enable_ept_ad_bits) {
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_ENABLE_PML;
|
|
|
|
msrs->ept_caps |= VMX_EPT_AD_BIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cpu_has_vmx_vmfunc()) {
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_ENABLE_VMFUNC;
|
|
|
|
/*
|
|
|
|
* Advertise EPTP switching unconditionally
|
|
|
|
* since we emulate it
|
|
|
|
*/
|
|
|
|
if (enable_ept)
|
|
|
|
msrs->vmfunc_controls =
|
|
|
|
VMX_VMFUNC_EPTP_SWITCHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Old versions of KVM use the single-context version without
|
|
|
|
* checking for support, so declare that it is supported even
|
|
|
|
* though it is treated as global context. The alternative is
|
|
|
|
* not failing the single-context invvpid, and it is worse.
|
|
|
|
*/
|
|
|
|
if (enable_vpid) {
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_ENABLE_VPID;
|
|
|
|
msrs->vpid_caps = VMX_VPID_INVVPID_BIT |
|
|
|
|
VMX_VPID_EXTENT_SUPPORTED_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enable_unrestricted_guest)
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_UNRESTRICTED_GUEST;
|
|
|
|
|
|
|
|
if (flexpriority_enabled)
|
|
|
|
msrs->secondary_ctls_high |=
|
|
|
|
SECONDARY_EXEC_VIRTUALIZE_APIC_ACCESSES;
|
|
|
|
|
|
|
|
/* miscellaneous data */
|
|
|
|
rdmsr(MSR_IA32_VMX_MISC,
|
|
|
|
msrs->misc_low,
|
|
|
|
msrs->misc_high);
|
|
|
|
msrs->misc_low &= VMX_MISC_SAVE_EFER_LMA;
|
|
|
|
msrs->misc_low |=
|
|
|
|
MSR_IA32_VMX_MISC_VMWRITE_SHADOW_RO_FIELDS |
|
|
|
|
VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE |
|
|
|
|
VMX_MISC_ACTIVITY_HLT;
|
|
|
|
msrs->misc_high = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This MSR reports some information about VMX support. We
|
|
|
|
* should return information about the VMX we emulate for the
|
|
|
|
* guest, and the VMCS structure we give it - not about the
|
|
|
|
* VMX support of the underlying hardware.
|
|
|
|
*/
|
|
|
|
msrs->basic =
|
|
|
|
VMCS12_REVISION |
|
|
|
|
VMX_BASIC_TRUE_CTLS |
|
|
|
|
((u64)VMCS12_SIZE << VMX_BASIC_VMCS_SIZE_SHIFT) |
|
|
|
|
(VMX_BASIC_MEM_TYPE_WB << VMX_BASIC_MEM_TYPE_SHIFT);
|
|
|
|
|
|
|
|
if (cpu_has_vmx_basic_inout())
|
|
|
|
msrs->basic |= VMX_BASIC_INOUT;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These MSRs specify bits which the guest must keep fixed on
|
|
|
|
* while L1 is in VMXON mode (in L1's root mode, or running an L2).
|
|
|
|
* We picked the standard core2 setting.
|
|
|
|
*/
|
|
|
|
#define VMXON_CR0_ALWAYSON (X86_CR0_PE | X86_CR0_PG | X86_CR0_NE)
|
|
|
|
#define VMXON_CR4_ALWAYSON X86_CR4_VMXE
|
|
|
|
msrs->cr0_fixed0 = VMXON_CR0_ALWAYSON;
|
|
|
|
msrs->cr4_fixed0 = VMXON_CR4_ALWAYSON;
|
|
|
|
|
|
|
|
/* These MSRs specify bits which the guest must keep fixed off. */
|
|
|
|
rdmsrl(MSR_IA32_VMX_CR0_FIXED1, msrs->cr0_fixed1);
|
|
|
|
rdmsrl(MSR_IA32_VMX_CR4_FIXED1, msrs->cr4_fixed1);
|
|
|
|
|
|
|
|
/* highest index: VMX_PREEMPTION_TIMER_VALUE */
|
|
|
|
msrs->vmcs_enum = VMCS12_MAX_FIELD_INDEX << 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nested_vmx_hardware_unsetup(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (enable_shadow_vmcs) {
|
|
|
|
for (i = 0; i < VMX_BITMAP_NR; i++)
|
|
|
|
free_page((unsigned long)vmx_bitmap[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__init int nested_vmx_hardware_setup(int (*exit_handlers[])(struct kvm_vcpu *))
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!cpu_has_vmx_shadow_vmcs())
|
|
|
|
enable_shadow_vmcs = 0;
|
|
|
|
if (enable_shadow_vmcs) {
|
|
|
|
for (i = 0; i < VMX_BITMAP_NR; i++) {
|
2019-02-12 02:02:52 +07:00
|
|
|
/*
|
|
|
|
* The vmx_bitmap is not tied to a VM and so should
|
|
|
|
* not be charged to a memcg.
|
|
|
|
*/
|
2018-12-04 04:53:18 +07:00
|
|
|
vmx_bitmap[i] = (unsigned long *)
|
|
|
|
__get_free_page(GFP_KERNEL);
|
|
|
|
if (!vmx_bitmap[i]) {
|
|
|
|
nested_vmx_hardware_unsetup();
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
init_vmcs_shadow_fields();
|
|
|
|
}
|
|
|
|
|
2019-11-19 02:11:21 +07:00
|
|
|
exit_handlers[EXIT_REASON_VMCLEAR] = handle_vmclear;
|
|
|
|
exit_handlers[EXIT_REASON_VMLAUNCH] = handle_vmlaunch;
|
|
|
|
exit_handlers[EXIT_REASON_VMPTRLD] = handle_vmptrld;
|
|
|
|
exit_handlers[EXIT_REASON_VMPTRST] = handle_vmptrst;
|
|
|
|
exit_handlers[EXIT_REASON_VMREAD] = handle_vmread;
|
|
|
|
exit_handlers[EXIT_REASON_VMRESUME] = handle_vmresume;
|
|
|
|
exit_handlers[EXIT_REASON_VMWRITE] = handle_vmwrite;
|
|
|
|
exit_handlers[EXIT_REASON_VMOFF] = handle_vmoff;
|
|
|
|
exit_handlers[EXIT_REASON_VMON] = handle_vmon;
|
|
|
|
exit_handlers[EXIT_REASON_INVEPT] = handle_invept;
|
|
|
|
exit_handlers[EXIT_REASON_INVVPID] = handle_invvpid;
|
|
|
|
exit_handlers[EXIT_REASON_VMFUNC] = handle_vmfunc;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
kvm_x86_ops->check_nested_events = vmx_check_nested_events;
|
|
|
|
kvm_x86_ops->get_nested_state = vmx_get_nested_state;
|
|
|
|
kvm_x86_ops->set_nested_state = vmx_set_nested_state;
|
2019-11-19 02:11:21 +07:00
|
|
|
kvm_x86_ops->get_vmcs12_pages = nested_get_vmcs12_pages;
|
2018-12-04 04:53:18 +07:00
|
|
|
kvm_x86_ops->nested_enable_evmcs = nested_enable_evmcs;
|
2018-12-11 00:21:55 +07:00
|
|
|
kvm_x86_ops->nested_get_evmcs_version = nested_get_evmcs_version;
|
2018-12-04 04:53:18 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|