mirror of
https://github.com/Mercantec-GHC/h4-projekt-gruppe-0-sm.git
synced 2025-04-28 00:34:06 +02:00
compiler: lower enums to mir
This commit is contained in:
parent
4c4b0da238
commit
66fc0ec0b5
@ -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();
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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 =
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -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))
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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" }
|
||||||
|
Loading…
x
Reference in New Issue
Block a user