Added asm meta data to IR.

This commit is contained in:
Daan Vanden Bosch 2019-10-02 19:57:43 +02:00
parent 7bde9988e8
commit ed571b9061
5 changed files with 135 additions and 73 deletions

View File

@ -8,6 +8,7 @@ import {
DataSegment, DataSegment,
Instruction, Instruction,
InstructionSegment, InstructionSegment,
new_arg,
new_instruction, new_instruction,
Segment, Segment,
SegmentType, SegmentType,
@ -638,50 +639,48 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
for (const param of opcode.params) { for (const param of opcode.params) {
switch (param.type.kind) { switch (param.type.kind) {
case Kind.Byte: case Kind.Byte:
args.push({ value: cursor.u8(), size: 1 }); args.push(new_arg(cursor.u8(), 1));
break; break;
case Kind.Word: case Kind.Word:
args.push({ value: cursor.u16(), size: 2 }); args.push(new_arg(cursor.u16(), 2));
break; break;
case Kind.DWord: case Kind.DWord:
args.push({ value: cursor.i32(), size: 4 }); args.push(new_arg(cursor.i32(), 4));
break; break;
case Kind.Float: case Kind.Float:
args.push({ value: cursor.f32(), size: 4 }); args.push(new_arg(cursor.f32(), 4));
break; break;
case Kind.Label: case Kind.Label:
case Kind.ILabel: case Kind.ILabel:
case Kind.DLabel: case Kind.DLabel:
case Kind.SLabel: case Kind.SLabel:
args.push({ value: cursor.u16(), size: 2 }); args.push(new_arg(cursor.u16(), 2));
break; break;
case Kind.String: case Kind.String:
{ {
const start_pos = cursor.position; const start_pos = cursor.position;
args.push({ args.push(
value: cursor.string_utf16( new_arg(
Math.min(4096, cursor.bytes_left), cursor.string_utf16(Math.min(4096, cursor.bytes_left), true, false),
true, cursor.position - start_pos,
false,
), ),
size: cursor.position - start_pos, );
});
} }
break; break;
case Kind.ILabelVar: case Kind.ILabelVar:
{ {
const arg_size = cursor.u8(); const arg_size = cursor.u8();
args.push(...cursor.u16_array(arg_size).map(value => ({ value, size: 2 }))); args.push(...cursor.u16_array(arg_size).map(value => new_arg(value, 2)));
} }
break; break;
case Kind.RegRef: case Kind.RegRef:
case Kind.RegTupRef: case Kind.RegTupRef:
args.push({ value: cursor.u8(), size: 1 }); args.push(new_arg(cursor.u8(), 1));
break; break;
case Kind.RegRefVar: case Kind.RegRefVar:
{ {
const arg_size = cursor.u8(); const arg_size = cursor.u8();
args.push(...cursor.u8_array(arg_size).map(value => ({ value, size: 1 }))); args.push(...cursor.u8_array(arg_size).map(value => new_arg(value, 1)));
} }
break; break;
default: default:

View File

@ -193,7 +193,7 @@ class Assembler {
}; };
} }
private add_instruction(opcode: Opcode, args: Arg[]): void { private add_instruction(opcode: Opcode, args: Arg[], token: IdentToken): void {
if (!this.segment) { if (!this.segment) {
// Unreachable code, technically valid. // Unreachable code, technically valid.
const instruction_segment: InstructionSegment = { const instruction_segment: InstructionSegment = {
@ -205,7 +205,9 @@ class Assembler {
this.segment = instruction_segment; this.segment = instruction_segment;
this.object_code.push(instruction_segment); this.object_code.push(instruction_segment);
} else if (this.segment.type === SegmentType.Instructions) { } else if (this.segment.type === SegmentType.Instructions) {
this.segment.instructions.push(new_instruction(opcode, args)); this.segment.instructions.push(
new_instruction(opcode, args, { col: token.col, len: token.len }),
);
} else { } else {
logger.error(`Line ${this.line_no}: Expected instructions segment.`); logger.error(`Line ${this.line_no}: Expected instructions segment.`);
} }
@ -418,7 +420,8 @@ class Assembler {
} }
} }
private parse_instruction({ col, len, value }: IdentToken): void { private parse_instruction(ident_token: IdentToken): void {
const { col, len, value } = ident_token;
const opcode = OPCODES_BY_MNEMONIC.get(value); const opcode = OPCODES_BY_MNEMONIC.get(value);
if (!opcode) { if (!opcode) {
@ -495,37 +498,45 @@ class Assembler {
if (token.type === TokenType.Register) { if (token.type === TokenType.Register) {
if (param.type.kind === Kind.RegTupRef) { if (param.type.kind === Kind.RegTupRef) {
this.add_instruction(OP_ARG_PUSHB, [arg]); this.add_instruction(OP_ARG_PUSHB, [arg], ident_token);
} else { } else {
this.add_instruction(OP_ARG_PUSHR, [arg]); this.add_instruction(OP_ARG_PUSHR, [arg], ident_token);
} }
} else { } else {
switch (param.type.kind) { switch (param.type.kind) {
case Kind.Byte: case Kind.Byte:
case Kind.RegRef: case Kind.RegRef:
case Kind.RegTupRef: case Kind.RegTupRef:
this.add_instruction(OP_ARG_PUSHB, [arg]); this.add_instruction(OP_ARG_PUSHB, [arg], ident_token);
break; break;
case Kind.Word: case Kind.Word:
case Kind.Label: case Kind.Label:
case Kind.ILabel: case Kind.ILabel:
case Kind.DLabel: case Kind.DLabel:
case Kind.SLabel: case Kind.SLabel:
this.add_instruction(OP_ARG_PUSHW, [arg]); this.add_instruction(OP_ARG_PUSHW, [arg], ident_token);
break; break;
case Kind.DWord: case Kind.DWord:
this.add_instruction(OP_ARG_PUSHL, [arg]); this.add_instruction(OP_ARG_PUSHL, [arg], ident_token);
break; break;
case Kind.Float: case Kind.Float:
this.add_instruction(OP_ARG_PUSHL, [ this.add_instruction(
{ OP_ARG_PUSHL,
value: reinterpret_f32_as_i32(arg.value), [
size: 4, {
}, value: reinterpret_f32_as_i32(arg.value),
]); size: 4,
asm: {
col: token.col,
len: token.len,
},
},
],
ident_token,
);
break; break;
case Kind.String: case Kind.String:
this.add_instruction(OP_ARG_PUSHS, [arg]); this.add_instruction(OP_ARG_PUSHS, [arg], ident_token);
break; break;
default: default:
logger.error( logger.error(
@ -539,7 +550,7 @@ class Assembler {
} }
} }
this.add_instruction(opcode, ins_args.map(([arg]) => arg)); this.add_instruction(opcode, ins_args.map(([arg]) => arg), ident_token);
} }
} }
@ -616,6 +627,10 @@ class Assembler {
{ {
value: token.value, value: token.value,
size: 4, size: 4,
asm: {
col: token.col,
len: token.len,
},
}, },
token, token,
]); ]);
@ -633,6 +648,10 @@ class Assembler {
{ {
value: token.value, value: token.value,
size: 4, size: 4,
asm: {
col: token.col,
len: token.len,
},
}, },
token, token,
]); ]);
@ -656,6 +675,10 @@ class Assembler {
{ {
value: token.value, value: token.value,
size: 2 * token.value.length + 2, size: 2 * token.value.length + 2,
asm: {
col: token.col,
len: token.len,
},
}, },
token, token,
]); ]);
@ -752,6 +775,10 @@ class Assembler {
{ {
value, value,
size, size,
asm: {
col: token.col,
len: token.len,
},
}, },
token, token,
]); ]);
@ -772,6 +799,10 @@ class Assembler {
{ {
value, value,
size: 1, size: 1,
asm: {
col: token.col,
len: token.len,
},
}, },
token, token,
]); ]);

