phantasmal-world/src/quest_editor/QuestRunner.ts

263 lines
8.8 KiB
TypeScript
Raw Normal View History

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,
} 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;
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.schedule_frame();
}
public resume(): void {
this.schedule_frame();
}
public step_over(): void {
const execloc = this.vm.get_current_execution_location();
defined(this.quest);
const src_segment = this.get_instruction_segment_by_index(execloc.seg_idx);
const cur_instr = src_segment.instructions[execloc.inst_idx];
const dst_label = this.get_step_innable_instruction_label_argument(cur_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 next_execloc = new ExecutionLocation(execloc.seg_idx, execloc.inst_idx + 1);
// next line is in the next segment
if (next_execloc.inst_idx >= src_segment.instructions.length) {
next_execloc.seg_idx++;
next_execloc.inst_idx = 0;
}
const dst_segment = this.get_instruction_segment_by_index(next_execloc.seg_idx);
const dst_instr = dst_segment.instructions[next_execloc.inst_idx];
if (dst_instr.asm && dst_instr.asm.mnemonic) {
this.stepping_breakpoints.push(dst_instr.asm.mnemonic.line_no);
}
}
}
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 cur_instr = src_segment.instructions[execloc.inst_idx];
const dst_label = this.get_step_innable_instruction_label_argument(cur_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];
if (dst_instr.asm && dst_instr.asm.mnemonic) {
this.stepping_breakpoints.push(dst_instr.asm.mnemonic.line_no);
}
this.schedule_frame();
}
}
public step_out(): void {
}
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) {
result = this.vm.execute();
2019-11-14 06:31:20 +08:00
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;
}
2019-11-14 06:31:20 +08:00
}
// first instruction after pause did not break, set state to unpaused
if (need_emit_unpause) {
need_emit_unpause = false;
this._paused.val = false;
}
2019-11-14 06:31:20 +08:00
switch (result) {
case ExecutionResult.WaitingVsync:
this.vm.vsync();
this.schedule_frame();
break exec_loop;
2019-11-14 06:31:20 +08:00
case ExecutionResult.WaitingInput:
// TODO: implement input from gui
this.schedule_frame();
break exec_loop;
2019-11-14 06:31:20 +08:00
case ExecutionResult.WaitingSelection:
// TODO: implement input from gui
this.vm.list_select(0);
this.schedule_frame();
break exec_loop;
2019-11-14 06:31:20 +08:00
case ExecutionResult.Halted:
this._running.val = false;
asm_editor_store?.unset_execution_location();
2019-11-14 06:31:20 +08:00
break exec_loop;
}
}
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;
}
}
}