2019-08-11 04:09:06 +08:00
|
|
|
import { reinterpret_i32_as_f32 } from "../../core/primitive_conversion";
|
2019-08-07 14:43:47 +08:00
|
|
|
import { Arg, Segment, SegmentType } from "./instructions";
|
|
|
|
import { Kind, Param, StackInteraction } from "./opcodes";
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-22 18:31:20 +08:00
|
|
|
/**
|
2019-07-29 07:02:22 +08:00
|
|
|
* @param manual_stack If true, will output stack management instructions (argpush variants). Otherwise the arguments of stack management instructions will be output as arguments to the instruction that pops them from the stack.
|
2019-07-22 18:31:20 +08:00
|
|
|
*/
|
2019-07-29 07:02:22 +08:00
|
|
|
export function disassemble(object_code: Segment[], manual_stack: boolean = false): string[] {
|
2019-07-22 02:44:34 +08:00
|
|
|
const lines: string[] = [];
|
|
|
|
const stack: Arg[] = [];
|
2019-08-05 04:59:32 +08:00
|
|
|
let section_type: SegmentType | undefined;
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-29 07:02:22 +08:00
|
|
|
for (const segment of object_code) {
|
2019-08-05 04:59:32 +08:00
|
|
|
// Section marker.
|
|
|
|
let section_marker!: string;
|
2019-07-30 05:46:11 +08:00
|
|
|
|
2019-08-05 04:59:32 +08:00
|
|
|
switch (segment.type) {
|
|
|
|
case SegmentType.Instructions:
|
|
|
|
section_marker = ".code";
|
|
|
|
break;
|
|
|
|
case SegmentType.Data:
|
|
|
|
section_marker = ".data";
|
|
|
|
break;
|
|
|
|
case SegmentType.String:
|
|
|
|
section_marker = ".string";
|
|
|
|
break;
|
|
|
|
}
|
2019-07-30 05:46:11 +08:00
|
|
|
|
2019-08-05 04:59:32 +08:00
|
|
|
if (section_type !== segment.type) {
|
|
|
|
section_type = segment.type;
|
2019-07-30 05:46:11 +08:00
|
|
|
|
2019-08-05 04:59:32 +08:00
|
|
|
if (lines.length) {
|
|
|
|
lines.push("");
|
2019-07-30 03:59:16 +08:00
|
|
|
}
|
2019-08-05 04:59:32 +08:00
|
|
|
|
|
|
|
lines.push(section_marker, "");
|
2019-07-29 07:02:22 +08:00
|
|
|
}
|
|
|
|
|
2019-08-05 04:59:32 +08:00
|
|
|
// Labels.
|
2019-07-31 05:15:23 +08:00
|
|
|
for (const label of segment.labels) {
|
|
|
|
lines.push(`${label}:`);
|
2019-07-29 07:02:22 +08:00
|
|
|
}
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-08-05 04:59:32 +08:00
|
|
|
// Code or data lines.
|
2019-07-30 03:59:16 +08:00
|
|
|
if (segment.type === SegmentType.Data) {
|
|
|
|
const bytes = new Uint8Array(segment.data);
|
|
|
|
let line = " ";
|
|
|
|
|
|
|
|
for (let i = 0; i < bytes.length; i++) {
|
|
|
|
line += "0x" + bytes[i].toString(16).padStart(2, "0");
|
|
|
|
|
|
|
|
if (i % 16 === 15) {
|
|
|
|
lines.push(line);
|
|
|
|
line = " ";
|
|
|
|
} else if (i < bytes.length - 1) {
|
|
|
|
line += " ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (line.length > 4) {
|
|
|
|
lines.push(line);
|
|
|
|
}
|
2019-08-05 01:50:02 +08:00
|
|
|
} else if (segment.type === SegmentType.String) {
|
2019-08-05 04:59:32 +08:00
|
|
|
lines.push(" " + JSON.stringify(segment.value));
|
2019-07-30 03:59:16 +08:00
|
|
|
} else {
|
|
|
|
for (const instruction of segment.instructions) {
|
2019-08-05 01:50:02 +08:00
|
|
|
if (!manual_stack && instruction.opcode.stack === StackInteraction.Push) {
|
2019-07-30 03:59:16 +08:00
|
|
|
stack.push(...instruction.args);
|
|
|
|
} else {
|
2019-08-05 01:50:02 +08:00
|
|
|
let args: string[] = [];
|
2019-07-30 03:59:16 +08:00
|
|
|
|
2019-08-05 01:50:02 +08:00
|
|
|
if (instruction.opcode.stack === StackInteraction.Pop) {
|
|
|
|
if (!manual_stack) {
|
|
|
|
args = args_to_strings(
|
|
|
|
instruction.opcode.params,
|
2019-07-30 03:59:16 +08:00
|
|
|
stack.splice(
|
2019-08-05 01:50:02 +08:00
|
|
|
Math.max(0, stack.length - instruction.opcode.params.length),
|
2019-08-11 04:09:06 +08:00
|
|
|
instruction.opcode.params.length,
|
2019-08-06 23:37:34 +08:00
|
|
|
),
|
2019-08-11 04:09:06 +08:00
|
|
|
true,
|
2019-08-05 01:50:02 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
2019-08-06 23:37:34 +08:00
|
|
|
args = args_to_strings(instruction.opcode.params, instruction.args, false);
|
2019-07-30 03:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
lines.push(
|
|
|
|
" " +
|
|
|
|
instruction.opcode.mnemonic +
|
2019-08-11 04:09:06 +08:00
|
|
|
(args.length ? " " + args.join(", ") : ""),
|
2019-07-29 07:02:22 +08:00
|
|
|
);
|
|
|
|
}
|
2019-07-22 02:44:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-29 07:02:22 +08:00
|
|
|
// Ensure newline at the end.
|
2019-07-22 18:31:20 +08:00
|
|
|
if (lines.length) {
|
|
|
|
lines.push("");
|
|
|
|
}
|
|
|
|
|
2019-07-23 21:54:42 +08:00
|
|
|
return lines;
|
2019-07-22 02:44:34 +08:00
|
|
|
}
|
|
|
|
|
2019-08-06 23:37:34 +08:00
|
|
|
function args_to_strings(params: Param[], args: Arg[], stack: boolean): string[] {
|
2019-07-22 18:31:20 +08:00
|
|
|
const arg_strings: string[] = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < params.length; i++) {
|
|
|
|
const type = params[i].type;
|
|
|
|
const arg = args[i];
|
|
|
|
|
|
|
|
if (arg == null) {
|
|
|
|
arg_strings.push("");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-08-07 14:43:47 +08:00
|
|
|
switch (type.kind) {
|
|
|
|
case Kind.Float:
|
2019-08-06 23:37:34 +08:00
|
|
|
// Floats are pushed onto the stack as integers with arg_pushl.
|
|
|
|
if (stack) {
|
|
|
|
arg_strings.push(reinterpret_i32_as_f32(arg.value).toString());
|
|
|
|
} else {
|
|
|
|
arg_strings.push(arg.value.toString());
|
|
|
|
}
|
|
|
|
break;
|
2019-08-07 14:43:47 +08:00
|
|
|
case Kind.ILabelVar:
|
2019-07-22 18:31:20 +08:00
|
|
|
for (; i < args.length; i++) {
|
|
|
|
arg_strings.push(args[i].value.toString());
|
|
|
|
}
|
|
|
|
break;
|
2019-08-07 14:43:47 +08:00
|
|
|
case Kind.RegRefVar:
|
2019-08-05 01:50:02 +08:00
|
|
|
for (; i < args.length; i++) {
|
|
|
|
arg_strings.push("r" + args[i].value);
|
|
|
|
}
|
|
|
|
break;
|
2019-08-07 14:43:47 +08:00
|
|
|
case Kind.RegRef:
|
|
|
|
case Kind.RegTupRef:
|
2019-07-22 18:31:20 +08:00
|
|
|
arg_strings.push("r" + arg.value);
|
|
|
|
break;
|
2019-08-07 14:43:47 +08:00
|
|
|
case Kind.String:
|
2019-07-22 18:31:20 +08:00
|
|
|
arg_strings.push(JSON.stringify(arg.value));
|
|
|
|
break;
|
|
|
|
default:
|
2019-08-07 14:43:47 +08:00
|
|
|
arg_strings.push(arg.value.toString());
|
2019-07-22 18:31:20 +08:00
|
|
|
break;
|
|
|
|
}
|
2019-07-22 02:44:34 +08:00
|
|
|
}
|
2019-07-22 18:31:20 +08:00
|
|
|
|
|
|
|
return arg_strings;
|
2019-07-22 02:44:34 +08:00
|
|
|
}
|