View File

@ -6,7 +6,13 @@ import { BufferCursor } from "../../core/data_formats/cursor/BufferCursor";
import { parse_bin, write_bin } from "../../core/data_formats/parsing/quest/bin"; import { parse_bin, write_bin } from "../../core/data_formats/parsing/quest/bin";
import { assemble } from "./assembly"; import { assemble } from "./assembly";
import { disassemble } from "./disassembly"; import { disassemble } from "./disassembly";
import { new_instruction, Segment, segment_arrays_equal, SegmentType } from "./instructions"; import {
new_arg,
new_instruction,
Segment,
segment_arrays_equal,
SegmentType,
} from "./instructions";
import { OP_ARG_PUSHW, OP_RET, OP_SWITCH_JMP, OP_VA_CALL, OP_VA_END, OP_VA_START } from "./opcodes"; import { OP_ARG_PUSHW, OP_RET, OP_SWITCH_JMP, OP_VA_CALL, OP_VA_END, OP_VA_START } from "./opcodes";
test("vararg instructions should be disassembled correctly", () => { test("vararg instructions should be disassembled correctly", () => {
@ -16,10 +22,10 @@ test("vararg instructions should be disassembled correctly", () => {
labels: [0], labels: [0],
instructions: [ instructions: [
new_instruction(OP_SWITCH_JMP, [ new_instruction(OP_SWITCH_JMP, [
{ value: 90, size: 1 }, new_arg(90, 1),
{ value: 100, size: 2 }, new_arg(100, 2),
{ value: 101, size: 2 }, new_arg(101, 2),
{ value: 102, size: 2 }, new_arg(102, 2),
]), ]),
new_instruction(OP_RET, []), new_instruction(OP_RET, []),
], ],
@ -45,8 +51,8 @@ test("va list instructions should be disassembled correctly", () => {
labels: [0], labels: [0],
instructions: [ instructions: [
new_instruction(OP_VA_START, []), new_instruction(OP_VA_START, []),
new_instruction(OP_ARG_PUSHW, [{ value: 1337, size: 2 }]), new_instruction(OP_ARG_PUSHW, [new_arg(1337, 2)]),
new_instruction(OP_VA_CALL, [{ value: 100, size: 2 }]), new_instruction(OP_VA_CALL, [new_arg(100, 2)]),
new_instruction(OP_VA_END, []), new_instruction(OP_VA_END, []),
new_instruction(OP_RET, []), new_instruction(OP_RET, []),
], ],

View File

@ -1,6 +1,14 @@
import { Kind, Opcode } from "./opcodes"; import { Kind, Opcode } from "./opcodes";
import { array_buffers_equal, arrays_equal } from "../../core/util"; import { array_buffers_equal, arrays_equal } from "../../core/util";
/**
* Dimensions of related assembly code.
*/
export type AsmToken = {
readonly col: number;
readonly len: number;
};
/** /**
* Instruction invocation. * Instruction invocation.
*/ */
@ -19,9 +27,17 @@ export type Instruction = {
* Maps each parameter by index to its arguments. * Maps each parameter by index to its arguments.
*/ */
readonly param_to_args: readonly Arg[][]; readonly param_to_args: readonly Arg[][];
/**
* Dimensions of the opcode's mnemonic in the related asm code.
*/
readonly asm: AsmToken;
}; };
export function new_instruction(opcode: Opcode, args: Arg[]): Instruction { export function new_instruction(
opcode: Opcode,
args: Arg[],
asm: AsmToken = { col: 0, len: 0 },
): Instruction {
const len = Math.min(opcode.params.length, args.length); const len = Math.min(opcode.params.length, args.length);
const param_to_args: Arg[][] = []; const param_to_args: Arg[][] = [];
let arg_size = 0; let arg_size = 0;
@ -55,6 +71,7 @@ export function new_instruction(opcode: Opcode, args: Arg[]): Instruction {
arg_size, arg_size,
size: opcode.size + arg_size, size: opcode.size + arg_size,
param_to_args, param_to_args,
asm,
}; };
} }
@ -66,10 +83,19 @@ function instructions_equal(a: Instruction, b: Instruction): boolean {
* Instruction argument. * Instruction argument.
*/ */
export type Arg = { export type Arg = {
value: any; readonly value: any;
size: number; readonly size: number;
readonly asm: AsmToken;
}; };
export function new_arg(value: any, size: number, asm: AsmToken = { col: 0, len: 0 }): Arg {
return {
value,
size,
asm,
};
}
function args_equal(a: Arg, b: Arg): boolean { function args_equal(a: Arg, b: Arg): boolean {
return a.value === b.value && a.size === b.size; return a.value === b.value && a.size === b.size;
} }

View File

@ -1,6 +1,6 @@
import { Episode } from "../../core/data_formats/parsing/quest/Episode"; import { Episode } from "../../core/data_formats/parsing/quest/Episode";
import { QuestModel } from "../model/QuestModel"; import { QuestModel } from "../model/QuestModel";
import { new_instruction, SegmentType } from "../scripting/instructions"; import { new_arg, new_instruction, SegmentType } from "../scripting/instructions";
import { ObjectType } from "../../core/data_formats/parsing/quest/object_types"; import { ObjectType } from "../../core/data_formats/parsing/quest/object_types";
import { NpcType } from "../../core/data_formats/parsing/quest/npc_types"; import { NpcType } from "../../core/data_formats/parsing/quest/npc_types";
import { import {
@ -38,15 +38,15 @@ export function create_new_quest(episode: Episode): QuestModel {
labels: [0], labels: [0],
type: SegmentType.Instructions, type: SegmentType.Instructions,
instructions: [ instructions: [
new_instruction(OP_SET_EPISODE, [{ value: 0, size: 4 }]), new_instruction(OP_SET_EPISODE, [new_arg(0, 4)]),
new_instruction(OP_ARG_PUSHL, [{ value: 0, size: 4 }]), new_instruction(OP_ARG_PUSHL, [new_arg(0, 4)]),
new_instruction(OP_ARG_PUSHW, [{ value: 150, size: 2 }]), new_instruction(OP_ARG_PUSHW, [new_arg(150, 2)]),
new_instruction(OP_SET_FLOOR_HANDLER, []), new_instruction(OP_SET_FLOOR_HANDLER, []),
new_instruction(OP_BB_MAP_DESIGNATE, [ new_instruction(OP_BB_MAP_DESIGNATE, [
{ value: 0, size: 1 }, new_arg(0, 1),
{ value: 0, size: 2 }, new_arg(0, 2),
{ value: 0, size: 1 }, new_arg(0, 1),
{ value: 0, size: 1 }, new_arg(0, 1),
]), ]),
new_instruction(OP_RET, []), new_instruction(OP_RET, []),
], ],
@ -55,33 +55,33 @@ export function create_new_quest(episode: Episode): QuestModel {
labels: [150], labels: [150],
type: SegmentType.Instructions, type: SegmentType.Instructions,
instructions: [ instructions: [
new_instruction(OP_LETI, [{ value: 60, size: 1 }, { value: 237, size: 4 }]), new_instruction(OP_LETI, [new_arg(60, 1), new_arg(237, 4)]),
new_instruction(OP_LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), new_instruction(OP_LETI, [new_arg(61, 1), new_arg(0, 4)]),
new_instruction(OP_LETI, [{ value: 62, size: 1 }, { value: 333, size: 4 }]), new_instruction(OP_LETI, [new_arg(62, 1), new_arg(333, 4)]),
new_instruction(OP_LETI, [{ value: 63, size: 1 }, { value: -15, size: 4 }]), new_instruction(OP_LETI, [new_arg(63, 1), new_arg(-15, 4)]),
new_instruction(OP_ARG_PUSHL, [{ value: 0, size: 4 }]), new_instruction(OP_ARG_PUSHL, [new_arg(0, 4)]),
new_instruction(OP_ARG_PUSHR, [{ value: 60, size: 1 }]), new_instruction(OP_ARG_PUSHR, [new_arg(60, 1)]),
new_instruction(OP_P_SETPOS, []), new_instruction(OP_P_SETPOS, []),
new_instruction(OP_LETI, [{ value: 60, size: 1 }, { value: 255, size: 4 }]), new_instruction(OP_LETI, [new_arg(60, 1), new_arg(255, 4)]),
new_instruction(OP_LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), new_instruction(OP_LETI, [new_arg(61, 1), new_arg(0, 4)]),
new_instruction(OP_LETI, [{ value: 62, size: 1 }, { value: 338, size: 4 }]), new_instruction(OP_LETI, [new_arg(62, 1), new_arg(338, 4)]),
new_instruction(OP_LETI, [{ value: 63, size: 1 }, { value: -43, size: 4 }]), new_instruction(OP_LETI, [new_arg(63, 1), new_arg(-43, 4)]),
new_instruction(OP_ARG_PUSHL, [{ value: 1, size: 4 }]), new_instruction(OP_ARG_PUSHL, [new_arg(1, 4)]),
new_instruction(OP_ARG_PUSHR, [{ value: 60, size: 1 }]), new_instruction(OP_ARG_PUSHR, [new_arg(60, 1)]),
new_instruction(OP_P_SETPOS, []), new_instruction(OP_P_SETPOS, []),
new_instruction(OP_LETI, [{ value: 60, size: 1 }, { value: 222, size: 4 }]), new_instruction(OP_LETI, [new_arg(60, 1), new_arg(222, 4)]),
new_instruction(OP_LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), new_instruction(OP_LETI, [new_arg(61, 1), new_arg(0, 4)]),
new_instruction(OP_LETI, [{ value: 62, size: 1 }, { value: 322, size: 4 }]), new_instruction(OP_LETI, [new_arg(62, 1), new_arg(322, 4)]),
new_instruction(OP_LETI, [{ value: 63, size: 1 }, { value: 25, size: 4 }]), new_instruction(OP_LETI, [new_arg(63, 1), new_arg(25, 4)]),
new_instruction(OP_ARG_PUSHL, [{ value: 2, size: 4 }]), new_instruction(OP_ARG_PUSHL, [new_arg(2, 4)]),
new_instruction(OP_ARG_PUSHR, [{ value: 60, size: 1 }]), new_instruction(OP_ARG_PUSHR, [new_arg(60, 1)]),
new_instruction(OP_P_SETPOS, []), new_instruction(OP_P_SETPOS, []),
new_instruction(OP_LETI, [{ value: 60, size: 1 }, { value: 248, size: 4 }]), new_instruction(OP_LETI, [new_arg(60, 1), new_arg(248, 4)]),
new_instruction(OP_LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), new_instruction(OP_LETI, [new_arg(61, 1), new_arg(0, 4)]),
new_instruction(OP_LETI, [{ value: 62, size: 1 }, { value: 323, size: 4 }]), new_instruction(OP_LETI, [new_arg(62, 1), new_arg(323, 4)]),
new_instruction(OP_LETI, [{ value: 63, size: 1 }, { value: -20, size: 4 }]), new_instruction(OP_LETI, [new_arg(63, 1), new_arg(-20, 4)]),
new_instruction(OP_ARG_PUSHL, [{ value: 3, size: 4 }]), new_instruction(OP_ARG_PUSHL, [new_arg(3, 4)]),
new_instruction(OP_ARG_PUSHR, [{ value: 60, size: 1 }]), new_instruction(OP_ARG_PUSHR, [new_arg(60, 1)]),
new_instruction(OP_P_SETPOS, []), new_instruction(OP_P_SETPOS, []),
new_instruction(OP_RET, []), new_instruction(OP_RET, []),
], ],