aboutsummaryrefslogtreecommitdiff
path: root/src/codegen.zig
diff options
context:
space:
mode:
authorJakub Konka <kubkon@jakubkonka.com>2022-02-21 23:31:55 +0100
committerJakub Konka <kubkon@jakubkonka.com>2022-02-22 21:56:34 +0100
commit261711722175447d402632fd7f55b371bfc72df2 (patch)
tree93a12c77f9d7929c1c3e08b031791f7b2093c5cf /src/codegen.zig
parent87a0d754be9a26aeabb90f25d2971d263680fe9e (diff)
downloadzig-261711722175447d402632fd7f55b371bfc72df2.tar.gz
zig-261711722175447d402632fd7f55b371bfc72df2.zip
x64: fix lowering of error unions (we didn't pad to alignment)
* fix returning large values on stack from procedure calls - we need to explicitly specify source and dest base registers for `genSetStack` as well
Diffstat (limited to 'src/codegen.zig')
-rw-r--r--src/codegen.zig41
1 files changed, 31 insertions, 10 deletions
diff --git a/src/codegen.zig b/src/codegen.zig
index 1c1881bc32..26a4478fb2 100644
--- a/src/codegen.zig
+++ b/src/codegen.zig
@@ -486,20 +486,34 @@ pub fn generateSymbol(
const payload_ty = typed_value.ty.errorUnionPayload();
const is_payload = typed_value.val.errorUnionIsPayload();
- const error_val = if (!is_payload) typed_value.val else Value.initTag(.zero);
- switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{
- .ty = error_ty,
- .val = error_val,
- }, code, debug_output)) {
- .appended => {},
- .externally_managed => |external_slice| {
- code.appendSliceAssumeCapacity(external_slice);
- },
- .fail => |em| return Result{ .fail = em },
+ const target = bin_file.options.target;
+ const abi_align = typed_value.ty.abiAlignment(target);
+
+ {
+ const error_val = if (!is_payload) typed_value.val else Value.initTag(.zero);
+ const begin = code.items.len;
+ switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{
+ .ty = error_ty,
+ .val = error_val,
+ }, code, debug_output)) {
+ .appended => {},
+ .externally_managed => |external_slice| {
+ code.appendSliceAssumeCapacity(external_slice);
+ },
+ .fail => |em| return Result{ .fail = em },
+ }
+ const unpadded_end = code.items.len - begin;
+ const padded_end = mem.alignForwardGeneric(u64, unpadded_end, abi_align);
+ const padding = try math.cast(usize, padded_end - unpadded_end);
+
+ if (padding > 0) {
+ try code.writer().writeByteNTimes(0, padding);
+ }
}
if (payload_ty.hasRuntimeBits()) {
const payload_val = if (typed_value.val.castTag(.eu_payload)) |val| val.data else Value.initTag(.undef);
+ const begin = code.items.len;
switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{
.ty = payload_ty,
.val = payload_val,
@@ -510,6 +524,13 @@ pub fn generateSymbol(
},
.fail => |em| return Result{ .fail = em },
}
+ const unpadded_end = code.items.len - begin;
+ const padded_end = mem.alignForwardGeneric(u64, unpadded_end, abi_align);
+ const padding = try math.cast(usize, padded_end - unpadded_end);
+
+ if (padding > 0) {
+ try code.writer().writeByteNTimes(0, padding);
+ }
}
return Result{ .appended = {} };