From 78ed5c80714f6f72f412b4e63830bd56a95a47d8 Mon Sep 17 00:00:00 2001 From: jtuu Date: Wed, 2 Oct 2019 18:26:45 +0300 Subject: [PATCH] [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);