2019-11-06 04:07:17 +08:00
|
|
|
import { ExecutionResult, VirtualMachine } from "./scripting/vm";
|
|
|
|
import { QuestModel } from "./model/QuestModel";
|
|
|
|
import { VirtualMachineIO } from "./scripting/vm/io";
|
|
|
|
import { AsmToken } from "./scripting/instructions";
|
2019-11-09 00:34:58 +08:00
|
|
|
import { quest_editor_store } from "./stores/QuestEditorStore";
|
2019-11-14 06:31:20 +08:00
|
|
|
import { asm_editor_store } from "./stores/AsmEditorStore";
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-11-09 00:34:58 +08:00
|
|
|
const logger = quest_editor_store.get_logger("quest_editor/QuestRunner");
|
|
|
|
|
|
|
|
function srcloc_to_string(srcloc: AsmToken): string {
|
|
|
|
return `[${srcloc.line_no}:${srcloc.col}]`;
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
|
|
|
|
export class QuestRunner {
|
|
|
|
private readonly vm: VirtualMachine;
|
|
|
|
private animation_frame?: number;
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
this.vm = new VirtualMachine(this.create_vm_io());
|
|
|
|
}
|
|
|
|
|
|
|
|
run(quest: QuestModel): void {
|
|
|
|
if (this.animation_frame != undefined) {
|
|
|
|
cancelAnimationFrame(this.animation_frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.vm.load_object_code(quest.object_code);
|
|
|
|
this.vm.start_thread(0);
|
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
this.schedule_frame();
|
|
|
|
}
|
|
|
|
|
|
|
|
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-14 06:31:20 +08:00
|
|
|
exec_loop:
|
|
|
|
while (true) {
|
2019-11-06 04:07:17 +08:00
|
|
|
result = this.vm.execute();
|
2019-11-14 06:31:20 +08:00
|
|
|
|
|
|
|
const srcloc = this.vm.get_current_source_location();
|
|
|
|
if (srcloc && asm_editor_store.breakpoints.val.includes(srcloc.line_no)) {
|
2019-11-14 06:34:07 +08:00
|
|
|
asm_editor_store.set_execution_location(srcloc.line_no);
|
2019-11-14 06:31:20 +08:00
|
|
|
break exec_loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (result) {
|
|
|
|
case ExecutionResult.WaitingVsync:
|
|
|
|
this.vm.vsync();
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
case ExecutionResult.WaitingInput:
|
|
|
|
// TODO: implement input from gui
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
case ExecutionResult.WaitingSelection:
|
|
|
|
// TODO: implement input from gui
|
|
|
|
this.vm.list_select(0);
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
case ExecutionResult.Halted:
|
2019-11-14 06:34:07 +08:00
|
|
|
asm_editor_store.unset_execution_location();
|
2019-11-14 06:31:20 +08:00
|
|
|
break exec_loop;
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
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 => {},
|
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
list: (list_items: string[]): void => {
|
|
|
|
logger.info(`list "[${list_items}]"`);
|
|
|
|
},
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
warning: (msg: string, srcloc?: AsmToken): void => {
|
2019-11-09 00:34:58 +08:00
|
|
|
logger.warning(msg, srcloc && srcloc_to_string(srcloc));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
error: (err: Error, srcloc?: AsmToken): void => {
|
2019-11-09 00:34:58 +08:00
|
|
|
logger.error(err, srcloc && srcloc_to_string(srcloc));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|