aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2023-01-05 12:29:16 +0000
committergingerBill <bill@gingerbill.org>2023-01-05 12:29:16 +0000
commit213a0499a1964e0bc5d2c48cd3b4450b45f59314 (patch)
tree0343fd1f2fdee38cdf0bff07e15ddcaab34832a3 /src
parent1517f1d7793c8985664600a820e3434dfdf83810 (diff)
Begin multithreading the llvm backend when `-use-separate-modules` is enabled
Diffstat (limited to 'src')
-rw-r--r--src/llvm_backend.cpp46
-rw-r--r--src/llvm_backend.hpp3
-rw-r--r--src/llvm_backend_general.cpp21
-rw-r--r--src/llvm_backend_stmt.cpp2
4 files changed, 53 insertions, 19 deletions
diff --git a/src/llvm_backend.cpp b/src/llvm_backend.cpp
index 192e5cc56..f3c4dd50d 100644
--- a/src/llvm_backend.cpp
+++ b/src/llvm_backend.cpp
@@ -730,6 +730,8 @@ gb_internal lbValue lb_map_set_proc_for_type(lbModule *m, Type *type) {
gb_internal lbValue lb_generate_anonymous_proc_lit(lbModule *m, String const &prefix_name, Ast *expr, lbProcedure *parent) {
+ MUTEX_GUARD(&m->gen->anonymous_proc_lits_mutex);
+
lbProcedure **found = map_get(&m->gen->anonymous_proc_lits, expr);
if (found) {
return lb_find_procedure_value_from_entity(m, (*found)->entity);
@@ -1526,6 +1528,10 @@ struct lbLLVMModulePassWorkerData {
gb_internal WORKER_TASK_PROC(lb_llvm_module_pass_worker_proc) {
auto wd = cast(lbLLVMModulePassWorkerData *)data;
+
+ lb_run_remove_unused_function_pass(wd->m);
+ lb_run_remove_unused_globals_pass(wd->m);
+
LLVMPassManagerRef module_pass_manager = LLVMCreatePassManager();
lb_populate_module_pass_manager(wd->target_machine, module_pass_manager, build_context.optimization_level);
LLVMRunPassManager(module_pass_manager, wd->m->mod);
@@ -2155,29 +2161,47 @@ gb_internal void lb_generate_code(lbGenerator *gen) {
}
}
+ isize non_empty_module_count = 0;
+ for (auto const &entry : gen->modules) {
+ lbModule *m = entry.value;
+ if (!lb_is_module_empty(m)) {
+ non_empty_module_count += 1;
+ }
+ }
+ if (non_empty_module_count <= 1) {
+ do_threading = false;
+ }
TIME_SECTION("LLVM Function Pass");
for (auto const &entry : gen->modules) {
lbModule *m = entry.value;
- lb_llvm_function_pass_worker_proc(m);
+ // if (do_threading) {
+ // thread_pool_add_task(lb_llvm_function_pass_worker_proc, m);
+ // } else {
+ lb_llvm_function_pass_worker_proc(m);
+ // }
}
+ thread_pool_wait();
TIME_SECTION("LLVM Module Pass");
-
for (auto const &entry : gen->modules) {
lbModule *m = entry.value;
- lb_run_remove_unused_function_pass(m);
- lb_run_remove_unused_globals_pass(m);
-
auto wd = gb_alloc_item(permanent_allocator(), lbLLVMModulePassWorkerData);
wd->m = m;
wd->target_machine = m->target_machine;
- lb_llvm_module_pass_worker_proc(wd);
+ if (do_threading) {
+ thread_pool_add_task(lb_llvm_module_pass_worker_proc, wd);
+ } else {
+ lb_llvm_module_pass_worker_proc(wd);
+ }
}
+ thread_pool_wait();
+ TIME_SECTION("LLVM Module Verification");
+
llvm_error = nullptr;
defer (LLVMDisposeMessage(llvm_error));
@@ -2245,21 +2269,13 @@ gb_internal void lb_generate_code(lbGenerator *gen) {
TIME_SECTION("LLVM Object Generation");
- isize non_empty_module_count = 0;
- for (auto const &entry : gen->modules) {
- lbModule *m = entry.value;
- if (!lb_is_module_empty(m)) {
- non_empty_module_count += 1;
- }
- }
-
if (build_context.ignore_llvm_build) {
gb_printf_err("LLVM SUCCESS!\n");
gb_exit(1);
return;
}
- if (do_threading && non_empty_module_count > 1) {
+ if (do_threading) {
for (auto const &entry : gen->modules) {
lbModule *m = entry.value;
if (lb_is_module_empty(m)) {
diff --git a/src/llvm_backend.hpp b/src/llvm_backend.hpp
index 9f7caa3bb..d824b99cf 100644
--- a/src/llvm_backend.hpp
+++ b/src/llvm_backend.hpp
@@ -132,6 +132,8 @@ struct lbModule {
PtrMap<void *, lbStructFieldRemapping> struct_field_remapping; // Key: LLVMTypeRef or Type *
i32 internal_type_level;
+ RecursiveMutex values_mutex;
+
PtrMap<Entity *, lbValue> values;
PtrMap<Entity *, lbAddr> soa_values;
StringMap<lbValue> members;
@@ -178,6 +180,7 @@ struct lbGenerator {
PtrMap<LLVMContextRef, lbModule *> modules_through_ctx;
lbModule default_module;
+ BlockingMutex anonymous_proc_lits_mutex;
PtrMap<Ast *, lbProcedure *> anonymous_proc_lits;
BlockingMutex foreign_mutex;
diff --git a/src/llvm_backend_general.cpp b/src/llvm_backend_general.cpp
index a849929f0..c09648825 100644
--- a/src/llvm_backend_general.cpp
+++ b/src/llvm_backend_general.cpp
@@ -316,6 +316,7 @@ gb_internal bool lb_is_instr_terminating(LLVMValueRef instr) {
gb_internal lbModule *lb_pkg_module(lbGenerator *gen, AstPackage *pkg) {
+ // NOTE(bill): no need for a mutex since it's immutable
auto *found = map_get(&gen->modules, pkg);
if (found) {
return *found;
@@ -1354,7 +1355,7 @@ gb_internal String lb_mangle_name(lbModule *m, Entity *e) {
return mangled_name;
}
-gb_internal String lb_set_nested_type_name_ir_mangled_name(Entity *e, lbProcedure *p) {
+gb_internal String lb_set_nested_type_name_ir_mangled_name(Entity *e, lbProcedure *p, lbModule *module) {
// NOTE(bill, 2020-03-08): A polymorphic procedure may take a nested type declaration
// and as a result, the declaration does not have time to determine what it should be
@@ -1421,7 +1422,7 @@ gb_internal String lb_get_entity_name(lbModule *m, Entity *e, String default_nam
}
if (e->kind == Entity_TypeName && (e->scope->flags & ScopeFlag_File) == 0) {
- return lb_set_nested_type_name_ir_mangled_name(e, nullptr);
+ return lb_set_nested_type_name_ir_mangled_name(e, nullptr, m);
}
String name = {};
@@ -2164,19 +2165,25 @@ gb_internal void lb_ensure_abi_function_type(lbModule *m, lbProcedure *p) {
gb_internal void lb_add_entity(lbModule *m, Entity *e, lbValue val) {
if (e != nullptr) {
+ mutex_lock(&m->values_mutex);
map_set(&m->values, e, val);
+ mutex_unlock(&m->values_mutex);
}
}
gb_internal void lb_add_member(lbModule *m, String const &name, lbValue val) {
if (name.len > 0) {
+ mutex_lock(&m->values_mutex);
string_map_set(&m->members, name, val);
+ mutex_unlock(&m->values_mutex);
}
}
gb_internal void lb_add_procedure_value(lbModule *m, lbProcedure *p) {
+ mutex_lock(&m->values_mutex);
if (p->entity != nullptr) {
map_set(&m->procedure_values, p->value, p->entity);
}
string_map_set(&m->procedures, p->name, p);
+ mutex_unlock(&m->values_mutex);
}
@@ -2519,6 +2526,8 @@ gb_internal lbValue lb_find_ident(lbProcedure *p, lbModule *m, Entity *e, Ast *e
return *found;
}
}
+ mutex_lock(&m->values_mutex);
+ defer (mutex_unlock(&m->values_mutex));
auto *found = map_get(&m->values, e);
if (found) {
@@ -2538,7 +2547,6 @@ gb_internal lbValue lb_find_ident(lbProcedure *p, lbModule *m, Entity *e, Ast *e
if (USE_SEPARATE_MODULES) {
lbModule *other_module = lb_pkg_module(m->gen, e->pkg);
if (other_module != m) {
-
String name = lb_get_entity_name(other_module, e);
lb_set_entity_from_other_modules_linkage_correctly(other_module, e, name);
@@ -2569,6 +2577,9 @@ gb_internal lbValue lb_find_procedure_value_from_entity(lbModule *m, Entity *e)
e = strip_entity_wrapping(e);
GB_ASSERT(e != nullptr);
+ mutex_lock(&m->values_mutex);
+ defer (mutex_unlock(&m->values_mutex));
+
auto *found = map_get(&m->values, e);
if (found) {
return *found;
@@ -2657,6 +2668,10 @@ gb_internal lbValue lb_find_value_from_entity(lbModule *m, Entity *e) {
return lb_find_procedure_value_from_entity(m, e);
}
+ mutex_lock(&m->values_mutex);
+ defer (mutex_unlock(&m->values_mutex));
+
+
auto *found = map_get(&m->values, e);
if (found) {
return *found;
diff --git a/src/llvm_backend_stmt.cpp b/src/llvm_backend_stmt.cpp
index 0e6f75118..73b4e251f 100644
--- a/src/llvm_backend_stmt.cpp
+++ b/src/llvm_backend_stmt.cpp
@@ -32,7 +32,7 @@ gb_internal void lb_build_constant_value_decl(lbProcedure *p, AstValueDecl *vd)
continue;
}
- lb_set_nested_type_name_ir_mangled_name(e, p);
+ lb_set_nested_type_name_ir_mangled_name(e, p, p->module);
}
for_array(i, vd->names) {