stage2: Pop error trace when storing error to var/const

In order to enforce a strict stack discipline for error return traces,
we cannot track error return traces that are stored in variables:

  ```zig
  const x = errorable(); // errorable()'s error return trace is killed here

  // v-- error trace starts here instead
  return x catch error.UnknownError;
  ```

In order to propagate error return traces, function calls need to be passed
directly to an error-handling expression (`if`, `catch`, `try` or `return`):

  ```zig
  // When passed directly to `catch`, the return trace is propagated
  return errorable() catch error.UnknownError;

  // Using a break also works
  return blk: {
      // code here
      break :blk errorable();
  } catch error.UnknownError;
  ```

Why do we need this restriction? Without it, multiple errors can co-exist
with their own error traces. Handling that situation correctly means either:
  a. Dynamically allocating trace memory and tracking lifetimes, OR
  b. Allowing the production of one error to interfere with the trace of another
     (which is the current status quo)

This is piece (3/3) of https://github.com/ziglang/zig/issues/1923#issuecomment-1218495574
This commit is contained in:
Cody Tapscott 2022-09-13 15:31:07 -07:00
parent 0c3a50fe1c
commit 3007fdde45
5 changed files with 147 additions and 42 deletions

View file

@ -44,23 +44,24 @@ pub fn main() void {
if (!have_tty) { if (!have_tty) {
std.debug.print("{d}/{d} {s}... ", .{ i + 1, test_fn_list.len, test_fn.name }); std.debug.print("{d}/{d} {s}... ", .{ i + 1, test_fn_list.len, test_fn.name });
} }
const result = if (test_fn.async_frame_size) |size| switch (io_mode) { if (result: {
.evented => blk: { if (test_fn.async_frame_size) |size| switch (io_mode) {
if (async_frame_buffer.len < size) { .evented => {
std.heap.page_allocator.free(async_frame_buffer); if (async_frame_buffer.len < size) {
async_frame_buffer = std.heap.page_allocator.alignedAlloc(u8, std.Target.stack_align, size) catch @panic("out of memory"); std.heap.page_allocator.free(async_frame_buffer);
} async_frame_buffer = std.heap.page_allocator.alignedAlloc(u8, std.Target.stack_align, size) catch @panic("out of memory");
const casted_fn = @ptrCast(fn () callconv(.Async) anyerror!void, test_fn.func); }
break :blk await @asyncCall(async_frame_buffer, {}, casted_fn, .{}); const casted_fn = @ptrCast(fn () callconv(.Async) anyerror!void, test_fn.func);
}, break :result await @asyncCall(async_frame_buffer, {}, casted_fn, .{});
.blocking => { },
skip_count += 1; .blocking => {
test_node.end(); skip_count += 1;
progress.log("SKIP (async test)\n", .{}); test_node.end();
continue; progress.log("SKIP (async test)\n", .{});
}, continue;
} else test_fn.func(); },
if (result) |_| { } else break :result test_fn.func();
}) |_| {
ok_count += 1; ok_count += 1;
test_node.end(); test_node.end();
if (!have_tty) std.debug.print("OK\n", .{}); if (!have_tty) std.debug.print("OK\n", .{});

View file

@ -8545,9 +8545,22 @@ fn callExpr(
scratch_index += 1; scratch_index += 1;
} }
// If our result location is a try/catch/error-union-if/return, the error trace propagates.
// Otherwise, it should always be popped (handled in Sema).
const propagate_error_trace = switch (rl) {
.catch_none, .catch_ref => true, // Propagate to try/catch/error-union-if
.ptr, .ty => |ref| b: { // Otherwise, propagate if result loc is a return
const inst = refToIndex(ref) orelse break :b false;
const zir_tags = astgen.instructions.items(.tag);
break :b zir_tags[inst] == .ret_ptr or zir_tags[inst] == .ret_type;
},
else => false,
};
const payload_index = try addExtra(astgen, Zir.Inst.Call{ const payload_index = try addExtra(astgen, Zir.Inst.Call{
.callee = callee, .callee = callee,
.flags = .{ .flags = .{
.pop_error_return_trace = !propagate_error_trace,
.packed_modifier = @intCast(Zir.Inst.Call.Flags.PackedModifier, @enumToInt(modifier)), .packed_modifier = @intCast(Zir.Inst.Call.Flags.PackedModifier, @enumToInt(modifier)),
.args_len = @intCast(Zir.Inst.Call.Flags.PackedArgsLen, call.ast.params.len), .args_len = @intCast(Zir.Inst.Call.Flags.PackedArgsLen, call.ast.params.len),
}, },

View file

@ -5664,6 +5664,7 @@ fn zirCall(
const modifier = @intToEnum(std.builtin.CallOptions.Modifier, extra.data.flags.packed_modifier); const modifier = @intToEnum(std.builtin.CallOptions.Modifier, extra.data.flags.packed_modifier);
const ensure_result_used = extra.data.flags.ensure_result_used; const ensure_result_used = extra.data.flags.ensure_result_used;
const pop_error_return_trace = extra.data.flags.pop_error_return_trace;
var func = try sema.resolveInst(extra.data.callee); var func = try sema.resolveInst(extra.data.callee);
var resolved_args: []Air.Inst.Ref = undefined; var resolved_args: []Air.Inst.Ref = undefined;
@ -5771,7 +5772,7 @@ fn zirCall(
resolved_args[arg_index] = resolved; resolved_args[arg_index] = resolved;
} }
return sema.analyzeCall(block, func, func_src, call_src, modifier, ensure_result_used, resolved_args, bound_arg_src); return sema.analyzeCall(block, func, func_src, call_src, modifier, ensure_result_used, pop_error_return_trace, resolved_args, bound_arg_src);
} }
const GenericCallAdapter = struct { const GenericCallAdapter = struct {
@ -5883,6 +5884,7 @@ fn analyzeCall(
call_src: LazySrcLoc, call_src: LazySrcLoc,
modifier: std.builtin.CallOptions.Modifier, modifier: std.builtin.CallOptions.Modifier,
ensure_result_used: bool, ensure_result_used: bool,
pop_error_return_trace: bool,
uncasted_args: []const Air.Inst.Ref, uncasted_args: []const Air.Inst.Ref,
bound_arg_src: ?LazySrcLoc, bound_arg_src: ?LazySrcLoc,
) CompileError!Air.Inst.Ref { ) CompileError!Air.Inst.Ref {
@ -6333,19 +6335,55 @@ fn analyzeCall(
sema.owner_func.?.calls_or_awaits_errorable_fn = true; sema.owner_func.?.calls_or_awaits_errorable_fn = true;
} }
try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.Call).Struct.fields.len + const backend_supports_error_return_tracing = sema.mod.comp.bin_file.options.use_llvm;
args.len); const emit_error_trace_save_restore = sema.mod.comp.bin_file.options.error_return_tracing and
const func_inst = try block.addInst(.{ backend_supports_error_return_tracing and
.tag = call_tag, pop_error_return_trace and func_ty_info.return_type.isError();
.data = .{ .pl_op = .{
.operand = func, if (emit_error_trace_save_restore) {
.payload = sema.addExtraAssumeCapacity(Air.Call{ // This function call is error-able (and so can generate an error trace), but AstGen determined
.args_len = @intCast(u32, args.len), // that its result does not go to an error-handling operator (try/catch/return etc.). We need to
}), // save and restore the error trace index here, effectively "popping" the new entries immediately.
} },
}); const unresolved_stack_trace_ty = try sema.getBuiltinType(block, call_src, "StackTrace");
sema.appendRefsAssumeCapacity(args); const stack_trace_ty = try sema.resolveTypeFields(block, call_src, unresolved_stack_trace_ty);
break :res func_inst; const ptr_stack_trace_ty = try Type.Tag.single_mut_pointer.create(sema.arena, stack_trace_ty);
const err_return_trace = try block.addTy(.err_return_trace, ptr_stack_trace_ty);
const field_ptr = try sema.structFieldPtr(block, call_src, err_return_trace, "index", call_src, stack_trace_ty, true);
const saved_index = try sema.analyzeLoad(block, call_src, field_ptr, call_src);
try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.Call).Struct.fields.len +
args.len);
const func_inst = try block.addInst(.{
.tag = call_tag,
.data = .{ .pl_op = .{
.operand = func,
.payload = sema.addExtraAssumeCapacity(Air.Call{
.args_len = @intCast(u32, args.len),
}),
} },
});
sema.appendRefsAssumeCapacity(args);
try sema.storePtr2(block, call_src, field_ptr, call_src, saved_index, call_src, .store);
break :res func_inst;
} else {
try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.Call).Struct.fields.len +
args.len);
const func_inst = try block.addInst(.{
.tag = call_tag,
.data = .{ .pl_op = .{
.operand = func,
.payload = sema.addExtraAssumeCapacity(Air.Call{
.args_len = @intCast(u32, args.len),
}),
} },
});
sema.appendRefsAssumeCapacity(args);
break :res func_inst;
}
}; };
if (ensure_result_used) { if (ensure_result_used) {
@ -10927,7 +10965,7 @@ fn maybeErrorUnwrap(sema: *Sema, block: *Block, body: []const Zir.Inst.Index, op
const panic_fn = try sema.getBuiltin(block, src, "panicUnwrapError"); const panic_fn = try sema.getBuiltin(block, src, "panicUnwrapError");
const err_return_trace = try sema.getErrorReturnTrace(block, src); const err_return_trace = try sema.getErrorReturnTrace(block, src);
const args: [2]Air.Inst.Ref = .{ err_return_trace, operand }; const args: [2]Air.Inst.Ref = .{ err_return_trace, operand };
_ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, false, &args, null);
return true; return true;
}, },
.panic => { .panic => {
@ -10938,7 +10976,7 @@ fn maybeErrorUnwrap(sema: *Sema, block: *Block, body: []const Zir.Inst.Index, op
const panic_fn = try sema.getBuiltin(block, src, "panic"); const panic_fn = try sema.getBuiltin(block, src, "panic");
const err_return_trace = try sema.getErrorReturnTrace(block, src); const err_return_trace = try sema.getErrorReturnTrace(block, src);
const args: [3]Air.Inst.Ref = .{ msg_inst, err_return_trace, .null_value }; const args: [3]Air.Inst.Ref = .{ msg_inst, err_return_trace, .null_value };
_ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, false, &args, null);
return true; return true;
}, },
else => unreachable, else => unreachable,
@ -16141,7 +16179,7 @@ fn retWithErrTracing(
const args: [1]Air.Inst.Ref = .{err_return_trace}; const args: [1]Air.Inst.Ref = .{err_return_trace};
if (!need_check) { if (!need_check) {
_ = try sema.analyzeCall(block, return_err_fn, src, src, .never_inline, false, &args, null); _ = try sema.analyzeCall(block, return_err_fn, src, src, .never_inline, false, false, &args, null);
_ = try block.addUnOp(ret_tag, operand); _ = try block.addUnOp(ret_tag, operand);
return always_noreturn; return always_noreturn;
} }
@ -16152,7 +16190,7 @@ fn retWithErrTracing(
var else_block = block.makeSubBlock(); var else_block = block.makeSubBlock();
defer else_block.instructions.deinit(gpa); defer else_block.instructions.deinit(gpa);
_ = try sema.analyzeCall(&else_block, return_err_fn, src, src, .never_inline, false, &args, null); _ = try sema.analyzeCall(&else_block, return_err_fn, src, src, .never_inline, false, false, &args, null);
_ = try else_block.addUnOp(ret_tag, operand); _ = try else_block.addUnOp(ret_tag, operand);
try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.CondBr).Struct.fields.len + try sema.air_extra.ensureUnusedCapacity(gpa, @typeInfo(Air.CondBr).Struct.fields.len +
@ -20369,7 +20407,7 @@ fn zirBuiltinCall(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError
} }
} }
const ensure_result_used = extra.flags.ensure_result_used; const ensure_result_used = extra.flags.ensure_result_used;
return sema.analyzeCall(block, func, func_src, call_src, modifier, ensure_result_used, resolved_args, bound_arg_src); return sema.analyzeCall(block, func, func_src, call_src, modifier, ensure_result_used, false, resolved_args, bound_arg_src);
} }
fn zirFieldParentPtr(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref { fn zirFieldParentPtr(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref {
@ -21803,7 +21841,7 @@ fn panicWithMsg(
Value.@"null", Value.@"null",
); );
const args: [3]Air.Inst.Ref = .{ msg_inst, null_stack_trace, .null_value }; const args: [3]Air.Inst.Ref = .{ msg_inst, null_stack_trace, .null_value };
_ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(block, panic_fn, src, src, .auto, false, false, &args, null);
return always_noreturn; return always_noreturn;
} }
@ -21844,7 +21882,7 @@ fn panicUnwrapError(
const err = try fail_block.addTyOp(unwrap_err_tag, Type.anyerror, operand); const err = try fail_block.addTyOp(unwrap_err_tag, Type.anyerror, operand);
const err_return_trace = try sema.getErrorReturnTrace(&fail_block, src); const err_return_trace = try sema.getErrorReturnTrace(&fail_block, src);
const args: [2]Air.Inst.Ref = .{ err_return_trace, err }; const args: [2]Air.Inst.Ref = .{ err_return_trace, err };
_ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, false, &args, null);
} }
} }
try sema.addSafetyCheckExtra(parent_block, ok, &fail_block); try sema.addSafetyCheckExtra(parent_block, ok, &fail_block);
@ -21885,7 +21923,7 @@ fn panicIndexOutOfBounds(
} else { } else {
const panic_fn = try sema.getBuiltin(&fail_block, src, "panicOutOfBounds"); const panic_fn = try sema.getBuiltin(&fail_block, src, "panicOutOfBounds");
const args: [2]Air.Inst.Ref = .{ index, len }; const args: [2]Air.Inst.Ref = .{ index, len };
_ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, false, &args, null);
} }
} }
try sema.addSafetyCheckExtra(parent_block, ok, &fail_block); try sema.addSafetyCheckExtra(parent_block, ok, &fail_block);
@ -21927,7 +21965,7 @@ fn panicSentinelMismatch(
else { else {
const panic_fn = try sema.getBuiltin(parent_block, src, "checkNonScalarSentinel"); const panic_fn = try sema.getBuiltin(parent_block, src, "checkNonScalarSentinel");
const args: [2]Air.Inst.Ref = .{ expected_sentinel, actual_sentinel }; const args: [2]Air.Inst.Ref = .{ expected_sentinel, actual_sentinel };
_ = try sema.analyzeCall(parent_block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(parent_block, panic_fn, src, src, .auto, false, false, &args, null);
return; return;
}; };
const gpa = sema.gpa; const gpa = sema.gpa;
@ -21956,7 +21994,7 @@ fn panicSentinelMismatch(
} else { } else {
const panic_fn = try sema.getBuiltin(&fail_block, src, "panicSentinelMismatch"); const panic_fn = try sema.getBuiltin(&fail_block, src, "panicSentinelMismatch");
const args: [2]Air.Inst.Ref = .{ expected_sentinel, actual_sentinel }; const args: [2]Air.Inst.Ref = .{ expected_sentinel, actual_sentinel };
_ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, &args, null); _ = try sema.analyzeCall(&fail_block, panic_fn, src, src, .auto, false, false, &args, null);
} }
} }
try sema.addSafetyCheckExtra(parent_block, ok, &fail_block); try sema.addSafetyCheckExtra(parent_block, ok, &fail_block);

