✓
Passing This code compiles and runs correctly.
Code
// NEGATIVE: a single re-issued <owned!> handle is consumed TWICE on the same
// back-edge path (use-after-discharge). The `again` branch issues one <owned!>
// (`v`); routing it into BOTH a disposing consume AND @loop consumes the same
// obligation twice. Linearity is one-issue/one-consume, so this MUST be rejected.
// Grammar grounded against 330_072_obligation_transfer_through_loop (transfer
// shape, <!owned> consume marker) and 210_123 (#loop / @loop fold form).
const std = @import("std");
const Handle = struct { n: i32 };
~event make {}
| made *Handle<owned!>
~proc make|zig {
const h = std.heap.page_allocator.create(Handle) catch unreachable;
h.* = .{ .n = 0 };
return .{ .made = h };
}
~event step { h: *Handle<!owned> }
| again *Handle<owned!>
| stop *Handle<owned!>
~proc step|zig {
h.n += 1;
if (h.n < 3) {
return .{ .again = h };
}
return .{ .stop = h };
}
// A consuming/disposing leaf.
~event drop { h: *Handle<!owned> }
~proc drop|zig {
std.heap.page_allocator.destroy(h);
}
~event spin { h: *Handle<!owned> }
| finished *Handle<owned!>
~spin = #loop step(h)
| again v |> drop(h: v) |> @loop(h: v) // consumes v twice -> double consume
| stop r => finished r
~make()
| made h0 |> spin(h: h0)
| finished hf |> drop(h: hf)
Backend must reject with:
CONTAINS error[KORU030]
CONTAINS carries no obligation here
CONTAINS this event consumesError Verification
Actual Compiler Output
error[KORU030]: Phantom state mismatch: argument 'h' carries no obligation here, but this event consumes '<!owned>'. Pass a value that still holds its '<owned!>' obligation.
--> phantom_semantic_check:42:0
❌ Compiler coordination error: Phantom semantic validation failed
error: CompilerCoordinationFailed
/Users/larsde/src/koru/tests/regression/300_ADVANCED_FEATURES/330_PHANTOM_TYPES/330_079_reject_double_consume_carried/backend.zig:94:13: 0x10335f20b in emit (backend)
return error.CompilerCoordinationFailed;
^
/Users/larsde/src/koru/tests/regression/300_ADVANCED_FEATURES/330_PHANTOM_TYPES/330_079_reject_double_consume_carried/backend.zig:190:28: 0x10335fef7 in main (backend)
const generated_code = try RuntimeEmitter.emit(compile_allocator, final_ast);
^Flows
subflow ~spin click a branch to expand · @labels scroll to their anchor
#loop step (h)
flow ~make click a branch to expand · @labels scroll to their anchor
make
Test Configuration
MUST_FAIL