diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2025-06-30 15:25:35 +0200 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2025-06-30 15:25:35 +0200 |
| commit | 79db39307bea14c9d43b365535e704d2a86f3b24 (patch) | |
| tree | e8e28732fab6a70ca261dbb13654be44a202881e | |
| parent | e8377659560497beb36ec1f981b024f74292c1b9 (diff) | |
| parent | d9742acc232a2af4f98e0fbb9f749737a126e6d1 (diff) | |
| download | zig-79db39307bea14c9d43b365535e704d2a86f3b24.tar.gz zig-79db39307bea14c9d43b365535e704d2a86f3b24.zip | |
Merge pull request #24297 from ziglang/optional-empty-error-set
Sema: correct OPV for optional empty error set
| -rw-r--r-- | src/Air.zig | 22 | ||||
| -rw-r--r-- | src/Sema.zig | 43 | ||||
| -rw-r--r-- | test/cases/compile_errors/optional_empty_error_set.zig | 13 | ||||
| -rw-r--r-- | test/cases/safety/optional_empty_error_set.zig | 22 |
4 files changed, 80 insertions, 20 deletions
diff --git a/src/Air.zig b/src/Air.zig index 561ab86405..5a5070276c 100644 --- a/src/Air.zig +++ b/src/Air.zig @@ -1141,6 +1141,20 @@ pub const Inst = struct { pub fn toType(ref: Ref) Type { return .fromInterned(ref.toInterned().?); } + + pub fn fromIntern(ip_index: InternPool.Index) Ref { + return switch (ip_index) { + .none => .none, + else => { + assert(@intFromEnum(ip_index) >> 31 == 0); + return @enumFromInt(@as(u31, @intCast(@intFromEnum(ip_index)))); + }, + }; + } + + pub fn fromValue(v: Value) Ref { + return .fromIntern(v.toIntern()); + } }; /// All instructions have an 8-byte payload, which is contained within @@ -1754,13 +1768,7 @@ pub fn deinit(air: *Air, gpa: std.mem.Allocator) void { } pub fn internedToRef(ip_index: InternPool.Index) Inst.Ref { - return switch (ip_index) { - .none => .none, - else => { - assert(@intFromEnum(ip_index) >> 31 == 0); - return @enumFromInt(@as(u31, @intCast(@intFromEnum(ip_index)))); - }, - }; + return .fromIntern(ip_index); } /// Returns `null` if runtime-known. diff --git a/src/Sema.zig b/src/Sema.zig index 20be9e7052..5b2e846ff0 100644 --- a/src/Sema.zig +++ b/src/Sema.zig @@ -8060,7 +8060,7 @@ fn analyzeCall( }; try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.Call).@"struct".fields.len + runtime_args.len); - const result = try block.addInst(.{ + const maybe_opv = try block.addInst(.{ .tag = call_tag, .data = .{ .pl_op = .{ .operand = runtime_func, @@ -8072,7 +8072,7 @@ fn analyzeCall( sema.appendRefsAssumeCapacity(runtime_args); if (ensure_result_used) { - try sema.ensureResultUsed(block, sema.typeOf(result), call_src); + try sema.ensureResultUsed(block, sema.typeOf(maybe_opv), call_src); } if (call_tag == .call_always_tail) { @@ -8082,10 +8082,10 @@ fn analyzeCall( .pointer => func_or_ptr_ty.childType(zcu), else => unreachable, }; - return sema.handleTailCall(block, call_src, runtime_func_ty, result); + return sema.handleTailCall(block, call_src, runtime_func_ty, maybe_opv); } - if (resolved_ret_ty.toIntern() == .noreturn_type) { + if (ip.isNoReturn(resolved_ret_ty.toIntern())) { const want_check = c: { if (!block.wantSafety()) break :c false; if (func_val != null) break :c false; @@ -8099,6 +8099,11 @@ fn analyzeCall( return .unreachable_value; } + const result: Air.Inst.Ref = if (try sema.typeHasOnePossibleValue(sema.typeOf(maybe_opv))) |opv| + .fromValue(opv) + else + maybe_opv; + return result; } @@ -8335,7 +8340,7 @@ fn analyzeCall( break :result try sema.resolveAnalyzedBlock(block, call_src, &child_block, &inlining.merges, need_debug_scope); }; - const result: Air.Inst.Ref = if (try sema.resolveValue(result_raw)) |result_val| r: { + const maybe_opv: Air.Inst.Ref = if (try sema.resolveValue(result_raw)) |result_val| r: { const val_resolved = try sema.resolveAdHocInferredErrorSet(block, call_src, result_val.toIntern()); break :r Air.internedToRef(val_resolved); } else r: { @@ -8347,7 +8352,7 @@ fn analyzeCall( }; if (block.isComptime()) { - const result_val = (try sema.resolveValue(result)).?; + const result_val = (try sema.resolveValue(maybe_opv)).?; if (want_memoize and sema.allow_memoize and !result_val.canMutateComptimeVarState(zcu)) { _ = try pt.intern(.{ .memoized_call = .{ .func = func_val.?.toIntern(), @@ -8359,10 +8364,10 @@ fn analyzeCall( } if (ensure_result_used) { - try sema.ensureResultUsed(block, sema.typeOf(result), call_src); + try sema.ensureResultUsed(block, sema.typeOf(maybe_opv), call_src); } - return result; + return maybe_opv; } fn handleTailCall(sema: *Sema, block: *Block, call_src: LazySrcLoc, func_ty: Type, result: Air.Inst.Ref) !Air.Inst.Ref { @@ -9065,10 +9070,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 +36452,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 +36663,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 |
