Support multiple build variants, add ARM variant

This commit is contained in:
2022-11-17 22:12:29 -05:00
parent e1048c3ca4
commit 5cf8a55156
2 changed files with 151 additions and 57 deletions

View File

@@ -1,17 +1,14 @@
# -*- python -*-
from pathlib import Path
from dataclasses import dataclass
#
# File lists
#
build_dir = "build/"
variant_dir = build_dir + "default/"
src_dir = "src/"
program_sources = ["src/main.c"]
lib_srcs = [
PROGRAM_SOURCES = ["src/main.c"]
LIB_SRCS = [
"src/arena.c",
"src/builtins.c",
"src/evaluate.c",
@@ -21,69 +18,159 @@ lib_srcs = [
"src/utility.c",
"src/parse.c",
]
lib_includes = ["src/"]
LIB_INCLUDES = ["src/"]
test_srcs = [
TEST_SRCS = [
"test/test_arena.c",
"test/test_e2e.c",
"test/test_parse.c",
"test/test_scope.c",
"test/test_utility.c",
]
test_lib_srcs = ["third-party/unity/src/unity.c"]
test_lib_includes = ["third-party/unity/src/"]
#
# Construct Environments
#
VariantDir(variant_dir, ".", duplicate=0)
# Minimization flags for later:
# CCFLAGS = ["-Oz", "-flto", "-ffunction-sections", "-fdata-sections", "-Wl,--gc-sections", "-lreadline"]
CCFLAGS = ["-ggdb", "-O0", "-Werror", "-Wall", "-Wextra", "-Wno-unused-parameter"]
env = Environment(
CPPPATH=lib_includes, COMPILATIONDB_USE_ABSPATH=True, CCFLAGS=CCFLAGS
)
env.Tool("compilation_db")
env.CompilationDatabase()
test_env = env.Clone()
test_env.Append(CPPPATH=test_lib_includes)
TEST_LIB_SRCS = ["third-party/unity/src/unity.c"]
TEST_LIB_INCLUDES = ["third-party/unity/src/"]
#
# Construct Environments
#
# Generate REPL
def with_suffix(path, suffix):
return str(Path(path).with_suffix(suffix))
lib_objs = [env.Object(p) for p in lib_srcs]
env.Program(variant_dir + "uclisp", lib_objs + program_sources, LIBS=["readline"])
@dataclass
class BuildVariant:
variant_name: str
cc: str
ccflags: list[str]
libs: list[str]
linkflags: list[str]
enable_compile_commands_db: bool = False
# Generate unit test executables
def built(self, path):
return self.variant_dir + path
test_lib_objs = [
test_env.Object(variant_dir + p, CPPPATH=lib_includes + test_lib_includes)
for p in test_lib_srcs
def enable_compile_commands(self):
self.env.Tool("compilation_db")
self.env.CompilationDatabase()
def configure_lib(self):
self.lib_objs = [
self.env.Object(target=self.built(with_suffix(p, ".o")), source=p)
for p in LIB_SRCS
]
test_deps = test_lib_objs + lib_objs
tests = [test_env.Program(variant_dir + p, [p] + test_deps) for p in test_srcs]
def configure_repl(self):
pgm_objs = [
self.env.Object(target=self.built(with_suffix(p, ".o")), source=p)
for p in PROGRAM_SOURCES
]
self.repl_program = self.env.Program(
self.built("uclisp"), self.lib_objs + pgm_objs
)
def configure_tests(self):
test_lib_objs = [
self.test_env.Object(target=self.built(with_suffix(p, ".o")), source=p)
for p in TEST_LIB_SRCS
]
test_deps = test_lib_objs + self.lib_objs
test_objs = [
self.test_env.Object(target=self.built(with_suffix(p, ".o")), source=p)
for p in TEST_SRCS
]
self.tests = [
self.test_env.Program(
self.built(with_suffix(p, "")),
test_deps + [self.built(with_suffix(p, ".o"))],
)
for p in TEST_SRCS
]
def configure(self):
self.variant_dir = f"build/{self.variant_name}/"
#
# Generate Test Runner script
# Construct Environments
#
env.Substfile(
"run_tests.sh.in",
self.env = Environment(
CC=self.cc,
CPPPATH=LIB_INCLUDES,
COMPILATIONDB_USE_ABSPATH=True,
CCFLAGS=self.ccflags,
LINKFLAGS=self.linkflags,
LIBS=self.libs,
)
if self.enable_compile_commands_db:
self.enable_compile_commands()
self.test_env = self.env.Clone()
self.test_env.Append(CPPPATH=TEST_LIB_INCLUDES)
self.configure_lib()
self.configure_repl()
self.configure_tests()
self.env.Substfile(
target=self.built("run_tests.sh"),
source="run_tests.sh.in",
SUBST_DICT={
"@tests@": " ".join(str(Path(str(test[0])).resolve()) for test in tests)
"@tests@": " ".join(
str(Path(str(test[0])).resolve()) for test in self.tests
)
},
)
env.Command(variant_dir + "run_tests", "run_tests.sh", Chmod("run_tests.sh", 0o755))
self.env.Command(
self.built("run_tests"),
self.built("run_tests.sh"),
Chmod(self.built("run_tests.sh"), 0o755),
)
# Copy default build compile commands to root, which is where tools seem to want
# it.
Copy("compile_commands.json", variant_dir + "compile_commands.json")
if self.enable_compile_commands_db:
Copy("compile_commands.json", self.built("compile_commands.json"))
self.env.Alias(self.variant_name, self.variant_dir)
base_ccflags = ["-Werror", "-Wall", "-Wextra", "-Wno-unused-parameter"]
debug_ccflags = ["-ggdb", "-O0"]
release_ccflags = [
"-Oz",
"-flto",
"-ffunction-sections",
"-fdata-sections",
"-Wl,--gc-sections",
]
variants = [
BuildVariant(
variant_name="debug",
cc="gcc",
ccflags=debug_ccflags + base_ccflags,
libs=["readline"],
linkflags=[],
enable_compile_commands_db=True,
),
BuildVariant(
variant_name="release",
cc="gcc",
ccflags=release_ccflags + base_ccflags,
linkflags=[],
libs=["readline"],
),
BuildVariant(
variant_name="arm-debug",
cc="arm-none-eabi-gcc",
ccflags=["-DNO_READLINE"] + debug_ccflags + base_ccflags,
libs=[],
linkflags=["--specs=nosys.specs"],
),
]
for variant in variants:
variant.configure()

View File

@@ -1,7 +1,10 @@
#include <stdio.h>
#include <stdlib.h>
#ifndef NO_READLINE
#include <readline/readline.h>
#include <readline/history.h>
#endif
#include "uclisp.h"
@@ -71,14 +74,18 @@ int main(int argc, const char **argv) {
if (argc < 2) {
while (1) {
char *line = readline("> ");
char *line = NULL;
#ifndef NO_READLINE
line = readline("> ");
if (line == NULL) {
break;
}
if (strlen(line) > 0) {
add_history(line);
}
#else
// TODO
#endif
struct ucl_object *sexp = ucl_parse(line);
struct ucl_object *result = NULL;
if (sexp->type == UCL_TYPE_ERROR) {