aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2023-01-27 11:47:00 +0000
committergingerBill <bill@gingerbill.org>2023-01-27 11:47:00 +0000
commitccf4b48865f8d3c6f78f72cda5cfd5e46b108c02 (patch)
tree01d633a6a33ea0169aba0e8ec91e66fe1cd6e2ba /src
parent96eae9410337854b1384d9868f7098603c9647c2 (diff)
Add extra checks for multiple assignments when emitting stores
Diffstat (limited to 'src')
-rw-r--r--src/llvm_backend.hpp1
-rw-r--r--src/llvm_backend_general.cpp12
-rw-r--r--src/llvm_backend_stmt.cpp14
3 files changed, 22 insertions, 5 deletions
diff --git a/src/llvm_backend.hpp b/src/llvm_backend.hpp
index 85f1078f2..e151d2be1 100644
--- a/src/llvm_backend.hpp
+++ b/src/llvm_backend.hpp
@@ -324,6 +324,7 @@ struct lbProcedure {
lbBlock * curr_block;
lbTargetList * target_list;
PtrMap<Entity *, lbValue> direct_parameters;
+ bool in_multi_assignment;
Ast *curr_stmt;
diff --git a/src/llvm_backend_general.cpp b/src/llvm_backend_general.cpp
index 2c171239a..73cf0d1b9 100644
--- a/src/llvm_backend_general.cpp
+++ b/src/llvm_backend_general.cpp
@@ -937,7 +937,7 @@ gb_internal void lb_emit_store(lbProcedure *p, lbValue ptr, lbValue value) {
enum {MAX_STORE_SIZE = 64};
- if (lb_sizeof(LLVMTypeOf(value.value)) > MAX_STORE_SIZE) {
+ if (!p->in_multi_assignment && lb_sizeof(LLVMTypeOf(value.value)) > MAX_STORE_SIZE) {
if (LLVMIsALoadInst(value.value)) {
LLVMValueRef dst_ptr = ptr.value;
LLVMValueRef src_ptr_original = LLVMGetOperand(value.value, 0);
@@ -964,15 +964,16 @@ gb_internal void lb_emit_store(lbProcedure *p, lbValue ptr, lbValue value) {
}
}
+ LLVMValueRef instr = nullptr;
if (lb_is_type_proc_recursive(a)) {
// NOTE(bill, 2020-11-11): Because of certain LLVM rules, a procedure value may be
// stored as regular pointer with no procedure information
LLVMTypeRef rawptr_type = lb_type(p->module, t_rawptr);
LLVMTypeRef rawptr_ptr_type = LLVMPointerType(rawptr_type, 0);
- LLVMBuildStore(p->builder,
- LLVMBuildPointerCast(p->builder, value.value, rawptr_type, ""),
- LLVMBuildPointerCast(p->builder, ptr.value, rawptr_ptr_type, ""));
+ instr = LLVMBuildStore(p->builder,
+ LLVMBuildPointerCast(p->builder, value.value, rawptr_type, ""),
+ LLVMBuildPointerCast(p->builder, ptr.value, rawptr_ptr_type, ""));
} else {
Type *ca = core_type(a);
if (ca->kind == Type_Basic || ca->kind == Type_Proc) {
@@ -981,8 +982,9 @@ gb_internal void lb_emit_store(lbProcedure *p, lbValue ptr, lbValue value) {
GB_ASSERT_MSG(are_types_identical(a, value.type), "%s != %s", type_to_string(a), type_to_string(value.type));
}
- LLVMBuildStore(p->builder, value.value, ptr.value);
+ instr = LLVMBuildStore(p->builder, value.value, ptr.value);
}
+ LLVMSetVolatile(instr, p->in_multi_assignment);
}
gb_internal LLVMTypeRef llvm_addr_type(lbModule *module, lbValue addr_val) {
diff --git a/src/llvm_backend_stmt.cpp b/src/llvm_backend_stmt.cpp
index c268ab09a..2284649e2 100644
--- a/src/llvm_backend_stmt.cpp
+++ b/src/llvm_backend_stmt.cpp
@@ -1555,12 +1555,25 @@ gb_internal void lb_build_assignment(lbProcedure *p, Array<lbAddr> &lvals, Slice
lb_append_tuple_values(p, &inits, init);
}
+ bool prev_in_assignment = p->in_multi_assignment;
+
+ isize lval_count = 0;
+ for (lbAddr const &lval : lvals) {
+ if (lval.addr.value != nullptr) {
+ // check if it is not a blank identifier
+ lval_count += 1;
+ }
+ }
+ p->in_multi_assignment = lval_count > 1;
+
GB_ASSERT(lvals.count == inits.count);
for_array(i, inits) {
lbAddr lval = lvals[i];
lbValue init = inits[i];
lb_addr_store(p, lval, init);
}
+
+ p->in_multi_assignment = prev_in_assignment;
}
gb_internal void lb_build_return_stmt_internal(lbProcedure *p, lbValue res) {
@@ -2047,6 +2060,7 @@ gb_internal void lb_build_assign_stmt(lbProcedure *p, AstAssignStmt *as) {
lb_build_assignment(p, lvals, as->rhs);
return;
}
+
GB_ASSERT(as->lhs.count == 1);
GB_ASSERT(as->rhs.count == 1);
// NOTE(bill): Only 1 += 1 is allowed, no tuples