Commit 0ddc84d2 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'for-linus' of git://git.kernel.org/pub/scm/virt/kvm/kvm

Pull kvm fixes from Paolo Bonzini:
 "ARM64:

   - Address a rather annoying bug w.r.t. guest timer offsetting. The
     synchronization of timer offsets between vCPUs was broken, leading
     to inconsistent timer reads within the VM.

  x86:

   - New tests for the slow path of the EVTCHNOP_send Xen hypercall

   - Add missing nVMX consistency checks for CR0 and CR4

   - Fix bug that broke AMD GATag on 512 vCPU machines

  Selftests:

   - Skip hugetlb tests if huge pages are not available

   - Sync KVM exit reasons"

* tag 'for-linus' of git://git.kernel.org/pub/scm/virt/kvm/kvm:
  KVM: selftests: Sync KVM exit reasons in selftests
  KVM: selftests: Add macro to generate KVM exit reason strings
  KVM: selftests: Print expected and actual exit reason in KVM exit reason assert
  KVM: selftests: Make vCPU exit reason test assertion common
  KVM: selftests: Add EVTCHNOP_send slow path test to xen_shinfo_test
  KVM: selftests: Use enum for test numbers in xen_shinfo_test
  KVM: selftests: Add helpers to make Xen-style VMCALL/VMMCALL hypercalls
  KVM: selftests: Move the guts of kvm_hypercall() to a separate macro
  KVM: SVM: WARN if GATag generation drops VM or vCPU ID information
  KVM: SVM: Modify AVIC GATag to support max number of 512 vCPUs
  KVM: SVM: Fix a benign off-by-one bug in AVIC physical table mask
  selftests: KVM: skip hugetlb tests if huge pages are not available
  KVM: VMX: Use tabs instead of spaces for indentation
  KVM: VMX: Fix indentation coding style issue
  KVM: nVMX: remove unnecessary #ifdef
  KVM: nVMX: add missing consistency checks for CR0 and CR4
  KVM: arm64: timers: Convert per-vcpu virtual offset to a global value
