mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Fixed bug in assembler.
This commit is contained in:
parent
d32c56ed02
commit
c8ae97d8f6
@ -57,7 +57,8 @@ class Assembler {
|
|||||||
private errors!: AssemblyError[];
|
private errors!: AssemblyError[];
|
||||||
// Encountered labels.
|
// Encountered labels.
|
||||||
private labels!: Set<number>;
|
private labels!: Set<number>;
|
||||||
private section: SegmentType = SegmentType.Instructions;
|
private section!: SegmentType;
|
||||||
|
private first_section_marker = true;
|
||||||
|
|
||||||
constructor(private assembly: string[], private manual_stack: boolean) {}
|
constructor(private assembly: string[], private manual_stack: boolean) {}
|
||||||
|
|
||||||
@ -73,6 +74,7 @@ class Assembler {
|
|||||||
this.labels = new Set();
|
this.labels = new Set();
|
||||||
// Need to cast SegmentType.Instructions because of TypeScript bug.
|
// Need to cast SegmentType.Instructions because of TypeScript bug.
|
||||||
this.section = SegmentType.Instructions as SegmentType;
|
this.section = SegmentType.Instructions as SegmentType;
|
||||||
|
this.first_section_marker = true;
|
||||||
|
|
||||||
for (const line of this.assembly) {
|
for (const line of this.assembly) {
|
||||||
this.tokens = this.lexer.tokenize_line(line);
|
this.tokens = this.lexer.tokenize_line(line);
|
||||||
@ -190,7 +192,7 @@ class Assembler {
|
|||||||
const arr = new Uint8Array(buf);
|
const arr = new Uint8Array(buf);
|
||||||
|
|
||||||
arr.set(new Uint8Array(this.segment.data));
|
arr.set(new Uint8Array(this.segment.data));
|
||||||
arr.set(new Uint8Array(bytes));
|
arr.set(new Uint8Array(bytes), this.segment.data.byteLength);
|
||||||
|
|
||||||
this.segment.data = buf;
|
this.segment.data = buf;
|
||||||
} else {
|
} else {
|
||||||
@ -349,7 +351,7 @@ class Assembler {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.section === section) {
|
if (this.section === section && !this.first_section_marker) {
|
||||||
this.add_warning({
|
this.add_warning({
|
||||||
col,
|
col,
|
||||||
length: len,
|
length: len,
|
||||||
@ -358,6 +360,7 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.section = section;
|
this.section = section;
|
||||||
|
this.first_section_marker = false;
|
||||||
|
|
||||||
const next_token = this.tokens.shift();
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
|
65
src/scripting/disassembly.test.ts
Normal file
65
src/scripting/disassembly.test.ts
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
import { readFileSync } from "fs";
|
||||||
|
import { Endianness } from "../data_formats";
|
||||||
|
import { prs_decompress } from "../data_formats/compression/prs/decompress";
|
||||||
|
import { ArrayBufferCursor } from "../data_formats/cursor/ArrayBufferCursor";
|
||||||
|
import { BufferCursor } from "../data_formats/cursor/BufferCursor";
|
||||||
|
import { parse_bin, write_bin } from "../data_formats/parsing/quest/bin";
|
||||||
|
import { assemble } from "./assembly";
|
||||||
|
import { disassemble } from "./disassembly";
|
||||||
|
|
||||||
|
// Roundtrip test.
|
||||||
|
test("assembling dissambled object code with manual stack management should result in the same object code", () => {
|
||||||
|
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
|
const bin = parse_bin(orig_bytes);
|
||||||
|
|
||||||
|
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, true), true);
|
||||||
|
|
||||||
|
expect(errors).toEqual([]);
|
||||||
|
expect(warnings).toEqual([]);
|
||||||
|
|
||||||
|
bin.object_code.splice(0, bin.object_code.length, ...object_code);
|
||||||
|
|
||||||
|
const test_bytes = new ArrayBufferCursor(write_bin(bin), Endianness.Little);
|
||||||
|
|
||||||
|
orig_bytes.seek_start(0);
|
||||||
|
expect(test_bytes.size).toBe(orig_bytes.size);
|
||||||
|
|
||||||
|
let matching_bytes = 0;
|
||||||
|
|
||||||
|
while (orig_bytes.bytes_left) {
|
||||||
|
const test_byte = test_bytes.u8();
|
||||||
|
const orig_byte = orig_bytes.u8();
|
||||||
|
|
||||||
|
if (test_byte !== orig_byte) {
|
||||||
|
throw new Error(
|
||||||
|
`Byte ${matching_bytes} didn't match, expected ${orig_byte}, got ${test_byte}.`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
matching_bytes++;
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(matching_bytes).toBe(orig_bytes.size);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Roundtrip test.
|
||||||
|
test("disassembling assembled assembly code with automatic stack management should result the same assembly code", () => {
|
||||||
|
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
|
const orig_asm = disassemble(parse_bin(orig_bytes).object_code);
|
||||||
|
|
||||||
|
const { object_code, warnings, errors } = assemble(orig_asm);
|
||||||
|
|
||||||
|
expect(errors).toEqual([]);
|
||||||
|
expect(warnings).toEqual([]);
|
||||||
|
|
||||||
|
const test_asm = disassemble(object_code);
|
||||||
|
const len = Math.min(orig_asm.length, test_asm.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
|
expect(test_asm[i]).toBe(orig_asm[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(test_asm.length).toBe(orig_asm.length);
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user