GameCube .qst files are now parsed correctly.

This commit is contained in:
Daan Vanden Bosch 2020-01-02 15:00:26 +01:00
parent 0035588e43
commit f968d0047c
7 changed files with 86 additions and 36 deletions

View File

@ -0,0 +1,15 @@
export enum Version {
/**
* Dreamcast
*/
DC,
/**
* GameCube
*/
GC,
PC,
/**
* BlueBurst
*/
BB,
}

View File

@ -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);

View File

@ -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,
), ),
); );

View File

@ -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,
); );

View File

@ -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");

View File

@ -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;

View File

@ -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);