View file

@ -2825,10 +2825,11 @@ pub const Inst = struct {
pub const Flags = packed struct { pub const Flags = packed struct {
/// std.builtin.CallOptions.Modifier in packed form /// std.builtin.CallOptions.Modifier in packed form
pub const PackedModifier = u3; pub const PackedModifier = u3;
pub const PackedArgsLen = u28; pub const PackedArgsLen = u27;
packed_modifier: PackedModifier, packed_modifier: PackedModifier,
ensure_result_used: bool = false, ensure_result_used: bool = false,
pop_error_return_trace: bool,
args_len: PackedArgsLen, args_len: PackedArgsLen,
comptime { comptime {

View file

@ -208,6 +208,58 @@ pub fn addCases(cases: *tests.StackTracesContext) void {
}, },
}); });
cases.addCase(.{
.name = "stored errors do not contribute to error trace",
.source =
\\fn foo() !void {
\\ return error.TheSkyIsFalling;
\\}
\\
\\pub fn main() !void {
\\ // Once an error is stored in a variable, it is popped from the trace
\\ var x = foo();
\\ x = {};
\\
\\ // As a result, this error trace will still be clean
\\ return error.SomethingUnrelatedWentWrong;
\\}
,
.Debug = .{
.expect =
\\error: SomethingUnrelatedWentWrong
\\source.zig:11:5: [address] in main (test)
\\ return error.SomethingUnrelatedWentWrong;
\\ ^
\\
,
},
.ReleaseSafe = .{
.exclude_os = .{
.windows, // TODO
.linux, // defeated by aggressive inlining
},
.expect =
\\error: SomethingUnrelatedWentWrong
\\source.zig:11:5: [address] in [function]
\\ return error.SomethingUnrelatedWentWrong;
\\ ^
\\
,
},
.ReleaseFast = .{
.expect =
\\error: SomethingUnrelatedWentWrong
\\
,
},
.ReleaseSmall = .{
.expect =
\\error: SomethingUnrelatedWentWrong
\\
,
},
});
cases.addCase(.{ cases.addCase(.{
.name = "try return from within catch", .name = "try return from within catch",
.source = .source =