using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.IO; using System.Runtime.InteropServices; using LibCRIComp; namespace LibCPK { public class CPK { public List FileTable; public Dictionary cpkdata; public UTF utf; Tools tools; UTF files; public CPK(Tools tool) { tools = tool; isUtfEncrypted = false; FileTable = new List(); } public bool ReadCPK(string sPath, Encoding encoding = null) { if (File.Exists(sPath)) { uint Files; ushort Align; EndianReader br = new EndianReader(File.OpenRead(sPath), true); MemoryStream ms; EndianReader utfr; if (tools.ReadCString(br, 4) != "CPK ") { br.Close(); return false; } ReadUTFData(br); CPK_packet = utf_packet; FileEntry CPAK_entry = new FileEntry { FileName = "CPK_HDR", FileOffsetPos = br.BaseStream.Position + 0x10, FileSize = CPK_packet.Length, Encrypted = isUtfEncrypted, FileType = "CPK" }; FileTable.Add(CPAK_entry); ms = new MemoryStream(utf_packet); utfr = new EndianReader(ms, false); utf = new UTF(tools); if (!utf.ReadUTF(utfr, encoding)) { br.Close(); return false; } utfr.Close(); ms.Close(); cpkdata = new Dictionary(); try { for (int i = 0; i < utf.columns.Count; i++) { cpkdata.Add(utf.columns[i].name, utf.rows[0].rows[i].GetValue()); } } catch (Exception ex) { //MessageBox.Show(ex.ToString()); Console.WriteLine(ex.ToString()); } TocOffset = (ulong)GetColumsData2(utf, 0, "TocOffset", 3); long TocOffsetPos = GetColumnPostion(utf, 0, "TocOffset"); EtocOffset = (ulong)GetColumsData2(utf, 0, "EtocOffset", 3); long ETocOffsetPos = GetColumnPostion(utf, 0, "EtocOffset"); ItocOffset = (ulong)GetColumsData2(utf, 0, "ItocOffset", 3); long ITocOffsetPos = GetColumnPostion(utf, 0, "ItocOffset"); GtocOffset = (ulong)GetColumsData2(utf, 0, "GtocOffset", 3); long GTocOffsetPos = GetColumnPostion(utf, 0, "GtocOffset"); ContentOffset = (ulong)GetColumsData2(utf, 0, "ContentOffset", 3); long ContentOffsetPos = GetColumnPostion(utf, 0, "ContentOffset"); FileTable.Add(CreateFileEntry("CONTENT_OFFSET", ContentOffset, typeof(ulong), ContentOffsetPos, "CPK", "CONTENT", false)); Files = (uint)GetColumsData2(utf, 0, "Files", 2); Align = (ushort)GetColumsData2(utf, 0, "Align", 1); if (TocOffset != 0xFFFFFFFFFFFFFFFF) { FileEntry entry = CreateFileEntry("TOC_HDR", TocOffset, typeof(ulong), TocOffsetPos, "CPK", "HDR", false); FileTable.Add(entry); if (!ReadTOC(br, TocOffset, ContentOffset, encoding)) return false; } if (EtocOffset != 0xFFFFFFFFFFFFFFFF) { FileEntry entry = CreateFileEntry("ETOC_HDR", EtocOffset, typeof(ulong), ETocOffsetPos, "CPK", "HDR", false); FileTable.Add(entry); if (!ReadETOC(br, EtocOffset)) return false; } if (ItocOffset != 0xFFFFFFFFFFFFFFFF) { //FileEntry ITOC_entry = new FileEntry { // FileName = "ITOC_HDR", // FileOffset = ItocOffset, FileOffsetType = typeof(ulong), FileOffsetPos = ITocOffsetPos, // TOCName = "CPK", // FileType = "FILE", Encrypted = true, //}; FileEntry entry = CreateFileEntry("ITOC_HDR", ItocOffset, typeof(ulong), ITocOffsetPos, "CPK", "HDR", false); FileTable.Add(entry); if (!ReadITOC(br, ItocOffset, ContentOffset, Align)) return false; } if (GtocOffset != 0xFFFFFFFFFFFFFFFF) { FileEntry entry = CreateFileEntry("GTOC_HDR", GtocOffset, typeof(ulong), GTocOffsetPos, "CPK", "HDR", false); FileTable.Add(entry); if (!ReadGTOC(br, GtocOffset)) return false; } br.Close(); // at this point, we should have all needed file info //utf = null; files = null; return true; } return false; } FileEntry CreateFileEntry(string FileName, ulong FileOffset, Type FileOffsetType, long FileOffsetPos, string TOCName, string FileType, bool encrypted) { FileEntry entry = new FileEntry { FileName = FileName, FileOffset = FileOffset, FileOffsetType = FileOffsetType, FileOffsetPos = FileOffsetPos, TOCName = TOCName, FileType = FileType, Encrypted = encrypted, }; return entry; } public bool ReadTOC(EndianReader br, ulong TocOffset, ulong ContentOffset, Encoding encoding = null) { ulong fTocOffset = TocOffset; ulong add_offset = 0; if (fTocOffset > (ulong)0x800) fTocOffset = (ulong)0x800; if (ContentOffset < 0) add_offset = fTocOffset; else { if (TocOffset < 0) add_offset = ContentOffset; else { if (ContentOffset < fTocOffset) add_offset = ContentOffset; else add_offset = fTocOffset; } } br.BaseStream.Seek((long)TocOffset, SeekOrigin.Begin); if (tools.ReadCString(br, 4) != "TOC ") { br.Close(); return false; } ReadUTFData(br); // Store unencrypted TOC TOC_packet = utf_packet; FileEntry toc_entry = FileTable.Where(x => x.FileName.ToString() == "TOC_HDR").Single(); toc_entry.Encrypted = isUtfEncrypted; toc_entry.FileSize = TOC_packet.Length; MemoryStream ms = new MemoryStream(utf_packet); EndianReader utfr = new EndianReader(ms, false); files = new UTF(tools); if (!files.ReadUTF(utfr, encoding)) { br.Close(); return false; } utfr.Close(); ms.Close(); FileEntry temp; for (int i = 0; i < files.num_rows; i++) { temp = new FileEntry(); temp.TOCName = "TOC"; temp.DirName = GetColumnData(files, i, "DirName"); temp.FileName = GetColumnData(files, i, "FileName"); temp.FileSize = GetColumnData(files, i, "FileSize"); temp.FileSizePos = GetColumnPostion(files, i, "FileSize"); temp.FileSizeType = GetColumnType(files, i, "FileSize"); temp.ExtractSize = GetColumnData(files, i, "ExtractSize"); temp.ExtractSizePos = GetColumnPostion(files, i, "ExtractSize"); temp.ExtractSizeType = GetColumnType(files, i, "ExtractSize"); temp.FileOffset = ((ulong)GetColumnData(files, i, "FileOffset") + (ulong)add_offset); temp.FileOffsetPos = GetColumnPostion(files, i, "FileOffset"); temp.FileOffsetType = GetColumnType(files, i, "FileOffset"); temp.FileType = "FILE"; temp.Offset = add_offset; temp.ID = GetColumnData(files, i, "ID"); temp.UserString = GetColumnData(files, i, "UserString"); FileTable.Add(temp); } files = null; return true; } public void WriteCPK(BinaryWriter cpk) { WritePacket(cpk, "CPK ", 0, CPK_packet); cpk.BaseStream.Seek(0x800 - 6, SeekOrigin.Begin); cpk.Write(Encoding.ASCII.GetBytes("(c)CRI")); if ((TocOffset > 0x800) && TocOffset < 0x8000) { //部分cpk是从0x2000开始TOC,所以 //需要计算 cpk padding cpk.Write(new byte[TocOffset - 0x800]); } } public void WriteTOC(BinaryWriter cpk) { WritePacket(cpk, "TOC ", TocOffset, TOC_packet); } public void WriteITOC(BinaryWriter cpk) { WritePacket(cpk, "ITOC", ItocOffset, ITOC_packet); } public void WriteETOC(BinaryWriter cpk, ulong currentEtocOffset) { WritePacket(cpk, "ETOC", currentEtocOffset, ETOC_packet); } public void WriteGTOC(BinaryWriter cpk) { WritePacket(cpk, "GTOC", GtocOffset, GTOC_packet); } public void WritePacket(BinaryWriter cpk, string ID, ulong position, byte[] packet) { if (position != 0xffffffffffffffff) { cpk.BaseStream.Seek((long)position, SeekOrigin.Begin); byte[] encrypted; if (isUtfEncrypted == true) { encrypted = DecryptUTF(packet); // Yes it says decrypt... } else { encrypted = packet; } cpk.Write(Encoding.ASCII.GetBytes(ID)); cpk.Write((Int32)0xff); cpk.Write((UInt64)encrypted.Length); cpk.Write(encrypted); } } public bool ReadITOC(EndianReader br, ulong startoffset, ulong ContentOffset, ushort Align) { br.BaseStream.Seek((long)startoffset, SeekOrigin.Begin); if (tools.ReadCString(br, 4) != "ITOC") { br.Close(); return false; } ReadUTFData(br); ITOC_packet = utf_packet; FileEntry itoc_entry = FileTable.Where(x => x.FileName.ToString() == "ITOC_HDR").Single(); itoc_entry.Encrypted = isUtfEncrypted; itoc_entry.FileSize = ITOC_packet.Length; MemoryStream ms = new MemoryStream(utf_packet); EndianReader utfr = new EndianReader(ms, false); files = new UTF(tools); if (!files.ReadUTF(utfr)) { br.Close(); return false; } utfr.Close(); ms.Close(); //uint FilesL = (uint)GetColumnData(files, 0, "FilesL"); //uint FilesH = (uint)GetColumnData(files, 0, "FilesH"); byte[] DataL = (byte[])GetColumnData(files, 0, "DataL"); long DataLPos = GetColumnPostion(files, 0, "DataL"); byte[] DataH = (byte[])GetColumnData(files, 0, "DataH"); long DataHPos = GetColumnPostion(files, 0, "DataH"); //MemoryStream ms; //EndianReader ir; UTF utfDataL, utfDataH; Dictionary SizeTable, CSizeTable; Dictionary SizePosTable, CSizePosTable; Dictionary SizeTypeTable, CSizeTypeTable; List IDs = new List(); SizeTable = new Dictionary(); SizePosTable = new Dictionary(); SizeTypeTable = new Dictionary(); CSizeTable = new Dictionary(); CSizePosTable = new Dictionary(); CSizeTypeTable = new Dictionary(); ushort ID, size1; uint size2; long pos; Type type; if (DataL != null) { ms = new MemoryStream(DataL); utfr = new EndianReader(ms, false); utfDataL = new UTF(tools); utfDataL.ReadUTF(utfr); for (int i = 0; i < utfDataL.num_rows; i++) { ID = (ushort)GetColumnData(utfDataL, i, "ID"); size1 = (ushort)GetColumnData(utfDataL, i, "FileSize"); SizeTable.Add((int)ID, (uint)size1); pos = GetColumnPostion(utfDataL, i, "FileSize"); SizePosTable.Add((int)ID, pos + DataLPos); type = GetColumnType(utfDataL, i, "FileSize"); SizeTypeTable.Add((int)ID, type); if ((GetColumnData(utfDataL, i, "ExtractSize")) != null) { size1 = (ushort)GetColumnData(utfDataL, i, "ExtractSize"); CSizeTable.Add((int)ID, (uint)size1); pos = GetColumnPostion(utfDataL, i, "ExtractSize"); CSizePosTable.Add((int)ID, pos + DataLPos); type = GetColumnType(utfDataL, i, "ExtractSize"); CSizeTypeTable.Add((int)ID, type); } IDs.Add(ID); } } if (DataH != null) { ms = new MemoryStream(DataH); utfr = new EndianReader(ms, false); utfDataH = new UTF(tools); utfDataH.ReadUTF(utfr); for (int i = 0; i < utfDataH.num_rows; i++) { ID = (ushort)GetColumnData(utfDataH, i, "ID"); size2 = (uint)GetColumnData(utfDataH, i, "FileSize"); SizeTable.Add(ID, size2); pos = GetColumnPostion(utfDataH, i, "FileSize"); SizePosTable.Add((int)ID, pos + DataHPos); type = GetColumnType(utfDataH, i, "FileSize"); SizeTypeTable.Add((int)ID, type); if ((GetColumnData(utfDataH, i, "ExtractSize")) != null) { size2 = (uint)GetColumnData(utfDataH, i, "ExtractSize"); CSizeTable.Add(ID, size2); pos = GetColumnPostion(utfDataH, i, "ExtractSize"); CSizePosTable.Add((int)ID, pos + DataHPos); type = GetColumnType(utfDataH, i, "ExtractSize"); CSizeTypeTable.Add((int)ID, type); } IDs.Add(ID); } } FileEntry temp; //int id = 0; uint value = 0, value2 = 0; ulong baseoffset = ContentOffset; // Seems ITOC can mix up the IDs..... but they'll alwaysy be in order... IDs = IDs.OrderBy(x => x).ToList(); for (int i = 0; i < IDs.Count; i++) { int id = IDs[i]; temp = new FileEntry(); SizeTable.TryGetValue(id, out value); CSizeTable.TryGetValue(id, out value2); temp.TOCName = "ITOC"; temp.DirName = null; temp.FileName = id.ToString() + ".bin" ; temp.FileSize = value; temp.FileSizePos = SizePosTable[id]; temp.FileSizeType = SizeTypeTable[id]; if (CSizeTable.Count > 0 && CSizeTable.ContainsKey(id)) { temp.ExtractSize = value2; temp.ExtractSizePos = CSizePosTable[id]; temp.ExtractSizeType = CSizeTypeTable[id]; } temp.FileType = "FILE"; temp.FileOffset = baseoffset; temp.ID = id; temp.UserString = null; FileTable.Add(temp); if ((value % Align) > 0) baseoffset += value + (Align - (value % Align)); else baseoffset += value; //id++; } files = null; utfDataL = null; utfDataH = null; ms.Close(); utfr.Close(); return true; } private void ReadUTFData(EndianReader br) { isUtfEncrypted = false; br.IsLittleEndian = true; unk1 = br.ReadInt32(); utf_size = br.ReadInt64(); utf_packet = br.ReadBytes((int)utf_size); if (utf_packet[0] != 0x40 && utf_packet[1] != 0x55 && utf_packet[2] != 0x54 && utf_packet[3] != 0x46) //@UTF { utf_packet = DecryptUTF(utf_packet); isUtfEncrypted = true; } br.IsLittleEndian = false; } public bool ReadGTOC(EndianReader br, ulong startoffset) { br.BaseStream.Seek((long)startoffset, SeekOrigin.Begin); if (tools.ReadCString(br, 4) != "GTOC") { br.Close(); return false; } //br.BaseStream.Seek(0xC, SeekOrigin.Current); //skip header data ReadUTFData(br); GTOC_packet = utf_packet; FileEntry gtoc_entry = FileTable.Where(x => x.FileName.ToString() == "GTOC_HDR").Single(); gtoc_entry.Encrypted = isUtfEncrypted; gtoc_entry.FileSize = GTOC_packet.Length; return true; } public bool ReadETOC(EndianReader br, ulong startoffset) { br.BaseStream.Seek((long)startoffset, SeekOrigin.Begin); if (tools.ReadCString(br, 4) != "ETOC") { br.Close(); return false; } //br.BaseStream.Seek(0xC, SeekOrigin.Current); //skip header data ReadUTFData(br); ETOC_packet = utf_packet; FileEntry etoc_entry = FileTable.Where(x => x.FileName.ToString() == "ETOC_HDR").Single(); etoc_entry.Encrypted = isUtfEncrypted; etoc_entry.FileSize = ETOC_packet.Length; MemoryStream ms = new MemoryStream(utf_packet); EndianReader utfr = new EndianReader(ms, false); files = new UTF(tools); if (!files.ReadUTF(utfr)) { br.Close(); return false; } utfr.Close(); ms.Close(); List fileEntries = FileTable.Where(x => x.FileType == "FILE").ToList(); for (int i = 0; i < fileEntries.Count; i++) { FileTable[i].LocalDir = GetColumnData(files, i, "LocalDir"); FileTable[i].UpdateDateTime = (ulong)GetColumnData(files, i, "UpdateDateTime"); } return true; } public byte[] DecryptUTF(byte[] input) { byte[] result = new byte[input.Length]; int m, t; byte d; m = 0x0000655f; t = 0x00004115; for (int i = 0; i < input.Length; i++) { d = input[i]; d = (byte)(d ^ (byte)(m & 0xff)); result[i] = d; m *= t; } return result; } /*unsafe public int CRICompress(byte* dest, int* destLen, byte* src, int srcLen) { int n = srcLen - 1, m = *destLen - 0x1, T = 0, d = 0; int p, q = 0, i, j, k; byte* odest = dest; for (; n >= 0x100; ) { j = n + 3 + 0x2000; if (j > srcLen) j = srcLen; for (i = n + 3, p = 0; i < j; i++) { for (k = 0; k <= n - 0x100; k++) { if (*(src + n - k) != *(src + i - k)) break; } if (k > p) { q = i - n - 3; p = k; } } if (p < 3) { d = (d << 9) | (*(src + n--)); T += 9; } else { d = (((d << 1) | 1) << 13) | q; T += 14; n -= p; if (p < 6) { d = (d << 2) | (p - 3); T += 2; } else if (p < 13) { d = (((d << 2) | 3) << 3) | (p - 6); T += 5; } else if (p < 44) { d = (((d << 5) | 0x1f) << 5) | (p - 13); T += 10; } else { d = ((d << 10) | 0x3ff); T += 10; p -= 44; for (; ; ) { for (; T >= 8; ) { *(dest + m--) = (byte)((d >> (T - 8)) & 0xff); T -= 8; d = d & ((1 << T) - 1); } if (p < 255) break; d = (d << 8) | 0xff; T += 8; p = p - 0xff; } d = (d << 8) | p; T += 8; } } for (; T >= 8; ) { *(dest + m--) = (byte)((d >> (T - 8)) & 0xff); T -= 8; d = d & ((1 << T) - 1); } } if (T != 0) { *(dest + m--) = (byte)(d << (8 - T)); } *(dest + m--) = 0; *(dest + m) = 0; for (; ; ) { if (((*destLen - m) & 3) == 0) break; *(dest + m--) = 0; } *destLen = *destLen - m; dest += m; int[] l = { 0x4c495243, 0x414c5941, srcLen - 0x100, *destLen }; for (j = 0; j < 4; j++) { for (i = 0; i < 4; i++) { *(odest + i + j * 4) = (byte)(l[j] & 0xff); l[j] >>= 8; } } for (j = 0, odest += 0x10; j < *destLen; j++) { *(odest++) = *(dest + j); } for (j = 0; j < 0x100; j++) { *(odest++) = *(src + j); } *destLen += 0x110; return *destLen; }*/ // unsafe指针在.NET 4.5下似乎很不稳定,有时出现“尝试读取或写入受保护的内存, 这通常指示其它内存已损坏”的错误 // 目前将CRICompress方法移动到了CLR类库中使用。 unsafe public byte[] CompressCRILAYLA(byte[] input) { unsafe { fixed (byte* src = input, dst = new byte[input.Length]) { //Move cricompress to CLR int destLength = (int)input.Length; int result = LibCRIComp.CriCompression.CRIcompress(dst, &destLength, src, input.Length); byte[] arr = new byte[destLength]; Marshal.Copy((IntPtr)dst, arr, 0, destLength); return arr; } } /*unsafe { int destLength = (int)input.Length; fixed (byte* src = input) fixed (byte* dest = new byte[input.Length]) { destLength = CRICompress(dest, &destLength, src, input.Length); byte[] arr = new byte[destLength]; Marshal.Copy((IntPtr)dest, arr, 0, destLength); return arr; } }*/ } public byte[] DecompressCRILAYLA(byte[] input, int USize) { byte[] result;// = new byte[USize]; MemoryStream ms = new MemoryStream(input); EndianReader br = new EndianReader(ms, true); br.BaseStream.Seek(8, SeekOrigin.Begin); // Skip CRILAYLA int uncompressed_size = br.ReadInt32(); int uncompressed_header_offset = br.ReadInt32(); result = new byte[uncompressed_size + 0x100]; // do some error checks here......... // copy uncompressed 0x100 header to start of file Array.Copy(input, uncompressed_header_offset + 0x10, result, 0, 0x100); int input_end = input.Length - 0x100 - 1; int input_offset = input_end; int output_end = 0x100 + uncompressed_size - 1; byte bit_pool = 0; int bits_left = 0, bytes_output = 0; int[] vle_lens = new int[4] { 2, 3, 5, 8 }; while (bytes_output < uncompressed_size) { if (get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 1) > 0) { int backreference_offset = output_end - bytes_output + get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 13) + 3; int backreference_length = 3; int vle_level; for (vle_level = 0; vle_level < vle_lens.Length; vle_level++) { int this_level = get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, vle_lens[vle_level]); backreference_length += this_level; if (this_level != ((1 << vle_lens[vle_level]) - 1)) break; } if (vle_level == vle_lens.Length) { int this_level; do { this_level = get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 8); backreference_length += this_level; } while (this_level == 255); } for (int i = 0; i < backreference_length; i++) { result[output_end - bytes_output] = result[backreference_offset--]; bytes_output++; } } else { // verbatim byte result[output_end - bytes_output] = (byte)get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 8); bytes_output++; } } br.Close(); ms.Close(); return result; } public byte[] DecompressLegacyCRI(byte[] input, int USize) { byte[] result;// = new byte[USize]; MemoryStream ms = new MemoryStream(input); EndianReader br = new EndianReader(ms, true); br.BaseStream.Seek(8, SeekOrigin.Begin); // Skip CRILAYLA int uncompressed_size = br.ReadInt32(); int uncompressed_header_offset = br.ReadInt32(); result = new byte[uncompressed_size + 0x100]; // do some error checks here......... // copy uncompressed 0x100 header to start of file Array.Copy(input, uncompressed_header_offset + 0x10, result, 0, 0x100); int input_end = input.Length - 0x100 - 1; int input_offset = input_end; int output_end = 0x100 + uncompressed_size - 1; byte bit_pool = 0; int bits_left = 0, bytes_output = 0; int[] vle_lens = new int[4] { 2, 3, 5, 8 }; while (bytes_output < uncompressed_size) { if (get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 1) > 0) { int backreference_offset = output_end - bytes_output + get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 13) + 3; int backreference_length = 3; int vle_level; for (vle_level = 0; vle_level < vle_lens.Length; vle_level++) { int this_level = get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, vle_lens[vle_level]); backreference_length += this_level; if (this_level != ((1 << vle_lens[vle_level]) - 1)) break; } if (vle_level == vle_lens.Length) { int this_level; do { this_level = get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 8); backreference_length += this_level; } while (this_level == 255); } for (int i = 0; i < backreference_length; i++) { result[output_end - bytes_output] = result[backreference_offset--]; bytes_output++; } } else { // verbatim byte result[output_end - bytes_output] = (byte)get_next_bits(input, ref input_offset, ref bit_pool, ref bits_left, 8); bytes_output++; } } br.Close(); ms.Close(); return result; } private ushort get_next_bits(byte[] input, ref int offset_p, ref byte bit_pool_p, ref int bits_left_p, int bit_count) { ushort out_bits = 0; int num_bits_produced = 0; int bits_this_round; while (num_bits_produced < bit_count) { if (bits_left_p == 0) { bit_pool_p = input[offset_p]; bits_left_p = 8; offset_p--; } if (bits_left_p > (bit_count - num_bits_produced)) bits_this_round = bit_count - num_bits_produced; else bits_this_round = bits_left_p; out_bits <<= bits_this_round; out_bits |= (ushort)((ushort)(bit_pool_p >> (bits_left_p - bits_this_round)) & ((1 << bits_this_round) - 1)); bits_left_p -= bits_this_round; num_bits_produced += bits_this_round; } return out_bits; } public object GetColumsData2(UTF utf, int row, string Name, int type) { object Temp = GetColumnData(utf, row, Name); if (Temp == null) { switch (type) { case 0: // byte return (byte)0xFF; case 1: // short return (ushort)0xFFFF; case 2: // int return 0xFFFFFFFF; case 3: // long return 0xFFFFFFFFFFFFFFFF; } } if (Temp is ulong) { return (Temp == null) ? 0xFFFFFFFFFFFFFFFF : (ulong)Temp; } if (Temp is uint) { return (Temp == null) ? 0xFFFFFFFF : (uint)Temp; } if (Temp is ushort) { return (Temp == null) ? (ushort)0xFFFF : (ushort)Temp; } return 0; } public object GetColumnData(UTF utf, int row, string Name) { object result = null; try { for (int i = 0; i < utf.num_columns; i++) { if (utf.columns[i].name == Name) { result = utf.rows[row].rows[i].GetValue(); break; } } } catch (Exception ex) { Console.WriteLine(ex.ToString()); return null; } return result; } public long GetColumnPostion(UTF utf, int row, string Name) { long result = -1; try { for (int i = 0; i < utf.num_columns; i++) { if (utf.columns[i].name == Name) { result = utf.rows[row].rows[i].position; break; } } } catch (Exception ex) { //MessageBox.Show(ex.ToString()); Console.WriteLine(ex.ToString()); return -1; } return result; } public Type GetColumnType(UTF utf, int row, string Name) { Type result = null; try { for (int i = 0; i < utf.num_columns; i++) { if (utf.columns[i].name == Name) { result = utf.rows[row].rows[i].GetType(); break; } } } catch (Exception ex) { //MessageBox.Show(ex.ToString()); Console.WriteLine(ex.ToString()); return null; } return result; } public void UpdateFileEntry(FileEntry fileEntry) { if (fileEntry.FileType == "FILE" || fileEntry.FileType == "HDR") { byte[] updateMe = null; switch (fileEntry.TOCName) { case "CPK": updateMe = CPK_packet; break; case "TOC": updateMe = TOC_packet; break; case "ITOC": updateMe = ITOC_packet; break; case "ETOC": updateMe = ETOC_packet; break; case "GTOC": updateMe = GTOC_packet; break; default: throw new Exception("I need to implement this TOC!"); } //Update ExtractSize if (fileEntry.ExtractSizePos > 0) UpdateValue(ref updateMe, fileEntry.ExtractSize, fileEntry.ExtractSizePos, fileEntry.ExtractSizeType); //Update FileSize if (fileEntry.FileSizePos > 0) UpdateValue(ref updateMe, fileEntry.FileSize, fileEntry.FileSizePos, fileEntry.FileSizeType); //Update FileOffset if (fileEntry.FileOffsetPos > 0) if (fileEntry.TOCName == "TOC") { UpdateValue(ref updateMe, fileEntry.FileOffset - (ulong)TocOffset, fileEntry.FileOffsetPos, fileEntry.FileOffsetType); } else { UpdateValue(ref updateMe, fileEntry.FileOffset, fileEntry.FileOffsetPos, fileEntry.FileOffsetType); } switch (fileEntry.TOCName) { case "CPK": CPK_packet = updateMe; break; case "TOC": TOC_packet = updateMe; break; case "ITOC": ITOC_packet = updateMe; break; case "ETOC": updateMe = ETOC_packet; break; case "GTOC": updateMe = GTOC_packet; break; default: throw new Exception("I need to implement this TOC!"); } } } public void UpdateValue(ref byte[] packet, object value, long pos, Type type) { MemoryStream temp = new MemoryStream(); temp.Write(packet, 0, packet.Length); EndianWriter toc = new EndianWriter(temp, false); toc.Seek((int)pos, SeekOrigin.Begin); value = Convert.ChangeType(value, type); if (type == typeof(Byte)) { toc.Write((Byte)value); } else if (type == typeof(UInt16)) { toc.Write((UInt16)value); } else if (type == typeof(UInt32)) { toc.Write((UInt32)value); } else if (type == typeof(UInt64)) { toc.Write((UInt64)value); } else if (type == typeof(Single)) { toc.Write((Single)value); } else { throw new Exception("Not supported type!"); } toc.Close(); MemoryStream myStream = (MemoryStream)toc.BaseStream; packet = myStream.ToArray(); } public bool isUtfEncrypted { get; set; } public int unk1 { get; set; } public long utf_size { get; set; } public byte[] utf_packet { get; set; } public byte[] CPK_packet { get; set; } public byte[] TOC_packet { get; set; } public byte[] ITOC_packet { get; set; } public byte[] ETOC_packet { get; set; } public byte[] GTOC_packet { get; set; } public ulong TocOffset, EtocOffset, ItocOffset, GtocOffset, ContentOffset; } public class UTF { public enum COLUMN_FLAGS : int { STORAGE_MASK = 0xf0, STORAGE_NONE = 0x00, STORAGE_ZERO = 0x10, STORAGE_CONSTANT = 0x30, STORAGE_PERROW = 0x50, TYPE_MASK = 0x0f, TYPE_DATA = 0x0b, TYPE_STRING = 0x0a, TYPE_FLOAT = 0x08, TYPE_8BYTE2 = 0x07, TYPE_8BYTE = 0x06, TYPE_4BYTE2 = 0x05, TYPE_4BYTE = 0x04, TYPE_2BYTE2 = 0x03, TYPE_2BYTE = 0x02, TYPE_1BYTE2 = 0x01, TYPE_1BYTE = 0x00, } public List columns; public List rows; Tools tools; public UTF(Tools tool) { tools = tool; } public bool ReadUTF(EndianReader br ,Encoding encoding = null) { long offset = br.BaseStream.Position; if (tools.ReadCString(br, 4) != "@UTF") { return false; } table_size = br.ReadInt32(); rows_offset = br.ReadInt32(); strings_offset = br.ReadInt32(); data_offset = br.ReadInt32(); // CPK Header & UTF Header are ignored, so add 8 to each offset rows_offset += (offset + 8); strings_offset += (offset + 8); data_offset += (offset + 8); table_name = br.ReadInt32(); num_columns = br.ReadInt16(); row_length = br.ReadInt16(); num_rows = br.ReadInt32(); //read Columns columns = new List(); COLUMN column; for (int i = 0; i < num_columns; i++) { column = new COLUMN(); column.flags = br.ReadByte(); if (column.flags == 0) { br.BaseStream.Seek(3, SeekOrigin.Current); column.flags = br.ReadByte(); } column.name = tools.ReadCString(br, -1, (long)(br.ReadInt32() + strings_offset), encoding); columns.Add(column); } //read Rows rows = new List(); ROWS current_entry; ROW current_row; int storage_flag; for (int j = 0; j < num_rows; j++) { br.BaseStream.Seek(rows_offset + (j * row_length), SeekOrigin.Begin); current_entry = new ROWS(); for (int i = 0; i < num_columns; i++) { current_row = new ROW(); storage_flag = (columns[i].flags & (int)COLUMN_FLAGS.STORAGE_MASK); if (storage_flag == (int)COLUMN_FLAGS.STORAGE_NONE) // 0x00 { current_entry.rows.Add(current_row); continue; } if (storage_flag == (int)COLUMN_FLAGS.STORAGE_ZERO) // 0x10 { current_entry.rows.Add(current_row); continue; } if (storage_flag == (int)COLUMN_FLAGS.STORAGE_CONSTANT) // 0x30 { current_entry.rows.Add(current_row); continue; } // 0x50 current_row.type = columns[i].flags & (int)COLUMN_FLAGS.TYPE_MASK; current_row.position = br.BaseStream.Position; switch (current_row.type) { case 0: case 1: current_row.uint8 = br.ReadByte(); break; case 2: case 3: current_row.uint16 = br.ReadUInt16(); break; case 4: case 5: current_row.uint32 = br.ReadUInt32(); break; case 6: case 7: current_row.uint64 = br.ReadUInt64(); break; case 8: current_row.ufloat = br.ReadSingle(); break; case 0xA: current_row.str = tools.ReadCString(br, -1, br.ReadInt32() + strings_offset , encoding); break; case 0xB: long position = br.ReadInt32() + data_offset; current_row.position = position; current_row.data = tools.GetData(br, position, br.ReadInt32()); break; default: throw new NotImplementedException(); } current_entry.rows.Add(current_row); } rows.Add(current_entry); } return true; } public int table_size { get; set; } public long rows_offset { get; set; } public long strings_offset { get; set; } public long data_offset { get; set; } public int table_name { get; set; } public short num_columns { get; set; } public short row_length { get; set; } public int num_rows { get; set; } } public class COLUMN { public COLUMN() { } public byte flags { get; set; } public string name { get; set; } } public class ROWS { public List rows; public ROWS() { rows = new List(); } } public class ROW { public ROW() { type = -1; } public int type { get; set; } public object GetValue() { object result = -1; switch (this.type) { case 0: case 1: return this.uint8; case 2: case 3: return this.uint16; case 4: case 5: return this.uint32; case 6: case 7: return this.uint64; case 8: return this.ufloat; case 0xA: return this.str; case 0xB: return this.data; default: return null; } } public new Type GetType() { object result = -1; switch (this.type) { case 0: case 1: return this.uint8.GetType(); case 2: case 3: return this.uint16.GetType(); case 4: case 5: return this.uint32.GetType(); case 6: case 7: return this.uint64.GetType(); case 8: return this.ufloat.GetType(); case 0xA: return this.str.GetType(); case 0xB: return this.data.GetType(); default: return null; } } //column based datatypes public byte uint8 { get; set; } public ushort uint16 { get; set; } public uint uint32 { get; set; } public ulong uint64 { get; set; } public float ufloat { get; set; } public string str { get; set; } public byte[] data { get; set; } public long position { get; set; } } public class FileEntry { public FileEntry() { DirName = null; FileName = null; FileSize = null; ExtractSize = null; ID = null; UserString = null; LocalDir = null; FileOffset = 0; UpdateDateTime = 0; } public object DirName { get; set; } // string public object FileName { get; set; } // string public object FileSize { get; set; } public long FileSizePos { get; set; } public Type FileSizeType { get; set; } public object ExtractSize { get; set; } // int public long ExtractSizePos { get; set; } public Type ExtractSizeType { get; set; } public ulong FileOffset { get; set; } public long FileOffsetPos { get; set; } public Type FileOffsetType { get; set; } public ulong Offset { get; set; } public object ID { get; set; } // int public object UserString { get; set; } // string public ulong UpdateDateTime { get; set; } public object LocalDir { get; set; } // string public string TOCName { get; set; } public bool Encrypted { get; set; } public string FileType { get; set; } } }