feat(stdlib): per-decl nominal identity + same-name shadows — close 0105 [stdlib E2]
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.
This commit is contained in:
@@ -51,10 +51,16 @@ pub const GenericResolver = struct {
|
||||
|
||||
const info = self.l.module.types.get(ty);
|
||||
return switch (info) {
|
||||
.@"struct" => |s| self.l.module.types.getString(s.name),
|
||||
.@"union" => |u| self.l.module.types.getString(u.name),
|
||||
.tagged_union => |u| self.l.module.types.getString(u.name),
|
||||
.@"enum" => |e| self.l.module.types.getString(e.name),
|
||||
// A nominal type's mangle includes its `nominal_id` when nonzero so two
|
||||
// same-DISPLAY-name authors in different sources (issue 0105) produce
|
||||
// DISTINCT monomorph symbols (`struct_to_string__Box` vs
|
||||
// `struct_to_string__Box__n1`) instead of one symbol with conflicting
|
||||
// signatures. `nominal_id == 0` (the single-author / structural case)
|
||||
// appends nothing — byte-identical to the pre-E2 mangle.
|
||||
.@"struct" => |s| self.mangleNominalName(self.l.module.types.getString(s.name), s.nominal_id),
|
||||
.@"union" => |u| self.mangleNominalName(self.l.module.types.getString(u.name), u.nominal_id),
|
||||
.tagged_union => |u| self.mangleNominalName(self.l.module.types.getString(u.name), u.nominal_id),
|
||||
.@"enum" => |e| self.mangleNominalName(self.l.module.types.getString(e.name), e.nominal_id),
|
||||
.pointer => |p| blk: {
|
||||
const inner = self.mangleTypeName(p.pointee);
|
||||
break :blk std.fmt.allocPrint(self.l.alloc, "ptr_{s}", .{inner}) catch "pointer";
|
||||
@@ -96,6 +102,14 @@ pub const GenericResolver = struct {
|
||||
};
|
||||
}
|
||||
|
||||
/// Append a `__n<id>` disambiguator to a nominal type's display name when its
|
||||
/// `nominal_id` is nonzero (a same-name shadow, issue 0105); id 0 returns the
|
||||
/// name unchanged so single-author mangling is byte-identical.
|
||||
fn mangleNominalName(self: GenericResolver, name: []const u8, nominal_id: u32) []const u8 {
|
||||
if (nominal_id == 0) return name;
|
||||
return std.fmt.allocPrint(self.l.alloc, "{s}__n{d}", .{ name, nominal_id }) catch name;
|
||||
}
|
||||
|
||||
fn mangleParamList(self: GenericResolver, prefix: []const u8, params: []const TypeId, ret: TypeId) []const u8 {
|
||||
var buf = std.ArrayList(u8).empty;
|
||||
buf.appendSlice(self.l.alloc, prefix) catch return prefix;
|
||||
|
||||
Reference in New Issue
Block a user