1 /* SPDX-License-Identifier: GPL-2.0-only */
2 #ifndef SELFTEST_KVM_FLDS_EMULATION_H
3 #define SELFTEST_KVM_FLDS_EMULATION_H
7 #define FLDS_MEM_EAX ".byte 0xd9, 0x00"
10 * flds is an instruction that the KVM instruction emulator is known not to
11 * support. This can be used in guest code along with a mechanism to force
12 * KVM to emulate the instruction (e.g. by providing an MMIO address) to
13 * exercise emulation failures.
15 static inline void flds(uint64_t address
)
17 __asm__
__volatile__(FLDS_MEM_EAX :: "a"(address
));
20 static inline void handle_flds_emulation_failure_exit(struct kvm_vcpu
*vcpu
)
22 struct kvm_run
*run
= vcpu
->run
;
27 TEST_ASSERT_KVM_EXIT_REASON(vcpu
, KVM_EXIT_INTERNAL_ERROR
);
29 TEST_ASSERT(run
->emulation_failure
.suberror
== KVM_INTERNAL_ERROR_EMULATION
,
30 "Unexpected suberror: %u",
31 run
->emulation_failure
.suberror
);
33 flags
= run
->emulation_failure
.flags
;
34 TEST_ASSERT(run
->emulation_failure
.ndata
>= 3 &&
35 flags
& KVM_INTERNAL_ERROR_EMULATION_FLAG_INSTRUCTION_BYTES
,
36 "run->emulation_failure is missing instruction bytes");
38 TEST_ASSERT(run
->emulation_failure
.insn_size
>= 2,
39 "Expected a 2-byte opcode for 'flds', got %d bytes",
40 run
->emulation_failure
.insn_size
);
42 insn_bytes
= run
->emulation_failure
.insn_bytes
;
43 TEST_ASSERT(insn_bytes
[0] == 0xd9 && insn_bytes
[1] == 0,
44 "Expected 'flds [eax]', opcode '0xd9 0x00', got opcode 0x%02x 0x%02x",
45 insn_bytes
[0], insn_bytes
[1]);
47 vcpu_regs_get(vcpu
, ®s
);
49 vcpu_regs_set(vcpu
, ®s
);
52 #endif /* !SELFTEST_KVM_FLDS_EMULATION_H */