blob: bdfe9c189e60ac1b38ca618b2189642ad20ef893 [file] [log] [blame]
/*
* Copyright 2017, Data61, CSIRO (ABN 41 687 119 230)
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
/* Include Kconfig variables. */
#include <autoconf.h>
#include <sel4test/gen_config.h>
#include <sel4/sel4.h>
#include <utils/attribute.h>
#include <sel4test/testutil.h>
#include <vka/vka.h>
#include <vspace/vspace.h>
#include <sel4platsupport/timer.h>
#include <sync/mutex.h>
#include <sel4utils/elf.h>
#include <sel4rpc/client.h>
#include <inttypes.h>
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
/* max test name size */
#define TEST_NAME_MAX (64 - 4 * sizeof(seL4_Word))
#define MAX_REGIONS 4
/* Contains information about the test environment for regular tests, bootstrap tests do
* not use this environment */
struct env {
/* An initialised vka that may be used by the test. */
vka_t vka;
/* virtual memory management interface */
vspace_t vspace;
/* abstract interface over application init */
simple_t simple;
/* notification for timer */
vka_object_t timer_notification;
/* RPC client for serial server resource allocation */
sel4rpc_client_t rpc_client;
/* caps for the current process */
seL4_CPtr cspace_root;
seL4_CPtr page_directory;
seL4_CPtr endpoint;
seL4_CPtr tcb;
seL4_CPtr timer_untyped;
seL4_CPtr asid_pool;
seL4_CPtr asid_ctrl;
seL4_CPtr sched_ctrl;
#ifdef CONFIG_IOMMU
seL4_CPtr io_space;
#endif /* CONFIG_IOMMU */
#ifdef CONFIG_TK1_SMMU
seL4_SlotRegion io_space_caps;
#endif
seL4_Word cores;
seL4_CPtr domain;
seL4_CPtr device_frame;
int priority;
int cspace_size_bits;
int num_regions;
sel4utils_elf_region_t regions[MAX_REGIONS];
};
typedef struct env *env_t;
/* Prototype of a test function. Returns false on failure. */
typedef int (*test_fn)(uintptr_t environment);
/* Test type definitions. */
typedef enum test_type_name {BOOTSTRAP = 0, BASIC} test_type_name_t;
typedef struct testcase testcase_t; // Forward type declaration.
typedef struct test_type {
/* Represents a single test type. See comment for `struct testcase` for info about ALIGN(32). */
const char *name;
test_type_name_t id;
// Function called before and after all the tests for this test type have been run.
void (*set_up_test_type)(uintptr_t e);
void (*tear_down_test_type)(uintptr_t e);
// Function called before and after each test for this test type.
void (*set_up)(uintptr_t e);
void (*tear_down)(uintptr_t e);
// Run the test. Different tests take different environments
test_result_t (*run_test)(struct testcase *test, uintptr_t e);
} ALIGN(32) test_type_t;
/* Declare a test type.
* For now, we put the test types in a separate elf section. */
#define DEFINE_TEST_TYPE(_name, _id, _set_up_test_type, _tear_down_test_type, _set_up, _tear_down, _run_test) \
__attribute__((used)) __attribute__((section("_test_type"))) struct test_type TEST_TYPE_ ##_name = { \
.name = #_name, \
.id = _id, \
.set_up_test_type = _set_up_test_type, \
.tear_down_test_type = _tear_down_test_type, \
.set_up = _set_up, \
.tear_down = _tear_down, \
.run_test = _run_test, \
};
/* Represents a single testcase.
* Because this struct is used to declare variables that get
* placed into custom sections, that we later treat as an array,
* we need to make sure the struct is aligned and filled to the
* nearest power of two to avoid gcc placing arbitrary padding between them.
*
* The declaration below ensures that the actual size of
* the objects in the section is the same as the size reported
* by sizeof(struct testcase), allowing as to treat the items
* in the section as an array */
struct testcase {
char name[TEST_NAME_MAX];
const char *description;
test_fn function;
seL4_Word test_type;
seL4_Word enabled;
} PACKED;
typedef struct testcase ALIGN(sizeof(struct testcase)) testcase_t;
/* Declare a testcase.
* Must be declared using C89 style (#_name, _desc, _func...) instead of
* C99 style (name = _name, desc = _desc, func = _func...) to make sure
* that it is accepted by C++ compilers.
*/
#define DEFINE_TEST_WITH_TYPE(_name, _description, _function, _test_type, _enabled) \
__attribute__((used)) __attribute__((section("_test_case"))) struct testcase TEST_ ## _name = { \
#_name, \
_description, \
(test_fn)_function, \
_test_type, \
_enabled, \
};
#define DEFINE_TEST(_name, _description, _function, _enabled) DEFINE_TEST_WITH_TYPE(_name, _description, _function, BASIC, _enabled)
#define DEFINE_TEST_BOOTSTRAP(_name, _description, _function, _enabled) DEFINE_TEST_WITH_TYPE(_name, _description, _function, BOOTSTRAP, _enabled)
/**/
/* Definitions so that we can find the test types */
extern struct test_type __start__test_type[];
extern struct test_type __stop__test_type[];
/* Definitions so that we can find the test cases */
extern testcase_t __start__test_case[];
extern testcase_t __stop__test_case[];
static inline int test_type_comparator(const void *a, const void *b)
{
const struct test_type **ta = (const struct test_type **) a;
const struct test_type **tb = (const struct test_type **) b;
if ((*ta)->id > (*tb)->id) {
return 1;
} else if ((*ta)->id < (*tb)->id) {
return -1;
}
return 0;
}
static inline int test_comparator(const void *a, const void *b)
{
const struct testcase **ta = (const struct testcase **)a;
const struct testcase **tb = (const struct testcase **)b;
return strcmp((*ta)->name, (*tb)->name);
}
/* Fails a test case, stop running the rest of the test, but keep running other tests. */
static inline test_result_t _test_fail(const char *condition, const char *file, int line)
{
_sel4test_failure(condition, file, line);
return FAILURE;
}
/* Fails a test case, keep running the rest of the test, then keep running other tests. */
static inline void _test_error(const char *condition, const char *file, int line)
{
_sel4test_report_error(condition, file, line);
}
/* Fails a test case, stop everything. */
static inline test_result_t _test_abort(const char *condition, const char *file, int line)
{
_sel4test_failure(condition, file, line);
return ABORT;
}
static inline void print_error_in_ipc(seL4_Error e)
{
#ifdef CONFIG_KERNEL_INVOCATION_REPORT_ERROR_IPC
// If it hasnt been printed already
if (!seL4_CanPrintError() && e != seL4_NoError) {
printf("%s", seL4_GetDebugError());
}
#endif
}
#define test_error_eq(e, c) \
if (!((e) == (c))) { \
print_error_in_ipc(e); \
return _test_fail(#e, __FILE__, __LINE__); \
}
#define test_assert(e) if (!(e)) return _test_fail(#e, __FILE__, __LINE__)
#define test_check(e) if (!(e)) _test_error(#e, __FILE__, __LINE__)
#define test_assert_fatal(e) if (!(e)) return _test_abort(#e, __FILE__, __LINE__)
#define __TEST_BUFFER_SIZE 200
#define test_op_type(a, b, op, t, name_a, name_b, cast) \
do {\
if (!(a op b)) { \
int len = snprintf(NULL, 0, "Check %s(" t ") %s %s(" t ") failed.",\
#name_a, (cast) a, #op, #name_b, (cast) b) + 1; \
char buffer[len]; \
snprintf(buffer, len, "Check %s(" t ") %s %s(" t ") failed.",\
#name_a, (cast) a, #op, #name_b, (cast) b); \
_test_error(buffer, __FILE__, __LINE__); \
}\
} while (0)
#define test_op(a, b, op) \
do { \
typeof (a) _a = (a); \
typeof (b) _b = (b); \
if (sizeof(_a) != sizeof(_b)) { \
int len = snprintf(NULL, 0, "%s (size %zu) != %s (size %zu), use of test_eq incorrect", #a,\
sizeof(_a), #b, sizeof(_b)) + 1;\
char buffer[len];\
snprintf(buffer, len, "%s (size %zu) != %s (size %zu), use of test_eq incorrect", #a, sizeof(_a),\
#b, sizeof(_b));\
_test_error(buffer, __FILE__, __LINE__);\
} else if (TYPES_COMPATIBLE(typeof(_a), int)) {\
test_op_type(_a, _b, op, "%d", a, b, int); \
} else if (TYPES_COMPATIBLE(typeof(_a), long)) {\
test_op_type(_a, _b, op, "%ld", a, b, long); \
} else if (TYPES_COMPATIBLE(typeof(_a), long long)) {\
test_op_type(_a, _b, op, "%lld", a, b, long long); \
} else if (TYPES_COMPATIBLE(typeof(_a), unsigned int)) {\
test_op_type(_a, _b, op, "%u", a, b, unsigned int); \
} else if (TYPES_COMPATIBLE(typeof(_a), unsigned long)) {\
test_op_type(_a, _b, op, "%lu", a, b, unsigned long); \
} else if (TYPES_COMPATIBLE(typeof(_a), unsigned long long)) {\
test_op_type(_a, _b, op, "%llu", a, b, unsigned long long); \
} else if (TYPES_COMPATIBLE(typeof(_a), char)) {\
test_op_type(_a, _b, op, "%c", a, b, char); \
} else if (TYPES_COMPATIBLE(typeof(_a), uintptr_t)) {\
test_op_type(_a, _b, op, "0x%" PRIxPTR, a, b, uintptr_t);\
} else { \
_test_error("Cannot use test_op on this type", __FILE__, __LINE__);\
}\
} while (0)
/* Pretty printed test_check wrapper macros for basic comparisons on base types,
* which output the values and variable names to aid debugging */
#define test_eq(a, b) test_op(a, b, ==)
#define test_neq(a, b) test_op(a, b, !=)
#define test_gt(a, b) test_op(a, b, >)
#define test_geq(a, b) test_op(a, b, >=)
#define test_lt(a, b) test_op(a, b, <)
#define test_leq(a, b) test_op(a, b, <=)
#define __TEST_MAX_STRING 50
#define test_strop(a, b, op) \
do {\
if (strnlen(a, __TEST_MAX_STRING) == __TEST_MAX_STRING) {\
_test_error("String " #a " too long for test_str* macros", __FILE__, __LINE__);\
} else if (strnlen(b, __TEST_MAX_STRING) == __TEST_MAX_STRING) {\
_test_error("String " #b " too long for test_str* macros", __FILE__, __LINE__);\
} else if (!(strncmp(a, b, __TEST_MAX_STRING)) op 0) {\
char buffer[__TEST_BUFFER_SIZE + 2 * __TEST_MAX_STRING];\
snprintf(buffer, sizeof(buffer),\
"Check %s(%s) %s %s(%s) failed.", #a, a, #op, #b, b);\
_test_error(buffer, __FILE__, __LINE__); \
}\
} while (0)
/* Pretty printed test_check wrapper macros for basic comparisons on c strings,
* which output the values and variable names to aid debugging */
#define test_streq(a, b) test_strop(a, b, ==)
#define test_strneq(a, b) test_strop(a, b, !=)
#define test_strge(a, b) test_strop(a, b, >)
#define test_strgeq(a, b) test_strop(a, b, >=)
#define test_strle(a, b) test_strop(a, b, <)
#define test_strleq(a, b) test_strop(a, b, <=)
env_t sel4test_get_env(void);