Make same-name top-level types in different sources DISTINCT nominal types instead of collapsing last-wins in the type table (issue 0105). Registration: - internNamedTypeDecl assigns a per-decl nominal_id and populates type_decl_tids. The first author of a name keeps nominal_id 0 (byte-identical to pre-E2); a genuine cross-module shadow (>=2 distinct normalized-path authors per the import facts) gets a fresh id -> a distinct TypeId. - mergeFlat/addOwnDecl stop first-wins-dropping per-source decls (named types + non-fn const_decls) so every same-name author reaches registration; functions and var_decls (incl. #foreign extern globals) keep first-wins. Resolution (selectNominalLeaf): - own-author wins; else flatTypeAuthorCount over the transitive flat closure: >=2 distinct -> .ambiguous (loud diagnostic + poison); exactly one -> resolved; a flat author not yet findByName-registered -> .undeclared stub (not a leak). - struct-literal type names route through the same source-aware leaf. - lazyLowerFunction pins the function's own source before resolving its return type, so a shadowed signature type resolves in its module, not the caller's. Codegen: - mangleTypeName appends __n<id> for nonzero nominal_id so same-name shadows get distinct monomorph symbols (struct_to_string__Box vs __Box__n1). Library hygiene: - rename trace.sx's compiler-contracted Frame -> TraceFrame (+ the two compiler findByName sites) so it never collides with a UI/geometry Frame; the layout is structural (getFrameStructType / SxFrame), name-independent. Examples: 0752-0756 pin the five 0105 cases (distinct fields / same fields / own-wins / ambiguous / alias per-source); 0170 pins the folded anon-struct-field regression.
99 lines
3.9 KiB
Plaintext
99 lines
3.9 KiB
Plaintext
#import "std.sx";
|
|
|
|
// =====================================================================
|
|
// trace.sx — error return-trace formatting (ERR step E3.3).
|
|
//
|
|
// Reads the thread-local return-trace buffer (ERR E3.1, populated by the
|
|
// push/clear wiring in E3.2) and renders it. A `raise` / propagating `try`
|
|
// pushes a frame; an absorbing site (`catch` / `or value` / destructure)
|
|
// clears the buffer. So at format time the buffer holds exactly the frames
|
|
// of failures that escaped to where you're formatting — typically inside a
|
|
// `catch` handler (the clear fires when the handler completes, so the body
|
|
// still sees the chain) or the (future) failable-`main` wrapper.
|
|
//
|
|
// Frame resolution (ERR E3.0 slice 3a): in compiled code a frame is a pointer
|
|
// to an interned `TraceFrame` the compiler stamped in at the push site, so the
|
|
// location resolves in-process with no DWARF and no symbolizer. (The comptime
|
|
// path — a packed `(func_id, ir_offset)` resolved via the interpreter's IR
|
|
// tables — lands with slice 3b.)
|
|
// =====================================================================
|
|
|
|
libc :: #library "c";
|
|
|
|
// The compiled return-trace frame. Named `TraceFrame` (not `Frame`) so it never
|
|
// collides with a UI / geometry `Frame` a consumer flat-imports — same-name
|
|
// types are now distinct nominal identities (issue 0105), so a bare `Frame` must
|
|
// resolve unambiguously to the consumer's own. Layout MUST match
|
|
// `getFrameStructType` in src/ir/emit_llvm.zig and `SxFrame` in
|
|
// library/vendors/sx_trace_runtime/sx_trace.c.
|
|
TraceFrame :: struct {
|
|
file: string;
|
|
line: s32;
|
|
col: s32;
|
|
func: string;
|
|
line_text: string; // the source line, for the snippet + caret
|
|
}
|
|
|
|
// `n` spaces — used to position the `^` caret under a column.
|
|
spaces :: (n: s32) -> string {
|
|
s := "";
|
|
i : s32 = 0;
|
|
while i < n {
|
|
s = concat(s, " ");
|
|
i = i + 1;
|
|
}
|
|
s
|
|
}
|
|
|
|
// The error-trace buffer C API (library/vendors/sx_trace_runtime/sx_trace.c),
|
|
// linked in for the JIT and auto-injected for AOT when traces are used.
|
|
// `frame_at` returns the raw stored `u64`; `__trace_resolve_frame` turns it
|
|
// into a `TraceFrame` — by reinterpreting the stamped `*TraceFrame` in compiled code, or
|
|
// by resolving the packed `(func_id, span.start)` in the comptime interpreter.
|
|
sx_trace_len :: () -> u32 #foreign;
|
|
sx_trace_truncated :: () -> u32 #foreign;
|
|
sx_trace_frame_at :: (i: u32) -> u64 #foreign;
|
|
|
|
write :: (fd: s32, buf: [*]u8, count: usize) -> isize #foreign libc;
|
|
|
|
// Render the current trace buffer to a string (allocated from
|
|
// context.allocator). Empty buffer → "" so callers can cheaply skip output.
|
|
to_string :: () -> string {
|
|
n := sx_trace_len();
|
|
if n == 0 { return ""; }
|
|
|
|
result := "error return trace (most recent call last):\n";
|
|
if sx_trace_truncated() != 0 {
|
|
result = concat(result, " ... older frames omitted (buffer full)\n");
|
|
}
|
|
|
|
i : u32 = 0;
|
|
while i < n {
|
|
f := __trace_resolve_frame(sx_trace_frame_at(i));
|
|
result = concat(result, format(" {} at {}:{}:{}\n", f.func, f.file, f.line, f.col));
|
|
if f.line_text.len > 0 {
|
|
result = concat(result, format(" {}\n", f.line_text));
|
|
result = concat(result, concat(" ", concat(spaces(f.col - 1), "^\n")));
|
|
}
|
|
i = i + 1;
|
|
}
|
|
result
|
|
}
|
|
|
|
// Write the current trace to stderr (fd 2). No-op when the buffer is empty.
|
|
print_current :: () {
|
|
s := to_string();
|
|
if s.len > 0 {
|
|
write(2, s.ptr, xx s.len);
|
|
}
|
|
}
|
|
|
|
// Dump the comptime (`#run`) interpreter call-frame chain (ERR E4.1). At
|
|
// comptime the interpreter walks its active sx frames and appends them to the
|
|
// build output; in compiled code this folds to nothing (there is no
|
|
// interpreter stack — the only caller is a dead `is_comptime()` branch).
|
|
// Frame source locations await IR-offset resolution, so only names print today.
|
|
print_interpreter_frames :: () {
|
|
__interp_print_frames();
|
|
}
|