mirror of
https://github.com/Mercantec-GHC/h4-projekt-gruppe-0-sm.git
synced 2025-04-28 00:34:06 +02:00
compiler: add tuple structs
This commit is contained in:
parent
5b109a5432
commit
4c4b0da238
@ -351,8 +351,29 @@ export class Checker {
|
|||||||
case "error":
|
case "error":
|
||||||
return Ty({ tag: "error" });
|
return Ty({ tag: "error" });
|
||||||
case "enum":
|
case "enum":
|
||||||
case "struct":
|
return todo("return enum type here");
|
||||||
return todo("return a ctor here");
|
case "struct": {
|
||||||
|
const data = res.kind.kind.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 this.structItemTy(res.kind.item, res.kind.kind);
|
||||||
|
case "tuple":
|
||||||
|
this.report(
|
||||||
|
"expected value, got struct type",
|
||||||
|
expr.span,
|
||||||
|
);
|
||||||
|
return Ty({ tag: "error" });
|
||||||
|
}
|
||||||
|
return exhausted(data.kind);
|
||||||
|
}
|
||||||
case "variant":
|
case "variant":
|
||||||
return todo("return a ctor here");
|
return todo("return a ctor here");
|
||||||
case "field":
|
case "field":
|
||||||
@ -450,6 +471,9 @@ export class Checker {
|
|||||||
kind: ast.CallExpr,
|
kind: ast.CallExpr,
|
||||||
expected: Ty,
|
expected: Ty,
|
||||||
): Ty {
|
): Ty {
|
||||||
|
if (this.callExprIsTupleStructCtor(kind)) {
|
||||||
|
return this.checkCallExprTupleStructCtor(expr, kind, expected);
|
||||||
|
}
|
||||||
const fnTy = this.exprTy(kind.expr);
|
const fnTy = this.exprTy(kind.expr);
|
||||||
if (fnTy.kind.tag !== "fn") {
|
if (fnTy.kind.tag !== "fn") {
|
||||||
if (fnTy.kind.tag === "error") {
|
if (fnTy.kind.tag === "error") {
|
||||||
@ -477,6 +501,55 @@ export class Checker {
|
|||||||
return ty;
|
return ty;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private callExprIsTupleStructCtor(kind: ast.CallExpr): boolean {
|
||||||
|
if (kind.expr.kind.tag !== "path") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const res = this.re.exprRes(kind.expr.id);
|
||||||
|
return res.kind.tag === "struct";
|
||||||
|
}
|
||||||
|
|
||||||
|
private checkCallExprTupleStructCtor(
|
||||||
|
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 !== "struct") {
|
||||||
|
throw new Error();
|
||||||
|
}
|
||||||
|
const ty = this.structItemTy(res.kind.item, res.kind.kind);
|
||||||
|
this.exprTys.set(expr.id, ty);
|
||||||
|
if (ty.kind.tag === "error") {
|
||||||
|
return ty;
|
||||||
|
}
|
||||||
|
if (ty.kind.tag !== "struct") {
|
||||||
|
throw new Error();
|
||||||
|
}
|
||||||
|
const data = ty.kind.data;
|
||||||
|
if (data.tag !== "tuple") {
|
||||||
|
this.report(
|
||||||
|
"struct 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 checkBinaryExpr(
|
private checkBinaryExpr(
|
||||||
expr: ast.Expr,
|
expr: ast.Expr,
|
||||||
kind: ast.BinaryExpr,
|
kind: ast.BinaryExpr,
|
||||||
|
@ -332,11 +332,32 @@ export class FnLowerer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private lowerCallExpr(expr: ast.Expr, kind: ast.CallExpr): RVal {
|
private lowerCallExpr(expr: ast.Expr, kind: ast.CallExpr): RVal {
|
||||||
|
if (this.callExprIsTupleStructCtor(kind)) {
|
||||||
|
return this.lowerCallExprTupleStructCtor(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 };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private callExprIsTupleStructCtor(kind: ast.CallExpr): boolean {
|
||||||
|
if (kind.expr.kind.tag !== "path") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const res = this.re.exprRes(kind.expr.id);
|
||||||
|
return res.kind.tag === "struct";
|
||||||
|
}
|
||||||
|
|
||||||
|
private lowerCallExprTupleStructCtor(
|
||||||
|
expr: ast.Expr,
|
||||||
|
kind: ast.CallExpr,
|
||||||
|
): RVal {
|
||||||
|
const ty = this.ch.exprTy(expr);
|
||||||
|
const fields = kind.args
|
||||||
|
.map((arg) => this.lowerExprToOperand(arg));
|
||||||
|
return { tag: "struct", ty, fields };
|
||||||
|
}
|
||||||
|
|
||||||
private lowerBinaryExpr(expr: ast.Expr, kind: ast.BinaryExpr): RVal {
|
private lowerBinaryExpr(expr: ast.Expr, kind: ast.BinaryExpr): RVal {
|
||||||
const left = this.lowerExprToOperand(kind.left);
|
const left = this.lowerExprToOperand(kind.left);
|
||||||
const right = this.lowerExprToOperand(kind.right);
|
const right = this.lowerExprToOperand(kind.right);
|
||||||
@ -632,7 +653,20 @@ export class FnLowerer {
|
|||||||
case "error":
|
case "error":
|
||||||
return { tag: "error" };
|
return { tag: "error" };
|
||||||
case "enum":
|
case "enum":
|
||||||
case "struct":
|
return todo();
|
||||||
|
case "struct": {
|
||||||
|
const data = re.kind.kind.data;
|
||||||
|
switch (data.kind.tag) {
|
||||||
|
case "error":
|
||||||
|
return { tag: "error" };
|
||||||
|
case "struct":
|
||||||
|
case "unit":
|
||||||
|
return todo(data.kind.tag);
|
||||||
|
case "tuple":
|
||||||
|
return todo(data.kind.tag);
|
||||||
|
}
|
||||||
|
return exhausted(data.kind);
|
||||||
|
}
|
||||||
case "variant":
|
case "variant":
|
||||||
case "field":
|
case "field":
|
||||||
return todo();
|
return todo();
|
||||||
|
@ -394,6 +394,9 @@ export class Parser {
|
|||||||
}
|
}
|
||||||
const ident = this.parseIdent();
|
const ident = this.parseIdent();
|
||||||
const data = this.parseVariantData();
|
const data = this.parseVariantData();
|
||||||
|
if (data.kind.tag === "unit" || data.kind.tag === "tuple") {
|
||||||
|
this.eatSemicolon();
|
||||||
|
}
|
||||||
const span = Span.fromto(begin, data.span);
|
const span = Span.fromto(begin, data.span);
|
||||||
return this.stmt({
|
return this.stmt({
|
||||||
tag: "item",
|
tag: "item",
|
||||||
|
@ -1,10 +1,18 @@
|
|||||||
|
|
||||||
struct A {
|
struct A(int);
|
||||||
|
struct B(int, int);
|
||||||
|
struct C {
|
||||||
|
a: A,
|
||||||
v: int,
|
v: int,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let a: A = A { v: 123 };
|
let a: A = A(123);
|
||||||
|
let b = B(1, 2);
|
||||||
|
let c = C {
|
||||||
|
a: a,
|
||||||
|
v: 123,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -241,11 +241,31 @@ export class Resolver implements ast.Visitor {
|
|||||||
return "stop";
|
return "stop";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
visitCallExpr(expr: ast.Expr, kind: ast.CallExpr): ast.VisitRes {
|
||||||
|
if (
|
||||||
|
kind.expr.kind.tag === "path" &&
|
||||||
|
kind.expr.kind.path.segments.length === 1
|
||||||
|
) {
|
||||||
|
const res = this.resolveTyPath(kind.expr.kind.path);
|
||||||
|
if (res.kind.tag === "struct") {
|
||||||
|
this.exprResols.set(kind.expr.id, res);
|
||||||
|
for (const arg of kind.args) {
|
||||||
|
ast.visitExpr(this, arg);
|
||||||
|
}
|
||||||
|
return "stop";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// otherwise, just continue as usual
|
||||||
|
}
|
||||||
|
|
||||||
visitStructExpr(expr: ast.Expr, kind: ast.StructExpr): ast.VisitRes {
|
visitStructExpr(expr: ast.Expr, kind: ast.StructExpr): ast.VisitRes {
|
||||||
if (!kind.path) {
|
if (!kind.path) {
|
||||||
return todo();
|
return todo();
|
||||||
}
|
}
|
||||||
this.resolveValPath(kind.path);
|
this.resolveTyPath(kind.path);
|
||||||
|
for (const field of kind.fields) {
|
||||||
|
ast.visitExpr(this, field.expr);
|
||||||
|
}
|
||||||
return "stop";
|
return "stop";
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -321,7 +341,7 @@ export class Resolver implements ast.Visitor {
|
|||||||
|
|
||||||
private resolveValPath(path: ast.Path): Resolve {
|
private resolveValPath(path: ast.Path): Resolve {
|
||||||
let res: Resolve;
|
let res: Resolve;
|
||||||
if (path.segments.length === 0) {
|
if (path.segments.length === 1) {
|
||||||
res = this.syms.getVal(path.segments[0].ident);
|
res = this.syms.getVal(path.segments[0].ident);
|
||||||
} else {
|
} else {
|
||||||
res = path.segments
|
res = path.segments
|
||||||
@ -343,6 +363,16 @@ export class Resolver implements ast.Visitor {
|
|||||||
exhausted();
|
exhausted();
|
||||||
}, this.syms.getTy(path.segments[0].ident));
|
}, this.syms.getTy(path.segments[0].ident));
|
||||||
}
|
}
|
||||||
|
if (res.kind.tag === "error") {
|
||||||
|
if (path.segments.length === 1) {
|
||||||
|
this.report(
|
||||||
|
`could not resolve symbol '${path.segments[0].ident.text}'`,
|
||||||
|
path.span,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
this.report(`could not path`, path.span);
|
||||||
|
}
|
||||||
|
}
|
||||||
this.pathResols.set(path.id, res);
|
this.pathResols.set(path.id, res);
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
@ -169,18 +169,31 @@ export class MirFnStringifyer {
|
|||||||
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;
|
||||||
if (datak.tag !== "struct") {
|
switch (datak.tag) {
|
||||||
throw new Error();
|
case "error":
|
||||||
|
return "<error>";
|
||||||
|
case "unit":
|
||||||
|
return todo();
|
||||||
|
case "tuple": {
|
||||||
|
const name = tyk.item.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} }`;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const name = tyk.item.ident.text;
|
return exhausted(datak);
|
||||||
const fields = rval.fields
|
|
||||||
.map((field, idx) =>
|
|
||||||
`${datak.fields[idx].ident!.text}: ${
|
|
||||||
this.operand(field)
|
|
||||||
}`
|
|
||||||
)
|
|
||||||
.join(", ");
|
|
||||||
return `${name} { ${fields} }`;
|
|
||||||
} else {
|
} else {
|
||||||
return todo();
|
return todo();
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user