From 5245687533b01dcac86112ccb8e0b59b6402fc54 Mon Sep 17 00:00:00 2001 From: morkt Date: Tue, 26 Jul 2016 07:07:01 +0400 Subject: [PATCH] implemented 'ARCFORM4 WBUG' archives. --- ArcFormats/WildBug/ArcWBP.cs | 83 +++++++++++++++++++++++++++++++++++- 1 file changed, 81 insertions(+), 2 deletions(-) diff --git a/ArcFormats/WildBug/ArcWBP.cs b/ArcFormats/WildBug/ArcWBP.cs index 0a34625d..1091d8f3 100644 --- a/ArcFormats/WildBug/ArcWBP.cs +++ b/ArcFormats/WildBug/ArcWBP.cs @@ -23,6 +23,7 @@ // IN THE SOFTWARE. // +using System; using System.Collections.Generic; using System.ComponentModel.Composition; using System.IO; @@ -35,12 +36,17 @@ namespace GameRes.Formats.WildBug public override string Tag { get { return "WBP"; } } public override string Description { get { return "Wild Bug's engine resource archive"; } } public override uint Signature { get { return 0x46435241; } } // 'ARCF' - public override bool IsHierarchic { get { return false; } } + public override bool IsHierarchic { get { return true; } } public override bool CanCreate { get { return false; } } public override ArcFile TryOpen (ArcView file) { - if (!file.View.AsciiEqual (0, "ARCFORM3 WBUG ")) + int version; + if (file.View.AsciiEqual (0, "ARCFORM3 WBUG ")) + version = 3; + else if (file.View.AsciiEqual (0, "ARCFORM4 WBUG ")) + version = 4; + else return null; int count = file.View.ReadInt32 (0x10); if (!IsSaneCount (count)) @@ -53,6 +59,14 @@ namespace GameRes.Formats.WildBug if (index_size > file.View.Reserve (index_offset, index_size)) return null; + if (3 == version) + return OpenV3 (file, count, index_offset); + else + return OpenV4 (file, count, index_offset, index_size); + } + + ArcFile OpenV3 (ArcView file, int count, uint index_offset) + { var dir = new List (count); for (int i = 0; i < count; ++i) { @@ -70,5 +84,70 @@ namespace GameRes.Formats.WildBug } return new ArcFile (file, this, dir); } + + ArcFile OpenV4 (ArcView file, int count, uint index_offset, uint index_size) + { + var buffer = new byte[0x400]; + var dir_names = new uint[0x100]; + file.View.Read (0x24, buffer, 0, 0x400); + Buffer.BlockCopy (buffer, 0, dir_names, 0, 0x400); + var res_names = new uint[0x100]; + file.View.Read (0x424, buffer, 0, 0x400); + Buffer.BlockCopy (buffer, 0, res_names, 0, 0x400); + + var dir_table = new Dictionary(); + for (int i = 0; i < 0x100; ++i) + { + if (0 == dir_names[i]) + continue; + uint dir_offset = dir_names[i]; + for (;;) + { + byte hash = file.View.ReadByte (dir_offset); + if (hash != i) + break; + byte name_length = file.View.ReadByte (dir_offset+1); + int dir_id = file.View.ReadUInt16 (dir_offset+2); + file.View.Read (dir_offset + 4, buffer, 0, name_length); + byte checksum = 0; + for (int j = 0; j < name_length; ++j) + checksum += buffer[j]; + if (hash != checksum) + return null; + dir_table[dir_id] = Encodings.cp932.GetString (buffer, 0, name_length).TrimStart ('\\'); + dir_offset += 5u + name_length; + } + } + uint index_end = index_offset + index_size; + var dir = new List (count); + for (int i = 0; i < 0x100; ++i) + { + if (0 == res_names[i]) + continue; + uint res_offset = res_names[i]; + while (res_offset < index_end) + { + byte hash = file.View.ReadByte (res_offset); + if (hash != i) + break; + byte name_length = file.View.ReadByte (res_offset+1); + int dir_id = file.View.ReadUInt16 (res_offset+2); + file.View.Read (res_offset+0x14, buffer, 0, name_length); + byte checksum = 0; + for (int j = 0; j < name_length; ++j) + checksum += buffer[j]; + if (hash != checksum) + return null; + var dir_name = dir_table[dir_id]; + var name = dir_table[dir_id] + Encodings.cp932.GetString (buffer, 0, name_length); + var entry = FormatCatalog.Instance.Create (name); + entry.Offset = file.View.ReadUInt32 (res_offset+4); + entry.Size = file.View.ReadUInt32 (res_offset+8); + res_offset += (0x18u + name_length) & ~3u; + dir.Add (entry); + } + } + return new ArcFile (file, this, dir); + } } }