2019-12-19 05:37:26 +08:00
|
|
|
import { ExecutionResult, VirtualMachine } from "./scripting/vm";
|
2019-11-06 04:07:17 +08:00
|
|
|
import { QuestModel } from "./model/QuestModel";
|
|
|
|
import { VirtualMachineIO } from "./scripting/vm/io";
|
2019-12-19 05:37:26 +08:00
|
|
|
import { AsmToken } from "./scripting/instructions";
|
2019-11-15 17:28:55 +08:00
|
|
|
import { WritableProperty } from "../core/observable/property/WritableProperty";
|
2019-12-19 05:37:26 +08:00
|
|
|
import { list_property, property } from "../core/observable";
|
2019-11-15 17:28:55 +08:00
|
|
|
import { Property } from "../core/observable/property/Property";
|
2019-12-19 03:04:57 +08:00
|
|
|
import Logger from "js-logger";
|
|
|
|
import { log_store } from "./stores/LogStore";
|
2019-12-19 05:37:26 +08:00
|
|
|
import { Debugger } from "./scripting/vm/Debugger";
|
|
|
|
import { WritableListProperty } from "../core/observable/property/list/WritableListProperty";
|
|
|
|
import { ListProperty } from "../core/observable/property/list/ListProperty";
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-19 03:04:57 +08:00
|
|
|
const logger = Logger.get("quest_editor/QuestRunner");
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
/**
|
|
|
|
* Orchestrates everything related to emulating a quest run. Delegates to {@link VirtualMachine}
|
|
|
|
* and {@link Debugger}.
|
|
|
|
*/
|
2019-11-06 04:07:17 +08:00
|
|
|
export class QuestRunner {
|
2019-12-19 03:04:57 +08:00
|
|
|
private quest_logger = log_store.get_logger("quest_editor/QuestRunner");
|
2019-11-14 22:01:33 +08:00
|
|
|
private quest?: QuestModel;
|
2019-11-06 04:07:17 +08:00
|
|
|
private animation_frame?: number;
|
|
|
|
|
2019-11-15 17:28:55 +08:00
|
|
|
private readonly _running: WritableProperty<boolean> = property(false);
|
|
|
|
private readonly _paused: WritableProperty<boolean> = property(false);
|
2019-12-19 05:37:26 +08:00
|
|
|
private readonly _breakpoints: WritableListProperty<number> = list_property();
|
|
|
|
private readonly _breakpoint_location: WritableProperty<number | undefined> = property(
|
|
|
|
undefined,
|
|
|
|
);
|
|
|
|
/**
|
|
|
|
* Have we executed since last advancing the instruction pointer?
|
|
|
|
*/
|
|
|
|
private executed_since_advance = true;
|
|
|
|
|
|
|
|
private execution_counter = 0;
|
|
|
|
private readonly execution_max_count = 100_000;
|
|
|
|
private readonly debugger: Debugger;
|
|
|
|
|
|
|
|
// TODO: make vm private again.
|
|
|
|
readonly vm: VirtualMachine;
|
2019-11-15 17:28:55 +08:00
|
|
|
/**
|
|
|
|
* There is a quest loaded and it is currently running.
|
|
|
|
*/
|
2019-12-19 03:04:57 +08:00
|
|
|
readonly running: Property<boolean> = this._running;
|
2019-11-15 17:28:55 +08:00
|
|
|
/**
|
|
|
|
* A quest is running but the execution is currently paused.
|
|
|
|
*/
|
2019-12-19 03:04:57 +08:00
|
|
|
readonly paused: Property<boolean> = this._paused;
|
2019-12-19 05:37:26 +08:00
|
|
|
readonly breakpoints: ListProperty<number> = this._breakpoints;
|
|
|
|
readonly breakpoint_location: Property<number | undefined> = this._breakpoint_location;
|
2019-11-23 23:01:04 +08:00
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
constructor() {
|
|
|
|
this.vm = new VirtualMachine(this.create_vm_io());
|
2019-12-19 05:37:26 +08:00
|
|
|
this.debugger = new Debugger(this.vm);
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
run(quest: QuestModel): void {
|
|
|
|
if (this.animation_frame != undefined) {
|
|
|
|
cancelAnimationFrame(this.animation_frame);
|
|
|
|
}
|
|
|
|
|
2019-11-14 22:01:33 +08:00
|
|
|
this.quest = quest;
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
this.vm.load_object_code(quest.object_code);
|
|
|
|
this.vm.start_thread(0);
|
|
|
|
|
2019-11-15 17:28:55 +08:00
|
|
|
this._running.val = true;
|
2019-11-16 01:30:51 +08:00
|
|
|
this._paused.val = false;
|
2019-11-21 21:22:47 +08:00
|
|
|
this.executed_since_advance = true;
|
2019-11-23 23:01:04 +08:00
|
|
|
this.execution_counter = 0;
|
2019-11-15 17:28:55 +08:00
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
this.schedule_frame();
|
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
resume(): void {
|
2019-11-14 22:01:33 +08:00
|
|
|
this.schedule_frame();
|
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
step_over(): void {
|
|
|
|
this.debugger.step_over();
|
|
|
|
this.schedule_frame();
|
|
|
|
}
|
2019-11-14 22:01:33 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
step_in(): void {
|
|
|
|
this.debugger.step_in();
|
|
|
|
this.schedule_frame();
|
|
|
|
}
|
2019-11-16 01:30:51 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
step_out(): void {
|
|
|
|
this.debugger.step_out();
|
2019-11-16 01:30:51 +08:00
|
|
|
this.schedule_frame();
|
2019-11-14 22:01:33 +08:00
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
stop(): void {
|
|
|
|
this.vm.halt();
|
|
|
|
this._running.val = false;
|
|
|
|
this._paused.val = false;
|
|
|
|
this._breakpoint_location.val = undefined;
|
|
|
|
}
|
2019-11-14 22:01:33 +08:00
|
|
|
|
2019-12-19 05:58:46 +08:00
|
|
|
/**
|
|
|
|
* @returns false if there already was a breakpoint.
|
|
|
|
*/
|
|
|
|
set_breakpoint(line_no: number): boolean {
|
|
|
|
const set = this.debugger.set_breakpoint(line_no);
|
2019-12-19 05:37:26 +08:00
|
|
|
this._breakpoints.splice(0, Infinity, ...this.debugger.breakpoints);
|
2019-12-19 05:58:46 +08:00
|
|
|
return set;
|
2019-12-19 05:37:26 +08:00
|
|
|
}
|
2019-11-14 22:01:33 +08:00
|
|
|
|
2019-12-19 05:58:46 +08:00
|
|
|
/**
|
|
|
|
* @returns false if there was no breakpoint to remove.
|
|
|
|
*/
|
|
|
|
remove_breakpoint(line_no: number): boolean {
|
|
|
|
const removed = this.debugger.remove_breakpoint(line_no);
|
2019-12-19 05:37:26 +08:00
|
|
|
this._breakpoints.splice(0, Infinity, ...this.debugger.breakpoints);
|
2019-12-19 05:58:46 +08:00
|
|
|
return removed;
|
2019-11-14 22:01:33 +08:00
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
toggle_breakpoint(line_no: number): void {
|
|
|
|
this.debugger.toggle_breakpoint(line_no);
|
|
|
|
this._breakpoints.splice(0, Infinity, ...this.debugger.breakpoints);
|
2019-11-15 17:28:55 +08:00
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
clear_breakpoints(): void {
|
|
|
|
this.debugger.clear_breakpoints();
|
|
|
|
this._breakpoints.splice(0, Infinity, ...this.debugger.breakpoints);
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
private schedule_frame(): void {
|
2019-11-06 04:07:17 +08:00
|
|
|
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
private execution_loop = (): void => {
|
|
|
|
let result: ExecutionResult;
|
|
|
|
|
2019-11-22 01:52:42 +08:00
|
|
|
this._paused.val = false;
|
2019-11-15 17:28:55 +08:00
|
|
|
|
2019-11-14 22:01:33 +08:00
|
|
|
exec_loop: while (true) {
|
2019-11-21 21:22:47 +08:00
|
|
|
if (this.executed_since_advance) {
|
|
|
|
this.vm.advance();
|
2019-11-17 00:07:21 +08:00
|
|
|
|
2019-11-21 21:22:47 +08:00
|
|
|
this.executed_since_advance = false;
|
2019-11-16 22:50:28 +08:00
|
|
|
|
2019-11-21 21:22:47 +08:00
|
|
|
if (this.vm.halted) {
|
|
|
|
this.stop();
|
2019-12-19 03:04:57 +08:00
|
|
|
break;
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
2019-11-14 06:31:20 +08:00
|
|
|
|
2019-11-16 01:30:51 +08:00
|
|
|
const srcloc = this.vm.get_current_source_location();
|
2019-11-15 17:28:55 +08:00
|
|
|
|
2019-11-16 01:30:51 +08:00
|
|
|
if (srcloc) {
|
|
|
|
// check if need to break
|
2019-12-19 05:37:26 +08:00
|
|
|
const hit_breakpoint = this.debugger.breakpoint_hit(srcloc);
|
2019-11-16 01:30:51 +08:00
|
|
|
|
|
|
|
if (hit_breakpoint) {
|
2019-12-19 05:37:26 +08:00
|
|
|
this._breakpoint_location.val = srcloc.line_no;
|
2019-12-19 03:04:57 +08:00
|
|
|
break;
|
2019-11-16 01:30:51 +08:00
|
|
|
}
|
2019-11-14 22:01:33 +08:00
|
|
|
}
|
2019-11-15 17:28:55 +08:00
|
|
|
}
|
2019-11-14 22:01:33 +08:00
|
|
|
|
2019-11-16 01:30:51 +08:00
|
|
|
result = this.vm.execute(false);
|
|
|
|
this.executed_since_advance = true;
|
|
|
|
|
2019-11-23 23:01:04 +08:00
|
|
|
// limit execution to prevent the browser from freezing
|
|
|
|
if (++this.execution_counter >= this.execution_max_count) {
|
|
|
|
this.stop();
|
2019-12-19 03:04:57 +08:00
|
|
|
logger.error("Terminated: Maximum execution count reached.");
|
|
|
|
break;
|
2019-11-23 23:01:04 +08:00
|
|
|
}
|
|
|
|
|
2019-11-14 06:31:20 +08:00
|
|
|
switch (result) {
|
|
|
|
case ExecutionResult.WaitingVsync:
|
|
|
|
this.vm.vsync();
|
|
|
|
this.schedule_frame();
|
2019-11-15 17:28:55 +08:00
|
|
|
break exec_loop;
|
2019-11-14 06:31:20 +08:00
|
|
|
case ExecutionResult.WaitingInput:
|
|
|
|
// TODO: implement input from gui
|
|
|
|
this.schedule_frame();
|
2019-11-15 17:28:55 +08:00
|
|
|
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();
|
2019-11-15 17:28:55 +08:00
|
|
|
break exec_loop;
|
2019-11-14 06:31:20 +08:00
|
|
|
case ExecutionResult.Halted:
|
2019-11-16 01:30:51 +08:00
|
|
|
this.stop();
|
2019-11-14 06:31:20 +08:00
|
|
|
break exec_loop;
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|
2019-11-15 17:28:55 +08:00
|
|
|
|
|
|
|
this._paused.val = true;
|
2019-11-23 23:01:04 +08:00
|
|
|
this.execution_counter = 0;
|
2019-11-06 04:07:17 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
private create_vm_io = (): VirtualMachineIO => {
|
2019-12-19 03:04:57 +08:00
|
|
|
function srcloc_to_string(srcloc?: AsmToken): string {
|
|
|
|
return srcloc ? ` [${srcloc.line_no}:${srcloc.col}]` : " ";
|
|
|
|
}
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
return {
|
|
|
|
window_msg: (msg: string): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.info(`window_msg "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
message: (msg: string): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.info(`message "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
add_msg: (msg: string): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.info(`add_msg "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
winend: (): void => {},
|
|
|
|
|
|
|
|
mesend: (): void => {},
|
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
list: (list_items: string[]): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.info(`list "[${list_items}]"`);
|
2019-11-11 21:21:13 +08:00
|
|
|
},
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
warning: (msg: string, srcloc?: AsmToken): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.warning(msg + srcloc_to_string(srcloc));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
error: (err: Error, srcloc?: AsmToken): void => {
|
2019-12-19 03:04:57 +08:00
|
|
|
this.quest_logger.error(err + srcloc_to_string(srcloc));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|