1/* SPDX-License-Identifier: GPL-2.0-only */ 2/* 3 * Copyright (C) 2018, Google LLC. 4 */ 5#ifndef SELFTEST_KVM_UCALL_COMMON_H 6#define SELFTEST_KVM_UCALL_COMMON_H 7#include "test_util.h" 8#include "ucall.h" 9 10/* Common ucalls */ 11enum { 12 UCALL_NONE, 13 UCALL_SYNC, 14 UCALL_ABORT, 15 UCALL_PRINTF, 16 UCALL_DONE, 17 UCALL_UNHANDLED, 18}; 19 20#define UCALL_MAX_ARGS 7 21#define UCALL_BUFFER_LEN 1024 22 23struct ucall { 24 uint64_t cmd; 25 uint64_t args[UCALL_MAX_ARGS]; 26 char buffer[UCALL_BUFFER_LEN]; 27 28 /* Host virtual address of this struct. */ 29 struct ucall *hva; 30}; 31 32void ucall_arch_init(struct kvm_vm *vm, vm_paddr_t mmio_gpa); 33void ucall_arch_do_ucall(vm_vaddr_t uc); 34void *ucall_arch_get_ucall(struct kvm_vcpu *vcpu); 35 36void ucall(uint64_t cmd, int nargs, ...); 37void ucall_fmt(uint64_t cmd, const char *fmt, ...); 38void ucall_assert(uint64_t cmd, const char *exp, const char *file, 39 unsigned int line, const char *fmt, ...); 40uint64_t get_ucall(struct kvm_vcpu *vcpu, struct ucall *uc); 41void ucall_init(struct kvm_vm *vm, vm_paddr_t mmio_gpa); 42int ucall_nr_pages_required(uint64_t page_size); 43 44/* 45 * Perform userspace call without any associated data. This bare call avoids 46 * allocating a ucall struct, which can be useful if the atomic operations in 47 * the full ucall() are problematic and/or unwanted. Note, this will come out 48 * as UCALL_NONE on the backend. 49 */ 50#define GUEST_UCALL_NONE() ucall_arch_do_ucall((vm_vaddr_t)NULL) 51 52#define GUEST_SYNC_ARGS(stage, arg1, arg2, arg3, arg4) \ 53 ucall(UCALL_SYNC, 6, "hello", stage, arg1, arg2, arg3, arg4) 54#define GUEST_SYNC(stage) ucall(UCALL_SYNC, 2, "hello", stage) 55#define GUEST_PRINTF(_fmt, _args...) ucall_fmt(UCALL_PRINTF, _fmt, ##_args) 56#define GUEST_DONE() ucall(UCALL_DONE, 0) 57 58#define REPORT_GUEST_PRINTF(ucall) pr_info("%s", (ucall).buffer) 59 60enum guest_assert_builtin_args { 61 GUEST_ERROR_STRING, 62 GUEST_FILE, 63 GUEST_LINE, 64 GUEST_ASSERT_BUILTIN_NARGS 65}; 66 67#define ____GUEST_ASSERT(_condition, _exp, _fmt, _args...) \ 68do { \ 69 if (!(_condition)) \ 70 ucall_assert(UCALL_ABORT, _exp, __FILE__, __LINE__, _fmt, ##_args); \ 71} while (0) 72 73#define __GUEST_ASSERT(_condition, _fmt, _args...) \ 74 ____GUEST_ASSERT(_condition, #_condition, _fmt, ##_args) 75 76#define GUEST_ASSERT(_condition) \ 77 __GUEST_ASSERT(_condition, #_condition) 78 79#define GUEST_FAIL(_fmt, _args...) \ 80 ucall_assert(UCALL_ABORT, "Unconditional guest failure", \ 81 __FILE__, __LINE__, _fmt, ##_args) 82 83#define GUEST_ASSERT_EQ(a, b) \ 84do { \ 85 typeof(a) __a = (a); \ 86 typeof(b) __b = (b); \ 87 ____GUEST_ASSERT(__a == __b, #a " == " #b, "%#lx != %#lx (%s != %s)", \ 88 (unsigned long)(__a), (unsigned long)(__b), #a, #b); \ 89} while (0) 90 91#define GUEST_ASSERT_NE(a, b) \ 92do { \ 93 typeof(a) __a = (a); \ 94 typeof(b) __b = (b); \ 95 ____GUEST_ASSERT(__a != __b, #a " != " #b, "%#lx == %#lx (%s == %s)", \ 96 (unsigned long)(__a), (unsigned long)(__b), #a, #b); \ 97} while (0) 98 99#define REPORT_GUEST_ASSERT(ucall) \ 100 test_assert(false, (const char *)(ucall).args[GUEST_ERROR_STRING], \ 101 (const char *)(ucall).args[GUEST_FILE], \ 102 (ucall).args[GUEST_LINE], "%s", (ucall).buffer) 103 104#endif /* SELFTEST_KVM_UCALL_COMMON_H */ 105