aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorJakub Konka <kubkon@jakubkonka.com>2022-02-20 16:35:24 +0100
committerJakub Konka <kubkon@jakubkonka.com>2022-02-22 21:56:34 +0100
commitf4f23e307c7a493435bed155359a5f3a2b57d965 (patch)
tree05eb45a407bf3703148aaf059a84634b8b166c12 /src
parent358b5441570916a48f2f7df9ebe8b9cb5f87676b (diff)
downloadzig-f4f23e307c7a493435bed155359a5f3a2b57d965.tar.gz
zig-f4f23e307c7a493435bed155359a5f3a2b57d965.zip
codegen: lower error_set and error_union
Diffstat (limited to 'src')
-rw-r--r--src/arch/x86_64/CodeGen.zig9
-rw-r--r--src/codegen.zig48
-rw-r--r--src/link/Elf.zig1
3 files changed, 52 insertions, 6 deletions
diff --git a/src/arch/x86_64/CodeGen.zig b/src/arch/x86_64/CodeGen.zig
index c172f998d6..be70c0be01 100644
--- a/src/arch/x86_64/CodeGen.zig
+++ b/src/arch/x86_64/CodeGen.zig
@@ -3509,6 +3509,7 @@ fn isErr(self: *Self, ty: Type, operand: MCValue) !MCValue {
return self.fail("TODO isErr for errors with size larger than register size", .{});
}
} else {
+ log.warn("operand = {}, payload_type = {}", .{ operand, payload_type });
return self.fail("TODO isErr for non-empty payloads", .{});
}
}
@@ -5108,22 +5109,18 @@ fn genTypedValue(self: *Self, typed_value: TypedValue) InnerError!MCValue {
const error_type = typed_value.ty.errorUnionSet();
const payload_type = typed_value.ty.errorUnionPayload();
- if (typed_value.val.castTag(.eu_payload)) |pl| {
+ if (typed_value.val.castTag(.eu_payload)) |_| {
if (!payload_type.hasRuntimeBits()) {
// We use the error type directly as the type.
return MCValue{ .immediate = 0 };
}
-
- _ = pl;
- return self.fail("TODO implement error union const of type '{}' (non-error)", .{typed_value.ty});
} else {
if (!payload_type.hasRuntimeBits()) {
// We use the error type directly as the type.
return self.genTypedValue(.{ .ty = error_type, .val = typed_value.val });
}
}
-
- return self.fail("TODO implement error union const of type '{}' (error)", .{typed_value.ty});
+ return self.lowerUnnamedConst(typed_value);
},
.Struct => {
return self.lowerUnnamedConst(typed_value);
diff --git a/src/codegen.zig b/src/codegen.zig
index fd4321fee9..9a8eb7abf2 100644
--- a/src/codegen.zig
+++ b/src/codegen.zig
@@ -432,6 +432,54 @@ 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 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 },
+ }
+
+ if (payload_ty.hasRuntimeBits()) {
+ const payload_val = if (typed_value.val.castTag(.eu_payload)) |val| val.data else Value.initTag(.undef);
+ 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 },
+ }
+ }
+
+ 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(
diff --git a/src/link/Elf.zig b/src/link/Elf.zig
index 08903f7c05..4be478331a 100644
--- a/src/link/Elf.zig
+++ b/src/link/Elf.zig
@@ -3127,6 +3127,7 @@ pub fn lowerUnnamedConst(self: *Elf, typed_value: TypedValue, decl: *Module.Decl
.fail => |em| {
decl.analysis = .codegen_failure;
try module.failed_decls.put(module.gpa, decl, em);
+ log.err("{s}", .{em.msg});
return error.AnalysisFail;
},
};