Fork the process when compiling libraries to avoid cross-contamination
of one library's bindings leaking into another's
This commit is contained in:
parent
d888bec409
commit
da4d18e7e8
49
src/tomo.c
49
src/tomo.c
@ -89,6 +89,7 @@ static void build_library(Text_t lib_dir_name);
|
||||
static void compile_files(env_t *env, Array_t files, Array_t *object_files, Array_t *ldlibs);
|
||||
static bool is_stale(Path_t path, Path_t relative_to);
|
||||
static Path_t build_file(Path_t path, const char *extension);
|
||||
static void wait_for_child_success(pid_t child);
|
||||
|
||||
typedef struct {
|
||||
bool h:1, c:1, o:1;
|
||||
@ -197,10 +198,13 @@ int main(int argc, char *argv[])
|
||||
}
|
||||
|
||||
for (int64_t i = 0; i < libraries.length; i++) {
|
||||
// Fork a child process to build the library to prevent cross-contamination
|
||||
// of side effects when building one library from affecting another library.
|
||||
// This *could* be done in parallel, but there may be some dependency issues.
|
||||
pid_t child = fork();
|
||||
if (child == 0) {
|
||||
Path_t *lib = (Path_t*)(libraries.data + i*libraries.stride);
|
||||
const char *lib_str = Path$as_c_string(*lib);
|
||||
char cwd[PATH_MAX];
|
||||
getcwd(cwd, sizeof(cwd));
|
||||
if (chdir(lib_str) != 0)
|
||||
print_err("Could not enter directory: ", lib_str);
|
||||
|
||||
@ -208,7 +212,9 @@ int main(int argc, char *argv[])
|
||||
getcwd(libdir, sizeof(libdir));
|
||||
char *libdirname = basename(libdir);
|
||||
build_library(Text$from_str(libdirname));
|
||||
chdir(cwd);
|
||||
_exit(0);
|
||||
}
|
||||
wait_for_child_success(child);
|
||||
}
|
||||
|
||||
// TODO: REPL
|
||||
@ -265,17 +271,7 @@ int main(int argc, char *argv[])
|
||||
print_err("Could not execute program: ", prog_args[0]);
|
||||
}
|
||||
|
||||
int status;
|
||||
while (waitpid(child, &status, 0) < 0 && errno == EINTR) {
|
||||
if (WIFEXITED(status) || WIFSIGNALED(status))
|
||||
break;
|
||||
else if (WIFSTOPPED(status))
|
||||
kill(child, SIGCONT);
|
||||
}
|
||||
|
||||
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
|
||||
_exit(WIFEXITED(status) ? WEXITSTATUS(status) : EXIT_FAILURE);
|
||||
}
|
||||
wait_for_child_success(child);
|
||||
}
|
||||
|
||||
if (compile_exe && should_install) {
|
||||
@ -291,6 +287,21 @@ int main(int argc, char *argv[])
|
||||
#pragma GCC diagnostic pop
|
||||
#endif
|
||||
|
||||
void wait_for_child_success(pid_t child)
|
||||
{
|
||||
int status;
|
||||
while (waitpid(child, &status, 0) < 0 && errno == EINTR) {
|
||||
if (WIFEXITED(status) || WIFSIGNALED(status))
|
||||
break;
|
||||
else if (WIFSTOPPED(status))
|
||||
kill(child, SIGCONT);
|
||||
}
|
||||
|
||||
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
|
||||
_exit(WIFEXITED(status) ? WEXITSTATUS(status) : EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
|
||||
Text_t escape_lib_name(Text_t lib_name)
|
||||
{
|
||||
char *libname_id = String(lib_name);
|
||||
@ -511,7 +522,6 @@ void compile_files(env_t *env, Array_t to_compile, Array_t *object_files, Array_
|
||||
build_file_dependency_graph(filename, &dependency_files, &to_link);
|
||||
}
|
||||
|
||||
int status;
|
||||
// (Re)compile header files, eagerly for explicitly passed in files, lazily
|
||||
// for downstream dependencies:
|
||||
for (int64_t i = 0; i < dependency_files.entries.length; i++) {
|
||||
@ -552,13 +562,8 @@ void compile_files(env_t *env, Array_t to_compile, Array_t *object_files, Array_
|
||||
child_processes = new(struct child_s, .next=child_processes, .pid=pid);
|
||||
}
|
||||
|
||||
for (; child_processes; child_processes = child_processes->next) {
|
||||
waitpid(child_processes->pid, &status, 0);
|
||||
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0)
|
||||
exit(EXIT_FAILURE);
|
||||
else if (WIFSTOPPED(status))
|
||||
kill(child_processes->pid, SIGCONT);
|
||||
}
|
||||
for (; child_processes; child_processes = child_processes->next)
|
||||
wait_for_child_success(child_processes->pid);
|
||||
|
||||
if (object_files) {
|
||||
for (int64_t i = 0; i < dependency_files.entries.length; i++) {
|
||||
|
Loading…
Reference in New Issue
Block a user