From b03a421ab58a7989e9c9921ac86b631869f2ba04 Mon Sep 17 00:00:00 2001 From: jtuu Date: Fri, 1 May 2020 23:02:29 +0300 Subject: [PATCH] Removed the ability to step in a yielded thread. Changed the way the thread select widget works to make the thread status text update properly. --- src/quest_editor/QuestRunner.ts | 41 ++++------ .../controllers/DebugController.ts | 34 ++++++-- src/quest_editor/gui/DebugView.ts | 12 +-- .../scripting/vm/VirtualMachine.ts | 77 +------------------ 4 files changed, 51 insertions(+), 113 deletions(-) diff --git a/src/quest_editor/QuestRunner.ts b/src/quest_editor/QuestRunner.ts index ad64ec33..b7d1f4f9 100644 --- a/src/quest_editor/QuestRunner.ts +++ b/src/quest_editor/QuestRunner.ts @@ -70,10 +70,8 @@ export class QuestRunner { private readonly _breakpoints: WritableListProperty = list_property(); private readonly _pause_location: WritableProperty = property(undefined); private readonly _thread_ids: WritableListProperty = list_property(); - private readonly _debugging_thread_id: WritableProperty = property( - undefined, - ); private readonly _active_thread_id: WritableProperty = property(undefined); + private readonly _shown_thread_id: WritableProperty = property(undefined); private readonly debugger: Debugger; @@ -96,8 +94,8 @@ export class QuestRunner { readonly breakpoints: ListProperty = this._breakpoints; readonly pause_location: Property = this._pause_location; readonly thread_ids: ListProperty = this._thread_ids; - readonly debugging_thread_id: Property = this._debugging_thread_id; readonly active_thread_id: Property = this._active_thread_id; + readonly shown_thread_id: Property = this._shown_thread_id; get game_state(): GameState { return this._game_state; @@ -171,8 +169,8 @@ export class QuestRunner { this.debugger.deactivate_breakpoints(); this._state.val = QuestRunnerState.Stopped; this._pause_location.val = undefined; - this._debugging_thread_id.val = undefined; this._active_thread_id.val = undefined; + this._shown_thread_id.val = undefined; this._thread_ids.clear(); this.npcs.splice(0, this.npcs.length); this.objects.splice(0, this.objects.length); @@ -207,24 +205,18 @@ export class QuestRunner { this._breakpoints.splice(0, Infinity, ...this.debugger.breakpoints); } - set_debugging_thread(thread_id: number): void { - if ( - thread_id !== this.debugging_thread_id.val && - this.thread_ids.val.indexOf(thread_id) > -1 - ) { - // Update pause location. - const ip = this.vm.get_instruction_pointer(thread_id); + show_thread_location(thread_id: number): void { + // Update highlight location. + const ip = this.vm.get_instruction_pointer(thread_id); - // Exists in source? - if (ip && ip.source_location) { - this._debugging_thread_id.val = thread_id; - this.vm.set_debugging_thread(thread_id); - this._pause_location.val = ip.source_location.line_no; - } else { - this.logger.warn( - `Failed to select thread #${thread_id} because its execution location is unknown.`, - ); - } + // Exists in source? + if (ip && ip.source_location) { + this._pause_location.val = ip.source_location.line_no; + this._shown_thread_id.val = thread_id; + } else { + this.logger.warn( + `Failed to select thread #${thread_id} because its execution location is unknown.`, + ); } } @@ -295,8 +287,9 @@ export class QuestRunner { }; private update_thread_info(): void { - this._debugging_thread_id.val = this.vm.get_debugging_thread_id(); - this._active_thread_id.val = this.vm.get_current_thread_id(); + const thread_id = this.vm.get_current_thread_id(); + this._active_thread_id.val = thread_id; + this._shown_thread_id.val = thread_id; this._thread_ids.splice(0, this._thread_ids.length.val, ...this.vm.get_thread_ids()); } diff --git a/src/quest_editor/controllers/DebugController.ts b/src/quest_editor/controllers/DebugController.ts index 84bb18fa..e9e2268a 100644 --- a/src/quest_editor/controllers/DebugController.ts +++ b/src/quest_editor/controllers/DebugController.ts @@ -6,14 +6,19 @@ import { GuiStore, GuiTool } from "../../core/stores/GuiStore"; import { LogEntry } from "../../core/Logger"; import { LogStore } from "../stores/LogStore"; import { Severity } from "../../core/Severity"; +import { map } from "../../core/observable"; + +type ThreadIdAndLabel = { + id: number; + label: string; +}; export class DebugController extends Controller { readonly can_debug: Property; readonly can_step: Property; readonly can_stop: Property; - readonly thread_ids: ListProperty; - readonly debugging_thread_id: Property; - readonly active_thread_id: Property; + readonly threads: Property; + readonly selected_thread_id: Property; readonly can_select_thread: Property; readonly log: ListProperty; readonly severity: Property; @@ -31,11 +36,24 @@ export class DebugController extends Controller { this.can_stop = quest_editor_store.quest_runner.running; - this.thread_ids = quest_editor_store.quest_runner.thread_ids; + this.threads = map( + (thread_ids, active_thread_id) => + thread_ids.map(id => { + const status = active_thread_id === id ? "Active" : "Yielded"; + return { + id, + label: `Thread #${id} (${status})`, + }; + }), + quest_editor_store.quest_runner.thread_ids, + quest_editor_store.quest_runner.active_thread_id, + ); - this.debugging_thread_id = quest_editor_store.quest_runner.debugging_thread_id; - - this.active_thread_id = quest_editor_store.quest_runner.active_thread_id; + this.selected_thread_id = map( + (threads, shown_thread_id) => threads.find(thread => thread.id === shown_thread_id)!, + this.threads, + quest_editor_store.quest_runner.shown_thread_id, + ); this.can_select_thread = quest_editor_store.quest_runner.thread_ids.map( ids => ids.length > 0 && quest_editor_store.quest_runner.running.val, @@ -92,6 +110,6 @@ export class DebugController extends Controller { }; select_thread = (thread_id: number): void => { - this.quest_editor_store.quest_runner.set_debugging_thread(thread_id); + this.quest_editor_store.quest_runner.show_thread_location(thread_id); }; } diff --git a/src/quest_editor/gui/DebugView.ts b/src/quest_editor/gui/DebugView.ts index ca6dce55..5ee21c6e 100644 --- a/src/quest_editor/gui/DebugView.ts +++ b/src/quest_editor/gui/DebugView.ts @@ -49,15 +49,12 @@ export class DebugView extends ResizableView { icon_left: Icon.Stop, tooltip: "Stop execution (Shift-F5)", }); - // TODO: ensure label is up-to-date. const thread_select = new Select({ class: "quest_editor_DebugView_thread_select", label: "Thread:", - items: ctrl.thread_ids, - to_label: id => { - const status = ctrl.active_thread_id.val === id ? "Active" : "Yielded"; - return `Thread #${id} (${status})`; - }, + items: ctrl.threads, + selected: ctrl.selected_thread_id, + to_label: id_and_label => id_and_label.label, }); const severity_select = new Select({ class: "quest_editor_DebugView_severity_select", @@ -122,8 +119,7 @@ export class DebugView extends ResizableView { stop_button.onclick.observe(ctrl.stop), stop_button.enabled.bind_to(ctrl.can_stop), - thread_select.selected.observe(({ value }) => ctrl.select_thread(value!)), - thread_select.selected.bind_to(ctrl.debugging_thread_id), + thread_select.selected.observe(({ value }) => ctrl.select_thread(value!.id)), thread_select.enabled.bind_to(ctrl.can_select_thread), severity_select.selected.observe( diff --git a/src/quest_editor/scripting/vm/VirtualMachine.ts b/src/quest_editor/scripting/vm/VirtualMachine.ts index c2d10799..3606b1b2 100644 --- a/src/quest_editor/scripting/vm/VirtualMachine.ts +++ b/src/quest_editor/scripting/vm/VirtualMachine.ts @@ -188,7 +188,6 @@ export class VirtualMachine { private readonly breakpoints: InstructionPointer[] = []; private paused = false; - private debugging_thread_id: number | undefined = undefined; /** * Set of unsupported opcodes that have already been logged. Each unsupported opcode will only @@ -206,7 +205,7 @@ export class VirtualMachine { set step_mode(step_mode: StepMode) { if (step_mode != undefined) { - const thread = this.threads.find(thread => thread.id === this.debugging_thread_id); + const thread = this.current_thread(); if (thread) { thread.step_mode = step_mode; @@ -262,15 +261,9 @@ export class VirtualMachine { ); } - const thread = new Thread( - this.io, - new InstructionPointer(seg_idx!, 0, this.object_code), - area_id, + this.threads.push( + new Thread(this.io, new InstructionPointer(seg_idx!, 0, this.object_code), area_id), ); - if (this.debugging_thread_id === undefined) { - this.debugging_thread_id = thread.id; - } - this.threads.push(thread); } /** @@ -296,19 +289,9 @@ export class VirtualMachine { const thread = this.current_thread(); if (!thread) { - this.ignore_pauses_until_after_line = undefined; return ExecutionResult.Suspended; } - // This thread is the one currently selected for debugging? - const debugging_current_thread = thread.id === this.debugging_thread_id; - - const allow_pausing = debugging_current_thread - ? // Debugging current thread: Allow pausing if not ignoring pauses. - this.ignore_pauses_until_after_line === undefined - : // Not debugging current thread: Only allow pausing on breakpoints. - thread.step_mode === StepMode.BreakPoint; - // Get current instruction. const frame = thread.current_stack_frame()!; inst_ptr = frame.instruction_pointer; @@ -316,15 +299,11 @@ export class VirtualMachine { // Check whether the VM needs to pause only if it's not already paused. In that case // it's resuming. - if (allow_pausing && !this.paused) { + if (!this.paused) { switch (thread.step_mode) { case StepMode.BreakPoint: if (this.breakpoints.findIndex(bp => bp.equals(inst_ptr!)) !== -1) { this.paused = true; - // A breakpoint should interrupt the pause ignoring process - // since we are now in a different execution location. - this.debugging_thread_id = thread.id; - this.ignore_pauses_until_after_line = undefined; return ExecutionResult.Paused; } break; @@ -360,15 +339,6 @@ export class VirtualMachine { } } - // Check if we can stop ignoring pauses. - if (debugging_current_thread && this.ignore_pauses_until_after_line !== undefined) { - const line = inst.asm?.mnemonic?.line_no; - // Reached line, allow pausing again. - if (this.ignore_pauses_until_after_line === line) { - this.ignore_pauses_until_after_line = undefined; - } - } - // Not paused, the next instruction can be executed. this.paused = false; @@ -427,9 +397,7 @@ export class VirtualMachine { this._halted = true; this.paused = false; this.breakpoints.splice(0, Infinity); - this.debugging_thread_id = undefined; this.unsupported_opcodes_logged.clear(); - this.ignore_pauses_until_after_line = undefined; Thread.reset_id_counter(); } } @@ -477,35 +445,6 @@ export class VirtualMachine { } } - set_debugging_thread(thread_id: number): void { - if (this.threads.find(thread => thread.id === thread_id)) { - this.debugging_thread_id = thread_id; - - const ip = this.get_instruction_pointer(thread_id); - - if (thread_id === this.current_thread()?.id) { - this.ignore_pauses_until_after_line = undefined; - } - // If switching away from the thread that is currently being executed - // it will look like we are paused but actually the VM has not yet - // processed the next instruction and is not yet actually paused. - // We shall ignore the next pause to prevent the user from having - // to press the step button twice. - else { - // Exists in source? - if (ip && ip.source_location) { - this.ignore_pauses_until_after_line = ip.source_location.line_no; - } else { - this.ignore_pauses_until_after_line = undefined; - } - } - } - } - - get_debugging_thread_id(): number | undefined { - return this.debugging_thread_id; - } - get_thread_ids(): number[] { return this.threads.map(thread => thread.id); } @@ -523,14 +462,6 @@ export class VirtualMachine { this.threads.splice(thread_idx, 1); - if (thread.id === this.debugging_thread_id) { - if (this.threads.length === 0) { - this.debugging_thread_id = undefined; - } else { - this.debugging_thread_id = this.threads[0].id; - } - } - if (this.thread_idx >= thread_idx && this.thread_idx > 0) { this.thread_idx--; }