mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
Modified QuestRunner's execution flow.
Now it goes like this: 1. Advance instruction pointer. 2. Check for breakpoints. 3. Execute instruction. Also fixed some bugs.
This commit is contained in:
parent
6e41b6fb79
commit
24abf3ae7f
@ -45,6 +45,11 @@ export class QuestRunner {
|
|||||||
* A quest is running but the execution is currently paused.
|
* A quest is running but the execution is currently paused.
|
||||||
*/
|
*/
|
||||||
public readonly paused: Property<boolean> = this._paused;
|
public readonly paused: Property<boolean> = this._paused;
|
||||||
|
/**
|
||||||
|
* Have we executed since last advancing the instruction pointer?
|
||||||
|
*/
|
||||||
|
private executed_since_advance = false;
|
||||||
|
private first_frame = true;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.vm = new VirtualMachine(this.create_vm_io());
|
this.vm = new VirtualMachine(this.create_vm_io());
|
||||||
@ -73,6 +78,9 @@ export class QuestRunner {
|
|||||||
this.vm.start_thread(0);
|
this.vm.start_thread(0);
|
||||||
|
|
||||||
this._running.val = true;
|
this._running.val = true;
|
||||||
|
this._paused.val = false;
|
||||||
|
this.executed_since_advance = false;
|
||||||
|
this.first_frame = true;
|
||||||
|
|
||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
}
|
}
|
||||||
@ -110,6 +118,8 @@ export class QuestRunner {
|
|||||||
this.stepping_breakpoints.push(dst_instr.asm.mnemonic.line_no);
|
this.stepping_breakpoints.push(dst_instr.asm.mnemonic.line_no);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.schedule_frame();
|
||||||
}
|
}
|
||||||
|
|
||||||
public step_in(): void {
|
public step_in(): void {
|
||||||
@ -139,6 +149,12 @@ export class QuestRunner {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public stop(): void {
|
||||||
|
this._running.val = false;
|
||||||
|
this._paused.val = false;
|
||||||
|
asm_editor_store?.unset_execution_location();
|
||||||
|
}
|
||||||
|
|
||||||
private schedule_frame(): void {
|
private schedule_frame(): void {
|
||||||
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
||||||
}
|
}
|
||||||
@ -149,30 +165,45 @@ export class QuestRunner {
|
|||||||
let need_emit_unpause = this.paused.val;
|
let need_emit_unpause = this.paused.val;
|
||||||
|
|
||||||
exec_loop: while (true) {
|
exec_loop: while (true) {
|
||||||
result = this.vm.execute();
|
if (this.first_frame || this.executed_since_advance) {
|
||||||
|
if (!this.first_frame) {
|
||||||
|
this.vm.advance();
|
||||||
|
|
||||||
|
if (this.vm.halted) {
|
||||||
|
this.stop();
|
||||||
|
break exec_loop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const srcloc = this.vm.get_current_source_location();
|
this.executed_since_advance = false;
|
||||||
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;
|
const srcloc = this.vm.get_current_source_location();
|
||||||
|
|
||||||
if (hit_breakpoint) {
|
if (srcloc) {
|
||||||
this.stepping_breakpoints.length = 0;
|
// check if need to break
|
||||||
asm_editor_store?.set_execution_location(srcloc.line_no);
|
const hit_breakpoint =
|
||||||
break exec_loop;
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// first instruction after pause did not break, set state to unpaused
|
||||||
|
if (need_emit_unpause) {
|
||||||
|
need_emit_unpause = false;
|
||||||
|
this._paused.val = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// first instruction after pause did not break, set state to unpaused
|
result = this.vm.execute(false);
|
||||||
if (need_emit_unpause) {
|
this.executed_since_advance = true;
|
||||||
need_emit_unpause = false;
|
|
||||||
this._paused.val = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (result) {
|
switch (result) {
|
||||||
case ExecutionResult.WaitingVsync:
|
case ExecutionResult.WaitingVsync:
|
||||||
@ -189,12 +220,12 @@ export class QuestRunner {
|
|||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
break exec_loop;
|
break exec_loop;
|
||||||
case ExecutionResult.Halted:
|
case ExecutionResult.Halted:
|
||||||
this._running.val = false;
|
this.stop();
|
||||||
asm_editor_store?.unset_execution_location();
|
|
||||||
break exec_loop;
|
break exec_loop;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.first_frame = false;
|
||||||
this._paused.val = true;
|
this._paused.val = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -159,6 +159,7 @@ export class VirtualMachine {
|
|||||||
private list_open = false;
|
private list_open = false;
|
||||||
private selection_reg = 0;
|
private selection_reg = 0;
|
||||||
private cur_srcloc?: AsmToken;
|
private cur_srcloc?: AsmToken;
|
||||||
|
private _halted = true;
|
||||||
|
|
||||||
constructor(private io: VirtualMachineIO = new VMIOStub()) {
|
constructor(private io: VirtualMachineIO = new VMIOStub()) {
|
||||||
srand(GetTickCount());
|
srand(GetTickCount());
|
||||||
@ -216,29 +217,81 @@ export class VirtualMachine {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.thread.push(
|
const thread = new Thread(
|
||||||
new Thread(
|
this.io,
|
||||||
this.io,
|
new ExecutionLocation(seg_idx!, 0),
|
||||||
new ExecutionLocation(seg_idx!, 0),
|
this.memory.allocate(ARG_STACK_SLOT_SIZE * ARG_STACK_LENGTH),
|
||||||
this.memory.allocate(ARG_STACK_SLOT_SIZE * ARG_STACK_LENGTH),
|
true,
|
||||||
true,
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
this.thread.push(thread);
|
||||||
|
|
||||||
|
this._halted = false;
|
||||||
|
|
||||||
|
this.update_source_location(thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
private dispose_thread(thread_idx: number): void {
|
private dispose_thread(thread_idx: number): void {
|
||||||
this.thread[thread_idx].dispose();
|
this.thread[thread_idx].dispose();
|
||||||
this.thread.splice(thread_idx, 1);
|
this.thread.splice(thread_idx, 1);
|
||||||
|
|
||||||
|
if (this.thread.length === 0) {
|
||||||
|
this.halt();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private update_source_location(exec: Thread) {
|
||||||
|
const inst = this.get_next_instruction_from_thread(exec);
|
||||||
|
|
||||||
|
if (inst.asm && inst.asm.mnemonic) {
|
||||||
|
this.cur_srcloc = inst.asm.mnemonic;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Move to next instruction.
|
||||||
|
*/
|
||||||
|
public advance(): void {
|
||||||
|
if (this._halted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const exec = this.thread[this.thread_idx];
|
||||||
|
if (exec.call_stack.length) {
|
||||||
|
const top = exec.call_stack_top();
|
||||||
|
const segment = this.object_code[top.seg_idx] as InstructionSegment;
|
||||||
|
|
||||||
|
// move to next instruction
|
||||||
|
if (++top.inst_idx >= segment.instructions.length) {
|
||||||
|
// segment ended, move to next segment
|
||||||
|
if (++top.seg_idx >= this.object_code.length) {
|
||||||
|
// eof
|
||||||
|
this.dispose_thread(this.thread_idx);
|
||||||
|
} else {
|
||||||
|
top.inst_idx = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.thread.length === 0) {
|
||||||
|
this.halt();
|
||||||
|
} else {
|
||||||
|
this.update_source_location(exec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public get halted(): boolean {
|
||||||
|
return this._halted;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Executes the next instruction if one is scheduled.
|
* Executes the next instruction if one is scheduled.
|
||||||
|
* @param auto_advance Should the instruction pointer be automatically advanced.
|
||||||
*/
|
*/
|
||||||
execute(): ExecutionResult {
|
execute(auto_advance: boolean = true): ExecutionResult {
|
||||||
let srcloc: AsmToken | undefined;
|
let srcloc: AsmToken | undefined;
|
||||||
|
|
||||||
if (this.thread.length === 0) {
|
if (this._halted) {
|
||||||
this.cur_srcloc = undefined;
|
|
||||||
return ExecutionResult.Halted;
|
return ExecutionResult.Halted;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,13 +299,7 @@ export class VirtualMachine {
|
|||||||
const exec = this.thread[this.thread_idx];
|
const exec = this.thread[this.thread_idx];
|
||||||
const inst = this.get_next_instruction_from_thread(exec);
|
const inst = this.get_next_instruction_from_thread(exec);
|
||||||
|
|
||||||
if (inst.asm && inst.asm.mnemonic) {
|
return this.execute_instruction(auto_advance, exec, inst, srcloc);
|
||||||
srcloc = inst.asm.mnemonic;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.cur_srcloc = srcloc;
|
|
||||||
|
|
||||||
return this.execute_instruction(exec, inst, srcloc);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (!(err instanceof Error)) {
|
if (!(err instanceof Error)) {
|
||||||
err = new Error(String(err));
|
err = new Error(String(err));
|
||||||
@ -266,6 +313,7 @@ export class VirtualMachine {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private execute_instruction(
|
private execute_instruction(
|
||||||
|
auto_advance: boolean,
|
||||||
exec: Thread,
|
exec: Thread,
|
||||||
inst: Instruction,
|
inst: Instruction,
|
||||||
srcloc?: AsmToken,
|
srcloc?: AsmToken,
|
||||||
@ -689,21 +737,8 @@ export class VirtualMachine {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// advance instruction "pointer"
|
if (auto_advance) {
|
||||||
if (exec.call_stack.length) {
|
this.advance();
|
||||||
const top = exec.call_stack_top();
|
|
||||||
const segment = this.object_code[top.seg_idx] as InstructionSegment;
|
|
||||||
|
|
||||||
// move to next instruction
|
|
||||||
if (++top.inst_idx >= segment.instructions.length) {
|
|
||||||
// segment ended, move to next segment
|
|
||||||
if (++top.seg_idx >= this.object_code.length) {
|
|
||||||
// eof
|
|
||||||
this.dispose_thread(this.thread_idx);
|
|
||||||
} else {
|
|
||||||
top.inst_idx = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
@ -726,6 +761,8 @@ export class VirtualMachine {
|
|||||||
thread.dispose();
|
thread.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.cur_srcloc = undefined;
|
||||||
|
this._halted = true;
|
||||||
this.window_msg_open = false;
|
this.window_msg_open = false;
|
||||||
|
|
||||||
this.thread = [];
|
this.thread = [];
|
||||||
@ -839,7 +876,7 @@ export class VirtualMachine {
|
|||||||
// popped off the last return address
|
// popped off the last return address
|
||||||
// which means this is the end of the function this thread was started on
|
// which means this is the end of the function this thread was started on
|
||||||
// which means this is the end of this thread
|
// which means this is the end of this thread
|
||||||
this.thread.splice(idx, 1);
|
this.dispose_thread(idx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user