Added running and paused properties to QuestRunner.

Also tried to improve the situation with the circular dependencies.
This commit is contained in:
jtuu 2019-11-15 11:28:55 +02:00
parent a219200291
commit b8f0cbfcb3
2 changed files with 76 additions and 26 deletions

View File

@ -2,16 +2,20 @@ import { ExecutionResult, VirtualMachine, ExecutionLocation } from "./scripting/
import { QuestModel } from "./model/QuestModel"; import { QuestModel } from "./model/QuestModel";
import { VirtualMachineIO } from "./scripting/vm/io"; import { VirtualMachineIO } from "./scripting/vm/io";
import { AsmToken, SegmentType, InstructionSegment, Segment, Instruction } from "./scripting/instructions"; import { AsmToken, SegmentType, InstructionSegment, Segment, Instruction } from "./scripting/instructions";
import { quest_editor_store } from "./stores/QuestEditorStore"; import { quest_editor_store, Logger } from "./stores/QuestEditorStore";
import { asm_editor_store } from "./stores/AsmEditorStore";
import { defined, assert } from "../core/util"; import { defined, assert } from "../core/util";
import { import {
OP_CALL, OP_CALL,
OP_VA_CALL, OP_VA_CALL,
OP_SWITCH_CALL, OP_SWITCH_CALL,
} from "./scripting/opcodes"; } 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";
const logger = quest_editor_store.get_logger("quest_editor/QuestRunner"); let asm_editor_store: AsmEditorStore | undefined;
let logger: Logger | undefined;
function srcloc_to_string(srcloc: AsmToken): string { function srcloc_to_string(srcloc: AsmToken): string {
return `[${srcloc.line_no}:${srcloc.col}]`; return `[${srcloc.line_no}:${srcloc.col}]`;
@ -31,11 +35,34 @@ export class QuestRunner {
private readonly stepping_breakpoints: number[] = []; private readonly stepping_breakpoints: number[] = [];
private break_on_next = false; 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() { constructor() {
this.vm = new VirtualMachine(this.create_vm_io()); this.vm = new VirtualMachine(this.create_vm_io());
} }
run(quest: QuestModel): void { 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) { if (this.animation_frame != undefined) {
cancelAnimationFrame(this.animation_frame); cancelAnimationFrame(this.animation_frame);
} }
@ -45,6 +72,8 @@ export class QuestRunner {
this.vm.load_object_code(quest.object_code); this.vm.load_object_code(quest.object_code);
this.vm.start_thread(0); this.vm.start_thread(0);
this._running.val = true;
this.schedule_frame(); this.schedule_frame();
} }
@ -106,6 +135,10 @@ export class QuestRunner {
} }
} }
public step_out(): void {
}
private schedule_frame(): void { private schedule_frame(): void {
this.animation_frame = requestAnimationFrame(this.execution_loop); this.animation_frame = requestAnimationFrame(this.execution_loop);
} }
@ -113,57 +146,70 @@ export class QuestRunner {
private execution_loop = (): void => { private execution_loop = (): void => {
let result: ExecutionResult; let result: ExecutionResult;
let need_emit_unpause = this.paused.val;
exec_loop: while (true) { exec_loop: while (true) {
result = this.vm.execute(); result = this.vm.execute();
const srcloc = this.vm.get_current_source_location(); const srcloc = this.vm.get_current_source_location();
if (srcloc) { if (srcloc) {
// check if need to break
const hit_breakpoint = const hit_breakpoint =
this.break_on_next || this.break_on_next ||
asm_editor_store.breakpoints.val.includes(srcloc.line_no) || asm_editor_store?.breakpoints.val.includes(srcloc.line_no) ||
this.stepping_breakpoints.includes(srcloc.line_no); this.stepping_breakpoints.includes(srcloc.line_no);
this.break_on_next = false;
if (hit_breakpoint) { if (hit_breakpoint) {
this.stepping_breakpoints.length = 0; this.stepping_breakpoints.length = 0;
asm_editor_store.set_execution_location(srcloc.line_no); asm_editor_store?.set_execution_location(srcloc.line_no);
break exec_loop; break exec_loop;
} }
} }
this.break_on_next = false; // first instruction after pause did not break, set state to unpaused
if (need_emit_unpause) {
need_emit_unpause = false;
this._paused.val = false;
}
switch (result) { switch (result) {
case ExecutionResult.WaitingVsync: case ExecutionResult.WaitingVsync:
this.vm.vsync(); this.vm.vsync();
this.schedule_frame(); this.schedule_frame();
break; break exec_loop;
case ExecutionResult.WaitingInput: case ExecutionResult.WaitingInput:
// TODO: implement input from gui // TODO: implement input from gui
this.schedule_frame(); this.schedule_frame();
break; break exec_loop;
case ExecutionResult.WaitingSelection: case ExecutionResult.WaitingSelection:
// TODO: implement input from gui // TODO: implement input from gui
this.vm.list_select(0); this.vm.list_select(0);
this.schedule_frame(); this.schedule_frame();
break; break exec_loop;
case ExecutionResult.Halted: case ExecutionResult.Halted:
asm_editor_store.unset_execution_location(); this._running.val = false;
asm_editor_store?.unset_execution_location();
break exec_loop; break exec_loop;
} }
} }
this._paused.val = true;
}; };
private create_vm_io = (): VirtualMachineIO => { private create_vm_io = (): VirtualMachineIO => {
return { return {
window_msg: (msg: string): void => { window_msg: (msg: string): void => {
logger.info(`window_msg "${msg}"`); logger?.info(`window_msg "${msg}"`);
}, },
message: (msg: string): void => { message: (msg: string): void => {
logger.info(`message "${msg}"`); logger?.info(`message "${msg}"`);
}, },
add_msg: (msg: string): void => { add_msg: (msg: string): void => {
logger.info(`add_msg "${msg}"`); logger?.info(`add_msg "${msg}"`);
}, },
winend: (): void => {}, winend: (): void => {},
@ -171,15 +217,15 @@ export class QuestRunner {
mesend: (): void => {}, mesend: (): void => {},
list: (list_items: string[]): void => { list: (list_items: string[]): void => {
logger.info(`list "[${list_items}]"`); logger?.info(`list "[${list_items}]"`);
}, },
warning: (msg: string, srcloc?: AsmToken): void => { warning: (msg: string, srcloc?: AsmToken): void => {
logger.warning(msg, srcloc && srcloc_to_string(srcloc)); logger?.warning(msg, srcloc && srcloc_to_string(srcloc));
}, },
error: (err: Error, srcloc?: AsmToken): void => { error: (err: Error, srcloc?: AsmToken): void => {
logger.error(err, srcloc && srcloc_to_string(srcloc)); logger?.error(err, srcloc && srcloc_to_string(srcloc));
}, },
}; };
}; };

View File

@ -31,12 +31,20 @@ import Logger = require("js-logger");
import { MessageLogStore, LogMessage, LogLevel, LogGroup } from "../../core/gui/MessageLog"; import { MessageLogStore, LogMessage, LogLevel, LogGroup } from "../../core/gui/MessageLog";
import { ListProperty } from "../../core/observable/property/list/ListProperty"; import { ListProperty } from "../../core/observable/property/list/ListProperty";
import { WritableProperty } from "../../core/observable/property/WritableProperty"; import { WritableProperty } from "../../core/observable/property/WritableProperty";
import { QuestRunner } from "../QuestRunner";
const logger = Logger.get("quest_editor/gui/QuestEditorStore"); const logger = Logger.get("quest_editor/gui/QuestEditorStore");
export interface Logger {
debug(...items: any[]): void;
info(...items: any[]): void;
warning(...items: any[]): void;
error(...items: any[]): void;
}
export class QuestEditorStore implements Disposable, MessageLogStore { export class QuestEditorStore implements Disposable, MessageLogStore {
private readonly disposer = new Disposer(); private readonly disposer = new Disposer();
public readonly quest_runner: QuestRunner = new QuestRunner();
/** /**
* Log levels as Record<name, LogLevel> * Log levels as Record<name, LogLevel>
*/ */
@ -309,15 +317,11 @@ export class QuestEditorStore implements Disposable, MessageLogStore {
}; };
run_current_quest = (): void => { run_current_quest = (): void => {
// workaround for circular dependency const quest = this.current_quest.val;
import("../QuestRunner").then(({ QuestRunner }) => {
const quest = this.current_quest.val;
const quest_runner = new QuestRunner();
if (quest) { if (quest) {
quest_runner.run(quest); this.quest_runner.run(quest);
} }
});
}; };
private log_message_predicate = (msg: LogMessage): boolean => { private log_message_predicate = (msg: LogMessage): boolean => {
@ -372,7 +376,7 @@ export class QuestEditorStore implements Disposable, MessageLogStore {
}); });
} }
public get_logger(group_name?: string) { public get_logger(group_name?: string): Logger {
let group = this.default_log_group; let group = this.default_log_group;
// find existing or create new // find existing or create new