| /* 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]; |
| }; |
| |
| void ucall_init(struct kvm_vm *vm, void *arg); |
| void ucall_uninit(struct kvm_vm *vm); |
| void ucall(uint64_t cmd, int nargs, ...); |
| uint64_t get_ucall(struct kvm_vcpu *vcpu, struct ucall *uc); |
| |
| #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 */ |