From fa76002d1c68dad8ff94bc87e685decf8d45d5b8 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Wed, 2 Oct 2019 13:44:55 +0200 Subject: [PATCH] Instruction is now structured cloning-safe. --- src/core/data_formats/parsing/quest/bin.ts | 5 +- src/core/util.ts | 4 +- src/quest_editor/scripting/assembly.ts | 6 +- .../scripting/disassembly.test.ts | 20 ++--- src/quest_editor/scripting/disassembly.ts | 4 +- src/quest_editor/scripting/instructions.ts | 80 +++++++++++-------- src/quest_editor/stores/quest_creation.ts | 72 ++++++++--------- 7 files changed, 101 insertions(+), 90 deletions(-) diff --git a/src/core/data_formats/parsing/quest/bin.ts b/src/core/data_formats/parsing/quest/bin.ts index 04efde71..155744f1 100644 --- a/src/core/data_formats/parsing/quest/bin.ts +++ b/src/core/data_formats/parsing/quest/bin.ts @@ -8,6 +8,7 @@ import { DataSegment, Instruction, InstructionSegment, + new_instruction, Segment, SegmentType, StringSegment, @@ -557,14 +558,14 @@ function parse_instructions_segment( // Parse the arguments. try { const args = parse_instruction_arguments(cursor, opcode); - instructions.push(new Instruction(opcode, args)); + instructions.push(new_instruction(opcode, args)); } catch (e) { if (lenient) { logger.error( `Exception occurred while parsing arguments for instruction ${opcode.mnemonic}.`, e, ); - instructions.push(new Instruction(opcode, [])); + instructions.push(new_instruction(opcode, [])); } else { throw e; } diff --git a/src/core/util.ts b/src/core/util.ts index 9ed2775d..f9b3dbbd 100644 --- a/src/core/util.ts +++ b/src/core/util.ts @@ -1,6 +1,6 @@ export function arrays_equal( - a: T[], - b: T[], + a: readonly T[], + b: readonly T[], equal: (element_a: T, element_b: T) => boolean = (a, b) => a === b, ): boolean { const len = a.length; diff --git a/src/quest_editor/scripting/assembly.ts b/src/quest_editor/scripting/assembly.ts index d8f89daa..7d2edb9c 100644 --- a/src/quest_editor/scripting/assembly.ts +++ b/src/quest_editor/scripting/assembly.ts @@ -16,8 +16,8 @@ import { import { Arg, DataSegment, - Instruction, InstructionSegment, + new_instruction, Segment, SegmentType, StringSegment, @@ -194,7 +194,7 @@ class Assembler { this.segment = instruction_segment; this.object_code.push(instruction_segment); } else if (this.segment.type === SegmentType.Instructions) { - this.segment.instructions.push(new Instruction(opcode, args)); + this.segment.instructions.push(new_instruction(opcode, args)); } else { logger.error(`Line ${this.line_no}: Expected instructions segment.`); } @@ -202,7 +202,7 @@ class Assembler { private add_bytes(bytes: number[]): void { if (!this.segment) { - // Unadressable data, technically valid. + // Unaddressable data, technically valid. const data_segment: DataSegment = { labels: [], type: SegmentType.Data, diff --git a/src/quest_editor/scripting/disassembly.test.ts b/src/quest_editor/scripting/disassembly.test.ts index 1ca42de4..2968ada0 100644 --- a/src/quest_editor/scripting/disassembly.test.ts +++ b/src/quest_editor/scripting/disassembly.test.ts @@ -6,7 +6,7 @@ import { BufferCursor } from "../../core/data_formats/cursor/BufferCursor"; import { parse_bin, write_bin } from "../../core/data_formats/parsing/quest/bin"; import { assemble } from "./assembly"; import { disassemble } from "./disassembly"; -import { Instruction, object_code_equal, Segment, SegmentType } from "./instructions"; +import { new_instruction, segment_arrays_equal, Segment, SegmentType } from "./instructions"; import { Opcode } from "./opcodes"; test("vararg instructions should be disassembled correctly", () => { @@ -15,13 +15,13 @@ test("vararg instructions should be disassembled correctly", () => { type: SegmentType.Instructions, labels: [0], instructions: [ - new Instruction(Opcode.SWITCH_JMP, [ + new_instruction(Opcode.SWITCH_JMP, [ { value: 90, size: 1 }, { value: 100, size: 2 }, { value: 101, size: 2 }, { value: 102, size: 2 }, ]), - new Instruction(Opcode.RET, []), + new_instruction(Opcode.RET, []), ], }, ]); @@ -44,11 +44,11 @@ test("va list instructions should be disassembled correctly", () => { type: SegmentType.Instructions, labels: [0], instructions: [ - new Instruction(Opcode.VA_START, []), - new Instruction(Opcode.ARG_PUSHW, [{ value: 1337, size: 2 }]), - new Instruction(Opcode.VA_CALL, [{ value: 100, size: 2 }]), - new Instruction(Opcode.VA_END, []), - new Instruction(Opcode.RET, []), + new_instruction(Opcode.VA_START, []), + new_instruction(Opcode.ARG_PUSHW, [{ value: 1337, size: 2 }]), + new_instruction(Opcode.VA_CALL, [{ value: 100, size: 2 }]), + new_instruction(Opcode.VA_END, []), + new_instruction(Opcode.RET, []), ], }, ]; @@ -81,7 +81,7 @@ test("assembling disassembled object code with manual stack management should re expect(errors).toEqual([]); expect(warnings).toEqual([]); - expect(object_code_equal(object_code, bin.object_code)).toBe(true); + expect(segment_arrays_equal(object_code, bin.object_code)).toBe(true); }); // Round-trip test. @@ -95,7 +95,7 @@ test("assembling disassembled object code with automatic stack management should expect(errors).toEqual([]); expect(warnings).toEqual([]); - expect(object_code_equal(object_code, bin.object_code)).toBe(true); + expect(segment_arrays_equal(object_code, bin.object_code)).toBe(true); }); // Round-trip test. diff --git a/src/quest_editor/scripting/disassembly.ts b/src/quest_editor/scripting/disassembly.ts index 2a023e2d..584fdb91 100644 --- a/src/quest_editor/scripting/disassembly.ts +++ b/src/quest_editor/scripting/disassembly.ts @@ -133,7 +133,7 @@ export function disassemble(object_code: Segment[], manual_stack = false): strin return lines; } -function add_type_to_args(params: Param[], args: Arg[]): ArgWithType[] { +function add_type_to_args(params: readonly Param[], args: readonly Arg[]): ArgWithType[] { const args_with_type: ArgWithType[] = []; const len = Math.min(params.length, args.length); @@ -158,7 +158,7 @@ function add_type_to_args(params: Param[], args: Arg[]): ArgWithType[] { return args_with_type; } -function args_to_strings(params: Param[], args: ArgWithType[], stack: boolean): string[] { +function args_to_strings(params: readonly Param[], args: ArgWithType[], stack: boolean): string[] { const arg_strings: string[] = []; for (let i = 0; i < params.length; i++) { diff --git a/src/quest_editor/scripting/instructions.ts b/src/quest_editor/scripting/instructions.ts index c21d4c0a..fe1d4452 100644 --- a/src/quest_editor/scripting/instructions.ts +++ b/src/quest_editor/scripting/instructions.ts @@ -4,11 +4,13 @@ import { array_buffers_equal, arrays_equal } from "../../core/util"; /** * Instruction invocation. */ -export class Instruction { +export type Instruction = { + readonly opcode: Opcode; + readonly args: readonly Arg[]; /** * Byte size of the argument list. */ - readonly arg_size: number = 0; + readonly arg_size: number; /** * Byte size of the entire instruction, i.e. the sum of the opcode size and all argument sizes. */ @@ -16,36 +18,48 @@ export class Instruction { /** * Maps each parameter by index to its arguments. */ - readonly param_to_args: Arg[][] = []; + readonly param_to_args: readonly Arg[][]; +}; - constructor(readonly opcode: Opcode, readonly args: Arg[]) { - const len = Math.min(opcode.params.length, args.length); +export function new_instruction(opcode: Opcode, args: Arg[]): Instruction { + const len = Math.min(opcode.params.length, args.length); + const param_to_args: Arg[][] = []; + let arg_size = 0; - for (let i = 0; i < len; i++) { - const type = opcode.params[i].type; - const arg = args[i]; - this.param_to_args[i] = []; + for (let i = 0; i < len; i++) { + const type = opcode.params[i].type; + const arg = args[i]; + param_to_args[i] = []; - switch (type.kind) { - case Kind.ILabelVar: - case Kind.RegRefVar: - this.arg_size++; + switch (type.kind) { + case Kind.ILabelVar: + case Kind.RegRefVar: + arg_size++; - for (let j = i; j < args.length; j++) { - this.param_to_args[i].push(args[j]); - this.arg_size += args[j].size; - } + for (let j = i; j < args.length; j++) { + param_to_args[i].push(args[j]); + arg_size += args[j].size; + } - break; - default: - this.arg_size += arg.size; - this.param_to_args[i].push(arg); - break; - } + break; + default: + arg_size += arg.size; + param_to_args[i].push(arg); + break; } - - this.size = opcode.size + this.arg_size; } + + return { + opcode, + args, + arg_size, + size: opcode.size + arg_size, + param_to_args, + }; +} + +function instructions_equal(a: Instruction, b: Instruction): boolean { + return a.opcode.code === b.opcode.code && arrays_equal(a.args, b.args, args_equal); } /** @@ -56,6 +70,10 @@ export type Arg = { size: number; }; +function args_equal(a: Arg, b: Arg): boolean { + return a.value === b.value && a.size === b.size; +} + export enum SegmentType { Instructions, Data, @@ -85,10 +103,6 @@ export type StringSegment = { value: string; }; -export function object_code_equal(a: Segment[], b: Segment[]): boolean { - return arrays_equal(a, b, segments_equal); -} - function segments_equal(a: Segment, b: Segment): boolean { if (a.type !== b.type || !arrays_equal(a.labels, b.labels)) return false; @@ -108,10 +122,6 @@ function segments_equal(a: Segment, b: Segment): boolean { } } -function instructions_equal(a: Instruction, b: Instruction): boolean { - return a.opcode.code === b.opcode.code && arrays_equal(a.args, b.args, args_equal); -} - -function args_equal(a: Arg, b: Arg): boolean { - return a.value === b.value && a.size === b.size; +export function segment_arrays_equal(a: Segment[], b: Segment[]): boolean { + return arrays_equal(a, b, segments_equal); } diff --git a/src/quest_editor/stores/quest_creation.ts b/src/quest_editor/stores/quest_creation.ts index c0c03cba..30c77266 100644 --- a/src/quest_editor/stores/quest_creation.ts +++ b/src/quest_editor/stores/quest_creation.ts @@ -1,6 +1,6 @@ import { Episode } from "../../core/data_formats/parsing/quest/Episode"; import { QuestModel } from "../model/QuestModel"; -import { Instruction, SegmentType } from "../scripting/instructions"; +import { new_instruction, SegmentType } from "../scripting/instructions"; import { ObjectType } from "../../core/data_formats/parsing/quest/object_types"; import { NpcType } from "../../core/data_formats/parsing/quest/npc_types"; import { Opcode } from "../scripting/opcodes"; @@ -28,52 +28,52 @@ export function create_new_quest(episode: Episode): QuestModel { labels: [0], type: SegmentType.Instructions, instructions: [ - new Instruction(Opcode.SET_EPISODE, [{ value: 0, size: 4 }]), - new Instruction(Opcode.ARG_PUSHL, [{ value: 0, size: 4 }]), - new Instruction(Opcode.ARG_PUSHW, [{ value: 150, size: 2 }]), - new Instruction(Opcode.SET_FLOOR_HANDLER, []), - new Instruction(Opcode.BB_MAP_DESIGNATE, [ + new_instruction(Opcode.SET_EPISODE, [{ value: 0, size: 4 }]), + new_instruction(Opcode.ARG_PUSHL, [{ value: 0, size: 4 }]), + new_instruction(Opcode.ARG_PUSHW, [{ value: 150, size: 2 }]), + new_instruction(Opcode.SET_FLOOR_HANDLER, []), + new_instruction(Opcode.BB_MAP_DESIGNATE, [ { value: 0, size: 1 }, { value: 0, size: 2 }, { value: 0, size: 1 }, { value: 0, size: 1 }, ]), - new Instruction(Opcode.RET, []), + new_instruction(Opcode.RET, []), ], }, { labels: [150], type: SegmentType.Instructions, instructions: [ - new Instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 237, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 333, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -15, size: 4 }]), - new Instruction(Opcode.ARG_PUSHL, [{ value: 0, size: 4 }]), - new Instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), - new Instruction(Opcode.P_SETPOS, []), - new Instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 255, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 338, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -43, size: 4 }]), - new Instruction(Opcode.ARG_PUSHL, [{ value: 1, size: 4 }]), - new Instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), - new Instruction(Opcode.P_SETPOS, []), - new Instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 222, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 322, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: 25, size: 4 }]), - new Instruction(Opcode.ARG_PUSHL, [{ value: 2, size: 4 }]), - new Instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), - new Instruction(Opcode.P_SETPOS, []), - new Instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 248, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 323, size: 4 }]), - new Instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -20, size: 4 }]), - new Instruction(Opcode.ARG_PUSHL, [{ value: 3, size: 4 }]), - new Instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), - new Instruction(Opcode.P_SETPOS, []), - new Instruction(Opcode.RET, []), + new_instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 237, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 333, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -15, size: 4 }]), + new_instruction(Opcode.ARG_PUSHL, [{ value: 0, size: 4 }]), + new_instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), + new_instruction(Opcode.P_SETPOS, []), + new_instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 255, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 338, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -43, size: 4 }]), + new_instruction(Opcode.ARG_PUSHL, [{ value: 1, size: 4 }]), + new_instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), + new_instruction(Opcode.P_SETPOS, []), + new_instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 222, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 322, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: 25, size: 4 }]), + new_instruction(Opcode.ARG_PUSHL, [{ value: 2, size: 4 }]), + new_instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), + new_instruction(Opcode.P_SETPOS, []), + new_instruction(Opcode.LETI, [{ value: 60, size: 1 }, { value: 248, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 61, size: 1 }, { value: 0, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 62, size: 1 }, { value: 323, size: 4 }]), + new_instruction(Opcode.LETI, [{ value: 63, size: 1 }, { value: -20, size: 4 }]), + new_instruction(Opcode.ARG_PUSHL, [{ value: 3, size: 4 }]), + new_instruction(Opcode.ARG_PUSHR, [{ value: 60, size: 1 }]), + new_instruction(Opcode.P_SETPOS, []), + new_instruction(Opcode.RET, []), ], }, ],