/* SPDX-License-Identifier: GPL-2.0-only */ /* * tools/testing/selftests/kvm/include/kvm_util.h * * Copyright (C) 2018, Google LLC. */ #ifndef SELFTEST_KVM_UCALL_COMMON_H #define SELFTEST_KVM_UCALL_COMMON_H #include "test_util.h" /* Common ucalls */ enum { UCALL_NONE, UCALL_SYNC, UCALL_ABORT, UCALL_DONE, UCALL_UNHANDLED, }; #define UCALL_MAX_ARGS 7 struct ucall { uint64_t cmd; uint64_t args[UCALL_MAX_ARGS]; /* Host virtual address of this struct. */ struct ucall *hva; }; void ucall_arch_init(struct kvm_vm *vm, vm_paddr_t mmio_gpa); void ucall_arch_do_ucall(vm_vaddr_t uc); void *ucall_arch_get_ucall(struct kvm_vcpu *vcpu); void ucall(uint64_t cmd, int nargs, ...); uint64_t get_ucall(struct kvm_vcpu *vcpu, struct ucall *uc); void ucall_init(struct kvm_vm *vm, vm_paddr_t mmio_gpa); /* * Perform userspace call without any associated data. This bare call avoids * allocating a ucall struct, which can be useful if the atomic operations in * the full ucall() are problematic and/or unwanted. Note, this will come out * as UCALL_NONE on the backend. */ #define GUEST_UCALL_NONE() ucall_arch_do_ucall((vm_vaddr_t)NULL) #define GUEST_SYNC_ARGS(stage, arg1, arg2, arg3, arg4) \ ucall(UCALL_SYNC, 6, "hello", stage, arg1, arg2, arg3, arg4) #define GUEST_SYNC(stage) ucall(UCALL_SYNC, 2, "hello", stage) #define GUEST_DONE() ucall(UCALL_DONE, 0) enum guest_assert_builtin_args { GUEST_ERROR_STRING, GUEST_FILE, GUEST_LINE, GUEST_ASSERT_BUILTIN_NARGS }; #define __GUEST_ASSERT(_condition, _condstr, _nargs, _args...) \ do { \ if (!(_condition)) \ ucall(UCALL_ABORT, GUEST_ASSERT_BUILTIN_NARGS + _nargs, \ "Failed guest assert: " _condstr, \ __FILE__, __LINE__, ##_args); \ } while (0) #define GUEST_ASSERT(_condition) \ __GUEST_ASSERT(_condition, #_condition, 0, 0) #define GUEST_ASSERT_1(_condition, arg1) \ __GUEST_ASSERT(_condition, #_condition, 1, (arg1)) #define GUEST_ASSERT_2(_condition, arg1, arg2) \ __GUEST_ASSERT(_condition, #_condition, 2, (arg1), (arg2)) #define GUEST_ASSERT_3(_condition, arg1, arg2, arg3) \ __GUEST_ASSERT(_condition, #_condition, 3, (arg1), (arg2), (arg3)) #define GUEST_ASSERT_4(_condition, arg1, arg2, arg3, arg4) \ __GUEST_ASSERT(_condition, #_condition, 4, (arg1), (arg2), (arg3), (arg4)) #define GUEST_ASSERT_EQ(a, b) __GUEST_ASSERT((a) == (b), #a " == " #b, 2, a, b) #define __REPORT_GUEST_ASSERT(_ucall, fmt, _args...) \ TEST_FAIL("%s at %s:%ld\n" fmt, \ (const char *)(_ucall).args[GUEST_ERROR_STRING], \ (const char *)(_ucall).args[GUEST_FILE], \ (_ucall).args[GUEST_LINE], \ ##_args) #define GUEST_ASSERT_ARG(ucall, i) ((ucall).args[GUEST_ASSERT_BUILTIN_NARGS + i]) #define REPORT_GUEST_ASSERT(ucall) \ __REPORT_GUEST_ASSERT((ucall), "") #define REPORT_GUEST_ASSERT_1(ucall, fmt) \ __REPORT_GUEST_ASSERT((ucall), \ fmt, \ GUEST_ASSERT_ARG((ucall), 0)) #define REPORT_GUEST_ASSERT_2(ucall, fmt) \ __REPORT_GUEST_ASSERT((ucall), \ fmt, \ GUEST_ASSERT_ARG((ucall), 0), \ GUEST_ASSERT_ARG((ucall), 1)) #define REPORT_GUEST_ASSERT_3(ucall, fmt) \ __REPORT_GUEST_ASSERT((ucall), \ fmt, \ GUEST_ASSERT_ARG((ucall), 0), \ GUEST_ASSERT_ARG((ucall), 1), \ GUEST_ASSERT_ARG((ucall), 2)) #define REPORT_GUEST_ASSERT_4(ucall, fmt) \ __REPORT_GUEST_ASSERT((ucall), \ fmt, \ GUEST_ASSERT_ARG((ucall), 0), \ GUEST_ASSERT_ARG((ucall), 1), \ GUEST_ASSERT_ARG((ucall), 2), \ GUEST_ASSERT_ARG((ucall), 3)) #define REPORT_GUEST_ASSERT_N(ucall, fmt, args...) \ __REPORT_GUEST_ASSERT((ucall), fmt, ##args) #endif /* SELFTEST_KVM_UCALL_COMMON_H */