RARC : Add rewritten code ported from reborn project
This commit is contained in:
parent
ed015677df
commit
c847d40d3f
BIN
File_Format_Library.zip
Normal file
BIN
File_Format_Library.zip
Normal file
Binary file not shown.
122
File_Format_Library/FileFormats/Archives/RARC/RARC.cs
Normal file
122
File_Format_Library/FileFormats/Archives/RARC/RARC.cs
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Toolbox;
|
||||||
|
using System.Windows.Forms;
|
||||||
|
using System.IO;
|
||||||
|
using Toolbox.Library.Forms;
|
||||||
|
using Toolbox.Library;
|
||||||
|
using Toolbox.Library.IO;
|
||||||
|
|
||||||
|
namespace FirstPlugin
|
||||||
|
{
|
||||||
|
public class RARC : IArchiveFile, IFileFormat, IContextMenuNode
|
||||||
|
{
|
||||||
|
public FileType FileType { get; set; } = FileType.Archive;
|
||||||
|
|
||||||
|
public bool CanSave { get; set; }
|
||||||
|
public string[] Description { get; set; } = new string[] { "RARC" };
|
||||||
|
public string[] Extension { get; set; } = new string[] { "*.rarc", "*.arc", "*.crar", "*.yaz0" };
|
||||||
|
public string FileName { get; set; }
|
||||||
|
public string FilePath { get; set; }
|
||||||
|
public IFileInfo IFileInfo { get; set; }
|
||||||
|
|
||||||
|
public bool CanAddFiles { get; set; }
|
||||||
|
public bool CanRenameFiles { get; set; } = true;
|
||||||
|
public bool CanReplaceFiles { get; set; }
|
||||||
|
public bool CanDeleteFiles { get; set; }
|
||||||
|
|
||||||
|
public bool Identify(System.IO.Stream stream)
|
||||||
|
{
|
||||||
|
using (var reader = new Toolbox.Library.IO.FileReader(stream, true))
|
||||||
|
{
|
||||||
|
return reader.CheckSignature(4, "RARC") || reader.CheckSignature(4, "CRAR");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type[] Types
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
List<Type> types = new List<Type>();
|
||||||
|
return types.ToArray();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RARC_Parser RarcFileData;
|
||||||
|
|
||||||
|
public IEnumerable<ArchiveFileInfo> Files => RarcFileData.Files;
|
||||||
|
|
||||||
|
public void ClearFiles() { RarcFileData.Files.Clear(); }
|
||||||
|
|
||||||
|
public ToolStripItem[] GetContextMenuItems()
|
||||||
|
{
|
||||||
|
List<ToolStripItem> Items = new List<ToolStripItem>();
|
||||||
|
Items.Add(new ToolStripMenuItem("Save", null, SaveAction, Keys.Control | Keys.S));
|
||||||
|
Items.Add(new ToolStripMenuItem("Batch Rename Galaxy (Mario Galaxy)", null, BatchRenameGalaxy, Keys.Control | Keys.S));
|
||||||
|
return Items.ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void BatchRenameGalaxy(object sender, EventArgs args)
|
||||||
|
{
|
||||||
|
string ActorName = Path.GetFileNameWithoutExtension(FileName);
|
||||||
|
|
||||||
|
RenameDialog dialog = new RenameDialog();
|
||||||
|
dialog.SetString(ActorName);
|
||||||
|
|
||||||
|
if (dialog.ShowDialog() == DialogResult.OK)
|
||||||
|
{
|
||||||
|
string NewActorName = dialog.textBox1.Text;
|
||||||
|
FileName = NewActorName + ".arc";
|
||||||
|
|
||||||
|
foreach (var file in RarcFileData.Files)
|
||||||
|
{
|
||||||
|
file.FileName = file.FileName.Replace(ActorName, NewActorName);
|
||||||
|
file.UpdateWrapper();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SaveAction(object sender, EventArgs args)
|
||||||
|
{
|
||||||
|
SaveFileDialog sfd = new SaveFileDialog();
|
||||||
|
sfd.Filter = Utils.GetAllFilters(this);
|
||||||
|
sfd.FileName = FileName;
|
||||||
|
|
||||||
|
if (sfd.ShowDialog() == DialogResult.OK)
|
||||||
|
{
|
||||||
|
STFileSaver.SaveFileFormat(this, sfd.FileName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsLittleEndian { get; set; } = false;
|
||||||
|
|
||||||
|
public void Load(System.IO.Stream stream)
|
||||||
|
{
|
||||||
|
CanSave = true;
|
||||||
|
CanRenameFiles = true;
|
||||||
|
CanReplaceFiles = true;
|
||||||
|
|
||||||
|
RarcFileData= new RARC_Parser(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Save(System.IO.Stream stream)
|
||||||
|
{
|
||||||
|
RarcFileData.Save(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Unload() { }
|
||||||
|
|
||||||
|
public bool AddFile(ArchiveFileInfo archiveFileInfo)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteFile(ArchiveFileInfo archiveFileInfo)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,150 +1,40 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Toolbox;
|
|
||||||
using System.Windows.Forms;
|
|
||||||
using System.IO;
|
|
||||||
using Toolbox.Library.Forms;
|
|
||||||
using Toolbox.Library;
|
using Toolbox.Library;
|
||||||
using Toolbox.Library.IO;
|
using Toolbox.Library.IO;
|
||||||
|
|
||||||
namespace FirstPlugin
|
namespace FirstPlugin
|
||||||
{
|
{
|
||||||
public class RARC : IArchiveFile, IFileFormat, IDirectoryContainer, IContextMenuNode
|
public class RARC_Parser
|
||||||
{
|
{
|
||||||
public FileType FileType { get; set; } = FileType.Archive;
|
public List<FileEntry> Files = new List<FileEntry>();
|
||||||
|
|
||||||
public bool CanSave { get; set; }
|
|
||||||
public string[] Description { get; set; } = new string[] { "RARC" };
|
|
||||||
public string[] Extension { get; set; } = new string[] { "*.rarc", "*.arc", "*.crar", "*.yaz0" };
|
|
||||||
public string FileName { get; set; }
|
|
||||||
public string FilePath { get; set; }
|
|
||||||
public IFileInfo IFileInfo { get; set; }
|
|
||||||
|
|
||||||
public bool CanAddFiles { get; set; }
|
|
||||||
public bool CanRenameFiles { get; set; } = true;
|
|
||||||
public bool CanReplaceFiles { get; set; }
|
|
||||||
public bool CanDeleteFiles { get; set; }
|
|
||||||
|
|
||||||
public bool Identify(System.IO.Stream stream)
|
|
||||||
{
|
|
||||||
using (var reader = new Toolbox.Library.IO.FileReader(stream, true))
|
|
||||||
{
|
|
||||||
return reader.CheckSignature(4, "RARC") || reader.CheckSignature(4, "CRAR");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Type[] Types
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
List<Type> types = new List<Type>();
|
|
||||||
return types.ToArray();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<FileEntry> files = new List<FileEntry>();
|
|
||||||
public List<INode> nodes = new List<INode>();
|
|
||||||
|
|
||||||
public IEnumerable<ArchiveFileInfo> Files => files;
|
|
||||||
public IEnumerable<INode> Nodes => nodes;
|
|
||||||
|
|
||||||
public void ClearFiles() { files.Clear(); }
|
|
||||||
|
|
||||||
public string Name
|
|
||||||
{
|
|
||||||
get { return FileName; }
|
|
||||||
set { FileName = value; }
|
|
||||||
}
|
|
||||||
|
|
||||||
private DirectoryEntry[] Directories;
|
private DirectoryEntry[] Directories;
|
||||||
|
|
||||||
|
public bool IsLittleEndian = false;
|
||||||
|
|
||||||
private uint HeaderSize = 32;
|
private uint HeaderSize = 32;
|
||||||
private uint Unknown = 256;
|
private uint Unknown = 256;
|
||||||
|
|
||||||
public RamAllocation RamType = RamAllocation.MRAM;
|
public RARC_Parser(Stream stream)
|
||||||
|
|
||||||
public enum RamAllocation
|
|
||||||
{
|
{
|
||||||
None,
|
using (var reader = new FileReader(stream)) {
|
||||||
ARAM,
|
reader.SetByteOrder(true);
|
||||||
MRAM,
|
|
||||||
}
|
|
||||||
|
|
||||||
public ToolStripItem[] GetContextMenuItems()
|
|
||||||
{
|
|
||||||
List<ToolStripItem> Items = new List<ToolStripItem>();
|
|
||||||
Items.Add(new ToolStripMenuItem("Save", null, SaveAction, Keys.Control | Keys.S));
|
|
||||||
Items.Add(new ToolStripMenuItem("Batch Rename Galaxy (Mario Galaxy)", null, BatchRenameGalaxy, Keys.Control | Keys.S));
|
|
||||||
return Items.ToArray();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void BatchRenameGalaxy(object sender, EventArgs args)
|
|
||||||
{
|
|
||||||
string ActorName = Path.GetFileNameWithoutExtension(FileName);
|
|
||||||
|
|
||||||
RenameDialog dialog = new RenameDialog();
|
|
||||||
dialog.SetString(ActorName);
|
|
||||||
|
|
||||||
if (dialog.ShowDialog() == DialogResult.OK)
|
|
||||||
{
|
|
||||||
string NewActorName = dialog.textBox1.Text;
|
|
||||||
FileName = NewActorName + ".arc";
|
|
||||||
|
|
||||||
foreach (var file in files)
|
|
||||||
{
|
|
||||||
file.FileName = file.FileName.Replace(ActorName, NewActorName);
|
|
||||||
file.UpdateWrapper();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void SaveAction(object sender, EventArgs args)
|
|
||||||
{
|
|
||||||
SaveFileDialog sfd = new SaveFileDialog();
|
|
||||||
sfd.Filter = Utils.GetAllFilters(this);
|
|
||||||
sfd.FileName = FileName;
|
|
||||||
|
|
||||||
if (sfd.ShowDialog() == DialogResult.OK)
|
|
||||||
{
|
|
||||||
STFileSaver.SaveFileFormat(this, sfd.FileName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsLittleEndian { get; set; } = false;
|
|
||||||
|
|
||||||
public void Load(System.IO.Stream stream)
|
|
||||||
{
|
|
||||||
CanSave = true;
|
|
||||||
CanRenameFiles = true;
|
|
||||||
CanReplaceFiles = true;
|
|
||||||
|
|
||||||
files.Clear();
|
|
||||||
|
|
||||||
using (var reader = new FileReader(stream))
|
|
||||||
{
|
|
||||||
_savedDirectories.Clear();
|
|
||||||
|
|
||||||
reader.ByteOrder = Syroot.BinaryData.ByteOrder.BigEndian;
|
|
||||||
string signature = reader.ReadString(4, Encoding.ASCII);
|
string signature = reader.ReadString(4, Encoding.ASCII);
|
||||||
if (signature == "CRAR") {
|
if (signature == "CRAR") {
|
||||||
IsLittleEndian = true;
|
IsLittleEndian = true;
|
||||||
reader.ByteOrder = Syroot.BinaryData.ByteOrder.LittleEndian;
|
reader.SetByteOrder(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint FileSize = reader.ReadUInt32();
|
uint FileSize = reader.ReadUInt32();
|
||||||
HeaderSize = reader.ReadUInt32();
|
HeaderSize = reader.ReadUInt32();
|
||||||
uint DataOffset = reader.ReadUInt32();
|
uint DataOffset = reader.ReadUInt32();
|
||||||
uint FileDataSize = reader.ReadUInt32();
|
uint FileDataSize = reader.ReadUInt32();
|
||||||
uint MRamSize = reader.ReadUInt32();
|
uint EndOfFileOffset = reader.ReadUInt32();
|
||||||
uint ARamSize = reader.ReadUInt32();
|
byte[] Padding = reader.ReadBytes(8);
|
||||||
byte[] Padding = reader.ReadBytes(4);
|
|
||||||
|
|
||||||
if (MRamSize != 0)
|
|
||||||
RamType |= RamAllocation.MRAM;
|
|
||||||
else if (ARamSize != 0)
|
|
||||||
RamType |= RamAllocation.ARAM;
|
|
||||||
|
|
||||||
//Info Block
|
//Info Block
|
||||||
long pos = reader.Position;
|
long pos = reader.Position;
|
||||||
@ -159,79 +49,72 @@ namespace FirstPlugin
|
|||||||
Unknown = reader.ReadUInt16();
|
Unknown = reader.ReadUInt16();
|
||||||
byte[] Padding2 = reader.ReadBytes(4);
|
byte[] Padding2 = reader.ReadBytes(4);
|
||||||
|
|
||||||
|
reader.SeekBegin(DirectoryOffset);
|
||||||
Directories = new DirectoryEntry[DirectoryCount];
|
Directories = new DirectoryEntry[DirectoryCount];
|
||||||
for (int dir = 0; dir < DirectoryCount; dir++)
|
|
||||||
Directories[dir] = new DirectoryEntry(this);
|
|
||||||
|
|
||||||
Console.WriteLine($"DirectoryCount {DirectoryCount}");
|
|
||||||
Console.WriteLine($"StringTablOffset {StringTablOffset}");
|
|
||||||
|
|
||||||
reader.SeekBegin(DirectoryOffset);
|
|
||||||
for (int dir = 0; dir < DirectoryCount; dir++)
|
for (int dir = 0; dir < DirectoryCount; dir++)
|
||||||
{
|
{
|
||||||
Directories[dir].Read(reader);
|
Directories[dir] = new DirectoryEntry();
|
||||||
|
Directories[dir].Identifier = reader.ReadUInt32();
|
||||||
|
Directories[dir].NameOffset = reader.ReadUInt32();
|
||||||
|
Directories[dir].Hash = reader.ReadUInt16();
|
||||||
|
Directories[dir].NodeCount = reader.ReadUInt16();
|
||||||
|
Directories[dir].FirstNodeIndex = reader.ReadUInt32();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int dir = 0; dir < DirectoryCount; dir++) {
|
||||||
|
uint NamePointer = StringTablOffset + Directories[dir].NameOffset;
|
||||||
|
Directories[dir].Name = ReadStringAtTable(reader, NamePointer);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int dir = 0; dir < DirectoryCount; dir++)
|
for (int dir = 0; dir < DirectoryCount; dir++)
|
||||||
{
|
{
|
||||||
uint NamePointer = StringTablOffset + Directories[dir].NameOffset;
|
|
||||||
Directories[dir].Name = ReadStringAtTable(reader, NamePointer);
|
|
||||||
|
|
||||||
Console.WriteLine($"{ Directories[dir].Name } {dir}");
|
|
||||||
|
|
||||||
for (int n = 0; n < Directories[dir].NodeCount; n++)
|
for (int n = 0; n < Directories[dir].NodeCount; n++)
|
||||||
{
|
{
|
||||||
|
//Seek to the child entry
|
||||||
reader.SeekBegin(NodeOffset + ((n + Directories[dir].FirstNodeIndex) * 0x14));
|
reader.SeekBegin(NodeOffset + ((n + Directories[dir].FirstNodeIndex) * 0x14));
|
||||||
FileEntry entry = new FileEntry();
|
FileEntry entry = new FileEntry();
|
||||||
entry.Read(reader, IsLittleEndian);
|
entry.Read(reader, IsLittleEndian);
|
||||||
NamePointer = StringTablOffset + entry.NameOffset;
|
var NamePointer = StringTablOffset + entry.NameOffset;
|
||||||
entry.Name = ReadStringAtTable(reader, NamePointer);
|
entry.Name = ReadStringAtTable(reader, NamePointer);
|
||||||
|
|
||||||
//Root and parent strings. Skip these unecessary nodes.
|
//Skip root strings
|
||||||
if (entry.Name == "." || entry.Name == "..")
|
if (entry.Name == "." || entry.Name == "..")
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (entry.IsDirectory)
|
if (entry.IsDirectory)
|
||||||
{
|
{
|
||||||
// Directories[entry.Offset].ID = entry.FileId; //0xFFFF or 0
|
|
||||||
Directories[dir].AddNode(Directories[entry.Offset]);
|
Directories[dir].AddNode(Directories[entry.Offset]);
|
||||||
_savedDirectories.Add(entry);
|
_savedDirectories.Add(entry);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
using (reader.TemporarySeek(pos + DataOffset + entry.Offset, System.IO.SeekOrigin.Begin))
|
long dataPos = pos + DataOffset + entry.Offset;
|
||||||
{
|
entry.FileData = reader.getSection((int)dataPos, (int)entry.Size);
|
||||||
entry.FileData = reader.ReadBytes((int)entry.Size);
|
|
||||||
}
|
|
||||||
entry.FileName = entry.Name;
|
|
||||||
files.Add(entry);
|
|
||||||
|
|
||||||
|
Files.Add(entry);
|
||||||
Directories[dir].AddNode(entry);
|
Directories[dir].AddNode(entry);
|
||||||
|
|
||||||
|
entry.FileName = GetFullDirectory(entry);
|
||||||
|
if (Utils.GetExtension(entry.FileName) == ".key")
|
||||||
|
entry.OpenFileFormatOnLoad = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.Name = Directories[0].Name;
|
|
||||||
nodes.AddRange(Directories[0].Nodes);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void LoadAllDirectoriesAndFiles(DirectoryEntry parentDir)
|
private string GetFullDirectory(INode file)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < parentDir.Children.Count; i++)
|
if (file.Parent != null)
|
||||||
{
|
return GetFullDirectory(file.Parent) + "/" + file.Name;
|
||||||
if (parentDir.Children[i] is DirectoryEntry)
|
else
|
||||||
{
|
return file.Name;
|
||||||
//_savedDirectories.Add((DirectoryEntry)parentDir.nodes[i]);
|
}
|
||||||
}
|
|
||||||
else
|
|
||||||
_savedFiles.Add((FileEntry)parentDir.Children[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < parentDir.Children.Count; i++)
|
public void Save(Stream stream)
|
||||||
{
|
{
|
||||||
if (parentDir.Children[i] is DirectoryEntry)
|
using (var writer = new FileWriter(stream)) {
|
||||||
LoadAllDirectoriesAndFiles((DirectoryEntry)parentDir.Children[i]);
|
SaveFile(writer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -260,14 +143,12 @@ namespace FirstPlugin
|
|||||||
writer.ByteOrder = Syroot.BinaryData.ByteOrder.LittleEndian;
|
writer.ByteOrder = Syroot.BinaryData.ByteOrder.LittleEndian;
|
||||||
|
|
||||||
writer.WriteSignature(IsLittleEndian ? "CRAR" : "RARC");
|
writer.WriteSignature(IsLittleEndian ? "CRAR" : "RARC");
|
||||||
|
|
||||||
writer.Write(uint.MaxValue); //FileSize
|
writer.Write(uint.MaxValue); //FileSize
|
||||||
writer.Write(HeaderSize);
|
writer.Write(HeaderSize);
|
||||||
writer.Write(uint.MaxValue); //DataOffset
|
writer.Write(uint.MaxValue); //DataOffset
|
||||||
writer.Write(uint.MaxValue); //File Size
|
writer.Write(uint.MaxValue); //File Size
|
||||||
writer.Write(0); //MRAM (saved later if used)
|
writer.Write(uint.MaxValue); //End of file
|
||||||
writer.Write(0); //ARAM (saved later if used)
|
writer.Seek(8); //padding
|
||||||
writer.Seek(4); //padding
|
|
||||||
|
|
||||||
writer.SeekBegin(HeaderSize);
|
writer.SeekBegin(HeaderSize);
|
||||||
long InfoPos = writer.Position;
|
long InfoPos = writer.Position;
|
||||||
@ -300,7 +181,7 @@ namespace FirstPlugin
|
|||||||
foreach (FileEntry entry in Files)
|
foreach (FileEntry entry in Files)
|
||||||
entry.SaveFileFormat();
|
entry.SaveFileFormat();
|
||||||
|
|
||||||
writer.Align(32);
|
writer.Align(32);
|
||||||
|
|
||||||
Console.WriteLine("Directories");
|
Console.WriteLine("Directories");
|
||||||
|
|
||||||
@ -316,20 +197,11 @@ namespace FirstPlugin
|
|||||||
var dirEntry = (DirectoryEntry)_savedNodes[dir].Children[n];
|
var dirEntry = (DirectoryEntry)_savedNodes[dir].Children[n];
|
||||||
int index = Directories.ToList().IndexOf(dirEntry);
|
int index = Directories.ToList().IndexOf(dirEntry);
|
||||||
|
|
||||||
writer.Write(dirEntry.ID);
|
writer.Write((ushort)0);
|
||||||
writer.Write(dirEntry.Hash);
|
writer.Write(dirEntry.Hash);
|
||||||
if (IsLittleEndian)
|
writer.Write((byte)0x2);
|
||||||
{
|
writer.Write((byte)0);
|
||||||
writer.Write((ushort)dirEntry.NameOffset);
|
writer.Write((ushort)dirEntry.NameOffset);
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((byte)0x2);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
writer.Write((byte)0x2);
|
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((ushort)dirEntry.NameOffset);
|
|
||||||
}
|
|
||||||
writer.Write((int)index);
|
writer.Write((int)index);
|
||||||
writer.Write((int)16);
|
writer.Write((int)16);
|
||||||
writer.Write((int)0);
|
writer.Write((int)0);
|
||||||
@ -339,21 +211,12 @@ namespace FirstPlugin
|
|||||||
var fileEntry = (FileEntry)entry;
|
var fileEntry = (FileEntry)entry;
|
||||||
writer.Write(fileEntry.FileId);
|
writer.Write(fileEntry.FileId);
|
||||||
writer.Write(fileEntry.Hash);
|
writer.Write(fileEntry.Hash);
|
||||||
if (IsLittleEndian)
|
writer.Write(fileEntry.Flags);
|
||||||
{
|
writer.Write((byte)0); //Padding
|
||||||
writer.Write((ushort)fileEntry.NameOffset);
|
writer.Write((ushort)fileEntry.NameOffset);
|
||||||
writer.Write((byte)0); //Padding
|
|
||||||
writer.Write(fileEntry.Flags);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
writer.Write(fileEntry.Flags);
|
|
||||||
writer.Write((byte)0); //Padding
|
|
||||||
writer.Write((ushort)fileEntry.NameOffset);
|
|
||||||
}
|
|
||||||
|
|
||||||
fileEntry._dataOffsetPos = writer.Position;
|
fileEntry._dataOffsetPos = writer.Position;
|
||||||
writer.Write(0);
|
writer.Write(0);
|
||||||
writer.Write((uint)fileEntry.FileData.Length);
|
writer.Write((uint)fileEntry.FileData.Length);
|
||||||
_savedFileData.Add(fileEntry.FileData);
|
_savedFileData.Add(fileEntry.FileData);
|
||||||
writer.Write(0);
|
writer.Write(0);
|
||||||
@ -363,18 +226,9 @@ namespace FirstPlugin
|
|||||||
//Save parent and root nodes at end
|
//Save parent and root nodes at end
|
||||||
writer.Write((ushort)0);
|
writer.Write((ushort)0);
|
||||||
writer.Write((ushort)46);
|
writer.Write((ushort)46);
|
||||||
if (IsLittleEndian)
|
writer.Write((byte)2);
|
||||||
{
|
writer.Write((byte)0);
|
||||||
writer.Write((ushort)0);
|
writer.Write((ushort)0);
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((byte)2);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
writer.Write((byte)2);
|
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((ushort)0);
|
|
||||||
}
|
|
||||||
writer.Write(dir);
|
writer.Write(dir);
|
||||||
writer.Write(16);
|
writer.Write(16);
|
||||||
writer.Write(0);
|
writer.Write(0);
|
||||||
@ -383,18 +237,9 @@ namespace FirstPlugin
|
|||||||
|
|
||||||
writer.Write((ushort)0);
|
writer.Write((ushort)0);
|
||||||
writer.Write((ushort)184);
|
writer.Write((ushort)184);
|
||||||
if (IsLittleEndian)
|
writer.Write((byte)2);
|
||||||
{
|
writer.Write((byte)0);
|
||||||
writer.Write((ushort)2);
|
writer.Write((ushort)2);
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((byte)2);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
writer.Write((byte)2);
|
|
||||||
writer.Write((byte)0);
|
|
||||||
writer.Write((ushort)2);
|
|
||||||
}
|
|
||||||
writer.Write(rootIndex);
|
writer.Write(rootIndex);
|
||||||
writer.Write(16);
|
writer.Write(16);
|
||||||
writer.Write(0);
|
writer.Write(0);
|
||||||
@ -404,21 +249,19 @@ namespace FirstPlugin
|
|||||||
|
|
||||||
long stringTablePos = writer.Position;
|
long stringTablePos = writer.Position;
|
||||||
WriteOffset(writer, 20, InfoPos);
|
WriteOffset(writer, 20, InfoPos);
|
||||||
foreach (var str in stringTable)
|
foreach (var str in stringTable) {
|
||||||
{
|
|
||||||
writer.WriteString(str);
|
writer.WriteString(str);
|
||||||
}
|
}
|
||||||
writer.Align(32);
|
writer.Align(32);
|
||||||
long stringTableSize = writer.Position - stringTablePos;
|
long stringTableSize = writer.Position - stringTablePos;
|
||||||
|
|
||||||
var dataPos = writer.Position;
|
var dataPos = writer.Position;
|
||||||
using (writer.TemporarySeek(pos + 12, System.IO.SeekOrigin.Begin))
|
using (writer.TemporarySeek(pos + 12, System.IO.SeekOrigin.Begin)) {
|
||||||
{
|
|
||||||
writer.Write((uint)(dataPos - InfoPos));
|
writer.Write((uint)(dataPos - InfoPos));
|
||||||
}
|
}
|
||||||
|
|
||||||
uint dataOffset = 0;
|
uint dataOffset = 0;
|
||||||
SaveFileData(writer, ref dataOffset, Directories[0]);
|
SaveFileData(writer,ref dataOffset, Directories[0]);
|
||||||
uint DataSize = (uint)(writer.Position - dataPos);
|
uint DataSize = (uint)(writer.Position - dataPos);
|
||||||
|
|
||||||
writer.Align(32);
|
writer.Align(32);
|
||||||
@ -426,28 +269,22 @@ namespace FirstPlugin
|
|||||||
uint EndFileSize = (uint)(writer.Position - dataPos);
|
uint EndFileSize = (uint)(writer.Position - dataPos);
|
||||||
|
|
||||||
//Write data size
|
//Write data size
|
||||||
using (writer.TemporarySeek(pos + 16, System.IO.SeekOrigin.Begin))
|
using (writer.TemporarySeek(pos + 16, System.IO.SeekOrigin.Begin)) {
|
||||||
{
|
|
||||||
writer.Write((uint)DataSize);
|
writer.Write((uint)DataSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
//Write ram size
|
//Write end of file size
|
||||||
if (RamType != RamAllocation.None)
|
using (writer.TemporarySeek(pos + 20, System.IO.SeekOrigin.Begin)) {
|
||||||
{
|
writer.Write((uint)EndFileSize);
|
||||||
using (writer.TemporarySeek(pos + (RamType.HasFlag(RamAllocation.MRAM) ? 20 : 24), System.IO.SeekOrigin.Begin)) {
|
|
||||||
writer.Write((uint)EndFileSize);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//Write file size
|
//Write file size
|
||||||
using (writer.TemporarySeek(pos + 0x4, System.IO.SeekOrigin.Begin))
|
using (writer.TemporarySeek(pos + 0x4, System.IO.SeekOrigin.Begin)) {
|
||||||
{
|
|
||||||
writer.Write((uint)writer.BaseStream.Length);
|
writer.Write((uint)writer.BaseStream.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
//Write string table size
|
//Write string table size
|
||||||
using (writer.TemporarySeek(InfoPos + 16, System.IO.SeekOrigin.Begin))
|
using (writer.TemporarySeek(InfoPos + 16, System.IO.SeekOrigin.Begin)) {
|
||||||
{
|
|
||||||
writer.Write((uint)stringTableSize);
|
writer.Write((uint)stringTableSize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -468,10 +305,9 @@ namespace FirstPlugin
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void CreateStringTable(List<string> stringList,
|
private void CreateStringTable(List<string> stringList,
|
||||||
ref uint stringPos, DirectoryEntry parentDir)
|
ref uint stringPos, DirectoryEntry parentDir)
|
||||||
{
|
{
|
||||||
if (!stringList.Contains(parentDir.Name))
|
if (!stringList.Contains(parentDir.Name)) {
|
||||||
{
|
|
||||||
parentDir.NameOffset = stringPos;
|
parentDir.NameOffset = stringPos;
|
||||||
stringList.Add(parentDir.Name);
|
stringList.Add(parentDir.Name);
|
||||||
stringPos += (uint)parentDir.Name.Length + 1;
|
stringPos += (uint)parentDir.Name.Length + 1;
|
||||||
@ -479,15 +315,9 @@ namespace FirstPlugin
|
|||||||
for (int i = 0; i < parentDir.Children.Count; i++)
|
for (int i = 0; i < parentDir.Children.Count; i++)
|
||||||
{
|
{
|
||||||
if (parentDir.Children[i] is FileEntry)
|
if (parentDir.Children[i] is FileEntry)
|
||||||
{
|
|
||||||
((FileEntry)parentDir.Children[i]).NameOffset = (ushort)stringPos;
|
((FileEntry)parentDir.Children[i]).NameOffset = (ushort)stringPos;
|
||||||
((FileEntry)parentDir.Children[i]).UpdateHash();
|
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
|
||||||
((DirectoryEntry)parentDir.Children[i]).NameOffset = (ushort)stringPos;
|
((DirectoryEntry)parentDir.Children[i]).NameOffset = (ushort)stringPos;
|
||||||
((DirectoryEntry)parentDir.Children[i]).UpdateHash();
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.WriteLine($"{parentDir.Children[i].Name} {stringPos}");
|
Console.WriteLine($"{parentDir.Children[i].Name} {stringPos}");
|
||||||
|
|
||||||
@ -500,16 +330,14 @@ namespace FirstPlugin
|
|||||||
}
|
}
|
||||||
|
|
||||||
//Saves data in recusive order
|
//Saves data in recusive order
|
||||||
private void SaveFileData(FileWriter writer, ref uint dataPos, DirectoryEntry parentDir)
|
private void SaveFileData(FileWriter writer, ref uint dataPos, DirectoryEntry parentDir)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < parentDir.Children.Count; i++)
|
for (int i = 0; i < parentDir.Children.Count; i++)
|
||||||
{
|
{
|
||||||
if (parentDir.Children[i] is FileEntry)
|
if (parentDir.Children[i] is FileEntry) {
|
||||||
{
|
|
||||||
var entry = (FileEntry)parentDir.Children[i];
|
var entry = (FileEntry)parentDir.Children[i];
|
||||||
long pos = writer.Position;
|
long pos = writer.Position;
|
||||||
using (writer.TemporarySeek(entry._dataOffsetPos, System.IO.SeekOrigin.Begin))
|
using (writer.TemporarySeek(entry._dataOffsetPos, SeekOrigin.Begin)) {
|
||||||
{
|
|
||||||
writer.Write((uint)dataPos);
|
writer.Write((uint)dataPos);
|
||||||
}
|
}
|
||||||
writer.Write(entry.FileData);
|
writer.Write(entry.FileData);
|
||||||
@ -522,6 +350,25 @@ namespace FirstPlugin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void LoadAllDirectoriesAndFiles(DirectoryEntry parentDir)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < parentDir.Children.Count; i++)
|
||||||
|
{
|
||||||
|
if (parentDir.Children[i] is DirectoryEntry)
|
||||||
|
{
|
||||||
|
//_savedDirectories.Add((DirectoryEntry)parentDir.nodes[i]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
_savedFiles.Add((FileEntry)parentDir.Children[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < parentDir.Children.Count; i++)
|
||||||
|
{
|
||||||
|
if (parentDir.Children[i] is DirectoryEntry)
|
||||||
|
LoadAllDirectoriesAndFiles((DirectoryEntry)parentDir.Children[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void WriteOffset(FileWriter writer, long Target, long RelativePosition)
|
private void WriteOffset(FileWriter writer, long Target, long RelativePosition)
|
||||||
{
|
{
|
||||||
long Position = writer.Position;
|
long Position = writer.Position;
|
||||||
@ -531,10 +378,8 @@ namespace FirstPlugin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private string ReadStringAtTable(FileReader reader, uint NameOffset)
|
private static string ReadStringAtTable(FileReader reader, uint NameOffset) {
|
||||||
{
|
using (reader.TemporarySeek(NameOffset, System.IO.SeekOrigin.Begin)) {
|
||||||
using (reader.TemporarySeek(NameOffset, System.IO.SeekOrigin.Begin))
|
|
||||||
{
|
|
||||||
return reader.ReadZeroTerminatedString();
|
return reader.ReadZeroTerminatedString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -550,79 +395,36 @@ namespace FirstPlugin
|
|||||||
return Hash;
|
return Hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void CreateDirectoryEntry()
|
class DirectoryEntry : INode
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public class DirectoryEntry : IDirectoryContainer
|
|
||||||
{
|
|
||||||
public RARC ParentArchive { get; }
|
|
||||||
|
|
||||||
public string Name { get; set; }
|
public string Name { get; set; }
|
||||||
|
|
||||||
public uint Identifier;
|
public uint Identifier;
|
||||||
internal uint NameOffset; //Relative to string table
|
internal uint NameOffset; //Relative to string table
|
||||||
public ushort Hash { get; set; }
|
public ushort Hash { get; set; }
|
||||||
public ushort NodeCount;
|
public ushort NodeCount;
|
||||||
public uint FirstNodeIndex { get; set; }
|
public uint FirstNodeIndex { get; set; }
|
||||||
|
|
||||||
public ushort ID { get; set; } = 0xFFFF;
|
|
||||||
|
|
||||||
public DirectoryEntry(RARC rarc) { ParentArchive = rarc; }
|
|
||||||
|
|
||||||
public IEnumerable<INode> Nodes { get { return Children; } }
|
|
||||||
public List<INode> Children = new List<INode>();
|
public List<INode> Children = new List<INode>();
|
||||||
|
|
||||||
public INode Parent { get; set; }
|
public INode Parent { get; set; }
|
||||||
|
|
||||||
public void AddNode(INode node)
|
public void AddNode(INode node)
|
||||||
{
|
{
|
||||||
if (node is FileEntry)
|
node.Parent = this;
|
||||||
((FileEntry)node).Parent = this;
|
|
||||||
else
|
|
||||||
((DirectoryEntry)node).Parent = this;
|
|
||||||
|
|
||||||
Children.Add(node);
|
Children.Add(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Read(FileReader reader)
|
|
||||||
{
|
|
||||||
Identifier = reader.ReadUInt32();
|
|
||||||
NameOffset = reader.ReadUInt32();
|
|
||||||
Hash = reader.ReadUInt16();
|
|
||||||
NodeCount = reader.ReadUInt16();
|
|
||||||
FirstNodeIndex = reader.ReadUInt32();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UpdateHash() {
|
public void UpdateHash() {
|
||||||
Hash = CalculateHash(Name);
|
Hash = CalculateHash(Name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Unload()
|
public class FileEntry : ArchiveFileInfo, INode
|
||||||
{
|
{
|
||||||
|
public string Name { get; set; }
|
||||||
|
|
||||||
}
|
public INode Parent { get; set; }
|
||||||
|
|
||||||
public void Save(System.IO.Stream stream)
|
|
||||||
{
|
|
||||||
SaveFile(new FileWriter(stream));
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool AddFile(ArchiveFileInfo archiveFileInfo)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool DeleteFile(ArchiveFileInfo archiveFileInfo)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public class FileEntry : ArchiveFileInfo
|
|
||||||
{
|
|
||||||
//According to this to determine directory or not
|
|
||||||
//https://github.com/LordNed/WArchive-Tools/blob/3c7fdefe54b4c7634a042847b7455de61705033f/ArchiveToolsLib/Archive/Archive.cs#L44
|
|
||||||
public bool IsDirectory { get { return (Flags & 2) >> 1 == 1; } }
|
public bool IsDirectory { get { return (Flags & 2) >> 1 == 1; } }
|
||||||
|
|
||||||
public ushort FileId { get; set; }
|
public ushort FileId { get; set; }
|
||||||
@ -635,13 +437,11 @@ namespace FirstPlugin
|
|||||||
|
|
||||||
internal long _dataOffsetPos;
|
internal long _dataOffsetPos;
|
||||||
|
|
||||||
public INode Parent { get; set; }
|
public void Read(FileReader reader, bool isLittleEndian)
|
||||||
|
|
||||||
public void Read(FileReader reader, bool IsLittleEndian)
|
|
||||||
{
|
{
|
||||||
FileId = reader.ReadUInt16();
|
FileId = reader.ReadUInt16();
|
||||||
Hash = reader.ReadUInt16();
|
Hash = reader.ReadUInt16();
|
||||||
if (IsLittleEndian)
|
if (isLittleEndian)
|
||||||
{
|
{
|
||||||
NameOffset = reader.ReadUInt16();
|
NameOffset = reader.ReadUInt16();
|
||||||
reader.Seek(1); //Padding
|
reader.Seek(1); //Padding
|
||||||
@ -657,10 +457,12 @@ namespace FirstPlugin
|
|||||||
Offset = reader.ReadUInt32();
|
Offset = reader.ReadUInt32();
|
||||||
Size = reader.ReadUInt32();
|
Size = reader.ReadUInt32();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void UpdateHash() {
|
public interface INode
|
||||||
Hash = CalculateHash(Name);
|
{
|
||||||
}
|
string Name { get; set; }
|
||||||
|
INode Parent { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -225,6 +225,7 @@
|
|||||||
<Compile Include="FileFormats\Archives\PKG.cs" />
|
<Compile Include="FileFormats\Archives\PKG.cs" />
|
||||||
<Compile Include="FileFormats\Archives\QuickAccess\QuickAccessFile.cs" />
|
<Compile Include="FileFormats\Archives\QuickAccess\QuickAccessFile.cs" />
|
||||||
<Compile Include="FileFormats\Archives\QuickAccess\QuickAccessFolder.cs" />
|
<Compile Include="FileFormats\Archives\QuickAccess\QuickAccessFolder.cs" />
|
||||||
|
<Compile Include="FileFormats\Archives\RARC\RARC_Parser.cs" />
|
||||||
<Compile Include="FileFormats\Archives\WTA.cs" />
|
<Compile Include="FileFormats\Archives\WTA.cs" />
|
||||||
<Compile Include="FileFormats\Audio\BARS\BarsFile.cs" />
|
<Compile Include="FileFormats\Audio\BARS\BarsFile.cs" />
|
||||||
<Compile Include="FileFormats\BCH\BCH.cs" />
|
<Compile Include="FileFormats\BCH\BCH.cs" />
|
||||||
@ -367,7 +368,7 @@
|
|||||||
<Compile Include="FileFormats\Archives\ME01.cs" />
|
<Compile Include="FileFormats\Archives\ME01.cs" />
|
||||||
<Compile Include="FileFormats\Archives\MKGPDX_PAC.cs" />
|
<Compile Include="FileFormats\Archives\MKGPDX_PAC.cs" />
|
||||||
<Compile Include="FileFormats\Archives\NXARC.cs" />
|
<Compile Include="FileFormats\Archives\NXARC.cs" />
|
||||||
<Compile Include="FileFormats\Archives\RARC.cs" />
|
<Compile Include="FileFormats\Archives\RARC\RARC.cs" />
|
||||||
<Compile Include="FileFormats\Archives\Sonic Racing\GameDataToc.cs" />
|
<Compile Include="FileFormats\Archives\Sonic Racing\GameDataToc.cs" />
|
||||||
<Compile Include="FileFormats\Archives\Sonic Racing\SPC.cs" />
|
<Compile Include="FileFormats\Archives\Sonic Racing\SPC.cs" />
|
||||||
<Compile Include="FileFormats\Archives\Sonic Racing\SWU.cs" />
|
<Compile Include="FileFormats\Archives\Sonic Racing\SWU.cs" />
|
||||||
|
Loading…
Reference in New Issue
Block a user