2019-12-19 06:40:38 +08:00
|
|
|
import { ExecutionResult, VirtualMachine } from "./scripting/vm/VirtualMachine";
|
2019-11-06 04:07:17 +08:00
|
|
|
import { QuestModel } from "./model/QuestModel";
|
|
|
|
import { VirtualMachineIO } from "./scripting/vm/io";
|
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 { log_store } from "./stores/LogStore";
|
2019-12-20 23:10:50 +08:00
|
|
|
import { Breakpoint, Debugger } from "./scripting/vm/Debugger";
|
2019-12-19 05:37:26 +08:00
|
|
|
import { WritableListProperty } from "../core/observable/property/list/WritableListProperty";
|
|
|
|
import { ListProperty } from "../core/observable/property/list/ListProperty";
|
2019-12-20 01:54:01 +08:00
|
|
|
import { AreaVariantModel } from "./model/AreaVariantModel";
|
|
|
|
import { Episode } from "../core/data_formats/parsing/quest/Episode";
|
|
|
|
import { QuestNpcModel } from "./model/QuestNpcModel";
|
|
|
|
import { QuestObjectModel } from "./model/QuestObjectModel";
|
2019-12-22 05:49:41 +08:00
|
|
|
import { AreaStore } from "./stores/AreaStore";
|
2019-12-23 06:15:05 +08:00
|
|
|
import { InstructionPointer } from "./scripting/vm/InstructionPointer";
|
2020-01-03 01:42:08 +08:00
|
|
|
import { clone_segment } from "../core/data_formats/asm/instructions";
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
export enum QuestRunnerState {
|
|
|
|
/**
|
|
|
|
* No quest is loading or loaded quest is not running.
|
|
|
|
*/
|
|
|
|
Stopped,
|
|
|
|
/**
|
|
|
|
* Quest has started up and is running nominally.
|
|
|
|
*/
|
|
|
|
Running,
|
|
|
|
/**
|
|
|
|
* Quest has started up and is paused.
|
|
|
|
*/
|
|
|
|
Paused,
|
|
|
|
}
|
|
|
|
|
2019-12-22 11:34:50 +08:00
|
|
|
class GameStateInternal {
|
2019-12-23 06:15:05 +08:00
|
|
|
constructor(public episode: Episode) {}
|
|
|
|
|
2019-12-20 05:14:59 +08:00
|
|
|
/**
|
|
|
|
* Maps area ids to function labels.
|
|
|
|
*/
|
2019-12-22 11:34:50 +08:00
|
|
|
readonly floor_handlers = new Map<number, number>();
|
2019-12-20 05:14:59 +08:00
|
|
|
/**
|
|
|
|
* Maps area ids to area variants.
|
|
|
|
*/
|
2019-12-22 11:34:50 +08:00
|
|
|
readonly area_variants = new Map<number, AreaVariantModel>();
|
|
|
|
readonly current_area_variant = property<AreaVariantModel | undefined>(undefined);
|
|
|
|
readonly npcs = list_property<QuestNpcModel>();
|
|
|
|
readonly objects = list_property<QuestObjectModel>();
|
|
|
|
}
|
|
|
|
|
|
|
|
export type GameState = Readonly<GameStateInternal>;
|
2019-12-20 01:54:01 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
/**
|
2019-12-22 20:50:55 +08:00
|
|
|
* Orchestrates everything related to emulating a quest run. Drives a {@link VirtualMachine} and
|
|
|
|
* delegates to {@link Debugger}.
|
2019-12-19 05:37:26 +08:00
|
|
|
*/
|
2019-11-06 04:07:17 +08:00
|
|
|
export class QuestRunner {
|
2019-12-25 07:17:02 +08:00
|
|
|
private logger = log_store.get_logger("quest_editor/QuestRunner");
|
2019-11-06 04:07:17 +08:00
|
|
|
private animation_frame?: number;
|
2019-12-21 00:57:34 +08:00
|
|
|
private startup = true;
|
2019-12-20 01:54:01 +08:00
|
|
|
private readonly _state: WritableProperty<QuestRunnerState> = property(
|
|
|
|
QuestRunnerState.Stopped,
|
|
|
|
);
|
|
|
|
|
|
|
|
private initial_area_id = 0;
|
2019-12-22 11:34:50 +08:00
|
|
|
private readonly npcs: QuestNpcModel[] = [];
|
|
|
|
private readonly objects: QuestObjectModel[] = [];
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
private readonly _breakpoints: WritableListProperty<Breakpoint> = list_property();
|
2019-12-20 07:11:54 +08:00
|
|
|
private readonly _pause_location: WritableProperty<number | undefined> = property(undefined);
|
2019-12-19 05:37:26 +08:00
|
|
|
|
|
|
|
private readonly debugger: Debugger;
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
private _game_state = new GameStateInternal(Episode.I);
|
2019-12-20 01:54:01 +08:00
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
// TODO: make vm private again.
|
|
|
|
readonly vm: VirtualMachine;
|
2019-11-15 17:28:55 +08:00
|
|
|
/**
|
2019-12-20 01:54:01 +08:00
|
|
|
* There is a quest loaded and it is currently running or paused.
|
2019-11-15 17:28:55 +08:00
|
|
|
*/
|
2019-12-20 01:54:01 +08:00
|
|
|
readonly running: Property<boolean> = this._state.map(
|
|
|
|
state => state !== QuestRunnerState.Stopped,
|
|
|
|
);
|
2019-11-15 17:28:55 +08:00
|
|
|
/**
|
2019-12-20 01:54:01 +08:00
|
|
|
* A quest is running but execution is currently paused.
|
2019-11-15 17:28:55 +08:00
|
|
|
*/
|
2019-12-20 01:54:01 +08:00
|
|
|
readonly paused: Property<boolean> = this._state.map(
|
2019-12-21 00:57:34 +08:00
|
|
|
state => state === QuestRunnerState.Paused,
|
2019-12-20 01:54:01 +08:00
|
|
|
);
|
2019-12-20 23:10:50 +08:00
|
|
|
readonly breakpoints: ListProperty<Breakpoint> = this._breakpoints;
|
2019-12-20 07:11:54 +08:00
|
|
|
readonly pause_location: Property<number | undefined> = this._pause_location;
|
2019-11-23 23:01:04 +08:00
|
|
|
|
2019-12-22 20:50:55 +08:00
|
|
|
get game_state(): GameState {
|
|
|
|
return this._game_state;
|
|
|
|
}
|
2019-12-20 01:54:01 +08:00
|
|
|
|
2019-12-22 05:49:41 +08:00
|
|
|
constructor(private readonly area_store: AreaStore) {
|
2019-11-06 04:07:17 +08:00
|
|
|
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 {
|
2019-12-23 00:24:22 +08:00
|
|
|
this.stop();
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
// Runner state.
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info("Starting debugger.");
|
2019-12-21 00:57:34 +08:00
|
|
|
this.startup = true;
|
2019-12-22 11:34:50 +08:00
|
|
|
this.initial_area_id = 0;
|
|
|
|
this.npcs.splice(0, this.npcs.length, ...quest.npcs.val);
|
|
|
|
this.objects.splice(0, this.objects.length, ...quest.objects.val);
|
2019-11-14 22:01:33 +08:00
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
// Current game state.
|
|
|
|
this._game_state = new GameStateInternal(quest.episode);
|
|
|
|
|
|
|
|
// Virtual machine.
|
|
|
|
this.vm.load_object_code(quest.object_code.map(clone_segment), this.game_state.episode);
|
2019-11-06 04:07:17 +08:00
|
|
|
this.vm.start_thread(0);
|
2019-12-23 06:15:05 +08:00
|
|
|
|
|
|
|
// Debugger.
|
2019-12-24 10:04:18 +08:00
|
|
|
this.debugger.activate_breakpoints();
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-21 00:57:34 +08:00
|
|
|
this._state.val = QuestRunnerState.Running;
|
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-12-20 23:10:50 +08:00
|
|
|
this.debugger.resume();
|
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-21 01:20:02 +08:00
|
|
|
step_into(): void {
|
2019-12-19 05:37:26 +08:00
|
|
|
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 {
|
2019-12-23 00:24:22 +08:00
|
|
|
if (!this.running.val) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info("Stopping debugger.");
|
2019-12-22 20:50:55 +08:00
|
|
|
|
|
|
|
if (this.animation_frame != undefined) {
|
|
|
|
cancelAnimationFrame(this.animation_frame);
|
|
|
|
this.animation_frame = undefined;
|
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
this.vm.halt();
|
2019-12-24 10:04:18 +08:00
|
|
|
this.debugger.deactivate_breakpoints();
|
2019-12-20 01:54:01 +08:00
|
|
|
this._state.val = QuestRunnerState.Stopped;
|
2019-12-20 07:11:54 +08:00
|
|
|
this._pause_location.val = undefined;
|
2019-12-22 11:34:50 +08:00
|
|
|
this.npcs.splice(0, this.npcs.length);
|
|
|
|
this.objects.splice(0, this.objects.length);
|
2019-12-23 06:15:05 +08:00
|
|
|
this._game_state = new GameStateInternal(Episode.I);
|
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 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-12-20 05:14:59 +08:00
|
|
|
if (this.animation_frame == undefined) {
|
|
|
|
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
/**
|
|
|
|
* Executes instructions until all threads have yielded or a breakpoint is hit.
|
|
|
|
*/
|
2019-11-06 04:07:17 +08:00
|
|
|
private execution_loop = (): void => {
|
2019-12-20 05:14:59 +08:00
|
|
|
this.animation_frame = undefined;
|
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
this.vm.vsync();
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
const result = this.vm.execute();
|
|
|
|
|
|
|
|
let pause_location: number | undefined;
|
|
|
|
|
|
|
|
switch (result) {
|
|
|
|
case ExecutionResult.Suspended:
|
|
|
|
this._state.val = QuestRunnerState.Running;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ExecutionResult.Paused:
|
2019-12-21 00:57:34 +08:00
|
|
|
this._state.val = QuestRunnerState.Paused;
|
2019-12-24 10:04:18 +08:00
|
|
|
pause_location = this.vm.get_instruction_pointer()?.source_location?.line_no;
|
2019-12-20 23:10:50 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ExecutionResult.WaitingVsync:
|
|
|
|
this._state.val = QuestRunnerState.Running;
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ExecutionResult.WaitingInput:
|
|
|
|
// TODO: implement input from gui
|
|
|
|
this._state.val = QuestRunnerState.Running;
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ExecutionResult.WaitingSelection:
|
|
|
|
// TODO: implement input from gui
|
|
|
|
this.vm.list_select(0);
|
|
|
|
this._state.val = QuestRunnerState.Running;
|
|
|
|
this.schedule_frame();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ExecutionResult.Halted:
|
2019-11-23 23:01:04 +08:00
|
|
|
this.stop();
|
2019-12-20 23:10:50 +08:00
|
|
|
break;
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|
2019-11-15 17:28:55 +08:00
|
|
|
|
2019-12-20 23:10:50 +08:00
|
|
|
this._pause_location.val = pause_location;
|
2019-12-20 01:54:01 +08:00
|
|
|
|
2019-12-21 00:57:34 +08:00
|
|
|
if (this.startup && this._state.val === QuestRunnerState.Running) {
|
|
|
|
this.startup = false;
|
2019-12-20 05:14:59 +08:00
|
|
|
// At this point we know function 0 has run. All area variants have been designated and
|
|
|
|
// all floor handlers have been registered.
|
|
|
|
this.run_floor_handler(
|
|
|
|
this._game_state.area_variants.get(this.initial_area_id) ||
|
2019-12-22 05:49:41 +08:00
|
|
|
this.area_store.get_variant(this._game_state.episode, this.initial_area_id, 0),
|
2019-12-20 01:54:01 +08:00
|
|
|
);
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
private create_vm_io = (): VirtualMachineIO => {
|
2019-12-23 06:15:05 +08:00
|
|
|
function message_with_inst_ptr(message: string, inst_ptr?: InstructionPointer): string {
|
|
|
|
const msg = [message];
|
|
|
|
|
|
|
|
if (inst_ptr) {
|
|
|
|
const { instruction, source_location } = inst_ptr;
|
|
|
|
|
|
|
|
msg.push(` [${instruction.opcode.mnemonic}`);
|
|
|
|
|
|
|
|
if (source_location) {
|
|
|
|
msg.push(` ${source_location.line_no}:${source_location.col}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
msg.push("]");
|
|
|
|
}
|
|
|
|
|
|
|
|
return msg.join("");
|
2019-12-19 03:04:57 +08:00
|
|
|
}
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
return {
|
2020-01-03 01:54:18 +08:00
|
|
|
map_designate: (area_id: number, area_variant_id: number): void => {
|
2019-12-20 07:11:54 +08:00
|
|
|
this._game_state.area_variants.set(
|
|
|
|
area_id,
|
2019-12-22 05:49:41 +08:00
|
|
|
this.area_store.get_variant(this._game_state.episode, area_id, area_variant_id),
|
2019-12-20 07:11:54 +08:00
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2019-12-20 05:14:59 +08:00
|
|
|
set_floor_handler: (area_id: number, label: number) => {
|
|
|
|
this._game_state.floor_handlers.set(area_id, label);
|
|
|
|
},
|
|
|
|
|
2019-11-06 04:07:17 +08:00
|
|
|
window_msg: (msg: string): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info(`window_msg "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
message: (msg: string): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info(`message "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
add_msg: (msg: string): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info(`add_msg "${msg}"`);
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
2019-12-22 05:49:41 +08:00
|
|
|
winend: (): void => {
|
|
|
|
// TODO
|
|
|
|
},
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-12-21 04:13:59 +08:00
|
|
|
p_dead_v3: (): boolean => {
|
|
|
|
// Players never die.
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
|
2019-12-22 05:49:41 +08:00
|
|
|
mesend: (): void => {
|
|
|
|
// TODO
|
|
|
|
},
|
2019-11-06 04:07:17 +08:00
|
|
|
|
2019-11-11 21:21:13 +08:00
|
|
|
list: (list_items: string[]): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.info(`list "[${list_items}]"`);
|
2019-11-11 21:21:13 +08:00
|
|
|
},
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
warning: (msg: string, inst_ptr?: InstructionPointer): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.warn(message_with_inst_ptr(msg, inst_ptr));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
|
2019-12-23 06:15:05 +08:00
|
|
|
error: (err: Error, inst_ptr?: InstructionPointer): void => {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.error(message_with_inst_ptr(err.message, inst_ptr));
|
2019-11-06 04:07:17 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
2019-12-20 05:14:59 +08:00
|
|
|
|
|
|
|
private run_floor_handler(area_variant: AreaVariantModel): void {
|
|
|
|
const area_id = area_variant.area.id;
|
|
|
|
|
|
|
|
this._game_state.current_area_variant.val = area_variant;
|
2019-12-22 11:34:50 +08:00
|
|
|
this._game_state.objects.push(...this.objects.filter(obj => obj.area_id === area_id));
|
2019-12-20 05:14:59 +08:00
|
|
|
|
|
|
|
const label = this._game_state.floor_handlers.get(area_id);
|
|
|
|
|
|
|
|
if (label == undefined) {
|
2019-12-25 07:17:02 +08:00
|
|
|
this.logger.debug(`No floor handler registered for floor ${area_id}.`);
|
2019-12-20 05:14:59 +08:00
|
|
|
} else {
|
|
|
|
this.vm.start_thread(label);
|
2019-12-20 23:10:50 +08:00
|
|
|
this.schedule_frame();
|
2019-12-20 05:14:59 +08:00
|
|
|
}
|
|
|
|
}
|
2019-11-06 04:07:17 +08:00
|
|
|
}
|