Files
sx/examples/221-try.sx
agra aa1aa63bb3 ERR/E1.4a: standalone try sema + pure-failable propagation + named widening
`try f()` (standalone form) now propagates a failable callee's error to the
enclosing failable function. E1.4 was split: E1.4a = standalone try (failure
target = function-propagation); E1.4b = fallback-target routing +
failable-`or` + whole-program SCC for inferred sets + empty-inferred warning.

- lowerExpr: `.try_expr` -> lowerTry
- lowerTry: (1) try legal only inside a failable fn; (2) the sole
  failable-operand check (errorChannelOf(inferExprType(operand))); (3)
  named-caller widening (checkErrorSetSubset at the propagation site); (4)
  pure-failable lowering — condBr on tag != 0: propagate (run defers + ret
  the widened tag) vs continue on success
- inferExprType: `.try_expr` arm (success type: void for pure-failable)
- lowerBinaryOp .or_op: bail loudly on a failable LHS (exprIsFailable);
  the optional-`or` path is unchanged for non-failable LHS
- value-carrying callee/caller `try` bail loudly (pending E2's tuple ABI)

Tests: examples/221-try.sx (positive propagation, exit 5),
examples/222-try-rejections.sx (3 stable rejections: outside-failable,
non-failable operand, named-widening miss; exit 1). Gates: zig build,
zig build test, 260/260 examples.
2026-05-31 19:47:19 +03:00

33 lines
1.2 KiB
Plaintext

// First runnable `try` (ERR step E1.4a). The STANDALONE form: a failable
// expression whose failure propagates to the enclosing function's error
// return (Zig-style). `outer` calls `try inner(n)` — on `inner`'s failure
// `outer` returns that error; on success it continues. Both are pure
// failable (`-> !E`). The error-channel tuple ABI for value-carrying
// `-> (T, !)` and `try` in an `or` chain land in ERR E1.4b/E2.
#import "modules/std.sx";
E :: error { Bad, Worse }
inner :: (n: s32) -> !E {
if n < 0 { raise error.Bad; }
return; // success — no error
}
// Propagates inner's error (standalone `try`, target = function return).
outer :: (n: s32) -> !E {
try inner(n);
return;
}
main :: () -> s32 {
bad := outer(-1); // inner raises Bad -> outer propagates
good := outer(7); // inner succeeds -> outer succeeds
r : s32 = 0;
if bad == error.Bad { r = r + 5; } // true -> +5
if good == error.Bad { r = r + 1; } // false (success = no error)
if bad == error.Worse { r = r + 2; } // false (propagated Bad)
print("try result: {}\n", r); // -> 5
return r;
}