phantasmal-world/src/quest_editor/QuestRunner.ts

347 lines
11 KiB
TypeScript
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

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);
}
}