aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAndrew Kelley <andrew@ziglang.org>2025-06-29 16:39:26 -0700
committerAndrew Kelley <andrew@ziglang.org>2025-06-29 17:20:23 -0700
commit7999374b21795a7b8fac7af0b2e326d39bd1e839 (patch)
tree14a3344acff87876a43897211912115f592b771d
parente8377659560497beb36ec1f981b024f74292c1b9 (diff)
downloadzig-7999374b21795a7b8fac7af0b2e326d39bd1e839.tar.gz
zig-7999374b21795a7b8fac7af0b2e326d39bd1e839.zip
Sema: correct OPV for optional empty error set
prevents crashes in backends; improves codegen; provides more comptime-ness.
-rw-r--r--src/Sema.zig22
-rw-r--r--test/cases/compile_errors/optional_empty_error_set.zig13
-rw-r--r--test/cases/safety/optional_empty_error_set.zig22
3 files changed, 52 insertions, 5 deletions
diff --git a/src/Sema.zig b/src/Sema.zig
index 20be9e7052..68890c1f6f 100644
--- a/src/Sema.zig
+++ b/src/Sema.zig
@@ -9065,10 +9065,14 @@ fn zirOptionalPayload(
};
if (try sema.resolveDefinedValue(block, src, operand)) |val| {
- return if (val.optionalValue(zcu)) |payload|
- Air.internedToRef(payload.toIntern())
- else
- sema.fail(block, src, "unable to unwrap null", .{});
+ if (val.optionalValue(zcu)) |payload| return Air.internedToRef(payload.toIntern());
+ if (block.isComptime()) return sema.fail(block, src, "unable to unwrap null", .{});
+ if (safety_check and block.wantSafety()) {
+ try sema.safetyPanic(block, src, .unwrap_null);
+ } else {
+ _ = try block.addNoOp(.unreach);
+ }
+ return .unreachable_value;
}
try sema.requireRuntimeBlock(block, src, null);
@@ -36443,7 +36447,6 @@ pub fn typeHasOnePossibleValue(sema: *Sema, ty: Type) CompileError!?Value {
.type_int_unsigned, // u0 handled above
.type_pointer,
.type_slice,
- .type_optional, // ?noreturn handled above
.type_anyframe,
.type_error_union,
.type_anyerror_union,
@@ -36655,6 +36658,15 @@ pub fn typeHasOnePossibleValue(sema: *Sema, ty: Type) CompileError!?Value {
else => unreachable,
},
+
+ .type_optional => {
+ const payload_ip = ip.indexToKey(ty.toIntern()).opt_type;
+ // Although ?noreturn is handled above, the element type
+ // can be effectively noreturn for example via an empty
+ // enum or error set.
+ if (ip.isNoReturn(payload_ip)) return try pt.nullValue(ty);
+ return null;
+ },
},
};
}
diff --git a/test/cases/compile_errors/optional_empty_error_set.zig b/test/cases/compile_errors/optional_empty_error_set.zig
new file mode 100644
index 0000000000..9fac5fde6a
--- /dev/null
+++ b/test/cases/compile_errors/optional_empty_error_set.zig
@@ -0,0 +1,13 @@
+export fn example() void {
+ comptime foo() catch |err| switch (err) {};
+}
+var x: ?error{} = null;
+fn foo() !void {
+ return x.?;
+}
+// error
+// backend=stage2
+// target=native
+//
+// :6:13: error: unable to unwrap null
+// :2:17: note: called at comptime here
diff --git a/test/cases/safety/optional_empty_error_set.zig b/test/cases/safety/optional_empty_error_set.zig
new file mode 100644
index 0000000000..dbe39d00c3
--- /dev/null
+++ b/test/cases/safety/optional_empty_error_set.zig
@@ -0,0 +1,22 @@
+const std = @import("std");
+
+pub fn panic(message: []const u8, stack_trace: ?*std.builtin.StackTrace, ra: ?usize) noreturn {
+ _ = stack_trace;
+ _ = ra;
+ if (std.mem.eql(u8, message, "attempt to use null value")) {
+ std.process.exit(0);
+ }
+ std.process.exit(1);
+}
+
+pub fn main() !void {
+ foo() catch |err| switch (err) {};
+ return error.TestFailed;
+}
+var x: ?error{} = null;
+fn foo() !void {
+ return x.?;
+}
+// run
+// backend=stage2,llvm
+// target=native