aboutsummaryrefslogtreecommitdiff
path: root/src/llvm_backend_proc.cpp
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2024-07-14 13:44:47 +0100
committergingerBill <bill@gingerbill.org>2024-07-14 13:44:47 +0100
commit6959554040d85597300ab2ce6c25852d18e61923 (patch)
tree43cd72fb83e5bdc03240805be3704964d51618a7 /src/llvm_backend_proc.cpp
parent0a530b5ce8642508c5df2fdb9ec43682be07d5cd (diff)
Calculate size and alignment, and reuse memory for all variadic calls within a procedure body
Diffstat (limited to 'src/llvm_backend_proc.cpp')
-rw-r--r--src/llvm_backend_proc.cpp25
1 files changed, 17 insertions, 8 deletions
diff --git a/src/llvm_backend_proc.cpp b/src/llvm_backend_proc.cpp
index bc85b14c2..7a895fbdd 100644
--- a/src/llvm_backend_proc.cpp
+++ b/src/llvm_backend_proc.cpp
@@ -3456,39 +3456,48 @@ gb_internal lbValue lb_build_call_expr_internal(lbProcedure *p, Ast *expr) {
}
isize slice_len = var_args.count;
if (slice_len > 0) {
- lbAddr base_array = {};
lbAddr slice = {};
for (auto const &vr : p->variadic_reuses) {
if (are_types_identical(vr.slice_type, slice_type)) {
- base_array = vr.base_array;
slice = vr.slice_addr;
break;
}
}
+
DeclInfo *d = decl_info_of_entity(p->entity);
- if (d != nullptr && base_array.addr.value == nullptr) {
+ if (d != nullptr && slice.addr.value == nullptr) {
for (auto const &vr : d->variadic_reuses) {
if (are_types_identical(vr.slice_type, slice_type)) {
- base_array = lb_add_local_generated(p, alloc_type_array(elem_type, vr.max_count), true);
slice = lb_add_local_generated(p, slice_type, true);
- array_add(&p->variadic_reuses, lbVariadicReuseData{slice_type, base_array, slice});
+ array_add(&p->variadic_reuses, lbVariadicReuseSlices{slice_type, slice});
break;
}
}
}
- GB_ASSERT(base_array.addr.value != nullptr);
+
+ lbValue base_array_ptr = p->variadic_reuse_base_array_ptr.addr;
+ if (d != nullptr && base_array_ptr.value == nullptr) {
+ i64 max_bytes = d->variadic_reuse_max_bytes;
+ i64 max_align = gb_max(d->variadic_reuse_max_align, 16);
+ p->variadic_reuse_base_array_ptr = lb_add_local_generated(p, alloc_type_array(t_u8, max_bytes), true);
+ lb_try_update_alignment(p->variadic_reuse_base_array_ptr.addr, cast(unsigned)max_align);
+ base_array_ptr = p->variadic_reuse_base_array_ptr.addr;
+ }
+
+ GB_ASSERT(base_array_ptr.value != nullptr);
GB_ASSERT(slice.addr.value != nullptr);
+ base_array_ptr = lb_emit_conv(p, base_array_ptr, alloc_type_pointer(alloc_type_array(elem_type, slice_len)));
for (isize i = 0; i < var_args.count; i++) {
- lbValue addr = lb_emit_array_epi(p, base_array.addr, cast(i32)i);
+ lbValue addr = lb_emit_array_epi(p, base_array_ptr, cast(i32)i);
lbValue var_arg = var_args[i];
var_arg = lb_emit_conv(p, var_arg, elem_type);
lb_emit_store(p, addr, var_arg);
}
- lbValue base_elem = lb_emit_array_epi(p, base_array.addr, 0);
+ lbValue base_elem = lb_emit_array_epi(p, base_array_ptr, 0);
lbValue len = lb_const_int(p->module, t_int, slice_len);
lb_fill_slice(p, slice, base_elem, len);