2018-07-20 01:31:35 +02:00
|
|
|
|
using System.IO;
|
2018-07-22 05:05:57 +02:00
|
|
|
|
using System.Security.Cryptography;
|
2018-07-20 01:31:35 +02:00
|
|
|
|
|
|
|
|
|
namespace libhac.Savefile
|
|
|
|
|
{
|
|
|
|
|
public class Header
|
|
|
|
|
{
|
|
|
|
|
public byte[] Cmac { get; set; }
|
|
|
|
|
public FsLayout Layout { get; set; }
|
2018-07-24 21:51:52 +02:00
|
|
|
|
public JournalHeader Journal { get; set; }
|
|
|
|
|
public SaveHeader Save { get; set; }
|
2018-07-20 01:31:35 +02:00
|
|
|
|
|
|
|
|
|
public RemapHeader FileRemap { get; set; }
|
|
|
|
|
public RemapHeader MetaRemap { get; set; }
|
|
|
|
|
|
|
|
|
|
public MapEntry[] FileMapEntries { get; set; }
|
|
|
|
|
public MapEntry[] MetaMapEntries { get; set; }
|
|
|
|
|
|
2018-08-25 00:59:45 +02:00
|
|
|
|
public byte[] MasterHashA { get; }
|
|
|
|
|
public byte[] MasterHashB { get; }
|
|
|
|
|
public byte[] DuplexMasterA { get; }
|
|
|
|
|
public byte[] DuplexMasterB { get; }
|
|
|
|
|
|
2018-07-22 05:05:57 +02:00
|
|
|
|
public byte[] Data { get; }
|
|
|
|
|
|
|
|
|
|
public Header(BinaryReader reader, IProgressReport logger = null)
|
2018-07-20 01:31:35 +02:00
|
|
|
|
{
|
2018-07-22 05:05:57 +02:00
|
|
|
|
reader.BaseStream.Position = 0;
|
|
|
|
|
Data = reader.ReadBytes(0x4000);
|
|
|
|
|
reader.BaseStream.Position = 0;
|
|
|
|
|
|
2018-07-20 01:31:35 +02:00
|
|
|
|
Cmac = reader.ReadBytes(0x10);
|
|
|
|
|
|
|
|
|
|
reader.BaseStream.Position = 0x100;
|
|
|
|
|
Layout = new FsLayout(reader);
|
|
|
|
|
|
2018-07-23 03:28:31 +02:00
|
|
|
|
reader.BaseStream.Position = 0x408;
|
|
|
|
|
Journal = new JournalHeader(reader);
|
|
|
|
|
|
2018-07-24 21:51:52 +02:00
|
|
|
|
reader.BaseStream.Position = 0x608;
|
|
|
|
|
Save = new SaveHeader(reader);
|
|
|
|
|
|
2018-07-20 01:31:35 +02:00
|
|
|
|
reader.BaseStream.Position = 0x650;
|
|
|
|
|
FileRemap = new RemapHeader(reader);
|
|
|
|
|
reader.BaseStream.Position = 0x690;
|
|
|
|
|
MetaRemap = new RemapHeader(reader);
|
|
|
|
|
|
2018-08-25 00:59:45 +02:00
|
|
|
|
reader.BaseStream.Position = Layout.MasterHashOffset0;
|
|
|
|
|
MasterHashA = reader.ReadBytes((int) Layout.MasterHashSize);
|
|
|
|
|
reader.BaseStream.Position = Layout.MasterHashOffset1;
|
|
|
|
|
MasterHashB = reader.ReadBytes((int) Layout.MasterHashSize);
|
|
|
|
|
|
|
|
|
|
reader.BaseStream.Position = Layout.L1BitmapOffset0;
|
|
|
|
|
DuplexMasterA = reader.ReadBytes((int) Layout.L1BitmapSize);
|
|
|
|
|
reader.BaseStream.Position = Layout.L1BitmapOffset1;
|
|
|
|
|
DuplexMasterB = reader.ReadBytes((int) Layout.L1BitmapSize);
|
|
|
|
|
|
2018-07-20 01:31:35 +02:00
|
|
|
|
reader.BaseStream.Position = Layout.FileMapEntryOffset;
|
|
|
|
|
FileMapEntries = new MapEntry[FileRemap.MapEntryCount];
|
|
|
|
|
for (int i = 0; i < FileRemap.MapEntryCount; i++)
|
|
|
|
|
{
|
|
|
|
|
FileMapEntries[i] = new MapEntry(reader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reader.BaseStream.Position = Layout.MetaMapEntryOffset;
|
|
|
|
|
MetaMapEntries = new MapEntry[MetaRemap.MapEntryCount];
|
|
|
|
|
for (int i = 0; i < MetaRemap.MapEntryCount; i++)
|
|
|
|
|
{
|
|
|
|
|
MetaMapEntries[i] = new MapEntry(reader);
|
|
|
|
|
}
|
2018-07-22 05:05:57 +02:00
|
|
|
|
|
|
|
|
|
var hashStatus = ValidateHeaderHash() ? "valid" : "invalid";
|
|
|
|
|
logger?.LogMessage($"Header hash is {hashStatus}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool ValidateHeaderHash()
|
|
|
|
|
{
|
|
|
|
|
using (SHA256 sha256 = SHA256.Create())
|
|
|
|
|
{
|
|
|
|
|
var hash = sha256.ComputeHash(Data, 0x300, 0x3d00);
|
|
|
|
|
return Util.ArraysEqual(hash, Layout.Hash);
|
|
|
|
|
}
|
2018-07-20 01:31:35 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class FsLayout
|
|
|
|
|
{
|
|
|
|
|
public string Magic { get; set; }
|
|
|
|
|
public uint MagicNum { get; set; }
|
|
|
|
|
public byte[] Hash { get; set; }
|
|
|
|
|
public long FileMapEntryOffset { get; set; }
|
|
|
|
|
public long FileMapEntrySize { get; set; }
|
|
|
|
|
public long MetaMapEntryOffset { get; set; }
|
|
|
|
|
public long MetaMapEntrySize { get; set; }
|
|
|
|
|
public long FileMapDataOffset { get; set; }
|
|
|
|
|
public long FileMapDataSize { get; set; }
|
|
|
|
|
public long DuplexL1OffsetA { get; set; }
|
|
|
|
|
public long DuplexL1OffsetB { get; set; }
|
|
|
|
|
public long DuplexL1Size { get; set; }
|
|
|
|
|
public long DuplexDataOffsetA { get; set; }
|
|
|
|
|
public long DuplexDataOffsetB { get; set; }
|
|
|
|
|
public long DuplexDataSize { get; set; }
|
|
|
|
|
public long JournalDataOffset { get; set; }
|
|
|
|
|
public long JournalDataSizeA { get; set; }
|
|
|
|
|
public long JournalDataSizeB { get; set; }
|
|
|
|
|
public long SizeReservedArea { get; set; }
|
2018-07-22 05:05:57 +02:00
|
|
|
|
public long L1BitmapOffset0 { get; set; }
|
|
|
|
|
public long L1BitmapOffset1 { get; set; }
|
|
|
|
|
public long L1BitmapSize { get; set; }
|
|
|
|
|
public long MasterHashOffset0 { get; set; }
|
|
|
|
|
public long MasterHashOffset1 { get; set; }
|
2018-07-20 01:31:35 +02:00
|
|
|
|
public long MasterHashSize { get; set; }
|
2018-07-23 03:28:31 +02:00
|
|
|
|
public long JournalTableOffset { get; set; }
|
|
|
|
|
public long JournalTableSize { get; set; }
|
2018-07-20 01:31:35 +02:00
|
|
|
|
public long JournalBitmapUpdatedPhysicalOffset { get; set; }
|
|
|
|
|
public long JournalBitmapUpdatedPhysicalSize { get; set; }
|
|
|
|
|
public long JournalBitmapUpdatedVirtualOffset { get; set; }
|
|
|
|
|
public long JournalBitmapUpdatedVirtualSize { get; set; }
|
|
|
|
|
public long JournalBitmapUnassignedOffset { get; set; }
|
|
|
|
|
public long JournalBitmapUnassignedSize { get; set; }
|
|
|
|
|
public long Layer1HashOffset { get; set; }
|
|
|
|
|
public long Layer1HashSize { get; set; }
|
|
|
|
|
public long Layer2HashOffset { get; set; }
|
|
|
|
|
public long Layer2HashSize { get; set; }
|
|
|
|
|
public long Layer3HashOffset { get; set; }
|
|
|
|
|
public long Layer3HashSize { get; set; }
|
|
|
|
|
public long Field148 { get; set; }
|
|
|
|
|
public long Field150 { get; set; }
|
2018-07-25 01:50:41 +02:00
|
|
|
|
public long DuplexIndex { get; set; }
|
2018-07-20 01:31:35 +02:00
|
|
|
|
|
|
|
|
|
public FsLayout(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
Magic = reader.ReadAscii(4);
|
|
|
|
|
MagicNum = reader.ReadUInt32();
|
|
|
|
|
Hash = reader.ReadBytes(0x20);
|
|
|
|
|
FileMapEntryOffset = reader.ReadInt64();
|
|
|
|
|
FileMapEntrySize = reader.ReadInt64();
|
|
|
|
|
MetaMapEntryOffset = reader.ReadInt64();
|
|
|
|
|
MetaMapEntrySize = reader.ReadInt64();
|
|
|
|
|
FileMapDataOffset = reader.ReadInt64();
|
|
|
|
|
FileMapDataSize = reader.ReadInt64();
|
|
|
|
|
DuplexL1OffsetA = reader.ReadInt64();
|
|
|
|
|
DuplexL1OffsetB = reader.ReadInt64();
|
|
|
|
|
DuplexL1Size = reader.ReadInt64();
|
|
|
|
|
DuplexDataOffsetA = reader.ReadInt64();
|
|
|
|
|
DuplexDataOffsetB = reader.ReadInt64();
|
|
|
|
|
DuplexDataSize = reader.ReadInt64();
|
|
|
|
|
JournalDataOffset = reader.ReadInt64();
|
|
|
|
|
JournalDataSizeA = reader.ReadInt64();
|
|
|
|
|
JournalDataSizeB = reader.ReadInt64();
|
|
|
|
|
SizeReservedArea = reader.ReadInt64();
|
2018-07-22 05:05:57 +02:00
|
|
|
|
L1BitmapOffset0 = reader.ReadInt64();
|
|
|
|
|
L1BitmapOffset1 = reader.ReadInt64();
|
|
|
|
|
L1BitmapSize = reader.ReadInt64();
|
|
|
|
|
MasterHashOffset0 = reader.ReadInt64();
|
|
|
|
|
MasterHashOffset1 = reader.ReadInt64();
|
2018-07-20 01:31:35 +02:00
|
|
|
|
MasterHashSize = reader.ReadInt64();
|
2018-07-23 03:28:31 +02:00
|
|
|
|
JournalTableOffset = reader.ReadInt64();
|
|
|
|
|
JournalTableSize = reader.ReadInt64();
|
2018-07-20 01:31:35 +02:00
|
|
|
|
JournalBitmapUpdatedPhysicalOffset = reader.ReadInt64();
|
|
|
|
|
JournalBitmapUpdatedPhysicalSize = reader.ReadInt64();
|
|
|
|
|
JournalBitmapUpdatedVirtualOffset = reader.ReadInt64();
|
|
|
|
|
JournalBitmapUpdatedVirtualSize = reader.ReadInt64();
|
|
|
|
|
JournalBitmapUnassignedOffset = reader.ReadInt64();
|
|
|
|
|
JournalBitmapUnassignedSize = reader.ReadInt64();
|
|
|
|
|
Layer1HashOffset = reader.ReadInt64();
|
|
|
|
|
Layer1HashSize = reader.ReadInt64();
|
|
|
|
|
Layer2HashOffset = reader.ReadInt64();
|
|
|
|
|
Layer2HashSize = reader.ReadInt64();
|
|
|
|
|
Layer3HashOffset = reader.ReadInt64();
|
|
|
|
|
Layer3HashSize = reader.ReadInt64();
|
|
|
|
|
Field148 = reader.ReadInt64();
|
|
|
|
|
Field150 = reader.ReadInt64();
|
2018-07-25 01:50:41 +02:00
|
|
|
|
DuplexIndex = reader.ReadInt64();
|
2018-07-20 01:31:35 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class RemapHeader
|
|
|
|
|
{
|
|
|
|
|
public string Magic { get; set; }
|
|
|
|
|
public uint MagicNum { get; set; }
|
|
|
|
|
public int MapEntryCount { get; set; }
|
|
|
|
|
public int MapSegmentCount { get; set; }
|
|
|
|
|
public int Field10 { get; set; }
|
|
|
|
|
|
|
|
|
|
public RemapHeader(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
Magic = reader.ReadAscii(4);
|
|
|
|
|
MagicNum = reader.ReadUInt32();
|
|
|
|
|
MapEntryCount = reader.ReadInt32();
|
|
|
|
|
MapSegmentCount = reader.ReadInt32();
|
|
|
|
|
Field10 = reader.ReadInt32();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-23 03:28:31 +02:00
|
|
|
|
public class JournalHeader
|
|
|
|
|
{
|
|
|
|
|
public string Magic { get; }
|
|
|
|
|
public uint MagicNum { get; }
|
|
|
|
|
public long Field8 { get; }
|
|
|
|
|
public long Field10 { get; }
|
|
|
|
|
public long BlockSize { get; }
|
|
|
|
|
public int Field20 { get; }
|
|
|
|
|
public int MappingEntryCount { get; }
|
|
|
|
|
public int Field28 { get; }
|
|
|
|
|
public int Field2C { get; }
|
|
|
|
|
|
|
|
|
|
public JournalHeader(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
Magic = reader.ReadAscii(4);
|
|
|
|
|
MagicNum = reader.ReadUInt32();
|
|
|
|
|
Field8 = reader.ReadInt64();
|
|
|
|
|
Field10 = reader.ReadInt64();
|
|
|
|
|
BlockSize = reader.ReadInt64();
|
|
|
|
|
Field20 = reader.ReadInt32();
|
|
|
|
|
MappingEntryCount = reader.ReadInt32();
|
|
|
|
|
Field28 = reader.ReadInt32();
|
|
|
|
|
Field2C = reader.ReadInt32();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 21:51:52 +02:00
|
|
|
|
public class SaveHeader
|
|
|
|
|
{
|
|
|
|
|
public string Magic { get; }
|
|
|
|
|
public uint MagicNum { get; }
|
|
|
|
|
public int Field8 { get; }
|
|
|
|
|
public int FieldC { get; }
|
|
|
|
|
public int Field10 { get; }
|
|
|
|
|
public int Field14 { get; }
|
|
|
|
|
public long BlockSize { get; }
|
|
|
|
|
public StorageInfo AllocationTableInfo { get; }
|
|
|
|
|
public StorageInfo DataInfo { get; }
|
|
|
|
|
public int DirectoryTableBlock { get; }
|
|
|
|
|
public int FileTableBlock { get; }
|
|
|
|
|
|
|
|
|
|
public SaveHeader(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
Magic = reader.ReadAscii(4);
|
|
|
|
|
MagicNum = reader.ReadUInt32();
|
|
|
|
|
Field8 = reader.ReadInt32();
|
|
|
|
|
FieldC = reader.ReadInt32();
|
|
|
|
|
Field10 = reader.ReadInt32();
|
|
|
|
|
Field14 = reader.ReadInt32();
|
|
|
|
|
BlockSize = reader.ReadInt64();
|
|
|
|
|
AllocationTableInfo = new StorageInfo(reader);
|
|
|
|
|
DataInfo = new StorageInfo(reader);
|
|
|
|
|
DirectoryTableBlock = reader.ReadInt32();
|
|
|
|
|
FileTableBlock = reader.ReadInt32();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class StorageInfo
|
|
|
|
|
{
|
|
|
|
|
public long Offset { get; }
|
|
|
|
|
public int Size { get; }
|
|
|
|
|
public int FieldC { get; }
|
|
|
|
|
|
|
|
|
|
public StorageInfo(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
Offset = reader.ReadInt64();
|
|
|
|
|
Size = reader.ReadInt32();
|
|
|
|
|
FieldC = reader.ReadInt32();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-20 01:31:35 +02:00
|
|
|
|
public class MapEntry
|
|
|
|
|
{
|
|
|
|
|
public long VirtualOffset { get; }
|
|
|
|
|
public long PhysicalOffset { get; }
|
|
|
|
|
public long Size { get; }
|
|
|
|
|
public int Alignment { get; }
|
|
|
|
|
public int StorageType { get; }
|
|
|
|
|
public long VirtualOffsetEnd => VirtualOffset + Size;
|
|
|
|
|
public long PhysicalOffsetEnd => PhysicalOffset + Size;
|
|
|
|
|
internal RemapSegment Segment { get; set; }
|
|
|
|
|
internal MapEntry Next { get; set; }
|
|
|
|
|
|
|
|
|
|
public MapEntry(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
VirtualOffset = reader.ReadInt64();
|
|
|
|
|
PhysicalOffset = reader.ReadInt64();
|
|
|
|
|
Size = reader.ReadInt64();
|
|
|
|
|
Alignment = reader.ReadInt32();
|
|
|
|
|
StorageType = reader.ReadInt32();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|