Browse Source

Handle missing procedures better

gingerBill 3 weeks ago
parent
commit
9b8771b475
4 changed files with 21 additions and 11 deletions
  1. 11 5
      src/llvm_backend.cpp
  2. 3 3
      src/llvm_backend.hpp
  3. 7 2
      src/llvm_backend_general.cpp
  4. 0 1
      src/llvm_backend_proc.cpp

+ 11 - 5
src/llvm_backend.cpp

@@ -12,7 +12,7 @@
 #endif
 #endif
 
 
 #ifndef LLVM_WEAK_MONOMORPHIZATION
 #ifndef LLVM_WEAK_MONOMORPHIZATION
-#define LLVM_WEAK_MONOMORPHIZATION build_context.internal_weak_monomorphization
+#define LLVM_WEAK_MONOMORPHIZATION 1
 #endif
 #endif
 
 
 
 
@@ -2478,8 +2478,14 @@ gb_internal void lb_generate_procedures(lbGenerator *gen, bool do_threading) {
 gb_internal WORKER_TASK_PROC(lb_generate_missing_procedures_to_check_worker_proc) {
 gb_internal WORKER_TASK_PROC(lb_generate_missing_procedures_to_check_worker_proc) {
 	lbModule *m = cast(lbModule *)data;
 	lbModule *m = cast(lbModule *)data;
 	for (lbProcedure *p = nullptr; mpsc_dequeue(&m->missing_procedures_to_check, &p); /**/) {
 	for (lbProcedure *p = nullptr; mpsc_dequeue(&m->missing_procedures_to_check, &p); /**/) {
-		debugf("Generate missing procedure: %.*s module %p\n", LIT(p->name), m);
-		lb_generate_procedure(m, p);
+		if (!p->is_done.load(std::memory_order_relaxed)) {
+			debugf("Generate missing procedure: %.*s module %p\n", LIT(p->name), m);
+			lb_generate_procedure(m, p);
+		}
+
+		for (lbProcedure *nested = nullptr; mpsc_dequeue(&m->procedures_to_generate, &nested); /**/) {
+			mpsc_enqueue(&m->missing_procedures_to_check, nested);
+		}
 	}
 	}
 	return 0;
 	return 0;
 }
 }
@@ -2860,7 +2866,7 @@ gb_internal lbProcedure *lb_create_main_procedure(lbModule *m, lbProcedure *star
 }
 }
 
 
 gb_internal void lb_generate_procedure(lbModule *m, lbProcedure *p) {
 gb_internal void lb_generate_procedure(lbModule *m, lbProcedure *p) {
-	if (p->is_done) {
+	if (p->is_done.load(std::memory_order_relaxed)) {
 		return;
 		return;
 	}
 	}
 
 
@@ -2869,7 +2875,7 @@ gb_internal void lb_generate_procedure(lbModule *m, lbProcedure *p) {
 		lb_begin_procedure_body(p);
 		lb_begin_procedure_body(p);
 		lb_build_stmt(p, p->body);
 		lb_build_stmt(p, p->body);
 		lb_end_procedure_body(p);
 		lb_end_procedure_body(p);
-		p->is_done = true;
+		p->is_done.store(true, std::memory_order_relaxed);
 		m->curr_procedure = nullptr;
 		m->curr_procedure = nullptr;
 	} else if (p->generate_body != nullptr) {
 	} else if (p->generate_body != nullptr) {
 		p->generate_body(m, p);
 		p->generate_body(m, p);

+ 3 - 3
src/llvm_backend.hpp

@@ -366,9 +366,9 @@ struct lbProcedure {
 
 
 	lbFunctionType *abi_function_type;
 	lbFunctionType *abi_function_type;
 
 
-	LLVMValueRef    value;
-	LLVMBuilderRef  builder;
-	bool            is_done;
+	LLVMValueRef      value;
+	LLVMBuilderRef    builder;
+	std::atomic<bool> is_done;
 
 
 	lbAddr           return_ptr;
 	lbAddr           return_ptr;
 	Array<lbDefer>   defer_stmts;
 	Array<lbDefer>   defer_stmts;

+ 7 - 2
src/llvm_backend_general.cpp

@@ -3082,9 +3082,12 @@ gb_internal lbValue lb_find_procedure_value_from_entity(lbModule *m, Entity *e)
 		if (found == nullptr) {
 		if (found == nullptr) {
 			// THIS IS THE RACE CONDITION
 			// THIS IS THE RACE CONDITION
 			lbProcedure *missing_proc_in_other_module = lb_create_procedure(other_module, e, false);
 			lbProcedure *missing_proc_in_other_module = lb_create_procedure(other_module, e, false);
-			mpsc_enqueue(&other_module->missing_procedures_to_check, missing_proc_in_other_module);
+			if (!missing_proc_in_other_module->is_done.load(std::memory_order_relaxed)) {
+				mpsc_enqueue(&other_module->missing_procedures_to_check, missing_proc_in_other_module);
+			}
 		}
 		}
 	} else {
 	} else {
+		GB_PANIC("missing procedure: %.*s", LIT(missing_proc->name));
 		mpsc_enqueue(&m->missing_procedures_to_check, missing_proc);
 		mpsc_enqueue(&m->missing_procedures_to_check, missing_proc);
 	}
 	}
 
 
@@ -3157,7 +3160,9 @@ gb_internal lbValue lb_generate_anonymous_proc_lit(lbModule *m, String const &pr
 		rw_mutex_shared_unlock(&target_module->values_mutex);
 		rw_mutex_shared_unlock(&target_module->values_mutex);
 		if (found == nullptr) {
 		if (found == nullptr) {
 			lbProcedure *missing_proc_in_target_module = lb_create_procedure(target_module, e, false);
 			lbProcedure *missing_proc_in_target_module = lb_create_procedure(target_module, e, false);
-			mpsc_enqueue(&target_module->missing_procedures_to_check, missing_proc_in_target_module);
+			if (!missing_proc_in_target_module->is_done.load(std::memory_order_relaxed)) {
+				mpsc_enqueue(&target_module->missing_procedures_to_check, missing_proc_in_target_module);
+			}
 		}
 		}
 
 
 		lbProcedure *p = lb_create_procedure(m, e, true);
 		lbProcedure *p = lb_create_procedure(m, e, true);

+ 0 - 1
src/llvm_backend_proc.cpp

@@ -99,7 +99,6 @@ gb_internal lbProcedure *lb_create_procedure(lbModule *m, Entity *entity, bool i
 		}
 		}
 	}
 	}
 
 
-
 	lbProcedure *p = gb_alloc_item(permanent_allocator(), lbProcedure);
 	lbProcedure *p = gb_alloc_item(permanent_allocator(), lbProcedure);
 
 
 	p->module = m;
 	p->module = m;