public override void Read(Stream source) { using (CriTableReader reader = CriTableReader.Create(source)) { switch (reader.TableName) { // ADX wrapper case "AAX": Mode = CriAaxArchiveMode.Adx; break; // DSP wrapper case "ADPCM_WII": Mode = CriAaxArchiveMode.Dsp; break; // WAV wrapper case "SWLPCM": Mode = CriAaxArchiveMode.Wav; break; // Unknown default: throw new InvalidDataException($"Unknown AAX type '{reader.TableName}'. Please report the error with the file(s)."); } if (Mode == CriAaxArchiveMode.Dsp) { // TODO throw new NotImplementedException("DSP files aren't supported yet."); } else { while (reader.Read()) { CriAaxEntry entry = new CriAaxEntry(); entry.Flag = (CriAaxEntryFlag)reader.GetByte("lpflg"); entry.Position = reader.GetPosition("data"); entry.Length = reader.GetLength("data"); entries.Add(entry); } } } }
static void Main(string[] args) { Settings.Default.Save(); if (args.Length < 1) { Console.WriteLine(Resources.Description); Console.ReadLine(); return; } #if !DEBUG try { #endif if (args[0].EndsWith(".csb", StringComparison.OrdinalIgnoreCase)) { var extractor = new DataExtractor(); extractor.ProgressChanged += OnProgressChanged; extractor.BufferSize = Settings.Default.BufferSize; extractor.EnableThreading = Settings.Default.EnableThreading; extractor.MaxThreads = Settings.Default.MaxThreads; string baseDirectory = Path.GetDirectoryName(args[0]); string outputDirectoryName = Path.Combine(baseDirectory, Path.GetFileNameWithoutExtension(args[0])); CriCpkArchive cpkArchive = null; string cpkPath = outputDirectoryName + ".cpk"; bool found = File.Exists(cpkPath); //This should fix "File not found" error in case-sensitive file systems. //Add new extensions when necessary. foreach (string extension in new string[] { "cpk", "CPK" }) { if (found) { break; } cpkPath = outputDirectoryName + "." + extension; found = File.Exists(cpkPath); } using (CriTableReader reader = CriTableReader.Create(args[0])) { while (reader.Read()) { if (reader.GetString("name") == "SOUND_ELEMENT") { long tablePosition = reader.GetPosition("utf"); using (CriTableReader sdlReader = CriTableReader.Create(reader.GetSubStream("utf"))) { while (sdlReader.Read()) { if (sdlReader.GetByte("fmt") != 0) { throw new Exception("The given CSB file contains an audio file which is not an ADX. Only CSB files with ADXs are supported."); } bool streaming = sdlReader.GetBoolean("stmflg"); if (streaming && !found) { throw new Exception("Cannot find the external .CPK file for this .CSB file. Please ensure that the external .CPK file is stored in the directory where the .CPK file is."); } else if (streaming && found && cpkArchive == null) { cpkArchive = new CriCpkArchive(); cpkArchive.Load(cpkPath, Settings.Default.BufferSize); } string sdlName = sdlReader.GetString("name"); DirectoryInfo destinationPath = new DirectoryInfo(Path.Combine(outputDirectoryName, sdlName)); destinationPath.Create(); CriAaxArchive aaxArchive = new CriAaxArchive(); if (streaming) { CriCpkEntry cpkEntry = cpkArchive.GetByPath(sdlName); if (cpkEntry != null) { using (Stream cpkSource = File.OpenRead(cpkPath)) using (Stream aaxSource = cpkEntry.Open(cpkSource)) { aaxArchive.Read(aaxSource); foreach (CriAaxEntry entry in aaxArchive) { extractor.Add(cpkPath, Path.Combine(destinationPath.FullName, entry.Flag == CriAaxEntryFlag.Intro ? "Intro.adx" : "Loop.adx"), cpkEntry.Position + entry.Position, entry.Length); } } } } else { long aaxPosition = sdlReader.GetPosition("data"); using (Stream aaxSource = sdlReader.GetSubStream("data")) { aaxArchive.Read(aaxSource); foreach (CriAaxEntry entry in aaxArchive) { extractor.Add(args[0], Path.Combine(destinationPath.FullName, entry.Flag == CriAaxEntryFlag.Intro ? "Intro.adx" : "Loop.adx"), tablePosition + aaxPosition + entry.Position, entry.Length); } } } } } break; } } } extractor.Run(); } else if (File.GetAttributes(args[0]).HasFlag(FileAttributes.Directory)) { string baseDirectory = Path.GetDirectoryName(args[0]); string csbPath = args[0] + ".csb"; foreach (string extension in new string[] { "csb", "CSB" }) { if (File.Exists(csbPath)) { break; } csbPath = args[0] + "." + extension; } if (!File.Exists(csbPath)) { throw new Exception("Cannot find the .CSB file for this directory. Please ensure that the .CSB file is stored in the directory where this directory is."); } CriCpkArchive cpkArchive = new CriCpkArchive(); cpkArchive.ProgressChanged += OnProgressChanged; CriTable csbFile = new CriTable(); csbFile.Load(csbPath, Settings.Default.BufferSize); CriRow soundElementRow = csbFile.Rows.First(row => (string)row["name"] == "SOUND_ELEMENT"); CriTable soundElementTable = new CriTable(); soundElementTable.Load((byte[])soundElementRow["utf"]); List <FileInfo> junks = new List <FileInfo>(); foreach (CriRow sdlRow in soundElementTable.Rows) { string sdlName = (string)sdlRow["name"]; DirectoryInfo sdlDirectory = new DirectoryInfo(Path.Combine(args[0], sdlName)); if (!sdlDirectory.Exists) { throw new Exception($"Cannot find sound element directory for replacement.\nPath attempt: {sdlDirectory.FullName}"); } bool streaming = (byte)sdlRow["stmflg"] != 0; uint sampleRate = (uint)sdlRow["sfreq"]; byte numberChannels = (byte)sdlRow["nch"]; CriAaxArchive aaxArchive = new CriAaxArchive(); foreach (FileInfo file in sdlDirectory.GetFiles("*.adx")) { CriAaxEntry entry = new CriAaxEntry(); if (file.Name.ToLower(CultureInfo.GetCultureInfo("en-US")) == "intro.adx") { entry.Flag = CriAaxEntryFlag.Intro; entry.FilePath = file; aaxArchive.Add(entry); ReadAdx(file, out sampleRate, out numberChannels); } else if (file.Name.ToLower(CultureInfo.GetCultureInfo("en-US")) == "loop.adx") { entry.Flag = CriAaxEntryFlag.Loop; entry.FilePath = file; aaxArchive.Add(entry); ReadAdx(file, out sampleRate, out numberChannels); } } if (streaming) { CriCpkEntry entry = new CriCpkEntry(); entry.Name = Path.GetFileName(sdlName); entry.DirectoryName = Path.GetDirectoryName(sdlName); entry.Id = (uint)cpkArchive.Count; entry.FilePath = new FileInfo(Path.GetTempFileName()); junks.Add(entry.FilePath); cpkArchive.Add(entry); aaxArchive.Save(entry.FilePath.FullName, Settings.Default.BufferSize); } else { sdlRow["data"] = aaxArchive.Save(); } sdlRow["sfreq"] = sampleRate; sdlRow["nch"] = numberChannels; } soundElementTable.WriterSettings = CriTableWriterSettings.AdxSettings; soundElementRow["utf"] = soundElementTable.Save(); csbFile.WriterSettings = CriTableWriterSettings.AdxSettings; csbFile.Save(csbPath, Settings.Default.BufferSize); if (cpkArchive.Count > 0) { string cpkPath = args[0] + ".cpk"; foreach (string extension in new string[] { "cpk", "CPK" }) { if (File.Exists(args[0] + "." + extension)) { cpkPath = args[0] + "." + extension; break; } } cpkArchive.Save(cpkPath, Settings.Default.BufferSize); } foreach (FileInfo junk in junks) { junk.Delete(); } } #if !DEBUG } catch (Exception exception) { MessageBox.Show($"{exception.Message}", "CSB Editor", MessageBoxButtons.OK, MessageBoxIcon.Error); } #endif }
// All taken from the code for CsbEditor: https://github.com/blueskythlikesclouds/SonicAudioTools/blob/master/Source/CsbEditor/Program.cs public static async Task CSBUnpack(string csbFile) { Encoding.RegisterProvider(CodePagesEncodingProvider.Instance); DataExtractor?extractor = new() { EnableThreading = false }; string baseDirectory = Path.GetDirectoryName(csbFile); string outputDirectoryName = Path.Combine(baseDirectory, Path.GetFileNameWithoutExtension(csbFile)); CriCpkArchive cpkArchive = null; string cpkPath = outputDirectoryName + ".cpk"; bool found = File.Exists(cpkPath); //This should fix "File not found" error in case-sensitive file systems. //Add new extensions when necessary. foreach (string extension in new string[] { "cpk", "CPK" }) { if (found) { break; } cpkPath = outputDirectoryName + "." + extension; found = File.Exists(cpkPath); } using (CriTableReader reader = CriTableReader.Create(csbFile)) { while (reader.Read()) { if (reader.GetString("name") == "SOUND_ELEMENT") { long tablePosition = reader.GetPosition("utf"); using (CriTableReader sdlReader = CriTableReader.Create(reader.GetSubStream("utf"))) { while (sdlReader.Read()) { if (sdlReader.GetByte("fmt") != 0) { throw new Exception("The given CSB file contains an audio file which is not an ADX. Only CSB files with ADXs are supported."); } bool streaming = sdlReader.GetBoolean("stmflg"); if (streaming && !found) { throw new Exception("Cannot find the external .CPK file for this .CSB file. Please ensure that the external .CPK file is stored in the directory where the .CPK file is."); } else if (streaming && found && cpkArchive == null) { cpkArchive = new CriCpkArchive(); cpkArchive.Load(cpkPath, 4096); } string sdlName = sdlReader.GetString("name"); DirectoryInfo destinationPath = new DirectoryInfo(Path.Combine(outputDirectoryName, sdlName)); destinationPath.Create(); CriAaxArchive aaxArchive = new CriAaxArchive(); if (streaming) { CriCpkEntry cpkEntry = cpkArchive.GetByPath(sdlName); if (cpkEntry != null) { using (Stream cpkSource = File.OpenRead(cpkPath)) using (Stream aaxSource = cpkEntry.Open(cpkSource)) { aaxArchive.Read(aaxSource); foreach (CriAaxEntry entry in aaxArchive) { extractor.Add(cpkPath, Path.Combine(destinationPath.FullName, entry.Flag == CriAaxEntryFlag.Intro ? "Intro.adx" : "Loop.adx"), cpkEntry.Position + entry.Position, entry.Length); } } } } else { long aaxPosition = sdlReader.GetPosition("data"); using (Stream aaxSource = sdlReader.GetSubStream("data")) { aaxArchive.Read(aaxSource); foreach (CriAaxEntry entry in aaxArchive) { extractor.Add(csbFile, Path.Combine(destinationPath.FullName, entry.Flag == CriAaxEntryFlag.Intro ? "Intro.adx" : "Loop.adx"), tablePosition + aaxPosition + entry.Position, entry.Length); } } } } } break; } } } extractor.Run(); }
static void Main(string[] args) { if (!File.Exists(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile)) { Settings.Default.Reset(); Settings.Default.Save(); } if (args.Length < 1) { Console.WriteLine(Resources.Description); Console.ReadLine(); return; } #if !DEBUG try { #endif if (args[0].EndsWith(".acb", StringComparison.OrdinalIgnoreCase)) { var extractor = new DataExtractor(); extractor.ProgressChanged += OnProgressChanged; extractor.BufferSize = Settings.Default.BufferSize; extractor.EnableThreading = Settings.Default.EnableThreading; extractor.MaxThreads = Settings.Default.MaxThreads; string baseDirectory = Path.GetDirectoryName(args[0]); string outputDirectoryPath = Path.ChangeExtension(args[0], null); string extAfs2ArchivePath = string.Empty; Directory.CreateDirectory(outputDirectoryPath); using (CriTableReader acbReader = CriTableReader.Create(args[0])) { acbReader.Read(); CriAfs2Archive afs2Archive = new CriAfs2Archive(); CriAfs2Archive extAfs2Archive = new CriAfs2Archive(); CriCpkArchive cpkArchive = new CriCpkArchive(); CriCpkArchive extCpkArchive = null; extAfs2ArchivePath = outputDirectoryPath + ".awb"; bool found = File.Exists(extAfs2ArchivePath); if (!found) { extAfs2ArchivePath = outputDirectoryPath + "_streamfiles.awb"; found = File.Exists(extAfs2ArchivePath); } if (!found) { extAfs2ArchivePath = outputDirectoryPath + "_STR.awb"; found = File.Exists(extAfs2ArchivePath); } bool cpkMode = true; long awbPosition = acbReader.GetPosition("AwbFile"); if (acbReader.GetLength("AwbFile") > 0) { using (SubStream afs2Stream = acbReader.GetSubStream("AwbFile")) { cpkMode = !CheckIfAfs2(afs2Stream); if (cpkMode) { cpkArchive.Read(afs2Stream); } else { afs2Archive.Read(afs2Stream); } } } if (acbReader.GetLength("StreamAwbAfs2Header") > 0) { cpkMode = false; using (SubStream extAfs2Stream = acbReader.GetSubStream("StreamAwbAfs2Header")) { bool utfMode = DataStream.ReadCString(extAfs2Stream, 4) == "@UTF"; extAfs2Stream.Seek(0, SeekOrigin.Begin); if (utfMode) { using (CriTableReader utfAfs2HeaderReader = CriTableReader.Create(extAfs2Stream)) { utfAfs2HeaderReader.Read(); using (SubStream extAfs2HeaderStream = utfAfs2HeaderReader.GetSubStream("Header")) { extAfs2Archive.Read(extAfs2HeaderStream); } } } else { extAfs2Archive.Read(extAfs2Stream); } } if (!found) { throw new FileNotFoundException("Cannot find the external .AWB file for this .ACB file. Please ensure that the external .AWB file is stored in the directory where the .ACB file is."); } } using (SubStream waveformTableStream = acbReader.GetSubStream("WaveformTable")) using (CriTableReader waveformReader = CriTableReader.Create(waveformTableStream)) { while (waveformReader.Read()) { byte encodeType = waveformReader.GetByte("EncodeType"); bool streaming = waveformReader.GetBoolean("Streaming"); ushort id = waveformReader.ContainsField("MemoryAwbId") ? streaming ? waveformReader.GetUInt16("StreamAwbId") : waveformReader.GetUInt16("MemoryAwbId") : waveformReader.GetUInt16("Id"); string outputName = id.ToString("D5"); if (streaming) { outputName += "_streaming"; } outputName += GetExtension(encodeType); outputName = Path.Combine(outputDirectoryPath, outputName); if (streaming) { if (!found) { throw new Exception("Cannot find the external .AWB file for this .ACB file. Please ensure that the external .AWB file is stored in the directory where the .ACB file is."); } else if (extCpkArchive == null && cpkMode) { extCpkArchive = new CriCpkArchive(); extCpkArchive.Load(extAfs2ArchivePath, Settings.Default.BufferSize); } EntryBase afs2Entry = null; if (cpkMode) { afs2Entry = extCpkArchive.GetById(id); } else { afs2Entry = extAfs2Archive.GetById(id); } extractor.Add(extAfs2ArchivePath, outputName, afs2Entry.Position, afs2Entry.Length); } else { EntryBase afs2Entry = null; if (cpkMode) { afs2Entry = cpkArchive.GetById(id); } else { afs2Entry = afs2Archive.GetById(id); } extractor.Add(args[0], outputName, awbPosition + afs2Entry.Position, afs2Entry.Length); } } } } extractor.Run(); } else if (File.GetAttributes(args[0]).HasFlag(FileAttributes.Directory)) { string baseDirectory = Path.GetDirectoryName(args[0]); string acbPath = args[0] + ".acb"; string awbPath = args[0] + "_streamfiles.awb"; bool found = File.Exists(awbPath); if (!found) { awbPath = args[0] + "_STR.awb"; found = File.Exists(awbPath); } if (!found) { awbPath = args[0] + ".awb"; } if (!File.Exists(acbPath)) { throw new FileNotFoundException("Cannot find the .ACB file for this directory. Please ensure that the .ACB file is stored in the directory where this directory is."); } CriTable acbFile = new CriTable(); acbFile.Load(acbPath, Settings.Default.BufferSize); CriAfs2Archive afs2Archive = new CriAfs2Archive(); CriAfs2Archive extAfs2Archive = new CriAfs2Archive(); CriCpkArchive cpkArchive = new CriCpkArchive(); CriCpkArchive extCpkArchive = new CriCpkArchive(); cpkArchive.Mode = extCpkArchive.Mode = CriCpkMode.Id; afs2Archive.ProgressChanged += OnProgressChanged; extAfs2Archive.ProgressChanged += OnProgressChanged; cpkArchive.ProgressChanged += OnProgressChanged; extCpkArchive.ProgressChanged += OnProgressChanged; bool cpkMode = true; byte[] awbFile = (byte[])acbFile.Rows[0]["AwbFile"]; byte[] streamAwbAfs2Header = (byte[])acbFile.Rows[0]["StreamAwbAfs2Header"]; cpkMode = !(awbFile != null && awbFile.Length >= 4 && Encoding.ASCII.GetString(awbFile, 0, 4) == "AFS2") && (streamAwbAfs2Header == null || streamAwbAfs2Header.Length == 0); using (CriTableReader reader = CriTableReader.Create((byte[])acbFile.Rows[0]["WaveformTable"])) { while (reader.Read()) { byte encodeType = reader.GetByte("EncodeType"); bool streaming = reader.GetBoolean("Streaming"); ushort id = reader.ContainsField("MemoryAwbId") ? streaming ? reader.GetUInt16("StreamAwbId") : reader.GetUInt16("MemoryAwbId") : reader.GetUInt16("Id"); string inputName = id.ToString("D5"); if (streaming) { inputName += "_streaming"; } inputName += GetExtension(encodeType); inputName = Path.Combine(args[0], inputName); if (!File.Exists(inputName)) { throw new FileNotFoundException($"Cannot find audio file with id {id} for replacement.\nPath attempt: {inputName}"); } if (cpkMode) { CriCpkEntry entry = new CriCpkEntry(); entry.FilePath = new FileInfo(inputName); entry.Id = id; if (streaming) { extCpkArchive.Add(entry); } else { cpkArchive.Add(entry); } } else { CriAfs2Entry entry = new CriAfs2Entry(); entry.FilePath = new FileInfo(inputName); entry.Id = id; if (streaming) { extAfs2Archive.Add(entry); } else { afs2Archive.Add(entry); } } } } acbFile.Rows[0]["AwbFile"] = null; acbFile.Rows[0]["StreamAwbAfs2Header"] = null; if (afs2Archive.Count > 0 || cpkArchive.Count > 0) { Console.WriteLine("Saving internal AWB..."); acbFile.Rows[0]["AwbFile"] = cpkMode ? cpkArchive.Save() : afs2Archive.Save(); Console.WriteLine(); } if (extAfs2Archive.Count > 0 || extCpkArchive.Count > 0) { Console.WriteLine("Saving external AWB..."); if (cpkMode) { extCpkArchive.Save(awbPath, Settings.Default.BufferSize); } else { extAfs2Archive.Save(awbPath, Settings.Default.BufferSize); if (Encoding.UTF8.GetString(streamAwbAfs2Header, 0, 4) == "@UTF") { CriTable headerTable = new CriTable(); headerTable.Load(streamAwbAfs2Header); headerTable.Rows[0]["Header"] = extAfs2Archive.Header; headerTable.WriterSettings = CriTableWriterSettings.Adx2Settings; acbFile.Rows[0]["StreamAwbAfs2Header"] = headerTable.Save(); } else { acbFile.Rows[0]["StreamAwbAfs2Header"] = extAfs2Archive.Header; } } } acbFile.WriterSettings = CriTableWriterSettings.Adx2Settings; acbFile.Save(acbPath, Settings.Default.BufferSize); } #if !DEBUG } catch (Exception exception) { MessageBox.Show($"{exception.Message}", "ACB Editor", MessageBoxButtons.OK, MessageBoxIcon.Error); } #endif }