Path: blob/master/tools/testing/selftests/kvm/x86/flds_emulation.h
38237 views
/* SPDX-License-Identifier: GPL-2.0-only */1#ifndef SELFTEST_KVM_FLDS_EMULATION_H2#define SELFTEST_KVM_FLDS_EMULATION_H34#include "kvm_util.h"56#define FLDS_MEM_EAX ".byte 0xd9, 0x00"78/*9* flds is an instruction that the KVM instruction emulator is known not to10* support. This can be used in guest code along with a mechanism to force11* KVM to emulate the instruction (e.g. by providing an MMIO address) to12* exercise emulation failures.13*/14static inline void flds(uint64_t address)15{16__asm__ __volatile__(FLDS_MEM_EAX :: "a"(address));17}1819static inline void handle_flds_emulation_failure_exit(struct kvm_vcpu *vcpu)20{21struct kvm_run *run = vcpu->run;22struct kvm_regs regs;23uint8_t *insn_bytes;24uint64_t flags;2526TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);2728TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION,29"Unexpected suberror: %u",30run->emulation_failure.suberror);3132flags = run->emulation_failure.flags;33TEST_ASSERT(run->emulation_failure.ndata >= 3 &&34flags & KVM_INTERNAL_ERROR_EMULATION_FLAG_INSTRUCTION_BYTES,35"run->emulation_failure is missing instruction bytes");3637TEST_ASSERT(run->emulation_failure.insn_size >= 2,38"Expected a 2-byte opcode for 'flds', got %d bytes",39run->emulation_failure.insn_size);4041insn_bytes = run->emulation_failure.insn_bytes;42TEST_ASSERT(insn_bytes[0] == 0xd9 && insn_bytes[1] == 0,43"Expected 'flds [eax]', opcode '0xd9 0x00', got opcode 0x%02x 0x%02x",44insn_bytes[0], insn_bytes[1]);4546vcpu_regs_get(vcpu, ®s);47regs.rip += 2;48vcpu_regs_set(vcpu, ®s);49}5051#endif /* !SELFTEST_KVM_FLDS_EMULATION_H */525354