2019-07-29 07:02:22 +08:00
|
|
|
import { Arg, Param, Segment, SegmentType, Type } from "../data_formats/parsing/quest/bin";
|
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-07-29 07:02:22 +08:00
|
|
|
for (const segment of object_code) {
|
|
|
|
if (segment.type === SegmentType.Data) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (segment.label !== -1) {
|
|
|
|
lines.push(`${segment.label}:`);
|
|
|
|
}
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-29 07:02:22 +08:00
|
|
|
for (const instruction of segment.instructions) {
|
|
|
|
if (!manual_stack && instruction.opcode.push_stack) {
|
|
|
|
stack.push(...instruction.args);
|
|
|
|
} else {
|
|
|
|
let args = args_to_strings(instruction.opcode.params, instruction.args);
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-29 07:02:22 +08:00
|
|
|
if (!manual_stack) {
|
|
|
|
args.push(
|
|
|
|
...args_to_strings(
|
|
|
|
instruction.opcode.stack_params,
|
|
|
|
stack.splice(
|
|
|
|
Math.max(0, stack.length - instruction.opcode.stack_params.length),
|
|
|
|
instruction.opcode.stack_params.length
|
|
|
|
)
|
2019-07-22 18:31:20 +08:00
|
|
|
)
|
2019-07-29 07:02:22 +08:00
|
|
|
);
|
|
|
|
}
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-29 07:02:22 +08:00
|
|
|
lines.push(
|
|
|
|
" " +
|
|
|
|
instruction.opcode.mnemonic +
|
|
|
|
(args.length ? " " + args.join(", ") : "")
|
|
|
|
);
|
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-07-22 18:31:20 +08:00
|
|
|
function args_to_strings(params: Param[], args: Arg[]): string[] {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case Type.U8Var:
|
2019-07-29 07:02:22 +08:00
|
|
|
case Type.ILabelVar:
|
2019-07-22 18:31:20 +08:00
|
|
|
for (; i < args.length; i++) {
|
|
|
|
arg_strings.push(args[i].value.toString());
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case Type.Register:
|
|
|
|
arg_strings.push("r" + arg.value);
|
|
|
|
break;
|
|
|
|
case Type.String:
|
|
|
|
arg_strings.push(JSON.stringify(arg.value));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
arg_strings.push(arg.value.toString());
|
|
|
|
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
|
|
|
}
|