2021-04-24 23:53:23 -06:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
|
|
* Copyright (c) 2021, Andrew Kaster <akaster@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <LibTest/Macros.h> // intentionally first -- we redefine VERIFY and friends in here
|
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
#include <AK/DeprecatedString.h>
|
2021-04-24 23:53:23 -06:00
|
|
|
#include <AK/Function.h>
|
|
|
|
#include <AK/NonnullRefPtr.h>
|
|
|
|
#include <AK/RefCounted.h>
|
|
|
|
|
|
|
|
namespace Test {
|
|
|
|
|
|
|
|
using TestFunction = Function<void()>;
|
|
|
|
|
|
|
|
class TestCase : public RefCounted<TestCase> {
|
|
|
|
public:
|
2022-12-04 18:02:33 +00:00
|
|
|
TestCase(DeprecatedString const& name, TestFunction&& fn, bool is_benchmark)
|
2021-04-24 23:53:23 -06:00
|
|
|
: m_name(name)
|
|
|
|
, m_function(move(fn))
|
|
|
|
, m_is_benchmark(is_benchmark)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_benchmark() const { return m_is_benchmark; }
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString const& name() const { return m_name; }
|
2022-04-01 20:58:27 +03:00
|
|
|
TestFunction const& func() const { return m_function; }
|
2021-04-24 23:53:23 -06:00
|
|
|
|
|
|
|
private:
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString m_name;
|
2021-04-24 23:53:23 -06:00
|
|
|
TestFunction m_function;
|
|
|
|
bool m_is_benchmark;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Helper to hide implementation of TestSuite from users
|
2022-04-01 20:58:27 +03:00
|
|
|
void add_test_case_to_suite(NonnullRefPtr<TestCase> const& test_case);
|
2021-08-29 11:55:05 -07:00
|
|
|
void set_suite_setup_function(Function<void()> setup);
|
2021-04-24 23:53:23 -06:00
|
|
|
}
|
|
|
|
|
2022-10-17 00:06:11 +02:00
|
|
|
#define TEST_SETUP \
|
|
|
|
static void __setup(); \
|
|
|
|
struct __setup_type { \
|
|
|
|
__setup_type() \
|
|
|
|
{ \
|
|
|
|
Test::set_suite_setup_function(__setup); \
|
|
|
|
} \
|
|
|
|
}; \
|
|
|
|
static struct __setup_type __setup_type; \
|
2021-08-29 11:55:05 -07:00
|
|
|
static void __setup()
|
|
|
|
|
2021-04-24 23:53:23 -06:00
|
|
|
#define __TESTCASE_FUNC(x) __test_##x
|
|
|
|
#define __TESTCASE_TYPE(x) __TestCase_##x
|
|
|
|
|
2022-10-17 00:06:11 +02:00
|
|
|
#define TEST_CASE(x) \
|
|
|
|
static void __TESTCASE_FUNC(x)(); \
|
|
|
|
struct __TESTCASE_TYPE(x) { \
|
|
|
|
__TESTCASE_TYPE(x) \
|
|
|
|
() \
|
|
|
|
{ \
|
|
|
|
add_test_case_to_suite(adopt_ref(*new ::Test::TestCase(#x, __TESTCASE_FUNC(x), false))); \
|
|
|
|
} \
|
|
|
|
}; \
|
|
|
|
static struct __TESTCASE_TYPE(x) __TESTCASE_TYPE(x); \
|
2021-04-24 23:53:23 -06:00
|
|
|
static void __TESTCASE_FUNC(x)()
|
|
|
|
|
|
|
|
#define __BENCHMARK_FUNC(x) __benchmark_##x
|
|
|
|
#define __BENCHMARK_TYPE(x) __BenchmarkCase_##x
|
|
|
|
|
2022-10-17 00:06:11 +02:00
|
|
|
#define BENCHMARK_CASE(x) \
|
|
|
|
static void __BENCHMARK_FUNC(x)(); \
|
|
|
|
struct __BENCHMARK_TYPE(x) { \
|
|
|
|
__BENCHMARK_TYPE(x) \
|
|
|
|
() \
|
|
|
|
{ \
|
|
|
|
add_test_case_to_suite(adopt_ref(*new ::Test::TestCase(#x, __BENCHMARK_FUNC(x), true))); \
|
|
|
|
} \
|
|
|
|
}; \
|
|
|
|
static struct __BENCHMARK_TYPE(x) __BENCHMARK_TYPE(x); \
|
2021-04-24 23:53:23 -06:00
|
|
|
static void __BENCHMARK_FUNC(x)()
|
2023-10-24 01:43:23 +02:00
|
|
|
|
|
|
|
// This allows us to print the generated locals in the test after a failure is fully shrunk.
|
|
|
|
#define GEN(identifier, value) \
|
|
|
|
auto identifier = (value); \
|
|
|
|
if (::Test::current_test_result() == ::Test::TestResult::Overrun) \
|
|
|
|
return; \
|
|
|
|
if (::Test::is_reporting_enabled()) \
|
|
|
|
::AK::warnln("{} = {}", #identifier, (identifier))
|