mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
GameCube .qst files are now parsed correctly.
This commit is contained in:
parent
0035588e43
commit
f968d0047c
15
src/core/data_formats/parsing/quest/Version.ts
Normal file
15
src/core/data_formats/parsing/quest/Version.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
export enum Version {
|
||||||
|
/**
|
||||||
|
* Dreamcast
|
||||||
|
*/
|
||||||
|
DC,
|
||||||
|
/**
|
||||||
|
* GameCube
|
||||||
|
*/
|
||||||
|
GC,
|
||||||
|
PC,
|
||||||
|
/**
|
||||||
|
* BlueBurst
|
||||||
|
*/
|
||||||
|
BB,
|
||||||
|
}
|
@ -4,6 +4,7 @@ import { prs_decompress } from "../../compression/prs/decompress";
|
|||||||
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
||||||
import { BufferCursor } from "../../cursor/BufferCursor";
|
import { BufferCursor } from "../../cursor/BufferCursor";
|
||||||
import { parse_bin, write_bin } from "./bin";
|
import { parse_bin, write_bin } from "./bin";
|
||||||
|
import { Version } from "./Version";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Parse a file, convert the resulting structure to BIN again and check whether the end result is equal to the original.
|
* Parse a file, convert the resulting structure to BIN again and check whether the end result is equal to the original.
|
||||||
@ -11,7 +12,7 @@ import { parse_bin, write_bin } from "./bin";
|
|||||||
function test_quest(path: string): void {
|
function test_quest(path: string): void {
|
||||||
const orig_buffer = readFileSync(path);
|
const orig_buffer = readFileSync(path);
|
||||||
const orig_bin = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
const orig_bin = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
const test_buffer = write_bin(parse_bin(orig_bin));
|
const test_buffer = write_bin(parse_bin(orig_bin, Version.BB));
|
||||||
const test_bin = new ArrayBufferCursor(test_buffer, Endianness.Little);
|
const test_bin = new ArrayBufferCursor(test_buffer, Endianness.Little);
|
||||||
|
|
||||||
orig_bin.seek_start(0);
|
orig_bin.seek_start(0);
|
||||||
|
@ -27,6 +27,7 @@ import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
|||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { LogManager } from "../../../Logger";
|
import { LogManager } from "../../../Logger";
|
||||||
|
import { Version } from "./Version";
|
||||||
|
|
||||||
const logger = LogManager.get("core/data_formats/parsing/quest/bin");
|
const logger = LogManager.get("core/data_formats/parsing/quest/bin");
|
||||||
|
|
||||||
@ -47,18 +48,35 @@ SEGMENT_PRIORITY[SegmentType.Data] = 0;
|
|||||||
|
|
||||||
export function parse_bin(
|
export function parse_bin(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
|
version: Version,
|
||||||
entry_labels: number[] = [0],
|
entry_labels: number[] = [0],
|
||||||
lenient: boolean = false,
|
lenient: boolean = false,
|
||||||
): BinFile {
|
): BinFile {
|
||||||
const object_code_offset = cursor.u32(); // Always 4652
|
const object_code_offset = cursor.u32();
|
||||||
const label_offset_table_offset = cursor.u32(); // Relative offsets
|
const label_offset_table_offset = cursor.u32(); // Relative offsets
|
||||||
const size = cursor.u32();
|
const size = cursor.u32();
|
||||||
cursor.seek(4); // Always seems to be 0xFFFFFFFF
|
cursor.seek(4); // Always seems to be 0xFFFFFFFF for BB.
|
||||||
const quest_id = cursor.u32();
|
const dc_gc_format = version === Version.DC || version === Version.GC;
|
||||||
const language = cursor.u32();
|
let quest_id: number;
|
||||||
const quest_name = cursor.string_utf16(64, true, true);
|
let language: number;
|
||||||
const short_description = cursor.string_utf16(256, true, true);
|
|
||||||
const long_description = cursor.string_utf16(576, true, true);
|
if (dc_gc_format) {
|
||||||
|
language = cursor.u16();
|
||||||
|
quest_id = cursor.u16();
|
||||||
|
} else {
|
||||||
|
quest_id = cursor.u32();
|
||||||
|
language = cursor.u32();
|
||||||
|
}
|
||||||
|
|
||||||
|
const quest_name = dc_gc_format
|
||||||
|
? cursor.string_ascii(32, true, true)
|
||||||
|
: cursor.string_utf16(64, true, true);
|
||||||
|
const short_description = dc_gc_format
|
||||||
|
? cursor.string_ascii(128, true, true)
|
||||||
|
: cursor.string_utf16(256, true, true);
|
||||||
|
const long_description = dc_gc_format
|
||||||
|
? cursor.string_ascii(288, true, true)
|
||||||
|
: cursor.string_utf16(576, true, true);
|
||||||
|
|
||||||
if (size !== cursor.size) {
|
if (size !== cursor.size) {
|
||||||
logger.warn(`Value ${size} in bin size field does not match actual size ${cursor.size}.`);
|
logger.warn(`Value ${size} in bin size field does not match actual size ${cursor.size}.`);
|
||||||
@ -78,7 +96,13 @@ export function parse_bin(
|
|||||||
.seek_start(object_code_offset)
|
.seek_start(object_code_offset)
|
||||||
.take(label_offset_table_offset - object_code_offset);
|
.take(label_offset_table_offset - object_code_offset);
|
||||||
|
|
||||||
const segments = parse_object_code(object_code, label_holder, entry_labels, lenient);
|
const segments = parse_object_code(
|
||||||
|
object_code,
|
||||||
|
label_holder,
|
||||||
|
entry_labels,
|
||||||
|
lenient,
|
||||||
|
dc_gc_format,
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
quest_id,
|
quest_id,
|
||||||
@ -220,6 +244,7 @@ function parse_object_code(
|
|||||||
label_holder: LabelHolder,
|
label_holder: LabelHolder,
|
||||||
entry_labels: number[],
|
entry_labels: number[],
|
||||||
lenient: boolean,
|
lenient: boolean,
|
||||||
|
dc_gc_format: boolean,
|
||||||
): Segment[] {
|
): Segment[] {
|
||||||
const offset_to_segment = new Map<number, Segment>();
|
const offset_to_segment = new Map<number, Segment>();
|
||||||
|
|
||||||
@ -229,6 +254,7 @@ function parse_object_code(
|
|||||||
entry_labels.reduce((m, l) => m.set(l, SegmentType.Instructions), new Map()),
|
entry_labels.reduce((m, l) => m.set(l, SegmentType.Instructions), new Map()),
|
||||||
offset_to_segment,
|
offset_to_segment,
|
||||||
lenient,
|
lenient,
|
||||||
|
dc_gc_format,
|
||||||
);
|
);
|
||||||
|
|
||||||
const segments: Segment[] = [];
|
const segments: Segment[] = [];
|
||||||
@ -333,6 +359,7 @@ function find_and_parse_segments(
|
|||||||
labels: Map<number, SegmentType>,
|
labels: Map<number, SegmentType>,
|
||||||
offset_to_segment: Map<number, Segment>,
|
offset_to_segment: Map<number, Segment>,
|
||||||
lenient: boolean,
|
lenient: boolean,
|
||||||
|
dc_gc_format: boolean,
|
||||||
): void {
|
): void {
|
||||||
let start_segment_count: number;
|
let start_segment_count: number;
|
||||||
|
|
||||||
@ -341,7 +368,15 @@ function find_and_parse_segments(
|
|||||||
start_segment_count = offset_to_segment.size;
|
start_segment_count = offset_to_segment.size;
|
||||||
|
|
||||||
for (const [label, type] of labels) {
|
for (const [label, type] of labels) {
|
||||||
parse_segment(offset_to_segment, label_holder, cursor, label, type, lenient);
|
parse_segment(
|
||||||
|
offset_to_segment,
|
||||||
|
label_holder,
|
||||||
|
cursor,
|
||||||
|
label,
|
||||||
|
type,
|
||||||
|
lenient,
|
||||||
|
dc_gc_format,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find label references.
|
// Find label references.
|
||||||
@ -460,6 +495,7 @@ function parse_segment(
|
|||||||
label: number,
|
label: number,
|
||||||
type: SegmentType,
|
type: SegmentType,
|
||||||
lenient: boolean,
|
lenient: boolean,
|
||||||
|
dc_gc_format: boolean,
|
||||||
): void {
|
): void {
|
||||||
try {
|
try {
|
||||||
const info = label_holder.get_info(label);
|
const info = label_holder.get_info(label);
|
||||||
@ -501,13 +537,14 @@ function parse_segment(
|
|||||||
labels,
|
labels,
|
||||||
info.next && info.next.label,
|
info.next && info.next.label,
|
||||||
lenient,
|
lenient,
|
||||||
|
dc_gc_format,
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
case SegmentType.Data:
|
case SegmentType.Data:
|
||||||
parse_data_segment(offset_to_segment, cursor, end_offset, labels);
|
parse_data_segment(offset_to_segment, cursor, end_offset, labels);
|
||||||
break;
|
break;
|
||||||
case SegmentType.String:
|
case SegmentType.String:
|
||||||
parse_string_segment(offset_to_segment, cursor, end_offset, labels);
|
parse_string_segment(offset_to_segment, cursor, end_offset, labels, dc_gc_format);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Segment type ${SegmentType[type]} not implemented.`);
|
throw new Error(`Segment type ${SegmentType[type]} not implemented.`);
|
||||||
@ -529,6 +566,7 @@ function parse_instructions_segment(
|
|||||||
labels: number[],
|
labels: number[],
|
||||||
next_label: number | undefined,
|
next_label: number | undefined,
|
||||||
lenient: boolean,
|
lenient: boolean,
|
||||||
|
dc_gc_format: boolean,
|
||||||
): void {
|
): void {
|
||||||
const instructions: Instruction[] = [];
|
const instructions: Instruction[] = [];
|
||||||
|
|
||||||
@ -559,7 +597,7 @@ function parse_instructions_segment(
|
|||||||
|
|
||||||
// Parse the arguments.
|
// Parse the arguments.
|
||||||
try {
|
try {
|
||||||
const args = parse_instruction_arguments(cursor, opcode);
|
const args = parse_instruction_arguments(cursor, opcode, dc_gc_format);
|
||||||
instructions.push(new_instruction(opcode, args));
|
instructions.push(new_instruction(opcode, args));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (lenient) {
|
if (lenient) {
|
||||||
@ -596,6 +634,7 @@ function parse_instructions_segment(
|
|||||||
next_label,
|
next_label,
|
||||||
SegmentType.Instructions,
|
SegmentType.Instructions,
|
||||||
lenient,
|
lenient,
|
||||||
|
dc_gc_format,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -622,18 +661,21 @@ function parse_string_segment(
|
|||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
end_offset: number,
|
end_offset: number,
|
||||||
labels: number[],
|
labels: number[],
|
||||||
|
dc_gc_format: boolean,
|
||||||
): void {
|
): void {
|
||||||
const start_offset = cursor.position;
|
const start_offset = cursor.position;
|
||||||
const segment: StringSegment = {
|
const segment: StringSegment = {
|
||||||
type: SegmentType.String,
|
type: SegmentType.String,
|
||||||
labels,
|
labels,
|
||||||
value: cursor.string_utf16(end_offset - start_offset, true, true),
|
value: dc_gc_format
|
||||||
|
? cursor.string_ascii(end_offset - start_offset, true, true)
|
||||||
|
: cursor.string_utf16(end_offset - start_offset, true, true),
|
||||||
asm: { labels: [] },
|
asm: { labels: [] },
|
||||||
};
|
};
|
||||||
offset_to_segment.set(start_offset, segment);
|
offset_to_segment.set(start_offset, segment);
|
||||||
}
|
}
|
||||||
|
|
||||||
function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
function parse_instruction_arguments(cursor: Cursor, opcode: Opcode, dc_gc_format: boolean): Arg[] {
|
||||||
const args: Arg[] = [];
|
const args: Arg[] = [];
|
||||||
|
|
||||||
if (opcode.stack !== StackInteraction.Pop) {
|
if (opcode.stack !== StackInteraction.Pop) {
|
||||||
@ -660,9 +702,12 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
|||||||
case Kind.String:
|
case Kind.String:
|
||||||
{
|
{
|
||||||
const start_pos = cursor.position;
|
const start_pos = cursor.position;
|
||||||
|
const max_bytes = Math.min(4096, cursor.bytes_left);
|
||||||
args.push(
|
args.push(
|
||||||
new_arg(
|
new_arg(
|
||||||
cursor.string_utf16(Math.min(4096, cursor.bytes_left), true, false),
|
dc_gc_format
|
||||||
|
? cursor.string_ascii(max_bytes, true, false)
|
||||||
|
: cursor.string_utf16(max_bytes, true, false),
|
||||||
cursor.position - start_pos,
|
cursor.position - start_pos,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
@ -20,6 +20,7 @@ import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
|||||||
import { npc_data, NpcType } from "./npc_types";
|
import { npc_data, NpcType } from "./npc_types";
|
||||||
import { reinterpret_f32_as_i32, reinterpret_i32_as_f32 } from "../../../primitive_conversion";
|
import { reinterpret_f32_as_i32, reinterpret_i32_as_f32 } from "../../../primitive_conversion";
|
||||||
import { LogManager } from "../../../Logger";
|
import { LogManager } from "../../../Logger";
|
||||||
|
import { Version } from "./Version";
|
||||||
|
|
||||||
const logger = LogManager.get("core/data_formats/parsing/quest");
|
const logger = LogManager.get("core/data_formats/parsing/quest");
|
||||||
|
|
||||||
@ -90,6 +91,7 @@ export function parse_quest(cursor: Cursor, lenient: boolean = false): Quest | u
|
|||||||
);
|
);
|
||||||
const bin = parse_bin(
|
const bin = parse_bin(
|
||||||
bin_decompressed,
|
bin_decompressed,
|
||||||
|
qst.version === Version.DC || qst.version === Version.GC ? 1 : 2,
|
||||||
extract_script_entry_points(objects, dat.npcs),
|
extract_script_entry_points(objects, dat.npcs),
|
||||||
lenient,
|
lenient,
|
||||||
);
|
);
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
import { walk_qst_files } from "../../../../../test/src/utils";
|
import { walk_qst_files } from "../../../../../test/src/utils";
|
||||||
import { parse_qst, Version, write_qst } from "./qst";
|
import { parse_qst, write_qst } from "./qst";
|
||||||
import { Endianness } from "../../Endianness";
|
import { Endianness } from "../../Endianness";
|
||||||
import { BufferCursor } from "../../cursor/BufferCursor";
|
import { BufferCursor } from "../../cursor/BufferCursor";
|
||||||
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
||||||
import * as fs from "fs";
|
import * as fs from "fs";
|
||||||
|
import { Version } from "./Version";
|
||||||
|
|
||||||
test("Parse a GC quest.", () => {
|
test("Parse a GC quest.", () => {
|
||||||
const buf = fs.readFileSync("test/resources/lost_heat_sword_gc.qst");
|
const buf = fs.readFileSync("test/resources/lost_heat_sword_gc.qst");
|
||||||
|
@ -6,6 +6,7 @@ import { WritableCursor } from "../../cursor/WritableCursor";
|
|||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { basename, defined } from "../../../util";
|
import { basename, defined } from "../../../util";
|
||||||
import { LogManager } from "../../../Logger";
|
import { LogManager } from "../../../Logger";
|
||||||
|
import { Version } from "./Version";
|
||||||
|
|
||||||
const logger = LogManager.get("core/data_formats/parsing/quest/qst");
|
const logger = LogManager.get("core/data_formats/parsing/quest/qst");
|
||||||
|
|
||||||
@ -15,22 +16,6 @@ const ONLINE_QUEST = 0x44;
|
|||||||
const DOWNLOAD_QUEST = 0xa6;
|
const DOWNLOAD_QUEST = 0xa6;
|
||||||
const CHUNK_BODY_SIZE = 1024;
|
const CHUNK_BODY_SIZE = 1024;
|
||||||
|
|
||||||
export enum Version {
|
|
||||||
/**
|
|
||||||
* Dreamcast
|
|
||||||
*/
|
|
||||||
DC,
|
|
||||||
/**
|
|
||||||
* GameCube
|
|
||||||
*/
|
|
||||||
GC,
|
|
||||||
PC,
|
|
||||||
/**
|
|
||||||
* BlueBurst
|
|
||||||
*/
|
|
||||||
BB,
|
|
||||||
}
|
|
||||||
|
|
||||||
export type QstContainedFile = {
|
export type QstContainedFile = {
|
||||||
readonly id?: number;
|
readonly id?: number;
|
||||||
readonly filename: string;
|
readonly filename: string;
|
||||||
|
@ -14,6 +14,7 @@ import {
|
|||||||
SegmentType,
|
SegmentType,
|
||||||
} from "./instructions";
|
} 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";
|
||||||
|
import { Version } from "../../core/data_formats/parsing/quest/Version";
|
||||||
|
|
||||||
test("vararg instructions should be disassembled correctly", () => {
|
test("vararg instructions should be disassembled correctly", () => {
|
||||||
const asm = disassemble([
|
const asm = disassemble([
|
||||||
@ -82,7 +83,7 @@ test("va list instructions should be disassembled correctly", () => {
|
|||||||
test("assembling disassembled object code with manual stack management should result in the same IR", () => {
|
test("assembling disassembled object code with manual stack management should result in the same IR", () => {
|
||||||
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
const bin = parse_bin(orig_bytes);
|
const bin = parse_bin(orig_bytes, Version.BB);
|
||||||
|
|
||||||
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, true), true);
|
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, true), true);
|
||||||
|
|
||||||
@ -96,7 +97,7 @@ test("assembling disassembled object code with manual stack management should re
|
|||||||
test("assembling disassembled object code with automatic stack management should result in the same IR", () => {
|
test("assembling disassembled object code with automatic stack management should result in the same IR", () => {
|
||||||
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
const bin = parse_bin(orig_bytes);
|
const bin = parse_bin(orig_bytes, Version.BB);
|
||||||
|
|
||||||
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, false), false);
|
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, false), false);
|
||||||
|
|
||||||
@ -110,7 +111,7 @@ test("assembling disassembled object code with automatic stack management should
|
|||||||
test("assembling disassembled object code with manual stack management should result in the same object code", () => {
|
test("assembling disassembled object code with manual stack management should result in the same object code", () => {
|
||||||
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
const orig_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
const bin = parse_bin(orig_bytes);
|
const bin = parse_bin(orig_bytes, Version.BB);
|
||||||
|
|
||||||
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, true), true);
|
const { object_code, warnings, errors } = assemble(disassemble(bin.object_code, true), true);
|
||||||
|
|
||||||
@ -144,7 +145,7 @@ test("assembling disassembled object code with manual stack management should re
|
|||||||
test("disassembling assembled assembly code with automatic stack management should result the same assembly code", () => {
|
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_buffer = readFileSync("test/resources/quest27_e.bin");
|
||||||
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
const orig_bytes = prs_decompress(new BufferCursor(orig_buffer, Endianness.Little));
|
||||||
const orig_asm = disassemble(parse_bin(orig_bytes).object_code, false);
|
const orig_asm = disassemble(parse_bin(orig_bytes, Version.BB).object_code, false);
|
||||||
|
|
||||||
const { object_code, warnings, errors } = assemble(orig_asm, false);
|
const { object_code, warnings, errors } = assemble(orig_asm, false);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user