2019-10-02 21:45:40 +08:00
|
|
|
import { Instruction, InstructionSegment, Segment, SegmentType, Arg, new_arg } from "../instructions";
|
2019-10-02 20:25:47 +08:00
|
|
|
import {
|
|
|
|
OP_CALL,
|
|
|
|
OP_CLEAR,
|
|
|
|
OP_EXIT,
|
|
|
|
OP_LET,
|
|
|
|
OP_LETB,
|
|
|
|
OP_LETI,
|
|
|
|
OP_LETW,
|
|
|
|
OP_NOP,
|
|
|
|
OP_RET,
|
|
|
|
OP_REV,
|
|
|
|
OP_SET,
|
|
|
|
OP_SYNC,
|
|
|
|
OP_THREAD,
|
2019-09-15 00:50:03 +08:00
|
|
|
OP_JMP,
|
2019-09-16 04:20:49 +08:00
|
|
|
OP_ARG_PUSHR,
|
|
|
|
OP_ARG_PUSHL,
|
|
|
|
OP_ARG_PUSHB,
|
|
|
|
OP_ARG_PUSHW,
|
|
|
|
OP_ARG_PUSHA,
|
|
|
|
OP_ARG_PUSHO,
|
|
|
|
OP_ARG_PUSHS,
|
2019-10-02 23:26:45 +08:00
|
|
|
OP_ADD,
|
|
|
|
OP_ADDI,
|
|
|
|
OP_SUB,
|
|
|
|
OP_SUBI,
|
|
|
|
OP_FADD,
|
|
|
|
OP_FADDI,
|
|
|
|
OP_FSUB,
|
|
|
|
OP_FSUBI,
|
|
|
|
OP_FMUL,
|
|
|
|
OP_MUL,
|
|
|
|
OP_MULI,
|
|
|
|
OP_FMULI,
|
|
|
|
OP_DIV,
|
|
|
|
OP_FDIV,
|
|
|
|
OP_DIVI,
|
|
|
|
OP_FDIVI,
|
|
|
|
OP_MOD,
|
|
|
|
OP_MODI,
|
|
|
|
OP_AND,
|
|
|
|
OP_ANDI,
|
|
|
|
OP_OR,
|
|
|
|
OP_ORI,
|
|
|
|
OP_XOR,
|
|
|
|
OP_XORI,
|
2019-10-02 20:25:47 +08:00
|
|
|
} from "../opcodes";
|
2019-08-06 23:07:12 +08:00
|
|
|
import Logger from "js-logger";
|
|
|
|
|
2019-08-26 21:42:12 +08:00
|
|
|
const logger = Logger.get("quest_editor/scripting/vm");
|
2019-08-06 23:07:12 +08:00
|
|
|
|
|
|
|
const REGISTER_COUNT = 256;
|
|
|
|
const REGISTER_SIZE = 4;
|
|
|
|
|
|
|
|
export enum ExecutionResult {
|
|
|
|
Ok,
|
|
|
|
WaitingVsync,
|
|
|
|
Halted,
|
|
|
|
}
|
|
|
|
|
2019-10-02 23:26:45 +08:00
|
|
|
type BinaryNumericOperation = (a: number, b: number) => number;
|
|
|
|
|
|
|
|
const numeric_ops: Record<"add" |
|
|
|
|
"sub" |
|
|
|
|
"mul" |
|
|
|
|
"div" |
|
|
|
|
"idiv" |
|
|
|
|
"mod" |
|
|
|
|
"and" |
|
|
|
|
"or" |
|
|
|
|
"xor",
|
|
|
|
BinaryNumericOperation> = {
|
|
|
|
add: (a, b) => a + b,
|
|
|
|
sub: (a, b) => a - b,
|
|
|
|
mul: (a, b) => a * b,
|
|
|
|
div: (a, b) => a / b,
|
|
|
|
idiv: (a, b) => Math.floor(a / b),
|
|
|
|
mod: (a, b) => a % b,
|
|
|
|
and: (a, b) => a & b,
|
|
|
|
or: (a, b) => a | b,
|
|
|
|
xor: (a, b) => a ^ b,
|
|
|
|
};
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
export class VirtualMachine {
|
|
|
|
private register_store = new ArrayBuffer(REGISTER_SIZE * REGISTER_COUNT);
|
|
|
|
private register_uint8_view = new Uint8Array(this.register_store);
|
|
|
|
private registers = new DataView(this.register_store);
|
|
|
|
private object_code: Segment[] = [];
|
|
|
|
private label_to_seg_idx: Map<number, number> = new Map();
|
|
|
|
private thread: Thread[] = [];
|
|
|
|
private thread_idx = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Halts and resets the VM, then loads new object code.
|
|
|
|
*/
|
|
|
|
load_object_code(object_code: Segment[]): void {
|
|
|
|
this.halt();
|
|
|
|
this.clear_registers();
|
|
|
|
this.object_code = object_code;
|
|
|
|
this.label_to_seg_idx.clear();
|
|
|
|
let i = 0;
|
|
|
|
|
|
|
|
for (const segment of this.object_code) {
|
|
|
|
for (const label of segment.labels) {
|
|
|
|
this.label_to_seg_idx.set(label, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Schedules concurrent execution of the code at the given label.
|
|
|
|
*/
|
|
|
|
start_thread(label: number): void {
|
|
|
|
const seg_idx = this.label_to_seg_idx.get(label);
|
|
|
|
const segment = seg_idx == undefined ? undefined : this.object_code[seg_idx];
|
|
|
|
|
|
|
|
if (segment == undefined) {
|
|
|
|
throw new Error(`Unknown label ${label}.`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (segment.type !== SegmentType.Instructions) {
|
|
|
|
throw new Error(
|
|
|
|
`Label ${label} points to a ${SegmentType[segment.type]} segment, expecting ${
|
|
|
|
SegmentType[SegmentType.Instructions]
|
2019-08-11 04:09:06 +08:00
|
|
|
}.`,
|
2019-08-06 23:07:12 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
this.thread.push(new Thread(new ExecutionLocation(seg_idx!, 0), true));
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes the next instruction if one is scheduled.
|
|
|
|
*
|
|
|
|
* @returns true if an instruction was executed, false otherwise.
|
|
|
|
*/
|
|
|
|
execute(): ExecutionResult {
|
|
|
|
if (this.thread.length === 0) return ExecutionResult.Halted;
|
|
|
|
if (this.thread_idx >= this.thread.length) return ExecutionResult.WaitingVsync;
|
|
|
|
|
|
|
|
const exec = this.thread[this.thread_idx];
|
|
|
|
const inst = this.get_next_instruction_from_thread(exec);
|
|
|
|
|
2019-10-02 22:39:32 +08:00
|
|
|
const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = inst.args.map(arg => arg.value);
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
switch (inst.opcode) {
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_NOP:
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_RET:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.pop_call_stack(this.thread_idx, exec);
|
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_SYNC:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.thread_idx++;
|
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_EXIT:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.halt();
|
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_THREAD:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.start_thread(arg0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_LET:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_sint(arg0, this.get_sint(arg1));
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_LETI:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_sint(arg0, arg1);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_LETB:
|
|
|
|
case OP_LETW:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_uint(arg0, arg1);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_SET:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_sint(arg0, 1);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_CLEAR:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_sint(arg0, 0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_REV:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.set_sint(arg0, this.get_sint(arg0) === 0 ? 1 : 0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 20:25:47 +08:00
|
|
|
case OP_CALL:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.push_call_stack(exec, arg0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-09-15 00:50:03 +08:00
|
|
|
case OP_JMP:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.jump_to_label(exec, arg0);
|
2019-09-15 00:50:03 +08:00
|
|
|
break;
|
2019-09-16 04:20:49 +08:00
|
|
|
case OP_ARG_PUSHR:
|
2019-10-02 21:45:40 +08:00
|
|
|
// deref given register ref
|
|
|
|
this.push_arg_stack(exec, new_arg(
|
2019-10-02 22:39:32 +08:00
|
|
|
this.get_sint(arg0),
|
2019-10-02 21:45:40 +08:00
|
|
|
REGISTER_SIZE,
|
|
|
|
inst.args[0].asm
|
|
|
|
));
|
|
|
|
break;
|
2019-09-16 04:20:49 +08:00
|
|
|
case OP_ARG_PUSHL:
|
|
|
|
case OP_ARG_PUSHB:
|
|
|
|
case OP_ARG_PUSHW:
|
|
|
|
case OP_ARG_PUSHS:
|
2019-10-02 21:45:40 +08:00
|
|
|
// push arg as-is
|
2019-10-02 22:39:32 +08:00
|
|
|
this.push_arg_stack(exec, arg0);
|
2019-09-16 04:20:49 +08:00
|
|
|
break;
|
2019-10-02 23:26:45 +08:00
|
|
|
// arithmetic operations
|
|
|
|
case OP_ADD:
|
|
|
|
case OP_FADD:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.add);
|
|
|
|
break;
|
|
|
|
case OP_ADDI:
|
|
|
|
case OP_FADDI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.add);
|
|
|
|
break;
|
|
|
|
case OP_SUB:
|
|
|
|
case OP_FSUB:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.sub);
|
|
|
|
break;
|
|
|
|
case OP_SUBI:
|
|
|
|
case OP_FSUBI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.sub);
|
|
|
|
break;
|
|
|
|
case OP_MUL:
|
|
|
|
case OP_FMUL:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mul);
|
|
|
|
break;
|
|
|
|
case OP_MULI:
|
|
|
|
case OP_FMULI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mul);
|
|
|
|
break;
|
|
|
|
case OP_DIV:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.idiv);
|
|
|
|
break;
|
|
|
|
case OP_FDIV:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.div);
|
|
|
|
break;
|
|
|
|
case OP_DIVI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.idiv);
|
|
|
|
break;
|
|
|
|
case OP_FDIVI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.div);
|
|
|
|
break;
|
|
|
|
case OP_MOD:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mod);
|
|
|
|
break;
|
|
|
|
case OP_MODI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mod);
|
|
|
|
break;
|
|
|
|
// bit operations
|
|
|
|
case OP_AND:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.and);
|
|
|
|
break;
|
|
|
|
case OP_ANDI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.and);
|
|
|
|
break;
|
|
|
|
case OP_OR:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.or);
|
|
|
|
break;
|
|
|
|
case OP_ORI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.or);
|
|
|
|
break;
|
|
|
|
case OP_XOR:
|
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.xor);
|
|
|
|
break;
|
|
|
|
case OP_XORI:
|
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.xor);
|
|
|
|
break;
|
2019-08-06 23:07:12 +08:00
|
|
|
default:
|
|
|
|
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
|
|
|
}
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
// advance instruction "pointer"
|
|
|
|
if (exec.call_stack.length) {
|
|
|
|
const top = exec.call_stack_top();
|
2019-08-06 23:07:12 +08:00
|
|
|
const segment = this.object_code[top.seg_idx] as InstructionSegment;
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
// move to next instruction
|
2019-08-06 23:07:12 +08:00
|
|
|
if (++top.inst_idx >= segment.instructions.length) {
|
2019-09-15 00:50:03 +08:00
|
|
|
// segment ended, move to next segment
|
|
|
|
if (++top.seg_idx >= this.object_code.length) {
|
|
|
|
// eof
|
|
|
|
this.thread.splice(this.thread_idx, 1);
|
|
|
|
} else {
|
|
|
|
top.inst_idx = 0;
|
|
|
|
}
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.thread.length === 0) return ExecutionResult.Halted;
|
|
|
|
if (this.thread_idx >= this.thread.length) return ExecutionResult.WaitingVsync;
|
|
|
|
return ExecutionResult.Ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signal to the VM that a vsync has happened.
|
|
|
|
*/
|
|
|
|
vsync(): void {
|
|
|
|
if (this.thread_idx >= this.thread.length) {
|
|
|
|
this.thread_idx = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Halts execution of all threads.
|
|
|
|
*/
|
|
|
|
halt(): void {
|
|
|
|
this.thread = [];
|
|
|
|
this.thread_idx = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private get_sint(reg: number): number {
|
|
|
|
return this.registers.getInt32(REGISTER_SIZE * reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
private set_sint(reg: number, value: number): void {
|
|
|
|
this.registers.setInt32(REGISTER_SIZE * reg, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
private set_uint(reg: number, value: number): void {
|
|
|
|
this.registers.setUint32(REGISTER_SIZE * reg, value);
|
|
|
|
}
|
|
|
|
|
2019-10-02 23:26:45 +08:00
|
|
|
private do_numeric_op_with_register(reg1: number, reg2: number, op: BinaryNumericOperation): void {
|
|
|
|
this.do_numeric_op_with_literal(reg1, this.get_sint(reg2), op);
|
|
|
|
}
|
|
|
|
|
|
|
|
private do_numeric_op_with_literal(reg: number, literal: number, op: BinaryNumericOperation): void {
|
|
|
|
this.set_sint(reg, op(this.get_sint(reg), literal));
|
|
|
|
}
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
private push_call_stack(exec: Thread, label: number): void {
|
|
|
|
const seg_idx = this.label_to_seg_idx.get(label);
|
|
|
|
|
|
|
|
if (seg_idx == undefined) {
|
|
|
|
logger.warn(`Invalid label called: ${label}.`);
|
|
|
|
} else {
|
|
|
|
const segment = this.object_code[seg_idx];
|
|
|
|
|
|
|
|
if (segment.type !== SegmentType.Instructions) {
|
|
|
|
logger.warn(
|
|
|
|
`Label ${label} points to a ${SegmentType[segment.type]} segment, expecting ${
|
|
|
|
SegmentType[SegmentType.Instructions]
|
2019-08-11 04:09:06 +08:00
|
|
|
}.`,
|
2019-08-06 23:07:12 +08:00
|
|
|
);
|
|
|
|
} else {
|
2019-09-15 00:50:03 +08:00
|
|
|
exec.call_stack.push(new ExecutionLocation(seg_idx, -1));
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private pop_call_stack(idx: number, exec: Thread): void {
|
2019-09-15 00:50:03 +08:00
|
|
|
exec.call_stack.pop();
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
if (exec.call_stack.length >= 1) {
|
|
|
|
const top = exec.call_stack_top();
|
2019-08-06 23:07:12 +08:00
|
|
|
const segment = this.object_code[top.seg_idx];
|
|
|
|
|
|
|
|
if (!segment || segment.type !== SegmentType.Instructions) {
|
|
|
|
throw new Error(`Invalid segment index ${top.seg_idx}.`);
|
|
|
|
}
|
|
|
|
} else {
|
2019-09-15 00:50:03 +08:00
|
|
|
// popped off the last return address
|
|
|
|
// which means this is the end of the function this thread was started on
|
|
|
|
// which means this is the end of this thread
|
2019-08-06 23:07:12 +08:00
|
|
|
this.thread.splice(idx, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
private jump_to_label(exec: Thread, label: number) {
|
|
|
|
const top = exec.call_stack_top();
|
|
|
|
const seg_idx = this.label_to_seg_idx.get(label);
|
|
|
|
|
|
|
|
if (seg_idx == undefined) {
|
|
|
|
logger.warn(`Invalid jump label: ${label}.`);
|
|
|
|
} else {
|
|
|
|
top.seg_idx = seg_idx;
|
|
|
|
top.inst_idx = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-16 04:20:49 +08:00
|
|
|
private push_arg_stack(exec: Thread, arg: Arg): void {
|
|
|
|
exec.arg_stack.push(arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
private pop_arg_stack(exec: Thread): Arg {
|
|
|
|
const arg = exec.arg_stack.pop();
|
|
|
|
|
|
|
|
if (!arg) {
|
|
|
|
throw new Error("Argument stack underflow.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return arg;
|
|
|
|
}
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
private get_next_instruction_from_thread(exec: Thread): Instruction {
|
2019-09-15 00:50:03 +08:00
|
|
|
if (exec.call_stack.length) {
|
|
|
|
const top = exec.call_stack_top();
|
2019-08-06 23:07:12 +08:00
|
|
|
const segment = this.object_code[top.seg_idx];
|
|
|
|
|
|
|
|
if (!segment || segment.type !== SegmentType.Instructions) {
|
|
|
|
throw new Error(`Invalid segment index ${top.seg_idx}.`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const inst = segment.instructions[top.inst_idx];
|
|
|
|
|
|
|
|
if (!inst) {
|
|
|
|
throw new Error(
|
2019-08-11 04:09:06 +08:00
|
|
|
`Invalid instruction index ${top.inst_idx} for segment ${top.seg_idx}.`,
|
2019-08-06 23:07:12 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return inst;
|
|
|
|
} else {
|
|
|
|
throw new Error(`Call stack is empty.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private clear_registers(): void {
|
|
|
|
this.register_uint8_view.fill(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
class ExecutionLocation {
|
2019-08-06 23:07:12 +08:00
|
|
|
constructor(public seg_idx: number, public inst_idx: number) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Thread {
|
|
|
|
/**
|
|
|
|
* Call stack. The top element describes the instruction about to be executed.
|
|
|
|
*/
|
2019-09-15 00:50:03 +08:00
|
|
|
public call_stack: ExecutionLocation[] = [];
|
2019-09-16 04:20:49 +08:00
|
|
|
public arg_stack: Arg[] = [];
|
2019-08-06 23:07:12 +08:00
|
|
|
/**
|
|
|
|
* Global or floor-local?
|
|
|
|
*/
|
|
|
|
public global: boolean;
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
call_stack_top(): ExecutionLocation {
|
|
|
|
return this.call_stack[this.call_stack.length - 1];
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-09-15 00:50:03 +08:00
|
|
|
constructor(next: ExecutionLocation, global: boolean) {
|
|
|
|
this.call_stack = [next];
|
2019-08-06 23:07:12 +08:00
|
|
|
this.global = global;
|
|
|
|
}
|
|
|
|
}
|