2019-08-11 04:09:06 +08:00
|
|
|
import { reinterpret_i32_as_f32 } from "../../core/primitive_conversion";
|
2020-01-03 01:42:08 +08:00
|
|
|
import { Arg, Segment, SegmentType } from "../../core/data_formats/asm/instructions";
|
|
|
|
import { AnyType, Kind, OP_VA_END, OP_VA_START, Param, StackInteraction } from "../../core/data_formats/asm/opcodes";
|
2019-12-25 07:17:02 +08:00
|
|
|
import { LogManager } from "../../core/Logger";
|
2019-12-27 10:16:52 +08:00
|
|
|
import { number_to_hex_string } from "../../core/util";
|
2019-09-17 00:52:16 +08:00
|
|
|
|
2019-12-25 07:17:02 +08:00
|
|
|
const logger = LogManager.get("quest_editor/scripting/disassembly");
|
2019-09-17 00:52:16 +08:00
|
|
|
|
|
|
|
type ArgWithType = Arg & {
|
|
|
|
/**
|
|
|
|
* Type inferred from the specific instruction used to push this argument onto the stack.
|
|
|
|
*/
|
|
|
|
type: AnyType;
|
|
|
|
};
|
2019-07-22 02:44:34 +08:00
|
|
|
|
2019-07-22 18:31:20 +08:00
|
|
|
/**
|
2019-09-17 00:52:16 +08:00
|
|
|
* @param object_code - The object code to disassemble.
|
|
|
|
* @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-10-10 19:47:43 +08:00
|
|
|
export function disassemble(object_code: readonly Segment[], manual_stack = false): string[] {
|
2019-09-17 00:52:16 +08:00
|
|
|
logger.trace("disassemble start");
|
|
|
|
|
2019-07-22 02:44:34 +08:00
|
|
|
const lines: string[] = [];
|
2019-09-17 00:52:16 +08:00
|
|
|
const stack: ArgWithType[] = [];
|
|
|
|
let section_type: SegmentType | undefined = 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++) {
|
2020-01-01 04:48:59 +08:00
|
|
|
line += "0x" + number_to_hex_string(bytes[i], 2);
|
2019-07-30 03:59:16 +08:00
|
|
|
|
|
|
|
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-10-03 23:06:23 +08:00
|
|
|
lines.push(" " + string_segment_to_string(segment.value));
|
2019-07-30 03:59:16 +08:00
|
|
|
} else {
|
2019-10-02 04:47:01 +08:00
|
|
|
// SegmentType.Instructions
|
|
|
|
let in_va_list = false;
|
|
|
|
|
2019-07-30 03:59:16 +08:00
|
|
|
for (const instruction of segment.instructions) {
|
2019-10-02 20:25:47 +08:00
|
|
|
if (instruction.opcode.code === OP_VA_START.code) {
|
2019-10-02 04:47:01 +08:00
|
|
|
in_va_list = true;
|
2019-10-02 20:25:47 +08:00
|
|
|
} else if (instruction.opcode.code === OP_VA_END.code) {
|
2019-10-02 04:47:01 +08:00
|
|
|
in_va_list = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
!manual_stack &&
|
|
|
|
!in_va_list &&
|
|
|
|
instruction.opcode.stack === StackInteraction.Push
|
|
|
|
) {
|
2019-09-17 00:52:16 +08:00
|
|
|
stack.push(...add_type_to_args(instruction.opcode.params, instruction.args));
|
2019-07-30 03:59:16 +08:00
|
|
|
} 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-09-17 00:52:16 +08:00
|
|
|
args = args_to_strings(
|
|
|
|
instruction.opcode.params,
|
|
|
|
add_type_to_args(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-09-17 00:52:16 +08:00
|
|
|
logger.trace(`disassemble end, line count: ${lines.length}`);
|
2019-07-23 21:54:42 +08:00
|
|
|
return lines;
|
2019-07-22 02:44:34 +08:00
|
|
|
}
|
|
|
|
|
2019-10-02 19:44:55 +08:00
|
|
|
function add_type_to_args(params: readonly Param[], args: readonly Arg[]): ArgWithType[] {
|
2019-09-17 00:52:16 +08:00
|
|
|
const args_with_type: ArgWithType[] = [];
|
|
|
|
const len = Math.min(params.length, args.length);
|
|
|
|
|
|
|
|
for (let i = 0; i < len; i++) {
|
|
|
|
args_with_type.push({ ...args[i], type: params[i].type });
|
|
|
|
}
|
|
|
|
|
2019-10-02 04:18:14 +08:00
|
|
|
// Deal with varargs.
|
|
|
|
const last_param = params[params.length - 1];
|
|
|
|
|
|
|
|
if (
|
|
|
|
last_param &&
|
|
|
|
(last_param.type.kind === Kind.ILabelVar || last_param.type.kind === Kind.RegRefVar)
|
|
|
|
) {
|
|
|
|
const len = args.length;
|
|
|
|
|
|
|
|
for (let i = args_with_type.length; i < len; i++) {
|
|
|
|
args_with_type.push({ ...args[i], type: last_param.type });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-17 00:52:16 +08:00
|
|
|
return args_with_type;
|
|
|
|
}
|
|
|
|
|
2019-10-02 19:44:55 +08:00
|
|
|
function args_to_strings(params: readonly Param[], args: ArgWithType[], 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];
|
|
|
|
|
2019-10-02 04:18:14 +08:00
|
|
|
if (arg == undefined) {
|
2019-07-22 18:31:20 +08:00
|
|
|
arg_strings.push("");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-09-17 00:52:16 +08:00
|
|
|
if (arg.type.kind === Kind.RegTupRef) {
|
|
|
|
arg_strings.push("r" + arg.value);
|
|
|
|
} else {
|
|
|
|
switch (type.kind) {
|
|
|
|
case Kind.Float:
|
|
|
|
// 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;
|
|
|
|
case Kind.ILabelVar:
|
|
|
|
for (; i < args.length; i++) {
|
|
|
|
arg_strings.push(args[i].value.toString());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Kind.RegRefVar:
|
|
|
|
for (; i < args.length; i++) {
|
|
|
|
arg_strings.push("r" + args[i].value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Kind.RegRef:
|
|
|
|
case Kind.RegTupRef:
|
|
|
|
arg_strings.push("r" + arg.value);
|
|
|
|
break;
|
|
|
|
case Kind.String:
|
|
|
|
arg_strings.push(JSON.stringify(arg.value));
|
|
|
|
break;
|
|
|
|
default:
|
2019-08-06 23:37:34 +08:00
|
|
|
arg_strings.push(arg.value.toString());
|
2019-09-17 00:52:16 +08:00
|
|
|
break;
|
|
|
|
}
|
2019-07-22 18:31:20 +08:00
|
|
|
}
|
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
|
|
|
}
|
2019-10-03 23:06:23 +08:00
|
|
|
|
|
|
|
function string_segment_to_string(str: string): string {
|
|
|
|
return JSON.stringify(str.replace(/<cr>/g, "\n"));
|
|
|
|
}
|