diff options
Diffstat (limited to 'src/codegen.zig')
| -rw-r--r-- | src/codegen.zig | 120 |
1 files changed, 119 insertions, 1 deletions
diff --git a/src/codegen.zig b/src/codegen.zig index fd4321fee9..26a4478fb2 100644 --- a/src/codegen.zig +++ b/src/codegen.zig @@ -205,13 +205,62 @@ pub fn generateSymbol( .appended => {}, .externally_managed => |slice| { code.appendSliceAssumeCapacity(slice); - return Result{ .appended = {} }; }, .fail => |em| return Result{ .fail = em }, } } return Result{ .appended = {} }; }, + .repeated => { + const array = typed_value.val.castTag(.repeated).?.data; + const elem_ty = typed_value.ty.childType(); + const sentinel = typed_value.ty.sentinel(); + const len = typed_value.ty.arrayLen(); + + var index: u64 = 0; + while (index < len) : (index += 1) { + switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{ + .ty = elem_ty, + .val = array, + }, code, debug_output)) { + .appended => {}, + .externally_managed => |slice| { + code.appendSliceAssumeCapacity(slice); + }, + .fail => |em| return Result{ .fail = em }, + } + } + + if (sentinel) |sentinel_val| { + switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{ + .ty = elem_ty, + .val = sentinel_val, + }, code, debug_output)) { + .appended => {}, + .externally_managed => |slice| { + code.appendSliceAssumeCapacity(slice); + }, + .fail => |em| return Result{ .fail = em }, + } + } + + return Result{ .appended = {} }; + }, + .empty_array_sentinel => { + const elem_ty = typed_value.ty.childType(); + const sentinel_val = typed_value.ty.sentinel().?; + switch (try generateSymbol(bin_file, parent_atom_index, src_loc, .{ + .ty = elem_ty, + .val = sentinel_val, + }, code, debug_output)) { + .appended => {}, + .externally_managed => |slice| { + code.appendSliceAssumeCapacity(slice); + }, + .fail => |em| return Result{ .fail = em }, + } + return Result{ .appended = {} }; + }, else => return Result{ .fail = try ErrorMsg.create( bin_file.allocator, @@ -432,6 +481,75 @@ pub fn generateSymbol( return Result{ .appended = {} }; }, + .ErrorUnion => { + const error_ty = typed_value.ty.errorUnionSet(); + const payload_ty = typed_value.ty.errorUnionPayload(); + const is_payload = typed_value.val.errorUnionIsPayload(); + + 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, + }, 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); + } + } + + return Result{ .appended = {} }; + }, + .ErrorSet => { + const target = bin_file.options.target; + switch (typed_value.val.tag()) { + .@"error" => { + const name = typed_value.val.getError().?; + const kv = try bin_file.options.module.?.getErrorValue(name); + const endian = target.cpu.arch.endian(); + try code.writer().writeInt(u32, kv.value, endian); + }, + else => { + try code.writer().writeByteNTimes(0, @intCast(usize, typed_value.ty.abiSize(target))); + }, + } + return Result{ .appended = {} }; + }, else => |t| { return Result{ .fail = try ErrorMsg.create( |
