2018-07-24 21:51:52 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
2018-07-20 01:31:35 +02:00
|
|
|
|
using System.Text;
|
2018-08-23 18:28:45 +02:00
|
|
|
|
using libhac.Streams;
|
2018-07-20 01:31:35 +02:00
|
|
|
|
|
|
|
|
|
namespace libhac.Savefile
|
|
|
|
|
{
|
|
|
|
|
public class Savefile
|
|
|
|
|
{
|
|
|
|
|
public Header Header { get; }
|
|
|
|
|
public RemapStream FileRemap { get; }
|
2018-07-23 03:28:31 +02:00
|
|
|
|
public RemapStream MetaRemap { get; }
|
|
|
|
|
private Stream FileStream { get; }
|
|
|
|
|
public JournalStream JournalStream { get; }
|
2018-07-20 01:31:35 +02:00
|
|
|
|
|
2018-07-23 03:28:31 +02:00
|
|
|
|
public byte[] DuplexL1A { get; }
|
|
|
|
|
public byte[] DuplexL1B { get; }
|
|
|
|
|
public byte[] DuplexDataA { get; }
|
|
|
|
|
public byte[] DuplexDataB { get; }
|
|
|
|
|
|
|
|
|
|
public byte[] JournalTable { get; }
|
|
|
|
|
public byte[] JournalBitmapUpdatedPhysical { get; }
|
|
|
|
|
public byte[] JournalBitmapUpdatedVirtual { get; }
|
|
|
|
|
public byte[] JournalBitmapUnassigned { get; }
|
|
|
|
|
public byte[] JournalLayer1Hash { get; }
|
|
|
|
|
public byte[] JournalLayer2Hash { get; }
|
|
|
|
|
public byte[] JournalLayer3Hash { get; }
|
2018-07-25 01:50:41 +02:00
|
|
|
|
public byte[] JournalFat { get; }
|
2018-07-23 03:28:31 +02:00
|
|
|
|
|
2018-07-24 21:51:52 +02:00
|
|
|
|
public FileEntry[] Files { get; private set; }
|
|
|
|
|
private Dictionary<string, FileEntry> FileDict { get; }
|
|
|
|
|
|
2018-07-23 03:28:31 +02:00
|
|
|
|
public Savefile(Stream file, IProgressReport logger = null)
|
2018-07-20 01:31:35 +02:00
|
|
|
|
{
|
2018-07-23 03:28:31 +02:00
|
|
|
|
FileStream = file;
|
2018-07-20 01:31:35 +02:00
|
|
|
|
using (var reader = new BinaryReader(file, Encoding.Default, true))
|
|
|
|
|
{
|
2018-07-23 03:28:31 +02:00
|
|
|
|
Header = new Header(reader, logger);
|
|
|
|
|
var layout = Header.Layout;
|
2018-07-20 01:31:35 +02:00
|
|
|
|
FileRemap = new RemapStream(
|
2018-07-23 03:28:31 +02:00
|
|
|
|
new SubStream(file, layout.FileMapDataOffset, layout.FileMapDataSize),
|
2018-07-20 01:31:35 +02:00
|
|
|
|
Header.FileMapEntries, Header.FileRemap.MapSegmentCount);
|
2018-07-23 03:28:31 +02:00
|
|
|
|
|
|
|
|
|
DuplexL1A = new byte[layout.DuplexL1Size];
|
|
|
|
|
DuplexL1B = new byte[layout.DuplexL1Size];
|
|
|
|
|
DuplexDataA = new byte[layout.DuplexDataSize];
|
|
|
|
|
DuplexDataB = new byte[layout.DuplexDataSize];
|
|
|
|
|
|
|
|
|
|
FileRemap.Position = layout.DuplexL1OffsetA;
|
|
|
|
|
FileRemap.Read(DuplexL1A, 0, DuplexL1A.Length);
|
|
|
|
|
FileRemap.Position = layout.DuplexL1OffsetB;
|
|
|
|
|
FileRemap.Read(DuplexL1B, 0, DuplexL1B.Length);
|
|
|
|
|
FileRemap.Position = layout.DuplexDataOffsetA;
|
|
|
|
|
FileRemap.Read(DuplexDataA, 0, DuplexDataA.Length);
|
|
|
|
|
FileRemap.Position = layout.DuplexDataOffsetB;
|
|
|
|
|
FileRemap.Read(DuplexDataB, 0, DuplexDataB.Length);
|
|
|
|
|
|
2018-07-25 01:50:41 +02:00
|
|
|
|
var duplexDataOffset = layout.DuplexIndex == 0 ? layout.DuplexDataOffsetA : layout.DuplexDataOffsetB;
|
|
|
|
|
var duplexData = new SubStream(FileRemap, duplexDataOffset, layout.DuplexDataSize);
|
2018-07-23 03:28:31 +02:00
|
|
|
|
MetaRemap = new RemapStream(duplexData, Header.MetaMapEntries, Header.MetaRemap.MapSegmentCount);
|
|
|
|
|
|
|
|
|
|
JournalTable = new byte[layout.JournalTableSize];
|
|
|
|
|
JournalBitmapUpdatedPhysical = new byte[layout.JournalBitmapUpdatedPhysicalSize];
|
|
|
|
|
JournalBitmapUpdatedVirtual = new byte[layout.JournalBitmapUpdatedVirtualSize];
|
|
|
|
|
JournalBitmapUnassigned = new byte[layout.JournalBitmapUnassignedSize];
|
|
|
|
|
JournalLayer1Hash = new byte[layout.Layer1HashSize];
|
|
|
|
|
JournalLayer2Hash = new byte[layout.Layer2HashSize];
|
|
|
|
|
JournalLayer3Hash = new byte[layout.Layer3HashSize];
|
2018-07-25 01:50:41 +02:00
|
|
|
|
JournalFat = new byte[layout.Field150];
|
2018-07-23 03:28:31 +02:00
|
|
|
|
|
|
|
|
|
MetaRemap.Position = layout.JournalTableOffset;
|
|
|
|
|
MetaRemap.Read(JournalTable, 0, JournalTable.Length);
|
|
|
|
|
MetaRemap.Position = layout.JournalBitmapUpdatedPhysicalOffset;
|
|
|
|
|
MetaRemap.Read(JournalBitmapUpdatedPhysical, 0, JournalBitmapUpdatedPhysical.Length);
|
|
|
|
|
MetaRemap.Position = layout.JournalBitmapUpdatedVirtualOffset;
|
|
|
|
|
MetaRemap.Read(JournalBitmapUpdatedVirtual, 0, JournalBitmapUpdatedVirtual.Length);
|
|
|
|
|
MetaRemap.Position = layout.JournalBitmapUnassignedOffset;
|
|
|
|
|
MetaRemap.Read(JournalBitmapUnassigned, 0, JournalBitmapUnassigned.Length);
|
|
|
|
|
MetaRemap.Position = layout.Layer1HashOffset;
|
|
|
|
|
MetaRemap.Read(JournalLayer1Hash, 0, JournalLayer1Hash.Length);
|
|
|
|
|
MetaRemap.Position = layout.Layer2HashOffset;
|
|
|
|
|
MetaRemap.Read(JournalLayer2Hash, 0, JournalLayer2Hash.Length);
|
|
|
|
|
MetaRemap.Position = layout.Layer3HashOffset;
|
|
|
|
|
MetaRemap.Read(JournalLayer3Hash, 0, JournalLayer3Hash.Length);
|
|
|
|
|
MetaRemap.Position = layout.Field148;
|
2018-07-25 01:50:41 +02:00
|
|
|
|
MetaRemap.Read(JournalFat, 0, JournalFat.Length);
|
2018-07-23 03:28:31 +02:00
|
|
|
|
|
|
|
|
|
var journalMap = JournalStream.ReadMappingEntries(JournalTable, JournalBitmapUpdatedPhysical,
|
|
|
|
|
JournalBitmapUpdatedVirtual, JournalBitmapUnassigned, Header.Journal.MappingEntryCount);
|
|
|
|
|
|
|
|
|
|
var journalData = new SubStream(FileRemap, layout.JournalDataOffset,
|
|
|
|
|
layout.JournalDataSizeB + layout.SizeReservedArea);
|
2018-07-24 21:51:52 +02:00
|
|
|
|
JournalStream = new JournalStream(journalData, journalMap, (int)Header.Journal.BlockSize);
|
|
|
|
|
ReadFileInfo();
|
2018-08-22 16:57:47 +02:00
|
|
|
|
Dictionary<string, FileEntry> dictionary = new Dictionary<string, FileEntry>();
|
|
|
|
|
foreach (FileEntry entry in Files)
|
|
|
|
|
{
|
|
|
|
|
dictionary[entry.FullPath] = entry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FileDict = dictionary;
|
2018-07-24 21:51:52 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream OpenFile(string filename)
|
|
|
|
|
{
|
|
|
|
|
if (!FileDict.TryGetValue(filename, out FileEntry file))
|
|
|
|
|
{
|
|
|
|
|
throw new FileNotFoundException();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return OpenFile(file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream OpenFile(FileEntry file)
|
|
|
|
|
{
|
2018-07-24 22:06:12 +02:00
|
|
|
|
return new SubStream(JournalStream, file.Offset, file.Size);
|
2018-07-24 21:51:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool FileExists(string filename) => FileDict.ContainsKey(filename);
|
|
|
|
|
|
|
|
|
|
private void ReadFileInfo()
|
|
|
|
|
{
|
|
|
|
|
var blockSize = Header.Save.BlockSize;
|
|
|
|
|
var dirOffset = Header.Save.DirectoryTableBlock * blockSize;
|
|
|
|
|
var fileOffset = Header.Save.FileTableBlock * blockSize;
|
|
|
|
|
|
|
|
|
|
FileEntry[] dirEntries;
|
|
|
|
|
FileEntry[] fileEntries;
|
|
|
|
|
using (var reader = new BinaryReader(JournalStream, Encoding.Default, true))
|
|
|
|
|
{
|
|
|
|
|
JournalStream.Position = dirOffset;
|
|
|
|
|
dirEntries = ReadFileEntries(reader);
|
|
|
|
|
|
|
|
|
|
JournalStream.Position = fileOffset;
|
|
|
|
|
fileEntries = ReadFileEntries(reader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var dir in dirEntries)
|
|
|
|
|
{
|
|
|
|
|
if (dir.NextIndex != 0) dir.Next = dirEntries[dir.NextIndex];
|
|
|
|
|
if (dir.ParentDirIndex != 0 && dir.ParentDirIndex < dirEntries.Length)
|
|
|
|
|
dir.ParentDir = dirEntries[dir.ParentDirIndex];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var file in fileEntries)
|
|
|
|
|
{
|
|
|
|
|
if (file.NextIndex != 0) file.Next = fileEntries[file.NextIndex];
|
|
|
|
|
if (file.ParentDirIndex != 0 && file.ParentDirIndex < dirEntries.Length)
|
|
|
|
|
file.ParentDir = dirEntries[file.ParentDirIndex];
|
2018-07-24 22:06:12 +02:00
|
|
|
|
file.Offset = file.BlockIndex < 0 ? 0 : file.BlockIndex * blockSize;
|
2018-07-24 21:51:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Files = new FileEntry[fileEntries.Length - 2];
|
|
|
|
|
Array.Copy(fileEntries, 2, Files, 0, Files.Length);
|
|
|
|
|
|
|
|
|
|
FileEntry.ResolveFilenames(Files);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private FileEntry[] ReadFileEntries(BinaryReader reader)
|
|
|
|
|
{
|
|
|
|
|
var count = reader.ReadInt32();
|
|
|
|
|
JournalStream.Position -= 4;
|
|
|
|
|
|
|
|
|
|
var entries = new FileEntry[count];
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
entries[i] = new FileEntry(reader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return entries;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static class SavefileExtensions
|
|
|
|
|
{
|
|
|
|
|
public static void Extract(this Savefile save, string outDir, IProgressReport logger = null)
|
|
|
|
|
{
|
|
|
|
|
foreach (var file in save.Files)
|
|
|
|
|
{
|
|
|
|
|
var stream = save.OpenFile(file);
|
|
|
|
|
var outName = outDir + file.FullPath;
|
|
|
|
|
var dir = Path.GetDirectoryName(outName);
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(dir)) Directory.CreateDirectory(dir);
|
|
|
|
|
|
|
|
|
|
using (var outFile = new FileStream(outName, FileMode.Create, FileAccess.ReadWrite))
|
|
|
|
|
{
|
|
|
|
|
logger?.LogMessage(file.FullPath);
|
|
|
|
|
stream.CopyStream(outFile, stream.Length, logger);
|
|
|
|
|
}
|
2018-07-20 01:31:35 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|