2019-12-19 06:40:38 +08:00
|
|
|
import { VirtualMachine } from "./VirtualMachine";
|
2019-12-19 05:37:26 +08:00
|
|
|
import { AsmToken, Instruction, InstructionSegment, SegmentType } from "../instructions";
|
|
|
|
import { assert } from "../../../core/util";
|
|
|
|
import {
|
|
|
|
OP_ARG_PUSHA,
|
|
|
|
OP_ARG_PUSHB,
|
|
|
|
OP_ARG_PUSHL,
|
|
|
|
OP_ARG_PUSHR,
|
|
|
|
OP_ARG_PUSHS,
|
|
|
|
OP_ARG_PUSHW,
|
|
|
|
OP_CALL,
|
|
|
|
OP_SWITCH_CALL,
|
|
|
|
OP_VA_CALL,
|
|
|
|
} from "../opcodes";
|
2019-12-19 06:40:38 +08:00
|
|
|
import { ExecutionLocation } from "./Thread";
|
2019-12-19 05:37:26 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Contains all logic pertaining to breakpoints and stepping through code.
|
|
|
|
*/
|
|
|
|
export class Debugger {
|
|
|
|
private readonly vm: VirtualMachine;
|
|
|
|
private break_on_next = false;
|
|
|
|
/**
|
|
|
|
* Invisible breakpoints that help with stepping over/in/out.
|
|
|
|
*/
|
|
|
|
private readonly stepping_breakpoints: number[] = [];
|
|
|
|
private readonly _breakpoints: number[] = [];
|
|
|
|
|
|
|
|
readonly breakpoints: readonly number[] = this._breakpoints;
|
|
|
|
|
|
|
|
constructor(vm: VirtualMachine) {
|
|
|
|
this.vm = vm;
|
|
|
|
}
|
|
|
|
|
|
|
|
step_over = (): void => {
|
|
|
|
const execloc = this.vm.get_current_execution_location();
|
|
|
|
|
|
|
|
const src_segment = this.get_instruction_segment_by_index(execloc.seg_idx);
|
|
|
|
const src_instr = src_segment.instructions[execloc.inst_idx];
|
|
|
|
const dst_label = this.get_step_innable_instruction_label_argument(src_instr);
|
|
|
|
|
|
|
|
// nothing to step over, just break on next instruction
|
|
|
|
if (dst_label === undefined) {
|
|
|
|
this.break_on_next = true;
|
|
|
|
}
|
|
|
|
// set a breakpoint on the next line
|
|
|
|
else {
|
|
|
|
const dst_srcloc = this.get_next_source_location(execloc);
|
|
|
|
|
|
|
|
// set breakpoint
|
|
|
|
if (dst_srcloc) {
|
|
|
|
this.stepping_breakpoints.push(dst_srcloc.line_no);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
step_in = (): void => {
|
|
|
|
const execloc = this.vm.get_current_execution_location();
|
|
|
|
const src_segment = this.get_instruction_segment_by_index(execloc.seg_idx);
|
|
|
|
const src_instr = src_segment.instructions[execloc.inst_idx];
|
|
|
|
const dst_label = this.get_step_innable_instruction_label_argument(src_instr);
|
|
|
|
|
|
|
|
// not a step-innable instruction, behave like step-over
|
|
|
|
if (dst_label === undefined) {
|
|
|
|
this.step_over();
|
|
|
|
}
|
|
|
|
// can step-in
|
|
|
|
else {
|
|
|
|
const dst_segment = this.get_instruction_segment_by_label(dst_label);
|
|
|
|
const dst_instr = dst_segment.instructions[0];
|
|
|
|
const dst_srcloc = this.get_source_location(dst_instr);
|
|
|
|
|
|
|
|
if (dst_srcloc) {
|
|
|
|
this.stepping_breakpoints.push(dst_srcloc.line_no);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
step_out = (): void => {
|
|
|
|
throw new Error("Not implemented.");
|
|
|
|
};
|
|
|
|
|
2019-12-19 05:58:46 +08:00
|
|
|
set_breakpoint(line_no: number): boolean {
|
2019-12-19 05:37:26 +08:00
|
|
|
if (!this._breakpoints.includes(line_no)) {
|
|
|
|
this._breakpoints.push(line_no);
|
2019-12-19 05:58:46 +08:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
2019-12-19 05:37:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 05:58:46 +08:00
|
|
|
remove_breakpoint(line_no: number): boolean {
|
2019-12-19 05:37:26 +08:00
|
|
|
const index = this._breakpoints.indexOf(line_no);
|
|
|
|
|
|
|
|
if (index != -1) {
|
|
|
|
this._breakpoints.splice(index, 1);
|
2019-12-19 05:58:46 +08:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
2019-12-19 05:37:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
toggle_breakpoint(line_no: number): void {
|
|
|
|
const index = this._breakpoints.indexOf(line_no);
|
|
|
|
|
|
|
|
if (index == -1) {
|
|
|
|
this._breakpoints.push(line_no);
|
|
|
|
} else {
|
|
|
|
this._breakpoints.splice(index, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
clear_breakpoints(): void {
|
|
|
|
this._breakpoints.splice(0, Infinity);
|
|
|
|
}
|
|
|
|
|
|
|
|
breakpoint_hit(srcloc: AsmToken): boolean {
|
|
|
|
const break_now =
|
|
|
|
this.break_on_next ||
|
|
|
|
this._breakpoints.includes(srcloc.line_no) ||
|
|
|
|
this.stepping_breakpoints.includes(srcloc.line_no);
|
|
|
|
|
|
|
|
this.break_on_next = false;
|
|
|
|
|
|
|
|
if (break_now) {
|
|
|
|
this.stepping_breakpoints.length = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return break_now;
|
|
|
|
}
|
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
reset(): void {
|
|
|
|
this.stepping_breakpoints.splice(0, Infinity);
|
|
|
|
}
|
|
|
|
|
2019-12-19 05:37:26 +08:00
|
|
|
private get_instruction_segment_by_label(label: number): InstructionSegment {
|
|
|
|
const seg_idx = this.vm.get_segment_index_by_label(label);
|
|
|
|
return this.get_instruction_segment_by_index(seg_idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
private get_instruction_segment_by_index(index: number): InstructionSegment {
|
|
|
|
const segment = this.vm.object_code[index];
|
|
|
|
|
|
|
|
assert(
|
|
|
|
segment.type === SegmentType.Instructions,
|
|
|
|
`Expected segment ${index} to be of type ${
|
|
|
|
SegmentType[SegmentType.Instructions]
|
|
|
|
}, but was ${SegmentType[segment.type]}.`,
|
|
|
|
);
|
|
|
|
|
|
|
|
return segment;
|
|
|
|
}
|
|
|
|
|
|
|
|
private get_step_innable_instruction_label_argument(inst: Instruction): number | undefined {
|
|
|
|
switch (inst.opcode.code) {
|
|
|
|
case OP_VA_CALL.code:
|
|
|
|
case OP_CALL.code:
|
|
|
|
return inst.args[0].value;
|
|
|
|
case OP_SWITCH_CALL.code:
|
|
|
|
return inst.args[1].value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private get_next_source_location(execloc: ExecutionLocation): AsmToken | undefined {
|
|
|
|
const next_loc = new ExecutionLocation(execloc.seg_idx, execloc.inst_idx);
|
|
|
|
const segment = this.vm.object_code[next_loc.seg_idx];
|
|
|
|
|
|
|
|
// can't go to non-code segments
|
|
|
|
if (segment.type !== SegmentType.Instructions) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
// move to next instruction
|
|
|
|
// move to next segment if segment ended
|
|
|
|
if (++next_loc.inst_idx >= segment.instructions.length) {
|
|
|
|
next_loc.seg_idx++;
|
|
|
|
next_loc.inst_idx = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// no more segments
|
|
|
|
if (next_loc.seg_idx >= this.vm.object_code.length) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const dst_instr = segment.instructions[next_loc.inst_idx];
|
|
|
|
return this.get_source_location(dst_instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
private get_source_location(inst: Instruction): AsmToken | undefined {
|
|
|
|
let dst_srcloc = inst.asm?.mnemonic;
|
|
|
|
|
|
|
|
// use the location of the arg of the arg_push opcode instead
|
|
|
|
if (this.is_arg_push_opcode(inst)) {
|
|
|
|
dst_srcloc = inst.asm?.args[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
return dst_srcloc;
|
|
|
|
}
|
|
|
|
|
|
|
|
private is_arg_push_opcode(inst: Instruction): boolean {
|
|
|
|
switch (inst.opcode.code) {
|
|
|
|
case OP_ARG_PUSHB.code:
|
|
|
|
case OP_ARG_PUSHL.code:
|
|
|
|
case OP_ARG_PUSHR.code:
|
|
|
|
case OP_ARG_PUSHW.code:
|
|
|
|
case OP_ARG_PUSHA.code:
|
|
|
|
case OP_ARG_PUSHS.code:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|