# CHECKPOINT-COMPILER-API — comptime `compiler` library (`#library "compiler"` + `abi(.zig) extern`) Companion to the design-of-record [../design/comptime-compiler-api.md](../design/comptime-compiler-api.md) (the plan + phased build order live there). This stream supersedes the metatype `declare`/`define`/`type_info` `#builtin`s and the `#compiler` struct attribute with ONE welded mechanism. Branch: `reify` (off `master`). Update after every step. ## ⏯ Resume (fresh session) > **✅ P5.7 DONE (2026-06-19) — the comptime VM is the SOLE evaluator; ZERO legacy.** `interp.zig` (the legacy > tagged-`Value` Interpreter) is DELETED; the `Value` result-DTO + `regToValue` materialization live in > `comptime_value.zig` (the VM↔host boundary), `valueToReg` is gone. NO fallback anywhere — a VM bail is always a > build-gating diagnostic (emit-time `#run`/const-init, lowering-time type-fn, `#insert`, inline comptime-call > fold all run on the VM only). `#compiler` attribute + `compiler_call` IR op + `compiler_hooks` `Registry`/`HookFn` > + all `hookXxx` are DELETED (`compiler_hooks.zig` keeps only `BuildConfig`/`BuildHooks`/`AssetDir`); `compiler_lib` > is just the name registry now. The metatype `declare`/`define` are sx over the compiler-API (`declare_type`/ > `register_type`); `type_info`/`field_type` stay builtins (documented). Bonus this arc: fixed issue 0141 (reject > the silent `[*]T → []T` coercion; land the corrected List-grown form 0640 + diagnostic 1183); empty-member types > are valid for all kinds (0641; never-defined `declare` still rejected 1179). 1654 reconciled to the VM wording. > **709/0 corpus + 476/476 unit, green; commits on `reify` from `5d25e23` (Step A) through `5383496`.** > REMAINING TAIL (P5.8, needs hardware/SDK): iOS-device + Android bundle validation + an `.apk` corpus smoke test. > OPTIONAL follow-up: re-express `type_info` as sx (reflect-into-value; kept as builtin to protect the 0619/0622/ > 0623 round-trips). See the newest `## Log` entries. > > **⚠ DIRECTION CHANGED (2026-06-17). The active plan is now > [`PLAN-COMPILER-VM.md`](PLAN-COMPILER-VM.md), NOT the weld.** > The **byte-weld + serialization/marshaling** approach is the wrong direction and is > being **stripped**. New foundation: a **bytecode VM over byte-addressable > memory** so comptime values are native bytes; then the compiler-API rides on it with > direct memory access (no weld, no validation, no marshaling). Everything below this > banner describes the now-superseded weld state (committed on `reify` through > `40d075c`) and is kept only to scope the Phase 0 strip. Read > `PLAN-COMPILER-VM.md` first. > > **Why the pivot:** the comptime evaluator (`src/ir/interp.zig`) represents values as > tagged `Value` unions, NOT native bytes — so a comptime `@ptrCast(*StructInfo)` > reads the `Value` union's memory, not a struct. The weld tried to bridge that with > hand-marshaling — exactly what the design set out to kill. Comptime memory makes comptime > values real bytes, so the bridge disappears. (JIT-native comptime was rejected: it > breaks cross-compilation — host vs target layout — and loses the sandbox. A > comptime VM keeps both while getting native bytes + speed.) > > **Next action (2026-06-18) — the WHOLE metatype surface is VM-native (steps 7+8, committed through > `d0ebc55`; step 8 uncommitted).** `declare`/`define`/`type_info` + tagged-union `enum_init` all run > NATIVELY on the VM (`.call_builtin` exec arm → `callBuiltinVm`; `defineFromInfo` decodes a > `TypeInfo` from comptime memory, `buildTypeInfo` reflects one INTO comptime memory — faithful ports of > legacy `defineEnum`/`Struct`/`Tuple`/`reflectTypeInfo`). The ENTIRE metatype range `0614`–`0624` + > `0632` runs **HANDLED with ZERO fallback** (incl. the `define(declare, type_info(T))` round-trips > `0619`/`0622`/`0623`); VM output byte-matches legacy. `enum_init`/`define`/`type_info` bail loudly > on a `backing_type` tagged union rather than silent-clobber. **697/0 BOTH gates + all unit tests.** > **THE NEXT STEP — Phase 4D.3 (`compiler_call` / #compiler hooks on the VM).** Phase 4 (legacy-interp > retirement) is PLANNED in `PLAN-COMPILER-VM.md`; **user decision: UNIFY** (the VM runs the post-link > bundler too, `interp.zig` fully deleted). DONE this arc (all green): **4A.1** box_any/unbox_any + > `.any` as a 16-byte aggregate (`1526d19`); **4D.0** comptime memory → an ARENA, `Addr` = real host > pointer, no buffer/cap/move (`625ba0f`); **4D.1** general host-FFI escape — `Vm.callHostExtern` > dlsym + host_ffi, any extern, args/returns pass untouched since Addr IS a host pointer (`e7a8708`, > example 0636 `toupper`); **4D.2** slice/string args → NUL-term `char*` + float-arg/return guards > (`6a7f690`, example 0637 `strlen([:0]u8)`). **699/0 BOTH gates.** > > **DIRECTION CORRECTION (2026-06-18, user): `#compiler` / `compiler_call` is DELETED, not bridged > on the VM.** `BuildOptions` is RE-EXPRESSED as **`abi(.zig) extern compiler`** functions (the > compiler-API surface the VM already dispatches via `callCompilerFn`); the `#compiler` attribute, the > `compiler_call` IR op, and the `Value`-based hook `Registry` (`compiler_hooks.zig`) all go away. > So there is **NO transitional `compiler_call`→hooks shim** on the VM (I started one — threading the > legacy interp into `tryEval` for the hook registry — and reverted it; tree clean at `b05c74f`). > `0602`/`0603` stay on legacy fallback until the BuildOptions migration lands. **Migration shape** > (end-state, shares the `BuildConfig`-on-the-VM prerequisite with the bundler 4E): (1) each > `BuildOptions` setter/getter becomes a `compiler` fn in `compiler_lib.bound_fns` + `Vm.callCompilerFn`, > reading comptime args + a `*BuildConfig` threaded into the `Vm` (the same `BuildConfig` > `main.zig` forwards); (2) `library/modules/build.sx` declares them `abi(.zig) extern compiler` > instead of `struct #compiler`; (3) delete the `compiler_call` op + `compiler_hooks.zig` > `HookFn`/`Registry` + the `#compiler` parse/lower path. See `PLAN-COMPILER-VM.md` Phase 4. > > **Corpus-driven remainder (independent of the BuildOptions migration):** ALL PURE ops are DONE: > `switch_br`, `type_name`, `error_tag_name_get`, `global_addr`, `type_is_unsigned`. **`out` DONE (2026-06-19, > newest Log entry):** removed the `out` builtin — it's a plain sx fn calling libc `write`, so the VM handles it > via host-FFI (no buffer, no special arm; no double-print because there's no `out` op to bail-then-fallback on). > `trace_resolve` PORTED (1035). 0613/1035/0522/1038 run VM-HANDLED. Remaining side-effect op: `interp_print_frames` > (1034 — writes the comptime frame chain; could likewise become a plain sx fn over the trace runtime). > · **4B VM diagnostics (1179/1180) — DONE** (strict renders the proper `comptime type construction failed:` > diagnostic; VM-gap strict bails are now ONLY the 4 `compiler_call`) · **4C** `#insert`. **BuildOptions migration — design settled + > foundation underway (2026-06-18, see the two newest Log entries):** `#compiler`/`compiler_call` is replaced > by `abi(.compiler)` (a compiler-domain ABI — runs in the comptime evaluator, never in the binary). **S1+S2 > DONE:** `abi(.compiler)` introduced, the old `abi(.zig) extern compiler` + `#library "compiler"` fully removed, > all compiler-API examples migrated. **S3 DONE:** emit_llvm skips BODIED `abi(.compiler)` (compiler-domain) > functions; a comptime-only call from a dead body emits `undef` (regression `0638`; 701/0 both gates). The > earlier "runtime-reachability gating" blocker is MOOT — a compiler-domain callback isn't LLVM-emitted, so its > `build_options()` calls never reach the `emitCall` gate. **S4 SKIPPED (optional ergonomics):** an > `abi(.compiler)` function is type-compatible with a plain `() -> R` param (the ABI marks the *function*, not > its *type*), so callbacks/registrars just declare themselves `abi(.compiler)` (S3) — no param-propagation > needed. **S5a DONE:** `build_options` + `set_post_link_callback` → `abi(.compiler)`, `BuildConfig` threaded > into the VM; `bundle_main` + the platform registrars marked `abi(.compiler)`; strict `compiler_call` bails > 6→2 (0602/0603/1604/1611 HANDLED). **S5a is a GREEN INTERMEDIATE — do NOT extend it.** **DESIGN PIVOT > (2026-06-18, user): the 37-hook BuildOptions port is DEAD — DRIVE THE BUILD PIPELINE FROM SX** (newest Log > entry + `PLAN-COMPILER-VM.md` → Phase 5). `BuildConfig` becomes plain sx data; the compiler shrinks to a few > `abi(.compiler)` primitives (`emit_object`/`link`/queries, explicit args, `-> !` not bool) + an `on_build` > slot (stdlib `default_build`, user override `#run on_build = build;`). **P5.1 (= 4E) DONE (2026-06-19, newest > Log entry):** `core.invokeByFuncId` (the post-link build-driver invocation) now runs the callback on the VM > with **NO fallback** (a side-effecting callback can't double-execute); `BuildConfig` + `import_sources` threaded > in; VM bail → hard build error (`comptime_vm.last_bail_reason` surfaced by `main.printInterpBailDiag`). Smoke > test `1661-platform-post-link-vm-list` (AOT) — a post-link callback that GROWS a `List` (0141: works on VM, > bails on legacy with `struct_get`); build succeeds (exit 0) only via the VM. `flushInterpOutput` deleted (VM > writes `out` direct via host-FFI). **702/0 both gates.** **P5.2 metadata queries DONE (2026-06-19, newest Log > entry):** `c_object_paths() -> List(string)` + `link_libraries() -> List(string)` are `abi(.compiler)` primitives > (new stdlib home `library/modules/compiler.sx`), serviced by `comptime_vm.callCompilerFn` reading `BuildConfig` > fields `main.zig` forwards (`c_object_paths`/`link_libraries`). New reusable VM helper `makeStringList` builds a > `List(string)` in comptime memory (target-aware via the result type's offsets); `invoke`/`callCompilerFn` now thread > the call's result type (`ins.ty`). Legacy handlers bail loudly (VM-only by nature — post-link). Smoke test > `1662-platform-build-pipeline-queries` (AOT, C companion → 1 object): a post-link callback checks the VM-built > list is well-formed; build exit 0 ONLY if so (negative-probe verified: wrong count → "post-link callback > returned false", exit 1). **`emit_object() -> string` ALSO landed** (a QUERY — the Zig driver emits eagerly, the > primitive returns `BuildConfig.object_path`; NO vtable). So all three QUERY primitives are done. **703/0 both > gates.** **P5.2b (`link` ACTION) DONE (2026-06-19, newest Log entry):** `link(objects, output, libraries, > frameworks, flags, target)` dispatches through a host-installed `compiler_hooks.BuildHooks` vtable (`main.zig` > `LinkHooksCtx` → `target.link`); **USER DECISION: the build callback is NOT fallible** — `link` is plain VOID, > a failure BAILS (hard build error), no `-> !`/failable-tuple needed. New VM readers `readStringList`/ > `readStringArg`. Smoke test `1663-platform-build-pipeline-link` (AOT): a post-link callback re-links the build's > objects to a temp output via sx `link` — the relinked binary RUNS; negative-probe verified (bad path → bail → > build exit 1). **P5.3 (`on_build` registrar) + P5.4 CORE DONE (2026-06-19, newest Log entries):** the whole > build is sx-driven via `default_pipeline` (force-lowered + auto-invoked; NO Zig auto-emit/auto-link); > `on_build(cb)` is the sole callback mechanism; `set_post_link_callback` deleted. **703/0 both gates.** > **NEXT — the FULL MIGRATION (no legacy left), spec'd as Phase 5 steps P5.5–P5.8 in `PLAN-COMPILER-VM.md`:** > **P5.5 DONE (2026-06-19, newest Log entry):** the 35 `BuildOptions` `#compiler` methods → VM-native > `abi(.compiler)` arms (`comptime_vm.callBuildOptionFn`, NO legacy handler); setter strings duped into the > persistent `Vm.gpa`; `#run`/const-init compiler-domain entries routed to the VM (`entryNeedsVm`, no fallback) > so gate-OFF stays green; 5 bundle.sx helpers marked `abi(.compiler)`. BuildOptions `compiler_call` bails GONE > (1609/1614/1615 strict-clean). 37 `.ir` regenerated (string-pool churn, behavior-identical). · **P5.6 prereq > DONE (`994d649`):** ported `bit_and`/`bit_or`/`bit_xor`/`bit_not`/`shl`/`shr` into the VM (the 1616 `shr` gap); > test `0639`. **704/0 BOTH gates.** · **P5.6 REMAINING (the big body, NEXT):** move `platform/bundle.sx`'s > per-target logic into / called by `default_pipeline` (call `bundle()` after `link` when `bundle_path()` is set), > reading the migrated `abi(.compiler)` getters + `fs`/`process` host-FFI; remove the `--bundle`/`post_link_module` > Zig shim (compiler keeps ONLY the linker primitive). ALL bundling + code signing for EVERY target > (macOS/iOS-device/iOS-sim/Android) in > the sx `default_pipeline` · P5.7 DELETE `#compiler`/`compiler_call`/`compiler_hooks`/`interp.zig` + the > `regToValue` bridge + VM→legacy fallback (drop gate-OFF; VM is the SOLE evaluator) · P5.8 build > `~/projects/m3te` + `~/projects/distribution` end-to-end as the acceptance test + add `.app`/`.apk` smoke tests. > **FINAL atomic step (4F):** (`out` already done — VM-native via libc `write`) handle `interp_print_frames` + > flip strict-to-default (remove the fallback) + delete `interp.zig`/`Value` + re-express `define`/`make_enum`. > See `PLAN-COMPILER-VM.md` → Phase 4 for the full plan + top risks (bundler test coverage). > Earlier landed: dedicated `Type` builtin TypeId (`6844fb9`/`94f60c5`/`554871b`); WRITE side > declare_type/register_type/pointer_to VM-native (`66005af`); real lowering-time Context (`eb68d9e`); > metatype construction declare/define/enum_init (`d0ebc55`). > > Done so far in Phase 3: > - **READ side (7 readers, dual-path):** `find_type`/`type_kind`/`type_field_count`/ > `type_nominal_name`/`type_field_name`/`type_field_type`/`type_field_value`, each backed by a > `TypeTable` query both the legacy handler and the VM call (no drift). Examples 0628–0630. > - **WRITE side (P3.3, legacy-only at lowering time):** `declare_type` + `pointer_to` + ONE > kind-branching `register_type` (subsumes `define`'s per-kind dispatch; codes match > `type_kind`: 1 struct · 2 actual `.@"enum"` · 3 tagged_union · 4 tuple). Idempotent re-fill > (two-edge import). Plus two fixes (issue 0142): all-void enum → real `.@"enum"` (was a > verifySizes panic); bare `EnumType.variant` qualified construction. Examples 0631–0635, 0187. > - **Lowering-time VM (P3.4):** hardened the VM against malformed lowering-time IR (`refTy`, > bailing `aggType`, bounds-checked branch targets — bails, never panics); wired `tryEval` > into `runComptimeTypeFunc` behind the flag with legacy fallback; materialized a zeroed > lowering-time `Context` (the global isn't built yet at lowering). All measured green. > > **THE WALL (next step):** a `Type` *value* is an 8-byte tid, but `.any` (the boxed-any) is a > 16-byte `{tag,value}` — and they share one TypeId (`.any`). So a `Type` in an aggregate > (`Member.ty`/`EnumVariant.payload`) is sized 16B while the value is 8B → every lowering-time > type-fn bails at `const_type` / the Member-array build. Can't make `kindOf(.any)` a word: > at EMIT time `.any` really is a 16B box (variadic any, 0603), so that would silently corrupt > it. The correct fix is a **dedicated `Type` builtin TypeId (8B), distinct from `.any`** — > measured at **~123 `.any` references across ~25 files** (pack.zig has 30), a ~100-touch-point > cross-cutting change → its own focused session (USER CHOSE to pause rather than rush it). > Rejected alternatives: a scoped "lowering-mode treats `.any` as a word" flag (silent-wrong on > a real Any box in a reflection type-fn); scalar-only Type-fns (safe but no real corpus type-fn > is scalar-only — they all build a Member/variant aggregate). > > **Decisions recorded:** `find_type` returns a non-optional `TypeId` using `unresolved`(0), NOT > `?Type`; reader names use the `type_*` family (avoid colliding with std `field_name`/`type_name`); > the write side is a single kind-branching `register_type`; the write side stays LEGACY-only > until the VM runs at lowering time (needs the `Type` TypeId). End-state guarantee: ONE > evaluator — `interp.zig` deleted; dual-path + fallback are transitional (see PLAN end state). > Build/verify: `zig build && zig build test` (**697**, gate OFF). Run the corpus ON the VM: > `zig build test -Dcomptime-flat` OR env `SX_COMPTIME_FLAT=1`. Coverage trace: > `SX_COMPTIME_FLAT_TRACE=1` (now also prints lowering-time `type-fn` HANDLED/fallback lines). ### (superseded) prior weld resume Phase 1 done; Phase 2 welded structs were working via reflection + memory-order validation (the `computeWeldPlan`/byte-blob "GEP engine" was explored + DROPPED even earlier). A welded `Name :: struct abi(.zig) extern compiler { … }` declared fields in the compiler type's MEMORY order; the compiler reflected the bound Zig type and VALIDATED the header. **This whole mechanism is now being stripped — see the banner.** > ⚠ Snapshot workflow: use `-Dname=examples/NNNN-foo.sx[,…] -Dupdate-goldens` to > regenerate ONLY the named example(s) — a full `-Dupdate-goldens` re-runs all ~690 > and a flaky/host-divergent example (AOT/cross-arch) can clobber good snapshots. > See CLAUDE.md → Snapshot integrity. ## Last completed step **Phase 2 — welded structs by reflection + memory-order validation (byte-identical, no GEP engine).** A welded `struct abi(.zig) extern compiler { … }` now works end-to-end as a byte-identical mirror of the bound Zig type. Design (locked, supersedes the byte-layout-override plan): - The sx header declares fields in the compiler type's MEMORY order. The compiler REFLECTS the bound Zig type — field names from `@typeInfo`, offsets from `@offsetOf`, size from `@sizeOf` — and validates the header matches. Nothing is maintained by hand; a `types.zig` change re-reflects on the next compiler build. - On pass it's an ORDINARY struct whose natural layout already equals the Zig layout → `@ptrCast` to the compiler type + deref is byte-identical. No byte-blob, no index/remap tables, no reorder, no special LLVM path. - Loud, precise diagnostics on any drift: *field not found* (+ memory order), *wrong field order at position N* (+ expected memory order), *type layout mismatch* (field size), *layout mismatch* (total size / count). What changed from the dropped plan: - `compiler_lib.zig`: `weldStruct` now REFLECTS field names (`@typeInfo`) and bakes `bound_types` fields in ascending-OFFSET (memory) order — no hand-listed names. Deleted `computeWeldPlan`/`WeldPlan`/`WeldElement`. `validateStructLayout` checks the sx header against the memory-ordered registry. - `nominal.zig` `validateWeldedStruct`: renders the precise diagnostics (+ `weldedFieldOrderStr`). - Examples: `0627` (StructInfo in memory order, byte-identical, usable); `1186` (source-order StructInfo → wrong-field-order diagnostic). `1183` message refreshed. - `zig build` + `zig build test` green (692 corpus, unit tests pass). ### Earlier — Phase 2.1 (weld-plan layout math, now removed) **The weld-plan offset math + `StructInfo` registered.** Was the core of the byte-layout-override engine; superseded by the reflection+validation design above. Decision (locked 2026-06-17): **full byte-layout weld** — a welded sx struct is laid out byte-identically to the bound Zig type (Zig's `@offsetOf`, reordering + padding included), so it passes to a Zig handler as raw memory with zero marshalling. (The alternative — handlers reading interp `Value` aggregates logically, no layout override — was rejected; welded types must also be usable as runtime data, and the design wants the literal byte weld.) - Measured: Zig reorders `StructInfo` to `fields`@0, `name`@16, `nominal_id`@20, `is_protocol`@24, size 32 — vs sx-natural `name`@0, `fields`@8, … So the override is genuinely required (`Field`'s two-u32 natural layout was the easy case). - `compiler_lib.zig`: registered `StructInfo` (`weldStruct`, the second `bound_types` entry). Added `WeldElement` / `WeldPlan` + `computeWeldPlan(alloc, fields, total)` — pure: orders fields by ascending byte offset, inserts padding elements for gaps + the alignment tail, and builds the sx-field → LLVM-element remap. This is what the LLVM type builder + struct-GEP sites will consume. - Unit-tested (`compiler_lib.test.zig`): `Field` → identity plan (2 elems, no pad); `StructInfo` → 5 elems `[fields@0, name@16, nominal_id@20, is_protocol@24, pad@25..32]`, remap `[1,0,3,2]`. - `zig build` + `zig build test` green. ### Earlier — Phase 1 polish (comptime-only enforcement) **A RUNTIME call to a `fn abi(.zig) extern compiler` is a clean build-gating error instead of an undefined-symbol link failure.** - `emitCall` (`src/backend/llvm/ops.zig`): when the callee is `compiler_welded` AND the ENCLOSING function is not `is_comptime` (i.e. genuine runtime code, not a `#run`/`::` initializer wrapper whose LLVM body is dead), print a clear "comptime-only … cannot be called at runtime" error and set `comptime_failed` (the driver halts before object/JIT emission). The enclosing `is_comptime` guard is what keeps the legitimate `#run` use (example 0626) green. - Corpus: `examples/1185-diagnostics-weld-fn-runtime-call.sx` (runtime `intern(…)` → clean error, exit 1, no link failure). - `zig build` + `zig build test` green (458 unit + 690 corpus). ### Earlier — fifth sub-step (host-call bridge) **A `fn abi(.zig) extern compiler` dispatches, under the comptime interpreter, to its registered Zig handler instead of dlsym.** - `compiler_lib.zig`: function registry — `BoundFn { sx_name, handler }`, `bound_fns` = `intern(string)->StringId` + `text_of(StringId)->string` (the string-pool round-trip), `findFn`, and `FnHandler` (`*Interpreter, []Value -> Value`). `intern` mutates via `interp.mint orelse @constCast(&module.types)` (the same mutable-table access the metatype mint path uses); `text_of` reads the const pool. Imports `interp.zig` (the compiler_hooks↔interp cycle pattern). - IR `Function` gained `compiler_welded: bool`. `declareFunction` (`src/ir/lower/decl.zig`) sets it via `weldedCompilerFn`, which also VALIDATES: the bound lib must be `compiler` and the name must be on the function-export list — else a build-gating `.err` (no silent fall-through to dlsym). - `interp.call()`: before the dlsym/extern path, a `compiler_welded` function routes to `compiler_lib.findFn(name).handler(self, args)` (clean bail off the export list). - Corpus: `examples/0626-comptime-weld-fn-intern-text-of.sx` (`#run text_of(intern("hello, compiler"))` folds to a string constant → prints it); `examples/1184-diagnostics-weld-fn-unexported.sx` (unexported welded-fn name → build error). `findFn` lookup unit-tested. - **Runtime-call rejection is NOT yet clean** — welded fns are comptime-only; a RUNTIME call would emit a reference to a non-existent extern symbol → a loud LINK error (not silent, but not a tidy diagnostic). The examples call welded fns only inside `#run`. A dedicated "comptime-only symbol" emit diagnostic is the immediate follow-up. - `zig build` + `zig build test` green (458 unit tests + 689 corpus). ### Earlier — fourth sub-step (welded-struct layout validation) **A `struct abi(.zig) extern compiler { … }` is validated against the binding registry as a *header checked against the implementation*.** - `compiler_lib.zig`: `validateStructLayout(bt, sx_fields, total)` — pure, returns the first `LayoutMismatch` (field count / name / size / total) or null. Plus `lib_name = "compiler"` and `SxField`. Unit-tested (faithful `Field` passes; each drift flagged as the right variant). - `registerStructDecl` (`src/ir/lower/nominal.zig`): for `sd.abi == .zig`, `validateWeldedStruct` checks the bound lib is `compiler`, the name is on the export list (`findType`), and the sx layout (field names + `typeSizeBytes` + total) matches the welded type — emitting a build-gating `.err` (good span into the struct body) on any failure. No silent reinterpretation. - `#library "compiler"` is the comptime-only internal surface, NOT a dylib — `src/main.zig`'s dlopen walker skips it (was emitting a spurious `libcompiler.so` load warning). - Corpus: `examples/0625-comptime-weld-struct-field.sx` (faithful `Field` welds, validates, usable as data → `name=7 ty=3`); `examples/1183-diagnostics-weld- struct-field-count.sx` (one-field `Field` → build-gating field-count diagnostic). - **Offset-override / GEP emission for non-natural Zig layouts is NOT here** — it isn't exercised by `Field` (two u32s = natural layout coincides with the weld). It arrives with `StructInfo` in Phase 2 (slices/reordering), where the bound offsets actually differ from the sx-natural ones. The validation already checks per-field size + total, so a layout drift is caught even before the override engine exists. - `zig build` + `zig build test` green (456 unit tests + 687 corpus). ### Earlier — third sub-step (binding registry) **The binding registry (welded-type lookup, layout baked from the real Zig type).** - New `src/ir/compiler_lib.zig` — the `compiler` library's binding registry, the curated safety boundary. `BoundType { sx_name, size, alignment, fields: []FieldLayout{name, offset, size} }`; `weldStruct` bakes the layout from a real Zig struct via `@sizeOf`/`@alignOf`/`@offsetOf` at compiler-build time (a sx-field-count mismatch is a `@compileError`, never a silent truncation). `bound_types` exports `Field` (welded to `types.TypeInfo.StructInfo.Field` — two `u32`s); `findType(sx_name) ?*const BoundType` is the lookup the welded-decl resolution path will consult (returns null off the export list — clean boundary, no silent default). - Registered in the barrel (`src/ir/ir.zig`): `compiler_lib` + `compiler_lib_tests`. - Tests (`src/ir/compiler_lib.test.zig`): `findType("Field")` equals the real `StructInfo.Field` `@sizeOf`/`@alignOf`/`@offsetOf` (8 bytes, two u32s at 0/4); an unexported name returns null. Break-verified (a wrong size → suite red, named `ir.compiler_lib.test...`). - `zig build` + `zig build test` green (454 unit tests). ### Earlier — second sub-step (struct-decl parse) **`abi(.zig) extern ` PARSES on a STRUCT decl (parse-only, no semantics).** - `ast.StructDecl` gained `abi: ABI` + `extern_lib: ?[]const u8` binding fields. - `parseStructDecl` (`src/parser.zig`): after `struct` (and the `#compiler` check), parse an optional `abi(...)` then optional `extern ` — same slot order as fn decls — and thread them onto the node. Ordinary structs are unperturbed (`parseOptionalAbi`/`parseOptionalExternExport` no-op when absent). - Parser unit tests (`src/parser.test.zig`): `Field :: struct abi(.zig) extern compiler { name: StringId; ty: Type; }` parses with `abi == .zig`, `extern_lib == "compiler"`, field list intact; a plain struct leaves `abi == .default` / `extern_lib == null`. Break-verified (a wrong-sentinel assert turns the suite red, confirming the test runs). - `zig build` + `zig build test` green. ### Earlier — first sub-step (fn decls) + the syntax pivot **`abi(.zig) extern ` PARSES on a fn decl (parse-only).** Plus the syntax pivot it required. Syntax decision (locked 2026-06-17, supersedes the doc's original `extern(.zig) ` single-qualifier form): the ABI/layout selector and the linkage keyword are two orthogonal annotations. - `abi(.x)` — ABI / calling-convention annotation in the slot **before** `extern`/`export`. **Unified replacement for `callconv(...)`, which is removed.** `ABI = { default, c, zig, pure }`: `.c` (C ABI), `.zig` (Zig-layout weld → the `compiler` library), `.pure` (naked asm), `.default` (unannotated). Can appear standalone (no extern) on any fn / fn-type / lambda. - `extern ` — linkage keyword + binding source (named library). So a welded binding is `text_of :: (id: StringId) -> string abi(.zig) extern compiler;`. What landed: - **AST** (`src/ast.zig`): `CallingConvention` → `ABI { default, c, zig, pure }`; the `call_conv` field → `abi: ABI` on `FnDecl` / `Lambda` / `FunctionTypeExpr`. - **Lexer/token** (`src/token.zig`, `src/lexer.zig`): `kw_callconv` → `kw_abi`, keyword string `"callconv"` → `"abi"`. - **Parser** (`src/parser.zig`): `parseOptionalCallConv` → `parseOptionalAbi` (parses `abi(.c|.zig|.pure)`); wired in the fn-decl postfix slot (before `extern`/`export`), the function-type-expr slot, and the lambda slot; `isFunctionDef`/`hasFnBodyAfterArrow` recognise `kw_abi`. - **AST→IR map** (`src/ir/type_resolver.zig`, `src/ir/lower/decl.zig`, `sema.zig`, `closure.zig`): the AST `.abi == .c` reads kept their C-ABI meaning; the function-type resolver maps `.zig`/`.pure` → IR `.default` (no fn-pointer-type CC for those decl-level ABIs; neither occurs in a function-TYPE position yet). - **CC-mismatch diagnostic** (`src/ir/lower/expr.zig`, `src/sema.zig`): the user-facing text `callconv(.c)` → `abi(.c)`. - **sx migration**: 52 `.sx` files `callconv(` → `abi(` (all were function-type callback annotations — none in the fn-decl postfix slot, so no reordering). - **Docs**: `readme.md`, `specs.md`, the design doc, snapshots (0114 / 1104 / 1200) regenerated for the rename. - **Tests**: parser unit tests in `src/parser.test.zig` — `abi(.zig) extern ` on a fn decl (asserts `abi == .zig`, `extern_export == .extern_`, `extern_lib == "compiler"`); bare `extern` leaves `abi == .default`; standalone `abi(.c)` / `abi(.pure)`. lexer/sema tests updated. `zig build` + `zig build test` green (450/450 unit + 685 corpus). ## Current state > **Pivoted — see the banner + `PLAN-COMPILER-VM.md`.** The items below are the weld > machinery as it stands on `reify` HEAD (`40d075c`); they are the **strip list** for > Phase 0, not the forward direction. The `#library`/`abi`/`extern` *syntax* stays; the > weld *semantics* (layout reflection/validation, marshaling dispatch) go. - `compiler :: #library "compiler";` parses + is recognised as the comptime-only internal surface (never dlopen'd). - `abi(.zig) extern compiler` STRUCTS: layout-validated against the registry (faithful → ok; drift → build-gating diagnostic). `Field` welds + usable. - `abi(.zig) extern compiler` FUNCTIONS: dispatched under the comptime interp to their registered Zig handler (`intern`/`text_of` round-trip works); unexported names rejected at declaration. Comptime-only. - A RUNTIME call to a welded fn is a clean build-gating error (comptime-only enforcement at `emitCall`); the legitimate `#run`/`::` use stays green. - The whole Phase 1 foundation (parse → registry → struct-layout validation → function host-call bridge → comptime-only enforcement) is in place for the two-u32 `Field` case + the two string readers. - **Deferred**: offset-override / LLVM byte-offset GEP for non-natural layouts (needed by `StructInfo`'s slice field, Phase 2). ## Next step — execute `PLAN-COMPILER-VM.md` > The weld is being stripped. The next step is **Phase 0 of > [`PLAN-COMPILER-VM.md`](PLAN-COMPILER-VM.md)** — remove the weld / serialize / > marshal machinery (`compiler_lib.zig` reflection+validation, `nominal.zig` > `validateWeldedStruct`, the `compiler_welded` dispatch, the weld examples/diagnostics > 0625/0627/1183/1184/1185/1186), keeping the `#library`/`abi`/`extern` *syntax*. Then > Phase 1 (byte-addressable value model). The weld-era "next step" below is **obsolete** — > kept only as a record of what the weld surface was about to do. ### (obsolete) weld-era next step Welded structs were byte-identical mirrors, so the API surface was set to grow: - **Bind `register_struct` / `find_type`** over the host-call bridge (`compiler_lib.zig` `bound_fns`, like `intern`/`text_of`). `register_struct` takes a welded `StructInfo` and mints a real `TypeId` (guarded: dup field names, kind well-formedness — the checks `define` does today). Because the welded `StructInfo` is byte-identical, the handler can read it as the real Zig `*StructInfo` (cast + deref) rather than marshalling a `Value` field-by-field — the payoff of the byte-weld. `find_type(StringId) -> ?Type` reads the table. Prove: build a struct programmatically + round-trip a source one. - **Re-express `type_info`/`define` (struct) as sx** over `register_struct`/ `find_type`; migrate `examples/0622`; delete the bespoke struct interp arms (`defineStruct` / the `reflectTypeInfo` struct path). Then Phase 3+: widen the welded types to `EnumInfo`/`TaggedUnionInfo`/`TupleInfo` (optional fields → sentinels) — each just needs an sx header in the compiler type's memory order + the matching `register_*` fn. Finally migrate `BuildOptions` to `abi(.zig) extern compiler` (re-home the `#compiler` registry) and delete `#compiler`. Note: a welded struct with an `?T` / `union(enum)` field (e.g. `EnumInfo`'s `backing_type: ?TypeId`, `explicit_values: ?[]const i64`) is the next layout wrinkle — the sx header must mirror Zig's optional/union representation. Handle when reached (sentinels or accessor fns; see the design doc Risks). ## Known issues - None for this stream. (Metatype's deferred enhancement is issue 0141 — comptime `List` growth; orthogonal, see `current/CHECKPOINT-METATYPE.md`.) ## Log - **P5.8 — Android + iOS-sim validated end-to-end on emulator/simulator; first Android bundler corpus coverage (2026-06-19).** The sx-driven build pipeline now validated on BOTH mobile targets via the local emulator/simulator. **Android (Pixel_10_Pro emulator):** `sx build --target android --apk … -o lib….so` produces a valid SIGNED `.apk` (AndroidManifest.xml + resources.arsc + `lib/arm64-v8a/lib….so` + classes.dex + apksigner META-INF) via `default_pipeline → bundle_main` (javac/d8/aapt2/zip/zipalign/apksigner all on the comptime VM); `adb install` + `am start` → the app **runs (no crash)** for the `super.onCreate(b)` example (1424). **iOS-sim (iPhone 17 Pro simulator):** `sx build --target ios-sim --bundle …` → signed `.app`; `simctl install` + `launch` → the sx AppDelegate fired (`[sx] application:didFinishLaunchingWithOptions: called`), UIApplicationMain run loop alive. **THREE comptime-VM host-FFI gaps fixed to make the Android bundler run on the VM** (`d8fb425`): (1) an extern returning an OPTIONAL-of-word (`getenv() -> ?cstring`) now wraps the bare C payload word into the `{payload, has}` aggregate (present iff non-null), mirroring emit_llvm's `char*`→`?cstring`; (2) `struct_init` of the builtin two-word aggregates `string`/`any` (e.g. `from_cstring`'s `string.{ ptr=, len= }`); (3) the non-word-return bail now names the symbol + type. **`.apk` corpus smoke test** (`2ba36f6`, `examples/1666-platform-android-apk-smoke`): a new `.build` `apk` directive (cross-compile `--target android --apk`, build+inspect, NO execution) GATED on Android-SDK + real-JDK presence — SKIPS cleanly on hosts without them (so normal `zig build test` stays green), RUNS+PASSES with `JAVA_HOME` set (apk built + `AndroidManifest.xml`/`classes.dex`/`lib/arm64-v8a/` asserted, then cleaned up). Requires a real JDK (the macOS `/usr/bin/javac` stub fails); Android Studio's JBR works. **709 ran + 1666 skipped (no JDK) / 0 failed + 476/476 unit.** macOS `.app` (1665) + Android `.apk` (1666) now both cover the bundler. > **REMAINING (true tail):** iOS-DEVICE codesigning/provisioning (needs a real device + signing identity — a > simulator can't exercise the device-identity flow). Everything else in the stream is done. - **Empty-member types are VALID for all kinds (user design decision, 2026-06-19).** A comptime-constructed type with NO members now mints for every kind (empty `struct`, empty `tuple`/unit, empty `enum`, empty `tagged_union`); ONLY a bare `declare("X")` never completed by a `define` stays rejected. `registerTypeVm` dropped the blanket "a type with no members is never valid" bail (the per-kind loops are vacuous for an empty list). To distinguish an EXPLICITLY-defined empty union from a never-defined `declare` PLACEHOLDER (both are 0-field `tagged_union`s), added `defined: bool = true` to `TaggedUnionInfo` — default true for every real construction (normal unions, error sets, `register_type` completion); set false ONLY at the two declare- placeholder sites (`comptime_vm.declareNominal`, `lower/comptime.preregisterForwardTypes`). `checkComptimeTypeResult` now rejects on `!defined` (not `fields.len == 0`). Codegen parity fix: `typeSizeBytes(tagged_union)` floors the payload area at 8B when no field carries a payload, mirroring the LLVM lowering (`backend/llvm/types.zig`) — fixes a `verifySizes` panic on an empty/all-void tagged_union. Tests: `examples/1179` repurposed from "empty enum rejected" (now valid) to the never-defined `declare` case (preserves its issue-0140 "diagnostic-not-panic" regression role); `examples/1180` (duplicate variant) unchanged; new `examples/0641-comptime-empty-types-valid` constructs all four empty kinds. **709/0 corpus + 476/476 unit** (`5383496`). - **P5.7 Step D — re-expressed the metatype `declare`/`define` as sx over the compiler-API (2026-06-19).** `declare(name)` → sx `{ return declare_type(name); }`; `define(handle, info)` → sx that matches the `TypeInfo` union and calls `register_type(handle, kind, members)` (kinds 1/2/3/4). DELETED the bespoke `callBuiltinVm` `.declare`/`.define` arms + the `defineFromInfo`/`decodeTypeSlice` helpers + the `declare`/`define` `BuiltinId` enum members + their `tryLowerReflectionCall` interceptions. The metatype DSL now rides the ONE compiler-API mechanism (`register_type`/`declare_type`, serviced by `callCompilerFn`), with the DSL in `meta.sx`. Supporting VM change: tagged-union VALUE support so the sx `define` can `match` a `TypeInfo` (`kindOf` treats `tagged_union` as a by-address aggregate; `enum_tag` reads the tag word; new `enum_payload` arm; all bail loudly on a `backing_type` union). **KEPT as builtins (documented):** `type_info($T)` (reflects a type INTO a byte-compatible `TypeInfo` value — `buildTypeInfo`; re-expressing risks the 0619/0622/0623 round-trips for no proportional gain) and `field_type($T, idx)` (a LOWER-time fold in `generic.zig`, so it composes inside type-arg slots — never was a `callBuiltinVm` arm). Diagnostics 1179/1180 regenerated (now name `register_type`). **708/0 corpus + 476/476 unit** (`8850fcc`/`7b1d8ce`/`ccba704`). Net: 2 of 4 metatype builtins are now sx; the bespoke Zig metatype surface shrank by two `callBuiltinVm` arms, two helpers, and two `BuiltinId`s. - **P5.7 Step C — DELETED `interp.zig` (the legacy tagged-`Value` interpreter); the VM is the SOLE comptime evaluator (2026-06-19).** Five green commits. **C1** (`#insert` → VM): `evalComptimeString` was the last caller of `Interpreter.call`; routed through `comptime_vm.tryEval` (the VM bails-not-panics on malformed lowering-time IR like 0737's `ret Ref.none`; `regToValue` dupes the result string into the lowering allocator). **C2a** (ops.zig inline comptime-call fold → VM): the `emitCall` zero-arg comptime-callee fold now uses `tryEval`. **C2b** (emit_llvm): dropped the `*const Interpreter` materialization param from `valueToLLVMConst`/ `serializeAggregateValue` (it was used ONLY for the `.heap_ptr` data arm, which the VM's `regToValue` never produces) + the dead `interp_inst`. **C3** (the atomic delete, done by a delegated worker + independently verified): moved the `Value` result-DTO + `decodeVariantElements` into a new `src/ir/comptime_value.zig` (the VM↔host materialization boundary type); repointed `comptime_vm`/`emit_llvm`/`ir.zig`-barrel `Value` to it and `BuildConfig` to `compiler_hooks`; deleted the dead `valueToReg` bridge; slimmed `compiler_lib.zig` to just the name registry (`BoundFn{sx_name}` + `bound_fns` + `findFn`, all names preserved — `weldedCompilerFn` only validates names; deleted `FnHandler` + all `handle*` + the `Interpreter`/`Value`/`InterpError` imports); simplified `main.printInterpBailDiag` to use only `comptime_vm.last_bail_reason`; dropped the unused `interp_mod` import in `lower.zig`; **`rm src/ir/interp.zig` (2383 lines) + `src/ir/interp.test.zig` (844 lines)** + their barrel entries. **DEVIATION from the plan's literal "delete Value":** `Value` is RELOCATED (not eliminated) as the slim result/materialization DTO — the byte-addressable VM executes natively, and `Value` survives ONLY at the VM→`valueToLLVMConst` boundary (the marshaling the pivot killed was at EXECUTION time, which is gone). Eliminating it entirely (materializing LLVM consts straight from VM `Machine` bytes) is a larger, riskier rewrite deferred as optional follow-up; the plan's PRIMARY goal — ONE evaluator, no legacy interpreter, no fallback — is fully met. **706/0 corpus + 476/476 unit** (−24 from the deleted interp unit tests + 1 `valueToReg` round-trip test). Also dropped dead `Value.asString`/`reflectTypeId` (no callers). NEXT: Step D — re-express `define`/`make_enum` as sx over the compiler-API (they were legacy interp arms); Step E — land the 0141 repro + finalize. - **P5.7 Step B — deleted the `#compiler`/`compiler_call`/hook-Registry mechanism end-to-end (2026-06-19).** All superseded by `abi(.compiler)` VM-native dispatch (P5.5) — no sx code emits any of it. Two green commits: **B1** (`e2971f2`) removed the `compiler_call` IR op: the op variant + `CompilerCall` struct (`inst.zig`), the `Builder.compilerCall` emitter (`module.zig`), the two dead producer blocks in `lower/call.zig` (the `compiler_expr`-bodied free-fn + method dispatch), every consumer arm (`emit_llvm`, `ops.emitCompilerCall`, `print`, the `interp.zig` hook-dispatch arm), and the `interp.hooks` field + init/deinit. Stripped `compiler_hooks.zig` to its still-live `BuildConfig` / `BuildHooks` (link/emit vtable, P5.2b) / `AssetDir` — deleted `HookError`/`HookFn`/`Registry`/`registerDefaults` + all 37 `hookXxx` fns + the now-unused `interp`/`Value` imports. The two VM unit tests that used `compiler_call` as a sample unported op now use `vec_splat`. **B2** removed the `#compiler` attribute + `compiler_expr` AST node: the `hash_compiler` token (`token.zig`/`lexer.zig`/`lsp/server.zig`), the `is_compiler_struct` / `struct_default_compiler` parser machinery + the two `compiler_expr` body-synthesis branches (`parser.zig`), the `compiler_expr: void` AST variant (`ast.zig`), and every `.builtin_expr, .compiler_expr =>` arm / `== .compiler_expr` check across sema/resolver/semantic_diagnostics/generic/decl/call/calls (dropped `.compiler_expr`, kept `.builtin_expr`). `abi(.compiler)` (the NEW mechanism) is untouched. Deleted the obsolete `calls.test.zig` `#compiler`-dispatch unit test. **500/500 unit (−1 obsolete test) + 706/0 corpus, no snapshot churn.** NEXT: Step C — delete `interp.zig` + the `regToValue`/`valueToReg` bridge; move `#insert` (`evalComptimeString`) to the VM. - **P5.7 Step A — the flip: VM is the SOLE comptime evaluator at the emit-time + type-fn sites; NO fallback (2026-06-19).** Removed the `if (self.comptime_flat or need_vm)` gate + the `vm_result orelse fallback` legacy-interp blocks from `emit_llvm.zig` (`runComptimeSideEffects` AND the const-init path in `emitGlobals`) and from `comptime.zig` `runComptimeTypeFunc` (type-fn). The VM now ALWAYS runs; a bail is ALWAYS a build-gating diagnostic (`comptime_vm.last_bail_reason`), never a fallback. Deleted `emit_llvm.entryNeedsVm` (moot — every entry runs on the VM now). `runComptimeSideEffects` no longer creates an `Interpreter` at all (VM writes `#run` `print` output direct to fd 1 via host-FFI); `emitGlobals` keeps a fresh `interp_inst` ONLY as the helper context `valueToLLVMConst` uses to materialize the VM's result Value (it evaluates nothing) — that's the `regToValue` bridge, removed in Step C with `interp.zig`. **`#insert` (`evalComptimeString`) still uses the legacy interp** — intentionally deferred to Step C (interp.zig still exists); it only needs the evaluator to bail without crashing (0737's real error is a lowering-time visibility diagnostic). The `comptime_flat*` LLVMEmitter fields are now set-but-unused (harmless; cosmetic cleanup later). **Snapshot reconcile:** only `1654` churned — the asm-global `#run` now reports the VM's clean `comptime init of 'COMPUTED' failed: comptime extern call: symbol not found via dlsym …` instead of the legacy `CannotEvalComptime (op=call: …)` wrapper (exit still 1); regenerated scoped via `-Dname`. `1179`/`1180` unchanged (the VM-strict `comptime type construction failed:` wording already matched). **501/501 unit + 706/0 corpus** (one gate now — `-Dcomptime-flat` is moot but still accepted). NEXT: Step B — delete the `#compiler` attribute (parse+lower) + the `compiler_call` IR op + `compiler_hooks.zig`. - **P5.7 Step 0 — strict sweep CLEAN; zero VM gaps to port (2026-06-19).** Gating prerequisite for deleting the legacy fallback. Confirmed both gates 706/0 + 501/501 unit (gate-OFF and `-Dcomptime-flat`). Then ran every corpus example (706) under `SX_COMPTIME_FLAT_STRICT=1` (VM, NO fallback) via `.sx-tmp/strict_sweep.sh` — `sx run` (JIT), `sx build` (aot/bundle), or `sx ir --target` (cross-arch). Only **3** examples emit a VM-bail signature, ALL expected-failures (strict exit == expected exit), NONE a real gap: **1179** (`enum has no variants`) + **1180** (`duplicate variant name`) render the SAME `comptime type construction failed:` diagnostic the VM-strict path in `comptime.zig` already emits (no snapshot churn at the flip); **1654** (asm-global called at `#run`) — the VM bails cleanly via `callHostExtern` dlsym ("symbol not found … target- specific binding called at compile time?"), only its `.stderr` WORDING changes (legacy `CannotEvalComptime (op=call…)` → VM strict form) and must be reconciled at the flip. **Key conclusion (the prompt's flagged risk): no SUCCESSFUL corpus example relies on the legacy VM→legacy fallback** — every passing example runs natively on the VM; the only fallback users are the 3 expected-failures above. Removing the fallback is therefore safe. No ops to port before flipping. NEXT: make `-Dcomptime-flat` permanent + delete the fallback (emit_llvm both sites + comptime.zig) + remove `entryNeedsVm`, reconcile 1654. - **P5.8 (partial) — real-project validation: m3te + distribution build with the new pipeline (2026-06-19).** Acceptance test for the sx-driven build pipeline. **m3te** (`~/projects/m3te`, an SDL3 match-3 game): migrated its `build.sx` off the deleted API — `configure_build :: ()` → `() abi(.compiler)`, `opts.set_post_link_callback( bundle_main)` → `on_build(bundle_main)` (the ONLY source change needed). Then `sx build main.sx` produces a valid SIGNED macOS `.app` (correct `Contents/{MacOS,Resources}` layout, `Resources/assets/` bundled, links Homebrew SDL3, passes `codesign`); `sx build --target ios-sim main.sx` produces the flat iOS `.app` (DTPlatformName= iPhoneSimulator; system-framework "not embedded" warnings are expected/benign). So BOTH the macOS and iOS-sim bundle paths validate end-to-end on real project code. **distribution** (`~/projects/distribution`, a server/CLI+sqlite project, NO bundling): `make build` clean (smoke + dist binaries via `default_pipeline`'s emit+link with C objects + vendored sqlite); `make test` 24/25 (the 1 fail, `publish_happy.sx`, is a PRE-EXISTING stale `#foreign` parse error — that keyword is gone from the parser; unrelated to the pipeline/this session). **m3te's `build.sx` edit is in its working tree, uncommitted (the user's repo) — reported, not committed.** > **macOS `.app` corpus smoke test — DONE (2026-06-19, `445ae97`):** added a `.build` `bundle` directive to the > corpus runner (after an `aot` build, assert each `expect` entry under the produced `.app`, then `rm -rf` it; > macOS-host only, skipped elsewhere). `examples/1665-platform-macos-bundle-smoke.sx` exercises > `default_pipeline`'s auto-bundle end-to-end; passes on BOTH gates (bundler runs on legacy interp AND VM), > negative-probe verified. **706/0 both gates.** This closes the stream's named top-risk gap (no bundler coverage). > **REMAINING P5.8:** iOS-device + Android paths still unvalidated on this host (no device identity round-trip / > Android SDK; an `.apk` corpus smoke test needs an Android SDK on the runner). m3te is the real-project macOS + > iOS-sim acceptance test. - **P5.6 (macOS path) — `default_pipeline` drives bundling; fix issue 0125 (2026-06-19).** `build.sx` now `#import`s `platform/bundle.sx` and `default_pipeline` delegates to `bundle_main` when `bundle_path()` is set (emit+link via the shared `emit_and_link` core, then wrap the `.app`/`.apk`); else just emit+link. **Removed the Zig `--bundle`/`post_link_module` dispatch shim** (main.zig) — CLI bundle flags only feed `BuildConfig`, `default_pipeline` branches on `bundle_path()`. **USER DECISION:** import the bundler directly (it's `abi(.compiler)`, emit-skipped, never in the binary; the build↔bundle import cycle resolves like std↔build) — NOT a registration slot. **Validated end-to-end on macOS** (the stream's top-risk gap — bundler had ZERO coverage): `sx build --bundle App.app --bundle-id … plain.sx` AND auto-bundle from `set_bundle_path` both produce a valid SIGNED `.app` (correct `Contents/MacOS/` layout, Info.plist, passes `codesign`, binary runs). Fixed a pre-existing host-build bug: `target_triple` was empty for host builds → `is_macos()` false → wrong flat layout; main.zig now exposes the host triple (`LLVMGetDefaultTargetTriple`) when `--target` is absent. `bundle_main` dropped its redundant `build_options()` re-fetch (uses its `opts` param). **Fix issue 0125** (surfaced because the bundler's `format("…{}…")` instantiates `any_to_string`, which over-materialized array types): the type-match dispatcher (`lowerRuntimeDispatchCall`) unboxed each interned array tag to the concrete array type (whole-array load) → LLVM scalarized to one DAG node/element (~12s / segfault at `[65536]u8`). Fix (route 1): `case array:` arm calls `slice_to_string`, dispatcher builds a `{ptr,len}` slice VIEW of the payload pointer (`unbox_any → [*]elem` = int-to-ptr, NO load) for an ARRAY tag bound to a SLICE param. Output byte-identical (`[a, b, c]`); 0055 drops 12s→0.2s. Pinned `examples/0056-basic-large-array-format-no-blowup.sx`; issue 0125 marked RESOLVED. 37 `.ir` regenerated (bundler types + array-format lowering); `.ir`-only. **705/0 both gates** (`48eb7bf`). > **REMAINING P5.6 / next:** iOS-device / iOS-sim / Android paths in `bundle_main` exist + now run on the VM (the > `shr` port let them through) but are NOT validated on this host (no iOS/Android SDK + no corpus bundle harness) > — that's P5.8 (build `m3te`/`distribution` end-to-end + add `.app`/`.apk` corpus smoke tests). Then P5.7 (delete > `#compiler`/`compiler_call`/`compiler_hooks`/`interp.zig` + the legacy fallback; VM becomes the sole evaluator). - **P5.6 prerequisite — bitwise/shift ops ported into the VM (2026-06-19).** `comptime_vm` exec now handles `bit_and`/`bit_or`/`bit_xor`/`bit_not`/`shl`/`shr` (new `bitwise` helper next to `arith`), mirroring the legacy interp's i64 model EXACTLY: shift amount clamps to `@min(rhs, 63)`, `shr` is an ARITHMETIC right shift (sign-extending). These were unported and bailed — the `shr` gap surfaced via the iOS-device bundler once P5.5 let it run further (1616). With the port, 1616's strict VM run reaches the real bundler logic (no more `shr` bail; it now stops only at the genuinely-unavailable iOS runtime on macOS — `_UIApplicationMain` / no linked binary under `sx run`, expected). New focused corpus test `examples/0639-comptime-bitwise-shift.sx` (`::` consts fold AND/OR/XOR/NOT/shl/shr/arith-shr; identical on both evaluators). **704/0 BOTH gates.** - **P5.5 — the 35 `BuildOptions` accessors migrated off `struct #compiler` onto VM-native `abi(.compiler)` (2026-06-19).** `BuildOptions :: struct #compiler { ...35 methods... }` → `BuildOptions :: struct { }` (an opaque null-sentinel handle) + 35 free `ufcs (self: BuildOptions, …) abi(.compiler)` decls in `library/modules/build.sx`, each serviced by a new `comptime_vm.callBuildOptionFn` arm (dispatched from `callCompilerFn`). **NO legacy `compiler_lib` handler** (per the full-migration direction): the 35 names are registered in `compiler_lib.bound_fns` only so `weldedCompilerFn` accepts them, with a single bailing stub `handleBuildOptionsAccessor` (never reached). **String lifetime:** setters dupe the arg string into the PERSISTENT `Vm.gpa` (the Compilation allocator threaded into both `tryEval` and `runBuildCallback` — NOT the per-eval VM arena, whose bytes die at `Vm.deinit`), so a `#run`-set path survives to post-link. Setters write/append the duped string to the threaded `BuildConfig` (`output_path`/`bundle_path`/…, the `link_flags`/ `frameworks`/`asset_dirs` ArrayLists); string getters return the field (or `""`); bool getters compute from the triple (`predIsMacOS`/`predIsIOS`/…, mirroring the legacy hooks); count/indexed getters read the `BuildConfig` slices. **Dispatch routing (Option B, chosen at start):** a `#run` / const-init entry that directly calls a compiler-domain / compiler-welded fn (`emit_llvm.entryNeedsVm`) is routed through the VM with NO legacy fallback — regardless of the `-Dcomptime-flat` gate — so gate-OFF stays green without a legacy BuildOptions handler (P5.7 retires the legacy interp entirely). The 5 `platform/bundle.sx` helpers that call getters (`build_info_plist`/`embed_framework`/`android_bundle_main`/`build_android_manifest`/`compile_jni_main_sources`) are marked `abi(.compiler)` too (they're comptime-only bundler code; without it their now-welded getter calls trip the runtime-call gate). **Snapshots:** 37 `.ir` churned (std transitively imports build.sx → string-pool/ type-table indices shift) — regen scoped via `-Dname`; verified ONLY `.ir` changed (zero behavior-stream diffs). **703/0 BOTH gates.** Strict sweep: the BuildOptions `compiler_call` bails are GONE (1609/1614/1615 strict-clean); 1616 now bails on `shr` (a pre-existing, separate VM gap — bitwise/shift ops `shl`/`shr`/`bit_and`/`bit_or`/ `bit_xor`/`bit_not` are unported in `comptime_vm`, surfaced now that the iOS-device bundler runs further; 1616 is unpinned + can't JIT-run on macOS anyway). **Also (per user): swept the outdated "flat memory" terminology** — the comptime VM is byte-addressable, ARENA-backed memory where `Addr` is a REAL host pointer, NOT a flat contiguous address space; "flat memory"/"flat-memory" → "comptime memory" / "byte-addressable" across `comptime_vm.zig` + the plan/checkpoint/CLAUDE docs (flag names `-Dcomptime-flat`/`SX_COMPTIME_FLAT` kept). > **NEXT — P5.6 (ALL bundling + code signing in `default_pipeline`).** First likely sub-task: port the > bitwise/shift ops (`shl`/`shr`/`bit_and`/`bit_or`/`bit_xor`/`bit_not`) into `comptime_vm` so the real bundler > path runs on the VM (the 1616 `shr` gap). Then move `platform/bundle.sx`'s per-target logic to read the > migrated `abi(.compiler)` getters + `fs`/`process` host-FFI, call `bundle()` from `default_pipeline` after > `link` when `bundle_path()` is set, and remove the `--bundle`/`post_link_module` Zig shim. - **P5.4 CORE — the whole build is sx-driven via `default_pipeline`; no Zig auto-emit/auto-link (2026-06-19).** The compiler's post-IR role is now: codegen → invoke the build callback. **There is NO auto-emit / auto-link.** Commits (all green): (1) **core** (`d178454`) — `emit_object()` is an ACTION (verify+emit via a host `BuildHooks` vtable; `main.BuildHooksCtx`); new query primitives `build_output`/`build_target`/ `build_frameworks`/`build_flags` (read the merged `BuildConfig`); `library/modules/build.sx` imports `compiler.sx` + defines `default_pipeline` (emit → gather c_objs → link); the compiler **force-lowers** `default_pipeline` (well-known name, `decl.isDefaultBuildPipeline`, force-lowered after Pass 2) and **auto-invokes** it post-codegen when no `on_build` override (`main` final fallback `invokeByName "default_pipeline"`); the BUILD path **auto-imports `modules/build.sx`** (prepends a synthetic import node in `compileWithTimer`) so prelude-less programs (asm tests) still get `default_pipeline`; removed the build cache short-circuits (a future cache can live in `default_pipeline`). (2) **on_build-only** (`65ac370`) — migrated all 9 `set_post_link_callback` callers to `on_build(cb)` (callback gains `opt: BuildOptions`); DELETED `set_post_link_callback`. **Override semantics changed:** an `on_build` callback REPLACES the build (must emit+link or `return default_pipeline(opt)` — delegation verified), unlike the old post-link callback that ran AFTER the auto-link. Reworked tests: 1662 (queries) + 1664 (override+List-grow) DELEGATE to `default_pipeline`; deleted 1661/1663 (primitives now exercised by EVERY AOT build). `sx run` (JIT) is UNTOUCHED (emits in-process, never invokes `default_pipeline`). Benign `.ir` churn each step; **703/0 both gates.** > **REMAINING P5.4 (the BuildOptions-surface migration — large, mechanical, dual-path, string-lifetime-sensitive; > NOT YET DONE):** **FINAL DIRECTION (user 2026-06-19): FULL MIGRATION — NO LEGACY. Drop gate-OFF entirely; > the VM is the SOLE evaluator; delete `interp.zig`. Migrate DIRECTLY to VM-native `abi(.compiler)` arms — NO > legacy `compiler_lib` handlers, NO dual-path.** See `PLAN-COMPILER-VM.md` → Phase 5 steps **P5.5–P5.8** for > the full spec. In brief: > - **P5.5** — migrate all 36 `BuildOptions :: struct #compiler` methods → free `ufcs … abi(.compiler)` decls + > `comptime_vm.callCompilerFn` arms (NO legacy handler). Setters dupe strings into a PERSISTENT allocator > (thread `emit_llvm.alloc` via e.g. `BuildConfig.string_alloc`). Kills the 4 strict `compiler_call` bails. > - **P5.6** — ALL bundling + code signing for EVERY target (macOS `.app`, iOS device/sim, Android `.apk`: > Info.plist/codesign/provisioning/entitlements/framework-embed/AndroidManifest/javac/d8/aapt2/zipalign/ > apksigner) runs in the sx `default_pipeline` (via the migrated getters + `fs`/`process` host-FFI). Remove the > `--bundle`/`post_link_module` Zig shim. Compiler keeps ONLY the linker primitive (Option B). > - **P5.7** — DELETE `#compiler` + `compiler_call` op + `compiler_hooks` (Registry/HookFn) + `interp.zig` > (Interpreter/Value/reflectTypeInfo/callExtern) + `regToValue`/`valueToReg` + the VM→legacy fallback; make > `-Dcomptime-flat` permanent. A VM bail is ALWAYS a build diagnostic now. Re-express `define`/`make_enum` as > sx. Land the 0141 repro; reconcile 1654. > - **P5.8** — build `~/projects/m3te` + `~/projects/distribution` end-to-end as the acceptance test that > `default_pipeline` covers all targets; add `.app` + `.apk` bundle smoke tests (no corpus coverage today). - **P5.3 (`on_build` registrar) — the build-callback registration mechanism; callback takes `BuildOptions` (2026-06-19).** Per the user's design: `on_build(cb)` is the build-callback registrar (a FREE fn), generalizing `set_post_link_callback` — the callback is `(opt: BuildOptions) -> bool abi(.compiler)` and the compiler invokes it post-codegen WITH the opaque `BuildOptions` handle. **Key simplification:** the handle is a single null-sentinel word, so passing it sidesteps the feared fat-`BuildConfig` marshaling. Changes: VM `callCompilerFn` `on_build` arm + legacy `handleOnBuild` (both set `post_link_callback_fn` + a new `BuildConfig.post_link_takes_options` flag); `comptime_vm` `runEntry`→`runEntryArgs(extra)` (implicit ctx + explicit args) + a public `runBuildCallback(..., pass_options)`; `core.invokeByFuncId`/`invokeByName` now take `pass_options` (was an always-empty args slice); `main.zig` passes `getPostLinkTakesOptions()`; `build.sx` `on_build` decl. Smoke test `1664-platform-on-build-callback` (AOT). Benign 37-`.ir` churn (type table +1 for the `on_build` fn type; behavior identical — verified only `.ir` streams changed). **705/0 both gates.** > **CONSOLIDATED REMAINING PLAN (P5.4 — from the user's 2026-06-19 direction; large + coupled + re-churns > snapshots; the bundler has NO corpus coverage = the stream's top risk):** > 1. **Migrate to `on_build` ONLY** — convert every `set_post_link_callback(cb)` caller (`platform/bundle.sx` > `bundle_main`, examples 1611/1614/1615/1616, 0602/0603) to `#run on_build(cb)` with `cb: (opt: > BuildOptions) -> bool`; DELETE `set_post_link_callback` (build.sx + compiler_lib + VM arm). > 2. **Bundle/Android config → sx data in the default script.** The `#compiler` accessors the user flagged — > `set_bundle_path`/`bundle_path`/`bundle_id`/`codesign_identity`/`provisioning_profile`, > `set_manifest_path`/`keystore_path`, `jni_main_count`/`jni_main_runtime_path_at`/`jni_main_java_source_at` > — move into the sx `BuildConfig`/default script (sx-owned data), not compiler hooks. > 3. **`default_pipeline` + override model.** `library/modules/build.sx` ships `#run on_build(default_pipeline)` > (the stdlib default); a user's `#run on_build(custom)` in main.sx OVERRIDES it (LAST-WINS — already the > behavior, since registration just overwrites `post_link_callback_fn`). `default_pipeline` calls > `emit_object`/`c_object_paths`/`link_libraries`/`link` + the sx bundler. > 4. **REMOVE the Zig driver's auto-emit/auto-link** (`main.compileWithTimer`) — COUPLED with (3): once > `default_pipeline` drives emit+link, the driver must stop doing them or it double-links. Riskiest piece > (whole build/bundle path; no corpus guard → needs dedicated bundle smoke tests). > 5. **Delete `#compiler`/`compiler_call`/`compiler_hooks`** + the S5a `build_options` once config is sx data → > kills the 4 strict `compiler_call` bails (1609/1614/1615/1616) → strict sweep green → `interp.zig` deletable. - **P5.2b (`link` ACTION) — the sx `link` primitive links on the VM via a host-installed vtable; build callback de-failable'd (2026-06-19).** Phase 5's one genuine ACTION primitive: `link(objects, output, libraries, frameworks, flags, target)` (in `library/modules/compiler.sx`). **USER DECISION this step: drop fallibility from the build callback** — so `link` is a plain VOID primitive (no `-> !`), and a link failure BAILS on the VM → hard build error (sidesteps the failable-tuple-return construction entirely). **The vtable:** `comptime_vm.zig` can't depend on the driver (`core`/`main`/`target`), so `link` dispatches through a new `compiler_hooks.BuildHooks { ctx, link_fn }` that `main.zig` installs into `BuildConfig.build_hooks` before the post-link callback. The driver side is `main.LinkHooksCtx` (holds allocator/io/base_config/has_jni_main; its `link` adapter unions the explicit `flags` with the CLI ones and calls `target.link(objects[0], objects[1..], …)` — the linker treats first-vs-rest as equal inputs). **New VM readers** (inverse of `makeStringList`): `readStringList` (a `List(string)` arg → `[][]const u8`, element bytes are views into stable comptime arena) + `readStringArg` (a `string` arg). Registered `link` on `bound_fns` (legacy stub bails — VM-only). **Smoke test** `examples/1663-platform-build-pipeline-link` (AOT): a post-link callback re-links the build's own objects (via `c_object_paths` + `emit_object`) into a temp output through the sx `link` primitive — and the **relinked binary is a FUNCTIONAL executable that runs** (verified manually). Build exit 0 only if the VM-driven link succeeds; **negative-probe verified** (bad output path → `ld` fails → `ComptimeVmBail: comptime link: linking failed`, build exit 1 — the P5.1 VM-reason diagnostic path). **The driver still auto-links too** (P5.2b does NOT remove the Zig driver's `target.link`; the test links to a SEPARATE temp output) — removing the auto-link + having `on_build` drive everything is P5.3/P5.4. **704/0 both gates.** - **P5.2 (metadata queries) — `c_object_paths` / `link_libraries` compiler primitives + the VM `List(string)` builder (2026-06-19).** Phase 5 step 2 (the read-only slice): two `abi(.compiler)` primitives that the sx build driver will pass to `link` — `c_object_paths() -> List(string)` (the `#import c` companion `.o`s) and `link_libraries() -> List(string)` (the `#library` names). They live in a NEW stdlib file `library/modules/compiler.sx` (the Phase 5 home the sx `default_build` grows into) and are serviced by `comptime_vm.callCompilerFn` reading two new `BuildConfig` fields (`c_object_paths`/`link_libraries`) that `main.zig` forwards before the post-link callback (alongside `binary_path`/`target_triple`/…). **Reusable new piece:** `Vm.makeStringList(table, list_ty, items)` builds a `List(string)` in comptime memory — backing array of `string` fat pointers + the `{items,len,cap}` struct, all laid out from the RESULT type's field offsets/types (target-aware, no hardcoded layout). To get the result type, `invoke`/`callCompilerFn` now thread the call instruction's `ins.ty` (the only call-result-type need so far). Legacy (`compiler_lib`) handlers for these bail loudly (`handleBuildPipelineQuery`) — they're VM-only by nature (the post-link callback always runs on the VM since P5.1), and a `List(string)` isn't faithfully buildable in the legacy `Value` model (0141). Registered on `bound_fns` so `weldedCompilerFn` recognizes them. **Smoke test** `examples/1662-platform-build-pipeline-queries` (AOT + a 1-line C `#source` → exactly one C object): a post-link callback asserts `c_object_paths().len == 1`, `items[0].len > 0`, and iterates `link_libraries()` (liveness touch) — build exit 0 only if the VM-built list is well-formed. **Negative-probe verified** a real guard (forcing `len != 2` → "post-link callback returned false", build exit 1). **`emit_object() -> string` ALSO landed (same step):** a QUERY, not an action — the compiler emits the object eagerly (the Zig driver, before the callback), so the primitive just returns the path from a new `BuildConfig.object_path` field `main.zig` forwards (no driver vtable needed). 1662's callback now also asserts `emit_object().len > 0`. So ALL THREE query primitives (`emit_object`/`c_object_paths`/`link_libraries`) are done; only `link` (the genuine ACTION) remains. **No unit test for `makeStringList`** — constructing a `List(string)` `TypeId` in the test harness needs generic instantiation; the corpus test exercises the real stdlib type end-to-end with a non-empty list + a negative guard instead. **`emit_object` + `link` (the ACTIONS) deferred to P5.2b** — they must replace the Zig driver's auto-emit/auto-link (not duplicate it), so they need the driver-restructuring + a host-installed callback vtable (the VM can't depend on `core`/`main`/`target`). **703/0 both gates** + strict JIT run clean (no `compiler_call` bail). - **P5.1 (= 4E) — the post-link build driver runs on the VM (NO fallback); smoke test 1661 (2026-06-19).** Phase 5 step 1: `core.invokeByFuncId` — the post-codegen / post-link callback invocation `main.zig` fires after `target.link` — now routes the callback through the **comptime VM** (`comptime_vm.tryEval`) instead of the legacy `Interpreter`. **REQUIRED** because the sx build driver allocates/grows `List`s, which the legacy interp can't do at comptime (issue 0141: `struct_get: base has no fields`); the VM can. **NO fallback** (user directive): a side-effecting post-link callback can't safely re-run on a second evaluator (double execution), so a VM bail is a HARD build error — `error.ComptimeVmBail`, with the reason in `comptime_vm.last_bail_reason` (now surfaced by `main.printInterpBailDiag`, which previously only read the legacy interp's `last_bail_*` statics). `BuildConfig` (`&emitter.build_config`) + `import_sources` are threaded into the VM call. Deleted the now-dead `flushInterpOutput` (the VM writes `out` directly via host-FFI — no buffer to flush). Non-empty `args` rejected loudly (`error.ComptimeVmArgsUnsupported`) — the `on_build(config)` arg-passing entry arrives in P5.3. **Verification:** a probe with a List-growing post-link callback FAILS on the pre-change legacy path (`sx build` exit 1, `OutOfBounds (op=struct_get)`) and SUCCEEDS after the change (exit 0). Formalized as `examples/1661-platform-post-link-vm-list` (`{ "aot": true }`): the callback grows a `List` to 3 + returns `len == 3`; the build links cleanly (exit 0) and the binary prints `runtime main`. AOT snapshots the binary's streams (build stdout discarded), so the VM-success is pinned via exit 0 + `runtime main` — a legacy regression would flip the build to exit 1 and mismatch. **No corpus example fires post-link** (none had AOT sidecars; the platform examples register a callback at `#run` time but run JIT) — so `invokeByFuncId` was previously untested by the corpus; 1661 is the first coverage. The 4 strict `compiler_call` bails (1609/1614/1615/1616) are UNAFFECTED — they bail at `#run configure()` on still-`#compiler` accessors (`set_bundle_path` etc.), killed by P5.4, not here. **702/0 both gates.** - **4B (VM-native diagnostics) — the metatype negative tests (1179/1180) render proper diagnostics under strict; strict gap-bails now ONLY `compiler_call` (2026-06-19).** The legacy and VM both BAIL on a `define()` validation failure with an identical detail string; only the host's STRICT rendering differed (generic "bailed on the VM (strict)" vs the proper "comptime type construction failed: " + span the non-strict legacy path emits). Fixed: (1) aligned the VM's `define` messages with the legacy's exact text — `comptime define():` (was `comptime define:`), and the duplicate variant/field cases now NAME the offender via a new `failFmt` helper (`'...' duplicate variant name 'value'`). (2) The strict type-fn path (`lower/comptime.zig`) now emits `d.addFmt(.err, span, "comptime type construction failed: {s}", .{vm_reason})` — the SAME diagnostic as the legacy fallback, so **1179/1180 produce their exact expected `.stderr` under strict with NO legacy interp involved**. Left the const-init/`#run` strict paths on the "bailed on the VM" wrapper ON PURPOSE — they still carry genuine VM-gap bails (`compiler_call`), so the burndown sweep must keep distinguishing those. **701/0 both gates.** **STRICT GAP-BAILS NOW: only the 4 `compiler_call` (1609/1614/1615/1616 → Phase 5 sx-build-pipeline)** + 1654 (a legitimate unresolvable-symbol diagnostic — an asm global called at comptime; the legacy can't resolve it either; reconciles to VM wording at the 4F flip). So: BuildOptions/Phase 5 is the ONLY thing between the VM and a green strict sweep. - **`out` is now a PLAIN SX FUNCTION (libc `write`), NOT a builtin — VM handles it via host-FFI; `trace_resolve` ported; 0522 fixed (2026-06-19).** Per user: removed the `out` `#builtin` entirely. `library/modules/std/core.sx` now defines `libc_write :: (fd, [*]u8, usize) -> isize extern libc "write"` + `out :: (str: string) { libc_write(1, str.ptr, xx str.len); }`. Deleted `BuiltinId.out` (`inst.zig`), the `resolveBuiltin` "out" mapping (`call.zig`), the sema builtins-list entry (`sema.zig`), and BOTH `.out` arms (`interp.zig` buffered-append, `ops.zig` LLVM `write` lowering). **At comptime `out` runs through the evaluator's host-FFI** (the VM's dlsym `write` path / the interp's extern call) — so the VM HANDLES `out` with NO special arm. Benign prelude `.ir` churn (`[*]u8` interned earlier + `@out`→`@write` + the `out` fn body) → regen'd 54 `.ir` snapshots (verified: only string-table renumber + the intended decl/fn-body change; zero stdout/exit changes). **This UNMASKED two latent VM gaps the `out`-bail was hiding** (the VM now runs past `out`): (1) **`trace_resolve`** (1035) — PORTED to the VM (`comptime_vm.zig`): unpack the `(func_id<<32|offset)` comptime frame, resolve func name + `file:line:col` + source line via a **`source_map` now threaded into the VM** (new `tryEval` param, `&import_sources` from emit_llvm), build the `{file,line,col,func,line_text}` `Frame` struct in comptime memory (`makeStringValue`/`writeField`/`fieldOffset`). (2) **0522** (bare-pack `[]Any`) — was a CRASH (`reflectArgTypeId` `@intCast` of a garbage word) → hardened to a loud bail (`typeIdxOf` checked cast; the VM must never panic). ROOT CAUSE: after the 0143 fix `$args` materializes as `[]type_value` (8-byte), but the example declared `describe(args: []Any)` (16-byte) → every element past the first read at the wrong stride; the legacy's loose Value model tolerated it, the byte-accurate VM didn't. The bare-pack elements ARE `Type`s, so the fix is the honest type — `describe(args: []Type)` (output identical). **Result: `out`/`trace_resolve`/the 0522 pack-reflection all run VM-HANDLED under strict** (0613/1035/0522/1038 no longer bail). **701/0 BOTH gates + full suite.** (Build-pipeline relevance: the sx `default_build` driver uses `out` for diagnostics — now VM-native; no compiler `out` builtin to special-case.) **THEN `interp_print_frames` ported to the VM too** (1034): unlike `out` it needs the live evaluator call-chain, so it's a VM arm (mirrors legacy `printInterpFrames`) — walks `call_stack` (skips the last frame), writes ` at ` lines straight to fd 1 (consistent with `out`'s direct `write`). 1034 matches; 701/0. **STRICT DELETION-GATE NOW DOWN TO 7 (all known categories):** `compiler_call` (4 — 1609/1614/1615/1616, the still-`#compiler` BuildOptions accessors → Phase 5 sx-build-pipeline) · VM-diagnostic negatives (2 — 1179/1180, the `define` bail IS the expected outcome → **4B**: surface as a proper build diagnostic) · target-specific dlsym (1 — 1654, an asm global called at comptime; legacy can't resolve it either → a clean diagnostic, not a bug). EVERY pure + side-effect op bail is cleared. - **DESIGN PIVOT (2026-06-18, user) — DRIVE THE BUILD PIPELINE FROM SX; the 37-hook BuildOptions port is dead.** Trigger: porting each `BuildOptions` accessor to an `abi(.compiler)` fn that delegates to a `compiler_hooks` hook just re-encodes sx-level logic (setters/getters, `is_macos` triple-matching, list appends) as compiler hooks — they need NOTHING from the compiler but the `BuildConfig` state. So instead: **`BuildConfig` becomes plain sx data** (ordinary struct, sx-owned, no `#compiler`/hooks/shared-state/weld), and the **build pipeline is an sx program** — the logical end of "bundling lives in sx". The compiler shrinks to a few `abi(.compiler)` PRIMITIVES taking EXPLICIT args (`emit_object() -> !string`, `link(objects, output, libs, fws, flags, target) -> !`, metadata queries) + an `on_build : (BuildConfig) -> ! abi(.compiler)` slot (stdlib default `default_build`; user overrides via `#run on_build = build;`). **Chosen boundary: Option B** (compiler keeps the Zig linker; sx owns config+orchestration+bundle); Option A (sx shells `cc`/`ld`) is a later refinement. **NO bool** — failures are the error channel (`-> !`); VERIFIED on the current build: void `#run`, `-> !`/`-> !E` failable `#run`, and a `raise` at `#run` fails the build with a return trace (+ suggests `#run … catch (e){…}`). `on_build` GENERALIZES today's `post_link_callback_fn` (assignable typed global w/ default, vs a setter). **Full design + step plan in `PLAN-COMPILER-VM.md` → Phase 5.** **S5a (below) is a green intermediate that the sx-pipeline replaces wholesale** (don't extend it; P5.4 deletes `build_options`/`set_post_link_callback` + all `#compiler`). **NEXT — P5.1 (= 4E):** route the post-codegen / `on_build` invocation through the VM (`core.invokeByFuncId` → VM), REQUIRED because the sx driver allocates `List`s and the legacy interp can't (0141, VERIFIED: comptime `List` growth works on the VM, fails on legacy with `struct_get: base has no fields`). Add dedicated bundle smoke tests (no corpus coverage today). Both gates **701/0**. - **S5a DONE — `build_options` + `set_post_link_callback` migrated off `#compiler` onto `abi(.compiler)`; `BuildConfig` threaded into the VM (2026-06-18).** The corpus-covered slice of the BuildOptions migration. (1) `comptime_vm.zig` — `Vm.build_config: ?*BuildConfig`, threaded via a new `tryEval` param (`&self.build_config` from emit_llvm's `#run`/const-init sites; `null` at lowering-time type-fn). (2) Two `callCompilerFn` arms: `build_options` (returns the null-sentinel handle) + `set_post_link_callback` (reads the cb `func_ref`, stores `post_link_callback_fn` on the threaded `BuildConfig`). (3) `compiler_lib.zig` — matching legacy `handleBuildOptions`/`handleSetPostLinkCallback` (gate-OFF dual path). (4) `build.sx` — `build_options :: () -> BuildOptions abi(.compiler);` and `set_post_link_callback` EXTRACTED from the `struct #compiler` as a free `ufcs (…) abi(.compiler)` (so `opts.set_post_link_callback(cb)` still resolves via UFCS); the other ~38 BuildOptions methods stay `#compiler` for now. (5) Registrars/callbacks that call these are now compiler-domain: `platform/bundle.sx` `bundle_main :: () -> bool abi(.compiler)`, and the six platform examples' `configure`/`configure_build` registrars marked `abi(.compiler)`; 0602/0603 reworked the same way. **KEY learning:** every example transitively imports `build.sx` via the prelude, so the `set_post_link_callback` method→free-function change is BENIGN `.ir` churn (declaration renumber + global `@str`/`@tag.str` suffix shift) in all 37 examples that have `.ir` snapshots — verified line-by-line that NO instruction/control-flow/payload changed (only auto-numbered global-name suffixes), then regen'd those 37 snapshots scoped with `-Dname`. **Strict-VM `compiler_call` bail set dropped 6→2:** 0602/0603/1604/1611 now fully VM-HANDLED; 1609/1615 still bail on the *other* (still-`#compiler`) BuildOptions methods they use → **S5b** (migrate the remaining ~38 setters/getters). **701/0 BOTH gates + all unit tests.** - **S3 DONE — emit_llvm skips BODIED `abi(.compiler)` (compiler-domain) functions; comptime-only calls emit `undef` (2026-06-18).** A BODIED `abi(.compiler)` function is a user compiler-domain function (post-link callback / compiler helper): the comptime evaluator runs its sx body, but it NEVER runs in the binary, so the backend skips it. Changes: (1) IR `Function` gained `is_compiler_domain: bool` (`inst.zig`). (2) `decl.zig` — new `fnIsBodilessCompiler` splits the API surface (bodiless → declare-only, `compiler_welded`, no implicit ctx — the S1 behavior) from a bodied `abi(.compiler)` function (lowers its body for VM eval; flagged `is_compiler_domain` + `is_comptime`; gets normal implicit-ctx). The four S1 guards now gate on `fnIsBodilessCompiler` not `fd.abi == .compiler`. (3) `emit_llvm.zig` — Pass 2 skips `is_compiler_domain` bodies; Pass 1 declares them EXTERNAL-linkage (an internal empty decl fails LLVM verification). (4) **KEY** `ops.zig` `emitCall` — a call to a comptime-only callee (`compiler_welded` OR `is_compiler_domain`) from a dead comptime body now emits `undef` instead of a real `call`; the runtime-call gate covers both. Without the undef, an AOT `sx build` left an undefined `_double`/`_intern` symbol — this ALSO fixed a pre-existing, untested AOT breakage of the bodiless compiler-API examples (the corpus runs them JIT). Diagnostic reworded "compiler-library" → "compiler-domain" (1185 snapshot regen'd). Regression: `examples/0638-comptime-domain-fn-not-emitted` (`double` folds a `#run` const → 84, absent from the binary via `nm`, JIT + AOT both run). **701/0 both gates + all unit tests.** **NEXT: S4** — an `abi(.compiler)` function-TYPE param (`cb: () -> bool abi(.compiler)`) flags the bound function compiler-domain (so a plain `bundle_main :: () -> bool { … }` becomes compiler-domain when passed to `set_post_link_callback`). Then S5 (BuildOptions migration + delete `#compiler`/`compiler_call`/`compiler_hooks`). - **S1+S2 DONE — `abi(.compiler)` replaces `abi(.zig) extern compiler` + `#library "compiler"` (clean cutover, no legacy path) (2026-06-18).** Per the design pivot below, and the user's "no legacy paths": REMOVED the `.zig` ABI variant entirely (`ast.ABI` is now `{ default, c, compiler, pure }`) and made `abi(.compiler)` the sole spelling for a compiler-domain / compiler-API function — the ABI alone marks it, no `extern `, no fake `#library "compiler"`. Changes: (1) `ast.zig` — `.zig` → `.compiler` (doc rewritten). (2) `parser.zig` — `parseOptionalAbi` accepts `.compiler` (drops `.zig`); a **bodiless `abi(.compiler)` decl** (ends in `;`, no `extern`) is now accepted — synthesizes the empty-block placeholder like an `extern` import (the Zig/VM handler is the impl). (3) `decl.zig` — `weldedCompilerFn` keys off `fd.abi == .compiler` + export-list membership (no `extern_lib == "compiler"` check); a bodiless `abi(.compiler)` decl lowers extern-like (`is_extern_decl`, and the two body-lowering paths `lowerFunction`/`lazyLowerFunction` skip it) so it is declared-not-defined; `funcWantsImplicitCtx` returns false for `abi == .compiler` (an implicit `__sx_ctx` prepend would shift args and break the handler arity — this was the live bug surfaced + fixed). (4) `type_resolver.zig` — the function-TYPE CC switch handles `.compiler` (sx-default CC). (5) Migrated ALL 8 compiler-API examples (0626/0628/0629/0630/0631/0633 + the 1184/1185 negatives) `… abi(.zig) extern compiler;` → `… abi(.compiler);` and deleted every `compiler :: #library "compiler";` line; regen'd the 1184 stderr snapshot (new "not a function exported by the compiler" wording + shifted line). (6) Updated the two parser unit tests. **All 8 examples run HANDLED on the strict VM with byte-correct output; 1184 (unexported name) + 1185 (runtime call) still error cleanly; gate-OFF legacy still works.** **700/0 BOTH gates + all unit tests.** NOTE: the general `#library`/`extern ` PARSE paths stay (used by `libc :: #library "c"` etc.) — only the compiler-API's USE of them is gone. `compiler_lib.lib_name` + the `main.zig` dlopen-skip for a "compiler" lib are now dead defensive code (harmless; a `#library "compiler"` is just meaningless now). The struct-`abi(...)` parse slot is vestigial (weld stripped) — parse-only test kept. **NEXT: S3** — emit_llvm skips BODIED `abi(.compiler)` functions (Pass 2, like `is_extern`); thread an `abi(.compiler)` flag onto the IR `Function` and refine the three "today every `abi(.compiler)` fn is bodiless" guards in `decl.zig` (marked with `S3 NOTE`) to allow a bodied callback's body to lower for VM eval while NOT emitting it. Then S4 (callback-param propagation) + S5 (BuildOptions migration). - **DESIGN PIVOT (2026-06-18, user) — `abi(.compiler)` is the compiler-domain ABI; DROP the fake `#library "compiler"`.** Supersedes both the `abi(.zig) extern compiler` + `#library "compiler"` binding mechanism AND the previous "runtime-reachability gating" idea for the BuildOptions blocker (entry below). **The unifying concept:** a function is *compiler-domain* (runs in the comptime evaluator, NEVER in the shipped binary) because its **ABI says so** — `abi(.compiler)` — not because it's "extern" to an imaginary library. One annotation covers BOTH roles: 1. **Compiler-API surface** (`intern`, `text_of`, `find_type`, `declare_type`, `register_type`, `build_options`, `set_post_link_callback`, …): bodiless `abi(.compiler)` decls (the Zig/VM handler IS the impl). Replaces `… abi(.zig) extern compiler;` + the `compiler :: #library "compiler";` line — both GO AWAY. 2. **User compiler-domain functions** (post-link callbacks like `platform.bundle.bundle_main`): BODIED `abi(.compiler)` functions. emit_llvm does NOT lower them (skip in Pass 2, like `is_extern`); the comptime VM/interp evaluates them. A callback PARAM type carries it too — `set_post_link_callback(self, cb: () -> bool abi(.compiler))` — so the bound function is flagged compiler-domain. **Why this dissolves the BuildOptions blocker:** the welded-call enforcement (`ops.zig` `emitCall`) only fired because comptime-only callback bodies (`bundle_main`, 0602's `configure`) were being LLVM-emitted. A bodied `abi(.compiler)` function is never emitted → its `build_options()`/`binary_path()` calls never reach `emitCall` as runtime code → no enforcement, no undefined-symbol risk. **1185 stays correct**: `main` is an ordinary runtime fn (not `abi(.compiler)`) calling a compiler-domain fn → still a clean build-gating error. (The registrar half is independently fine via the idiomatic `#run { … }` block — the welded calls sit in the `is_comptime` `__run` wrapper; 0602/0603 only tripped via an intermediate `configure()`, a test-shape artifact.) **Staged plan (each its own step, both gates green):** - **S1 — introduce `abi(.compiler)`** as a new `ABI` variant that marks a function `compiler_welded` (export-list checked) WITHOUT requiring `extern compiler`/`#library`. Add it ALONGSIDE the existing `.zig extern compiler` path so migration is incremental; prove with one example (0626 → `abi(.compiler)`). (`.zig` is a misnomer — "we don't really have a zig abi"; it becomes `.compiler`, ultimately replacing `.zig` once all callers move.) - **S2 — migrate the rest of the compiler-API decls** (0628–0633, 1184/1185) to `abi(.compiler)`; drop the `#library "compiler"` lines; regen snapshots (the 1184 unexported-name + 1185 runtime-call diagnostics must stay red with refreshed wording). Then retire the `.zig extern compiler` parse path + `#library "compiler"`. - **S3 — emit_llvm skips bodied `abi(.compiler)` functions** (Pass 2 `continue`, like `is_extern`); thread the `abi(.compiler)` flag onto the IR `Function`. Prove a bodied compiler-domain function isn't emitted. - **S4 — callback-param propagation**: an `abi(.compiler)` function-type PARAM flags the bound function compiler-domain. - **S5 — BuildOptions migration** (now unblocked): `build_options`/`set_post_link_callback`/… become `abi(.compiler)` (+ VM `callCompilerFn` arms / legacy `compiler_lib` handlers; `BuildConfig` threaded into the VM — the bundler 4E shares this); callbacks declared/typed `abi(.compiler)`; delete `#compiler`/`compiler_call`/ `compiler_hooks` Registry. Then **4E** bundler on the VM. **Reusable facts from the reverted attempt:** only `build.sx` uses `#compiler`; VM dual-path bail-to-fallback means the VM needs only corpus-covered fns; UFCS on a free fn needs the `ufcs` marker (composes with the ABI annotation); the binding mechanism currently lives in `decl.zig` `weldedCompilerFn` (keys off `extern_lib == "compiler"` — S1 makes it key off `abi == .compiler`). Mechanism files: `ast.zig` (`ABI` enum), `parser.zig` (`parseOptionalAbi` + the extern-compiler postfix), `decl.zig` (`weldedCompilerFn`), `compiler_lib.zig` (export list), `comptime_vm.zig` (`callCompilerFn`), `emit_llvm.zig` (Pass-2 skip), `ops.zig` (`emitCall` gate). - **Phase 4 — BuildOptions→`abi(.zig) extern compiler` migration ATTEMPTED, then REVERTED; BLOCKER found: the comptime-only welded-call enforcement (2026-06-18).** Scoped an incremental slice (migrate only the corpus-covered `build_options()` + `set_post_link_callback`, leaving the 38 bundler accessors on `#compiler` → VM bails → legacy fallback). Built it end-to-end: threaded `BuildConfig` into the `Vm` (`tryEval` gained a `?*BuildConfig` param, passed `&self.build_config` from emit_llvm's `#run`/const-init sites); added `callCompilerFn` arms + legacy `compiler_lib` bound-handlers for both; rewrote `build.sx` (`build_options` → `abi(.zig) extern compiler`; extracted `set_post_link_callback` out of the `struct #compiler` as a free `ufcs (...) abi(.zig) extern compiler` fn so `opts.set_post_link_callback(cb)` still resolves via UFCS; added `compiler :: #library "compiler";`). All COMPILED and the welded dispatch fired. **BLOCKED at LLVM emission, NOT a bug — a design limitation the migration surfaces:** a `compiler_welded` call inside a NON-`is_comptime` function is a hard build-gating error (`ops.zig` `emitCall`, the Phase-1 enforcement guarding genuine runtime misuse — example 1185). But the post-link callback idiom calls comptime-only-API functions (`build_options()`, `binary_path()`, `bundle_path()`, …) **inside callback bodies** (`platform/bundle.sx`'s `bundle_main :: () -> bool`, and 0602's `configure`) that run ONLY at comptime (post-link interp/VM) yet are still LLVM-emitted as real `() -> bool` bodies. The OLD `#compiler`/`compiler_call` path emitted those as dead `undef` (`emitCompilerCall`), so no error; the welded enforcement instead halts the build, and it CANNOT distinguish a dead comptime-reachable body from genuine runtime use (1185, reachable from `main`) without runtime-reachability analysis. **Reverted the whole attempt** (kept only the green pure-ops work); both gates back to **700/0**. **THE DECISION the next session must make FIRST (before any BuildOptions migration):** how to emit a welded call in a comptime-only-but-LLVM-emitted function. Recommended path **A — runtime-reachability gating:** in `emit_llvm`, mark functions reachable from runtime roots (`main` / exported runtime fns); a welded call in an UNREACHABLE function emits `undef` (dead, like `compiler_call` did) instead of erroring, while a reachable one still errors (1185 stays red). This is also the right foundation for eventually NOT emitting comptime-only bodies at all. Rejected: (B) marking callbacks `is_comptime` — can't statically identify which `func_ref`s become post-link callbacks; (C) blanket softening to `undef` — would silently swallow genuine runtime misuse (1185). **Other migration facts confirmed this attempt (reuse next session):** only `build.sx` uses `#compiler` (the `issues/*.md` hits are doc text); the VM dual-path bail-to-fallback means the VM needs only the corpus-covered fns, the 38 bundler accessors can ride legacy; UFCS on a free fn requires the `ufcs` marker, which composes with `abi(.zig) extern compiler`; `build.sx` must declare `compiler :: #library "compiler";`. Do the reachability fix as its OWN step (verify 1185 still errors + a comptime-only-body welded call now emits clean), THEN redo the BuildOptions slice on top. - **Phase 4 burndown — three PURE comptime ops ported (`error_tag_name_get` + `global_addr` + `type_is_unsigned`); `interp_print_frames` correctly DEFERRED (2026-06-18).** Also ported `type_is_unsigned` (a `BuiltinId` via `callBuiltinVm`): resolves the queried `TypeId` the same way as `type_name` (a `.type_value` word, or an Any box `{tag@0,value@8}` whose tag IS the boxed value's type) then returns `table.isUnsignedInt(tid)`. Extracted the shared resolution into a `reflectArgTypeId` helper (VM-native `Value.reflectTypeId` mirror) so `type_name` + `type_is_unsigned` can't drift. MATCH-verified by a new VM unit test (`type_is_unsigned(u32) - type_is_unsigned(i64) == 1`). Strict sweep: 0600 `type_is_unsigned`→`out` (now its only remaining bail); no `type_is_unsigned` bails remain in the corpus. **With this, all PURE comptime ops are ported** — the remaining strict bails are side-effect (`out`/`interp_print_frames`), `compiler_call` (the BuildOptions migration), VM diagnostics (1179/1180), and `#insert`/bundler. Ported two side-effect-free ops onto the VM (`comptime_vm.zig` exec switch): (1) `error_tag_name_get` — a runtime tag-id word → its name string via `table.getTagName` + `makeStringValue` (uses the table, not the module, so it's unit-testable; `self.table == &module.types`); (2) `global_addr` — name-matches `__sx_default_context` and returns the already-tested `materializeDefaultContext` Addr (an aggregate value IS its address, so a downstream `load` sees the materialised Context), bailing for any other global exactly like legacy. **MATCH verification:** `error_tag_name_get` locked in by a new VM unit test (tag id → `"Bad"`, via `regToValue`); `global_addr` proven by the strict sweep (0600's first bail moved past it) and reuses `materializeDefaultContext`, already exercised by every implicit-ctx comptime call on the VM. **KEY CORRECTION to the handover's "three PURE ops" plan:** `interp_print_frames` (1034) is NOT pure — it WRITES the comptime call-frame chain to the build output, a side effect in the SAME bucket as `out` (the VM has no output buffer; output is direct-write, so a print-then-bail double-prints under the legacy fallback). It must land atomically in the FINAL `out`/strict-default step, NOT now. **Strict-sweep burndown:** 1035 `error_tag_name_get`→`out`; 0600 `global_addr`→ `type_is_unsigned` (a NEW pure-op bail surfaced — still a known pure op, next to port); 1034 stays at `interp_print_frames` (deferred, as it should). Also fixed the stale `comptime_vm.zig` header comment (it still said "bump/stack allocator"; the memory model is an ARENA of stable host allocations since 4D.0). **700/0 BOTH gates + all unit tests.** On `reify`. - **Phase 4 burndown — issue 0143 FIXED (pack-as-`[]Type` stride) + regression test (2026-06-18).** Root cause was a stale consequence of the `.type_value` migration: `buildPackSliceValue` (`lower/pack.zig`) materialized a bare `$` `[]Type` slice as `[]Any` (16-byte elements) while `const_type` now yields an 8-byte `.type_value` and `[]Type` resolves to `[]type_value` — so 8-byte words sat in 16-byte slots and an 8-byte-stride reader got `[t0, pad, t1, …]`. Fixed by building the array+slice as `.type_value` (8 bytes). Removed the stopgap `type_name` `.unresolved` guard (its whole reason is gone; dropping it keeps any future stride bug VISIBLE as wrong output rather than a silent fallback). Sibling `materialisePackSlice` checked — it genuinely boxes values into `[]Any` (correct, not the same bug). Regression test `examples/0525-packs-pack-as-type-slice-arg`. **700/0 both gates.** 0114 (and 0521/0522/0524) now bail ONLY at `out` (the deferred end-state op) — the type bug is gone. issue 0143 RESOLVED. - **Phase 4 burndown — switch_br + type_name ported; issue 0143 filed; KEY sequencing insight: `out` is end-state-only (2026-06-18).** Ported two PURE comptime ops (`379ed05`): `switch_br` (i64-discriminant multi-way branch — enum/error tag or `.type_value` index) and `type_name` (Type value / Any box → `table.typeName`, with an `.unresolved`-bail guard). Correct in isolation; 0520–0524 run GREEN under strict. **Two blockers found:** 1. **issue 0143 (FILED, OPEN) — pack-as-`[]Type` stride mismatch.** A `..$args` pack forwarded as a `[]Type` ARGUMENT across a call is backed by a `[N x Any]` (16B) array but viewed as `[]type_value` (8B) → half-stride reads (`[i64 string]` vs legacy `[i64 string bool]`). A LOWERING bug the legacy's Value model masks; the byte-accurate VM exposes it. Blocks `examples/0114` from running HANDLED. **Per CLAUDE.md: filed, NOT worked around** (the `type_name` `.unresolved` guard just makes the VM decline rather than emit garbage). Repro + fix-prompt in `issues/0143-…md`. 2. **`out` (comptime print) is an END-STATE op — it cannot land while the fallback exists.** Under the legacy fallback, an eval that prints via `out` then BAILS double-prints (the VM wrote to fd 1, then legacy re-runs the whole eval — no rewind). 0114 demonstrated it. So a direct-write `out` is only safe once the fallback is GONE (strict-by-default). **Revised ordering:** land the PURE ops (switch_br/type_name/type_is_unsigned/error_tag_name_get/global_addr/interp_print_frames) + the BuildOptions migration + #insert + bundler FIRST; then in the FINAL step flip strict-to-default (removing the fallback) AND add `out` together — at which point every `out`-using example flips atomically with deletion. (Most of the gap-list examples print, so they stay on fallback until that final flip — that's expected, not a regression.) 699/0 both default gates. - **Phase 4 — STRICT no-fallback mode (the interp-retirement enumeration gate) + full gap list (2026-06-18).** Added `-Dcomptime-flat-strict` / env `SX_COMPTIME_FLAT_STRICT` (implies `comptime_flat`): at all THREE comptime sites (type-fn in `lower/comptime.zig`, const-init + `#run` in `emit_llvm.zig`) a VM bail becomes a build-gating error naming the reason INSTEAD of falling back to legacy. This forces every comptime eval onto the VM so the complete gap set is enumerable in one sweep; when the corpus is green under strict mode AND every example MATCHES legacy, the VM handles everything and `interp.zig` can be deleted (4F). Default behaviour unchanged — **699/0 both default gates**. (Fixed a wiring bug: the type-fn site's local `comptime_flat` didn't include the strict flag, so every type-fn falsely reported ``; now strict implies flat there too.) **THE DELETION CHECKLIST (19 strict bails, swept via `SX_COMPTIME_FLAT_STRICT=1` over examples+issues; 0103/0800 "WRONG" were false positives — raw heap-pointer addresses the corpus normalizes):** - `switch_br` (5): 0114, 0521, 0522, 0524, 1035 — port the type-category multi-way branch (trivial jump). **CAUTION:** porting it (+`type_name`) UNMASKS a silent-wrong in 0114 — a `[]Type` slice materialized when a pack (`$args`) is passed ACROSS A CALL reads its `string` element as ``. Must fix that VM pack-Type-materialization bug, not just add the op. - `compiler_call` (6): 0602, 0603, 1604, 1609, 1611, 1615 — the **BuildOptions → `abi(.zig) extern compiler`** migration (delete `#compiler`/`compiler_call`; thread `BuildConfig` into the VM). Big. - `out` (2): 0613, 1038 — comptime print. Direct write to fd 1, BUT only safe when the WHOLE eval is VM-handled (a print-then-bail double-prints under the legacy re-run — 0613). Flip atomically. - `type_name` (1): 0520 — reflection reader (`.type_value` word / Any-box tag → `table.typeName`). - `global_addr` (1): 0600 — only `&__sx_default_context` is materialised (mirror legacy). - `interp_print_frames` (1): 1034 — return-trace frame printing. - VM-native diagnostics (4B) (2): 1179, 1180 — NEGATIVE tests; the VM bail (`define: enum has no variants` / `duplicate variant name`) IS the expected outcome → must surface as the proper build-gating diagnostic, not the generic strict error. - dlsym not found (1): 1654 — a target-specific `extern` (asm global) called at comptime; likely a legitimately-unresolvable case → confirm it stays a clean diagnostic. **Sweep command:** `SX_COMPTIME_FLAT_STRICT=1 ./zig-out/bin/sx run ` per example, diff vs legacy; a strict bail prints `... bailed on the VM (strict, no fallback): `. - **Phase 4D.2 (VM plan) — extern SLICE/string args (→ NUL-terminated `char*`) + float guards (2026-06-18).** Extracted `marshalExternArg`: a scalar/pointer WORD passes verbatim (a `cstring` arg already works as a pointer word via 4D.1); a `string`/slice `{ptr,len}` fat pointer is copied into a NUL-terminated arena buffer and its `char*` passed (mirrors legacy `marshalExternArg` — what the bundler's `popen(cmd: [:0]u8, …)` needs). Added FLOAT guards on args AND returns: floats are `kindOf == .word` but the host_ffi trampolines have no float variant, so they bail loudly rather than miscall through an integer register (the legacy interp doesn't support float FFI either, so parity holds — no corpus float-FFI example exists). New example `0637-comptime-extern-slice-arg` (`#run strlen("hello, world")` with a `[:0]u8` param → 12) runs **HANDLED on the VM**, byte-matching legacy. **699/0 BOTH gates.** On `reify`. The FFI escape is now complete for scalar/pointer/cstring/ slice args + scalar/pointer returns — enough for the bundler's libc surface. **Next (4D.3):** `compiler_call` (#compiler hooks — 0602/0603), the last legacy-only role besides #insert/bundler. - **Phase 4D.1 (VM plan) — general host-FFI escape: the VM calls any extern libc fn via dlsym + host_ffi (2026-06-18).** Replaced the "extern not ported → bail" stub in `Vm.invoke` with `callHostExtern`: resolve the symbol via `host_ffi.lookupSymbol` (dlsym RTLD_DEFAULT) and dispatch through the `host_ffi` trampolines, exactly like the legacy `interp.callExtern`. **Marshalling is now trivial because `Addr` is a real host pointer (4D.0):** every WORD-kind arg passes as `usize` verbatim — a scalar's bits OR a pointer, no translation — and a pointer return is a valid `Addr`. Picks `callPtrRet` (void*-ABI) for pointer-ish returns, `callIntRet` (i64-ABI) otherwise; honors variadic (`is_variadic and args > fixed`). Non-word (aggregate/string/float) args+returns bail loudly (no silent miscall — 4D.2 adds NUL-term cstring marshalling + float). NOT per-builtin: ONE general mechanism for all externs. New example `0636-comptime-extern-libc` (`#run toupper(97)`/ `tolower(90)` fold to 65/122) runs **HANDLED on the VM**, output byte-matching legacy. (`abs` doesn't dlsym-resolve on macOS — a compiler builtin — and the VM fails identically to legacy, confirming parity.) **698/0 BOTH gates** (one new example). On `reify`. **Next (4D.2):** string/aggregate extern args (string→NUL-term cstring) + float args/returns, then `compiler_call` (#compiler hooks, 4D.3). - **Phase 4D.0 (VM plan) — comptime VM memory = an ARENA of stable host allocations; `Addr` = real host pointer (2026-06-18).** Replaced the growable `ArrayList(u8)` flat buffer (which reallocs/MOVES on growth) with a `std.heap.ArenaAllocator`: each `allocBytes` is a separate arena allocation that never moves and is freed wholesale on `deinit` (no per-object free, no cap, no fixed buffer). **`Addr` is now the allocation's absolute host pointer** (`@intFromPtr`), not an offset — so a comptime pointer and an FFI-returned host pointer are the SAME kind of value, and the FFI bridge (4D.1) can pass them to/from libc with ZERO translation and no per-call pinning (the original moving-buffer hazard is gone by construction). `Machine.readWord/writeWord/bytes` deref the absolute pointer directly, keeping the null-check bail (the malformed-IR / null-deref safety contract). Dropped the offset-based upper-bounds check (can't bound an absolute pointer; the `Frame.bad_ref` guard still catches the dominant malformed-IR vector) and the test-only `mark`/`reset` (the arena has no cheap reset-to-mark; the VM never used them outside tests). Decision rationale (user): use a GPA-like allocator, no artificial buffer limits. **697/0 BOTH gates + all unit tests** (rewrote the two Machine tests: null-deref bail + arena-stability-across-grows). Pure refactor, no comptime behavior change. **Next (4D.1):** extern-call dispatch in `Vm.invoke` — marshal args (scalars by value, pointers as the host pointer they already are), call via `host_ffi` trampolines, return scalars/pointers; a new `#run` libc example as the corpus guard. - **Phase 4A.1 (VM plan) — `box_any`/`unbox_any` on the VM + `.any` as a 16-byte aggregate (2026-06-18).** Ported the Any-boxing conversion pair: `box_any` allocates the 16-byte `{ type_tag@0, value@8 }` box (tag = source TypeId index, matching the legacy comptime interp), writing a word source's scalar via `writeField(source_type)` (so f32 round-trips) or an aggregate source's comptime ADDR (the runtime pointer-in-value-slot shape); `unbox_any` reads the value slot back (word → `readField`, aggregate → the stored ADDR). **Required making `.any` a first-class comptime aggregate** (it was `kindOf → .unsupported`): `kindOf(.any) = .aggregate` (16B, by-address) + `fieldOffset` special-cases `.any` to the `{@0, @8}` layout (shared with string/slice) — without the latter, a `struct_get` on an Any panicked (`union field 'struct' while 'any' is active`), caught + fixed (no crash; "never crash" upheld). Updated two unit tests that used `unbox_any` as the "unported op" example → now `compiler_call`; added a box→unbox round-trip test. **697/0 BOTH gates + all unit tests.** On `reify`. The 6 box_any examples (0114/0520–0524/1035) no longer bail at box_any and produce VM output byte-matching legacy, but are not YET fully HANDLED — they now fall back further at `switch_br` (comptime Any-tag type-switch), `type_name`, and `out`/print (4A.2+/later steps). **Next (4A.2):** comptime `out`/print (VM output buffer + flush). - **Phase 3 P3.4 step 8 (VM plan) — VM-native `type_info` REFLECTION → the whole metatype surface is HANDLED (2026-06-18).** Ported `type_info($T)` into the VM (`callBuiltinVm` `.type_info` arm → new `buildTypeInfo`), the inverse of step 7's `define`: reflect a type INTO a `TypeInfo` VALUE built in FLAT MEMORY (the VM-native mirror of legacy `reflectTypeInfo`). Decodes the source type into a tag + members (tagged-union/struct field & enum variant → `{ name, ty }`, a payloadless variant → `void`; tuple → bare positional `Type`s), then lays out the nested value bottom-up using layouts derived from the `TypeInfo` RESULT type (`ins.ty`, now threaded into `callBuiltinVm`): element array → `{ptr,len}` slice → info struct (`EnumInfo`/`StructInfo`/`TupleInfo`) → `TypeInfo { tag, payload }` tagged union (reusing step 7's tagged-union write). Variant/field names materialize via a `makeStringValue` helper extracted from `text_of`. Same `backing_type` guard as step 7. **Result: the ENTIRE metatype surface runs HANDLED on the VM with ZERO fallback** — `0614`–`0624` + `0632` (0616 `field_type` folds at lower time, no comptime eval); the `define(declare, type_info(T))` round-trips (`0619`/`0622`/`0623`) mint byte-identical copies on the VM. VM output byte-matches legacy for all. **697/0 BOTH gates + all unit tests.** On `reify`. **Remaining VM fallbacks in the comptime corpus are now genuinely-non-metatype** emit-time side effects: `print`/`out` (0613), `global_addr` (0600), `compiler_call` #compiler hooks (0602/0603), and the inline-asm global (1654). **Next:** port those (or confirm each is a legitimately-non-comptime case) to drive the fallback list to empty, then — with user go-ahead — flip the VM to default + delete `interp.zig`. - **Phase 3 P3.4 step 7 (VM plan) — VM-native metatype CONSTRUCTION: `declare`/`define` + tagged-union `enum_init` (2026-06-18).** Ported the metatype type-CONSTRUCTION builtins into the VM so the construction examples run HANDLED end-to-end (no `call_builtin` fallback). Three pieces: (1) **tagged-union `enum_init` with payload** — the arm previously bailed; now allocates the value (zeroed), writes the tag at offset 0 (`{ header(tag)@0, [N x i8] payload@tag_size }`, the LLVM `backend/llvm/types.zig` layout) and copies the payload at `tag_size`. (2) A **`.call_builtin` exec arm** → new `callBuiltinVm`, the VM-native mirror of the legacy `execBuiltinInner`: `declare(name)` mints an empty forward nominal slot (shared `declareNominal` helper, also used by `declare_type`); `define(handle, info)` reads the `TypeInfo` tagged-union VALUE from FLAT MEMORY (tag@0, active payload `EnumInfo`/`StructInfo`/`TupleInfo` struct at `tag_size`, its single slice field) and mints via `defineFromInfo`, a faithful port of legacy `defineEnum`/`defineStruct`/`defineTuple` (all-void enum → real `.@"enum"` per issue 0142, dup-name rejection, `updatePreservingKey` vs `replaceKeyedInfo`). (3) Refactored the `[]{name,ty}` decode out of `registerTypeVm` into a shared `decodeMemberSlice` (+ `decodeTypeSlice` for bare-`Type` tuple elements), keyed to the module-level `NamedMember`. Unmodeled builtins (`type_info`/`type_name`/…) return null → bail with the builtin name → legacy fallback (dual-path parity). **Correctness guard (caught via review):** `enum_init`/`define` assume a tag-headed layout, which is WRONG for a `backing_type` tagged union (laid out as the backing struct) — both now bail loudly on `backing_type != null` rather than silent-clobber. **Result:** examples `0614`/`0620`/`0621`/`0624`/`0632` run **fully HANDLED** on the VM (define is the whole eval); `0622`/`0623` run define HANDLED then fall back cleanly at the still-unported `type_info` reflection. VM output byte-matches legacy for all 7. **697/0 BOTH gates + all unit tests (added: tagged-union `enum_init` payload layout).** On `reify`. **Next:** port `type_info` (REFLECT a type → build a `TypeInfo` value in comptime memory, the inverse — reuses the tagged-union `enum_init` write) so `0619`/`0622`/`0623` go fully HANDLED; then the rest of the comptime corpus (drive the SX_COMPTIME_FLAT_TRACE fallback list toward the genuinely-non-comptime cases) before the VM-default flip + legacy deletion. - **Phase 3 P3.4 step 6 (VM plan) — REAL lowering-time Context: allocating + List-building type-fns now run HANDLED on the VM (2026-06-18).** The VM can now evaluate a comptime type-fn that ALLOCATES at lowering time (the 0141 family) — the legacy interp cannot. Four changes: (1) `runComptimeTypeFunc` (lower/comptime.zig) FORCES the CAllocator→Allocator thunks to exist (`getOrCreateThunks`, idempotent, guarded by Allocator/ CAllocator registered) BEFORE eval — a type-fn const runs at scanDecls (Pass 1), before Pass 1c builds the default-context global + thunks, so the comptime allocator was otherwise null; (2) `materializeDefaultContext` builds a REAL context at lowering time when the global is absent — finds the two thunks by name (`findFuncByName`) and lays their func-refs into the inline `Allocator` value `{ctx=null, alloc_fn@+ptr, dealloc_fn@+2*ptr}` at the head of `Context`, so `context.allocator.alloc_bytes` dispatches `call_indirect` → thunk → native VM `malloc`; (3) `aggType` now DEREFS a pointer `base_type` (the List write path emits `struct_gep` with `base_type = *Struct` — `fieldOffset` panicked on the pointer; now derefs to the pointee, no panic); (4) `subslice` handles a `[*]T` many-pointer / `*T` base (a List's `items` field — the base IS the data pointer). **Verified end-to-end (manual probe):** a compiler-API type-fn that builds its `[]Member` in a `List(Member)` (`.append` ×3, then `register_type(handle, kind, vs.items[0..vs.len])`) runs **HANDLED on the VM** and mints correctly (`green=7`) — the exact 0141 List-growth pattern, on the VM. **Can't be a corpus test yet** (gate-OFF/legacy still can't allocate at lowering time — the dual-path bind), so locked in via VM unit tests instead (many-pointer subslice; `struct_gep` with a pointer `base_type`). **697/0 BOTH gates + all unit tests, EXIT=0.** On `reify`. **Remaining for the original 0141 repro (uses metatype `define`/ `make_enum` → `call_builtin` → legacy fallback → legacy fails):** re-express the metatype over the compiler-API so the whole type-fn runs on the VM (no `call_builtin`). THEN the repro works on the VM — and the dual-path bind resolves only at the VM-default-flip + legacy-deletion end-state. - **Phase 3 P3.4 — investigation: the "real lowering-time Context" is BLOCKED by issue 0141 (2026-06-18).** Probed whether the VM needs a REAL lowering-time `Context` (CAllocator thunk func-refs) for allocating type-fns. **Finding: lowering-time comptime ALLOCATION fails in the LEGACY interp too** — a type-fn that calls `context.allocator.alloc_bytes` at lowering time bails in legacy with `comptime call_indirect: callee is not a func_ref Value (raw fn-pointers from extern calls aren't dispatchable in interp)`, and the VM bails at parity (`call_indirect through a null function pointer`). This is exactly issue **0141**'s root cause (its analysis already notes "the null allocator is the same story for the CAllocator thunks") — an OPEN deferred issue. So: (1) the VM is CORRECT (parity — both bail; no regression); (2) the real-context work is PREMATURE — its only consumer (allocating lowering-time type-fns) can't pass gate-OFF, so no corpus test can validate it, and even a more-capable VM can't ship a divergence during the dual-path phase. **Consequence for the metatype re-expression:** re-expressing `define`/`make_enum` over the compiler-API needs to BUILD `[]Member` slices dynamically (allocation) — which is blocked by 0141 at lowering time. The viable paths are: (a) avoid allocation by passing the caller's existing slice through (needs `EnumVariant`/`StructField` to be usable AS `Member` — they're layout-identical `{string, Type}`, but distinct nominal types — a metatype-API decision), or (b) wait for 0141. **No code change this step** (the VM already bails correctly). Recorded so the next session doesn't re-derive it. 697/0 both gates unchanged. - **Phase 3 P3.4 step 5 (VM plan) — WRITE side ported to the VM → FIRST HANDLED lowering-time type-fns (2026-06-18).** Ported `declare_type` / `pointer_to` / `register_type` into `Vm.callCompilerFn`, mirroring the legacy `compiler_lib` handlers (mint via `@constCast(table)` — the same mutable access the read-side `intern` uses; the lowering-time mint target IS `&module.types`). `register_type` reads the `[]Member` slice from FLAT MEMORY: threaded `ref_types` through `invoke` → `callCompilerFn` so the slice's element type (`Member = {name: string, ty: Type}`) gives the field offsets + stride; decodes each `{name, ty}` and branches on `kind` (1 struct · 2 enum · 3 tagged_union · 4 tuple) exactly as legacy (dup-name / payload-on-enum rejections, idempotent re-fill via `nominalIdentOf`). **Key unblock:** the synthesized comptime type-fn wrapper (`createComptimeFunction`/`…WithPrelude`) was built with return type `.any` → `regToValue` bailed at the VM↔legacy boundary; changed to `.type_value` (the legacy path reads via `asTypeId` regardless, so no legacy change). **Result: the compiler-API write type-fns now run HANDLED end-to-end on the VM at LOWERING time** — `0631` (register-graph: 2 HANDLED, A↔B cycle via forward handles + `pointer_to`) and `0635` (multi-edge import: 2 HANDLED), parity-correct. They run on the ZEROED lowering-time context (fixed `.[…]` member arrays, no allocation). The metatype `make_enum`/`define` examples (`0632`) still fall back CLEANLY through `call_builtin(define)` (the separate metatype path — re-expressing it onto the compiler-API is the other half of P3.4). **697/0 BOTH gates + EXIT=0.** On `reify`. **Next:** (optional, deferred) a REAL lowering-time Context (CAllocator thunk func-refs) for List-growing type-fns; and re-express the metatype `define`/`make_enum` over the compiler-API to delete the bespoke interp arms (the end-state: ONE evaluator). - **Phase 3 P3.4 step 4 (VM plan) — model `.type_value` natively in the comptime VM (2026-06-18).** The VM now HANDLES Type values instead of bailing: `kindOf(.type_value)` → `.word`; a new `const_type` exec arm → the word `TypeId.index()`; `regToValue` maps a `.type_value` word back to a `.type_tag` Value at the legacy boundary (`valueToReg` already mapped `.type_tag` → index). Surfaced + fixed a VM PANIC (forbidden): `struct_init` assumed a `.@"struct"` result type and union-access-panicked on an ARRAY literal (`EnumVariant.[ … ]`, reached now that Type args no longer bail early) — it's the generic aggregate-literal op, so it now dispatches on the result kind (struct / array / tuple) and BAILS loudly on anything else, never panics. **697/0 both gates** (the make_enum type-fns now run further on the VM, then bail cleanly at the `define`/`make_enum` `call_builtin` → legacy mints — no mutation before the bail, parity holds). VM unit test added (const_type → word → regToValue → `.type_tag`). On `reify`. **Next (the payoff):** port the WRITE side (declare_type / register_type / pointer_to) into `Vm.callCompilerFn` + give the lowering-time path a REAL Context (CAllocator thunk func-refs, not zeroed) → the first HANDLED lowering-time type-fn end-to-end on the VM. - **Phase 3 P3.4 step 3 (VM plan) — dedicated `Type` builtin TypeId: RESOLVER FLIPPED + `.any` migration (2026-06-18).** Flipped `type_resolver:64` (`"Type"` → `.type_value`), `module.zig` `constType` (result type → `.type_value`), and `emitConstType` (a bare i64 carrying `tid.index()`, NOT a 16-byte Any box). Then migrated every `.any` reference that means "a Type value", classified per CLAUDE.md (leave the real boxed-Any refs): (a) the "Any holds a Type" **meta-marker tag** moved `.any` → `.type_value` at all 4 consumers — `reflectArgTypeId` (LLVM), `reflectTypeId` + the `.type_tag`-as-struct-field comptime path (interp), and `resolveTypeCategoryTags("type")` (generic.zig); (b) reflection-builtin RETURN types `.any` → `.type_value` (`type_of`/`declare`/ `define`); the runtime `type_of(any)` now reads the tag AS a `.type_value` (no re-box); (c) expr_typer infers a bare type-name expr as `.type_value` (with a `is_raw` backtick exemption — `` `string `` is a value, never the reserved type); (d) `reflectionArgIsType` accepts `.type_value` OR `.any` (a reflection arg can be a bare Type OR a boxed Any — the over-narrow `==.type_value` was the catastrophic-regression cause, caught + fixed); (e) the comptime `switch_br` accepts a `.type_tag` discriminant (type-category match); (f) a bare function name in a `Type` slot now lowers to `const_type(its real function type)` instead of a func-ref (fixed a JIT crash — was a func-ref word read as a TypeId), keeping the old string-box path only for genuine `Any` params; (g) the field-not-found diagnostic + `formatTypeName` render `.type_value` as "Type". Fixed 3 unit tests asserting the old `.any` Type behavior. **697/0 BOTH gates** + all 494 unit tests (EXIT=0). Gate ON stays green because the VM's `kindOf(.type_value)` → `.unsupported` → bails CLEANLY to legacy (no silent-wrong) — the VM doesn't model `Type` values YET (next step), but parity holds. Regenerated 24 snapshots (22 `.ir` const_type-shape; 2 `.stderr` Any→Type — diff reviewed, only the intended changes). On `reify`. **Next:** model `.type_value` natively in the VM (`kindOf` → word, `const_type` → word = `TypeId.index()`, `regToValue` word → `.type_tag`) for COVERAGE, then port the WRITE side into `callCompilerFn` + a real lowering-time Context → the first HANDLED lowering-time type-fn. - **Phase 3 P3.4 step 2 (VM plan) — dedicated `Type` builtin TypeId: FOUNDATION landed (dead/additive) (2026-06-18).** Added `TypeId.type_value` (slot 19) + a matching `TypeInfo.type_value` variant + the builtins init entry — an **8-byte type handle distinct from the 16-byte boxed `.any`** (THE WALL). All `types.zig` layout handlers wired: `sizeOf`/`typeSizeBytes` → 8, `typeAlignBytes` → 8, `typeName` → "Type", `hashTypeInfo`/`typeInfoEql` no-payload arms. Only ONE exhaustive switch needed a new arm (`backend/llvm/types.zig` `toLLVMTypeInfo` → `cached_i64`); every other `switch(TypeInfo)` site has an `else` (audited when the resolver flips). **`first_user` 19 → 100** (per the user): slots 20–99 are RESERVED builtin headroom (infos padded with the `unresolved` tripwire), so future builtins don't renumber user TypeIds / churn `sx ir` snapshots. Cost: ~80 default entries in each binary's per-type reflection arrays (user opted in). **Still dead:** `type_resolver.zig:64` STILL returns `.any` for "Type" — nothing produces `.type_value` yet, so NO behavior change. Regenerated 22 IR snapshots (pure TypeId renumber to 100-base; `git diff --name-only` confirmed ONLY `.ir` files + the 2 source files changed — no stdout/stderr/exit). **697/0 both gates** (OFF and `-Dcomptime-flat`). **Next:** flip `type_resolver:64` → `.type_value`, then migrate the `.any` refs that mean "a Type value" (const_type result / reflection returns / metatype `Type` params / `.type_tag` checks) — leave the real boxed-Any refs — file-by-file with a build after each. - **Phase 3 P3.4 step 1 (VM plan) — lowering-time default context; first blocker cleared (2026-06-18).** `materializeDefaultContext` now falls back to a ZEROED `Context` (found by name) when the `__sx_default_context` global is absent — i.e. at LOWERING time, where the global isn't emitted yet. A type-fn that never touches the allocator now runs past context setup; one that allocates reads a null `alloc_fn` (zeroed) → `call_indirect` on the null func-ref bails → legacy fallback (a REAL lowering-time context with the CAllocator thunk func-refs, so allocating type-fns also run on the VM, is a follow-up). **Measurement: the bail moved deeper** — metatype `make_enum` now bails at `const_type` (the `Type`-literal op, unported); `register_type` type-fns bail at the welded write call (declare_type/register_type aren't in `callCompilerFn`). No table mutation happens before either bail (the write fns bail before minting), so parity holds: both gates **697/0**, no crashes. **Next blockers (the "model Type" chunk):** (a) the `const_type` op → a word = `TypeId.index()`; (b) the Type-return bridge (`regToValue` for a `Type`/`.any` word → `.type_tag`); (c) the VM-native write side (declare_type/register_type/pointer_to in `callCompilerFn`) + a real lowering-time context. Only once those land does a type-fn actually run end-to-end on the VM (a HANDLED case). - **Phase 3 P3.4 (VM plan) — wire the VM at the LOWERING-time site + measure (2026-06-18).** Routed `runComptimeTypeFunc` (the type-fn fold — the THIRD comptime call site) through `comptime_vm.tryEval` behind `-Dcomptime-flat`/`SX_COMPTIME_FLAT` with legacy fallback, mirroring the two emit-time folds. Extracted the shared post-check (`checkComptimeTypeResult` — the declared-but-never-defined zero-field guard) so both paths use it. **Measurement (SX_COMPTIME_FLAT_TRACE):** every metatype/compiler-API type-fn currently bails CLEANLY with `no __sx_default_context global to materialize the implicit context` — at lowering time the default-context global doesn't exist yet (it's built at emit time), so the VM bails at context materialization, BEFORE running the body (no partial mint, no crash → legacy mints). The hardening holds: **no crashes** across the corpus on the VM lowering-time path. Both gates **697/0**. **So the FIRST lowering-time blocker is the implicit context, not `Type` modeling** — the VM needs a way to materialize/skip the default context at lowering time (most type-fns get an implicit ctx for potential `List`-growth alloc; many don't use it). Next: materialize a lowering-time default context for the VM (or pass a null ctx + bail only if the allocator is actually used), THEN model `Type` values + the VM-native write side. This is near-pure fallback today — permanent scaffolding that lights up as those land. - **Phase 3 P3.4-prep (VM plan) — harden the VM against malformed lowering-time IR (2026-06-18).** Prerequisite for wiring the VM at the LOWERING-time comptime site (`runComptimeTypeFunc`), where IR can be malformed (an unresolved name lowers to a dangling / `Ref.none` operand — the 0737 crash). Closed the remaining panic vectors so the VM BAILS (→ legacy fallback) instead of aborting: (1) a checked `Vm.refTy(ref_types, r)` replaces every raw `ref_types[ref.index()]` in `exec` (the type-side companion to `Frame.get`'s `bad_ref` value-side guard); (2) `aggType` is now a bailing method (`Error!TypeId`) using `refTy`; (3) the block-dispatch loop bounds-checks the branch target before indexing `func.blocks.items`. `global_get` was already guarded. No behavior change — gate OFF and ON both **697/0**; unit test added (a `cmp_lt` with a `Ref.none` operand bails, not panics). **Next:** wire `tryEval` into `runComptimeTypeFunc` behind the flag with legacy fallback and measure (most minting type-fns will still bail at the welded-write call / `Type`-result conversion until the VM models `Type` values + the VM-native write side land — those are the steps that actually move lowering-time comptime onto the VM, toward deleting legacy). - **Phase 3 P3.3 (VM plan) — WRITE side: declare_type + pointer_to + ONE kind-branching register_type (2026-06-18).** The mutating compiler-API: `declare_type(name) -> Type` (forward handle), `pointer_to(t) -> Type` (build `*T`), and `register_type(handle, kind, members: []Member) -> Type` which branches on `kind` IN THE COMPILER (subsuming define's per-kind dispatch). Take/return real `Type` values (matching meta.sx declare/define). **Timing (per user): mint LAZILY at lowering time, single pass** (the existing `runComptimeTypeFunc`), so the write side is **legacy-only** (`compiler_lib` handlers) — the VM isn't wired at lowering time, no VM mirror needed; readers stay dual-path. A non-generic `-> Type` builder is now flagged `is_comptime` (decl.zig) so its dead body permits the welded calls. **Graph:** forward handles + `pointer_to` express mutually-recursive A↔B (`*A`, `*B`, B-by-value); `register_type` is **idempotent** (re-fill a nominal slot reached via two import edges — `nominalIdent`). `kind` codes match `type_kind` (1 struct · 2 actual `.@"enum"` · 3 tagged_union · 4 tuple). **Fixed two bugs (issue 0142):** (a) a fully payloadless minted enum was an all-void tagged_union → verifySizes panic; now a real `.@"enum"` (register_type kind 2 AND metatype `defineEnum`); (b) bare `EnumType.variant` payloadless qualified construction wasn't supported (failed for hand-written enums too) — added in `lowerFieldAccess` (`isPayloadlessVariant`). Examples 0631 (graph + actual enum + reflection), 0632 (make_enum all-void), 0633/0634/0635 (namespaced / bare / multi-edge import of a minted type), 0187 (qualified variant construction). **Parity 697/697** (gate ON and OFF); unit tests added. **Next (P3.4):** re-express declare/define/type_info as sx over the compiler-API + delete the bespoke interp arms (needs the VM hardened for lowering-time IR, or the metatype migrated onto the legacy compiler-API calls). - **Phase 3 P3.2b (VM plan) — kind + enum-value readers: `type_kind` + `type_field_value`; READ side complete (2026-06-18).** The last two read-only readers the metatype's `type_info(T)` needs (added to `compiler_lib.bound_fns` AND `Vm.callCompilerFn`, each backed by a `TypeTable` query both call): `type_kind(t) -> i64` (`kindCode` — a stable, compiler-owned discriminant: 0 other · 1 struct · 2 enum · 3 tagged_union · 4 tuple · 5 union · 6 array · 7 vector · 8 error_set; TOTAL, never bails) and `type_field_value(t, idx) -> i64` (`memberValue` — an enum variant's explicit value or ordinal; mirrors the `field_value_int` builtin; loud-bail for non-enum / out-of-range). Example `0630-comptime-compiler-type-kind` reflects `Color` / `WindowFlags` (flags) / `Point`. **The READ side is now COMPLETE** — `find_type` + `type_kind` + `type_field_count` + `type_field_name`/`type_field_type`/`type_nominal_name` + `type_field_value` cover everything `reflectTypeInfo` reads. VM unit test added. **Parity 691/691** (gate ON and OFF). **Revised forward direction (per the user):** the WRITE side is ONE `register_type(info)` fn that branches on the kind IN THE COMPILER (subsuming `define`'s per-kind dispatch), not a per-kind `register_struct`. - **Phase 3 P3.2 (VM plan) — field-level reflection readers: `type_nominal_name` + `type_field_name` + `type_field_type` (2026-06-18).** Three more `compiler`-library readers on the same `TypeId`-handle shape (added to `compiler_lib.bound_fns` AND `Vm.callCompilerFn`), each backed by a new `TypeTable` query BOTH paths call (no drift): `nominalName` (a named type's own name handle; loud-bail for unnamed types like `i64`/pointers), `memberName` (struct/union/tagged-union field, enum variant, named-tuple element), `memberType` (struct/tuple/array/vector member type). All loud-bail on out-of-range idx / no-member (no silent default). First MULTI-ARG compiler fns — `callCompilerFn` reads arg 1 = idx; added `Vm.argHandle`/`argTypeId` (range-checked u32/TypeId arg reads) and refactored `find_type`/`type_field_count` onto them. Named `type_*` to avoid clashing with the std metatype builtins (`field_name`/`type_name` exist in core.sx); `nominalName` (the TypeTable method) is distinct from the existing `typeName(id) []const u8` display-string renderer. Example `0629-comptime-compiler-field-reflect` reflects `Pair { lo: Point; hi: Point }` — each field name + the nominal name of a field's type, all `#run`-folded, all VM-HANDLED natively. VM unit test added (type_field_name → "hi"; type_nominal_name(type_field_type(Pair,0)) → "Point"). **Parity 690/690** (gate ON and OFF). - **Phase 3 P3.1 (VM plan) — first read-only reflection readers: `find_type` + `type_field_count` (2026-06-18).** Two more `compiler`-library fns, bound the same way as the `intern`/`text_of` seed (added to `compiler_lib.bound_fns` for the legacy handler + the welded-decl export check, AND to `Vm.callCompilerFn` for the native comptime path — NO marshaling). A **type handle is a plain `u32` `TypeId`** (like `StringId`), so both keep the seed's clean scalar shape: `find_type(name: StringId) -> TypeId` (`TypeTable.findByName`, `unresolved`/0 if absent) and `type_field_count(t: TypeId) -> i64` (a NEW `TypeTable.memberCount` query — struct/union/ tagged-union fields, enum variants, array/vector length — called by BOTH paths so they can't drift; bails loudly, never a silent 0). New example `0628-comptime-compiler-find-type` chains `intern → find_type → type_field_count` (and a not-found lookup → 0), both folded at `#run`, both VM-HANDLED natively (trace confirms no fallback). VM unit test added (`find_type` + `type_field_count`, struct found → 3 fields, missing → `unresolved`). **Parity 689/689** (gate ON and OFF). **Decision (resolves the plan's `find_type → ?Type` sketch):** return a NON-optional `TypeId` with the `unresolved` (0) sentinel for not-found, NOT `?Type` — a `Type` value resolves to `.any` (which the comptime VM doesn't represent) and an optional can't cross the legacy↔VM eval boundary; `unresolved` is the project-blessed unmistakable "no type" marker. Forward (P3.2): more readers on the same handle shape (`type_name`/`field_name`/`field_type`/kind), then `register_struct` (first mutating fn). - **VM robustness — `Frame` bounds-check; lowering-time `#insert` wiring explored + reverted (2026-06-18).** Explored wiring the VM at the LOWERING-time comptime site (`evalComptimeString`, the `#insert` string fold). 12/13 `#insert` examples ran on the VM with parity, but `0737` (an `#insert` of an unresolved `secret()`) CRASHED the VM (SIGABRT): lowering-time IR can be malformed (a `ret Ref.none` from the unresolved name) and `Frame.get` panicked on the out-of-range index. **Decision: reverted the lowering-time wiring** — unlike the emit-time folds (fully lowered IR), lowering-time IR can be erroneous, and hardening the VM against ALL malformed IR (every `ref_types[...]` / `aggType` access, not just `Frame`) is out of scope here. The emit-time sites already give full corpus coverage. **KEPT** the defensive fix regardless (CLAUDE.md "never crash"): `Frame.get`/`set` now bounds-check and flip a `bad_ref` flag; the `run` loop bails (`badRef`) instead of panicking. Unit test added (malformed `ret Ref.none` → bail, not crash). Parity **688/688** both ways. - **Phase 3 SEED (VM plan) — compiler-call path: `intern`/`text_of` native on the VM (2026-06-18).** `invoke` now dispatches a welded `compiler`-library fn (gated on `compiler_welded`) to `Vm.callCompilerFn`, serviced NATIVELY on comptime memory (no legacy `Interpreter`): `intern(string)->StringId` reads the comptime string bytes and `internString`s into the const-cast table (pool-only — doesn't touch type layout, so cached sizes stay valid); `text_of(StringId)->string` materializes the pooled text back into comptime memory. Unlocked `0626`; the ONLY remaining const-init fallback is now the inline-asm global (`1654`). Parity **688/688** (gate ON and OFF); unit test added. This is the mechanism Phase 3 grows — the next compiler fns (`find_type`, `register_struct`, reflection readers) bind the same way (comptime pointer in, handle/pointer out, no marshaling). - **Phase 1.final step 9 (VM plan) — `-Dcomptime-flat` build flag (the "swap behind a build flag" step) (2026-06-18).** Added the `-Dcomptime-flat` build option (build.zig → a `build_opts` options module on `mod`; `emit_llvm.init` reads `build_opts.comptime_flat or SX_COMPTIME_FLAT env`). This is the plan's "reach parity → swap behind a build flag → delete the old path" mechanism. `zig build test -Dcomptime-flat` runs the FULL corpus on the VM (688/0). Verified the flag toggles the binary: flag-built `sx` reports VM HANDLED with no env var; default-built does not. Default OFF — `zig build test` unchanged (688/0). Env var still works for ad-hoc runs. Next (forward): Phase 2 (bytecode) / Phase 3 (compiler-API on comptime memory); eventual default-flip + legacy deletion. - **Phase 1.final step 8 (VM plan) — wire the `#run` side-effect path + trace-clear-on-fallback (2026-06-18).** Wired the SECOND comptime call site (`runComptimeSideEffects`, top-level `#run ;`) through `tryEval` with legacy fallback, mirroring the const-init fold. `tryEval` now handles void/noreturn entries (→ `.void_val`) so a void side-effect doesn't bail at the result conversion. **Fixed a trace-corruption** the new site exposed (`1035`): a side-effect that pushes return-trace frames and then bails (e.g. on `print`) had the legacy re-run DOUBLE-push them (`sx_trace_push` is a side effect on the shared buffer). Both wiring sites now `sx_trace_clear()` right before the legacy fallback, discarding the VM's partial pushes. **Parity 688/688** (gate ON and OFF). Most side-effects still bail (print/global_addr/call_builtin) → legacy, but the path is now uniform. All comptime evaluation routes through the VM-with-fallback. - **Phase 1.final step 7 (VM plan) — is_comptime + failable/error cluster + signed-load fix; coverage 31→36 (2026-06-18).** `is_comptime` → 1 (unlocked `1030`). Ported the failable/error-channel cluster (`1037` escape, `1038` handled): `kindOf(error_set)→word`, `regToValue` bridges TUPLES (the failable `(value…,tag)` shape `checkComptimeFailable` reads), `trace_frame` packs `(func_id<<32|span.start)` from a new `call_stack` (pushed by invoke/runEntry), and `sx_trace_push`/`sx_trace_clear` serviced NATIVELY (the VM calls the real sx_trace.c functions linked into the compiler, so the return-trace buffer is populated identically to legacy). raise/catch/or now run on the VM. **Surfaced + fixed a real GENERAL bug:** `readField` was ZERO-extending signed sub-64-bit loads, so a stored `i32 -1` reloaded as `0xFFFFFFFF` (+4.29e9) and `< 0` was false — silently hiding `raise error.Bad`; now SIGN-extends `i8`/`i16`/`i32`/`isize` (gate-ON parity confirms it's a strict fix; unit test added). VM HANDLES **36** corpus const-inits (was 31); **parity 688/688** (gate ON and OFF). Only **2 fallbacks** remain, both principled: `intern` (`0626`, welded compiler-API fn — Phase 3) + inline-asm global (`1654`). Forward work: Phase 2 (bytecode), Phase 3 (compiler-API on comptime memory). - **Phase 1.final step 6 (VM plan) — real default context + call_indirect + func_ref + global_get; coverage 27→31 (2026-06-17).** Per the user's direction ("the VM can set up a default context"), `runEntry` now materializes the REAL default context instead of a zeroed one. The implicit-ctx param is an opaque `*void`, so `materializeDefaultContext` finds the `__sx_default_context` global and lays its initializer (`{ {null, alloc_fn, dealloc_fn}, null }`, the CAllocator thunk func-refs) into comptime memory via a new recursive `layoutConst`. With `func_ref` (function value encoded `FuncId.index()+1`, reserving word 0 for the null fn-ptr) and `call_indirect` (decode word → FuncId → dispatch; 0 → bail) ported, the whole allocator protocol runs on the VM: `context.allocator.alloc_bytes` → call_indirect → thunk → `CAllocator.alloc_bytes` → `libc_malloc` → native comptime malloc. Unlocked `0606` (string global). Also: `global_get` lazily evaluates a comptime global's `comptime_func` (memoized) — unlocked `CT_CHAIN`; field access (`fieldOffset`/`struct_get`) handles string/slice `{ptr@0,len@8}` fat pointers (needed by `alloc_string`); `regToValue` maps function-typed words → `.func_ref` (kept `1128`'s rejection byte-identical). Native `malloc` is still required (the thunk bottoms out at it; a host pointer can't be used with comptime load/store). VM HANDLES **31** corpus const-inits (was 27); **parity 688/688** (gate ON and OFF). Unit tests: global_get, func_ref+call_indirect. Remaining fallbacks (7): `.unsupported` aggregates (3× — `1037`/`1038`), extern/builtin `intern`+asm (2×), `trace_frame`, `is_comptime`. - **Phase 1.final step 5 cont. (VM plan) — libc memory builtins + f32 fix; coverage 16→27 (2026-06-17).** Identified the dominant fallback (`call to extern/builtin`) as **11× `malloc`** (0604) + 1× `intern`. Modeled a curated set of libc MEMORY builtins natively on comptime memory (`Vm.callMemBuiltin`): `malloc`/`calloc` → `allocBytes` (16-aligned, 256-MiB cap → bail), `free` → no-op, `memcpy`/`memmove`/`memset` on comptime bytes — sandboxed (no host heap/dlsym), target-aware; the computed result is byte-identical to legacy (which calls real libc). This surfaced a **real latent f32 bug**: float registers hold f64 bits, but f32 MEMORY is the 4-byte single — `readField`/`writeField` were truncating the f64 bits (writing zeros for `1.0`); now they `@floatCast` on f32 load/store (mirrors legacy `storeAtRawPtr`). Result: VM HANDLES **27** corpus const-inits (was 16); **parity 688/688** (gate ON and OFF). Unit tests added (f32 round-trip; malloc → usable comptime memory). Next: the `kindOf` `.unsupported` aggregates (3×), `global_get` (2×), the rest. - **Phase 1.final step 5 (VM plan) — implicit-context materialization; coverage 0→16 (2026-06-17).** `tryEval` now MATERIALIZES the implicit ctx instead of skipping it: a `has_implicit_ctx` comptime entry (sole param `*Context`) gets a zeroed `Context` of the right size/align in comptime memory, its address passed as arg 0. Const bodies that ignore the ctx run; a body that uses the allocator hits unported `call_indirect` → bails → legacy. No func-ref materialization needed (handled bodies don't read ctx contents; parity is the guard). Fixed a real bug surfaced by the coverage pass: storing a `null` non-pointer optional (the `null_addr` sentinel) into an aggregate slot OOB-bailed — `writeField` now ZEROES the destination for a `null_addr` aggregate source (= none/empty); unit-test regression added. Result: VM HANDLES **16** corpus const-inits (was 0); **parity 688/688 both gate ON and OFF**. Next: port the ops the trace names — `call_builtin`/`compiler_call`/ extern (13×, via the bridge), `kindOf` `.unsupported` aggregates (3×), `global_get` (2×), func_ref / call_indirect / trace_frame / is_comptime. - **Phase 1.final steps 1–4 (VM plan) — host wiring landed; coverage measured (2026-06-17).** (1) **Hardening:** `Machine.readWord`/`writeWord`/`bytes` now return `error.OutOfBounds` (null / out-of-range / oversized / overflow-safe) instead of `assert`-panicking; `OutOfBounds` added to `Vm.Error`; `try` threaded through every helper + exec arm + the bridge. New unit tests (accessor OOB returns; null-deref → `tryEval` null, not a crash). (2) **Implicit context:** `tryEval` returns null for `has_implicit_ctx` funcs (legacy fallback) — conservative; full ctx materialization deferred to step 5. (3) **Wiring:** const-init fold in `emit_llvm.zig` `emitGlobals` is `(if comptime_flat) tryEval else null) orelse interp.call(...)`, gated by env `SX_COMPTIME_FLAT` (read once into `LLVMEmitter.comptime_flat`). Default OFF. (4) **Parity + coverage:** gate ON → full corpus byte-identical (688, 0 failed) + manual 0605/0606/0607 byte-identical. **Finding: 0 of 37 measured corpus const-inits are VM-handled — ALL are `has_implicit_ctx`-gated.** Added a coverage-trace facility (`comptime_vm.last_bail_reason` + env `SX_COMPTIME_FLAT_TRACE`). **Next: step 5 = implicit-context materialization** (the unblocker), then port the deferred ops. 688 corpus green (gate OFF). - **Phase 1.final start (VM plan) — wiring entry point `tryEval` (2026-06-17).** `comptime_vm.tryEval(gpa, module, func_id) ?Value` runs a comptime function entirely on the VM, returns a legacy `Value` (deep-copied to `gpa`) or `null` to fall back. Unit-tested (pure 6*7 → 42; unbox_any → null). NOT yet routed into the host: needs (1) panic→error hardening of `Machine` accessors so arbitrary funcs bail instead of crashing, (2) implicit-ctx handling, (3) wiring at `emit_llvm` const-init behind `SX_COMPTIME_FLAT`, (4) corpus parity run. See `PLAN-COMPILER-VM.md` Phase 1.final. 688 corpus green. - **Phase 1 sub-step 1.5b (VM plan) — Reg↔Value boundary bridge (2026-06-17).** Builtin/compiler_call/extern handlers are coupled to the legacy `Interpreter`, so the wiring will use WHOLE-FUNCTION fallback (VM runs pure functions; bail → legacy re-runs the whole eval). Built the boundary bridge that enables it: `valueToReg` (Value arg → Reg, aggregates into comptime memory) + `regToValue` (VM result → Value, deep-copied). Covers scalars/strings/structs; other shapes bail. Transitional. Round-trip unit-tested. 688 corpus green. Next: the wiring (flag + route a comptime entry through the VM with legacy fallback). - **Phase 1 sub-step 1.5 (VM plan) — direct `call` + stack-lifetime change (2026-06-17).** `Vm` gained `module` (callee resolution) + `depth`/`max_depth` guard. `call` marshals arg Refs → Reg and recursively runs the callee; aggregates pass as Addrs over shared comptime memory. `Frame` no longer reclaims the machine on exit (else a returned aggregate Addr dangles) — allocations live to `Vm.deinit`. Extern/builtin callees bail (1.5b). Unit-tested: direct call (142), recursion sum(0..n) (15/55). 688 corpus green. Next: 1.5b (call_builtin/compiler_call/extern), then hybrid wiring. - **Phase 1 sub-step 4d (VM plan) — deref/addr_of; pivot decision (2026-06-17).** Ported `addr_of` (pass-through) + `deref` (readField through pointer), unit-tested (deref *i64 → 77, addr_of struct + field → 80). DECIDED to stop porting rarer ops (tagged-union payload/any/closures) blind — their byte semantics are ambiguous without real call sites — and pivot to CALLS (sub-step 1.5: `call`, then builtin/compiler) + HYBRID WIRING (`-Dcomptime-flat` → VM with legacy fallback on `error.Unsupported`), so the VM runs the real corpus and surfaces exactly what's needed. Key design point for calls: aggregate-return lifetime → drop per-frame stack reclaim (let a comptime eval's allocations live to `Vm.deinit`). 688 corpus green. See `PLAN-COMPILER-VM.md` decision block. - **Phase 1 sub-step 4c (VM plan) — optionals + payloadless enums (2026-06-17).** `kindOf`: enum → word; `?T` → word (pointer-child, null==0) or `{T@0,i1@sizeof(T)}` aggregate. Ported optional_wrap/unwrap/has_value/coalesce (`optChildIsPtr`/`optHas`; const_null reads as none) + payloadless enum_init/enum_tag. Unit-tested (?i64 → 91, ?*i64 null==0 → 99, enum tag → 11). 688 corpus green. Next: 4d (tagged unions, any, closures). - **Phase 1 sub-step 4b (VM plan) — slices + strings on comptime memory (2026-06-17).** `{ptr@0(pointer_size), len@8(i64)}` fat pointers (kindOf: string/slice → aggregate). Ported `const_string` (text+NUL + fat pointer in comptime memory), `length`/`data_ptr`, `array_to_slice`, `subslice`, index-through-slice (`elemAddr` loads `.ptr`), and `str_eq`/`str_ne` (memcmp). Unit-tested (str length+eq/ne, array→slice index sum=23, subslice sum=43). 688 corpus green. Next: 4c (optionals/enums/any/closures). - **Phase 1 sub-step 4a (VM plan) — tuples + arrays on comptime memory (2026-06-17).** `kindOf` widened (tuple/array → aggregate). Ported `tuple_init`/`tuple_get` (`tupleFieldOffset`), `index_get`/`index_gep` (`elemAddr` = base + idx*elem_size over array/pointer/many_pointer; slice/string bases bail), `length` on array values. Unit-tested (mixed tuple, [3]i64 index sum=42, length=3). 688 corpus green. Next: sub-step 4b (slices/strings, then optionals/enums/any/closures). - **Phase 1 sub-step 3 (VM plan) — memory + structs on comptime memory (2026-06-17).** `Vm` gained optional `table: *const TypeTable` (target-aware layout). Ported `alloca`/`load`/`store` + `struct_init`/`struct_get`/`struct_gep`, laying structs out at the table's natural offsets. Value model: scalar/pointer → register word; struct → lives in comptime memory, its value IS its address (read→addr, write→memcpy), so nested structs compose and `struct_gep` = base+offset. `kindOf` bails loudly on not-yet-ported types. Addr-based values survive allocator realloc. Unit-tested (struct round-trip, alloca+gep+store+load, nested struct). 688 corpus green. Next: sub-step 4 (arrays/slices/strings/optionals/enums/tuples/any/closures, then calls). - **Phase 1 sub-step 2 (VM plan) — comptime executor: scalars + control flow (2026-06-17).** Added `Vm` to `comptime_vm.zig`: walks the same IR `Inst` over comptime frames (register `Reg` = scalar bits or `Addr`), mirroring the legacy interp's scalar semantics (i64 wrapping/signed, f64). Ported constants, arithmetic, comparison, logical, conversions, terminators (`br`/`cond_br`/`ret`/`ret_void`) and `block_param`; every other op bails loudly (`error.Unsupported` + op name in `detail`). Unit-tested on hand-built tiny IR (`Fb` builder): int add, f64 arithmetic, cond_br selection, a block-param loop, div-by-zero + unsupported-op bails. Corpus untouched (688 green). Next: sub-step 3 (memory + aggregates on comptime memory, where target-aware layout enters). - **Phase 1 sub-step 1 (VM plan) — comptime machine substrate (2026-06-17).** New `src/ir/comptime_vm.zig`: `Machine` (linear byte memory + bump/stack allocator with `mark`/`reset`, scalar `readWord`/`writeWord` 1/2/4/8 LE, `bytes` views, addr 0 reserved as `null_addr`) + `Frame` (Ref-indexed register file, stack reclamation on deinit). `Reg` = raw u64 (immediate scalar OR `Addr`). Unit-tested (`comptime_vm.test.zig`), registered in the barrel; standalone — the legacy interpreter stays live, corpus untouched (688 green). Next: sub-step 2 (executor + scalar/branch ops over the same IR). Also removed the "~500 lines / split step" rule from CLAUDE.md per request. - **Phase 0 (VM plan) — struct-weld stripped; `intern`/`text_of` bridge kept (2026-06-17).** Removed the struct-weld registry from `compiler_lib.zig` (`weldStruct`/`bound_types`/`BoundType`/`FieldLayout`/`findType`/`SxField`/ `LayoutMismatch`/`validateStructLayout`), `validateWeldedStruct`/`weldedFieldOrderStr` + the `sd.abi == .zig` call from `nominal.zig`, the struct-weld unit tests, and examples `0625`/`0627`/`1183`/`1186`. KEPT (decision) the `intern`/`text_of` function host-call bridge — a clean scalar dispatch, not weld/serialize/marshal, the Phase-3 compiler-call seed — so `weldedCompilerFn`, the `compiler_welded` dispatch, the `emitCall` comptime-only gate, the `#library`/`abi`/`extern` syntax, and examples `0626`/`1184`/`1185` remain. `zig build test` green (688 corpus, 0 failed). Next: Phase 1 (byte-addressable value model) per `PLAN-COMPILER-VM.md`. - **DIRECTION CHANGE — pivot off the byte-weld to a byte-addressable bytecode VM (2026-06-17).** Decided the weld + serialization/marshaling bridge is the wrong direction (it hand-marshals onto a comptime value model that isn't bytes — exactly what the design set out to kill). New foundation: a bytecode VM over comptime memory so comptime values are native bytes; the compiler-API then rides on it via direct memory (no weld/validation/marshaling). JIT-native comptime was weighed and rejected (breaks cross-compilation, loses the sandbox). Wrote `current/PLAN-COMPILER-VM.md` (Phase 0 strip → Phase 1 byte-addressable value model → Phase 2 bytecode → Phase 3 compiler-API on comptime memory). Banner added to `design/comptime-compiler-api.md` (superseded). Reverted the session's uncommitted `register_struct`/`find_type` marshaling experiment back to `reify` HEAD (40d075c). No code stripped yet — Phase 0 is the next action. - **Phase 2 — welded structs by reflection + memory-order validation.** Dropped the byte-layout-override engine (computeWeldPlan / offset-ordered LLVM struct / byte-blob — all explored, all unnecessary). Instead: the sx header declares fields in the compiler type's memory order; the compiler reflects the bound Zig type (`@typeInfo`/`@offsetOf`/`@sizeOf`) and validates the header matches with loud diagnostics (field-not-found, wrong-order+expected-order, size mismatch). On pass it's an ordinary byte-identical struct — cast + deref just works. Examples 0627 (usable) / 1186 (wrong-order diagnostic). Suite green (692). - **Phase 2.1 — weld-plan layout math (REMOVED).** The byte-layout-override math; superseded by the reflection+validation design and deleted. - **Phase 1 polish — comptime-only enforcement.** A runtime call to a welded fn is a clean build-gating error (`emitCall` gate, guarded by enclosing-`is_comptime` so `#run`/`::` uses stay green), not a link failure. Example 1185. Build + suite green (458 unit, 690 corpus). - **Phase 1.1 fifth sub-step — host-call bridge (welded functions).** `compiler_lib` function registry (`intern`/`text_of`) + `findFn`; IR `Function` `compiler_welded` flag set/validated in `declareFunction` (`weldedCompilerFn`); `interp.call()` dispatches welded calls to the Zig handler. Examples 0626 (round- trip) + 1184 (unexported-fn diagnostic); `findFn` unit-tested. Runtime-call clean rejection deferred (loud link error today). Build + suite green (458 unit, 689 corpus). - **Phase 1.1 fourth sub-step — welded-struct layout validation.** `validateStructLayout` (pure, unit-tested) + `validateWeldedStruct` wired into `registerStructDecl`: a `struct abi(.zig) extern compiler` is validated against the registry (lib == compiler, name exported, layout matches) with build-gating diagnostics. `#library "compiler"` no longer dlopen'd. Examples 0625 (faithful Field) + 1183 (field-count mismatch diagnostic). Offset-override/GEP deferred to Phase 2 (not exercised by Field's natural layout). Build + suite green (456 unit, 687 corpus). - **Phase 1.1 third sub-step — binding registry.** New `src/ir/compiler_lib.zig`: the `compiler` lib's welded-type registry; `Field` welded to `StructInfo.Field` with layout baked from the real Zig type (`@offsetOf`/`@sizeOf`/`@alignOf`); `findType` lookup proven by unit test (+ null off the export list). Standalone island — not yet consumed by lowering. Build + suite green (454 unit tests). Break-verified. - **Phase 1.1 second sub-step — struct-decl binding parses.** `ast.StructDecl` gained `abi` + `extern_lib`; `parseStructDecl` parses `abi(.zig) extern ` after `struct`. Parser unit tests (welded `Field` + plain struct), break-verified. Build + suite green. Parse-only sub-step (fns + structs) of Phase 1.1 complete. - **Phase 1.1 first sub-step + `callconv`→`abi` unification.** Parsed `abi(.zig) extern ` on fn decls; unified `callconv` into `abi(.c|.zig|.pure)` (removed the `callconv` keyword), migrated 52 sx files + compiler diagnostics + docs + snapshots. Build + suite green. The original design's `extern(.zig)` single qualifier was split into `abi(.zig)` (ABI/layout, before extern) + `extern ` (linkage + source) — recorded in the design doc's syntax-decision note.