111 lines
3.1 KiB
C
111 lines
3.1 KiB
C
|
/* 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 */
|