REIFY Phase 3.1. Add RecvResult($T) and TryResult($T) to meta.sx as type-fns over reify (value-or-closed; value-or-empty-or-closed). They need NO new compiler machinery — reify-of-a-literal in a type-fn body is exactly the Phase 1 path — so the channel result types are pure sx library code. examples/0617 green (both construct + match, incl. payload-less .closed / .empty). Suite green (673 examples, 447 unit). make_enum(variants) (3.2) and type_info (2.2) remain — both blocked on a generalized reify reader (reifyType currently AST-walks a literal TypeInfo). Plan/checkpoint updated.
11 KiB
CHECKPOINT-REIFY — comptime type_info / reify (async-first foundation, step 3)
Companion to PLAN-REIFY.md. Update after every step (one step at a time, per the cadence rule).
Last completed step
Phase 3.1 (green) — RecvResult/TryResult done. Added RecvResult($T) and
TryResult($T) to meta.sx as type-fns over reify (a value-or-closed, and a
value-or-empty-or-closed enum). They needed NO new compiler machinery — they're
reify-of-a-literal in a type-fn body, i.e. exactly the Phase 1 path — so the channel
result types are pure sx library code. examples/0617 green (both construct +
match, incl. payload-less .closed/.empty). Full suite green (673 examples).
Cadence: 3.0 xfail (undefined → RED) → 3.1 green.
make_enum(variants) NOT done (Phase 3.2): it takes a RUNTIME []EnumVariant,
so reify(.enum(.{ variants = variants })) is a non-literal arg — blocked on the
generalized reify reader (pairs with 2.2). type_info NOT done (Phase 2.2).
(prior) Phase 2.1 (green) — field_type done. field_type($T, i) -> Type is
implemented over the type table (fieldTypeOf in lower/generic.zig, re-exported
on Lowering): struct field / tagged-union + union variant payload (.void for
a tagless variant) / tuple element / array + vector element; OOB and memberless
types poison to .unresolved with a loud diagnostic (never a silent default).
It folds at lower time, so it composes inside type_eq / type_name / any type-arg
slot. examples/0616 green (struct fields name+type, type_eq fold, tagged-union
payloads incl. quit → void). Full suite green (672 examples, 447 unit). Cadence:
2.0 xfail (empty marker, RED) → 2.1 green (this commit).
type_info is NOT done — it still bails loudly in
call.zig:tryLowerReflectionCall. It builds a full TypeInfo value (inverse of
reify) and is the larger Phase 2.2 step (see Next step).
(prior) Phase 1 (type-fn → reify identity) — COMPLETE. A type-fn body that returns
reify(...) now mints the enum under the instantiation's name:
instantiateTypeFunction (lower/generic.zig) detects a reify-returning body
(findReturnReifyCall) and routes it to reifyType(<mangled-or-alias name>, reify_call) with the type-arg bindings active (so payload = T resolves to the
bound arg) — placed BEFORE the general case, which would otherwise route the
reify call to the inline-position loud bail. Registering under the mangled name
lets the fn's top-of-instantiation cache return the SAME TypeId on the second
instantiation → Box(i64) at two independent sites is ONE type (Contract 1).
examples/0615 green (build()→consume() cross-site, plus b : Box(i64) = .none); full suite green (671 examples, 447 unit). Cadence: 1.0 xfail (empty
marker, RED) → 1.1 green (this commit).
(prior) Phase 0.2 (green) — Phase 0 COMPLETE. Implemented reify(.enum(...)):
Lowering.reifyType (in lower/nominal.zig) reads the flat-enum TypeInfo
literal off the AST, synthesizes an ast.EnumDecl, and feeds it through the
SAME type_bridge.buildEnumInfo path source enums use → the minted type is
byte-identical to a hand-written enum { value: i64; closed; } and flows
through enum codegen unmodified (Contract 2 confirmed — a source enum
exhibits the exact same construct/match behavior, verified by probe). Wired at
the E :: reify(...) const-decl hook in lower/decl.zig. examples/0614
green (value 3 / closed, exit 0); snapshots captured; full suite green
(670 examples, 447 unit). Unsupported reify shapes bail loudly via
reifyBail (never a silent default).
Also landed (user-directed, separate commit feat(parser):): reserved keyword
as a member name after . (.enum, case .enum:, x.enum) — so the reify
example reads reify(.enum(...)) without a backtick. readme updated.
Two earlier Phase-0 commits
- 0.1 (xfail). Added
examples/0614+ empty.exitmarker → RED. - 0.0 (lock). Added the comptime type-metaprogramming surface as the
on-demand module
library/modules/std/meta.sx(NOT the prelude — see decision below):EnumVariant/EnumInfo/TypeInfodata types + bodyless#builtindeclsreify/type_info/field_type. Each builtin bails LOUDLY when reached unimplemented (no silent default). Unit testsrc/parser.test.zig(registered insrc/root.zig) locks that the decls parse.zig build testgreen (447/447 unit, 669/669 examples).
Current state
modules/std/meta.sxdeclares the surface; the variant uses the backtick raw escape`enum(reads as the keyword, not a mangledenum_).- Loud bails wired (unimplemented → diagnostic, never a silent type):
reify(...)in a::type-alias position →decl.zig(the.callconst-decl branch) emits "reify is not yet implemented (REIFY Phase 0.2)" and poisons the alias to.unresolved. This is also where Phase 0.2 will hook the real construction.reify/field_typein any other type position →generic.zig:resolveTypeCallWithBindings(defense-in-depth).type_info(...)in expression position →call.zig:tryLowerReflectionCall.
- No interpreter-side construction yet —
reifymints nothing.
Decision (0.0)
Meta lives in modules/std/meta.sx, not the prelude (core.sx). Declaring
the data types in the always-loaded prelude interns them into every module's
type table and shifts every .ir snapshot (broke 37 examples in a trial). An
on-demand import keeps the prelude clean; reify users #import "modules/std/meta.sx". (User-directed.)
Next step
Two pending pieces, both blocked on the SAME thing — a generalized reify reader
(reifyType currently AST-walks a LITERAL TypeInfo; it can't read a computed /
variable TypeInfo). Do that generalization (interp-evaluate the TypeInfo value,
or a broader AST reader) and both unlock:
- Phase 3.2 —
make_enum(variants: []EnumVariant) -> Type(sx lib;variantsis a runtime value, not a literal). - Phase 2.2 —
type_info($T) -> TypeInfo(reflect → aTypeInfovalue; also needs theTypeInfodata model widened past`enumfor struct/tuple). Bails loudly today incall.zig:tryLowerReflectionCall.
Remaining after that: Phase 4 (self-reference via declare()/define() — API
already decided), Phase 5 (validation + loud diagnostics: dup variants, bad
backing, by-value self-ref). Both are independent of the reader generalization.
(superseded) earlier Next step — type_info value construction Reflect a type into a TypeInfo
value — the inverse of reify. Two sub-pieces, both non-trivial: (a) widen the
meta.sx TypeInfo data model beyond `enum (struct / tuple variants); (b)
build the value at comptime — a []EnumVariant-style slice of structs holding
strings (name) + Type tags (payload), populated from the type table. This is
comptime value-CONSTRUCTION (allocating slice/struct/string/type-tag values in the
interpreter), materially larger than 2.1's fold-to-a-TypeId. Currently bails loudly
in call.zig:tryLowerReflectionCall. Best done as its own session for context room.
NOTE (round-tripping): reifyType still reads a LITERAL TypeInfo off the AST
(fine for the inline-literal + type-fn-over-literal cases of Phases 0–1). Once 2.2
produces a COMPUTED TypeInfo, feeding it back into reify needs the reader
generalized (or interp evaluation of the TypeInfo value) — handle that when 2.2
enables round-tripping.
Alternatively jump to Phase 3 (make_enum + RecvResult/TryResult sx lib over
reify) — that only needs reify (have it) + type-fns (have them), not type_info.
SELF-REFERENCE = Phase 4, API DECIDED (user-directed): explicit declare() →
define(h, info) (the declaration-vs-definition split; NOT a reify_rec((self)=>…)
closure). declare() returns a forward nominal Type handle (named from the ::
LHS) usable freely in any later TypeInfo — *List, []List, and across types for
MUTUAL recursion the one-self closure couldn't express; define(handle, info) fills
the body. reify(info) stays as the one-shot sugar. Maps onto existing machinery:
declare = empty nominal slot (reserveShadowEnumSlot-style), define = buildEnumInfo
updatePreservingKey, struct-stub→tagged_union re-key already handled (internNamedTypeDecl'sadoptsForwardStructStub). Loud invariants: never-defined declare = hard error; by-value self-inclusion rejected (infinite size). Full write-up in PLAN-REIFY Phase 4.
Known issues
None yet.
Log
- 3.1 (green) —
RecvResult/TryResultdone. Added as type-fns overreifyinmeta.sx(no new machinery — Phase 1 reify-of-literal-in-type-fn);0617green.make_enum(3.2) pending on the generalized reify reader. - 3.0 (xfail).
0617+ empty marker → RED (RecvResult undefined). - 2.1 (green) —
field_typedone.fieldTypeOfover the type table (struct/tagged-union/union/tuple/array/vector; OOB+memberless = loud poison); folds at lower time, composes intype_eq/type_name;0616green.type_infostill pending (Phase 2.2 — builds a TypeInfo value). - 2.0 (xfail).
0616+ empty.exitmarker → RED (field_type bailed). - 1.1 (green) — Phase 1 done. Type-fn body
return reify(...)routes throughreifyTypeunder the instantiation name (findReturnReifyCall+ mangled-name registration);Box(i64)at two sites = one type (Contract 1);0615green. - 1.0 (xfail).
0615+ empty.exitmarker → RED (reify bailed in type-fn body). - 0.2 (green) — Phase 0 done.
reifyTypemints a flat enum from a literalTypeInfovia the sharedbuildEnumInfopath;0614green; Contract 2 confirmed (reify'd enum == source enum, same construct/match). Payload-less variant idiom in the example isc : E = .closed;(enum-literal target), same as a source enum. - parser (user-directed). Keyword as member name after
.— seefeat(parser). - 0.1 (xfail).
0614+ empty.exitmarker → RED. - 0.0 (lock). Meta surface in
modules/std/meta.sx(data types + 3 bodyless#builtindecls), loud bails at all three reach points,src/parser.test.zigparse-lock. Two user-directed refinements folded in: variant uses`enumraw escape; surface moved out of the prelude into its own module to avoid type-table /.ir-snapshot churn.zig build testgreen. - Stream carved. Selected as the first async-first foundation:
reifygates both channel result types (RecvResult($T)) andrace's synthesized union, is fully validated (3 reviewers), and is a self-contained compiler/type-system feature testable in isolation (06xxcomptime). Generic-enum syntax dropped in its favor.