slige/compiler/lowerer.ts

640 lines
21 KiB
TypeScript
Raw Normal View History

2024-12-15 00:07:36 +01:00
import { Builtins, Ops } from "./arch.ts";
2024-12-26 02:38:32 +01:00
import { Assembler, Label } from "./assembler.ts";
2024-12-31 00:12:34 +01:00
import { AnnoView, Expr, Stmt } from "./ast.ts";
2024-12-10 14:36:41 +01:00
import { LocalLeaf, Locals, LocalsFnRoot } from "./lowerer_locals.ts";
2024-12-26 02:38:32 +01:00
import { MonoCallNameGenMap, MonoFn, MonoFnsMap } from "./mono.ts";
2024-12-13 09:55:09 +01:00
import { Pos } from "./token.ts";
2024-12-26 02:38:32 +01:00
import { vtypeToString } from "./vtype.ts";
2024-12-10 09:03:03 +01:00
2024-12-15 00:07:36 +01:00
export type FnNamesMap = { [pc: number]: string };
2024-12-10 14:36:41 +01:00
export class Lowerer {
2024-12-17 02:10:11 +01:00
private program = Assembler.newRoot();
2024-12-10 09:03:03 +01:00
2024-12-26 02:38:32 +01:00
public constructor(
private monoFns: MonoFnsMap,
private callMap: MonoCallNameGenMap,
private lastPos: Pos,
) {}
2024-12-13 09:55:09 +01:00
2024-12-26 02:38:32 +01:00
public lower(): { program: number[]; fnNames: FnNamesMap } {
const fnLabelNameMap: FnLabelMap = {};
for (const nameGen in this.monoFns) {
fnLabelNameMap[nameGen] = nameGen;
2024-12-10 09:03:03 +01:00
}
2024-12-10 10:39:12 +01:00
2024-12-26 02:38:32 +01:00
this.addPrelimiary();
for (const fn of Object.values(this.monoFns)) {
const fnProgram = new MonoFnLowerer(
fn,
this.program.fork(),
this.callMap,
).lower();
this.program.join(fnProgram);
}
this.addConcluding();
2024-12-15 00:07:36 +01:00
const { program, locs } = this.program.assemble();
const fnNames: FnNamesMap = {};
for (const label in locs) {
2024-12-26 02:38:32 +01:00
if (label in fnLabelNameMap) {
fnNames[locs[label]] = fnLabelNameMap[label];
2024-12-15 00:07:36 +01:00
}
}
return { program, fnNames };
2024-12-10 10:39:12 +01:00
}
2024-12-26 02:38:32 +01:00
private addPrelimiary() {
this.addClearingSourceMap();
this.program.add(Ops.PushPtr, { label: "main" });
this.program.add(Ops.Call, 0);
this.program.add(Ops.PushPtr, { label: "_exit" });
this.program.add(Ops.Jump);
}
private addConcluding() {
this.program.setLabel({ label: "_exit" });
this.addSourceMap(this.lastPos);
this.program.add(Ops.Pop);
}
2024-12-13 09:55:09 +01:00
private addSourceMap({ index, line, col }: Pos) {
this.program.add(Ops.SourceMap, index, line, col);
}
2024-12-13 11:09:16 +01:00
private addClearingSourceMap() {
this.program.add(Ops.SourceMap, 0, 1, 1);
}
2024-12-25 05:19:32 +01:00
2024-12-26 02:38:32 +01:00
public printProgram() {
this.program.printProgram();
2024-12-25 05:19:32 +01:00
}
2024-12-26 02:38:32 +01:00
}
2024-12-25 05:19:32 +01:00
2024-12-26 02:38:32 +01:00
type FnLabelMap = { [nameGen: string]: string };
class MonoFnLowerer {
private locals: Locals = new LocalsFnRoot();
private returnStack: Label[] = [];
private breakStack: Label[] = [];
public constructor(
private fn: MonoFn,
private program: Assembler,
private callMap: MonoCallNameGenMap,
) {}
public lower(): Assembler {
this.lowerFnStmt(this.fn.stmt);
return this.program;
}
private lowerFnStmt(stmt: Stmt) {
if (stmt.kind.type !== "fn") {
throw new Error();
}
const label = this.fn.nameGen;
this.program.setLabel({ label });
this.addSourceMap(stmt.pos);
const outerLocals = this.locals;
const fnRoot = new LocalsFnRoot(outerLocals);
const outerProgram = this.program;
const returnLabel = this.program.makeLabel();
this.returnStack.push(returnLabel);
this.program = outerProgram.fork();
this.locals = fnRoot;
for (const { ident } of stmt.kind.params) {
this.locals.allocSym(ident);
}
2024-12-31 00:12:34 +01:00
const annos = new AnnoView(stmt.details);
if (annos.has("builtin")) {
const anno = annos.get("builtin");
if (!anno) {
throw new Error();
}
this.lowerFnBuiltinBody(anno.args);
} else if (annos.has("remainder")) {
2024-12-26 02:38:32 +01:00
this.program.add(Ops.Remainder);
} else {
this.lowerExpr(stmt.kind.body);
}
2024-12-26 02:38:32 +01:00
this.locals = outerLocals;
const localAmount = fnRoot.stackReserved() -
stmt.kind.params.length;
for (let i = 0; i < localAmount; ++i) {
outerProgram.add(Ops.PushNull);
}
this.returnStack.pop();
this.program.setLabel(returnLabel);
this.program.add(Ops.Return);
outerProgram.join(this.program);
this.program = outerProgram;
}
private addSourceMap({ index, line, col }: Pos) {
this.program.add(Ops.SourceMap, index, line, col);
}
private addClearingSourceMap() {
this.program.add(Ops.SourceMap, 0, 1, 1);
2024-12-10 09:03:03 +01:00
}
2024-12-10 14:36:41 +01:00
private lowerStmt(stmt: Stmt) {
2024-12-10 09:03:03 +01:00
switch (stmt.kind.type) {
case "error":
2024-12-10 23:30:15 +01:00
break;
2024-12-10 09:03:03 +01:00
case "break":
2024-12-10 23:30:15 +01:00
return this.lowerBreakStmt(stmt);
2024-12-10 09:03:03 +01:00
case "return":
2024-12-15 04:24:07 +01:00
return this.lowerReturnStmt(stmt);
2024-12-10 14:36:41 +01:00
case "fn":
return this.lowerFnStmt(stmt);
2024-12-10 09:03:03 +01:00
case "let":
return this.lowerLetStmt(stmt);
case "assign":
2024-12-10 23:30:15 +01:00
return this.lowerAssignStmt(stmt);
2024-12-10 09:03:03 +01:00
case "expr":
2024-12-10 14:36:41 +01:00
this.lowerExpr(stmt.kind.expr);
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Pop);
2024-12-10 14:36:41 +01:00
return;
2024-12-10 10:39:12 +01:00
}
2024-12-10 14:36:41 +01:00
throw new Error(`unhandled stmt '${stmt.kind.type}'`);
2024-12-10 09:03:03 +01:00
}
2024-12-10 23:30:15 +01:00
private lowerAssignStmt(stmt: Stmt) {
if (stmt.kind.type !== "assign") {
throw new Error();
}
this.lowerExpr(stmt.kind.value);
switch (stmt.kind.subject.kind.type) {
case "field": {
this.lowerExpr(stmt.kind.subject.kind.subject);
this.program.add(Ops.PushString, stmt.kind.subject.kind.ident);
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Builtin, Builtins.StructSet);
2024-12-10 23:30:15 +01:00
return;
}
case "index": {
this.lowerExpr(stmt.kind.subject.kind.subject);
this.lowerExpr(stmt.kind.subject.kind.value);
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Builtin, Builtins.ArraySet);
2024-12-10 23:30:15 +01:00
return;
}
case "sym": {
2024-12-11 00:03:19 +01:00
this.program.add(
Ops.StoreLocal,
2024-12-10 23:30:15 +01:00
this.locals.symId(stmt.kind.subject.kind.sym.ident),
);
return;
}
default:
throw new Error();
}
}
2024-12-15 04:24:07 +01:00
private lowerReturnStmt(stmt: Stmt) {
if (stmt.kind.type !== "return") {
throw new Error();
}
if (stmt.kind.expr) {
this.lowerExpr(stmt.kind.expr);
}
this.addClearingSourceMap();
this.program.add(Ops.PushPtr, this.returnStack.at(-1)!);
this.program.add(Ops.Jump);
}
2024-12-10 23:30:15 +01:00
private lowerBreakStmt(stmt: Stmt) {
if (stmt.kind.type !== "break") {
throw new Error();
}
if (stmt.kind.expr) {
this.lowerExpr(stmt.kind.expr);
}
2024-12-13 11:09:16 +01:00
this.addClearingSourceMap();
2024-12-13 06:09:10 +01:00
this.program.add(Ops.PushPtr, this.breakStack.at(-1)!);
this.program.add(Ops.Jump);
2024-12-10 23:30:15 +01:00
}
2024-12-13 09:55:09 +01:00
private lowerFnBuiltinBody(annoArgs: Expr[]) {
if (annoArgs.length !== 1) {
throw new Error("invalid # of arguments to builtin annotation");
}
const anno = annoArgs[0];
if (anno.kind.type !== "ident") {
throw new Error(
`unexpected argument type '${anno.kind.type}' expected 'ident'`,
);
}
const value = anno.kind.ident;
2024-12-13 09:55:09 +01:00
const builtin = Object.entries(Builtins).find((entry) =>
entry[0] === value
)?.[1];
if (builtin === undefined) {
throw new Error(
`unrecognized builtin '${value}'`,
);
}
this.program.add(Ops.Builtin, builtin);
}
2024-12-10 14:36:41 +01:00
private lowerLetStmt(stmt: Stmt) {
2024-12-10 09:03:03 +01:00
if (stmt.kind.type !== "let") {
throw new Error();
}
2024-12-10 10:39:12 +01:00
this.lowerExpr(stmt.kind.value);
2024-12-11 00:03:19 +01:00
this.locals.allocSym(stmt.kind.param.ident);
this.program.add(
Ops.StoreLocal,
this.locals.symId(stmt.kind.param.ident),
);
2024-12-10 09:03:03 +01:00
}
2024-12-10 14:36:41 +01:00
private lowerExpr(expr: Expr) {
2024-12-10 09:03:03 +01:00
switch (expr.kind.type) {
case "error":
break;
2024-12-10 14:36:41 +01:00
case "sym":
return this.lowerSymExpr(expr);
case "null":
break;
2024-12-10 09:03:03 +01:00
case "int":
2024-12-10 14:36:41 +01:00
return this.lowerIntExpr(expr);
case "bool":
2024-12-15 04:24:07 +01:00
return this.lowerBoolExpr(expr);
2024-12-10 14:36:41 +01:00
case "string":
return this.lowerStringExpr(expr);
2024-12-10 09:03:03 +01:00
case "ident":
2024-12-10 14:36:41 +01:00
break;
2024-12-10 09:03:03 +01:00
case "group":
2024-12-13 06:09:10 +01:00
return void this.lowerExpr(expr.kind.expr);
2024-12-10 09:03:03 +01:00
case "field":
2024-12-31 03:38:38 +01:00
return this.lowerFieldExpr(expr);
2024-12-10 09:03:03 +01:00
case "index":
2024-12-13 23:22:08 +01:00
return this.lowerIndexExpr(expr);
2024-12-10 09:03:03 +01:00
case "call":
2024-12-10 14:36:41 +01:00
return this.lowerCallExpr(expr);
2024-12-26 01:51:05 +01:00
case "etype_args":
return this.lowerETypeArgsExpr(expr);
2024-12-10 09:03:03 +01:00
case "unary":
2024-12-13 06:09:10 +01:00
return this.lowerUnaryExpr(expr);
2024-12-10 09:03:03 +01:00
case "binary":
2024-12-10 10:39:12 +01:00
return this.lowerBinaryExpr(expr);
2024-12-10 09:03:03 +01:00
case "if":
2024-12-10 14:36:41 +01:00
return this.lowerIfExpr(expr);
2024-12-10 09:03:03 +01:00
case "loop":
2024-12-10 23:30:15 +01:00
return this.lowerLoopExpr(expr);
2024-12-10 14:36:41 +01:00
case "block":
return this.lowerBlockExpr(expr);
}
throw new Error(`unhandled expr '${expr.kind.type}'`);
}
2024-12-31 03:38:38 +01:00
private lowerFieldExpr(expr: Expr) {
if (expr.kind.type !== "field") {
throw new Error();
}
this.lowerExpr(expr.kind.subject);
this.program.add(Ops.PushString, expr.kind.ident);
if (expr.kind.subject.vtype?.type == "struct") {
this.program.add(Ops.Builtin, Builtins.StructAt);
return;
}
throw new Error(`unhandled field subject type '${expr.kind.subject}'`);
}
2024-12-13 16:03:01 +01:00
private lowerIndexExpr(expr: Expr) {
if (expr.kind.type !== "index") {
throw new Error();
}
2024-12-13 23:22:08 +01:00
this.lowerExpr(expr.kind.subject);
this.lowerExpr(expr.kind.value);
2024-12-13 16:03:01 +01:00
if (expr.kind.subject.vtype?.type == "array") {
this.program.add(Ops.Builtin, Builtins.ArrayAt);
return;
}
if (expr.kind.subject.vtype?.type == "string") {
this.program.add(Ops.Builtin, Builtins.StringCharAt);
return;
}
throw new Error(`unhandled index subject type '${expr.kind.subject}'`);
}
2024-12-10 14:36:41 +01:00
private lowerSymExpr(expr: Expr) {
if (expr.kind.type !== "sym") {
throw new Error();
}
if (expr.kind.sym.type === "let") {
2024-12-13 09:55:09 +01:00
const symId = this.locals.symId(expr.kind.ident);
this.program.add(Ops.LoadLocal, symId);
2024-12-10 14:36:41 +01:00
return;
}
if (expr.kind.sym.type === "fn_param") {
2024-12-11 00:03:19 +01:00
this.program.add(
2024-12-10 14:36:41 +01:00
Ops.LoadLocal,
this.locals.symId(expr.kind.ident),
);
return;
2024-12-10 09:03:03 +01:00
}
2024-12-10 14:36:41 +01:00
if (expr.kind.sym.type === "fn") {
2024-12-26 02:38:32 +01:00
// Is this smart? Well, my presumption is
// that it isn't. The underlying problem, which
// this solutions raison d'être is to solve, is
// that the compiler, as it d'être's currently
// doesn't support checking and infering generic
// fn args all the way down to the sym. Therefore,
// when a sym is checked in a call expr, we can't
// really do anything useful. Instead the actual
// function pointer pointing to the actual
// monomorphized function is emplaced when
// lowering the call expression itself. But what
// should we do then, if the user decides to
// assign a function to a local? You might ask.
// You see, that's where the problem lies.
// My current, very thought out solution, as
// you can read below, is to push a null pointer,
// for it to then be replaced later. This will
// probably cause many hastles in the future
// for myself in particular, when trying to
// decipher the lowerer's output. So if you're
// the unlucky girl, who has tried for ages to
// decipher why a zero value is pushed and then
// later replaced, and then you finally
// stumbled upon this here implementation,
// let me first say, I'm so sorry. At the time
// of writing, I really haven't thought out
// very well, how the generic call system should
// work, and it's therefore a bit flaky, and the
// implementation kinda looks like it was
// implementated by a girl who didn't really
// understand very well what they were
// implementing at the time that they were
// implementing it. Anyway, I just wanted to
// apologize. Happy coding.
// -Your favorite compiler girl.
this.program.add(Ops.PushPtr, 0);
2024-12-10 14:36:41 +01:00
return;
}
throw new Error(`unhandled sym type '${expr.kind.sym.type}'`);
2024-12-10 09:03:03 +01:00
}
2024-12-10 14:36:41 +01:00
private lowerIntExpr(expr: Expr) {
2024-12-10 09:03:03 +01:00
if (expr.kind.type !== "int") {
throw new Error();
}
2024-12-11 00:03:19 +01:00
this.program.add(Ops.PushInt, expr.kind.value);
2024-12-10 09:03:03 +01:00
}
2024-12-15 04:24:07 +01:00
private lowerBoolExpr(expr: Expr) {
if (expr.kind.type !== "bool") {
throw new Error();
}
this.program.add(Ops.PushBool, expr.kind.value);
}
2024-12-10 14:36:41 +01:00
private lowerStringExpr(expr: Expr) {
if (expr.kind.type !== "string") {
2024-12-10 09:03:03 +01:00
throw new Error();
}
2024-12-11 00:03:19 +01:00
this.program.add(Ops.PushString, expr.kind.value);
2024-12-10 09:03:03 +01:00
}
2024-12-13 06:09:10 +01:00
private lowerUnaryExpr(expr: Expr) {
if (expr.kind.type !== "unary") {
throw new Error();
}
this.lowerExpr(expr.kind.subject);
const vtype = expr.kind.subject.vtype!;
if (vtype.type === "bool") {
switch (expr.kind.unaryType) {
case "not":
this.program.add(Ops.Not);
return;
default:
}
}
if (vtype.type === "int") {
switch (expr.kind.unaryType) {
case "-": {
this.program.add(Ops.PushInt, 0);
2024-12-17 02:10:11 +01:00
this.program.add(Ops.Swap);
this.program.add(Ops.Subtract);
return;
}
default:
}
}
2024-12-13 06:09:10 +01:00
throw new Error(
`unhandled unary` +
` '${vtypeToString(expr.vtype!)}' aka. ` +
` ${expr.kind.unaryType}` +
` '${vtypeToString(expr.kind.subject.vtype!)}'`,
);
}
2024-12-10 14:36:41 +01:00
private lowerBinaryExpr(expr: Expr) {
2024-12-10 09:03:03 +01:00
if (expr.kind.type !== "binary") {
throw new Error();
}
2024-12-17 02:10:11 +01:00
const vtype = expr.kind.left.vtype!;
if (vtype.type === "bool") {
if (["or", "and"].includes(expr.kind.binaryType)) {
const shortCircuitLabel = this.program.makeLabel();
this.lowerExpr(expr.kind.left);
this.program.add(Ops.Duplicate);
if (expr.kind.binaryType === "and") {
this.program.add(Ops.Not);
}
this.program.add(Ops.PushPtr, shortCircuitLabel);
this.program.add(Ops.JumpIfTrue);
this.program.add(Ops.Pop);
this.lowerExpr(expr.kind.right);
this.program.setLabel(shortCircuitLabel);
return;
}
}
2024-12-10 09:03:03 +01:00
this.lowerExpr(expr.kind.left);
this.lowerExpr(expr.kind.right);
2024-12-10 23:30:15 +01:00
if (vtype.type === "int") {
2024-12-10 09:03:03 +01:00
switch (expr.kind.binaryType) {
case "+":
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Add);
2024-12-10 10:39:12 +01:00
return;
2024-12-12 16:07:59 +01:00
case "-":
this.program.add(Ops.Subtract);
return;
2024-12-10 09:03:03 +01:00
case "*":
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Multiply);
2024-12-10 10:39:12 +01:00
return;
2024-12-17 02:10:11 +01:00
case "/":
this.program.add(Ops.Multiply);
return;
2024-12-10 09:03:03 +01:00
case "==":
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Equal);
2024-12-10 23:30:15 +01:00
return;
case "!=":
this.program.add(Ops.Equal);
this.program.add(Ops.Not);
return;
2024-12-13 06:09:10 +01:00
case "<":
this.program.add(Ops.LessThan);
return;
2024-12-17 02:10:11 +01:00
case ">":
this.program.add(Ops.Swap);
this.program.add(Ops.LessThan);
return;
case "<=":
this.program.add(Ops.Swap);
this.program.add(Ops.LessThan);
this.program.add(Ops.Not);
return;
2024-12-10 09:03:03 +01:00
case ">=":
2024-12-11 00:03:19 +01:00
this.program.add(Ops.LessThan);
this.program.add(Ops.Not);
2024-12-10 23:30:15 +01:00
return;
default:
2024-12-10 09:03:03 +01:00
}
}
2024-12-26 02:38:32 +01:00
if (vtype.type === "bool") {
switch (expr.kind.binaryType) {
case "==":
this.program.add(Ops.And);
return;
case "!=":
this.program.add(Ops.And);
this.program.add(Ops.Not);
return;
default:
}
}
2024-12-10 23:30:15 +01:00
if (vtype.type === "string") {
2024-12-10 14:36:41 +01:00
if (expr.kind.binaryType === "+") {
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Builtin, Builtins.StringConcat);
2024-12-10 23:30:15 +01:00
return;
}
if (expr.kind.binaryType === "==") {
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Builtin, Builtins.StringEqual);
2024-12-10 23:30:15 +01:00
return;
}
if (expr.kind.binaryType === "!=") {
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Builtin, Builtins.StringEqual);
this.program.add(Ops.Not);
2024-12-10 14:36:41 +01:00
return;
}
}
2024-12-10 10:39:12 +01:00
throw new Error(
2024-12-10 14:36:41 +01:00
`unhandled binaryType` +
2024-12-10 23:30:15 +01:00
` '${vtypeToString(expr.vtype!)}' aka. ` +
2024-12-10 14:36:41 +01:00
` '${vtypeToString(expr.kind.left.vtype!)}'` +
` ${expr.kind.binaryType}` +
` '${vtypeToString(expr.kind.left.vtype!)}'`,
2024-12-10 10:39:12 +01:00
);
2024-12-10 09:03:03 +01:00
}
2024-12-10 14:36:41 +01:00
private lowerCallExpr(expr: Expr) {
if (expr.kind.type !== "call") {
throw new Error();
}
for (const arg of expr.kind.args) {
this.lowerExpr(arg);
}
this.lowerExpr(expr.kind.subject);
2024-12-26 02:38:32 +01:00
this.program.add(Ops.Pop);
this.program.add(Ops.PushPtr, { label: this.callMap[expr.id] });
2024-12-11 03:11:00 +01:00
this.program.add(Ops.Call, expr.kind.args.length);
2024-12-10 14:36:41 +01:00
}
2024-12-26 01:51:05 +01:00
private lowerETypeArgsExpr(expr: Expr) {
if (expr.kind.type !== "etype_args") {
throw new Error();
}
2024-12-26 02:38:32 +01:00
this.lowerExpr(expr.kind.subject);
2024-12-26 01:51:05 +01:00
}
2024-12-10 14:36:41 +01:00
private lowerIfExpr(expr: Expr) {
if (expr.kind.type !== "if") {
throw new Error();
}
2024-12-10 15:45:19 +01:00
const falseLabel = this.program.makeLabel();
const doneLabel = this.program.makeLabel();
2024-12-10 14:36:41 +01:00
this.lowerExpr(expr.kind.cond);
2024-12-10 15:45:19 +01:00
2024-12-11 00:03:19 +01:00
this.program.add(Ops.Not);
2024-12-13 11:09:16 +01:00
this.addClearingSourceMap();
2024-12-11 03:11:00 +01:00
this.program.add(Ops.PushPtr, falseLabel);
this.program.add(Ops.JumpIfTrue);
2024-12-10 15:45:19 +01:00
2024-12-13 10:26:34 +01:00
this.addSourceMap(expr.kind.truthy.pos);
2024-12-10 14:36:41 +01:00
this.lowerExpr(expr.kind.truthy);
2024-12-10 15:45:19 +01:00
2024-12-13 11:09:16 +01:00
this.addClearingSourceMap();
2024-12-11 03:11:00 +01:00
this.program.add(Ops.PushPtr, doneLabel);
this.program.add(Ops.Jump);
2024-12-10 15:45:19 +01:00
this.program.setLabel(falseLabel);
2024-12-10 23:30:15 +01:00
if (expr.kind.falsy) {
2024-12-17 10:24:18 +01:00
this.addSourceMap(expr.kind.elsePos!);
2024-12-13 10:33:51 +01:00
this.lowerExpr(expr.kind.falsy);
2024-12-11 12:36:19 +01:00
} else {
this.program.add(Ops.PushNull);
2024-12-10 23:30:15 +01:00
}
2024-12-10 15:45:19 +01:00
this.program.setLabel(doneLabel);
2024-12-10 14:36:41 +01:00
}
2024-12-10 23:30:15 +01:00
private lowerLoopExpr(expr: Expr) {
if (expr.kind.type !== "loop") {
throw new Error();
}
2024-12-13 12:12:16 +01:00
const continueLabel = this.program.makeLabel();
2024-12-10 23:30:15 +01:00
const breakLabel = this.program.makeLabel();
this.breakStack.push(breakLabel);
2024-12-13 12:12:16 +01:00
this.program.setLabel(continueLabel);
2024-12-13 10:26:34 +01:00
this.addSourceMap(expr.kind.body.pos);
2024-12-10 23:30:15 +01:00
this.lowerExpr(expr.kind.body);
2024-12-13 06:09:10 +01:00
this.program.add(Ops.Pop);
2024-12-13 11:09:16 +01:00
this.addClearingSourceMap();
2024-12-13 12:12:16 +01:00
this.program.add(Ops.PushPtr, continueLabel);
2024-12-11 03:11:00 +01:00
this.program.add(Ops.Jump);
2024-12-10 23:30:15 +01:00
this.program.setLabel(breakLabel);
if (expr.vtype!.type === "null") {
2024-12-11 00:03:19 +01:00
this.program.add(Ops.PushNull);
2024-12-10 23:30:15 +01:00
}
this.breakStack.pop();
}
2024-12-10 14:36:41 +01:00
private lowerBlockExpr(expr: Expr) {
if (expr.kind.type !== "block") {
throw new Error();
}
const outerLocals = this.locals;
this.locals = new LocalLeaf(this.locals);
for (const stmt of expr.kind.stmts) {
2024-12-13 12:12:16 +01:00
this.addSourceMap(stmt.pos);
2024-12-10 14:36:41 +01:00
this.lowerStmt(stmt);
}
if (expr.kind.expr) {
2024-12-13 12:12:16 +01:00
this.addSourceMap(expr.kind.expr.pos);
2024-12-10 14:36:41 +01:00
this.lowerExpr(expr.kind.expr);
} else {
2024-12-11 00:03:19 +01:00
this.program.add(Ops.PushNull);
2024-12-10 14:36:41 +01:00
}
this.locals = outerLocals;
}
2024-12-10 10:39:12 +01:00
}