mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
135 lines
3.5 KiB
TypeScript
135 lines
3.5 KiB
TypeScript
import { VirtualMachine } from "./VirtualMachine";
|
|
import { SegmentType } from "../instructions";
|
|
import { InstructionPointer } from "./InstructionPointer";
|
|
import { StepMode } from "./Thread";
|
|
|
|
/**
|
|
* Ensures consistency between source-level breakpoints and VM breakpoints.
|
|
*/
|
|
export class Debugger {
|
|
private readonly vm: VirtualMachine;
|
|
private readonly _breakpoints: Breakpoint[] = [];
|
|
|
|
readonly breakpoints: readonly Breakpoint[] = this._breakpoints;
|
|
|
|
constructor(vm: VirtualMachine) {
|
|
this.vm = vm;
|
|
}
|
|
|
|
resume(): void {
|
|
this.vm.step_mode = StepMode.BreakPoint;
|
|
}
|
|
|
|
step_over(): void {
|
|
this.vm.step_mode = StepMode.Over;
|
|
}
|
|
|
|
step_in(): void {
|
|
this.vm.step_mode = StepMode.In;
|
|
}
|
|
|
|
step_out(): void {
|
|
this.vm.step_mode = StepMode.Out;
|
|
}
|
|
|
|
set_breakpoint(line_no: number): boolean {
|
|
if (this._breakpoints.findIndex(bp => bp.line_no === line_no) === -1) {
|
|
this._breakpoints.push(new Breakpoint(line_no, undefined, this.vm));
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
remove_breakpoint(line_no: number): boolean {
|
|
const index = this._breakpoints.findIndex(bp => bp.line_no === line_no);
|
|
|
|
if (index != -1) {
|
|
this._breakpoints.splice(index, 1)[0].deactivate();
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
toggle_breakpoint(line_no: number): void {
|
|
const index = this._breakpoints.findIndex(bp => bp.line_no === line_no);
|
|
|
|
if (index == -1) {
|
|
this._breakpoints.push(new Breakpoint(line_no, undefined, this.vm));
|
|
} else {
|
|
this._breakpoints.splice(index, 1)[0].deactivate();
|
|
}
|
|
}
|
|
|
|
clear_breakpoints(): void {
|
|
for (const bp of this._breakpoints.splice(0, Infinity)) {
|
|
bp.deactivate();
|
|
}
|
|
}
|
|
|
|
activate_breakpoints(): void {
|
|
for (const bp of this._breakpoints) {
|
|
bp.activate();
|
|
}
|
|
}
|
|
|
|
deactivate_breakpoints(): void {
|
|
for (const bp of this._breakpoints) {
|
|
bp.deactivate();
|
|
}
|
|
}
|
|
}
|
|
|
|
export class Breakpoint {
|
|
constructor(
|
|
readonly line_no: number,
|
|
private ptr: InstructionPointer | undefined,
|
|
private vm: VirtualMachine,
|
|
) {
|
|
if (ptr == undefined) {
|
|
this.activate();
|
|
} else {
|
|
this.vm.set_breakpoint(ptr);
|
|
}
|
|
}
|
|
|
|
get active(): boolean {
|
|
return this.ptr != undefined;
|
|
}
|
|
|
|
activate(): void {
|
|
this.ptr = this.line_no_to_inst_pointer(this.line_no);
|
|
|
|
if (this.ptr) {
|
|
this.vm.set_breakpoint(this.ptr);
|
|
}
|
|
}
|
|
|
|
deactivate(): void {
|
|
if (this.ptr) {
|
|
this.vm.remove_breakpoint(this.ptr);
|
|
}
|
|
}
|
|
|
|
private line_no_to_inst_pointer(line_no: number): InstructionPointer | undefined {
|
|
if (this.vm.halted) return undefined;
|
|
|
|
for (let seg_idx = 0; seg_idx < this.vm.object_code.length; seg_idx++) {
|
|
const segment = this.vm.object_code[seg_idx];
|
|
|
|
if (segment.type === SegmentType.Instructions) {
|
|
for (let inst_idx = 0; inst_idx < segment.instructions.length; inst_idx++) {
|
|
const inst = segment.instructions[inst_idx];
|
|
|
|
if (inst.asm?.mnemonic?.line_no === line_no) {
|
|
return new InstructionPointer(seg_idx, inst_idx, this.vm.object_code);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
}
|