From 80a4aa784f83277e2edba6e045f21f6b8d0ec9bb Mon Sep 17 00:00:00 2001 From: jtuu Date: Sat, 14 Sep 2019 19:50:03 +0300 Subject: [PATCH 01/10] [VM] Added a button to run the current quest in the VM. Made the VM work with output from the assembler. Added some comments to the VM and cleaned it up a little bit. --- src/quest_editor/gui/QuestEditorToolBar.ts | 7 +++ src/quest_editor/scripting/vm/index.ts | 60 +++++++++++++++------ src/quest_editor/stores/QuestEditorStore.ts | 30 +++++++++++ 3 files changed, 80 insertions(+), 17 deletions(-) diff --git a/src/quest_editor/gui/QuestEditorToolBar.ts b/src/quest_editor/gui/QuestEditorToolBar.ts index 396dd8df..8d517aab 100644 --- a/src/quest_editor/gui/QuestEditorToolBar.ts +++ b/src/quest_editor/gui/QuestEditorToolBar.ts @@ -69,6 +69,10 @@ export class QuestEditorToolBar extends ToolBar { } }, ); + const run_button = new Button("Run in VM", { + icon_left: Icon.Play, + tooltip: "[Experimental] Run the current quest in a virtual machine." + }); super({ children: [ @@ -78,6 +82,7 @@ export class QuestEditorToolBar extends ToolBar { undo_button, redo_button, area_select, + run_button, ], }); @@ -109,6 +114,8 @@ export class QuestEditorToolBar extends ToolBar { quest_editor_store.set_current_area(area), ), + run_button.click.observe(() => quest_editor_store.run_current_quest_in_vm()), + gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-O", () => open_file_button.click(), ), diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 8b5f647c..02bbdcee 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -13,6 +13,7 @@ import { OP_SET, OP_SYNC, OP_THREAD, + OP_JMP, } from "../opcodes"; import Logger from "js-logger"; @@ -74,7 +75,7 @@ export class VirtualMachine { ); } - this.thread.push(new Thread(new StackElement(seg_idx!, 0), true)); + this.thread.push(new Thread(new ExecutionLocation(seg_idx!, 0), true)); } /** @@ -126,17 +127,27 @@ export class VirtualMachine { case OP_CALL: this.push_call_stack(exec, inst.args[0].value); break; + case OP_JMP: + this.jump_to_label(exec, inst.args[0].value); + break; default: throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`); } - if (exec.stack.length) { - const top = exec.stack_top(); + // advance instruction "pointer" + if (exec.call_stack.length) { + const top = exec.call_stack_top(); const segment = this.object_code[top.seg_idx] as InstructionSegment; + // move to next instruction if (++top.inst_idx >= segment.instructions.length) { - top.seg_idx++; - top.inst_idx = 0; + // 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; + } } } @@ -189,29 +200,44 @@ export class VirtualMachine { }.`, ); } else { - exec.stack.push(new StackElement(seg_idx, -1)); + exec.call_stack.push(new ExecutionLocation(seg_idx, -1)); } } } private pop_call_stack(idx: number, exec: Thread): void { - exec.stack.pop(); + exec.call_stack.pop(); - if (exec.stack.length >= 1) { - const top = exec.stack_top(); + if (exec.call_stack.length >= 1) { + const top = exec.call_stack_top(); const segment = this.object_code[top.seg_idx]; if (!segment || segment.type !== SegmentType.Instructions) { throw new Error(`Invalid segment index ${top.seg_idx}.`); } } else { + // 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 this.thread.splice(idx, 1); } } + 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; + } + } + private get_next_instruction_from_thread(exec: Thread): Instruction { - if (exec.stack.length) { - const top = exec.stack_top(); + if (exec.call_stack.length) { + const top = exec.call_stack_top(); const segment = this.object_code[top.seg_idx]; if (!segment || segment.type !== SegmentType.Instructions) { @@ -237,7 +263,7 @@ export class VirtualMachine { } } -class StackElement { +class ExecutionLocation { constructor(public seg_idx: number, public inst_idx: number) {} } @@ -245,18 +271,18 @@ class Thread { /** * Call stack. The top element describes the instruction about to be executed. */ - public stack: StackElement[] = []; + public call_stack: ExecutionLocation[] = []; /** * Global or floor-local? */ public global: boolean; - stack_top(): StackElement { - return this.stack[this.stack.length - 1]; + call_stack_top(): ExecutionLocation { + return this.call_stack[this.call_stack.length - 1]; } - constructor(next: StackElement, global: boolean) { - this.stack = [next]; + constructor(next: ExecutionLocation, global: boolean) { + this.call_stack = [next]; this.global = global; } } diff --git a/src/quest_editor/stores/QuestEditorStore.ts b/src/quest_editor/stores/QuestEditorStore.ts index e5a53fe6..48c18cea 100644 --- a/src/quest_editor/stores/QuestEditorStore.ts +++ b/src/quest_editor/stores/QuestEditorStore.ts @@ -28,6 +28,7 @@ import { RemoveEntityAction } from "../actions/RemoveEntityAction"; import { Euler, Vector3 } from "three"; import { vec3_to_threejs } from "../../core/rendering/conversion"; import { RotateEntityAction } from "../actions/RotateEntityAction"; +import { VirtualMachine, ExecutionResult } from "../scripting/vm"; import Logger = require("js-logger"); const logger = Logger.get("quest_editor/gui/QuestEditorStore"); @@ -350,6 +351,35 @@ export class QuestEditorStore implements Disposable { logger.warn(`Section ${entity.section_id.val} not found.`); } }; + + run_current_quest_in_vm = () => { + logger.setLevel(logger.TRACE); + + const quest = this.current_quest.val; + + if (!quest) { + throw new Error("No quest"); + } + + const vm = new VirtualMachine(); + vm.load_object_code(quest.object_code); + vm.start_thread(0); + + exec_loop: + while (true) { + const exec_result = vm.execute(); + + switch (exec_result) { + case ExecutionResult.Ok: + break; + case ExecutionResult.WaitingVsync: + vm.vsync(); + break; + case ExecutionResult.Halted: + break exec_loop; + } + } + } } export const quest_editor_store = new QuestEditorStore(); From 5458d06d8e5d72040454917f31179babecafde63 Mon Sep 17 00:00:00 2001 From: jtuu Date: Sat, 14 Sep 2019 21:58:15 +0300 Subject: [PATCH 02/10] [VM] Changed the icon for the VM button. --- src/core/gui/dom.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/core/gui/dom.ts b/src/core/gui/dom.ts index c7d573ab..43f2ce7d 100644 --- a/src/core/gui/dom.ts +++ b/src/core/gui/dom.ts @@ -146,6 +146,7 @@ export enum Icon { Redo, Remove, GitHub, + Play, } export function icon(icon: Icon): HTMLElement { @@ -179,6 +180,9 @@ export function icon(icon: Icon): HTMLElement { case Icon.GitHub: icon_str = "fab fa-github"; break; + case Icon.Play: + icon_str = "fas fa-play"; + break; } return el.span({ class: icon_str }); From 98acdbf7f99dbd25dcaa738101afcadbe02edc7a Mon Sep 17 00:00:00 2001 From: jtuu Date: Sun, 15 Sep 2019 23:20:49 +0300 Subject: [PATCH 03/10] [VM] Added arg stack. --- src/quest_editor/scripting/vm/index.ts | 33 +++++++++++++++++++++++++- 1 file changed, 32 insertions(+), 1 deletion(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 02bbdcee..426b8755 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -1,4 +1,4 @@ -import { Instruction, InstructionSegment, Segment, SegmentType } from "../instructions"; +import { Instruction, InstructionSegment, Segment, SegmentType, Arg } from "../instructions"; import { OP_CALL, OP_CLEAR, @@ -14,6 +14,13 @@ import { OP_SYNC, OP_THREAD, OP_JMP, + OP_ARG_PUSHR, + OP_ARG_PUSHL, + OP_ARG_PUSHB, + OP_ARG_PUSHW, + OP_ARG_PUSHA, + OP_ARG_PUSHO, + OP_ARG_PUSHS, } from "../opcodes"; import Logger from "js-logger"; @@ -130,6 +137,15 @@ export class VirtualMachine { case OP_JMP: this.jump_to_label(exec, inst.args[0].value); break; + case OP_ARG_PUSHR: + case OP_ARG_PUSHL: + case OP_ARG_PUSHB: + case OP_ARG_PUSHW: + case OP_ARG_PUSHA: + case OP_ARG_PUSHO: + case OP_ARG_PUSHS: + this.push_arg_stack(exec, inst.args[0].value); + break; default: throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`); } @@ -235,6 +251,20 @@ export class VirtualMachine { } } + 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; + } + private get_next_instruction_from_thread(exec: Thread): Instruction { if (exec.call_stack.length) { const top = exec.call_stack_top(); @@ -272,6 +302,7 @@ class Thread { * Call stack. The top element describes the instruction about to be executed. */ public call_stack: ExecutionLocation[] = []; + public arg_stack: Arg[] = []; /** * Global or floor-local? */ From 806ab0ddd5a5e799bad32e705a3e769d9c7c169d Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 16:45:40 +0300 Subject: [PATCH 04/10] [VM] Implement arg_pushr, b, w, and s correctly. Remove arg_pusha and o because they are tricky and low priority. --- src/quest_editor/scripting/vm/index.ts | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 426b8755..794afe5c 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -1,4 +1,4 @@ -import { Instruction, InstructionSegment, Segment, SegmentType, Arg } from "../instructions"; +import { Instruction, InstructionSegment, Segment, SegmentType, Arg, new_arg } from "../instructions"; import { OP_CALL, OP_CLEAR, @@ -138,12 +138,18 @@ export class VirtualMachine { this.jump_to_label(exec, inst.args[0].value); break; case OP_ARG_PUSHR: + // deref given register ref + this.push_arg_stack(exec, new_arg( + this.get_sint(inst.args[0].value), + REGISTER_SIZE, + inst.args[0].asm + )); + break; case OP_ARG_PUSHL: case OP_ARG_PUSHB: case OP_ARG_PUSHW: - case OP_ARG_PUSHA: - case OP_ARG_PUSHO: case OP_ARG_PUSHS: + // push arg as-is this.push_arg_stack(exec, inst.args[0].value); break; default: From bdd7e8b174036c404b189708f426420cc3c6554a Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 17:39:32 +0300 Subject: [PATCH 05/10] [VM] Assign arguments to variables to make accessing them terser. --- src/quest_editor/scripting/vm/index.ts | 24 +++++++++++++----------- 1 file changed, 13 insertions(+), 11 deletions(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 794afe5c..26551c51 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -97,6 +97,8 @@ export class VirtualMachine { const exec = this.thread[this.thread_idx]; const inst = this.get_next_instruction_from_thread(exec); + const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = inst.args.map(arg => arg.value); + switch (inst.opcode) { case OP_NOP: break; @@ -110,37 +112,37 @@ export class VirtualMachine { this.halt(); break; case OP_THREAD: - this.start_thread(inst.args[0].value); + this.start_thread(arg0); break; case OP_LET: - this.set_sint(inst.args[0].value, this.get_sint(inst.args[1].value)); + this.set_sint(arg0, this.get_sint(arg1)); break; case OP_LETI: - this.set_sint(inst.args[0].value, inst.args[1].value); + this.set_sint(arg0, arg1); break; case OP_LETB: case OP_LETW: - this.set_uint(inst.args[0].value, inst.args[1].value); + this.set_uint(arg0, arg1); break; case OP_SET: - this.set_sint(inst.args[0].value, 1); + this.set_sint(arg0, 1); break; case OP_CLEAR: - this.set_sint(inst.args[0].value, 0); + this.set_sint(arg0, 0); break; case OP_REV: - this.set_sint(inst.args[0].value, this.get_sint(inst.args[0].value) === 0 ? 1 : 0); + this.set_sint(arg0, this.get_sint(arg0) === 0 ? 1 : 0); break; case OP_CALL: - this.push_call_stack(exec, inst.args[0].value); + this.push_call_stack(exec, arg0); break; case OP_JMP: - this.jump_to_label(exec, inst.args[0].value); + this.jump_to_label(exec, arg0); break; case OP_ARG_PUSHR: // deref given register ref this.push_arg_stack(exec, new_arg( - this.get_sint(inst.args[0].value), + this.get_sint(arg0), REGISTER_SIZE, inst.args[0].asm )); @@ -150,7 +152,7 @@ export class VirtualMachine { case OP_ARG_PUSHW: case OP_ARG_PUSHS: // push arg as-is - this.push_arg_stack(exec, inst.args[0].value); + this.push_arg_stack(exec, arg0); break; default: throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`); From 78ed5c80714f6f72f412b4e63830bd56a95a47d8 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 18:26:45 +0300 Subject: [PATCH 06/10] [VM] Implemented opcodes for basic numeric operations. --- src/quest_editor/scripting/vm/index.ts | 117 +++++++++++++++++++++++++ 1 file changed, 117 insertions(+) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 26551c51..0ff29f1b 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -21,6 +21,30 @@ import { OP_ARG_PUSHA, OP_ARG_PUSHO, OP_ARG_PUSHS, + 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, } from "../opcodes"; import Logger from "js-logger"; @@ -35,6 +59,29 @@ export enum ExecutionResult { Halted, } +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, +}; + export class VirtualMachine { private register_store = new ArrayBuffer(REGISTER_SIZE * REGISTER_COUNT); private register_uint8_view = new Uint8Array(this.register_store); @@ -154,6 +201,68 @@ export class VirtualMachine { // push arg as-is this.push_arg_stack(exec, arg0); break; + // 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; default: throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`); } @@ -209,6 +318,14 @@ export class VirtualMachine { this.registers.setUint32(REGISTER_SIZE * reg, value); } + 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)); + } + private push_call_stack(exec: Thread, label: number): void { const seg_idx = this.label_to_seg_idx.get(label); From c628c30e1bdbbe9e455d52361807f1ddb83afa05 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 18:52:59 +0300 Subject: [PATCH 07/10] [VM] Compare opcode numbers not the objects. --- src/quest_editor/scripting/vm/index.ts | 88 +++++++++++++------------- 1 file changed, 44 insertions(+), 44 deletions(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 0ff29f1b..3b14956e 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -146,47 +146,47 @@ export class VirtualMachine { const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = inst.args.map(arg => arg.value); - switch (inst.opcode) { - case OP_NOP: + switch (inst.opcode.code) { + case OP_NOP.code: break; - case OP_RET: + case OP_RET.code: this.pop_call_stack(this.thread_idx, exec); break; - case OP_SYNC: + case OP_SYNC.code: this.thread_idx++; break; - case OP_EXIT: + case OP_EXIT.code: this.halt(); break; - case OP_THREAD: + case OP_THREAD.code: this.start_thread(arg0); break; - case OP_LET: + case OP_LET.code: this.set_sint(arg0, this.get_sint(arg1)); break; - case OP_LETI: + case OP_LETI.code: this.set_sint(arg0, arg1); break; - case OP_LETB: - case OP_LETW: + case OP_LETB.code: + case OP_LETW.code: this.set_uint(arg0, arg1); break; - case OP_SET: + case OP_SET.code: this.set_sint(arg0, 1); break; - case OP_CLEAR: + case OP_CLEAR.code: this.set_sint(arg0, 0); break; - case OP_REV: + case OP_REV.code: this.set_sint(arg0, this.get_sint(arg0) === 0 ? 1 : 0); break; - case OP_CALL: + case OP_CALL.code: this.push_call_stack(exec, arg0); break; - case OP_JMP: + case OP_JMP.code: this.jump_to_label(exec, arg0); break; - case OP_ARG_PUSHR: + case OP_ARG_PUSHR.code: // deref given register ref this.push_arg_stack(exec, new_arg( this.get_sint(arg0), @@ -194,73 +194,73 @@ export class VirtualMachine { inst.args[0].asm )); break; - case OP_ARG_PUSHL: - case OP_ARG_PUSHB: - case OP_ARG_PUSHW: - case OP_ARG_PUSHS: + case OP_ARG_PUSHL.code: + case OP_ARG_PUSHB.code: + case OP_ARG_PUSHW.code: + case OP_ARG_PUSHS.code: // push arg as-is this.push_arg_stack(exec, arg0); break; // arithmetic operations - case OP_ADD: - case OP_FADD: + case OP_ADD.code: + case OP_FADD.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.add); break; - case OP_ADDI: - case OP_FADDI: + case OP_ADDI.code: + case OP_FADDI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.add); break; - case OP_SUB: - case OP_FSUB: + case OP_SUB.code: + case OP_FSUB.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.sub); break; - case OP_SUBI: - case OP_FSUBI: + case OP_SUBI.code: + case OP_FSUBI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.sub); break; - case OP_MUL: - case OP_FMUL: + case OP_MUL.code: + case OP_FMUL.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mul); break; - case OP_MULI: - case OP_FMULI: + case OP_MULI.code: + case OP_FMULI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mul); break; - case OP_DIV: + case OP_DIV.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.idiv); break; - case OP_FDIV: + case OP_FDIV.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.div); break; - case OP_DIVI: + case OP_DIVI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.idiv); break; - case OP_FDIVI: + case OP_FDIVI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.div); break; - case OP_MOD: + case OP_MOD.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mod); break; - case OP_MODI: + case OP_MODI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mod); break; // bit operations - case OP_AND: + case OP_AND.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.and); break; - case OP_ANDI: + case OP_ANDI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.and); break; - case OP_OR: + case OP_OR.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.or); break; - case OP_ORI: + case OP_ORI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.or); break; - case OP_XOR: + case OP_XOR.code: this.do_numeric_op_with_register(arg0, arg1, numeric_ops.xor); break; - case OP_XORI: + case OP_XORI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.xor); break; default: From 2e672c12b2f3556ac21ece50d48271dc2b47c6e1 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 20:16:05 +0300 Subject: [PATCH 08/10] [VM] Implemented shift_left and shift_right opcodes. --- src/quest_editor/scripting/vm/index.ts | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index 3b14956e..de824378 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -45,6 +45,8 @@ import { OP_ORI, OP_XOR, OP_XORI, + OP_SHIFT_LEFT, + OP_SHIFT_RIGHT, } from "../opcodes"; import Logger from "js-logger"; @@ -69,7 +71,9 @@ const numeric_ops: Record<"add" | "mod" | "and" | "or" | - "xor", + "xor" | + "shl" | + "shr", BinaryNumericOperation> = { add: (a, b) => a + b, sub: (a, b) => a - b, @@ -80,6 +84,8 @@ const numeric_ops: Record<"add" | and: (a, b) => a & b, or: (a, b) => a | b, xor: (a, b) => a ^ b, + shl: (a, b) => a << b, + shr: (a, b) => a >>> b, }; export class VirtualMachine { @@ -263,6 +269,13 @@ export class VirtualMachine { case OP_XORI.code: this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.xor); break; + // shift operations + case OP_SHIFT_LEFT.code: + this.do_numeric_op_with_register(arg0, arg1, numeric_ops.shl); + break; + case OP_SHIFT_RIGHT.code: + this.do_numeric_op_with_register(arg0, arg1, numeric_ops.shr); + break; default: throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`); } From d9110f0041d5b3a7df295fbff6fd856aef414a63 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 20:16:54 +0300 Subject: [PATCH 09/10] [VM] Push entire Arg object to stack instead of just its value. --- src/quest_editor/scripting/vm/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/quest_editor/scripting/vm/index.ts b/src/quest_editor/scripting/vm/index.ts index de824378..800d27cb 100644 --- a/src/quest_editor/scripting/vm/index.ts +++ b/src/quest_editor/scripting/vm/index.ts @@ -205,7 +205,7 @@ export class VirtualMachine { case OP_ARG_PUSHW.code: case OP_ARG_PUSHS.code: // push arg as-is - this.push_arg_stack(exec, arg0); + this.push_arg_stack(exec, inst.args[0]); break; // arithmetic operations case OP_ADD.code: From 1a7e9ee9acc732fb4d5c895657a5802651a5eab5 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 21:34:57 +0300 Subject: [PATCH 10/10] [VM] Put the "run in VM" button behind a feature flag called "vm". --- src/quest_editor/gui/QuestEditorToolBar.ts | 25 ++++++++++++---------- 1 file changed, 14 insertions(+), 11 deletions(-) diff --git a/src/quest_editor/gui/QuestEditorToolBar.ts b/src/quest_editor/gui/QuestEditorToolBar.ts index 8d517aab..a99e6a81 100644 --- a/src/quest_editor/gui/QuestEditorToolBar.ts +++ b/src/quest_editor/gui/QuestEditorToolBar.ts @@ -74,17 +74,20 @@ export class QuestEditorToolBar extends ToolBar { tooltip: "[Experimental] Run the current quest in a virtual machine." }); - super({ - children: [ - new_quest_button, - open_file_button, - save_as_button, - undo_button, - redo_button, - area_select, - run_button, - ], - }); + const children = [ + new_quest_button, + open_file_button, + save_as_button, + undo_button, + redo_button, + area_select, + ]; + + if (gui_store.feature_active("vm")) { + children.push(run_button); + } + + super({ children }); const quest_loaded = quest_editor_store.current_quest.map(q => q != undefined);