diff options
| author | Jakub Konka <kubkon@jakubkonka.com> | 2022-02-21 23:31:55 +0100 |
|---|---|---|
| committer | Jakub Konka <kubkon@jakubkonka.com> | 2022-02-22 21:56:34 +0100 |
| commit | 261711722175447d402632fd7f55b371bfc72df2 (patch) | |
| tree | 93a12c77f9d7929c1c3e08b031791f7b2093c5cf /src/codegen.zig | |
| parent | 87a0d754be9a26aeabb90f25d2971d263680fe9e (diff) | |
| download | zig-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.zig | 41 |
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 = {} }; |
