712 lines
20 KiB
TypeScript
712 lines
20 KiB
TypeScript
import type {
|
|
ASTCtx,
|
|
LuaExpression,
|
|
LuaLValue,
|
|
LuaStatement,
|
|
} from "$common/space_lua/ast.ts";
|
|
import { evalPromiseValues } from "$common/space_lua/util.ts";
|
|
import {
|
|
luaCall,
|
|
luaSet,
|
|
type LuaStackFrame,
|
|
} from "$common/space_lua/runtime.ts";
|
|
import {
|
|
type ILuaFunction,
|
|
type ILuaGettable,
|
|
type ILuaSettable,
|
|
LuaBreak,
|
|
LuaEnv,
|
|
LuaFunction,
|
|
luaGet,
|
|
luaLen,
|
|
type LuaLValueContainer,
|
|
LuaMultiRes,
|
|
LuaReturn,
|
|
LuaRuntimeError,
|
|
LuaTable,
|
|
luaToString,
|
|
luaTruthy,
|
|
type LuaValue,
|
|
singleResult,
|
|
} from "./runtime.ts";
|
|
|
|
export function evalExpression(
|
|
e: LuaExpression,
|
|
env: LuaEnv,
|
|
sf: LuaStackFrame,
|
|
): Promise<LuaValue> | LuaValue {
|
|
try {
|
|
switch (e.type) {
|
|
case "String":
|
|
return e.value;
|
|
case "Number":
|
|
return e.value;
|
|
case "Boolean":
|
|
return e.value;
|
|
case "Nil":
|
|
return null;
|
|
case "Binary": {
|
|
const values = evalPromiseValues([
|
|
evalExpression(e.left, env, sf),
|
|
evalExpression(e.right, env, sf),
|
|
]);
|
|
if (values instanceof Promise) {
|
|
return values.then(([left, right]) =>
|
|
luaOp(
|
|
e.operator,
|
|
singleResult(left),
|
|
singleResult(right),
|
|
e.ctx,
|
|
sf,
|
|
)
|
|
);
|
|
} else {
|
|
return luaOp(
|
|
e.operator,
|
|
singleResult(values[0]),
|
|
singleResult(values[1]),
|
|
e.ctx,
|
|
sf,
|
|
);
|
|
}
|
|
}
|
|
case "Unary": {
|
|
const value = evalExpression(e.argument, env, sf);
|
|
if (value instanceof Promise) {
|
|
return value.then((value) => {
|
|
switch (e.operator) {
|
|
case "-":
|
|
return -singleResult(value);
|
|
case "+":
|
|
return +singleResult(value);
|
|
case "not":
|
|
return !singleResult(value);
|
|
case "#":
|
|
return luaLen(singleResult(value));
|
|
default:
|
|
throw new Error(
|
|
`Unknown unary operator ${e.operator}`,
|
|
);
|
|
}
|
|
});
|
|
} else {
|
|
switch (e.operator) {
|
|
case "-":
|
|
return -singleResult(value);
|
|
case "+":
|
|
return +singleResult(value);
|
|
case "not":
|
|
return !singleResult(value);
|
|
case "#":
|
|
return luaLen(singleResult(value));
|
|
default:
|
|
throw new Error(
|
|
`Unknown unary operator ${e.operator}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
case "Variable":
|
|
case "FunctionCall":
|
|
case "TableAccess":
|
|
case "PropertyAccess":
|
|
return evalPrefixExpression(e, env, sf);
|
|
case "TableConstructor": {
|
|
const table = new LuaTable();
|
|
const promises: Promise<void>[] = [];
|
|
for (const field of e.fields) {
|
|
switch (field.type) {
|
|
case "PropField": {
|
|
const value = evalExpression(field.value, env, sf);
|
|
if (value instanceof Promise) {
|
|
promises.push(value.then((value) => {
|
|
table.set(
|
|
field.key,
|
|
singleResult(value),
|
|
sf,
|
|
);
|
|
}));
|
|
} else {
|
|
table.set(field.key, singleResult(value), sf);
|
|
}
|
|
break;
|
|
}
|
|
case "DynamicField": {
|
|
const key = evalExpression(field.key, env, sf);
|
|
const value = evalExpression(field.value, env, sf);
|
|
if (
|
|
key instanceof Promise || value instanceof Promise
|
|
) {
|
|
promises.push(
|
|
Promise.all([
|
|
key instanceof Promise ? key : Promise.resolve(key),
|
|
value instanceof Promise ? value : Promise.resolve(value),
|
|
]).then(([key, value]) => {
|
|
table.set(
|
|
singleResult(key),
|
|
singleResult(value),
|
|
sf,
|
|
);
|
|
}),
|
|
);
|
|
} else {
|
|
table.set(
|
|
singleResult(key),
|
|
singleResult(value),
|
|
sf,
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
case "ExpressionField": {
|
|
const value = evalExpression(field.value, env, sf);
|
|
if (value instanceof Promise) {
|
|
promises.push(value.then((value) => {
|
|
// +1 because Lua tables are 1-indexed
|
|
table.set(
|
|
table.length + 1,
|
|
singleResult(value),
|
|
sf,
|
|
);
|
|
}));
|
|
} else {
|
|
// +1 because Lua tables are 1-indexed
|
|
table.set(
|
|
table.length + 1,
|
|
singleResult(value),
|
|
sf,
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (promises.length > 0) {
|
|
return Promise.all(promises).then(() => table);
|
|
} else {
|
|
return table;
|
|
}
|
|
}
|
|
case "FunctionDefinition": {
|
|
return new LuaFunction(e.body, env);
|
|
}
|
|
default:
|
|
throw new Error(`Unknown expression type ${e.type}`);
|
|
}
|
|
} catch (err: any) {
|
|
// Repackage any non Lua-specific exceptions with some position information
|
|
if (!err.constructor.name.startsWith("Lua")) {
|
|
throw new LuaRuntimeError(err.message, sf.withCtx(e.ctx), err);
|
|
} else {
|
|
throw err;
|
|
}
|
|
}
|
|
}
|
|
|
|
function evalPrefixExpression(
|
|
e: LuaExpression,
|
|
env: LuaEnv,
|
|
sf: LuaStackFrame,
|
|
): Promise<LuaValue> | LuaValue {
|
|
switch (e.type) {
|
|
case "Variable": {
|
|
const value = env.get(e.name);
|
|
if (value === undefined) {
|
|
return null;
|
|
} else {
|
|
return value;
|
|
}
|
|
}
|
|
case "Parenthesized":
|
|
return evalExpression(e.expression, env, sf);
|
|
// <<expr>>[<<expr>>]
|
|
case "TableAccess": {
|
|
const values = evalPromiseValues([
|
|
evalPrefixExpression(e.object, env, sf),
|
|
evalExpression(e.key, env, sf),
|
|
]);
|
|
if (values instanceof Promise) {
|
|
return values.then(([table, key]) => {
|
|
table = singleResult(table);
|
|
key = singleResult(key);
|
|
|
|
return luaGet(table, key, sf.withCtx(e.ctx));
|
|
});
|
|
} else {
|
|
const table = singleResult(values[0]);
|
|
const key = singleResult(values[1]);
|
|
return luaGet(table, singleResult(key), sf.withCtx(e.ctx));
|
|
}
|
|
}
|
|
// <expr>.property
|
|
case "PropertyAccess": {
|
|
const obj = evalPrefixExpression(e.object, env, sf);
|
|
if (obj instanceof Promise) {
|
|
return obj.then((obj) => {
|
|
return luaGet(obj, e.property, sf.withCtx(e.ctx));
|
|
});
|
|
} else {
|
|
return luaGet(obj, e.property, sf.withCtx(e.ctx));
|
|
}
|
|
}
|
|
case "FunctionCall": {
|
|
let prefixValue = evalPrefixExpression(e.prefix, env, sf);
|
|
if (!prefixValue) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to call nil as a function`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
}
|
|
if (prefixValue instanceof Promise) {
|
|
return prefixValue.then((prefixValue) => {
|
|
if (!prefixValue) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to call a nil value`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
}
|
|
let selfArgs: LuaValue[] = [];
|
|
// Handling a:b() syntax (b is kept in .name)
|
|
if (e.name && !prefixValue.get) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to index a non-table: ${prefixValue}`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
} else if (e.name) {
|
|
// Two things need to happen: the actual function be called needs to be looked up in the table, and the table itself needs to be passed as the first argument
|
|
selfArgs = [prefixValue];
|
|
prefixValue = prefixValue.get(e.name);
|
|
}
|
|
if (!prefixValue.call) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to call ${prefixValue} as a function`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
}
|
|
const args = evalPromiseValues(
|
|
e.args.map((arg) => evalExpression(arg, env, sf)),
|
|
);
|
|
if (args instanceof Promise) {
|
|
return args.then((args) =>
|
|
luaCall(prefixValue, [...selfArgs, ...args], e.ctx, sf)
|
|
);
|
|
} else {
|
|
return luaCall(prefixValue, [...selfArgs, ...args], e.ctx, sf);
|
|
}
|
|
});
|
|
} else {
|
|
let selfArgs: LuaValue[] = [];
|
|
// Handling a:b() syntax (b is kept in .name)
|
|
if (e.name && !prefixValue.get) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to index a non-table: ${prefixValue}`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
} else if (e.name) {
|
|
// Two things need to happen: the actual function be called needs to be looked up in the table, and the table itself needs to be passed as the first argument
|
|
selfArgs = [prefixValue];
|
|
prefixValue = prefixValue.get(e.name);
|
|
}
|
|
if (!prefixValue.call) {
|
|
throw new LuaRuntimeError(
|
|
`Attempting to call ${prefixValue} as a function`,
|
|
sf.withCtx(e.prefix.ctx),
|
|
);
|
|
}
|
|
const args = evalPromiseValues(
|
|
e.args.map((arg) => evalExpression(arg, env, sf)),
|
|
);
|
|
if (args instanceof Promise) {
|
|
return args.then((args) =>
|
|
luaCall(prefixValue, [...selfArgs, ...args], e.ctx, sf)
|
|
);
|
|
} else {
|
|
return luaCall(prefixValue, [...selfArgs, ...args], e.ctx, sf);
|
|
}
|
|
}
|
|
}
|
|
default:
|
|
throw new Error(`Unknown prefix expression type ${e.type}`);
|
|
}
|
|
}
|
|
|
|
// Mapping table of operators meta-methods to their corresponding operator
|
|
|
|
type LuaMetaMethod = Record<string, {
|
|
metaMethod?: string;
|
|
nativeImplementation: (
|
|
a: LuaValue,
|
|
b: LuaValue,
|
|
ctx: ASTCtx,
|
|
sf: LuaStackFrame,
|
|
) => LuaValue;
|
|
}>;
|
|
|
|
const operatorsMetaMethods: LuaMetaMethod = {
|
|
"+": {
|
|
metaMethod: "__add",
|
|
nativeImplementation: (a, b) => a + b,
|
|
},
|
|
"-": {
|
|
metaMethod: "__sub",
|
|
nativeImplementation: (a, b) => a - b,
|
|
},
|
|
"*": {
|
|
metaMethod: "__mul",
|
|
nativeImplementation: (a, b) => a * b,
|
|
},
|
|
"/": {
|
|
metaMethod: "__div",
|
|
nativeImplementation: (a, b) => a / b,
|
|
},
|
|
"//": {
|
|
metaMethod: "__idiv",
|
|
nativeImplementation: (a, b) => Math.floor(a / b),
|
|
},
|
|
"%": {
|
|
metaMethod: "__mod",
|
|
nativeImplementation: (a, b) => a % b,
|
|
},
|
|
"^": {
|
|
metaMethod: "__pow",
|
|
nativeImplementation: (a, b) => a ** b,
|
|
},
|
|
"..": {
|
|
metaMethod: "__concat",
|
|
nativeImplementation: (a, b) => {
|
|
const aString = luaToString(a);
|
|
const bString = luaToString(b);
|
|
if (aString instanceof Promise || bString instanceof Promise) {
|
|
return Promise.all([aString, bString]).then(([aString, bString]) =>
|
|
aString + bString
|
|
);
|
|
} else {
|
|
return aString + bString;
|
|
}
|
|
},
|
|
},
|
|
"==": {
|
|
metaMethod: "__eq",
|
|
nativeImplementation: (a, b) => a === b,
|
|
},
|
|
"~=": {
|
|
metaMethod: "__ne",
|
|
nativeImplementation: (a, b) => a !== b,
|
|
},
|
|
"!=": {
|
|
metaMethod: "__ne",
|
|
nativeImplementation: (a, b) => a !== b,
|
|
},
|
|
"<": {
|
|
metaMethod: "__lt",
|
|
nativeImplementation: (a, b) => a < b,
|
|
},
|
|
"<=": {
|
|
metaMethod: "__le",
|
|
nativeImplementation: (a, b) => a <= b,
|
|
},
|
|
">": {
|
|
nativeImplementation: (a, b, ctx, sf) => !luaOp("<=", a, b, ctx, sf),
|
|
},
|
|
">=": {
|
|
nativeImplementation: (a, b, ctx, cf) => !luaOp("<", a, b, ctx, cf),
|
|
},
|
|
and: {
|
|
metaMethod: "__and",
|
|
nativeImplementation: (a, b) => a && b,
|
|
},
|
|
or: {
|
|
metaMethod: "__or",
|
|
nativeImplementation: (a, b) => a || b,
|
|
},
|
|
};
|
|
|
|
function luaOp(
|
|
op: string,
|
|
left: any,
|
|
right: any,
|
|
ctx: ASTCtx,
|
|
sf: LuaStackFrame,
|
|
): any {
|
|
const operatorHandler = operatorsMetaMethods[op];
|
|
if (!operatorHandler) {
|
|
throw new LuaRuntimeError(`Unknown operator ${op}`, sf.withCtx(ctx));
|
|
}
|
|
if (operatorHandler.metaMethod) {
|
|
if (left?.metatable?.has(operatorHandler.metaMethod)) {
|
|
const fn = left.metatable.get(operatorHandler.metaMethod);
|
|
return luaCall(fn, [left, right], ctx, sf);
|
|
} else if (right?.metatable?.has(operatorHandler.metaMethod)) {
|
|
const fn = right.metatable.get(operatorHandler.metaMethod);
|
|
return luaCall(fn, [left, right], ctx, sf);
|
|
}
|
|
}
|
|
return operatorHandler.nativeImplementation(left, right, ctx, sf);
|
|
}
|
|
|
|
async function evalExpressions(
|
|
es: LuaExpression[],
|
|
env: LuaEnv,
|
|
sf: LuaStackFrame,
|
|
): Promise<LuaValue[]> {
|
|
return new LuaMultiRes(
|
|
await Promise.all(es.map((e) => evalExpression(e, env, sf))),
|
|
).flatten().values;
|
|
}
|
|
|
|
export async function evalStatement(
|
|
s: LuaStatement,
|
|
env: LuaEnv,
|
|
sf: LuaStackFrame,
|
|
): Promise<void> {
|
|
switch (s.type) {
|
|
case "Assignment": {
|
|
const values = await evalExpressions(s.expressions, env, sf);
|
|
const lvalues = await evalPromiseValues(s.variables
|
|
.map((lval) => evalLValue(lval, env, sf)));
|
|
|
|
for (let i = 0; i < lvalues.length; i++) {
|
|
luaSet(lvalues[i].env, lvalues[i].key, values[i], sf.withCtx(s.ctx));
|
|
}
|
|
|
|
break;
|
|
}
|
|
case "Local": {
|
|
if (s.expressions) {
|
|
const values = await evalExpressions(s.expressions, env, sf);
|
|
for (let i = 0; i < s.names.length; i++) {
|
|
env.setLocal(s.names[i].name, values[i]);
|
|
}
|
|
} else {
|
|
for (let i = 0; i < s.names.length; i++) {
|
|
env.setLocal(s.names[i].name, null);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case "Semicolon":
|
|
break;
|
|
case "Label":
|
|
case "Goto":
|
|
throw new Error("Labels and gotos are not supported yet");
|
|
case "Block": {
|
|
const newEnv = new LuaEnv(env);
|
|
for (const statement of s.statements) {
|
|
await evalStatement(statement, newEnv, sf);
|
|
}
|
|
break;
|
|
}
|
|
case "If": {
|
|
for (const cond of s.conditions) {
|
|
if (luaTruthy(await evalExpression(cond.condition, env, sf))) {
|
|
return evalStatement(cond.block, env, sf);
|
|
}
|
|
}
|
|
if (s.elseBlock) {
|
|
return evalStatement(s.elseBlock, env, sf);
|
|
}
|
|
break;
|
|
}
|
|
case "While": {
|
|
while (luaTruthy(await evalExpression(s.condition, env, sf))) {
|
|
try {
|
|
await evalStatement(s.block, env, sf);
|
|
} catch (e: any) {
|
|
if (e instanceof LuaBreak) {
|
|
break;
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case "Repeat": {
|
|
do {
|
|
try {
|
|
await evalStatement(s.block, env, sf);
|
|
} catch (e: any) {
|
|
if (e instanceof LuaBreak) {
|
|
break;
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
} while (!luaTruthy(await evalExpression(s.condition, env, sf)));
|
|
break;
|
|
}
|
|
case "Break":
|
|
throw new LuaBreak();
|
|
case "FunctionCallStatement": {
|
|
return evalExpression(s.call, env, sf);
|
|
}
|
|
case "Function": {
|
|
let body = s.body;
|
|
let propNames = s.name.propNames;
|
|
if (s.name.colonName) {
|
|
// function hello:there() -> function hello.there(self) transformation
|
|
body = {
|
|
...s.body,
|
|
parameters: ["self", ...s.body.parameters],
|
|
};
|
|
propNames = [...s.name.propNames, s.name.colonName];
|
|
}
|
|
let settable: ILuaSettable & ILuaGettable = env;
|
|
for (let i = 0; i < propNames.length - 1; i++) {
|
|
settable = settable.get(propNames[i]);
|
|
if (!settable) {
|
|
throw new LuaRuntimeError(
|
|
`Cannot find property ${propNames[i]}`,
|
|
sf.withCtx(s.name.ctx),
|
|
);
|
|
}
|
|
}
|
|
settable.set(
|
|
propNames[propNames.length - 1],
|
|
new LuaFunction(body, env),
|
|
);
|
|
break;
|
|
}
|
|
case "LocalFunction": {
|
|
env.setLocal(
|
|
s.name,
|
|
new LuaFunction(s.body, env),
|
|
);
|
|
break;
|
|
}
|
|
case "Return": {
|
|
// A return statement for now is implemented by throwing the value as an exception, this should
|
|
// be optimized for the common case later
|
|
throw new LuaReturn(
|
|
await evalPromiseValues(
|
|
s.expressions.map((value) => evalExpression(value, env, sf)),
|
|
),
|
|
);
|
|
}
|
|
case "For": {
|
|
const start = await evalExpression(s.start, env, sf);
|
|
const end = await evalExpression(s.end, env, sf);
|
|
const step = s.step ? await evalExpression(s.step, env, sf) : 1;
|
|
const localEnv = new LuaEnv(env);
|
|
for (
|
|
let i = start;
|
|
step > 0 ? i <= end : i >= end;
|
|
i += step
|
|
) {
|
|
localEnv.setLocal(s.name, i);
|
|
try {
|
|
await evalStatement(s.block, localEnv, sf);
|
|
} catch (e: any) {
|
|
if (e instanceof LuaBreak) {
|
|
break;
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case "ForIn": {
|
|
const iteratorMultiRes = new LuaMultiRes(
|
|
await evalPromiseValues(
|
|
s.expressions.map((e) => evalExpression(e, env, sf)),
|
|
),
|
|
).flatten();
|
|
const iteratorFunction: ILuaFunction | undefined =
|
|
iteratorMultiRes.values[0];
|
|
if (!iteratorFunction?.call) {
|
|
console.error("Cannot iterate over", iteratorMultiRes.values[0]);
|
|
throw new LuaRuntimeError(
|
|
`Cannot iterate over ${iteratorMultiRes.values[0]}`,
|
|
sf.withCtx(s.ctx),
|
|
);
|
|
}
|
|
|
|
const state: LuaValue = iteratorMultiRes.values[1] || null;
|
|
const control: LuaValue = iteratorMultiRes.values[2] || null;
|
|
|
|
while (true) {
|
|
const iterResult = new LuaMultiRes(
|
|
await luaCall(iteratorFunction, [state, control], s.ctx, sf),
|
|
).flatten();
|
|
if (
|
|
iterResult.values[0] === null || iterResult.values[0] === undefined
|
|
) {
|
|
break;
|
|
}
|
|
const localEnv = new LuaEnv(env);
|
|
for (let i = 0; i < s.names.length; i++) {
|
|
localEnv.setLocal(s.names[i], iterResult.values[i]);
|
|
}
|
|
try {
|
|
await evalStatement(s.block, localEnv, sf);
|
|
} catch (e: any) {
|
|
if (e instanceof LuaBreak) {
|
|
break;
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function evalLValue(
|
|
lval: LuaLValue,
|
|
env: LuaEnv,
|
|
sf: LuaStackFrame,
|
|
): LuaLValueContainer | Promise<LuaLValueContainer> {
|
|
switch (lval.type) {
|
|
case "Variable":
|
|
return { env, key: lval.name };
|
|
case "TableAccess": {
|
|
const objValue = evalExpression(
|
|
lval.object,
|
|
env,
|
|
sf,
|
|
);
|
|
const keyValue = evalExpression(lval.key, env, sf);
|
|
if (
|
|
objValue instanceof Promise ||
|
|
keyValue instanceof Promise
|
|
) {
|
|
return Promise.all([
|
|
objValue instanceof Promise ? objValue : Promise.resolve(objValue),
|
|
keyValue instanceof Promise ? keyValue : Promise.resolve(keyValue),
|
|
]).then(([objValue, keyValue]) => ({
|
|
env: singleResult(objValue),
|
|
key: singleResult(keyValue),
|
|
}));
|
|
} else {
|
|
return {
|
|
env: singleResult(objValue),
|
|
key: singleResult(keyValue),
|
|
};
|
|
}
|
|
}
|
|
case "PropertyAccess": {
|
|
const objValue = evalExpression(
|
|
lval.object,
|
|
env,
|
|
sf,
|
|
);
|
|
if (objValue instanceof Promise) {
|
|
return objValue.then((objValue) => {
|
|
return {
|
|
env: objValue,
|
|
key: lval.property,
|
|
};
|
|
});
|
|
} else {
|
|
return {
|
|
env: objValue,
|
|
key: lval.property,
|
|
};
|
|
}
|
|
}
|
|
}
|
|
}
|