parents 9c1bec9c f3e70741
...@@ -193,6 +193,9 @@ struct kvm_arch { ...@@ -193,6 +193,9 @@ struct kvm_arch {
/* Interrupt controller */ /* Interrupt controller */
struct vgic_dist vgic; struct vgic_dist vgic;
/* Timers */
struct arch_timer_vm_data timer_data;
/* Mandated version of PSCI */ /* Mandated version of PSCI */
u32 psci_version; u32 psci_version;
......
...@@ -84,14 +84,10 @@ u64 timer_get_cval(struct arch_timer_context *ctxt) ...@@ -84,14 +84,10 @@ u64 timer_get_cval(struct arch_timer_context *ctxt)
static u64 timer_get_offset(struct arch_timer_context *ctxt) static u64 timer_get_offset(struct arch_timer_context *ctxt)
{ {
struct kvm_vcpu *vcpu = ctxt->vcpu; if (ctxt->offset.vm_offset)
return *ctxt->offset.vm_offset;
switch(arch_timer_ctx_index(ctxt)) { return 0;
case TIMER_VTIMER:
return __vcpu_sys_reg(vcpu, CNTVOFF_EL2);
default:
return 0;
}
} }
static void timer_set_ctl(struct arch_timer_context *ctxt, u32 ctl) static void timer_set_ctl(struct arch_timer_context *ctxt, u32 ctl)
...@@ -128,15 +124,12 @@ static void timer_set_cval(struct arch_timer_context *ctxt, u64 cval) ...@@ -128,15 +124,12 @@ static void timer_set_cval(struct arch_timer_context *ctxt, u64 cval)
static void timer_set_offset(struct arch_timer_context *ctxt, u64 offset) static void timer_set_offset(struct arch_timer_context *ctxt, u64 offset)
{ {
struct kvm_vcpu *vcpu = ctxt->vcpu; if (!ctxt->offset.vm_offset) {
switch(arch_timer_ctx_index(ctxt)) {
case TIMER_VTIMER:
__vcpu_sys_reg(vcpu, CNTVOFF_EL2) = offset;
break;
default:
WARN(offset, "timer %ld\n", arch_timer_ctx_index(ctxt)); WARN(offset, "timer %ld\n", arch_timer_ctx_index(ctxt));
return;
} }
WRITE_ONCE(*ctxt->offset.vm_offset, offset);
} }
u64 kvm_phys_timer_read(void) u64 kvm_phys_timer_read(void)
...@@ -765,25 +758,6 @@ int kvm_timer_vcpu_reset(struct kvm_vcpu *vcpu) ...@@ -765,25 +758,6 @@ int kvm_timer_vcpu_reset(struct kvm_vcpu *vcpu)
return 0; return 0;
} }
/* Make the updates of cntvoff for all vtimer contexts atomic */
static void update_vtimer_cntvoff(struct kvm_vcpu *vcpu, u64 cntvoff)
{
unsigned long i;
struct kvm *kvm = vcpu->kvm;
struct kvm_vcpu *tmp;
mutex_lock(&kvm->lock);
kvm_for_each_vcpu(i, tmp, kvm)
timer_set_offset(vcpu_vtimer(tmp), cntvoff);
/*
* When called from the vcpu create path, the CPU being created is not
* included in the loop above, so we just set it here as well.
*/
timer_set_offset(vcpu_vtimer(vcpu), cntvoff);
mutex_unlock(&kvm->lock);
}
void kvm_timer_vcpu_init(struct kvm_vcpu *vcpu) void kvm_timer_vcpu_init(struct kvm_vcpu *vcpu)
{ {
struct arch_timer_cpu *timer = vcpu_timer(vcpu); struct arch_timer_cpu *timer = vcpu_timer(vcpu);
...@@ -791,10 +765,11 @@ void kvm_timer_vcpu_init(struct kvm_vcpu *vcpu) ...@@ -791,10 +765,11 @@ void kvm_timer_vcpu_init(struct kvm_vcpu *vcpu)
struct arch_timer_context *ptimer = vcpu_ptimer(vcpu); struct arch_timer_context *ptimer = vcpu_ptimer(vcpu);
vtimer->vcpu = vcpu; vtimer->vcpu = vcpu;
vtimer->offset.vm_offset = &vcpu->kvm->arch.timer_data.voffset;
ptimer->vcpu = vcpu; ptimer->vcpu = vcpu;
/* Synchronize cntvoff across all vtimers of a VM. */ /* Synchronize cntvoff across all vtimers of a VM. */
update_vtimer_cntvoff(vcpu, kvm_phys_timer_read()); timer_set_offset(vtimer, kvm_phys_timer_read());
timer_set_offset(ptimer, 0); timer_set_offset(ptimer, 0);
hrtimer_init(&timer->bg_timer, CLOCK_MONOTONIC, HRTIMER_MODE_ABS_HARD); hrtimer_init(&timer->bg_timer, CLOCK_MONOTONIC, HRTIMER_MODE_ABS_HARD);
...@@ -840,7 +815,7 @@ int kvm_arm_timer_set_reg(struct kvm_vcpu *vcpu, u64 regid, u64 value) ...@@ -840,7 +815,7 @@ int kvm_arm_timer_set_reg(struct kvm_vcpu *vcpu, u64 regid, u64 value)
break; break;
case KVM_REG_ARM_TIMER_CNT: case KVM_REG_ARM_TIMER_CNT:
timer = vcpu_vtimer(vcpu); timer = vcpu_vtimer(vcpu);
update_vtimer_cntvoff(vcpu, kvm_phys_timer_read() - value); timer_set_offset(timer, kvm_phys_timer_read() - value);
break; break;
case KVM_REG_ARM_TIMER_CVAL: case KVM_REG_ARM_TIMER_CVAL:
timer = vcpu_vtimer(vcpu); timer = vcpu_vtimer(vcpu);
......
...@@ -44,7 +44,7 @@ static void kvm_ptp_get_time(struct kvm_vcpu *vcpu, u64 *val) ...@@ -44,7 +44,7 @@ static void kvm_ptp_get_time(struct kvm_vcpu *vcpu, u64 *val)
feature = smccc_get_arg1(vcpu); feature = smccc_get_arg1(vcpu);
switch (feature) { switch (feature) {
case KVM_PTP_VIRT_COUNTER: case KVM_PTP_VIRT_COUNTER:
cycles = systime_snapshot.cycles - vcpu_read_sys_reg(vcpu, CNTVOFF_EL2); cycles = systime_snapshot.cycles - vcpu->kvm->arch.timer_data.voffset;
break; break;
case KVM_PTP_PHYS_COUNTER: case KVM_PTP_PHYS_COUNTER:
cycles = systime_snapshot.cycles; cycles = systime_snapshot.cycles;
......
...@@ -261,20 +261,22 @@ enum avic_ipi_failure_cause { ...@@ -261,20 +261,22 @@ enum avic_ipi_failure_cause {
AVIC_IPI_FAILURE_INVALID_BACKING_PAGE, AVIC_IPI_FAILURE_INVALID_BACKING_PAGE,
}; };
#define AVIC_PHYSICAL_MAX_INDEX_MASK GENMASK_ULL(9, 0) #define AVIC_PHYSICAL_MAX_INDEX_MASK GENMASK_ULL(8, 0)
/* /*
* For AVIC, the max index allowed for physical APIC ID * For AVIC, the max index allowed for physical APIC ID table is 0xfe (254), as
* table is 0xff (255). * 0xff is a broadcast to all CPUs, i.e. can't be targeted individually.
*/ */
#define AVIC_MAX_PHYSICAL_ID 0XFEULL #define AVIC_MAX_PHYSICAL_ID 0XFEULL
/* /*
* For x2AVIC, the max index allowed for physical APIC ID * For x2AVIC, the max index allowed for physical APIC ID table is 0x1ff (511).
* table is 0x1ff (511).
*/ */
#define X2AVIC_MAX_PHYSICAL_ID 0x1FFUL #define X2AVIC_MAX_PHYSICAL_ID 0x1FFUL
static_assert((AVIC_MAX_PHYSICAL_ID & AVIC_PHYSICAL_MAX_INDEX_MASK) == AVIC_MAX_PHYSICAL_ID);
static_assert((X2AVIC_MAX_PHYSICAL_ID & AVIC_PHYSICAL_MAX_INDEX_MASK) == X2AVIC_MAX_PHYSICAL_ID);
#define AVIC_HPA_MASK ~((0xFFFULL << 52) | 0xFFF) #define AVIC_HPA_MASK ~((0xFFFULL << 52) | 0xFFF)
#define VMCB_AVIC_APIC_BAR_MASK 0xFFFFFFFFFF000ULL #define VMCB_AVIC_APIC_BAR_MASK 0xFFFFFFFFFF000ULL
......
...@@ -27,19 +27,38 @@ ...@@ -27,19 +27,38 @@
#include "irq.h" #include "irq.h"
#include "svm.h" #include "svm.h"
/* AVIC GATAG is encoded using VM and VCPU IDs */ /*
#define AVIC_VCPU_ID_BITS 8 * Encode the arbitrary VM ID and the vCPU's default APIC ID, i.e the vCPU ID,
#define AVIC_VCPU_ID_MASK ((1 << AVIC_VCPU_ID_BITS) - 1) * into the GATag so that KVM can retrieve the correct vCPU from a GALog entry
* if an interrupt can't be delivered, e.g. because the vCPU isn't running.
*
* For the vCPU ID, use however many bits are currently allowed for the max
* guest physical APIC ID (limited by the size of the physical ID table), and
* use whatever bits remain to assign arbitrary AVIC IDs to VMs. Note, the
* size of the GATag is defined by hardware (32 bits), but is an opaque value
* as far as hardware is concerned.
*/
#define AVIC_VCPU_ID_MASK AVIC_PHYSICAL_MAX_INDEX_MASK
#define AVIC_VM_ID_BITS 24 #define AVIC_VM_ID_SHIFT HWEIGHT32(AVIC_PHYSICAL_MAX_INDEX_MASK)
#define AVIC_VM_ID_NR (1 << AVIC_VM_ID_BITS) #define AVIC_VM_ID_MASK (GENMASK(31, AVIC_VM_ID_SHIFT) >> AVIC_VM_ID_SHIFT)
#define AVIC_VM_ID_MASK ((1 << AVIC_VM_ID_BITS) - 1)
#define AVIC_GATAG(x, y) (((x & AVIC_VM_ID_MASK) << AVIC_VCPU_ID_BITS) | \ #define AVIC_GATAG_TO_VMID(x) ((x >> AVIC_VM_ID_SHIFT) & AVIC_VM_ID_MASK)
(y & AVIC_VCPU_ID_MASK))
#define AVIC_GATAG_TO_VMID(x) ((x >> AVIC_VCPU_ID_BITS) & AVIC_VM_ID_MASK)
#define AVIC_GATAG_TO_VCPUID(x) (x & AVIC_VCPU_ID_MASK) #define AVIC_GATAG_TO_VCPUID(x) (x & AVIC_VCPU_ID_MASK)
#define __AVIC_GATAG(vm_id, vcpu_id) ((((vm_id) & AVIC_VM_ID_MASK) << AVIC_VM_ID_SHIFT) | \
((vcpu_id) & AVIC_VCPU_ID_MASK))
#define AVIC_GATAG(vm_id, vcpu_id) \
({ \
u32 ga_tag = __AVIC_GATAG(vm_id, vcpu_id); \
\
WARN_ON_ONCE(AVIC_GATAG_TO_VCPUID(ga_tag) != (vcpu_id)); \
WARN_ON_ONCE(AVIC_GATAG_TO_VMID(ga_tag) != (vm_id)); \
ga_tag; \
})
static_assert(__AVIC_GATAG(AVIC_VM_ID_MASK, AVIC_VCPU_ID_MASK) == -1u);
static bool force_avic; static bool force_avic;
module_param_unsafe(force_avic, bool, 0444); module_param_unsafe(force_avic, bool, 0444);
......
...@@ -2903,7 +2903,7 @@ static int nested_vmx_check_address_space_size(struct kvm_vcpu *vcpu, ...@@ -2903,7 +2903,7 @@ static int nested_vmx_check_address_space_size(struct kvm_vcpu *vcpu,
static int nested_vmx_check_host_state(struct kvm_vcpu *vcpu, static int nested_vmx_check_host_state(struct kvm_vcpu *vcpu,
struct vmcs12 *vmcs12) struct vmcs12 *vmcs12)
{ {
bool ia32e; bool ia32e = !!(vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE);
if (CC(!nested_host_cr0_valid(vcpu, vmcs12->host_cr0)) || if (CC(!nested_host_cr0_valid(vcpu, vmcs12->host_cr0)) ||
CC(!nested_host_cr4_valid(vcpu, vmcs12->host_cr4)) || CC(!nested_host_cr4_valid(vcpu, vmcs12->host_cr4)) ||
...@@ -2923,12 +2923,6 @@ static int nested_vmx_check_host_state(struct kvm_vcpu *vcpu, ...@@ -2923,12 +2923,6 @@ static int nested_vmx_check_host_state(struct kvm_vcpu *vcpu,
vmcs12->host_ia32_perf_global_ctrl))) vmcs12->host_ia32_perf_global_ctrl)))
return -EINVAL; return -EINVAL;
#ifdef CONFIG_X86_64
ia32e = !!(vmcs12->vm_exit_controls & VM_EXIT_HOST_ADDR_SPACE_SIZE);
#else
ia32e = false;
#endif
if (ia32e) { if (ia32e) {
if (CC(!(vmcs12->host_cr4 & X86_CR4_PAE))) if (CC(!(vmcs12->host_cr4 & X86_CR4_PAE)))
return -EINVAL; return -EINVAL;
...@@ -3022,7 +3016,7 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu, ...@@ -3022,7 +3016,7 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu,
struct vmcs12 *vmcs12, struct vmcs12 *vmcs12,
enum vm_entry_failure_code *entry_failure_code) enum vm_entry_failure_code *entry_failure_code)
{ {
bool ia32e; bool ia32e = !!(vmcs12->vm_entry_controls & VM_ENTRY_IA32E_MODE);
*entry_failure_code = ENTRY_FAIL_DEFAULT; *entry_failure_code = ENTRY_FAIL_DEFAULT;
...@@ -3048,6 +3042,13 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu, ...@@ -3048,6 +3042,13 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu,
vmcs12->guest_ia32_perf_global_ctrl))) vmcs12->guest_ia32_perf_global_ctrl)))
return -EINVAL; return -EINVAL;
if (CC((vmcs12->guest_cr0 & (X86_CR0_PG | X86_CR0_PE)) == X86_CR0_PG))
return -EINVAL;
if (CC(ia32e && !(vmcs12->guest_cr4 & X86_CR4_PAE)) ||
CC(ia32e && !(vmcs12->guest_cr0 & X86_CR0_PG)))
return -EINVAL;
/* /*
* If the load IA32_EFER VM-entry control is 1, the following checks * If the load IA32_EFER VM-entry control is 1, the following checks
* are performed on the field for the IA32_EFER MSR: * are performed on the field for the IA32_EFER MSR:
...@@ -3059,7 +3060,6 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu, ...@@ -3059,7 +3060,6 @@ static int nested_vmx_check_guest_state(struct kvm_vcpu *vcpu,
*/ */
if (to_vmx(vcpu)->nested.nested_run_pending && if (to_vmx(vcpu)->nested.nested_run_pending &&
(vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_EFER)) { (vmcs12->vm_entry_controls & VM_ENTRY_LOAD_IA32_EFER)) {
ia32e = (vmcs12->vm_entry_controls & VM_ENTRY_IA32E_MODE) != 0;
if (CC(!kvm_valid_efer(vcpu, vmcs12->guest_ia32_efer)) || if (CC(!kvm_valid_efer(vcpu, vmcs12->guest_ia32_efer)) ||
CC(ia32e != !!(vmcs12->guest_ia32_efer & EFER_LMA)) || CC(ia32e != !!(vmcs12->guest_ia32_efer & EFER_LMA)) ||
CC(((vmcs12->guest_cr0 & X86_CR0_PG) && CC(((vmcs12->guest_cr0 & X86_CR0_PG) &&
......
...@@ -262,7 +262,7 @@ SYM_INNER_LABEL(vmx_vmexit, SYM_L_GLOBAL) ...@@ -262,7 +262,7 @@ SYM_INNER_LABEL(vmx_vmexit, SYM_L_GLOBAL)
* eIBRS has its own protection against poisoned RSB, so it doesn't * eIBRS has its own protection against poisoned RSB, so it doesn't
* need the RSB filling sequence. But it does need to be enabled, and a * need the RSB filling sequence. But it does need to be enabled, and a
* single call to retire, before the first unbalanced RET. * single call to retire, before the first unbalanced RET.
*/ */
FILL_RETURN_BUFFER %_ASM_CX, RSB_CLEAR_LOOPS, X86_FEATURE_RSB_VMEXIT,\ FILL_RETURN_BUFFER %_ASM_CX, RSB_CLEAR_LOOPS, X86_FEATURE_RSB_VMEXIT,\
X86_FEATURE_RSB_VMEXIT_LITE X86_FEATURE_RSB_VMEXIT_LITE
...@@ -311,7 +311,7 @@ SYM_FUNC_END(vmx_do_nmi_irqoff) ...@@ -311,7 +311,7 @@ SYM_FUNC_END(vmx_do_nmi_irqoff)
* vmread_error_trampoline - Trampoline from inline asm to vmread_error() * vmread_error_trampoline - Trampoline from inline asm to vmread_error()
* @field: VMCS field encoding that failed * @field: VMCS field encoding that failed
* @fault: %true if the VMREAD faulted, %false if it failed * @fault: %true if the VMREAD faulted, %false if it failed
*
* Save and restore volatile registers across a call to vmread_error(). Note, * Save and restore volatile registers across a call to vmread_error(). Note,
* all parameters are passed on the stack. * all parameters are passed on the stack.
*/ */
......
...@@ -874,7 +874,7 @@ void vmx_update_exception_bitmap(struct kvm_vcpu *vcpu) ...@@ -874,7 +874,7 @@ void vmx_update_exception_bitmap(struct kvm_vcpu *vcpu)
*/ */
if (is_guest_mode(vcpu)) if (is_guest_mode(vcpu))
eb |= get_vmcs12(vcpu)->exception_bitmap; eb |= get_vmcs12(vcpu)->exception_bitmap;
else { else {
int mask = 0, match = 0; int mask = 0, match = 0;
if (enable_ept && (eb & (1u << PF_VECTOR))) { if (enable_ept && (eb & (1u << PF_VECTOR))) {
...@@ -1282,7 +1282,7 @@ void vmx_prepare_switch_to_guest(struct kvm_vcpu *vcpu) ...@@ -1282,7 +1282,7 @@ void vmx_prepare_switch_to_guest(struct kvm_vcpu *vcpu)
} }
} }
if (vmx->nested.need_vmcs12_to_shadow_sync) if (vmx->nested.need_vmcs12_to_shadow_sync)
nested_sync_vmcs12_to_shadow(vcpu); nested_sync_vmcs12_to_shadow(vcpu);
if (vmx->guest_state_loaded) if (vmx->guest_state_loaded)
...@@ -5049,10 +5049,10 @@ static int vmx_interrupt_allowed(struct kvm_vcpu *vcpu, bool for_injection) ...@@ -5049,10 +5049,10 @@ static int vmx_interrupt_allowed(struct kvm_vcpu *vcpu, bool for_injection)
if (to_vmx(vcpu)->nested.nested_run_pending) if (to_vmx(vcpu)->nested.nested_run_pending)
return -EBUSY; return -EBUSY;
/* /*
* An IRQ must not be injected into L2 if it's supposed to VM-Exit, * An IRQ must not be injected into L2 if it's supposed to VM-Exit,
* e.g. if the IRQ arrived asynchronously after checking nested events. * e.g. if the IRQ arrived asynchronously after checking nested events.
*/ */
if (for_injection && is_guest_mode(vcpu) && nested_exit_on_intr(vcpu)) if (for_injection && is_guest_mode(vcpu) && nested_exit_on_intr(vcpu))
return -EBUSY; return -EBUSY;
......
...@@ -23,6 +23,19 @@ enum kvm_arch_timer_regs { ...@@ -23,6 +23,19 @@ enum kvm_arch_timer_regs {
TIMER_REG_CTL, TIMER_REG_CTL,
}; };
struct arch_timer_offset {
/*
* If set, pointer to one of the offsets in the kvm's offset
* structure. If NULL, assume a zero offset.
*/
u64 *vm_offset;
};
struct arch_timer_vm_data {
/* Offset applied to the virtual timer/counter */
u64 voffset;
};
struct arch_timer_context { struct arch_timer_context {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
...@@ -32,6 +45,8 @@ struct arch_timer_context { ...@@ -32,6 +45,8 @@ struct arch_timer_context {
/* Emulated Timer (may be unused) */ /* Emulated Timer (may be unused) */
struct hrtimer hrtimer; struct hrtimer hrtimer;
/* Offset for this counter/timer */
struct arch_timer_offset offset;
/* /*
* We have multiple paths which can save/restore the timer state onto * We have multiple paths which can save/restore the timer state onto
* the hardware, so we need some way of keeping track of where the * the hardware, so we need some way of keeping track of where the
......
...@@ -180,9 +180,7 @@ static void host_test_system_suspend(void) ...@@ -180,9 +180,7 @@ static void host_test_system_suspend(void)
enter_guest(source); enter_guest(source);
TEST_ASSERT(run->exit_reason == KVM_EXIT_SYSTEM_EVENT, TEST_ASSERT_KVM_EXIT_REASON(source, KVM_EXIT_SYSTEM_EVENT);
"Unhandled exit reason: %u (%s)",
run->exit_reason, exit_reason_str(run->exit_reason));
TEST_ASSERT(run->system_event.type == KVM_SYSTEM_EVENT_SUSPEND, TEST_ASSERT(run->system_event.type == KVM_SYSTEM_EVENT_SUSPEND,
"Unhandled system event: %u (expected: %u)", "Unhandled system event: %u (expected: %u)",
run->system_event.type, KVM_SYSTEM_EVENT_SUSPEND); run->system_event.type, KVM_SYSTEM_EVENT_SUSPEND);
......
...@@ -63,6 +63,15 @@ void test_assert(bool exp, const char *exp_str, ...@@ -63,6 +63,15 @@ void test_assert(bool exp, const char *exp_str,
#a, #b, #a, (unsigned long) __a, #b, (unsigned long) __b); \ #a, #b, #a, (unsigned long) __a, #b, (unsigned long) __b); \
} while (0) } while (0)
#define TEST_ASSERT_KVM_EXIT_REASON(vcpu, expected) do { \
__u32 exit_reason = (vcpu)->run->exit_reason; \
\
TEST_ASSERT(exit_reason == (expected), \
"Wanted KVM exit reason: %u (%s), got: %u (%s)", \
(expected), exit_reason_str((expected)), \
exit_reason, exit_reason_str(exit_reason)); \
} while (0)
#define TEST_FAIL(fmt, ...) do { \ #define TEST_FAIL(fmt, ...) do { \
TEST_ASSERT(false, fmt, ##__VA_ARGS__); \ TEST_ASSERT(false, fmt, ##__VA_ARGS__); \
__builtin_unreachable(); \ __builtin_unreachable(); \
......
...@@ -1063,6 +1063,8 @@ uint64_t *vm_get_page_table_entry(struct kvm_vm *vm, uint64_t vaddr); ...@@ -1063,6 +1063,8 @@ uint64_t *vm_get_page_table_entry(struct kvm_vm *vm, uint64_t vaddr);
uint64_t kvm_hypercall(uint64_t nr, uint64_t a0, uint64_t a1, uint64_t a2, uint64_t kvm_hypercall(uint64_t nr, uint64_t a0, uint64_t a1, uint64_t a2,
uint64_t a3); uint64_t a3);
uint64_t __xen_hypercall(uint64_t nr, uint64_t a0, void *a1);
void xen_hypercall(uint64_t nr, uint64_t a0, void *a1);
void __vm_xsave_require_permission(int bit, const char *name); void __vm_xsave_require_permission(int bit, const char *name);
......
...@@ -1815,38 +1815,53 @@ void vm_dump(FILE *stream, struct kvm_vm *vm, uint8_t indent) ...@@ -1815,38 +1815,53 @@ void vm_dump(FILE *stream, struct kvm_vm *vm, uint8_t indent)
vcpu_dump(stream, vcpu, indent + 2); vcpu_dump(stream, vcpu, indent + 2);
} }
#define KVM_EXIT_STRING(x) {KVM_EXIT_##x, #x}
/* Known KVM exit reasons */ /* Known KVM exit reasons */
static struct exit_reason { static struct exit_reason {
unsigned int reason; unsigned int reason;
const char *name; const char *name;
} exit_reasons_known[] = { } exit_reasons_known[] = {
{KVM_EXIT_UNKNOWN, "UNKNOWN"}, KVM_EXIT_STRING(UNKNOWN),
{KVM_EXIT_EXCEPTION, "EXCEPTION"}, KVM_EXIT_STRING(EXCEPTION),
{KVM_EXIT_IO, "IO"}, KVM_EXIT_STRING(IO),
{KVM_EXIT_HYPERCALL, "HYPERCALL"}, KVM_EXIT_STRING(HYPERCALL),
{KVM_EXIT_DEBUG, "DEBUG"}, KVM_EXIT_STRING(DEBUG),
{KVM_EXIT_HLT, "HLT"}, KVM_EXIT_STRING(HLT),
{KVM_EXIT_MMIO, "MMIO"}, KVM_EXIT_STRING(MMIO),
{KVM_EXIT_IRQ_WINDOW_OPEN, "IRQ_WINDOW_OPEN"}, KVM_EXIT_STRING(IRQ_WINDOW_OPEN),
{KVM_EXIT_SHUTDOWN, "SHUTDOWN"}, KVM_EXIT_STRING(SHUTDOWN),
{KVM_EXIT_FAIL_ENTRY, "FAIL_ENTRY"}, KVM_EXIT_STRING(FAIL_ENTRY),
{KVM_EXIT_INTR, "INTR"}, KVM_EXIT_STRING(INTR),
{KVM_EXIT_SET_TPR, "SET_TPR"}, KVM_EXIT_STRING(SET_TPR),
{KVM_EXIT_TPR_ACCESS, "TPR_ACCESS"}, KVM_EXIT_STRING(TPR_ACCESS),
{KVM_EXIT_S390_SIEIC, "S390_SIEIC"}, KVM_EXIT_STRING(S390_SIEIC),
{KVM_EXIT_S390_RESET, "S390_RESET"}, KVM_EXIT_STRING(S390_RESET),
{KVM_EXIT_DCR, "DCR"}, KVM_EXIT_STRING(DCR),
{KVM_EXIT_NMI, "NMI"}, KVM_EXIT_STRING(NMI),
{KVM_EXIT_INTERNAL_ERROR, "INTERNAL_ERROR"}, KVM_EXIT_STRING(INTERNAL_ERROR),
{KVM_EXIT_OSI, "OSI"}, KVM_EXIT_STRING(OSI),
{KVM_EXIT_PAPR_HCALL, "PAPR_HCALL"}, KVM_EXIT_STRING(PAPR_HCALL),
{KVM_EXIT_DIRTY_RING_FULL, "DIRTY_RING_FULL"}, KVM_EXIT_STRING(S390_UCONTROL),
{KVM_EXIT_X86_RDMSR, "RDMSR"}, KVM_EXIT_STRING(WATCHDOG),
{KVM_EXIT_X86_WRMSR, "WRMSR"}, KVM_EXIT_STRING(S390_TSCH),
{KVM_EXIT_XEN, "XEN"}, KVM_EXIT_STRING(EPR),
{KVM_EXIT_HYPERV, "HYPERV"}, KVM_EXIT_STRING(SYSTEM_EVENT),
KVM_EXIT_STRING(S390_STSI),
KVM_EXIT_STRING(IOAPIC_EOI),
KVM_EXIT_STRING(HYPERV),
KVM_EXIT_STRING(ARM_NISV),
KVM_EXIT_STRING(X86_RDMSR),
KVM_EXIT_STRING(X86_WRMSR),
KVM_EXIT_STRING(DIRTY_RING_FULL),
KVM_EXIT_STRING(AP_RESET_HOLD),
KVM_EXIT_STRING(X86_BUS_LOCK),
KVM_EXIT_STRING(XEN),
KVM_EXIT_STRING(RISCV_SBI),
KVM_EXIT_STRING(RISCV_CSR),
KVM_EXIT_STRING(NOTIFY),
#ifdef KVM_EXIT_MEMORY_NOT_PRESENT #ifdef KVM_EXIT_MEMORY_NOT_PRESENT
{KVM_EXIT_MEMORY_NOT_PRESENT, "MEMORY_NOT_PRESENT"}, KVM_EXIT_STRING(MEMORY_NOT_PRESENT),
#endif #endif
}; };
......
...@@ -35,8 +35,7 @@ static uint64_t diag318_handler(void) ...@@ -35,8 +35,7 @@ static uint64_t diag318_handler(void)
vcpu_run(vcpu); vcpu_run(vcpu);
run = vcpu->run; run = vcpu->run;
TEST_ASSERT(run->exit_reason == KVM_EXIT_S390_SIEIC, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_S390_SIEIC);
"DIAGNOSE 0x0318 instruction was not intercepted");
TEST_ASSERT(run->s390_sieic.icptcode == ICPT_INSTRUCTION, TEST_ASSERT(run->s390_sieic.icptcode == ICPT_INSTRUCTION,
"Unexpected intercept code: 0x%x", run->s390_sieic.icptcode); "Unexpected intercept code: 0x%x", run->s390_sieic.icptcode);
TEST_ASSERT((run->s390_sieic.ipa & 0xff00) == IPA0_DIAG, TEST_ASSERT((run->s390_sieic.ipa & 0xff00) == IPA0_DIAG,
......
...@@ -165,26 +165,33 @@ size_t get_trans_hugepagesz(void) ...@@ -165,26 +165,33 @@ size_t get_trans_hugepagesz(void)
size_t get_def_hugetlb_pagesz(void) size_t get_def_hugetlb_pagesz(void)
{ {
char buf[64]; char buf[64];
const char *tag = "Hugepagesize:"; const char *hugepagesize = "Hugepagesize:";
const char *hugepages_total = "HugePages_Total:";
FILE *f; FILE *f;
f = fopen("/proc/meminfo", "r"); f = fopen("/proc/meminfo", "r");
TEST_ASSERT(f != NULL, "Error in opening /proc/meminfo"); TEST_ASSERT(f != NULL, "Error in opening /proc/meminfo");
while (fgets(buf, sizeof(buf), f) != NULL) { while (fgets(buf, sizeof(buf), f) != NULL) {
if (strstr(buf, tag) == buf) { if (strstr(buf, hugepages_total) == buf) {
unsigned long long total = strtoull(buf + strlen(hugepages_total), NULL, 10);
if (!total) {
fprintf(stderr, "HUGETLB is not enabled in /proc/sys/vm/nr_hugepages\n");
exit(KSFT_SKIP);
}
}
if (strstr(buf, hugepagesize) == buf) {
fclose(f); fclose(f);
return strtoull(buf + strlen(tag), NULL, 10) << 10; return strtoull(buf + strlen(hugepagesize), NULL, 10) << 10;
} }
} }
if (feof(f)) if (feof(f)) {
TEST_FAIL("HUGETLB is not configured in host kernel"); fprintf(stderr, "HUGETLB is not configured in host kernel");
else exit(KSFT_SKIP);
TEST_FAIL("Error in reading /proc/meminfo"); }
fclose(f); TEST_FAIL("Error in reading /proc/meminfo");
return 0;
} }
#define ANON_FLAGS (MAP_PRIVATE | MAP_ANONYMOUS) #define ANON_FLAGS (MAP_PRIVATE | MAP_ANONYMOUS)
......
...@@ -1139,21 +1139,36 @@ const struct kvm_cpuid_entry2 *get_cpuid_entry(const struct kvm_cpuid2 *cpuid, ...@@ -1139,21 +1139,36 @@ const struct kvm_cpuid_entry2 *get_cpuid_entry(const struct kvm_cpuid2 *cpuid,
return NULL; return NULL;
} }
#define X86_HYPERCALL(inputs...) \
({ \
uint64_t r; \
\
asm volatile("test %[use_vmmcall], %[use_vmmcall]\n\t" \
"jnz 1f\n\t" \
"vmcall\n\t" \
"jmp 2f\n\t" \
"1: vmmcall\n\t" \
"2:" \
: "=a"(r) \
: [use_vmmcall] "r" (host_cpu_is_amd), inputs); \
\
r; \
})
uint64_t kvm_hypercall(uint64_t nr, uint64_t a0, uint64_t a1, uint64_t a2, uint64_t kvm_hypercall(uint64_t nr, uint64_t a0, uint64_t a1, uint64_t a2,
uint64_t a3) uint64_t a3)
{ {
uint64_t r; return X86_HYPERCALL("a"(nr), "b"(a0), "c"(a1), "d"(a2), "S"(a3));
}
asm volatile("test %[use_vmmcall], %[use_vmmcall]\n\t"
"jnz 1f\n\t" uint64_t __xen_hypercall(uint64_t nr, uint64_t a0, void *a1)
"vmcall\n\t" {
"jmp 2f\n\t" return X86_HYPERCALL("a"(nr), "D"(a0), "S"(a1));
"1: vmmcall\n\t" }
"2:"
: "=a"(r) void xen_hypercall(uint64_t nr, uint64_t a0, void *a1)
: "a"(nr), "b"(a0), "c"(a1), "d"(a2), "S"(a3), {
[use_vmmcall] "r" (host_cpu_is_amd)); GUEST_ASSERT(!__xen_hypercall(nr, a0, a1));
return r;
} }
const struct kvm_cpuid2 *kvm_get_supported_hv_cpuid(void) const struct kvm_cpuid2 *kvm_get_supported_hv_cpuid(void)
......
...@@ -126,10 +126,7 @@ void test_req_and_verify_all_valid_regs(struct kvm_vcpu *vcpu) ...@@ -126,10 +126,7 @@ void test_req_and_verify_all_valid_regs(struct kvm_vcpu *vcpu)
run->kvm_valid_regs = TEST_SYNC_FIELDS; run->kvm_valid_regs = TEST_SYNC_FIELDS;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv); TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv);
TEST_ASSERT(run->exit_reason == KVM_EXIT_S390_SIEIC, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_S390_SIEIC);
"Unexpected exit reason: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s390_sieic.icptcode == 4 && TEST_ASSERT(run->s390_sieic.icptcode == 4 &&
(run->s390_sieic.ipa >> 8) == 0x83 && (run->s390_sieic.ipa >> 8) == 0x83 &&
(run->s390_sieic.ipb >> 16) == 0x501, (run->s390_sieic.ipb >> 16) == 0x501,
...@@ -165,10 +162,7 @@ void test_set_and_verify_various_reg_values(struct kvm_vcpu *vcpu) ...@@ -165,10 +162,7 @@ void test_set_and_verify_various_reg_values(struct kvm_vcpu *vcpu)
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv); TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv);
TEST_ASSERT(run->exit_reason == KVM_EXIT_S390_SIEIC, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_S390_SIEIC);
"Unexpected exit reason: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.gprs[11] == 0xBAD1DEA + 1, TEST_ASSERT(run->s.regs.gprs[11] == 0xBAD1DEA + 1,
"r11 sync regs value incorrect 0x%llx.", "r11 sync regs value incorrect 0x%llx.",
run->s.regs.gprs[11]); run->s.regs.gprs[11]);
...@@ -200,10 +194,7 @@ void test_clear_kvm_dirty_regs_bits(struct kvm_vcpu *vcpu) ...@@ -200,10 +194,7 @@ void test_clear_kvm_dirty_regs_bits(struct kvm_vcpu *vcpu)
run->s.regs.diag318 = 0x4B1D; run->s.regs.diag318 = 0x4B1D;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv); TEST_ASSERT(rv == 0, "vcpu_run failed: %d\n", rv);
TEST_ASSERT(run->exit_reason == KVM_EXIT_S390_SIEIC, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_S390_SIEIC);
"Unexpected exit reason: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.gprs[11] != 0xDEADBEEF, TEST_ASSERT(run->s.regs.gprs[11] != 0xDEADBEEF,
"r11 sync regs value incorrect 0x%llx.", "r11 sync regs value incorrect 0x%llx.",
run->s.regs.gprs[11]); run->s.regs.gprs[11]);
......
...@@ -308,7 +308,6 @@ static void test_delete_memory_region(void) ...@@ -308,7 +308,6 @@ static void test_delete_memory_region(void)
static void test_zero_memory_regions(void) static void test_zero_memory_regions(void)
{ {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
struct kvm_vm *vm; struct kvm_vm *vm;
pr_info("Testing KVM_RUN with zero added memory regions\n"); pr_info("Testing KVM_RUN with zero added memory regions\n");
...@@ -318,10 +317,7 @@ static void test_zero_memory_regions(void) ...@@ -318,10 +317,7 @@ static void test_zero_memory_regions(void)
vm_ioctl(vm, KVM_SET_NR_MMU_PAGES, (void *)64ul); vm_ioctl(vm, KVM_SET_NR_MMU_PAGES, (void *)64ul);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);
run = vcpu->run;
TEST_ASSERT(run->exit_reason == KVM_EXIT_INTERNAL_ERROR,
"Unexpected exit_reason = %u\n", run->exit_reason);
kvm_vm_free(vm); kvm_vm_free(vm);
} }
......
...@@ -241,7 +241,6 @@ int main(int argc, char *argv[]) ...@@ -241,7 +241,6 @@ int main(int argc, char *argv[])
struct kvm_regs regs1, regs2; struct kvm_regs regs1, regs2;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct kvm_x86_state *state; struct kvm_x86_state *state;
int xsave_restore_size; int xsave_restore_size;
vm_vaddr_t amx_cfg, tiledata, xsavedata; vm_vaddr_t amx_cfg, tiledata, xsavedata;
...@@ -268,7 +267,6 @@ int main(int argc, char *argv[]) ...@@ -268,7 +267,6 @@ int main(int argc, char *argv[])
"KVM should enumerate max XSAVE size when XSAVE is supported"); "KVM should enumerate max XSAVE size when XSAVE is supported");
xsave_restore_size = kvm_cpu_property(X86_PROPERTY_XSTATE_MAX_SIZE); xsave_restore_size = kvm_cpu_property(X86_PROPERTY_XSTATE_MAX_SIZE);
run = vcpu->run;
vcpu_regs_get(vcpu, &regs1); vcpu_regs_get(vcpu, &regs1);
/* Register #NM handler */ /* Register #NM handler */
...@@ -291,10 +289,7 @@ int main(int argc, char *argv[]) ...@@ -291,10 +289,7 @@ int main(int argc, char *argv[])
for (stage = 1; ; stage++) { for (stage = 1; ; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
...@@ -350,7 +345,6 @@ int main(int argc, char *argv[]) ...@@ -350,7 +345,6 @@ int main(int argc, char *argv[])
/* Restore state in a new VM. */ /* Restore state in a new VM. */
vcpu = vm_recreate_with_one_vcpu(vm); vcpu = vm_recreate_with_one_vcpu(vm);
vcpu_load_state(vcpu, state); vcpu_load_state(vcpu, state);
run = vcpu->run;
kvm_x86_state_cleanup(state); kvm_x86_state_cleanup(state);
memset(&regs2, 0, sizeof(regs2)); memset(&regs2, 0, sizeof(regs2));
......
...@@ -50,7 +50,6 @@ static void guest_code(void) ...@@ -50,7 +50,6 @@ static void guest_code(void)
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_sregs sregs; struct kvm_sregs sregs;
struct ucall uc; struct ucall uc;
...@@ -58,15 +57,10 @@ int main(int argc, char *argv[]) ...@@ -58,15 +57,10 @@ int main(int argc, char *argv[])
TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XSAVE)); TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XSAVE));
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
run = vcpu->run;
while (1) { while (1) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO,
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -204,7 +204,7 @@ int main(void) ...@@ -204,7 +204,7 @@ int main(void)
vcpu_guest_debug_set(vcpu, &debug); vcpu_guest_debug_set(vcpu, &debug);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, "KVM_EXIT_IO"); TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
cmd = get_ucall(vcpu, &uc); cmd = get_ucall(vcpu, &uc);
TEST_ASSERT(cmd == UCALL_DONE, "UCALL_DONE"); TEST_ASSERT(cmd == UCALL_DONE, "UCALL_DONE");
......
...@@ -24,10 +24,7 @@ static inline void handle_flds_emulation_failure_exit(struct kvm_vcpu *vcpu) ...@@ -24,10 +24,7 @@ static inline void handle_flds_emulation_failure_exit(struct kvm_vcpu *vcpu)
uint8_t *insn_bytes; uint8_t *insn_bytes;
uint64_t flags; uint64_t flags;
TEST_ASSERT(run->exit_reason == KVM_EXIT_INTERNAL_ERROR, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);
"Unexpected exit reason: %u (%s)",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION, TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION,
"Unexpected suberror: %u", "Unexpected suberror: %u",
......
...@@ -207,13 +207,11 @@ int main(void) ...@@ -207,13 +207,11 @@ int main(void)
{ {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct ucall uc; struct ucall uc;
vm_vaddr_t tsc_page_gva; vm_vaddr_t tsc_page_gva;
int stage; int stage;
vm = vm_create_with_one_vcpu(&vcpu, guest_main); vm = vm_create_with_one_vcpu(&vcpu, guest_main);
run = vcpu->run;
vcpu_set_hv_cpuid(vcpu); vcpu_set_hv_cpuid(vcpu);
...@@ -227,10 +225,7 @@ int main(void) ...@@ -227,10 +225,7 @@ int main(void)
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -237,7 +237,6 @@ int main(int argc, char *argv[]) ...@@ -237,7 +237,6 @@ int main(int argc, char *argv[])
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct ucall uc; struct ucall uc;
int stage; int stage;
...@@ -266,13 +265,8 @@ int main(int argc, char *argv[]) ...@@ -266,13 +265,8 @@ int main(int argc, char *argv[])
pr_info("Running L1 which uses EVMCS to run L2\n"); pr_info("Running L1 which uses EVMCS to run L2\n");
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
run = vcpu->run;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -122,7 +122,6 @@ static void guest_test_msrs_access(void) ...@@ -122,7 +122,6 @@ static void guest_test_msrs_access(void)
{ {
struct kvm_cpuid2 *prev_cpuid = NULL; struct kvm_cpuid2 *prev_cpuid = NULL;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
struct kvm_vm *vm; struct kvm_vm *vm;
struct ucall uc; struct ucall uc;
int stage = 0; int stage = 0;
...@@ -151,8 +150,6 @@ static void guest_test_msrs_access(void) ...@@ -151,8 +150,6 @@ static void guest_test_msrs_access(void)
vm_init_descriptor_tables(vm); vm_init_descriptor_tables(vm);
vcpu_init_descriptor_tables(vcpu); vcpu_init_descriptor_tables(vcpu);
run = vcpu->run;
/* TODO: Make this entire test easier to maintain. */ /* TODO: Make this entire test easier to maintain. */
if (stage >= 21) if (stage >= 21)
vcpu_enable_cap(vcpu, KVM_CAP_HYPERV_SYNIC2, 0); vcpu_enable_cap(vcpu, KVM_CAP_HYPERV_SYNIC2, 0);
...@@ -494,9 +491,7 @@ static void guest_test_msrs_access(void) ...@@ -494,9 +491,7 @@ static void guest_test_msrs_access(void)
msr->idx, msr->write ? "write" : "read"); msr->idx, msr->write ? "write" : "read");
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"unexpected exit reason: %u (%s)",
run->exit_reason, exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
...@@ -518,7 +513,6 @@ static void guest_test_hcalls_access(void) ...@@ -518,7 +513,6 @@ static void guest_test_hcalls_access(void)
{ {
struct kvm_cpuid2 *prev_cpuid = NULL; struct kvm_cpuid2 *prev_cpuid = NULL;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
struct kvm_vm *vm; struct kvm_vm *vm;
struct ucall uc; struct ucall uc;
int stage = 0; int stage = 0;
...@@ -550,8 +544,6 @@ static void guest_test_hcalls_access(void) ...@@ -550,8 +544,6 @@ static void guest_test_hcalls_access(void)
vcpu_init_cpuid(vcpu, prev_cpuid); vcpu_init_cpuid(vcpu, prev_cpuid);
} }
run = vcpu->run;
switch (stage) { switch (stage) {
case 0: case 0:
vcpu_set_cpuid_feature(vcpu, HV_MSR_HYPERCALL_AVAILABLE); vcpu_set_cpuid_feature(vcpu, HV_MSR_HYPERCALL_AVAILABLE);
...@@ -669,9 +661,7 @@ static void guest_test_hcalls_access(void) ...@@ -669,9 +661,7 @@ static void guest_test_hcalls_access(void)
pr_debug("Stage %d: testing hcall: 0x%lx\n", stage, hcall->control); pr_debug("Stage %d: testing hcall: 0x%lx\n", stage, hcall->control);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"unexpected exit reason: %u (%s)",
run->exit_reason, exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -243,7 +243,6 @@ int main(int argc, char *argv[]) ...@@ -243,7 +243,6 @@ int main(int argc, char *argv[])
{ {
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_vcpu *vcpu[3]; struct kvm_vcpu *vcpu[3];
unsigned int exit_reason;
vm_vaddr_t hcall_page; vm_vaddr_t hcall_page;
pthread_t threads[2]; pthread_t threads[2];
int stage = 1, r; int stage = 1, r;
...@@ -283,10 +282,7 @@ int main(int argc, char *argv[]) ...@@ -283,10 +282,7 @@ int main(int argc, char *argv[])
while (true) { while (true) {
vcpu_run(vcpu[0]); vcpu_run(vcpu[0]);
exit_reason = vcpu[0]->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(vcpu[0], KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"unexpected exit reason: %u (%s)",
exit_reason, exit_reason_str(exit_reason));
switch (get_ucall(vcpu[0], &uc)) { switch (get_ucall(vcpu[0], &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -156,7 +156,6 @@ int main(int argc, char *argv[]) ...@@ -156,7 +156,6 @@ int main(int argc, char *argv[])
vm_vaddr_t hcall_page; vm_vaddr_t hcall_page;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct ucall uc; struct ucall uc;
int stage; int stage;
...@@ -165,7 +164,6 @@ int main(int argc, char *argv[]) ...@@ -165,7 +164,6 @@ int main(int argc, char *argv[])
/* Create VM */ /* Create VM */
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
vcpu_set_hv_cpuid(vcpu); vcpu_set_hv_cpuid(vcpu);
run = vcpu->run;
vcpu_alloc_svm(vm, &nested_gva); vcpu_alloc_svm(vm, &nested_gva);
vcpu_alloc_hyperv_test_pages(vm, &hv_pages_gva); vcpu_alloc_hyperv_test_pages(vm, &hv_pages_gva);
...@@ -177,10 +175,7 @@ int main(int argc, char *argv[]) ...@@ -177,10 +175,7 @@ int main(int argc, char *argv[])
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -542,18 +542,13 @@ static void *vcpu_thread(void *arg) ...@@ -542,18 +542,13 @@ static void *vcpu_thread(void *arg)
struct ucall uc; struct ucall uc;
int old; int old;
int r; int r;
unsigned int exit_reason;
r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old); r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old);
TEST_ASSERT(!r, "pthread_setcanceltype failed on vcpu_id=%u with errno=%d", TEST_ASSERT(!r, "pthread_setcanceltype failed on vcpu_id=%u with errno=%d",
vcpu->id, r); vcpu->id, r);
vcpu_run(vcpu); vcpu_run(vcpu);
exit_reason = vcpu->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"vCPU %u exited with unexpected exit reason %u-%s, expected KVM_EXIT_IO",
vcpu->id, exit_reason, exit_reason_str(exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
...@@ -587,7 +582,6 @@ int main(int argc, char *argv[]) ...@@ -587,7 +582,6 @@ int main(int argc, char *argv[])
{ {
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_vcpu *vcpu[3]; struct kvm_vcpu *vcpu[3];
unsigned int exit_reason;
pthread_t threads[2]; pthread_t threads[2];
vm_vaddr_t test_data_page, gva; vm_vaddr_t test_data_page, gva;
vm_paddr_t gpa; vm_paddr_t gpa;
...@@ -657,11 +651,7 @@ int main(int argc, char *argv[]) ...@@ -657,11 +651,7 @@ int main(int argc, char *argv[])
while (true) { while (true) {
vcpu_run(vcpu[0]); vcpu_run(vcpu[0]);
exit_reason = vcpu[0]->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(vcpu[0], KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"unexpected exit reason: %u (%s)",
exit_reason, exit_reason_str(exit_reason));
switch (get_ucall(vcpu[0], &uc)) { switch (get_ucall(vcpu[0], &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -105,7 +105,6 @@ static void setup_clock(struct kvm_vm *vm, struct test_case *test_case) ...@@ -105,7 +105,6 @@ static void setup_clock(struct kvm_vm *vm, struct test_case *test_case)
static void enter_guest(struct kvm_vcpu *vcpu) static void enter_guest(struct kvm_vcpu *vcpu)
{ {
struct kvm_clock_data start, end; struct kvm_clock_data start, end;
struct kvm_run *run = vcpu->run;
struct kvm_vm *vm = vcpu->vm; struct kvm_vm *vm = vcpu->vm;
struct ucall uc; struct ucall uc;
int i; int i;
...@@ -118,9 +117,7 @@ static void enter_guest(struct kvm_vcpu *vcpu) ...@@ -118,9 +117,7 @@ static void enter_guest(struct kvm_vcpu *vcpu)
vcpu_run(vcpu); vcpu_run(vcpu);
vm_ioctl(vm, KVM_GET_CLOCK, &end); vm_ioctl(vm, KVM_GET_CLOCK, &end);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"unexpected exit reason: %u (%s)",
run->exit_reason, exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -111,14 +111,11 @@ static void pr_hcall(struct ucall *uc) ...@@ -111,14 +111,11 @@ static void pr_hcall(struct ucall *uc)
static void enter_guest(struct kvm_vcpu *vcpu) static void enter_guest(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
while (true) { while (true) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"unexpected exit reason: %u (%s)",
run->exit_reason, exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_PR_MSR: case UCALL_PR_MSR:
......
...@@ -64,7 +64,6 @@ int main(int argc, char *argv[]) ...@@ -64,7 +64,6 @@ int main(int argc, char *argv[])
{ {
uint64_t disabled_quirks; uint64_t disabled_quirks;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
struct kvm_vm *vm; struct kvm_vm *vm;
struct ucall uc; struct ucall uc;
int testcase; int testcase;
...@@ -74,18 +73,12 @@ int main(int argc, char *argv[]) ...@@ -74,18 +73,12 @@ int main(int argc, char *argv[])
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
vcpu_clear_cpuid_feature(vcpu, X86_FEATURE_MWAIT); vcpu_clear_cpuid_feature(vcpu, X86_FEATURE_MWAIT);
run = vcpu->run;
vm_init_descriptor_tables(vm); vm_init_descriptor_tables(vm);
vcpu_init_descriptor_tables(vcpu); vcpu_init_descriptor_tables(vcpu);
while (1) { while (1) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO,
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -166,12 +166,9 @@ static void __attribute__((__flatten__)) l1_guest_code(void *test_data) ...@@ -166,12 +166,9 @@ static void __attribute__((__flatten__)) l1_guest_code(void *test_data)
static void assert_ucall_vector(struct kvm_vcpu *vcpu, int vector) static void assert_ucall_vector(struct kvm_vcpu *vcpu, int vector)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason, exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -36,15 +36,12 @@ static void guest_code(void) ...@@ -36,15 +36,12 @@ static void guest_code(void)
static void test_msr_platform_info_enabled(struct kvm_vcpu *vcpu) static void test_msr_platform_info_enabled(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vm_enable_cap(vcpu->vm, KVM_CAP_MSR_PLATFORM_INFO, true); vm_enable_cap(vcpu->vm, KVM_CAP_MSR_PLATFORM_INFO, true);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Exit_reason other than KVM_EXIT_IO: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
get_ucall(vcpu, &uc); get_ucall(vcpu, &uc);
TEST_ASSERT(uc.cmd == UCALL_SYNC, TEST_ASSERT(uc.cmd == UCALL_SYNC,
"Received ucall other than UCALL_SYNC: %lu\n", uc.cmd); "Received ucall other than UCALL_SYNC: %lu\n", uc.cmd);
...@@ -56,14 +53,9 @@ static void test_msr_platform_info_enabled(struct kvm_vcpu *vcpu) ...@@ -56,14 +53,9 @@ static void test_msr_platform_info_enabled(struct kvm_vcpu *vcpu)
static void test_msr_platform_info_disabled(struct kvm_vcpu *vcpu) static void test_msr_platform_info_disabled(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
vm_enable_cap(vcpu->vm, KVM_CAP_MSR_PLATFORM_INFO, false); vm_enable_cap(vcpu->vm, KVM_CAP_MSR_PLATFORM_INFO, false);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_SHUTDOWN, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_SHUTDOWN);
"Exit_reason other than KVM_EXIT_SHUTDOWN: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
} }
int main(int argc, char *argv[]) int main(int argc, char *argv[])
......
...@@ -151,14 +151,10 @@ static void amd_guest_code(void) ...@@ -151,14 +151,10 @@ static void amd_guest_code(void)
*/ */
static uint64_t run_vcpu_to_sync(struct kvm_vcpu *vcpu) static uint64_t run_vcpu_to_sync(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
get_ucall(vcpu, &uc); get_ucall(vcpu, &uc);
TEST_ASSERT(uc.cmd == UCALL_SYNC, TEST_ASSERT(uc.cmd == UCALL_SYNC,
"Received ucall other than UCALL_SYNC: %lu", uc.cmd); "Received ucall other than UCALL_SYNC: %lu", uc.cmd);
......
...@@ -133,7 +133,6 @@ int main(int argc, char *argv[]) ...@@ -133,7 +133,6 @@ int main(int argc, char *argv[])
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_regs regs; struct kvm_regs regs;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct kvm_x86_state *state; struct kvm_x86_state *state;
int stage, stage_reported; int stage, stage_reported;
...@@ -142,8 +141,6 @@ int main(int argc, char *argv[]) ...@@ -142,8 +141,6 @@ int main(int argc, char *argv[])
/* Create VM */ /* Create VM */
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
run = vcpu->run;
vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, SMRAM_GPA, vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, SMRAM_GPA,
SMRAM_MEMSLOT, SMRAM_PAGES, 0); SMRAM_MEMSLOT, SMRAM_PAGES, 0);
TEST_ASSERT(vm_phy_pages_alloc(vm, SMRAM_PAGES, SMRAM_GPA, SMRAM_MEMSLOT) TEST_ASSERT(vm_phy_pages_alloc(vm, SMRAM_PAGES, SMRAM_GPA, SMRAM_MEMSLOT)
...@@ -169,10 +166,7 @@ int main(int argc, char *argv[]) ...@@ -169,10 +166,7 @@ int main(int argc, char *argv[])
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
memset(&regs, 0, sizeof(regs)); memset(&regs, 0, sizeof(regs));
vcpu_regs_get(vcpu, &regs); vcpu_regs_get(vcpu, &regs);
...@@ -208,7 +202,6 @@ int main(int argc, char *argv[]) ...@@ -208,7 +202,6 @@ int main(int argc, char *argv[])
vcpu = vm_recreate_with_one_vcpu(vm); vcpu = vm_recreate_with_one_vcpu(vm);
vcpu_load_state(vcpu, state); vcpu_load_state(vcpu, state);
run = vcpu->run;
kvm_x86_state_cleanup(state); kvm_x86_state_cleanup(state);
} }
......
...@@ -158,14 +158,12 @@ int main(int argc, char *argv[]) ...@@ -158,14 +158,12 @@ int main(int argc, char *argv[])
struct kvm_regs regs1, regs2; struct kvm_regs regs1, regs2;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct kvm_x86_state *state; struct kvm_x86_state *state;
struct ucall uc; struct ucall uc;
int stage; int stage;
/* Create VM */ /* Create VM */
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
run = vcpu->run;
vcpu_regs_get(vcpu, &regs1); vcpu_regs_get(vcpu, &regs1);
...@@ -183,10 +181,7 @@ int main(int argc, char *argv[]) ...@@ -183,10 +181,7 @@ int main(int argc, char *argv[])
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
...@@ -214,7 +209,6 @@ int main(int argc, char *argv[]) ...@@ -214,7 +209,6 @@ int main(int argc, char *argv[])
/* Restore state in a new VM. */ /* Restore state in a new VM. */
vcpu = vm_recreate_with_one_vcpu(vm); vcpu = vm_recreate_with_one_vcpu(vm);
vcpu_load_state(vcpu, state); vcpu_load_state(vcpu, state);
run = vcpu->run;
kvm_x86_state_cleanup(state); kvm_x86_state_cleanup(state);
memset(&regs2, 0, sizeof(regs2)); memset(&regs2, 0, sizeof(regs2));
......
...@@ -85,7 +85,6 @@ static void l1_guest_code(struct svm_test_data *svm) ...@@ -85,7 +85,6 @@ static void l1_guest_code(struct svm_test_data *svm)
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
vm_vaddr_t svm_gva; vm_vaddr_t svm_gva;
struct kvm_vm *vm; struct kvm_vm *vm;
struct ucall uc; struct ucall uc;
...@@ -103,13 +102,8 @@ int main(int argc, char *argv[]) ...@@ -103,13 +102,8 @@ int main(int argc, char *argv[])
vcpu_alloc_svm(vm, &svm_gva); vcpu_alloc_svm(vm, &svm_gva);
vcpu_args_set(vcpu, 1, svm_gva); vcpu_args_set(vcpu, 1, svm_gva);
run = vcpu->run;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -42,7 +42,6 @@ static void l1_guest_code(struct svm_test_data *svm, struct idt_entry *idt) ...@@ -42,7 +42,6 @@ static void l1_guest_code(struct svm_test_data *svm, struct idt_entry *idt)
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_run *run;
vm_vaddr_t svm_gva; vm_vaddr_t svm_gva;
struct kvm_vm *vm; struct kvm_vm *vm;
...@@ -55,13 +54,9 @@ int main(int argc, char *argv[]) ...@@ -55,13 +54,9 @@ int main(int argc, char *argv[])
vcpu_alloc_svm(vm, &svm_gva); vcpu_alloc_svm(vm, &svm_gva);
vcpu_args_set(vcpu, 2, svm_gva, vm->idt); vcpu_args_set(vcpu, 2, svm_gva, vm->idt);
run = vcpu->run;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_SHUTDOWN, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_SHUTDOWN);
"Got exit_reason other than KVM_EXIT_SHUTDOWN: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
kvm_vm_free(vm); kvm_vm_free(vm);
} }
...@@ -176,16 +176,12 @@ static void run_test(bool is_nmi) ...@@ -176,16 +176,12 @@ static void run_test(bool is_nmi)
memset(&debug, 0, sizeof(debug)); memset(&debug, 0, sizeof(debug));
vcpu_guest_debug_set(vcpu, &debug); vcpu_guest_debug_set(vcpu, &debug);
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
alarm(2); alarm(2);
vcpu_run(vcpu); vcpu_run(vcpu);
alarm(0); alarm(0);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -47,14 +47,10 @@ int main(int argc, char *argv[]) ...@@ -47,14 +47,10 @@ int main(int argc, char *argv[])
vcpu_args_set(vcpu, 1, svm_gva); vcpu_args_set(vcpu, 1, svm_gva);
for (;;) { for (;;) {
volatile struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -132,10 +132,7 @@ int main(int argc, char *argv[]) ...@@ -132,10 +132,7 @@ int main(int argc, char *argv[])
/* TODO: BUILD TIME CHECK: TEST_ASSERT(KVM_SYNC_X86_NUM_FIELDS != 3); */ /* TODO: BUILD TIME CHECK: TEST_ASSERT(KVM_SYNC_X86_NUM_FIELDS != 3); */
run->kvm_valid_regs = TEST_SYNC_FIELDS; run->kvm_valid_regs = TEST_SYNC_FIELDS;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
vcpu_regs_get(vcpu, &regs); vcpu_regs_get(vcpu, &regs);
compare_regs(&regs, &run->s.regs.regs); compare_regs(&regs, &run->s.regs.regs);
...@@ -154,10 +151,7 @@ int main(int argc, char *argv[]) ...@@ -154,10 +151,7 @@ int main(int argc, char *argv[])
run->kvm_valid_regs = TEST_SYNC_FIELDS; run->kvm_valid_regs = TEST_SYNC_FIELDS;
run->kvm_dirty_regs = KVM_SYNC_X86_REGS | KVM_SYNC_X86_SREGS; run->kvm_dirty_regs = KVM_SYNC_X86_REGS | KVM_SYNC_X86_SREGS;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.regs.rbx == 0xBAD1DEA + 1, TEST_ASSERT(run->s.regs.regs.rbx == 0xBAD1DEA + 1,
"rbx sync regs value incorrect 0x%llx.", "rbx sync regs value incorrect 0x%llx.",
run->s.regs.regs.rbx); run->s.regs.regs.rbx);
...@@ -181,10 +175,7 @@ int main(int argc, char *argv[]) ...@@ -181,10 +175,7 @@ int main(int argc, char *argv[])
run->kvm_dirty_regs = 0; run->kvm_dirty_regs = 0;
run->s.regs.regs.rbx = 0xDEADBEEF; run->s.regs.regs.rbx = 0xDEADBEEF;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.regs.rbx != 0xDEADBEEF, TEST_ASSERT(run->s.regs.regs.rbx != 0xDEADBEEF,
"rbx sync regs value incorrect 0x%llx.", "rbx sync regs value incorrect 0x%llx.",
run->s.regs.regs.rbx); run->s.regs.regs.rbx);
...@@ -199,10 +190,7 @@ int main(int argc, char *argv[]) ...@@ -199,10 +190,7 @@ int main(int argc, char *argv[])
regs.rbx = 0xBAC0; regs.rbx = 0xBAC0;
vcpu_regs_set(vcpu, &regs); vcpu_regs_set(vcpu, &regs);
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.regs.rbx == 0xAAAA, TEST_ASSERT(run->s.regs.regs.rbx == 0xAAAA,
"rbx sync regs value incorrect 0x%llx.", "rbx sync regs value incorrect 0x%llx.",
run->s.regs.regs.rbx); run->s.regs.regs.rbx);
...@@ -219,10 +207,7 @@ int main(int argc, char *argv[]) ...@@ -219,10 +207,7 @@ int main(int argc, char *argv[])
run->kvm_dirty_regs = TEST_SYNC_FIELDS; run->kvm_dirty_regs = TEST_SYNC_FIELDS;
run->s.regs.regs.rbx = 0xBBBB; run->s.regs.regs.rbx = 0xBBBB;
rv = _vcpu_run(vcpu); rv = _vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->s.regs.regs.rbx == 0xBBBB, TEST_ASSERT(run->s.regs.regs.rbx == 0xBBBB,
"rbx sync regs value incorrect 0x%llx.", "rbx sync regs value incorrect 0x%llx.",
run->s.regs.regs.rbx); run->s.regs.regs.rbx);
......
...@@ -89,9 +89,7 @@ int main(void) ...@@ -89,9 +89,7 @@ int main(void)
run = vcpu->run; run = vcpu->run;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Expected KVM_EXIT_IO, got: %u (%s)\n",
run->exit_reason, exit_reason_str(run->exit_reason));
TEST_ASSERT(run->io.port == ARBITRARY_IO_PORT, TEST_ASSERT(run->io.port == ARBITRARY_IO_PORT,
"Expected IN from port %d from L2, got port %d", "Expected IN from port %d from L2, got port %d",
ARBITRARY_IO_PORT, run->io.port); ARBITRARY_IO_PORT, run->io.port);
...@@ -111,10 +109,7 @@ int main(void) ...@@ -111,10 +109,7 @@ int main(void)
if (has_svm) { if (has_svm) {
TEST_ASSERT(run->exit_reason == KVM_EXIT_SHUTDOWN, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_SHUTDOWN);
"Got exit_reason other than KVM_EXIT_SHUTDOWN: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
} else { } else {
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_DONE: case UCALL_DONE:
......
...@@ -64,14 +64,10 @@ static void *run_vcpu(void *_cpu_nr) ...@@ -64,14 +64,10 @@ static void *run_vcpu(void *_cpu_nr)
pthread_spin_unlock(&create_lock); pthread_spin_unlock(&create_lock);
for (;;) { for (;;) {
volatile struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_DONE: case UCALL_DONE:
......
...@@ -137,15 +137,11 @@ static void guest_gp_handler(struct ex_regs *regs) ...@@ -137,15 +137,11 @@ static void guest_gp_handler(struct ex_regs *regs)
static void run_vcpu_expect_gp(struct kvm_vcpu *vcpu) static void run_vcpu_expect_gp(struct kvm_vcpu *vcpu)
{ {
unsigned int exit_reason;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
exit_reason = vcpu->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"exited with unexpected exit reason %u-%s, expected KVM_EXIT_IO",
exit_reason, exit_reason_str(exit_reason));
TEST_ASSERT(get_ucall(vcpu, &uc) == UCALL_SYNC, TEST_ASSERT(get_ucall(vcpu, &uc) == UCALL_SYNC,
"Expect UCALL_SYNC\n"); "Expect UCALL_SYNC\n");
TEST_ASSERT(uc.args[1] == SYNC_GP, "#GP is expected."); TEST_ASSERT(uc.args[1] == SYNC_GP, "#GP is expected.");
...@@ -182,7 +178,6 @@ static void *run_ucna_injection(void *arg) ...@@ -182,7 +178,6 @@ static void *run_ucna_injection(void *arg)
struct ucall uc; struct ucall uc;
int old; int old;
int r; int r;
unsigned int exit_reason;
r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old); r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old);
TEST_ASSERT(r == 0, TEST_ASSERT(r == 0,
...@@ -191,10 +186,7 @@ static void *run_ucna_injection(void *arg) ...@@ -191,10 +186,7 @@ static void *run_ucna_injection(void *arg)
vcpu_run(params->vcpu); vcpu_run(params->vcpu);
exit_reason = params->vcpu->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(params->vcpu, KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"unexpected exit reason %u-%s, expected KVM_EXIT_IO",
exit_reason, exit_reason_str(exit_reason));
TEST_ASSERT(get_ucall(params->vcpu, &uc) == UCALL_SYNC, TEST_ASSERT(get_ucall(params->vcpu, &uc) == UCALL_SYNC,
"Expect UCALL_SYNC\n"); "Expect UCALL_SYNC\n");
TEST_ASSERT(uc.args[1] == SYNC_FIRST_UCNA, "Injecting first UCNA."); TEST_ASSERT(uc.args[1] == SYNC_FIRST_UCNA, "Injecting first UCNA.");
...@@ -204,10 +196,7 @@ static void *run_ucna_injection(void *arg) ...@@ -204,10 +196,7 @@ static void *run_ucna_injection(void *arg)
inject_ucna(params->vcpu, FIRST_UCNA_ADDR); inject_ucna(params->vcpu, FIRST_UCNA_ADDR);
vcpu_run(params->vcpu); vcpu_run(params->vcpu);
exit_reason = params->vcpu->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(params->vcpu, KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"unexpected exit reason %u-%s, expected KVM_EXIT_IO",
exit_reason, exit_reason_str(exit_reason));
TEST_ASSERT(get_ucall(params->vcpu, &uc) == UCALL_SYNC, TEST_ASSERT(get_ucall(params->vcpu, &uc) == UCALL_SYNC,
"Expect UCALL_SYNC\n"); "Expect UCALL_SYNC\n");
TEST_ASSERT(uc.args[1] == SYNC_SECOND_UCNA, "Injecting second UCNA."); TEST_ASSERT(uc.args[1] == SYNC_SECOND_UCNA, "Injecting second UCNA.");
...@@ -217,10 +206,7 @@ static void *run_ucna_injection(void *arg) ...@@ -217,10 +206,7 @@ static void *run_ucna_injection(void *arg)
inject_ucna(params->vcpu, SECOND_UCNA_ADDR); inject_ucna(params->vcpu, SECOND_UCNA_ADDR);
vcpu_run(params->vcpu); vcpu_run(params->vcpu);
exit_reason = params->vcpu->run->exit_reason; TEST_ASSERT_KVM_EXIT_REASON(params->vcpu, KVM_EXIT_IO);
TEST_ASSERT(exit_reason == KVM_EXIT_IO,
"unexpected exit reason %u-%s, expected KVM_EXIT_IO",
exit_reason, exit_reason_str(exit_reason));
if (get_ucall(params->vcpu, &uc) == UCALL_ABORT) { if (get_ucall(params->vcpu, &uc) == UCALL_ABORT) {
TEST_ASSERT(false, "vCPU assertion failure: %s.\n", TEST_ASSERT(false, "vCPU assertion failure: %s.\n",
(const char *)uc.args[0]); (const char *)uc.args[0]);
......
...@@ -63,11 +63,7 @@ int main(int argc, char *argv[]) ...@@ -63,11 +63,7 @@ int main(int argc, char *argv[])
while (1) { while (1) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO,
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
if (get_ucall(vcpu, &uc)) if (get_ucall(vcpu, &uc))
break; break;
......
...@@ -410,10 +410,7 @@ static void process_rdmsr(struct kvm_vcpu *vcpu, uint32_t msr_index) ...@@ -410,10 +410,7 @@ static void process_rdmsr(struct kvm_vcpu *vcpu, uint32_t msr_index)
check_for_guest_assert(vcpu); check_for_guest_assert(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_X86_RDMSR, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_X86_RDMSR);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->msr.index == msr_index, TEST_ASSERT(run->msr.index == msr_index,
"Unexpected msr (0x%04x), expected 0x%04x", "Unexpected msr (0x%04x), expected 0x%04x",
run->msr.index, msr_index); run->msr.index, msr_index);
...@@ -445,10 +442,7 @@ static void process_wrmsr(struct kvm_vcpu *vcpu, uint32_t msr_index) ...@@ -445,10 +442,7 @@ static void process_wrmsr(struct kvm_vcpu *vcpu, uint32_t msr_index)
check_for_guest_assert(vcpu); check_for_guest_assert(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_X86_WRMSR, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_X86_WRMSR);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->msr.index == msr_index, TEST_ASSERT(run->msr.index == msr_index,
"Unexpected msr (0x%04x), expected 0x%04x", "Unexpected msr (0x%04x), expected 0x%04x",
run->msr.index, msr_index); run->msr.index, msr_index);
...@@ -472,15 +466,11 @@ static void process_wrmsr(struct kvm_vcpu *vcpu, uint32_t msr_index) ...@@ -472,15 +466,11 @@ static void process_wrmsr(struct kvm_vcpu *vcpu, uint32_t msr_index)
static void process_ucall_done(struct kvm_vcpu *vcpu) static void process_ucall_done(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
check_for_guest_assert(vcpu); check_for_guest_assert(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s)",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(get_ucall(vcpu, &uc) == UCALL_DONE, TEST_ASSERT(get_ucall(vcpu, &uc) == UCALL_DONE,
"Unexpected ucall command: %lu, expected UCALL_DONE (%d)", "Unexpected ucall command: %lu, expected UCALL_DONE (%d)",
...@@ -489,15 +479,11 @@ static void process_ucall_done(struct kvm_vcpu *vcpu) ...@@ -489,15 +479,11 @@ static void process_ucall_done(struct kvm_vcpu *vcpu)
static uint64_t process_ucall(struct kvm_vcpu *vcpu) static uint64_t process_ucall(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *run = vcpu->run;
struct ucall uc = {}; struct ucall uc = {};
check_for_guest_assert(vcpu); check_for_guest_assert(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s)",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_SYNC: case UCALL_SYNC:
......
...@@ -96,21 +96,14 @@ int main(int argc, char *argv[]) ...@@ -96,21 +96,14 @@ int main(int argc, char *argv[])
vcpu_run(vcpu); vcpu_run(vcpu);
if (apic_access_addr == high_gpa) { if (apic_access_addr == high_gpa) {
TEST_ASSERT(run->exit_reason == TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);
KVM_EXIT_INTERNAL_ERROR,
"Got exit reason other than KVM_EXIT_INTERNAL_ERROR: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
TEST_ASSERT(run->internal.suberror == TEST_ASSERT(run->internal.suberror ==
KVM_INTERNAL_ERROR_EMULATION, KVM_INTERNAL_ERROR_EMULATION,
"Got internal suberror other than KVM_INTERNAL_ERROR_EMULATION: %u\n", "Got internal suberror other than KVM_INTERNAL_ERROR_EMULATION: %u\n",
run->internal.suberror); run->internal.suberror);
break; break;
} }
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -64,10 +64,7 @@ int main(int argc, char *argv[]) ...@@ -64,10 +64,7 @@ int main(int argc, char *argv[])
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
if (run->io.port == PORT_L0_EXIT) if (run->io.port == PORT_L0_EXIT)
break; break;
......
...@@ -73,7 +73,6 @@ int main(int argc, char *argv[]) ...@@ -73,7 +73,6 @@ int main(int argc, char *argv[])
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct ucall uc; struct ucall uc;
bool done = false; bool done = false;
...@@ -84,7 +83,6 @@ int main(int argc, char *argv[]) ...@@ -84,7 +83,6 @@ int main(int argc, char *argv[])
vm = vm_create_with_one_vcpu(&vcpu, l1_guest_code); vm = vm_create_with_one_vcpu(&vcpu, l1_guest_code);
vmx = vcpu_alloc_vmx(vm, &vmx_pages_gva); vmx = vcpu_alloc_vmx(vm, &vmx_pages_gva);
vcpu_args_set(vcpu, 1, vmx_pages_gva); vcpu_args_set(vcpu, 1, vmx_pages_gva);
run = vcpu->run;
/* Add an extra memory slot for testing dirty logging */ /* Add an extra memory slot for testing dirty logging */
vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS,
...@@ -117,10 +115,7 @@ int main(int argc, char *argv[]) ...@@ -117,10 +115,7 @@ int main(int argc, char *argv[])
while (!done) { while (!done) {
memset(host_test_mem, 0xaa, TEST_MEM_PAGES * 4096); memset(host_test_mem, 0xaa, TEST_MEM_PAGES * 4096);
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Unexpected exit reason: %u (%s),\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -26,9 +26,7 @@ static void __run_vcpu_with_invalid_state(struct kvm_vcpu *vcpu) ...@@ -26,9 +26,7 @@ static void __run_vcpu_with_invalid_state(struct kvm_vcpu *vcpu)
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_INTERNAL_ERROR, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);
"Expected KVM_EXIT_INTERNAL_ERROR, got %d (%s)\n",
run->exit_reason, exit_reason_str(run->exit_reason));
TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION, TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION,
"Expected emulation failure, got %d\n", "Expected emulation failure, got %d\n",
run->emulation_failure.suberror); run->emulation_failure.suberror);
......
...@@ -74,9 +74,7 @@ int main(int argc, char *argv[]) ...@@ -74,9 +74,7 @@ int main(int argc, char *argv[])
* The first exit to L0 userspace should be an I/O access from L2. * The first exit to L0 userspace should be an I/O access from L2.
* Running L1 should launch L2 without triggering an exit to userspace. * Running L1 should launch L2 without triggering an exit to userspace.
*/ */
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Expected KVM_EXIT_IO, got: %u (%s)\n",
run->exit_reason, exit_reason_str(run->exit_reason));
TEST_ASSERT(run->io.port == ARBITRARY_IO_PORT, TEST_ASSERT(run->io.port == ARBITRARY_IO_PORT,
"Expected IN from port %d from L2, got port %d", "Expected IN from port %d from L2, got port %d",
......
...@@ -183,14 +183,10 @@ int main(int argc, char *argv[]) ...@@ -183,14 +183,10 @@ int main(int argc, char *argv[])
vcpu_ioctl(vcpu, KVM_SET_TSC_KHZ, (void *) (tsc_khz / l1_scale_factor)); vcpu_ioctl(vcpu, KVM_SET_TSC_KHZ, (void *) (tsc_khz / l1_scale_factor));
for (;;) { for (;;) {
volatile struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -157,7 +157,6 @@ int main(int argc, char *argv[]) ...@@ -157,7 +157,6 @@ int main(int argc, char *argv[])
struct kvm_regs regs1, regs2; struct kvm_regs regs1, regs2;
struct kvm_vm *vm; struct kvm_vm *vm;
struct kvm_run *run;
struct kvm_vcpu *vcpu; struct kvm_vcpu *vcpu;
struct kvm_x86_state *state; struct kvm_x86_state *state;
struct ucall uc; struct ucall uc;
...@@ -173,7 +172,6 @@ int main(int argc, char *argv[]) ...@@ -173,7 +172,6 @@ int main(int argc, char *argv[])
/* Create VM */ /* Create VM */
vm = vm_create_with_one_vcpu(&vcpu, guest_code); vm = vm_create_with_one_vcpu(&vcpu, guest_code);
run = vcpu->run;
vcpu_regs_get(vcpu, &regs1); vcpu_regs_get(vcpu, &regs1);
...@@ -182,10 +180,7 @@ int main(int argc, char *argv[]) ...@@ -182,10 +180,7 @@ int main(int argc, char *argv[])
for (stage = 1;; stage++) { for (stage = 1;; stage++) {
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Stage %d: unexpected exit reason: %u (%s),\n",
stage, run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
...@@ -237,7 +232,6 @@ int main(int argc, char *argv[]) ...@@ -237,7 +232,6 @@ int main(int argc, char *argv[])
/* Restore state in a new VM. */ /* Restore state in a new VM. */
vcpu = vm_recreate_with_one_vcpu(vm); vcpu = vm_recreate_with_one_vcpu(vm);
vcpu_load_state(vcpu, state); vcpu_load_state(vcpu, state);
run = vcpu->run;
kvm_x86_state_cleanup(state); kvm_x86_state_cleanup(state);
memset(&regs2, 0, sizeof(regs2)); memset(&regs2, 0, sizeof(regs2));
......
...@@ -131,14 +131,10 @@ int main(int argc, char *argv[]) ...@@ -131,14 +131,10 @@ int main(int argc, char *argv[])
vcpu_args_set(vcpu, 1, vmx_pages_gva); vcpu_args_set(vcpu, 1, vmx_pages_gva);
for (;;) { for (;;) {
volatile struct kvm_run *run = vcpu->run;
struct ucall uc; struct ucall uc;
vcpu_run(vcpu); vcpu_run(vcpu);
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
...@@ -198,7 +198,6 @@ static void *vcpu_thread(void *arg) ...@@ -198,7 +198,6 @@ static void *vcpu_thread(void *arg)
struct ucall uc; struct ucall uc;
int old; int old;
int r; int r;
unsigned int exit_reason;
r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old); r = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &old);
TEST_ASSERT(r == 0, TEST_ASSERT(r == 0,
...@@ -207,11 +206,8 @@ static void *vcpu_thread(void *arg) ...@@ -207,11 +206,8 @@ static void *vcpu_thread(void *arg)
fprintf(stderr, "vCPU thread running vCPU %u\n", vcpu->id); fprintf(stderr, "vCPU thread running vCPU %u\n", vcpu->id);
vcpu_run(vcpu); vcpu_run(vcpu);
exit_reason = vcpu->run->exit_reason;
TEST_ASSERT(exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"vCPU %u exited with unexpected exit reason %u-%s, expected KVM_EXIT_IO",
vcpu->id, exit_reason, exit_reason_str(exit_reason));
if (get_ucall(vcpu, &uc) == UCALL_ABORT) { if (get_ucall(vcpu, &uc) == UCALL_ABORT) {
TEST_ASSERT(false, TEST_ASSERT(false,
......
...@@ -122,10 +122,7 @@ int main(int argc, char *argv[]) ...@@ -122,10 +122,7 @@ int main(int argc, char *argv[])
continue; continue;
} }
TEST_ASSERT(run->exit_reason == KVM_EXIT_IO, TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);
"Got exit_reason other than KVM_EXIT_IO: %u (%s)\n",
run->exit_reason,
exit_reason_str(run->exit_reason));
switch (get_ucall(vcpu, &uc)) { switch (get_ucall(vcpu, &uc)) {
case UCALL_ABORT: case UCALL_ABORT:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment