private static void UnpackRegulationFile(Options options) { using (FileStream inputStream = new FileStream(options.InputPath, FileMode.Open, FileAccess.Read)) { EncFile encryptedFile = EncFile.ReadEncFile(inputStream, DecryptionKeys.RegulationFileKey); DcxFile compressedRegulationFile = DcxFile.Read(encryptedFile.Data); UnpackBndFile(new MemoryStream(compressedRegulationFile.Decompress()), options.OutputPath); } }
private static void UnpackBdtFile(Options options) { FileNameDictionary dictionary = FileNameDictionary.OpenFromFile(options.InputGameVersion); string fileNameWithoutExtension = Path.GetFileName(options.InputPath).Replace("Ebl.bdt", "").Replace(".bdt", ""); string archiveName = fileNameWithoutExtension.ToLower(); using (Bdt5FileStream bdtStream = Bdt5FileStream.OpenFile(options.InputPath, FileMode.Open, FileAccess.Read)) { Bhd5File bhdFile = Bhd5File.Read( inputStream: DecryptBhdFile( filePath: Path.ChangeExtension(options.InputPath, "bhd"), version: options.InputGameVersion), version: options.InputGameVersion ); foreach (var bucket in bhdFile.GetBuckets()) { foreach (var entry in bucket.GetEntries()) { MemoryStream data; if (entry.FileSize == 0) { long fileSize; if (!TryReadFileSize(entry, bdtStream, out fileSize)) { Console.WriteLine($"Unable to determine the length of file '{entry.FileNameHash:D10}'"); continue; } entry.FileSize = fileSize; } if (entry.IsEncrypted) { data = bdtStream.Read(entry.FileOffset, entry.PaddedFileSize); CryptographyUtility.DecryptAesEcb(data, entry.AesKey.Key, entry.AesKey.Ranges); data.Position = 0; data.SetLength(entry.FileSize); } else { data = bdtStream.Read(entry.FileOffset, entry.FileSize); } string fileName; string dataExtension = GetDataExtension(data); bool fileNameFound = dictionary.TryGetFileName(entry.FileNameHash, archiveName, out fileName); if (!fileNameFound) { fileNameFound = dictionary.TryGetFileName(entry.FileNameHash, archiveName, dataExtension, out fileName); } string extension; if (fileNameFound) { extension = Path.GetExtension(fileName); if (dataExtension == ".dcx" && extension != ".dcx") { extension = ".dcx"; fileName += ".dcx"; } } else { extension = dataExtension; fileName = $"{entry.FileNameHash:D10}_{fileNameWithoutExtension}{extension}"; } if (extension == ".enc") { byte[] decryptionKey; if (DecryptionKeys.TryGetAesFileKey(Path.GetFileName(fileName), out decryptionKey)) { EncFile encFile = EncFile.ReadEncFile(data, decryptionKey); data = encFile.Data; fileName = Path.Combine(Path.GetDirectoryName(fileName), Path.GetFileNameWithoutExtension(fileName)); extension = Path.GetExtension(fileName); } else { Debug.WriteLine($"No decryption key for file \'{fileName}\' found."); } } if (extension == ".dcx") { DcxFile dcxFile = DcxFile.Read(data); data = new MemoryStream(dcxFile.Decompress()); fileName = Path.Combine(Path.GetDirectoryName(fileName), Path.GetFileNameWithoutExtension(fileName)); if (fileNameFound) { extension = Path.GetExtension(fileName); } else { extension = GetDataExtension(data); fileName += extension; } } if (extension == ".bnd") { UnpackBnd3File(data, options.OutputPath); continue; // don't actually unpack bnds } Debug.WriteLine( "{0}\t{1}\t{2}\t{3}\t{4}\t{5}\t{6}\t{7}\t{8}", fileNameWithoutExtension, fileName, extension, entry.FileNameHash, entry.FileOffset, entry.FileSize, entry.PaddedFileSize, entry.IsEncrypted, fileNameFound); string newFileNamePath = Path.Combine(options.OutputPath, fileName); Directory.CreateDirectory(Path.GetDirectoryName(newFileNamePath)); File.WriteAllBytes(newFileNamePath, data.ToArray()); } } } }