mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
347 lines
11 KiB
TypeScript
347 lines
11 KiB
TypeScript
import { ExecutionResult, VirtualMachine, ExecutionLocation } from "./scripting/vm";
|
||
import { QuestModel } from "./model/QuestModel";
|
||
import { VirtualMachineIO } from "./scripting/vm/io";
|
||
import { AsmToken, SegmentType, InstructionSegment, Segment, Instruction } from "./scripting/instructions";
|
||
import { quest_editor_store, Logger } from "./stores/QuestEditorStore";
|
||
import { defined, assert } from "../core/util";
|
||
import {
|
||
OP_CALL,
|
||
OP_VA_CALL,
|
||
OP_SWITCH_CALL,
|
||
OP_ARG_PUSHB,
|
||
OP_ARG_PUSHL,
|
||
OP_ARG_PUSHR,
|
||
OP_ARG_PUSHW,
|
||
OP_ARG_PUSHA,
|
||
OP_ARG_PUSHS,
|
||
} from "./scripting/opcodes";
|
||
import { WritableProperty } from "../core/observable/property/WritableProperty";
|
||
import { property } from "../core/observable";
|
||
import { Property } from "../core/observable/property/Property";
|
||
import { AsmEditorStore } from "./stores/AsmEditorStore";
|
||
|
||
let asm_editor_store: AsmEditorStore | undefined;
|
||
let logger: Logger | undefined;
|
||
|
||
function srcloc_to_string(srcloc: AsmToken): string {
|
||
return `[${srcloc.line_no}:${srcloc.col}]`;
|
||
}
|
||
|
||
function execloc_to_string(execloc: ExecutionLocation) {
|
||
return `[${execloc.seg_idx}:${execloc.inst_idx}]`;
|
||
}
|
||
|
||
export class QuestRunner {
|
||
private readonly vm: VirtualMachine;
|
||
private quest?: QuestModel;
|
||
private animation_frame?: number;
|
||
/**
|
||
* Invisible breakpoints that help with stepping over/in/out.
|
||
*/
|
||
private readonly stepping_breakpoints: number[] = [];
|
||
private break_on_next = false;
|
||
|
||
private readonly _running: WritableProperty<boolean> = property(false);
|
||
private readonly _paused: WritableProperty<boolean> = property(false);
|
||
/**
|
||
* There is a quest loaded and it is currently running.
|
||
*/
|
||
public readonly running: Property<boolean> = this._running;
|
||
/**
|
||
* A quest is running but the execution is currently paused.
|
||
*/
|
||
public readonly paused: Property<boolean> = this._paused;
|
||
/**
|
||
* Have we executed since last advancing the instruction pointer?
|
||
*/
|
||
private executed_since_advance = false;
|
||
private first_frame = true;
|
||
|
||
constructor() {
|
||
this.vm = new VirtualMachine(this.create_vm_io());
|
||
}
|
||
|
||
run(quest: QuestModel): void {
|
||
if (logger === undefined) {
|
||
// defer creation of logger to prevent problems caused by circular dependency with QuestEditorStore
|
||
logger = quest_editor_store.get_logger("quest_editor/QuestRunner");
|
||
}
|
||
|
||
if (asm_editor_store === undefined) {
|
||
// same here... circular dependency problem
|
||
import("./stores/AsmEditorStore").then(imports => {
|
||
asm_editor_store = imports.asm_editor_store;
|
||
});
|
||
}
|
||
|
||
if (this.animation_frame != undefined) {
|
||
cancelAnimationFrame(this.animation_frame);
|
||
}
|
||
|
||
this.quest = quest;
|
||
|
||
this.vm.load_object_code(quest.object_code);
|
||
this.vm.start_thread(0);
|
||
|
||
this._running.val = true;
|
||
this._paused.val = false;
|
||
this.executed_since_advance = false;
|
||
this.first_frame = true;
|
||
|
||
this.schedule_frame();
|
||
}
|
||
|
||
public resume(): void {
|
||
this.schedule_frame();
|
||
}
|
||
|
||
public step_over(): void {
|
||
const execloc = this.vm.get_current_execution_location();
|
||
|
||
const src_segment = this.get_instruction_segment_by_index(execloc.seg_idx);
|
||
const src_instr = src_segment.instructions[execloc.inst_idx];
|
||
const dst_label = this.get_step_innable_instruction_label_argument(src_instr);
|
||
|
||
// nothing to step over, just break on next instruction
|
||
if (dst_label === undefined) {
|
||
this.break_on_next = true;
|
||
}
|
||
// set a breakpoint on the next line
|
||
else {
|
||
const dst_srcloc = this.get_next_source_location(execloc);
|
||
|
||
// set breakpoint
|
||
if (dst_srcloc) {
|
||
this.stepping_breakpoints.push(dst_srcloc.line_no);
|
||
}
|
||
}
|
||
|
||
this.schedule_frame();
|
||
}
|
||
|
||
public step_in(): void {
|
||
const execloc = this.vm.get_current_execution_location();
|
||
const src_segment = this.get_instruction_segment_by_index(execloc.seg_idx);
|
||
const src_instr = src_segment.instructions[execloc.inst_idx];
|
||
const dst_label = this.get_step_innable_instruction_label_argument(src_instr);
|
||
|
||
// not a step-innable instruction, behave like step-over
|
||
if (dst_label === undefined) {
|
||
this.step_over();
|
||
}
|
||
// can step-in
|
||
else {
|
||
const dst_segment = this.get_instruction_segment_by_label(dst_label);
|
||
const dst_instr = dst_segment.instructions[0];
|
||
const dst_srcloc = this.get_source_location(dst_instr);
|
||
|
||
if (dst_srcloc) {
|
||
this.stepping_breakpoints.push(dst_srcloc.line_no);
|
||
}
|
||
|
||
this.schedule_frame();
|
||
}
|
||
}
|
||
|
||
public step_out(): void {
|
||
|
||
}
|
||
|
||
public stop(): void {
|
||
this.vm.halt();
|
||
this._running.val = false;
|
||
this._paused.val = false;
|
||
asm_editor_store?.unset_execution_location();
|
||
}
|
||
|
||
private schedule_frame(): void {
|
||
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
||
}
|
||
|
||
private execution_loop = (): void => {
|
||
let result: ExecutionResult;
|
||
|
||
let need_emit_unpause = this.paused.val;
|
||
|
||
exec_loop: while (true) {
|
||
if (this.first_frame || this.executed_since_advance) {
|
||
if (!this.first_frame) {
|
||
this.vm.advance();
|
||
|
||
this.executed_since_advance = false;
|
||
|
||
if (this.vm.halted) {
|
||
this.stop();
|
||
break exec_loop;
|
||
}
|
||
}
|
||
|
||
const srcloc = this.vm.get_current_source_location();
|
||
|
||
if (srcloc) {
|
||
// check if need to break
|
||
const hit_breakpoint =
|
||
this.break_on_next ||
|
||
asm_editor_store?.breakpoints.val.includes(srcloc.line_no) ||
|
||
this.stepping_breakpoints.includes(srcloc.line_no);
|
||
|
||
this.break_on_next = false;
|
||
|
||
if (hit_breakpoint) {
|
||
this.stepping_breakpoints.length = 0;
|
||
asm_editor_store?.set_execution_location(srcloc.line_no);
|
||
break exec_loop;
|
||
}
|
||
}
|
||
|
||
// first instruction after pause did not break, set state to unpaused
|
||
if (need_emit_unpause) {
|
||
need_emit_unpause = false;
|
||
this._paused.val = false;
|
||
}
|
||
}
|
||
|
||
result = this.vm.execute(false);
|
||
this.executed_since_advance = true;
|
||
this.first_frame = false;
|
||
|
||
switch (result) {
|
||
case ExecutionResult.WaitingVsync:
|
||
this.vm.vsync();
|
||
this.schedule_frame();
|
||
break exec_loop;
|
||
case ExecutionResult.WaitingInput:
|
||
// TODO: implement input from gui
|
||
this.schedule_frame();
|
||
break exec_loop;
|
||
case ExecutionResult.WaitingSelection:
|
||
// TODO: implement input from gui
|
||
this.vm.list_select(0);
|
||
this.schedule_frame();
|
||
break exec_loop;
|
||
case ExecutionResult.Halted:
|
||
this.stop();
|
||
break exec_loop;
|
||
}
|
||
}
|
||
|
||
this.first_frame = false;
|
||
this._paused.val = true;
|
||
};
|
||
|
||
private create_vm_io = (): VirtualMachineIO => {
|
||
return {
|
||
window_msg: (msg: string): void => {
|
||
logger?.info(`window_msg "${msg}"`);
|
||
},
|
||
|
||
message: (msg: string): void => {
|
||
logger?.info(`message "${msg}"`);
|
||
},
|
||
|
||
add_msg: (msg: string): void => {
|
||
logger?.info(`add_msg "${msg}"`);
|
||
},
|
||
|
||
winend: (): void => {},
|
||
|
||
mesend: (): void => {},
|
||
|
||
list: (list_items: string[]): void => {
|
||
logger?.info(`list "[${list_items}]"`);
|
||
},
|
||
|
||
warning: (msg: string, srcloc?: AsmToken): void => {
|
||
logger?.warning(msg, srcloc && srcloc_to_string(srcloc));
|
||
},
|
||
|
||
error: (err: Error, srcloc?: AsmToken): void => {
|
||
logger?.error(err, srcloc && srcloc_to_string(srcloc));
|
||
},
|
||
};
|
||
};
|
||
|
||
private get_instruction_segment_by_index(index: number): InstructionSegment {
|
||
defined(this.quest);
|
||
|
||
const segment = this.quest.object_code[index];
|
||
|
||
assert(
|
||
segment.type === SegmentType.Instructions,
|
||
`Expected segment ${index} to be of type ${
|
||
SegmentType[SegmentType.Instructions]
|
||
}, but was ${SegmentType[segment.type]}.`,
|
||
);
|
||
|
||
return segment;
|
||
}
|
||
|
||
private get_instruction_segment_by_label(label: number): InstructionSegment {
|
||
const seg_idx = this.vm.get_segment_index_by_label(label);
|
||
return this.get_instruction_segment_by_index(seg_idx);
|
||
}
|
||
|
||
private get_step_innable_instruction_label_argument(instr: Instruction): number | undefined {
|
||
switch (instr.opcode.code) {
|
||
case OP_VA_CALL.code:
|
||
case OP_CALL.code:
|
||
return instr.args[0].value;
|
||
case OP_SWITCH_CALL.code:
|
||
return instr.args[1].value;
|
||
}
|
||
}
|
||
|
||
private is_arg_push_opcode(instr: Instruction): boolean {
|
||
switch (instr.opcode.code) {
|
||
case OP_ARG_PUSHB.code:
|
||
case OP_ARG_PUSHL.code:
|
||
case OP_ARG_PUSHR.code:
|
||
case OP_ARG_PUSHW.code:
|
||
case OP_ARG_PUSHA.code:
|
||
case OP_ARG_PUSHS.code:
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
private get_source_location(instr: Instruction): AsmToken | undefined {
|
||
let dst_srcloc = instr.asm?.mnemonic;
|
||
|
||
// use the location of the arg of the arg_push opcode instead
|
||
if (asm_editor_store?.inline_args_mode.val) {
|
||
if (this.is_arg_push_opcode(instr)) {
|
||
dst_srcloc = instr.asm?.args[0];
|
||
}
|
||
}
|
||
|
||
return dst_srcloc;
|
||
}
|
||
|
||
private get_next_source_location(execloc: ExecutionLocation): AsmToken | undefined {
|
||
defined(this.quest);
|
||
|
||
const next_loc = new ExecutionLocation(execloc.seg_idx, execloc.inst_idx);
|
||
const segment = this.quest.object_code[next_loc.seg_idx];
|
||
|
||
// can't go to non-code segments
|
||
if (segment.type !== SegmentType.Instructions) {
|
||
return undefined;
|
||
}
|
||
|
||
// move to next instruction
|
||
// move to next segment if segment ended
|
||
if (++next_loc.inst_idx >= segment.instructions.length) {
|
||
next_loc.seg_idx++;
|
||
next_loc.inst_idx = 0;
|
||
}
|
||
|
||
// no more segments
|
||
if (next_loc.seg_idx >= this.quest.object_code.length) {
|
||
return undefined;
|
||
}
|
||
|
||
const dst_instr = segment.instructions[next_loc.inst_idx];
|
||
return this.get_source_location(dst_instr);
|
||
}
|
||
}
|