314 lines
9.6 KiB
C
314 lines
9.6 KiB
C
#include <std/base.h>
|
|
#include <std/os.h>
|
|
#include <std/entry_point.h>
|
|
#include <std/virtual_buffer.h>
|
|
#include <std/string.h>
|
|
|
|
#include <bloat-buster/base.h>
|
|
|
|
declare_slice(CompilerBackend);
|
|
|
|
typedef enum CMakeBuildType
|
|
{
|
|
CMAKE_BUILD_TYPE_DEBUG,
|
|
CMAKE_BUILD_TYPE_MIN_SIZE_RELEASE,
|
|
CMAKE_BUILD_TYPE_RELEASE_WITH_DEBUG_INFO,
|
|
CMAKE_BUILD_TYPE_RELEASE,
|
|
CMAKE_BUILD_TYPE_COUNT,
|
|
} CMakeBuildType;
|
|
|
|
fn void run(Arena* arena, char** envp, String compiler_path, CompilerBackend compiler_backend, u8 debug, char* bb_source_path)
|
|
{
|
|
CStringSlice args = {};
|
|
auto compiler_backend_string = compiler_backend_to_one_char_string(compiler_backend);
|
|
|
|
#define common_compile_and_run_args \
|
|
string_to_c(compiler_path), \
|
|
bb_source_path, \
|
|
string_to_c(compiler_backend_string), \
|
|
0,
|
|
|
|
if (debug)
|
|
{
|
|
#if _WIN32
|
|
args = (CStringSlice) array_to_slice(((char*[]){
|
|
"C:\\Users\\David\\Downloads\\remedybg_0_4_0_8\\remedybg.exe",
|
|
"-g",
|
|
common_compile_and_run_args
|
|
}));
|
|
#elif defined(__linux__)
|
|
args = (CStringSlice) array_to_slice(((char*[]){
|
|
"/home/david/source/gf/gf2",
|
|
"-ex",
|
|
"set auto-solib-add off",
|
|
"-ex",
|
|
"r",
|
|
"--args",
|
|
common_compile_and_run_args
|
|
}));
|
|
#elif defined(__APPLE__)
|
|
args = (CStringSlice) array_to_slice(((char*[]){
|
|
"/usr/bin/lldb",
|
|
"-o",
|
|
"run",
|
|
"--",
|
|
common_compile_and_run_args
|
|
}));
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
args = (CStringSlice) array_to_slice(((char*[]){
|
|
common_compile_and_run_args
|
|
}));
|
|
}
|
|
|
|
run_command(arena, args, envp);
|
|
}
|
|
|
|
typedef enum Command : u8
|
|
{
|
|
COMMAND_DEBUG,
|
|
COMMAND_RUN_TESTS,
|
|
COMMAND_COUNT,
|
|
} Command;
|
|
|
|
STRUCT(TestOptions)
|
|
{
|
|
Slice(String) test_paths;
|
|
Slice(CompilerBackend) compiler_backends;
|
|
};
|
|
|
|
fn void run_tests(Arena* arena, String compiler_path, TestOptions const * const test_options, char** envp)
|
|
{
|
|
Target target = native_target_get();
|
|
|
|
for (u32 test_i = 0; test_i < test_options->test_paths.length; test_i += 1)
|
|
{
|
|
String test_path = test_options->test_paths.pointer[test_i];
|
|
char* test_path_c = string_to_c(test_path);
|
|
auto test_dir = path_no_extension(test_path);
|
|
auto test_name = path_base(test_dir);
|
|
|
|
for (u32 engine_i = 0; engine_i < test_options->compiler_backends.length; engine_i += 1)
|
|
{
|
|
CompilerBackend compiler_backend = test_options->compiler_backends.pointer[engine_i];
|
|
auto compiler_backend_string = compiler_backend_to_one_char_string(compiler_backend);
|
|
|
|
char* arguments[] = {
|
|
string_to_c(compiler_path),
|
|
test_path_c,
|
|
string_to_c(compiler_backend_string),
|
|
0,
|
|
};
|
|
|
|
run_command(arena, (CStringSlice) array_to_slice(arguments), envp);
|
|
|
|
auto run_tests = BB_CI;
|
|
#ifndef __x86_64__
|
|
run_tests = run_tests && compiler_backend != COMPILER_BACKEND_BB;
|
|
#endif
|
|
if (run_tests)
|
|
{
|
|
auto executable = binary_path_from_options(arena, (BinaryPathOptions) {
|
|
.build_directory = strlit(BB_DIR),
|
|
.name = test_name,
|
|
.target = target,
|
|
.backend = compiler_backend,
|
|
.binary_file_type = BINARY_FILE_EXECUTABLE,
|
|
});
|
|
char* run_arguments[] = {
|
|
string_to_c(executable),
|
|
0,
|
|
};
|
|
run_command(arena, (CStringSlice) array_to_slice(run_arguments), envp);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void entry_point(int argc, char* argv[], char* envp[])
|
|
{
|
|
if (argc < 2)
|
|
{
|
|
print("Expected some arguments\n");
|
|
failed_execution();
|
|
}
|
|
|
|
Arena* arena = arena_init_default(KB(64));
|
|
|
|
CompilerBackend preferred_compiler_backend = COMPILER_BACKEND_COUNT;
|
|
Command command = COMMAND_COUNT;
|
|
u8 test_every_config = 0;
|
|
String source_file_path = {};
|
|
CMakeBuildType build_type = CMAKE_BUILD_TYPE_COUNT;
|
|
String release_strings[CMAKE_BUILD_TYPE_COUNT] = {
|
|
[CMAKE_BUILD_TYPE_DEBUG] = strlit("Debug"),
|
|
[CMAKE_BUILD_TYPE_MIN_SIZE_RELEASE] = strlit("MinSizeRel"),
|
|
[CMAKE_BUILD_TYPE_RELEASE_WITH_DEBUG_INFO] = strlit("RelWithDebInfo"),
|
|
[CMAKE_BUILD_TYPE_RELEASE] = strlit("Release"),
|
|
};
|
|
|
|
for (int i = 1; i < argc; i += 1)
|
|
{
|
|
char* c_argument = argv[i];
|
|
auto argument = cstr(c_argument);
|
|
|
|
if (one_char_string_to_compiler_backend(argument) != COMPILER_BACKEND_COUNT)
|
|
{
|
|
preferred_compiler_backend = one_char_string_to_compiler_backend(argument);
|
|
}
|
|
else if (string_starts_with(argument, strlit("build_type=")))
|
|
{
|
|
auto release_start = cast_to(u32, s32, string_first_ch(argument, '=') + 1);
|
|
auto release_string = s_get_slice(u8, argument, release_start, argument.length);
|
|
|
|
for (u64 i = 0; i < array_length(release_strings); i += 1)
|
|
{
|
|
if (s_equal(release_string, release_strings[i]))
|
|
{
|
|
build_type = (CMakeBuildType)i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
assert(build_type != CMAKE_BUILD_TYPE_COUNT);
|
|
}
|
|
else if (s_equal(argument, strlit("test")))
|
|
{
|
|
command = COMMAND_RUN_TESTS;
|
|
}
|
|
else if (s_equal(argument, strlit("debug")))
|
|
{
|
|
command = COMMAND_DEBUG;
|
|
}
|
|
else if (s_equal(argument, strlit("all")))
|
|
{
|
|
test_every_config = 1;
|
|
}
|
|
}
|
|
|
|
auto index = 2 - (command == COMMAND_COUNT);
|
|
if (argc > index)
|
|
{
|
|
auto* c_argument = argv[index];
|
|
auto argument = cstr(c_argument);
|
|
String expected_starts[] = {
|
|
strlit("tests/"),
|
|
strlit("tests\\"),
|
|
strlit("./tests/"),
|
|
strlit(".\\tests\\"),
|
|
strlit("src/"),
|
|
strlit("src\\"),
|
|
strlit("./src/"),
|
|
strlit(".\\src\\"),
|
|
};
|
|
|
|
for (u32 i = 0; i < array_length(expected_starts); i += 1)
|
|
{
|
|
auto expected_start = expected_starts[i];
|
|
if (expected_start.length < argument.length)
|
|
{
|
|
// TODO: make our own function
|
|
if (strncmp(c_argument, string_to_c(expected_start), expected_start.length) == 0)
|
|
{
|
|
source_file_path = argument;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (command == COMMAND_COUNT && !source_file_path.pointer)
|
|
{
|
|
print("Expected a command\n");
|
|
failed_execution();
|
|
}
|
|
|
|
if (command == COMMAND_COUNT)
|
|
{
|
|
command = COMMAND_RUN_TESTS;
|
|
test_every_config = 1;
|
|
}
|
|
|
|
if ((command == COMMAND_DEBUG) | ((command == COMMAND_RUN_TESTS) & (test_every_config == 0)))
|
|
{
|
|
if (preferred_compiler_backend == COMPILER_BACKEND_COUNT)
|
|
{
|
|
preferred_compiler_backend = COMPILER_BACKEND_BB;
|
|
}
|
|
}
|
|
|
|
if (build_type == CMAKE_BUILD_TYPE_COUNT)
|
|
{
|
|
build_type = CMAKE_BUILD_TYPE_DEBUG;
|
|
}
|
|
|
|
String compiler_path = strlit(BUILD_DIR "/" COMPILER_NAME);
|
|
|
|
switch (command)
|
|
{
|
|
case COMMAND_DEBUG:
|
|
if (!source_file_path.pointer)
|
|
{
|
|
source_file_path = strlit("foo");
|
|
// failed_execution();
|
|
}
|
|
|
|
run(arena, envp, compiler_path, preferred_compiler_backend, 1, string_to_c(source_file_path));
|
|
break;
|
|
case COMMAND_RUN_TESTS:
|
|
{
|
|
String every_single_test[] = {
|
|
strlit("tests/first.nat"),
|
|
// strlit("tests/add_sub.nat"),
|
|
// strlit("tests/mul.nat"),
|
|
// strlit("tests/div.nat"),
|
|
// strlit("tests/and.nat"),
|
|
// strlit("tests/or.nat"),
|
|
// strlit("tests/xor.nat"),
|
|
// strlit("tests/return_var.nat"),
|
|
// strlit("tests/return_mod_scope.nat"),
|
|
// strlit("tests/shift_left.nat"),
|
|
// strlit("tests/shift_right.nat"),
|
|
// strlit("tests/thousand_simple_functions.nat"),
|
|
// strlit("tests/simple_arg.nat"),
|
|
// strlit("tests/comparison.nat"),
|
|
};
|
|
CompilerBackend all_compiler_backends[] = {
|
|
COMPILER_BACKEND_BB,
|
|
COMPILER_BACKEND_LLVM,
|
|
};
|
|
static_assert(array_length(all_compiler_backends) == COMPILER_BACKEND_COUNT);
|
|
|
|
Slice(CompilerBackend) compiler_backend_selection;
|
|
|
|
if (test_every_config)
|
|
{
|
|
compiler_backend_selection = (Slice(CompilerBackend)) array_to_slice(all_compiler_backends);
|
|
}
|
|
else
|
|
{
|
|
compiler_backend_selection = (Slice(CompilerBackend)) { .pointer = &preferred_compiler_backend, .length = 1 };
|
|
}
|
|
|
|
Slice(String) test_selection;
|
|
if (source_file_path.pointer)
|
|
{
|
|
test_selection = (Slice(String)) { .pointer = &source_file_path, .length = 1 };
|
|
}
|
|
else
|
|
{
|
|
test_selection = (Slice(String)) array_to_slice(every_single_test);
|
|
}
|
|
|
|
run_tests(arena, compiler_path, &(TestOptions) {
|
|
.test_paths = test_selection,
|
|
.compiler_backends = compiler_backend_selection,
|
|
}, envp);
|
|
} break;
|
|
case COMMAND_COUNT:
|
|
unreachable();
|
|
}
|
|
}
|