2020-01-03 01:42:08 +08:00
|
|
|
import { Segment, SegmentType } from "../../../core/data_formats/asm/instructions";
|
2019-10-03 23:28:48 +08:00
|
|
|
import {
|
2019-12-19 05:37:26 +08:00
|
|
|
Kind,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_ADD,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_ADD_MSG,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_ADDI,
|
|
|
|
OP_AND,
|
|
|
|
OP_ANDI,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_ARG_PUSHA,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_ARG_PUSHB,
|
|
|
|
OP_ARG_PUSHL,
|
|
|
|
OP_ARG_PUSHR,
|
2019-10-11 16:11:06 +08:00
|
|
|
OP_ARG_PUSHS,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_ARG_PUSHW,
|
2019-12-20 07:11:54 +08:00
|
|
|
OP_BB_MAP_DESIGNATE,
|
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,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_FLET,
|
|
|
|
OP_FLETI,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_FMUL,
|
|
|
|
OP_FMULI,
|
|
|
|
OP_FSUB,
|
|
|
|
OP_FSUBI,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_GET_RANDOM,
|
|
|
|
OP_GETTIME,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_JMP,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_JMP_E,
|
|
|
|
OP_JMP_G,
|
|
|
|
OP_JMP_GE,
|
|
|
|
OP_JMP_L,
|
|
|
|
OP_JMP_LE,
|
|
|
|
OP_JMP_NE,
|
|
|
|
OP_JMP_OFF,
|
|
|
|
OP_JMP_ON,
|
|
|
|
OP_JMPI_E,
|
|
|
|
OP_JMPI_G,
|
|
|
|
OP_JMPI_GE,
|
|
|
|
OP_JMPI_L,
|
|
|
|
OP_JMPI_LE,
|
|
|
|
OP_JMPI_NE,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_LET,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_LETA,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_LETB,
|
|
|
|
OP_LETI,
|
|
|
|
OP_LETW,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_LIST,
|
2020-01-03 01:54:18 +08:00
|
|
|
OP_MAP_DESIGNATE,
|
|
|
|
OP_MAP_DESIGNATE_EX,
|
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-12-21 04:13:59 +08:00
|
|
|
OP_P_DEAD_V3,
|
2019-10-02 20:25:47 +08:00
|
|
|
OP_RET,
|
|
|
|
OP_REV,
|
|
|
|
OP_SET,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_SET_EPISODE,
|
2019-12-20 05:14:59 +08:00
|
|
|
OP_SET_FLOOR_HANDLER,
|
2019-10-03 23:28:48 +08:00
|
|
|
OP_SHIFT_LEFT,
|
|
|
|
OP_SHIFT_RIGHT,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_STACK_POP,
|
|
|
|
OP_STACK_POPM,
|
|
|
|
OP_STACK_PUSH,
|
|
|
|
OP_STACK_PUSHM,
|
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-12-21 04:03:15 +08:00
|
|
|
OP_THREAD_STG,
|
[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_UJMP_G,
|
|
|
|
OP_UJMP_GE,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_UJMP_L,
|
[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_UJMP_LE,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_UJMPI_G,
|
|
|
|
OP_UJMPI_GE,
|
|
|
|
OP_UJMPI_L,
|
[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_UJMPI_LE,
|
2019-10-11 17:49:20 +08:00
|
|
|
OP_WINDOW_MSG,
|
|
|
|
OP_WINEND,
|
2019-12-19 05:37:26 +08:00
|
|
|
OP_XOR,
|
|
|
|
OP_XORI,
|
2020-01-03 01:42:08 +08:00
|
|
|
} from "../../../core/data_formats/asm/opcodes";
|
2019-10-15 20:34:30 +08:00
|
|
|
import {
|
2019-12-19 05:37:26 +08:00
|
|
|
andreduce,
|
|
|
|
andsecond,
|
|
|
|
BinaryNumericOperation,
|
2019-10-15 20:34:30 +08:00
|
|
|
ComparisonOperation,
|
|
|
|
numeric_ops,
|
|
|
|
rest,
|
|
|
|
} from "./utils";
|
2019-12-20 07:11:54 +08:00
|
|
|
import { DefaultVirtualMachineIO, VirtualMachineIO } from "./io";
|
2019-11-02 02:20:25 +08:00
|
|
|
import { Episode } from "../../../core/data_formats/parsing/quest/Episode";
|
2019-11-16 06:16:09 +08:00
|
|
|
import { Endianness } from "../../../core/data_formats/Endianness";
|
2019-12-19 06:40:38 +08:00
|
|
|
import { Random } from "./Random";
|
2019-12-19 20:58:11 +08:00
|
|
|
import { Memory } from "./Memory";
|
2019-12-20 23:10:50 +08:00
|
|
|
import { InstructionPointer } from "./InstructionPointer";
|
2019-12-21 00:57:34 +08:00
|
|
|
import { StackFrame, StepMode, Thread } from "./Thread";
|
2019-12-25 07:17:02 +08:00
|
|
|
import { LogManager } from "../../../core/Logger";
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-11-22 05:04:16 +08:00
|
|
|
export const REGISTER_COUNT = 256;
|
2019-12-20 03:45:02 +08:00
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
const REGISTER_SIZE = 4;
|
2019-10-08 01:52:44 +08:00
|
|
|
const VARIABLE_STACK_LENGTH = 16; // TODO: verify this value
|
2019-10-11 16:11:06 +08:00
|
|
|
const STRING_ARG_STORE_ADDRESS = 0x00a92700;
|
|
|
|
const STRING_ARG_STORE_SIZE = 1024; // TODO: verify this value
|
2019-11-02 02:20:25 +08:00
|
|
|
const ENTRY_SEGMENT = 0;
|
2019-11-11 21:21:13 +08:00
|
|
|
const LIST_ITEM_DELIMITER = "\n";
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-12-25 07:17:02 +08:00
|
|
|
const logger = LogManager.get("quest_editor/scripting/vm/VirtualMachine");
|
2019-12-20 03:45:02 +08:00
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
export enum ExecutionResult {
|
2019-12-20 03:45:02 +08:00
|
|
|
/**
|
2019-12-20 23:10:50 +08:00
|
|
|
* There are no live threads, nothing to do.
|
2019-12-20 03:45:02 +08:00
|
|
|
*/
|
2019-12-20 23:10:50 +08:00
|
|
|
Suspended,
|
2019-12-20 03:45:02 +08:00
|
|
|
/**
|
2019-12-20 23:10:50 +08:00
|
|
|
* Execution is paused due to hitting a breakpoint or because the VM is executing in
|
|
|
|
* stepping mode.
|
2019-12-20 03:45:02 +08:00
|
|
|
*/
|
2019-12-20 23:10:50 +08:00
|
|
|
Paused,
|
2019-12-20 03:45:02 +08:00
|
|
|
/**
|
2019-12-20 23:10:50 +08:00
|
|
|
* All threads have yielded.
|
2019-12-20 03:45:02 +08:00
|
|
|
*/
|
2019-08-06 23:07:12 +08:00
|
|
|
WaitingVsync,
|
2019-11-11 21:21:13 +08:00
|
|
|
/**
|
|
|
|
* Waiting for any keypress. No method call required.
|
|
|
|
*/
|
|
|
|
WaitingInput,
|
|
|
|
/**
|
|
|
|
* Waiting for a value to be selected in a list.
|
|
|
|
* Call `list_select` to set selection.
|
|
|
|
*/
|
|
|
|
WaitingSelection,
|
2019-12-20 23:10:50 +08:00
|
|
|
/**
|
|
|
|
* Execution has halted because the VM encountered an `exit` instruction, a fatal error was
|
|
|
|
* raised or the VM was halted from outside.
|
|
|
|
*/
|
2019-12-20 03:45:02 +08:00
|
|
|
Halted,
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-11-02 02:20:25 +08:00
|
|
|
function encode_episode_number(ep: Episode): number {
|
|
|
|
switch (ep) {
|
|
|
|
case Episode.I:
|
|
|
|
return 0;
|
|
|
|
case Episode.II:
|
|
|
|
return 1;
|
|
|
|
case Episode.IV:
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
/**
|
|
|
|
* This class emulates the PSO script engine. It's in charge of memory, threading and executing
|
|
|
|
* instructions.
|
|
|
|
*/
|
2019-08-06 23:07:12 +08:00
|
|
|
export class VirtualMachine {
|
2019-12-23 06:15:05 +08:00
|
|
|
// Quest details.
|
|
|
|
|
2019-12-19 21:35:13 +08:00
|
|
|
private episode: Episode = Episode.I;
|
2019-12-23 06:15:05 +08:00
|
|
|
private _object_code: readonly Segment[] = [];
|
2019-12-20 23:10:50 +08:00
|
|
|
private readonly label_to_seg_idx: Map<number, number> = new Map();
|
2019-12-23 06:15:05 +08:00
|
|
|
|
|
|
|
// VM state.
|
|
|
|
|
|
|
|
private readonly registers = new Memory(REGISTER_COUNT * REGISTER_SIZE, Endianness.Little);
|
|
|
|
private string_arg_store = "";
|
2019-12-20 23:10:50 +08:00
|
|
|
private threads: Thread[] = [];
|
2019-08-06 23:07:12 +08:00
|
|
|
private thread_idx = 0;
|
2019-10-11 17:49:20 +08:00
|
|
|
private window_msg_open = false;
|
2019-11-02 02:20:25 +08:00
|
|
|
private set_episode_called = false;
|
2019-11-11 21:21:13 +08:00
|
|
|
private list_open = false;
|
|
|
|
private selection_reg = 0;
|
2019-12-20 23:10:50 +08:00
|
|
|
private _halted = true;
|
2019-12-23 06:15:05 +08:00
|
|
|
|
|
|
|
// Debugging.
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private readonly breakpoints: InstructionPointer[] = [];
|
2019-12-23 06:15:05 +08:00
|
|
|
private paused = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set of unsupported opcodes that have already been logged. Each unsupported opcode will only
|
|
|
|
* be logged once to avoid flooding the log with duplicate log messages.
|
|
|
|
*/
|
|
|
|
private readonly unsupported_opcodes_logged: Set<number> = new Set();
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
get object_code(): readonly Segment[] {
|
|
|
|
return this._object_code;
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
get halted(): boolean {
|
|
|
|
return this._halted;
|
|
|
|
}
|
|
|
|
|
2019-12-21 00:57:34 +08:00
|
|
|
set step_mode(step_mode: StepMode) {
|
|
|
|
if (step_mode != undefined) {
|
|
|
|
const thread = this.current_thread();
|
|
|
|
|
|
|
|
if (thread) {
|
|
|
|
thread.step_mode = step_mode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 06:40:38 +08:00
|
|
|
constructor(
|
2019-12-20 07:11:54 +08:00
|
|
|
private io: VirtualMachineIO = new DefaultVirtualMachineIO(),
|
2019-12-19 06:40:38 +08:00
|
|
|
private random: Random = new Random(),
|
|
|
|
) {}
|
2019-10-17 19:21:45 +08:00
|
|
|
|
2019-08-06 23:07:12 +08:00
|
|
|
/**
|
|
|
|
* Halts and resets the VM, then loads new object code.
|
|
|
|
*/
|
2019-12-19 21:35:13 +08:00
|
|
|
load_object_code(object_code: readonly Segment[], episode: Episode): void {
|
2019-08-06 23:07:12 +08:00
|
|
|
this.halt();
|
2019-12-20 03:45:02 +08:00
|
|
|
|
|
|
|
logger.debug("Starting.");
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
this._object_code = object_code;
|
2019-12-19 21:35:13 +08:00
|
|
|
this.episode = episode;
|
2019-11-02 02:20:25 +08:00
|
|
|
|
2019-12-24 10:04:18 +08:00
|
|
|
this.label_to_seg_idx.clear();
|
2019-08-06 23:07:12 +08:00
|
|
|
let i = 0;
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
for (const segment of this._object_code) {
|
2019-08-06 23:07:12 +08:00
|
|
|
for (const label of segment.labels) {
|
|
|
|
this.label_to_seg_idx.set(label, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
2019-12-20 23:10:50 +08:00
|
|
|
|
|
|
|
this._halted = false;
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Schedules concurrent execution of the code at the given label.
|
2019-12-21 04:03:15 +08:00
|
|
|
*
|
|
|
|
* @param label - instruction label to start thread execution at.
|
|
|
|
* @param area_id - if an area_id is passed in, the thread is floor-local.
|
2019-08-06 23:07:12 +08:00
|
|
|
*/
|
2019-12-21 04:03:15 +08:00
|
|
|
start_thread(label: number, area_id?: number): void {
|
2019-11-15 01:46:53 +08:00
|
|
|
const seg_idx = this.get_segment_index_by_label(label);
|
2019-12-19 05:37:26 +08:00
|
|
|
const segment = this._object_code[seg_idx];
|
2019-08-06 23:07:12 +08:00
|
|
|
|
|
|
|
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-12-20 23:10:50 +08:00
|
|
|
this.threads.push(
|
2019-12-21 04:03:15 +08:00
|
|
|
new Thread(this.io, new InstructionPointer(seg_idx!, 0, this.object_code), area_id),
|
2019-12-20 23:10:50 +08:00
|
|
|
);
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-20 23:10:50 +08:00
|
|
|
* Executes instructions while possible.
|
2019-11-16 01:30:51 +08:00
|
|
|
*/
|
2019-12-20 23:10:50 +08:00
|
|
|
execute(): ExecutionResult {
|
|
|
|
if (this._halted) {
|
|
|
|
return ExecutionResult.Halted;
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
let inst_ptr: InstructionPointer | undefined;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Limit amount of instructions executed to prevent infinite loops.
|
|
|
|
let execution_counter = 0;
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
while (execution_counter++ < 10_000) {
|
2019-12-21 04:03:15 +08:00
|
|
|
// Check whether VM is waiting for vsync or is suspended.
|
2019-12-21 00:57:34 +08:00
|
|
|
if (this.threads.length >= 1 && this.thread_idx >= this.threads.length) {
|
|
|
|
return ExecutionResult.WaitingVsync;
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
const thread = this.current_thread();
|
|
|
|
|
|
|
|
if (!thread) {
|
|
|
|
return ExecutionResult.Suspended;
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
// Get current instruction.
|
|
|
|
const frame = thread.current_stack_frame()!;
|
2019-12-20 23:10:50 +08:00
|
|
|
inst_ptr = frame.instruction_pointer;
|
|
|
|
const inst = inst_ptr.instruction;
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
// Check whether the VM needs to pause only if it's not already paused. In that case
|
|
|
|
// it's resuming.
|
2019-12-21 00:57:34 +08:00
|
|
|
if (!this.paused) {
|
|
|
|
switch (thread.step_mode) {
|
|
|
|
case StepMode.BreakPoint:
|
2019-12-20 23:10:50 +08:00
|
|
|
if (this.breakpoints.findIndex(bp => bp.equals(inst_ptr!)) !== -1) {
|
2019-12-21 00:57:34 +08:00
|
|
|
this.paused = true;
|
2019-12-20 23:10:50 +08:00
|
|
|
return ExecutionResult.Paused;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-12-21 00:57:34 +08:00
|
|
|
case StepMode.Over:
|
|
|
|
if (
|
|
|
|
thread.step_frame &&
|
|
|
|
frame.idx <= thread.step_frame.idx &&
|
|
|
|
inst.asm?.mnemonic
|
|
|
|
) {
|
|
|
|
this.paused = true;
|
2019-12-20 23:10:50 +08:00
|
|
|
return ExecutionResult.Paused;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-12-21 00:57:34 +08:00
|
|
|
case StepMode.In:
|
|
|
|
if (inst.asm?.mnemonic) {
|
|
|
|
this.paused = true;
|
|
|
|
return ExecutionResult.Paused;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case StepMode.Out:
|
|
|
|
if (
|
|
|
|
thread.step_frame &&
|
|
|
|
frame.idx < thread.step_frame.idx &&
|
|
|
|
inst.asm?.mnemonic
|
|
|
|
) {
|
|
|
|
this.paused = true;
|
2019-12-20 23:10:50 +08:00
|
|
|
return ExecutionResult.Paused;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
// Not paused, the next instruction can be executed.
|
2019-12-21 01:08:04 +08:00
|
|
|
this.paused = false;
|
|
|
|
|
2019-12-24 10:04:18 +08:00
|
|
|
const result = this.execute_instruction(thread, inst_ptr);
|
2019-11-12 00:07:26 +08:00
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
// Only return WaitingVsync when all threads have yielded.
|
|
|
|
if (result != undefined && result !== ExecutionResult.WaitingVsync) {
|
2019-12-21 00:57:34 +08:00
|
|
|
return result;
|
|
|
|
}
|
2019-12-20 23:10:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error(
|
|
|
|
"Maximum execution count reached. The code probably contains an infinite loop.",
|
|
|
|
);
|
2019-11-17 00:07:21 +08:00
|
|
|
} catch (thrown) {
|
|
|
|
let err = thrown;
|
|
|
|
|
2019-10-17 22:50:19 +08:00
|
|
|
if (!(err instanceof Error)) {
|
|
|
|
err = new Error(String(err));
|
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
try {
|
2019-12-23 06:15:05 +08:00
|
|
|
this.io.error(err, inst_ptr);
|
2019-12-21 04:03:15 +08:00
|
|
|
} finally {
|
|
|
|
this.halt();
|
|
|
|
}
|
2019-10-17 19:21:45 +08:00
|
|
|
|
|
|
|
return ExecutionResult.Halted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 07:11:54 +08:00
|
|
|
/**
|
|
|
|
* Signal to the VM that a vsync has happened.
|
|
|
|
*/
|
|
|
|
vsync(): void {
|
2019-12-20 23:10:50 +08:00
|
|
|
if (this.thread_idx >= this.threads.length) {
|
2019-12-20 07:11:54 +08:00
|
|
|
this.thread_idx = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Halts execution of all threads.
|
|
|
|
*/
|
|
|
|
halt(): void {
|
2019-12-20 23:10:50 +08:00
|
|
|
if (!this._halted) {
|
2019-12-20 07:11:54 +08:00
|
|
|
logger.debug("Halting.");
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
this.registers.zero();
|
|
|
|
this.string_arg_store = "";
|
|
|
|
this.threads = [];
|
2019-12-20 07:11:54 +08:00
|
|
|
this.thread_idx = 0;
|
2019-12-20 23:10:50 +08:00
|
|
|
this.window_msg_open = false;
|
|
|
|
this.set_episode_called = false;
|
|
|
|
this.list_open = false;
|
|
|
|
this.selection_reg = 0;
|
|
|
|
this._halted = true;
|
2019-12-21 00:57:34 +08:00
|
|
|
this.paused = false;
|
2019-12-20 23:10:50 +08:00
|
|
|
this.breakpoints.splice(0, Infinity);
|
2019-12-23 06:15:05 +08:00
|
|
|
this.unsupported_opcodes_logged.clear();
|
2019-12-20 07:11:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
get_current_stack_frame(): StackFrame | undefined {
|
|
|
|
return this.current_thread()?.current_stack_frame();
|
2019-12-20 07:11:54 +08:00
|
|
|
}
|
|
|
|
|
2019-12-24 10:04:18 +08:00
|
|
|
get_instruction_pointer(): InstructionPointer | undefined {
|
2019-12-20 23:10:50 +08:00
|
|
|
return this.get_current_stack_frame()?.instruction_pointer;
|
2019-12-20 07:11:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
get_segment_index_by_label(label: number): number {
|
|
|
|
if (!this.label_to_seg_idx.has(label)) {
|
|
|
|
throw new Error(`Invalid argument: No such label ${label}.`);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.label_to_seg_idx.get(label)!;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When the list opcode is used, call this method to select a value in the list.
|
|
|
|
*/
|
|
|
|
list_select(idx: number): void {
|
|
|
|
if (!this.list_open) {
|
|
|
|
throw new Error("list_select may not be called if there is no list open");
|
|
|
|
}
|
|
|
|
this.set_register_unsigned(this.selection_reg, idx);
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
set_breakpoint(breakpoint: InstructionPointer): void {
|
|
|
|
if (this.breakpoints.findIndex(bp => bp.equals(breakpoint)) === -1) {
|
|
|
|
this.breakpoints.push(breakpoint);
|
|
|
|
}
|
|
|
|
}
|
2019-12-20 07:11:54 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
remove_breakpoint(breakpoint: InstructionPointer): void {
|
|
|
|
const index = this.breakpoints.findIndex(bp => bp.equals(breakpoint));
|
|
|
|
|
|
|
|
if (index !== -1) {
|
|
|
|
this.breakpoints.splice(index, 1);
|
2019-12-20 07:11:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private current_thread(): Thread | undefined {
|
|
|
|
return this.threads[this.thread_idx];
|
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
private terminate_thread(thread_idx: number): void {
|
2019-12-20 23:10:50 +08:00
|
|
|
this.threads.splice(thread_idx, 1);
|
|
|
|
|
|
|
|
if (this.thread_idx >= thread_idx && this.thread_idx > 0) {
|
|
|
|
this.thread_idx--;
|
2019-12-20 07:11:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
/**
|
|
|
|
* Advance to the next instruction.
|
|
|
|
*/
|
|
|
|
private advance(thread: Thread): void {
|
|
|
|
const frame = thread.current_stack_frame();
|
|
|
|
if (!frame) return; // Thread already terminated.
|
|
|
|
|
|
|
|
const next = frame.instruction_pointer.next();
|
|
|
|
|
|
|
|
if (next) {
|
|
|
|
thread.set_current_instruction_pointer(next);
|
|
|
|
} else {
|
|
|
|
// Reached EOF.
|
2020-04-27 02:55:18 +08:00
|
|
|
// Game will crash if call stack is not empty.
|
|
|
|
if (thread.call_stack.length > 0) {
|
|
|
|
throw new Error("Reached EOF but call stack was not empty");
|
|
|
|
}
|
2019-12-21 04:03:15 +08:00
|
|
|
thread.pop_call_stack();
|
|
|
|
this.terminate_thread(this.thread_idx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-02 02:20:25 +08:00
|
|
|
private execute_instruction(
|
2019-12-20 23:10:50 +08:00
|
|
|
thread: Thread,
|
2019-12-23 06:15:05 +08:00
|
|
|
inst_ptr: InstructionPointer,
|
2019-12-20 23:10:50 +08:00
|
|
|
): ExecutionResult | undefined {
|
2019-12-23 06:15:05 +08:00
|
|
|
const inst = inst_ptr.instruction;
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
let result: ExecutionResult | undefined = undefined;
|
2019-12-21 04:03:15 +08:00
|
|
|
let advance = true;
|
2019-11-11 21:21:13 +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
|
|
|
const arg_vals = inst.args.map(arg => arg.value);
|
2020-01-03 01:54:18 +08:00
|
|
|
const [arg0, arg1, arg2] = arg_vals;
|
[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-11-11 21:21:13 +08:00
|
|
|
// previous instruction must've been `list`.
|
|
|
|
// list may not exist after the instruction
|
|
|
|
if (this.list_open) {
|
|
|
|
this.list_open = false;
|
|
|
|
}
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
const stack_args = thread.fetch_args(inst_ptr);
|
2019-12-19 05:47:53 +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-12-20 23:10:50 +08:00
|
|
|
this.pop_call_stack(this.thread_idx);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SYNC.code:
|
2019-12-21 00:57:34 +08:00
|
|
|
result = ExecutionResult.WaitingVsync;
|
2019-12-21 04:03:15 +08:00
|
|
|
this.advance(thread);
|
|
|
|
this.thread_idx++;
|
|
|
|
advance = false;
|
2019-08-06 23:07:12 +08:00
|
|
|
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-18 00:58:31 +08:00
|
|
|
// integer lets
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_LET.code:
|
2019-10-17 19:45:59 +08:00
|
|
|
this.set_register_signed(arg0, this.get_register_signed(arg1));
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_LETI.code:
|
2019-10-17 19:45:59 +08:00
|
|
|
this.set_register_signed(arg0, arg1);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_LETB.code:
|
2019-10-18 01:02:53 +08:00
|
|
|
this.set_register_byte(arg0, arg1);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_LETW.code:
|
2019-10-18 01:02:53 +08:00
|
|
|
this.set_register_word(arg0, arg1);
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-18 00:58:31 +08:00
|
|
|
case OP_LETA.code:
|
|
|
|
this.set_register_unsigned(arg0, this.get_register_address(arg0));
|
|
|
|
break;
|
|
|
|
// float lets
|
|
|
|
case OP_FLET.code:
|
|
|
|
this.set_register_float(arg0, this.get_register_float(arg1));
|
|
|
|
break;
|
|
|
|
case OP_FLETI.code:
|
|
|
|
this.set_register_float(arg0, arg1);
|
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SET.code:
|
2019-10-17 19:45:59 +08:00
|
|
|
this.set_register_signed(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-17 19:45:59 +08:00
|
|
|
this.set_register_signed(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-17 19:45:59 +08:00
|
|
|
this.set_register_signed(arg0, this.get_register_signed(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-12-20 23:10:50 +08:00
|
|
|
this.push_call_stack(thread, arg0);
|
2019-12-21 04:03:15 +08:00
|
|
|
advance = false;
|
2019-08-06 23:07:12 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_JMP.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
this.jump_to_label(thread, arg0);
|
2019-12-23 06:28:11 +08:00
|
|
|
advance = false;
|
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-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(this.get_register_signed(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-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(inst.args[0].value, Kind.DWord);
|
2019-10-10 21:02:41 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHB.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(inst.args[0].value, Kind.Byte);
|
2019-10-10 21:02:41 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ARG_PUSHW.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(inst.args[0].value, Kind.Word);
|
2019-09-16 04:20:49 +08:00
|
|
|
break;
|
2019-10-11 16:11:06 +08:00
|
|
|
case OP_ARG_PUSHA.code:
|
|
|
|
// push address of register
|
2019-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(this.get_register_address(inst.args[0].value), Kind.DWord);
|
2019-10-11 16:11:06 +08:00
|
|
|
break;
|
|
|
|
case OP_ARG_PUSHS.code:
|
2019-11-14 01:10:30 +08:00
|
|
|
if (typeof arg0 === "string") {
|
|
|
|
// process tags
|
|
|
|
const string_arg = this.parse_template_string(arg0);
|
2019-10-11 16:11:06 +08:00
|
|
|
// store string and push its address
|
2019-11-16 06:16:09 +08:00
|
|
|
this.string_arg_store = string_arg.slice(0, STRING_ARG_STORE_SIZE / 2);
|
2019-12-20 23:10:50 +08:00
|
|
|
thread.push_arg(STRING_ARG_STORE_ADDRESS, Kind.String);
|
2019-10-11 16:11:06 +08:00
|
|
|
}
|
|
|
|
break;
|
2019-10-18 03:11:59 +08:00
|
|
|
// integer arithmetic operations
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ADD.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.add);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ADDI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.add);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SUB.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.sub);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_SUBI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.sub);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MUL.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.mul);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MULI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.mul);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_DIV.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.idiv);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_DIVI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.idiv);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MOD.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.mod);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_MODI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.mod);
|
|
|
|
break;
|
|
|
|
// float arithmetic operations
|
|
|
|
case OP_FADD.code:
|
|
|
|
this.do_float_op_with_register(arg0, arg1, numeric_ops.add);
|
|
|
|
break;
|
|
|
|
case OP_FADDI.code:
|
2019-10-18 04:49:27 +08:00
|
|
|
this.do_float_op_with_literal(arg0, Math.fround(arg1), numeric_ops.add);
|
2019-10-18 03:11:59 +08:00
|
|
|
break;
|
|
|
|
case OP_FSUB.code:
|
|
|
|
this.do_float_op_with_register(arg0, arg1, numeric_ops.sub);
|
|
|
|
break;
|
|
|
|
case OP_FSUBI.code:
|
2019-10-18 04:49:27 +08:00
|
|
|
this.do_float_op_with_literal(arg0, Math.fround(arg1), numeric_ops.sub);
|
2019-10-18 03:11:59 +08:00
|
|
|
break;
|
|
|
|
case OP_FMUL.code:
|
|
|
|
this.do_float_op_with_register(arg0, arg1, numeric_ops.mul);
|
|
|
|
break;
|
|
|
|
case OP_FMULI.code:
|
2019-10-18 04:49:27 +08:00
|
|
|
this.do_float_op_with_literal(arg0, Math.fround(arg1), numeric_ops.mul);
|
2019-10-18 03:11:59 +08:00
|
|
|
break;
|
|
|
|
case OP_FDIV.code:
|
|
|
|
this.do_float_op_with_register(arg0, arg1, numeric_ops.div);
|
|
|
|
break;
|
|
|
|
case OP_FDIVI.code:
|
2019-10-18 04:49:27 +08:00
|
|
|
this.do_float_op_with_literal(arg0, Math.fround(arg1), numeric_ops.div);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
|
|
|
// bit operations
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_AND.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.and);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ANDI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.and);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_OR.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.or);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_ORI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.or);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_XOR.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.xor);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-02 23:52:59 +08:00
|
|
|
case OP_XORI.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(arg0, arg1, numeric_ops.xor);
|
2019-10-02 23:26:45 +08:00
|
|
|
break;
|
2019-10-03 01:16:05 +08:00
|
|
|
// shift operations
|
|
|
|
case OP_SHIFT_LEFT.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.shl);
|
2019-10-03 01:16:05 +08:00
|
|
|
break;
|
|
|
|
case OP_SHIFT_RIGHT.code:
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_register(arg0, arg1, numeric_ops.shr);
|
2019-10-03 01:16:05 +08:00
|
|
|
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:
|
2020-04-27 02:21:10 +08:00
|
|
|
// all nonzero?
|
2019-10-04 19:21:51 +08:00
|
|
|
this.conditional_jump(
|
2019-12-20 23:10:50 +08:00
|
|
|
thread,
|
2019-10-04 19:21:51 +08:00
|
|
|
arg0,
|
2020-04-27 02:21:10 +08:00
|
|
|
(_, b) => b !== 0,
|
2019-10-04 19:21:51 +08:00
|
|
|
1,
|
2019-10-17 19:45:59 +08:00
|
|
|
...rest(arg_vals).map(reg => this.get_register_signed(reg)),
|
2019-10-04 19:21:51 +08:00
|
|
|
);
|
2019-12-21 04:03:15 +08:00
|
|
|
advance = false;
|
[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(
|
2019-12-20 23:10:50 +08:00
|
|
|
thread,
|
2019-10-04 19:21:51 +08:00
|
|
|
arg0,
|
2019-12-21 04:03:15 +08:00
|
|
|
(a, b) => a === b,
|
2019-10-04 19:21:51 +08:00
|
|
|
0,
|
2019-10-17 19:45:59 +08:00
|
|
|
...rest(arg_vals).map(reg => this.get_register_signed(reg)),
|
2019-10-04 19:21:51 +08:00
|
|
|
);
|
2019-12-21 04:03:15 +08:00
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a === b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a === b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a !== b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a !== b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_register(thread, arg2, (a, b) => a > b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_literal(thread, arg2, (a, b) => a > b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a > b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a > b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_register(thread, arg2, (a, b) => a < b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_literal(thread, arg2, (a, b) => a < b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a < b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a < b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_register(thread, arg2, (a, b) => a >= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_literal(thread, arg2, (a, b) => a >= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a >= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a >= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_register(thread, arg2, (a, b) => a <= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.unsigned_cond_jump_with_literal(thread, arg2, (a, b) => a <= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_register(thread, arg2, (a, b) => a <= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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-12-21 04:03:15 +08:00
|
|
|
this.signed_cond_jump_with_literal(thread, arg2, (a, b) => a <= b, arg0, arg1);
|
|
|
|
advance = false;
|
[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:
|
2019-12-20 23:10:50 +08:00
|
|
|
this.push_variable_stack(thread, arg0, 1);
|
2019-10-08 01:52:44 +08:00
|
|
|
break;
|
|
|
|
case OP_STACK_POP.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
this.pop_variable_stack(thread, arg0, 1);
|
2019-10-08 01:52:44 +08:00
|
|
|
break;
|
|
|
|
case OP_STACK_PUSHM.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
this.push_variable_stack(thread, arg0, arg1);
|
2019-10-08 01:52:44 +08:00
|
|
|
break;
|
|
|
|
case OP_STACK_POPM.code:
|
2019-12-20 23:10:50 +08:00
|
|
|
this.pop_variable_stack(thread, arg0, arg1);
|
2019-10-08 01:52:44 +08:00
|
|
|
break;
|
2019-11-11 21:21:13 +08:00
|
|
|
case OP_LIST.code:
|
|
|
|
if (!this.window_msg_open) {
|
2019-12-19 05:47:53 +08:00
|
|
|
const list_items = this.deref_string(stack_args[1]).split(LIST_ITEM_DELIMITER);
|
2019-11-11 21:21:13 +08:00
|
|
|
|
|
|
|
result = ExecutionResult.WaitingSelection;
|
|
|
|
this.list_open = true;
|
2019-12-19 05:47:53 +08:00
|
|
|
this.selection_reg = stack_args[0];
|
2019-11-11 21:21:13 +08:00
|
|
|
this.io.list(list_items);
|
|
|
|
}
|
|
|
|
break;
|
2019-10-11 17:49:20 +08:00
|
|
|
case OP_WINDOW_MSG.code:
|
|
|
|
if (!this.window_msg_open) {
|
2019-12-19 05:47:53 +08:00
|
|
|
const str = this.deref_string(stack_args[0]);
|
2019-10-11 17:49:20 +08:00
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
result = ExecutionResult.WaitingInput;
|
2019-10-11 17:49:20 +08:00
|
|
|
this.window_msg_open = true;
|
2019-10-17 19:21:45 +08:00
|
|
|
this.io.window_msg(str);
|
2019-10-11 17:49:20 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case OP_ADD_MSG.code:
|
|
|
|
if (this.window_msg_open) {
|
2019-12-19 05:47:53 +08:00
|
|
|
const str = this.deref_string(stack_args[0]);
|
2019-10-11 17:49:20 +08:00
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
result = ExecutionResult.WaitingInput;
|
2019-10-17 19:21:45 +08:00
|
|
|
this.io.add_msg(str);
|
2019-10-11 17:49:20 +08:00
|
|
|
}
|
|
|
|
break;
|
2019-10-31 05:00:06 +08:00
|
|
|
case OP_GETTIME.code:
|
|
|
|
this.set_register_unsigned(arg0, Math.floor(Date.now() / 1000));
|
|
|
|
break;
|
2019-10-11 17:49:20 +08:00
|
|
|
case OP_WINEND.code:
|
|
|
|
if (this.window_msg_open) {
|
|
|
|
this.window_msg_open = false;
|
2019-10-17 19:21:45 +08:00
|
|
|
this.io.winend();
|
2019-10-11 17:49:20 +08:00
|
|
|
}
|
|
|
|
break;
|
2019-12-21 04:13:59 +08:00
|
|
|
case OP_P_DEAD_V3.code:
|
|
|
|
this.set_register_signed(stack_args[0], this.io.p_dead_v3(stack_args[1]) ? 1 : 0);
|
|
|
|
break;
|
2019-12-20 05:14:59 +08:00
|
|
|
case OP_SET_FLOOR_HANDLER.code:
|
|
|
|
this.io.set_floor_handler(stack_args[0], stack_args[1]);
|
|
|
|
break;
|
2019-12-21 04:03:15 +08:00
|
|
|
case OP_THREAD_STG.code:
|
|
|
|
this.start_thread(arg0);
|
|
|
|
break;
|
2020-01-03 01:54:18 +08:00
|
|
|
case OP_MAP_DESIGNATE.code:
|
|
|
|
this.io.map_designate(
|
|
|
|
this.get_register_signed(arg0),
|
|
|
|
this.get_register_signed(arg0 + 2),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case OP_MAP_DESIGNATE_EX.code:
|
|
|
|
this.io.map_designate(
|
|
|
|
this.get_register_signed(arg0),
|
|
|
|
this.get_register_signed(arg0 + 3),
|
|
|
|
);
|
|
|
|
break;
|
2019-10-31 03:02:35 +08:00
|
|
|
case OP_GET_RANDOM.code:
|
|
|
|
{
|
|
|
|
const low = this.get_register_signed(arg0);
|
|
|
|
const hi = this.get_register_signed(arg0 + 1);
|
|
|
|
|
2019-12-19 06:40:38 +08:00
|
|
|
const r = this.random.next();
|
2019-10-31 03:02:35 +08:00
|
|
|
let result = Math.floor(Math.fround(r / 32768.0) * hi);
|
|
|
|
|
|
|
|
// intentional. this is how the game does it.
|
|
|
|
if (low >= result) {
|
|
|
|
result = low;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set_register_signed(arg1, result);
|
|
|
|
}
|
2019-11-02 02:20:25 +08:00
|
|
|
break;
|
|
|
|
case OP_SET_EPISODE.code:
|
|
|
|
if (this.set_episode_called) {
|
2019-12-23 06:15:05 +08:00
|
|
|
this.io.warning(
|
|
|
|
"Calling set_episode more than once is not supported.",
|
|
|
|
inst_ptr,
|
|
|
|
);
|
2019-11-02 02:20:25 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set_episode_called = true;
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
if (!this._object_code[inst_ptr.seg_idx].labels.includes(ENTRY_SEGMENT)) {
|
2019-11-02 02:20:25 +08:00
|
|
|
this.io.warning(
|
|
|
|
`Calling set_episode outside of segment ${ENTRY_SEGMENT} is not supported.`,
|
2019-12-23 06:15:05 +08:00
|
|
|
inst_ptr,
|
2019-11-02 02:20:25 +08:00
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-12-19 21:35:13 +08:00
|
|
|
if (encode_episode_number(this.episode) !== arg0) {
|
2019-11-02 02:20:25 +08:00
|
|
|
this.io.warning(
|
2020-04-27 02:20:47 +08:00
|
|
|
"Calling set_episode with an argument that does not " +
|
2019-11-02 02:20:25 +08:00
|
|
|
"match the quest's designated episode is not supported.",
|
2019-12-23 06:15:05 +08:00
|
|
|
inst_ptr,
|
2019-11-02 02:20:25 +08:00
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-12-20 07:11:54 +08:00
|
|
|
break;
|
|
|
|
case OP_BB_MAP_DESIGNATE.code:
|
2020-01-03 01:54:18 +08:00
|
|
|
this.io.map_designate(arg0, arg2);
|
2019-10-31 03:02:35 +08:00
|
|
|
break;
|
2019-08-06 23:07:12 +08:00
|
|
|
default:
|
2019-12-23 06:15:05 +08:00
|
|
|
if (!this.unsupported_opcodes_logged.has(inst.opcode.code)) {
|
|
|
|
this.unsupported_opcodes_logged.add(inst.opcode.code);
|
|
|
|
this.io.warning(`Unsupported instruction.`, inst_ptr);
|
|
|
|
}
|
2019-11-02 03:05:20 +08:00
|
|
|
break;
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
if (advance) {
|
|
|
|
this.advance(thread);
|
|
|
|
}
|
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
return result;
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-10-17 19:45:59 +08:00
|
|
|
public get_register_signed(reg: number): number {
|
2019-10-11 15:49:47 +08:00
|
|
|
return this.registers.i32_at(REGISTER_SIZE * reg);
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-10-17 19:45:59 +08:00
|
|
|
private set_register_signed(reg: number, value: number): void {
|
2019-10-11 15:49:47 +08:00
|
|
|
this.registers.write_i32_at(REGISTER_SIZE * reg, value);
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-10-17 19:45:59 +08:00
|
|
|
public get_register_unsigned(reg: number): number {
|
2019-10-11 15:49:47 +08:00
|
|
|
return this.registers.u32_at(REGISTER_SIZE * 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
|
|
|
}
|
|
|
|
|
2019-10-17 19:45:59 +08:00
|
|
|
private set_register_unsigned(reg: number, value: number): void {
|
2019-10-11 15:49:47 +08:00
|
|
|
this.registers.write_u32_at(REGISTER_SIZE * reg, value);
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 01:02:53 +08:00
|
|
|
public get_register_word(reg: number): number {
|
|
|
|
return this.registers.u16_at(REGISTER_SIZE * reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
private set_register_word(reg: number, value: number): void {
|
|
|
|
this.registers.write_u16_at(REGISTER_SIZE * reg, value);
|
|
|
|
}
|
2019-10-31 05:00:06 +08:00
|
|
|
|
2019-10-18 01:02:53 +08:00
|
|
|
public get_register_byte(reg: number): number {
|
|
|
|
return this.registers.u8_at(REGISTER_SIZE * reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
public set_register_byte(reg: number, value: number): void {
|
2019-10-31 05:00:06 +08:00
|
|
|
this.registers.write_u8_at(REGISTER_SIZE * reg, value);
|
2019-10-18 01:02:53 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 00:58:31 +08:00
|
|
|
public get_register_float(reg: number): number {
|
|
|
|
return this.registers.f32_at(REGISTER_SIZE * reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
private set_register_float(reg: number, value: number): void {
|
|
|
|
this.registers.write_f32_at(REGISTER_SIZE * reg, value);
|
|
|
|
}
|
|
|
|
|
2019-10-18 03:11:59 +08:00
|
|
|
private do_integer_op_with_register(
|
2019-10-03 23:28:48 +08:00
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
2019-10-18 03:11:59 +08:00
|
|
|
this.do_integer_op_with_literal(reg1, this.get_register_signed(reg2), op);
|
2019-10-02 23:26:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 03:11:59 +08:00
|
|
|
private do_integer_op_with_literal(
|
2019-10-03 23:28:48 +08:00
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
2019-10-18 02:38:47 +08:00
|
|
|
if ((op === numeric_ops.div || op === numeric_ops.idiv) && literal === 0) {
|
|
|
|
throw new Error("Division by zero");
|
|
|
|
}
|
2019-10-17 19:45:59 +08:00
|
|
|
this.set_register_signed(reg, op(this.get_register_signed(reg), literal));
|
2019-10-02 23:26:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 03:11:59 +08:00
|
|
|
private do_float_op_with_register(
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
|
|
|
this.do_float_op_with_literal(reg1, this.get_register_float(reg2), op);
|
|
|
|
}
|
|
|
|
|
|
|
|
private do_float_op_with_literal(
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
op: BinaryNumericOperation,
|
|
|
|
): void {
|
|
|
|
if ((op === numeric_ops.div || op === numeric_ops.idiv) && literal === 0) {
|
|
|
|
throw new Error("Division by zero");
|
|
|
|
}
|
|
|
|
this.set_register_float(reg, op(this.get_register_float(reg), literal));
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private push_call_stack(thread: Thread, label: number): void {
|
2019-11-15 01:46:53 +08:00
|
|
|
const seg_idx = this.get_segment_index_by_label(label);
|
2019-12-19 05:37:26 +08:00
|
|
|
const segment = this._object_code[seg_idx];
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-11-15 01:46:53 +08:00
|
|
|
if (segment.type !== SegmentType.Instructions) {
|
|
|
|
throw new Error(
|
|
|
|
`Label ${label} points to a ${SegmentType[segment.type]} segment, expecting ${
|
|
|
|
SegmentType[SegmentType.Instructions]
|
|
|
|
}.`,
|
|
|
|
);
|
2019-08-06 23:07:12 +08:00
|
|
|
} else {
|
2019-12-20 23:10:50 +08:00
|
|
|
thread.push_frame(new InstructionPointer(seg_idx, 0, this.object_code));
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private pop_call_stack(idx: number): void {
|
|
|
|
this.threads[idx].pop_call_stack();
|
2019-08-06 23:07:12 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
if (this.threads[idx].call_stack.length === 0) {
|
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-12-21 04:03:15 +08:00
|
|
|
this.terminate_thread(idx);
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private jump_to_label(thread: Thread, label: number): void {
|
|
|
|
thread.set_current_instruction_pointer(
|
|
|
|
new InstructionPointer(this.get_segment_index_by_label(label), 0, this.object_code),
|
|
|
|
);
|
2019-09-15 00:50:03 +08:00
|
|
|
}
|
|
|
|
|
2019-12-21 04:03:15 +08:00
|
|
|
private signed_cond_jump_with_register(
|
[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
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
): void {
|
2019-10-17 22:50:19 +08:00
|
|
|
this.conditional_jump(
|
|
|
|
exec,
|
|
|
|
label,
|
|
|
|
condition,
|
|
|
|
this.get_register_signed(reg1),
|
|
|
|
this.get_register_signed(reg2),
|
|
|
|
);
|
[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-12-21 04:03:15 +08:00
|
|
|
private signed_cond_jump_with_literal(
|
[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
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
): void {
|
2019-10-17 19:45:59 +08:00
|
|
|
this.conditional_jump(exec, label, condition, this.get_register_signed(reg), literal);
|
[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-12-21 04:03:15 +08:00
|
|
|
private unsigned_cond_jump_with_register(
|
[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
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg1: number,
|
|
|
|
reg2: number,
|
|
|
|
): void {
|
2019-10-17 22:50:19 +08:00
|
|
|
this.conditional_jump(
|
|
|
|
exec,
|
|
|
|
label,
|
|
|
|
condition,
|
|
|
|
this.get_register_unsigned(reg1),
|
|
|
|
this.get_register_unsigned(reg2),
|
|
|
|
);
|
[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-12-21 04:03:15 +08:00
|
|
|
private unsigned_cond_jump_with_literal(
|
[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
|
|
|
exec: Thread,
|
|
|
|
label: number,
|
|
|
|
condition: ComparisonOperation,
|
|
|
|
reg: number,
|
|
|
|
literal: number,
|
|
|
|
): void {
|
2019-10-17 19:45:59 +08:00
|
|
|
this.conditional_jump(exec, label, condition, this.get_register_unsigned(reg), literal);
|
[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 conditional_jump(
|
2019-12-21 04:03:15 +08:00
|
|
|
thread: Thread,
|
[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
|
|
|
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-12-21 04:03:15 +08:00
|
|
|
|
2019-10-11 14:14:34 +08:00
|
|
|
if (andreduce(chain_cmp, vals) !== undefined) {
|
2019-12-21 04:03:15 +08:00
|
|
|
this.jump_to_label(thread, label);
|
|
|
|
} else {
|
|
|
|
this.advance(thread);
|
[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-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++) {
|
2019-10-17 19:45:59 +08:00
|
|
|
exec.variable_stack.push(this.get_register_unsigned(r));
|
2019-10-08 01:52:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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--) {
|
2019-10-17 19:45:59 +08:00
|
|
|
this.set_register_unsigned(r, exec.variable_stack.pop()!);
|
2019-10-08 01:52:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-10 17:56:43 +08:00
|
|
|
private get_register_address(reg: number): number {
|
2019-11-16 06:16:09 +08:00
|
|
|
return reg * REGISTER_SIZE;
|
2019-10-10 17:56:43 +08:00
|
|
|
}
|
2019-10-11 17:49:20 +08:00
|
|
|
|
|
|
|
private deref_string(address: number): string {
|
2019-11-16 06:16:09 +08:00
|
|
|
if (address === STRING_ARG_STORE_ADDRESS) {
|
|
|
|
return this.string_arg_store;
|
|
|
|
}
|
2019-10-11 17:49:20 +08:00
|
|
|
|
2019-11-16 06:16:09 +08:00
|
|
|
if (address > 0 && address < REGISTER_COUNT * REGISTER_SIZE) {
|
|
|
|
return this.registers.string_utf16_at(address, REGISTER_COUNT * REGISTER_SIZE, true);
|
2019-10-11 17:49:20 +08:00
|
|
|
}
|
|
|
|
|
2019-11-16 06:16:09 +08:00
|
|
|
throw new Error(`Failed to dereference string: Invalid address ${address}`);
|
2019-10-11 17:49:20 +08:00
|
|
|
}
|
2019-11-02 02:20:25 +08:00
|
|
|
|
2019-11-14 01:10:30 +08:00
|
|
|
private parse_template_string(template: string): string {
|
|
|
|
const exact_tags: Record<string, string | (() => string)> = {
|
|
|
|
// TODO: get real values for these
|
|
|
|
"hero name": "PLACEHOLDER",
|
|
|
|
"hero job": "PLACEHOLDER",
|
|
|
|
"name hero": "PLACEHOLDER",
|
|
|
|
"name job": "PLACEHOLDER",
|
|
|
|
// intentionally hardcoded
|
|
|
|
time: "01:12",
|
|
|
|
// used in cmode
|
|
|
|
"award item": "PLACEHOLDER",
|
|
|
|
"challenge title": "PLACEHOLDER",
|
|
|
|
// set by opcode get_pl_name
|
|
|
|
pl_name: "PLACEHOLDER",
|
|
|
|
pl_job: "PLACEHOLDER",
|
|
|
|
last_word: "PLACEHOLDER",
|
|
|
|
last_chat: "PLACEHOLDER",
|
|
|
|
team_name: "PLACEHOLDER",
|
|
|
|
// does not appear to be used in any sega quests
|
|
|
|
meseta_slot_prize: "PLACEHOLDER",
|
|
|
|
};
|
|
|
|
const pattern_tags: [RegExp, (arg: string) => string][] = [
|
|
|
|
[
|
|
|
|
/^color ([0-9]+)$/,
|
|
|
|
arg => {
|
|
|
|
// TODO: decide how to handle this
|
|
|
|
return `<color ${arg}>`;
|
|
|
|
},
|
|
|
|
],
|
|
|
|
[
|
|
|
|
/^r([0-9]{1,3})$/,
|
|
|
|
arg => {
|
|
|
|
const num = parseInt(arg);
|
|
|
|
|
|
|
|
if (isNaN(num)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.get_register_unsigned(num).toString();
|
|
|
|
},
|
|
|
|
],
|
|
|
|
[
|
|
|
|
/^f([0-9]{1,3})$/,
|
|
|
|
arg => {
|
|
|
|
const num = parseInt(arg);
|
|
|
|
|
|
|
|
if (isNaN(num)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.get_register_float(num).toFixed(6);
|
|
|
|
},
|
|
|
|
],
|
|
|
|
];
|
|
|
|
|
|
|
|
const tag_start_char = "<";
|
|
|
|
const tag_end_char = ">";
|
|
|
|
|
|
|
|
let tag_open = false;
|
|
|
|
let tag_start_idx = -1;
|
|
|
|
|
|
|
|
let i = 0;
|
|
|
|
let len = template.length;
|
|
|
|
// iterate through template
|
|
|
|
while (i < len) {
|
|
|
|
const char = template[i];
|
|
|
|
|
|
|
|
// end of tag
|
|
|
|
if (tag_open && char === tag_end_char) {
|
|
|
|
tag_open = false;
|
|
|
|
|
|
|
|
// extract key from tag
|
|
|
|
const tag_end_idx = i;
|
|
|
|
const key = template.slice(tag_start_idx + 1, tag_end_idx);
|
|
|
|
|
|
|
|
// get value
|
|
|
|
let val: string | (() => string) | undefined = undefined;
|
|
|
|
|
|
|
|
// check if matches pattern
|
|
|
|
for (const [pattern, handler] of pattern_tags) {
|
|
|
|
const match = pattern.exec(key);
|
|
|
|
if (match && match[1] !== undefined) {
|
|
|
|
val = handler(match[1]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if matches tag
|
|
|
|
if (val === undefined) {
|
|
|
|
val = exact_tags[key];
|
|
|
|
}
|
|
|
|
|
|
|
|
// not a valid tag, replace with empty string
|
|
|
|
if (val === undefined) {
|
|
|
|
val = "";
|
|
|
|
}
|
|
|
|
// run function and memoize result
|
|
|
|
else if (typeof val === "function") {
|
|
|
|
const memo = val();
|
|
|
|
exact_tags[key] = memo;
|
|
|
|
val = memo;
|
|
|
|
}
|
|
|
|
|
|
|
|
// replace tag with value in template
|
|
|
|
template = template.slice(0, tag_start_idx) + val + template.slice(tag_end_idx + 1);
|
|
|
|
|
|
|
|
// adjust position
|
|
|
|
const offset = val.length - (key.length + 2);
|
|
|
|
i += offset;
|
|
|
|
len += offset;
|
|
|
|
}
|
|
|
|
// mark start of tag
|
|
|
|
else if (char === tag_start_char) {
|
|
|
|
tag_open = true;
|
|
|
|
tag_start_idx = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove open tag if it was not closed until the end
|
|
|
|
if (tag_open) {
|
|
|
|
template = template.slice(0, tag_start_idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
return template;
|
|
|
|
}
|
2019-08-06 23:07:12 +08:00
|
|
|
}
|