2024-12-20 13:47:15 -06:00

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();
}
}