119a2b32fSDavid Matlack /* SPDX-License-Identifier: GPL-2.0-only */
219a2b32fSDavid Matlack #ifndef SELFTEST_KVM_FLDS_EMULATION_H
319a2b32fSDavid Matlack #define SELFTEST_KVM_FLDS_EMULATION_H
419a2b32fSDavid Matlack 
519a2b32fSDavid Matlack #include "kvm_util.h"
619a2b32fSDavid Matlack 
719a2b32fSDavid Matlack #define FLDS_MEM_EAX ".byte 0xd9, 0x00"
819a2b32fSDavid Matlack 
919a2b32fSDavid Matlack /*
1019a2b32fSDavid Matlack  * flds is an instruction that the KVM instruction emulator is known not to
1119a2b32fSDavid Matlack  * support. This can be used in guest code along with a mechanism to force
1219a2b32fSDavid Matlack  * KVM to emulate the instruction (e.g. by providing an MMIO address) to
1319a2b32fSDavid Matlack  * exercise emulation failures.
1419a2b32fSDavid Matlack  */
flds(uint64_t address)1519a2b32fSDavid Matlack static inline void flds(uint64_t address)
1619a2b32fSDavid Matlack {
1719a2b32fSDavid Matlack 	__asm__ __volatile__(FLDS_MEM_EAX :: "a"(address));
1819a2b32fSDavid Matlack }
1919a2b32fSDavid Matlack 
handle_flds_emulation_failure_exit(struct kvm_vcpu * vcpu)2019a2b32fSDavid Matlack static inline void handle_flds_emulation_failure_exit(struct kvm_vcpu *vcpu)
2119a2b32fSDavid Matlack {
2219a2b32fSDavid Matlack 	struct kvm_run *run = vcpu->run;
2319a2b32fSDavid Matlack 	struct kvm_regs regs;
2419a2b32fSDavid Matlack 	uint8_t *insn_bytes;
2519a2b32fSDavid Matlack 	uint64_t flags;
2619a2b32fSDavid Matlack 
27*c96f57b0SVipin Sharma 	TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_INTERNAL_ERROR);
2819a2b32fSDavid Matlack 
2919a2b32fSDavid Matlack 	TEST_ASSERT(run->emulation_failure.suberror == KVM_INTERNAL_ERROR_EMULATION,
3019a2b32fSDavid Matlack 		    "Unexpected suberror: %u",
3119a2b32fSDavid Matlack 		    run->emulation_failure.suberror);
3219a2b32fSDavid Matlack 
3319a2b32fSDavid Matlack 	flags = run->emulation_failure.flags;
3419a2b32fSDavid Matlack 	TEST_ASSERT(run->emulation_failure.ndata >= 3 &&
3519a2b32fSDavid Matlack 		    flags & KVM_INTERNAL_ERROR_EMULATION_FLAG_INSTRUCTION_BYTES,
3619a2b32fSDavid Matlack 		    "run->emulation_failure is missing instruction bytes");
3719a2b32fSDavid Matlack 
3819a2b32fSDavid Matlack 	TEST_ASSERT(run->emulation_failure.insn_size >= 2,
3919a2b32fSDavid Matlack 		    "Expected a 2-byte opcode for 'flds', got %d bytes",
4019a2b32fSDavid Matlack 		    run->emulation_failure.insn_size);
4119a2b32fSDavid Matlack 
4219a2b32fSDavid Matlack 	insn_bytes = run->emulation_failure.insn_bytes;
4319a2b32fSDavid Matlack 	TEST_ASSERT(insn_bytes[0] == 0xd9 && insn_bytes[1] == 0,
4419a2b32fSDavid Matlack 		    "Expected 'flds [eax]', opcode '0xd9 0x00', got opcode 0x%02x 0x%02x\n",
4519a2b32fSDavid Matlack 		    insn_bytes[0], insn_bytes[1]);
4619a2b32fSDavid Matlack 
4719a2b32fSDavid Matlack 	vcpu_regs_get(vcpu, &regs);
4819a2b32fSDavid Matlack 	regs.rip += 2;
4919a2b32fSDavid Matlack 	vcpu_regs_set(vcpu, &regs);
5019a2b32fSDavid Matlack }
5119a2b32fSDavid Matlack 
5219a2b32fSDavid Matlack #endif /* !SELFTEST_KVM_FLDS_EMULATION_H */
53