2022-12-24 06:13:42 +01:00
|
|
|
#ifndef GUARD_TEST_H
|
|
|
|
#define GUARD_TEST_H
|
|
|
|
|
|
|
|
#include "test_runner.h"
|
|
|
|
|
|
|
|
#define MAX_PROCESSES 32 // See also tools/mgba-rom-test-hydra/main.c
|
|
|
|
|
|
|
|
enum TestResult
|
|
|
|
{
|
|
|
|
TEST_RESULT_FAIL,
|
|
|
|
TEST_RESULT_PASS,
|
2023-03-08 12:26:33 +01:00
|
|
|
TEST_RESULT_ASSUMPTION_FAIL,
|
2022-12-24 06:13:42 +01:00
|
|
|
TEST_RESULT_INVALID,
|
|
|
|
TEST_RESULT_ERROR,
|
|
|
|
TEST_RESULT_TIMEOUT,
|
2023-03-24 02:34:08 +01:00
|
|
|
TEST_RESULT_TODO,
|
2022-12-24 06:13:42 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct TestRunner
|
|
|
|
{
|
|
|
|
u32 (*estimateCost)(void *);
|
|
|
|
void (*setUp)(void *);
|
|
|
|
void (*run)(void *);
|
|
|
|
void (*tearDown)(void *);
|
|
|
|
bool32 (*checkProgress)(void *);
|
|
|
|
bool32 (*handleExitWithResult)(void *, enum TestResult);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Test
|
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
const char *filename;
|
|
|
|
const struct TestRunner *runner;
|
|
|
|
void *data;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TestRunnerState
|
|
|
|
{
|
|
|
|
u8 state;
|
|
|
|
u8 exitCode;
|
|
|
|
s32 tests;
|
|
|
|
s32 passes;
|
|
|
|
const char *skipFilename;
|
|
|
|
const struct Test *test;
|
|
|
|
u32 processCosts[MAX_PROCESSES];
|
|
|
|
|
|
|
|
u8 result;
|
|
|
|
u8 expectedResult;
|
2023-04-20 21:35:22 +02:00
|
|
|
bool8 expectLeaks:1;
|
2022-12-24 06:13:42 +01:00
|
|
|
u32 timeoutSeconds;
|
|
|
|
};
|
|
|
|
|
|
|
|
extern const u8 gTestRunnerN;
|
|
|
|
extern const u8 gTestRunnerI;
|
|
|
|
extern const char gTestRunnerArgv[256];
|
|
|
|
|
|
|
|
extern const struct TestRunner gAssumptionsRunner;
|
2023-02-03 12:21:08 +01:00
|
|
|
|
|
|
|
struct FunctionTestRunnerState
|
|
|
|
{
|
|
|
|
u8 parameters;
|
|
|
|
u8 runParameter;
|
|
|
|
};
|
|
|
|
|
|
|
|
extern const struct TestRunner gFunctionTestRunner;
|
|
|
|
extern struct FunctionTestRunnerState *gFunctionTestRunnerState;
|
|
|
|
|
2022-12-24 06:13:42 +01:00
|
|
|
extern struct TestRunnerState gTestRunnerState;
|
|
|
|
|
|
|
|
void CB2_TestRunner(void);
|
|
|
|
|
|
|
|
void Test_ExpectedResult(enum TestResult);
|
2023-04-20 21:35:22 +02:00
|
|
|
void Test_ExpectLeaks(bool32);
|
2022-12-24 06:13:42 +01:00
|
|
|
void Test_ExitWithResult(enum TestResult, const char *fmt, ...);
|
|
|
|
|
|
|
|
s32 MgbaPrintf_(const char *fmt, ...);
|
|
|
|
|
2023-02-03 12:21:08 +01:00
|
|
|
#define TEST(_name) \
|
|
|
|
static void CAT(Test, __LINE__)(void); \
|
|
|
|
__attribute__((section(".tests"))) static const struct Test CAT(sTest, __LINE__) = \
|
|
|
|
{ \
|
|
|
|
.name = _name, \
|
|
|
|
.filename = __FILE__, \
|
|
|
|
.runner = &gFunctionTestRunner, \
|
|
|
|
.data = (void *)CAT(Test, __LINE__), \
|
|
|
|
}; \
|
|
|
|
static void CAT(Test, __LINE__)(void)
|
|
|
|
|
2022-12-24 06:13:42 +01:00
|
|
|
#define ASSUMPTIONS \
|
|
|
|
static void Assumptions(void); \
|
|
|
|
__attribute__((section(".tests"))) static const struct Test sAssumptions = \
|
|
|
|
{ \
|
|
|
|
.name = "ASSUMPTIONS: " __FILE__, \
|
|
|
|
.filename = __FILE__, \
|
|
|
|
.runner = &gAssumptionsRunner, \
|
|
|
|
.data = Assumptions, \
|
|
|
|
}; \
|
|
|
|
static void Assumptions(void)
|
|
|
|
|
|
|
|
#define ASSUME(c) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
if (!(c)) \
|
2023-03-08 12:26:33 +01:00
|
|
|
Test_ExitWithResult(TEST_RESULT_ASSUMPTION_FAIL, "%s:%d: ASSUME failed", gTestRunnerState.test->filename, __LINE__); \
|
2022-12-24 06:13:42 +01:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT(c) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
if (!(c)) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT failed", gTestRunnerState.test->filename, __LINE__); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_EQ(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a != _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_EQ(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_NE(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a == _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_NE(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_LT(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a >= _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_LT(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_LE(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a > _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_LE(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_GT(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a <= _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_GT(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define EXPECT_GE(a, b) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
typeof(a) _a = (a), _b = (b); \
|
|
|
|
if (_a < _b) \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_FAIL, "%s:%d: EXPECT_GE(%d, %d) failed", gTestRunnerState.test->filename, __LINE__, _a, _b); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define KNOWN_FAILING \
|
|
|
|
Test_ExpectedResult(TEST_RESULT_FAIL)
|
|
|
|
|
2023-04-20 21:35:22 +02:00
|
|
|
#define KNOWN_LEAKING \
|
|
|
|
Test_ExpectLeaks(TRUE)
|
|
|
|
|
2023-02-03 12:21:08 +01:00
|
|
|
#define PARAMETRIZE if (gFunctionTestRunnerState->parameters++ == gFunctionTestRunnerState->runParameter)
|
|
|
|
|
2023-03-24 02:34:08 +01:00
|
|
|
#define TO_DO \
|
2023-06-22 16:09:16 +02:00
|
|
|
do { \
|
|
|
|
Test_ExpectedResult(TEST_RESULT_TODO); \
|
|
|
|
Test_ExitWithResult(TEST_RESULT_TODO, "%s:%d: EXPECT_TO_DO", gTestRunnerState.test->filename, __LINE__); \
|
|
|
|
} while (0)
|
2023-03-24 02:34:08 +01:00
|
|
|
|
2022-12-24 06:13:42 +01:00
|
|
|
#endif
|