compiler: lower enums to mir

This commit is contained in:
sfja 2025-02-22 03:44:35 +01:00
parent 4c4b0da238
commit 66fc0ec0b5
9 changed files with 237 additions and 28 deletions

View File

@ -4,6 +4,7 @@ import {
Ctx, Ctx,
exhausted, exhausted,
File, File,
IdentId,
IdMap, IdMap,
IdSet, IdSet,
Ok, Ok,
@ -12,7 +13,14 @@ import {
todo, todo,
} from "@slige/common"; } from "@slige/common";
import * as resolve from "@slige/resolve"; import * as resolve from "@slige/resolve";
import { ElemDef, FieldDef, Ty, tyToString, VariantData } from "@slige/ty"; import {
ElemDef,
FieldDef,
Ty,
tyToString,
Variant,
VariantData,
} from "@slige/ty";
export class Checker { export class Checker {
private stmtChecked = new IdSet<AstId>(); private stmtChecked = new IdSet<AstId>();
@ -232,6 +240,27 @@ export class Checker {
exhausted(k); exhausted(k);
} }
public enumItemTy(item: ast.Item, kind: ast.EnumItem): Ty {
return this.itemTys.get(item.id) ?? this.checkEnumItem(item, kind);
}
private checkEnumItem(item: ast.Item, kind: ast.EnumItem): Ty {
const variantIdents = new Set<IdentId>();
const variants: Variant[] = [];
for (const variant of kind.variants) {
if (variantIdents.has(variant.ident.id)) {
this.report(`variant name already defined`, variant.span);
return Ty({ tag: "error" });
}
variantIdents.add(variant.ident.id);
variants.push({
ident: variant.ident.id,
data: this.checkVariantData(variant.data),
});
}
return Ty({ tag: "enum", item, kind, variants });
}
public structItemTy(item: ast.Item, kind: ast.StructItem): Ty { public structItemTy(item: ast.Item, kind: ast.StructItem): Ty {
return this.itemTys.get(item.id) ?? this.checkStructItem(item, kind); return this.itemTys.get(item.id) ?? this.checkStructItem(item, kind);
} }
@ -364,7 +393,30 @@ export class Checker {
); );
return Ty({ tag: "error" }); return Ty({ tag: "error" });
case "unit": case "unit":
return this.structItemTy(res.kind.item, res.kind.kind); //return this.structItemTy(res.kind.item, res.kind.kind);
return todo();
case "tuple":
this.report(
"expected value, got struct type",
expr.span,
);
return Ty({ tag: "error" });
}
return exhausted(data.kind);
}
case "variant": {
const data = res.kind.variant.data;
switch (data.kind.tag) {
case "error":
return Ty({ tag: "error" });
case "struct":
this.report(
"expected value, got struct type",
expr.span,
);
return Ty({ tag: "error" });
case "unit":
return todo();
case "tuple": case "tuple":
this.report( this.report(
"expected value, got struct type", "expected value, got struct type",
@ -374,8 +426,6 @@ export class Checker {
} }
return exhausted(data.kind); return exhausted(data.kind);
} }
case "variant":
return todo("return a ctor here");
case "field": case "field":
throw new Error(); throw new Error();
case "fn": { case "fn": {
@ -411,21 +461,32 @@ export class Checker {
return todo(); return todo();
} }
const res = this.re.pathRes(kind.path.id); const res = this.re.pathRes(kind.path.id);
if (res.kind.tag !== "struct") { let ty: Ty;
if (res.kind.tag === "struct") {
ty = this.structItemTy(res.kind.item, res.kind.kind);
} else if (res.kind.tag === "variant") {
ty = this.enumItemTy(res.kind.item, res.kind.kind);
} else {
this.report("type is not a struct", kind.path.span); this.report("type is not a struct", kind.path.span);
const ty = Ty({ tag: "error" }); const ty = Ty({ tag: "error" });
this.exprTys.set(expr.id, ty); this.exprTys.set(expr.id, ty);
return ty; return ty;
} }
const ty = this.structItemTy(res.kind.item, res.kind.kind);
this.exprTys.set(expr.id, ty); this.exprTys.set(expr.id, ty);
if (ty.kind.tag === "error") { if (ty.kind.tag === "error") {
return ty; return ty;
} }
if (ty.kind.tag !== "struct") { let data: VariantData;
if (ty.kind.tag === "struct") {
data = ty.kind.data;
} else if (ty.kind.tag === "enum") {
if (res.kind.tag !== "variant") {
throw new Error();
}
data = ty.kind.variants[res.kind.variantIdx].data;
} else {
throw new Error(); throw new Error();
} }
const data = ty.kind.data;
if (data.tag !== "struct") { if (data.tag !== "struct") {
this.report("struct data not a struct", kind.path.span); this.report("struct data not a struct", kind.path.span);
const ty = Ty({ tag: "error" }); const ty = Ty({ tag: "error" });
@ -471,6 +532,9 @@ export class Checker {
kind: ast.CallExpr, kind: ast.CallExpr,
expected: Ty, expected: Ty,
): Ty { ): Ty {
if (this.callExprIsTupleVariantCtor(kind)) {
return this.checkCallExprTupleVariantCtor(expr, kind, expected);
}
if (this.callExprIsTupleStructCtor(kind)) { if (this.callExprIsTupleStructCtor(kind)) {
return this.checkCallExprTupleStructCtor(expr, kind, expected); return this.checkCallExprTupleStructCtor(expr, kind, expected);
} }
@ -501,6 +565,56 @@ export class Checker {
return ty; return ty;
} }
private callExprIsTupleVariantCtor(kind: ast.CallExpr): boolean {
if (kind.expr.kind.tag !== "path") {
return false;
}
const res = this.re.exprRes(kind.expr.id);
return res.kind.tag === "variant" &&
res.kind.variant.data.kind.tag === "tuple";
}
private checkCallExprTupleVariantCtor(
expr: ast.Expr,
kind: ast.CallExpr,
expected: Ty,
): Ty {
if (kind.expr.kind.tag !== "path") {
throw new Error();
}
const res = this.re.exprRes(kind.expr.id);
if (res.kind.tag !== "variant") {
throw new Error();
}
const ty = this.enumItemTy(res.kind.item, res.kind.kind);
this.exprTys.set(expr.id, ty);
if (ty.kind.tag === "error") {
return ty;
}
if (ty.kind.tag !== "enum") {
throw new Error();
}
const data = ty.kind.variants[res.kind.variantIdx].data;
if (data.tag !== "tuple") {
this.report(
"variant data not a tuple",
kind.expr.kind.path.span,
);
const ty = Ty({ tag: "error" });
this.exprTys.set(expr.id, ty);
return ty;
}
for (const [i, arg] of kind.args.entries()) {
const argTy = this.exprTy(arg);
const tyRes = this.resolveTys(argTy, data.elems[i].ty);
if (!tyRes.ok) {
this.report(tyRes.val, arg.span);
return ty;
}
}
return ty;
}
private callExprIsTupleStructCtor(kind: ast.CallExpr): boolean { private callExprIsTupleStructCtor(kind: ast.CallExpr): boolean {
if (kind.expr.kind.tag !== "path") { if (kind.expr.kind.tag !== "path") {
return false; return false;
@ -742,7 +856,7 @@ export class Checker {
case "error": case "error":
return Ty({ tag: "error" }); return Ty({ tag: "error" });
case "enum": case "enum":
return todo(); return this.enumItemTy(k.item, k.kind);
case "struct": case "struct":
return this.structItemTy(k.item, k.kind); return this.structItemTy(k.item, k.kind);
case "fn": case "fn":
@ -854,6 +968,15 @@ export class Checker {
} }
return Res.Ok(a); return Res.Ok(a);
} }
case "enum": {
if (b.kind.tag !== "enum") {
return incompat();
}
if (a.kind.item.id !== b.kind.item.id) {
return incompat();
}
return Res.Ok(a);
}
case "struct": { case "struct": {
if (b.kind.tag !== "struct") { if (b.kind.tag !== "struct") {
return incompat(); return incompat();

View File

@ -326,15 +326,26 @@ export class FnLowerer {
private lowerStructExpr(expr: ast.Expr, kind: ast.StructExpr): RVal { private lowerStructExpr(expr: ast.Expr, kind: ast.StructExpr): RVal {
const ty = this.ch.exprTy(expr); const ty = this.ch.exprTy(expr);
let variant: ast.Variant | undefined = undefined;
if (ty.kind.tag === "enum") {
const res = this.re.pathRes(kind.path!.id);
if (res.kind.tag !== "variant") {
throw new Error();
}
variant = res.kind.variant;
}
const fields = kind.fields const fields = kind.fields
.map((field) => this.lowerExprToOperand(field.expr)); .map((field) => this.lowerExprToOperand(field.expr));
return { tag: "struct", ty, fields }; return { tag: "adt", ty, fields, variant };
} }
private lowerCallExpr(expr: ast.Expr, kind: ast.CallExpr): RVal { private lowerCallExpr(expr: ast.Expr, kind: ast.CallExpr): RVal {
if (this.callExprIsTupleStructCtor(kind)) { if (this.callExprIsTupleStructCtor(kind)) {
return this.lowerCallExprTupleStructCtor(expr, kind); return this.lowerCallExprTupleStructCtor(expr, kind);
} }
if (this.callExprIsTupleVariantCtor(kind)) {
return this.lowerCallExprTupleVariantCtor(expr, kind);
}
const args = kind.args.map((arg) => this.lowerExprToOperand(arg)); const args = kind.args.map((arg) => this.lowerExprToOperand(arg));
const func = this.lowerExprToOperand(kind.expr); const func = this.lowerExprToOperand(kind.expr);
return { tag: "call", func, args }; return { tag: "call", func, args };
@ -355,7 +366,31 @@ export class FnLowerer {
const ty = this.ch.exprTy(expr); const ty = this.ch.exprTy(expr);
const fields = kind.args const fields = kind.args
.map((arg) => this.lowerExprToOperand(arg)); .map((arg) => this.lowerExprToOperand(arg));
return { tag: "struct", ty, fields }; return { tag: "adt", ty, fields };
}
private callExprIsTupleVariantCtor(kind: ast.CallExpr): boolean {
if (kind.expr.kind.tag !== "path") {
return false;
}
const res = this.re.exprRes(kind.expr.id);
return res.kind.tag === "variant" &&
res.kind.variant.data.kind.tag === "tuple";
}
private lowerCallExprTupleVariantCtor(
expr: ast.Expr,
kind: ast.CallExpr,
): RVal {
const res = this.re.exprRes(kind.expr.id);
if (res.kind.tag !== "variant") {
throw new Error();
}
const variant = res.kind.variant;
const ty = this.ch.exprTy(expr);
const fields = kind.args
.map((arg) => this.lowerExprToOperand(arg));
return { tag: "adt", ty, fields, variant };
} }
private lowerBinaryExpr(expr: ast.Expr, kind: ast.BinaryExpr): RVal { private lowerBinaryExpr(expr: ast.Expr, kind: ast.BinaryExpr): RVal {
@ -661,13 +696,23 @@ export class FnLowerer {
return { tag: "error" }; return { tag: "error" };
case "struct": case "struct":
case "unit": case "unit":
return todo(data.kind.tag);
case "tuple": case "tuple":
return todo(data.kind.tag); return todo(data.kind.tag);
} }
return exhausted(data.kind); return exhausted(data.kind);
} }
case "variant": case "variant": {
const data = re.kind.variant.data;
switch (data.kind.tag) {
case "error":
return { tag: "error" };
case "struct":
case "unit":
case "tuple":
return todo(data.kind.tag);
}
return exhausted(data.kind);
}
case "field": case "field":
return todo(); return todo();
case "local": { case "local": {
@ -704,6 +749,7 @@ export class FnLowerer {
case "bool": case "bool":
return true; return true;
case "fn": case "fn":
case "enum":
case "struct": case "struct":
return false; return false;
} }

View File

@ -85,7 +85,7 @@ export type RVal =
| { tag: "ptr"; place: Place; mut: boolean } | { tag: "ptr"; place: Place; mut: boolean }
| { tag: "binary"; binaryType: BinaryType; left: Operand; right: Operand } | { tag: "binary"; binaryType: BinaryType; left: Operand; right: Operand }
| { tag: "unary"; unaryType: UnaryType; operand: Operand } | { tag: "unary"; unaryType: UnaryType; operand: Operand }
| { tag: "struct"; ty: Ty; fields: Operand[] } | { tag: "adt"; ty: Ty; fields: Operand[]; variant?: ast.Variant }
| { tag: "call"; func: Operand; args: Operand[] }; | { tag: "call"; func: Operand; args: Operand[] };
export type BinaryType = export type BinaryType =

View File

@ -349,7 +349,6 @@ export class Parser {
this.report("expected '{'"); this.report("expected '{'");
return this.stmt({ tag: "error" }, begin); return this.stmt({ tag: "error" }, begin);
} }
this.step();
const endSpan: [Span] = [begin]; const endSpan: [Span] = [begin];
const variants = this.parseDelimitedList( const variants = this.parseDelimitedList(
this.parseVariant, this.parseVariant,

View File

@ -1,18 +1,13 @@
struct A(int); enum Abc {
struct B(int, int); B(int),
struct C { C { v: int },
a: A,
v: int,
} }
fn main() { fn main() {
let a: A = A(123); let b: Abc = Abc::B(123);
let b = B(1, 2);
let c = C { let c = Abc::C { v: 123 };
a: a,
v: 123,
};
} }

View File

@ -57,7 +57,9 @@ export class HirStringifyer {
this.file(k.ast!, depth + 1) this.file(k.ast!, depth + 1)
}\n}`; }\n}`;
case "enum": case "enum":
return todo(); return `enum ${ident}: ${
this.ty(this.ch.enumItemTy(item, k))
};`;
case "struct": case "struct":
return `struct ${ident}: ${ return `struct ${ident}: ${
this.ty(this.ch.structItemTy(item, k)) this.ty(this.ch.structItemTy(item, k))

View File

@ -165,7 +165,7 @@ export class MirFnStringifyer {
} }
case "unary": case "unary":
return todo(rval.tag); return todo(rval.tag);
case "struct": { case "adt": {
const tyk = rval.ty.kind; const tyk = rval.ty.kind;
if (tyk.tag === "struct") { if (tyk.tag === "struct") {
const datak = tyk.kind.data.kind; const datak = tyk.kind.data.kind;
@ -194,6 +194,35 @@ export class MirFnStringifyer {
} }
} }
return exhausted(datak); return exhausted(datak);
} else if (tyk.tag === "enum") {
const datak = rval.variant!.data.kind;
switch (datak.tag) {
case "error":
return "<error>";
case "unit":
return todo();
case "tuple": {
const name = `${tyk.item.ident.text}::${
rval.variant!.ident.text
}`;
const fields = rval.fields
.map((field, idx) => this.operand(field))
.join(", ");
return `${name}(${fields})`;
}
case "struct": {
const name = tyk.item.ident.text;
const fields = rval.fields
.map((field, idx) =>
`${datak.fields[idx].ident!.text}: ${
this.operand(field)
}`
)
.join(", ");
return `${name} { ${fields} }`;
}
}
return exhausted(datak);
} else { } else {
return todo(); return todo();
} }

View File

@ -22,6 +22,10 @@ export function tyToString(ctx: Ctx, ty: Ty): string {
const reTy = tyToString(ctx, k.returnTy); const reTy = tyToString(ctx, k.returnTy);
return `fn ${identText}(${params}) -> ${reTy}`; return `fn ${identText}(${params}) -> ${reTy}`;
} }
case "enum": {
const identText = ctx.identText(k.item.ident.id);
return identText;
}
case "struct": { case "struct": {
const identText = ctx.identText(k.item.ident.id); const identText = ctx.identText(k.item.ident.id);
return identText; return identText;

View File

@ -20,6 +20,12 @@ export type TyKind =
params: Ty[]; params: Ty[];
returnTy: Ty; returnTy: Ty;
} }
| {
tag: "enum";
item: ast.Item;
kind: ast.EnumItem;
variants: Variant[];
}
| { | {
tag: "struct"; tag: "struct";
item: ast.Item; item: ast.Item;
@ -27,6 +33,11 @@ export type TyKind =
data: VariantData; data: VariantData;
}; };
export type Variant = {
ident: IdentId;
data: VariantData;
};
export type VariantData = export type VariantData =
| { tag: "error" } | { tag: "error" }
| { tag: "unit" } | { tag: "unit" }