2019-10-10 21:02:41 +08:00
|
|
|
import { Instruction, InstructionSegment, Segment, SegmentType } from "../instructions";
|
2019-10-03 23:28:48 +08:00
|
|
|
import {
|
|
|
|
OP_ADD,
|
|
|
|
OP_ADDI,
|
|
|
|
OP_AND,
|
|
|
|
OP_ANDI,
|
|
|
|
OP_ARG_PUSHB,
|
|
|
|
OP_ARG_PUSHL,
|
|
|
|
OP_ARG_PUSHR,
|
|
|
|
OP_ARG_PUSHW,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_CALL,
|
|
|
|
OP_CLEAR,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_DIV,
|
|
|
|
OP_DIVI,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_EXIT,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_FADD,
|
|
|
|
OP_FADDI,
|
|
|
|
OP_FDIV,
|
|
|
|
OP_FDIVI,
|
|
|
|
OP_FMUL,
|
|
|
|
OP_FMULI,
|
|
|
|
OP_FSUB,
|
|
|
|
OP_FSUBI,
|
|
|
|
OP_JMP,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_LET,
|
|
|
|
OP_LETB,
|
|
|
|
OP_LETI,
|
|
|
|
OP_LETW,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_MOD,
|
|
|
|
OP_MODI,
|
|
|
|
OP_MUL,
|
|
|
|
OP_MULI,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_NOP,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_OR,
|
|
|
|
OP_ORI,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_RET,
|
|
|
|
OP_REV,
|
|
|
|
OP_SET,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_SHIFT_LEFT,
|
|
|
|
OP_SHIFT_RIGHT,
|
2019-10-02 23:26:45 +08:00
|
|
|
OP_SUB,
|
|
|
|
OP_SUBI,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_SYNC,
|
|
|
|
OP_THREAD,
|
2019-10-02 23:26:45 +08:00
|
|
|
OP_XOR,
|
|
|
|
OP_XORI,
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
OP_JMP_E,
|
|
|
|
OP_JMPI_E,
|
|
|
|
OP_JMP_ON,
|
|
|
|
OP_JMP_OFF,
|
|
|
|
OP_JMP_NE,
|
|
|
|
OP_JMPI_NE,
|
|
|
|
OP_UJMP_G,
|
|
|
|
OP_UJMPI_G,
|
|
|
|
OP_JMP_G,
|
|
|
|
OP_JMPI_G,
|
|
|
|
OP_UJMP_L,
|
|
|
|
OP_UJMPI_L,
|
|
|
|
OP_JMP_L,
|
|
|
|
OP_JMPI_L,
|
|
|
|
OP_UJMP_GE,
|
|
|
|
OP_UJMPI_GE,
|
|
|
|
OP_JMP_GE,
|
|
|
|
OP_JMPI_GE,
|
|
|
|
OP_UJMP_LE,
|
|
|
|
OP_UJMPI_LE,
|
|
|
|
OP_JMP_LE,
|
|
|
|
OP_JMPI_LE,
|
2019-10-08 01:52:44 +08:00
|
|
|
OP_STACK_POP,
|
|
|
|
OP_STACK_PUSH,
|
|
|
|
OP_STACK_PUSHM,
|
|
|
|
OP_STACK_POPM,
|
2019-10-10 21:02:41 +08:00
|
|
|
Param,
|
|
|
|
Kind,
|
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
|
|
|
|
2019-10-10 17:56:43 +08:00
|
|
|
const REGISTERS_BASE_ADDRESS = 0x00a954b0;
|
2019-08-06 23:07:12 +08:00
|
|
|
const REGISTER_COUNT = 256;
|
|
|
|
const REGISTER_SIZE = 4;
|
2019-10-08 01:52:44 +08:00
|
|
|
const VARIABLE_STACK_LENGTH = 16; // TODO: verify this value
|
2019-10-10 21:02:41 +08:00
|
|
|
const ARG_STACK_SLOT_SIZE = 4;
|
|
|
|
const ARG_STACK_LENGTH = 8;
|
2019-08-06 23:07:12 +08:00
|
|
|
|
|
|
|
export enum ExecutionResult {
|
|
|
|
Ok,
|
|
|
|
WaitingVsync,
|
|
|
|
Halted,
|
|
|
|
}
|
|
|
|
|
2019-10-02 23:26:45 +08:00
|
|
|
type BinaryNumericOperation = (a: number, b: number) => number;
|
|
|
|
|
2019-10-03 23:28:48 +08:00
|
|
|
const numeric_ops: Record<
|
|
|
|
"add" | "sub" | "mul" | "div" | "idiv" | "mod" | "and" | "or" | "xor" | "shl" | "shr",
|
|
|
|
BinaryNumericOperation
|
|
|
|
> = {
|
2019-10-02 23:26:45 +08:00
|
|
|
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-10-03 01:16:05 +08:00
|
|
|
shl: (a, b) => a << b,
|
|
|
|
shr: (a, b) => a >>> b,
|
2019-10-02 23:26:45 +08:00
|
|
|
};
|
|
|
|
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
type ComparisonOperation = (a: number, b: number) => boolean;
|
|
|
|
|
2019-10-04 19:21:51 +08:00
|
|
|
const comparison_ops: Record<"eq" | "neq" | "gt" | "lt" | "gte" | "lte", ComparisonOperation> = {
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
eq: (a, b) => a === b,
|
|
|
|
neq: (a, b) => a !== b,
|
|
|
|
gt: (a, b) => a > b,
|
|
|
|
lt: (a, b) => a < b,
|
|
|
|
gte: (a, b) => a >= b,
|
|
|
|
lte: (a, b) => a <= b,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Short-circuiting fold.
|
|
|
|
*/
|
2019-10-11 14:14:34 +08:00
|
|
|
function andfold<T, A>(fn: (acc: A, cur: T) => A | undefined, init: A, lst: T[]): A | undefined {
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
let acc = init;
|
|
|
|
|
|
|
|
for (const item of lst) {
|
|
|
|
const new_val = fn(acc, item);
|
|
|
|
|
2019-10-11 14:14:34 +08:00
|
|
|
if (new_val === undefined) {
|
|
|
|
return undefined;
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
} else {
|
|
|
|
acc = new_val;
|
|
|
|
}
|
|
|
|
}
|
2019-10-04 19:21:51 +08:00
|
|
|
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
return acc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Short-circuiting reduce.
|
|
|
|
*/
|
2019-10-11 14:14:34 +08:00
|
|
|
function andreduce<T>(fn: (acc: T, cur: T) => T | undefined, lst: T[]): T | undefined {
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
return andfold(fn, lst[0], lst.slice(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Applies the given arguments to the given function.
|
2019-10-11 14:14:34 +08:00
|
|
|
* Returns the second argument if the function returns a truthy value, else undefined.
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
*/
|
2019-10-11 14:14:34 +08:00
|
|
|
function andsecond<T>(fn: (first: T, second: T) => any, first: T, second: T): T | undefined {
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
if (fn(first, second)) {
|
|
|
|
return second;
|
|
|
|
}
|
2019-10-11 14:14:34 +08:00
|
|
|
return undefined;
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
function rest<T>(lst: T[]): T[] {
|
|
|
|
return lst.slice(1);
|
|
|
|
}
|
|
|
|
|
2019-10-10 17:56:43 +08:00
|
|
|
type Range = [number, number];
|
|
|
|
|
|
|
|
function ranges_overlap(a: Range, b: Range): boolean {
|
|
|
|
return a[0] <= b[1] && b[0] <= a[1];
|
|
|
|
}
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
class VirtualMachineMemoryBuffer extends ArrayBuffer {
|
|
|
|
/**
|
|
|
|
* The memory this buffer belongs to.
|
|
|
|
*/
|
|
|
|
public readonly memory: VirtualMachineMemory;
|
|
|
|
/**
|
|
|
|
* The memory address of this buffer.
|
|
|
|
*/
|
|
|
|
public readonly address: number;
|
|
|
|
|
|
|
|
constructor(memory: VirtualMachineMemory, address: number, size: number) {
|
|
|
|
super(size);
|
|
|
|
this.memory = memory;
|
|
|
|
this.address = address;
|
|
|
|
}
|
|
|
|
|
2019-10-11 14:14:34 +08:00
|
|
|
public get_offset(byte_offset: number): VirtualMachineMemorySlot | undefined {
|
2019-10-10 21:02:41 +08:00
|
|
|
return this.memory.get(this.address + byte_offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
public free(): void {
|
|
|
|
this.memory.free(this.address);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-10 17:56:43 +08:00
|
|
|
/**
|
|
|
|
* Represents a single location in memory.
|
|
|
|
*/
|
|
|
|
class VirtualMachineMemorySlot {
|
2019-10-10 21:02:41 +08:00
|
|
|
/**
|
|
|
|
* The memory this slot belongs to.
|
|
|
|
*/
|
|
|
|
public readonly memory: VirtualMachineMemory;
|
|
|
|
/**
|
|
|
|
* The memory address this slots represents.
|
|
|
|
*/
|
|
|
|
public readonly address: number;
|
|
|
|
/**
|
|
|
|
* The allocated buffer this slot is a part of.
|
|
|
|
*/
|
|
|
|
public readonly buffer: VirtualMachineMemoryBuffer;
|
|
|
|
/**
|
|
|
|
* The offset that this slot represents in the buffer.
|
|
|
|
*/
|
|
|
|
public readonly byte_offset: number;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
memory: VirtualMachineMemory,
|
|
|
|
address: number,
|
|
|
|
buffer: VirtualMachineMemoryBuffer,
|
|
|
|
byte_offset: number,
|
|
|
|
) {
|
|
|
|
this.memory = memory;
|
|
|
|
this.address = address;
|
|
|
|
this.buffer = buffer;
|
|
|
|
this.byte_offset = byte_offset;
|
|
|
|
}
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps memory addresses to buffers.
|
|
|
|
*/
|
|
|
|
class VirtualMachineMemory {
|
|
|
|
private allocated_ranges: Range[] = [];
|
|
|
|
private ranges_sorted: boolean = true;
|
|
|
|
private memory: Map<number, VirtualMachineMemorySlot> = new Map();
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
private sort_ranges(): void {
|
2019-10-10 17:56:43 +08:00
|
|
|
this.allocated_ranges.sort((a, b) => a[0] - b[0]);
|
|
|
|
|
|
|
|
this.ranges_sorted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Would a buffer of the given size fit at the given address?
|
|
|
|
*/
|
|
|
|
private will_fit(address: number, size: number): boolean {
|
|
|
|
const fit_range: Range = [address, address + size - 1];
|
|
|
|
|
|
|
|
if (!this.ranges_sorted) {
|
|
|
|
this.sort_ranges();
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if it would overlap any already allocated space
|
|
|
|
for (const alloc_range of this.allocated_ranges) {
|
|
|
|
if (ranges_overlap(alloc_range, fit_range)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an address where a buffer of the given size would fit.
|
|
|
|
*/
|
|
|
|
private find_free_space(size: number): number {
|
|
|
|
let address = 0;
|
|
|
|
|
|
|
|
// nothing yet allocated, we can place it wherever
|
|
|
|
if (this.allocated_ranges.length < 1) {
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.ranges_sorted) {
|
|
|
|
this.sort_ranges();
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if buffer could fit in between allocated buffers
|
|
|
|
for (const alloc_range of this.allocated_ranges) {
|
|
|
|
if (!ranges_overlap(alloc_range, [address, address + size - 1])) {
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
|
|
|
address = alloc_range[1] + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// just place it at the end
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allocate a buffer of the given size at the given address.
|
|
|
|
* If the address is omitted a suitable location is chosen.
|
2019-10-10 21:02:41 +08:00
|
|
|
* @returns The allocated buffer.
|
2019-10-10 17:56:43 +08:00
|
|
|
*/
|
2019-10-10 21:02:41 +08:00
|
|
|
public allocate(size: number, address?: number): VirtualMachineMemoryBuffer {
|
2019-10-10 17:56:43 +08:00
|
|
|
if (size <= 0) {
|
|
|
|
throw new Error("Allocation failed: The size of the buffer must be greater than 0");
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if given address is good or find an address if none was given
|
|
|
|
if (address === undefined) {
|
|
|
|
address = this.find_free_space(size);
|
|
|
|
} else {
|
|
|
|
if (!this.will_fit(address, size)) {
|
|
|
|
throw new Error(
|
|
|
|
"Allocation failed: Cannot fit a buffer of the given size at the given address",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// save the range of allocated memory
|
|
|
|
this.allocated_ranges.push([address, address + size - 1]);
|
|
|
|
this.ranges_sorted = false;
|
|
|
|
|
|
|
|
// the actual buffer
|
2019-10-10 21:02:41 +08:00
|
|
|
const buf = new VirtualMachineMemoryBuffer(this, address, size);
|
2019-10-10 17:56:43 +08:00
|
|
|
|
|
|
|
// set addresses to correct buffer offsets
|
|
|
|
for (let offset = 0; offset < size; offset++) {
|
2019-10-10 21:02:41 +08:00
|
|
|
this.memory.set(
|
|
|
|
address + offset,
|
|
|
|
new VirtualMachineMemorySlot(this, address, buf, offset),
|
|
|
|
);
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
return buf;
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Free the memory allocated for the buffer at the given address.
|
|
|
|
*/
|
2019-10-10 21:02:41 +08:00
|
|
|
public free(address: number): void {
|
2019-10-10 17:56:43 +08:00
|
|
|
// check if address is a valid allocated buffer
|
2019-10-11 14:14:34 +08:00
|
|
|
let range: Range | undefined = undefined;
|
2019-10-10 17:56:43 +08:00
|
|
|
let range_idx = -1;
|
|
|
|
|
|
|
|
for (let i = 0; i < this.allocated_ranges.length; i++) {
|
|
|
|
const cur = this.allocated_ranges[i];
|
|
|
|
if (cur[0] === address) {
|
|
|
|
range = cur;
|
|
|
|
range_idx = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-11 14:14:34 +08:00
|
|
|
if (range === undefined) {
|
2019-10-10 17:56:43 +08:00
|
|
|
throw new Error("Free failed: Given address is not the start of an allocated buffer");
|
|
|
|
}
|
|
|
|
|
|
|
|
const [alloc_start, alloc_end] = range;
|
|
|
|
|
|
|
|
// remove addresses
|
|
|
|
for (let addr = alloc_start; addr <= alloc_end; addr++) {
|
|
|
|
this.memory.delete(addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove range
|
|
|
|
this.allocated_ranges.splice(range_idx, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-11 14:14:34 +08:00
|
|
|
* Gets the memory at the given address. Returns undefined if
|
2019-10-10 17:56:43 +08:00
|
|
|
* there is nothing allocated at the given address.
|
|
|
|
*/
|
2019-10-11 14:14:34 +08:00
|
|
|
public get(address: number): VirtualMachineMemorySlot | undefined {
|
2019-10-10 17:56:43 +08:00
|
|
|
if (this.memory.has(address)) {
|
|
|
|
return this.memory.get(address)!;
|
|
|
|
}
|
|
|
|
|
2019-10-11 14:14:34 +08:00
|
|
|
return undefined;
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
export class VirtualMachine {
|
2019-10-10 17:56:43 +08:00
|
|
|
private memory = new VirtualMachineMemory();
|
2019-10-10 21:02:41 +08:00
|
|
|
private register_store = this.memory.allocate(
|
2019-10-10 17:56:43 +08:00
|
|
|
REGISTER_SIZE * REGISTER_COUNT,
|
|
|
|
REGISTERS_BASE_ADDRESS,
|
2019-10-10 21:02:41 +08:00
|
|
|
)!;
|
2019-08-06 23:07:12 +08:00
|
|
|
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-10-10 21:02:41 +08:00
|
|
|
this.thread.push(
|
|
|
|
new Thread(
|
|
|
|
new ExecutionLocation(seg_idx!, 0),
|
|
|
|
this.memory.allocate(ARG_STACK_SLOT_SIZE * ARG_STACK_LENGTH),
|
|
|
|
true,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
private dispose_thread(thread_idx: number): void {
|
|
|
|
this.thread[thread_idx].dispose();
|
|
|
|
this.thread.splice(thread_idx, 1);
|
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);
|
|
|
|
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
const arg_vals = inst.args.map(arg => arg.value);
|
2019-10-10 21:02:41 +08:00
|
|
|
// eslint-disable-next-line
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = arg_vals;
|
|
|
|
|
|
|
|
// helper for conditional jump opcodes
|
2019-10-04 19:21:51 +08:00
|
|
|
const conditional_jump_args: (
|
|
|
|
cond: ComparisonOperation,
|
|
|
|
) => [Thread, number, ComparisonOperation, number, number] = cond => [
|
|
|
|
exec,
|
|
|
|
arg2,
|
|
|
|
cond,
|
|
|
|
arg0,
|
|
|
|
arg1,
|
|
|
|
];
|
2019-10-02 22:39:32 +08:00
|
|
|
|
2019-10-02 23:52:59 +08:00
|
|
|
switch (inst.opcode.code) {
|
|
|
|
case OP_NOP.code:
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_RET.code:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.pop_call_stack(this.thread_idx, exec);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SYNC.code:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.thread_idx++;
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_EXIT.code:
|
2019-08-06 23:07:12 +08:00
|
|
|
this.halt();
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_THREAD.code:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.start_thread(arg0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_LET.code:
|
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 23:52:59 +08:00
|
|
|
case OP_LETI.code:
|
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 23:52:59 +08:00
|
|
|
case OP_LETB.code:
|
|
|
|
case OP_LETW.code:
|
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 23:52:59 +08:00
|
|
|
case OP_SET.code:
|
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 23:52:59 +08:00
|
|
|
case OP_CLEAR.code:
|
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 23:52:59 +08:00
|
|
|
case OP_REV.code:
|
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 23:52:59 +08:00
|
|
|
case OP_CALL.code:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.push_call_stack(exec, arg0);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_JMP.code:
|
2019-10-02 22:39:32 +08:00
|
|
|
this.jump_to_label(exec, arg0);
|
2019-09-15 00:50:03 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHR.code:
|
2019-10-02 21:45:40 +08:00
|
|
|
// deref given register ref
|
2019-10-10 21:02:41 +08:00
|
|
|
exec.push_arg(this.get_sint(arg0), Kind.DWord);
|
2019-10-02 21:45:40 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHL.code:
|
2019-10-10 21:02:41 +08:00
|
|
|
exec.push_arg(inst.args[0].value, Kind.DWord);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHB.code:
|
2019-10-10 21:02:41 +08:00
|
|
|
exec.push_arg(inst.args[0].value, Kind.Byte);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHW.code:
|
2019-10-10 21:02:41 +08:00
|
|
|
exec.push_arg(inst.args[0].value, Kind.Word);
|
2019-09-16 04:20:49 +08:00
|
|
|
break;
|
2019-10-02 23:26:45 +08:00
|
|
|
// arithmetic operations
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ADD.code:
|
|
|
|
case OP_FADD.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.add);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ADDI.code:
|
|
|
|
case OP_FADDI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.add);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SUB.code:
|
|
|
|
case OP_FSUB.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.sub);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SUBI.code:
|
|
|
|
case OP_FSUBI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.sub);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MUL.code:
|
|
|
|
case OP_FMUL.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mul);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MULI.code:
|
|
|
|
case OP_FMULI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mul);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_DIV.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.idiv);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_FDIV.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.div);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_DIVI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.idiv);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_FDIVI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.div);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MOD.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mod);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MODI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mod);
|
|
|
|
break;
|
|
|
|
// bit operations
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_AND.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.and);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ANDI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.and);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_OR.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.or);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ORI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.or);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_XOR.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.xor);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_XORI.code:
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.xor);
|
|
|
|
break;
|
2019-10-03 01:16:05 +08:00
|
|
|
// 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;
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
// conditional jumps
|
|
|
|
case OP_JMP_ON.code:
|
|
|
|
// all eq 1?
|
2019-10-04 19:21:51 +08:00
|
|
|
this.conditional_jump(
|
|
|
|
exec,
|
|
|
|
arg0,
|
|
|
|
comparison_ops.eq,
|
|
|
|
1,
|
|
|
|
...rest(arg_vals).map(reg => this.get_sint(reg)),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_OFF.code:
|
|
|
|
// all eq 0?
|
2019-10-04 19:21:51 +08:00
|
|
|
this.conditional_jump(
|
|
|
|
exec,
|
|
|
|
arg0,
|
|
|
|
comparison_ops.eq,
|
|
|
|
0,
|
|
|
|
...rest(arg_vals).map(reg => this.get_sint(reg)),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_E.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.eq),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_E.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.eq),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_NE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.neq),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_NE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.neq),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMP_G.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.gt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMPI_G.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.gt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_G.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.gt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_G.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.gt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMP_L.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.lt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMPI_L.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.lt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_L.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.lt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_L.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.lt),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMP_GE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.gte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMPI_GE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.gte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_GE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.gte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_GE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.gte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMP_LE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.lte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_UJMPI_LE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.unsigned_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.lte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMP_LE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_register(
|
|
|
|
...conditional_jump_args(comparison_ops.lte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
|
|
|
case OP_JMPI_LE.code:
|
2019-10-04 19:21:51 +08:00
|
|
|
this.signed_conditional_jump_with_literal(
|
|
|
|
...conditional_jump_args(comparison_ops.lte),
|
|
|
|
);
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
break;
|
2019-10-08 01:52:44 +08:00
|
|
|
// variable stack operations
|
|
|
|
case OP_STACK_PUSH.code:
|
|
|
|
this.push_variable_stack(exec, arg0, 1);
|
|
|
|
break;
|
|
|
|
case OP_STACK_POP.code:
|
|
|
|
this.pop_variable_stack(exec, arg0, 1);
|
|
|
|
break;
|
|
|
|
case OP_STACK_PUSHM.code:
|
|
|
|
this.push_variable_stack(exec, arg0, arg1);
|
|
|
|
break;
|
|
|
|
case OP_STACK_POPM.code:
|
|
|
|
this.pop_variable_stack(exec, arg0, arg1);
|
|
|
|
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
|
2019-10-10 21:02:41 +08:00
|
|
|
this.dispose_thread(this.thread_idx);
|
2019-09-15 00:50:03 +08:00
|
|
|
} 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);
|
|
|
|
}
|
|
|
|
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
private get_uint(reg: number): number {
|
|
|
|
return this.registers.getUint32(REGISTER_SIZE * reg);
|
|
|
|
}
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
private set_uint(reg: number, value: number): void {
|
|
|
|
this.registers.setUint32(REGISTER_SIZE * reg, value);
|
|
|
|
}
|
|
|
|
|
2019-10-03 23:28:48 +08:00
|
|
|
private do_numeric_op_with_register(
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
2019-10-02 23:26:45 +08:00
|
|
|
this.do_numeric_op_with_literal(reg1, this.get_sint(reg2), op);
|
|
|
|
}
|
|
|
|
|
2019-10-03 23:28:48 +08:00
|
|
|
private do_numeric_op_with_literal(
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
2019-10-02 23:26:45 +08:00
|
|
|
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-10-03 23:28:48 +08:00
|
|
|
private jump_to_label(exec: Thread, label: number): void {
|
2019-09-15 00:50:03 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
private signed_conditional_jump_with_register(
|
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
): void {
|
|
|
|
this.conditional_jump(exec, label, condition, this.get_sint(reg1), this.get_sint(reg2));
|
|
|
|
}
|
|
|
|
|
|
|
|
private signed_conditional_jump_with_literal(
|
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
): void {
|
|
|
|
this.conditional_jump(exec, label, condition, this.get_sint(reg), literal);
|
|
|
|
}
|
|
|
|
|
|
|
|
private unsigned_conditional_jump_with_register(
|
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
): void {
|
|
|
|
this.conditional_jump(exec, label, condition, this.get_uint(reg1), this.get_uint(reg2));
|
|
|
|
}
|
|
|
|
|
|
|
|
private unsigned_conditional_jump_with_literal(
|
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
): void {
|
|
|
|
this.conditional_jump(exec, label, condition, this.get_uint(reg), literal);
|
|
|
|
}
|
|
|
|
|
|
|
|
private conditional_jump(
|
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
...vals: number[]
|
|
|
|
): void {
|
2019-10-04 19:21:51 +08:00
|
|
|
const chain_cmp = andsecond.bind<
|
|
|
|
null,
|
|
|
|
ComparisonOperation,
|
|
|
|
Parameters<ComparisonOperation>,
|
|
|
|
any
|
|
|
|
>(null, condition);
|
2019-10-11 14:14:34 +08:00
|
|
|
if (andreduce(chain_cmp, vals) !== undefined) {
|
[VM] Implemented conditional jump opcodes.
jmp_on, jmp_off, jmp_=, jmpi_=, jmp_!=, jmpi_!=, ujmp_>, ujmpi_>, jmp_>, jmpi_>, ujmp_<, ujmpi_<, jmp_<, jmpi_<, ujmp_>=, ujmpi_>=, jmp_>=, jmpi_>=, ujmp_<=, ujmpi_<=, jmp_<=, jmpi_<=
2019-10-04 05:18:57 +08:00
|
|
|
this.jump_to_label(exec, label);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-08 01:52:44 +08:00
|
|
|
private push_variable_stack(exec: Thread, base_reg: number, num_push: number): void {
|
|
|
|
const end = base_reg + num_push;
|
|
|
|
|
|
|
|
if (end > REGISTER_COUNT) {
|
|
|
|
throw new Error("Variable stack: Invalid register");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exec.variable_stack.length + num_push > VARIABLE_STACK_LENGTH) {
|
|
|
|
throw new Error("Variable stack: Stack overflow");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let r = base_reg; r < end; r++) {
|
|
|
|
exec.variable_stack.push(this.get_uint(r));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private pop_variable_stack(exec: Thread, base_reg: number, num_pop: number): void {
|
|
|
|
const end = base_reg + num_pop;
|
|
|
|
|
|
|
|
if (end > REGISTER_COUNT) {
|
|
|
|
throw new Error("Variable stack: Invalid register");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exec.variable_stack.length < num_pop) {
|
|
|
|
throw new Error("Variable stack: Stack underflow");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let r = end - 1; r >= base_reg; r--) {
|
|
|
|
this.set_uint(r, exec.variable_stack.pop()!);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-10-10 17:56:43 +08:00
|
|
|
|
|
|
|
private get_register_address(reg: number): number {
|
2019-10-10 21:02:41 +08:00
|
|
|
return this.register_store.address + reg * REGISTER_SIZE;
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
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) {}
|
|
|
|
}
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
type ArgStackTypeList = [Kind, Kind, Kind, Kind, Kind, Kind, Kind, Kind];
|
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
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-10-10 21:02:41 +08:00
|
|
|
|
|
|
|
private arg_stack_buf: VirtualMachineMemoryBuffer;
|
|
|
|
private arg_stack_counter: number = 0;
|
|
|
|
private arg_stack: DataView;
|
|
|
|
private arg_stack_types: ArgStackTypeList = Array(ARG_STACK_LENGTH).fill(
|
|
|
|
Kind.Any,
|
|
|
|
) as ArgStackTypeList;
|
|
|
|
|
2019-10-08 01:52:44 +08:00
|
|
|
public variable_stack: number[] = [];
|
2019-08-06 23:07:12 +08:00
|
|
|
/**
|
|
|
|
* Global or floor-local?
|
|
|
|
*/
|
|
|
|
public global: boolean;
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
constructor(
|
|
|
|
next: ExecutionLocation,
|
|
|
|
arg_stack_buf: VirtualMachineMemoryBuffer,
|
|
|
|
global: boolean,
|
|
|
|
) {
|
|
|
|
this.call_stack = [next];
|
|
|
|
this.global = global;
|
|
|
|
|
|
|
|
this.arg_stack_buf = arg_stack_buf;
|
|
|
|
this.arg_stack = new DataView(arg_stack_buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
public call_stack_top(): ExecutionLocation {
|
2019-09-15 00:50:03 +08:00
|
|
|
return this.call_stack[this.call_stack.length - 1];
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-10-10 21:02:41 +08:00
|
|
|
public push_arg(data: number, type: Kind): void {
|
|
|
|
if (this.arg_stack_counter >= ARG_STACK_LENGTH) {
|
|
|
|
throw new Error("Argument stack: Stack overflow");
|
|
|
|
}
|
|
|
|
|
|
|
|
this.arg_stack.setUint32(this.arg_stack_counter * ARG_STACK_SLOT_SIZE, data, true);
|
|
|
|
this.arg_stack_types[this.arg_stack_counter] = type;
|
|
|
|
|
|
|
|
this.arg_stack_counter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
public fetch_args(params: readonly Param[]): number[] {
|
|
|
|
const args: number[] = [];
|
|
|
|
|
|
|
|
if (params.length !== this.arg_stack_counter) {
|
|
|
|
logger.warn("Argument stack: Argument count mismatch");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < params.length; i++) {
|
|
|
|
const param = params[i];
|
|
|
|
|
|
|
|
if (param.type.kind !== this.arg_stack_types[i]) {
|
|
|
|
logger.warn("Argument stack: Argument type mismatch");
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (param.type.kind) {
|
|
|
|
case Kind.Byte:
|
|
|
|
args.push(this.arg_stack.getUint8(i));
|
|
|
|
break;
|
|
|
|
case Kind.Word:
|
|
|
|
args.push(this.arg_stack.getUint16(i, true));
|
|
|
|
break;
|
|
|
|
case Kind.DWord:
|
|
|
|
case Kind.String:
|
|
|
|
args.push(this.arg_stack.getUint32(i, true));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(`Unhandled param kind: Kind.${Kind[param.type.kind]}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.arg_stack_counter = 0;
|
|
|
|
|
|
|
|
return args;
|
|
|
|
}
|
|
|
|
|
|
|
|
public dispose(): void {
|
|
|
|
this.arg_stack_buf.free();
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
}
|