2018-07-23 03:28:31 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
|
|
|
|
|
namespace libhac.Savefile
|
|
|
|
|
{
|
|
|
|
|
public class JournalStream : Stream
|
|
|
|
|
{
|
|
|
|
|
private long _position;
|
|
|
|
|
private Stream BaseStream { get; }
|
|
|
|
|
public MappingEntry[] Map { get; }
|
|
|
|
|
public int BlockSize { get; }
|
|
|
|
|
private MappingEntry CurrentMapEntry { get; set; }
|
|
|
|
|
|
|
|
|
|
public JournalStream(Stream baseStream, MappingEntry[] map, int blockSize)
|
|
|
|
|
{
|
|
|
|
|
BaseStream = baseStream;
|
|
|
|
|
Map = map;
|
|
|
|
|
BlockSize = blockSize;
|
|
|
|
|
Length = map.Length * BlockSize;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override int Read(byte[] buffer, int offset, int count)
|
|
|
|
|
{
|
|
|
|
|
long remaining = Length - Position;
|
|
|
|
|
if (remaining <= 0) return 0;
|
|
|
|
|
if (remaining < count) count = (int)remaining;
|
|
|
|
|
|
|
|
|
|
var toOutput = count;
|
|
|
|
|
int outPos = offset;
|
|
|
|
|
|
|
|
|
|
while (toOutput > 0)
|
|
|
|
|
{
|
|
|
|
|
var remainInEntry = BlockSize - Position % BlockSize;
|
|
|
|
|
int toRead = (int)Math.Min(toOutput, remainInEntry);
|
|
|
|
|
BaseStream.Read(buffer, outPos, toRead);
|
|
|
|
|
|
|
|
|
|
outPos += toRead;
|
|
|
|
|
toOutput -= toRead;
|
|
|
|
|
Position += toRead;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override long Seek(long offset, SeekOrigin origin)
|
|
|
|
|
{
|
|
|
|
|
switch (origin)
|
|
|
|
|
{
|
|
|
|
|
case SeekOrigin.Begin:
|
|
|
|
|
Position = offset;
|
|
|
|
|
break;
|
|
|
|
|
case SeekOrigin.Current:
|
|
|
|
|
Position += offset;
|
|
|
|
|
break;
|
|
|
|
|
case SeekOrigin.End:
|
|
|
|
|
Position = Length - offset;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Position;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void SetLength(long value) => throw new NotSupportedException();
|
|
|
|
|
public override void Write(byte[] buffer, int offset, int count) => throw new NotSupportedException();
|
|
|
|
|
public override void Flush() => throw new NotSupportedException();
|
|
|
|
|
public override bool CanRead => true;
|
|
|
|
|
public override bool CanSeek => true;
|
|
|
|
|
public override bool CanWrite => false;
|
|
|
|
|
public override long Length { get; }
|
|
|
|
|
public override long Position
|
|
|
|
|
{
|
|
|
|
|
get => _position;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
_position = value;
|
|
|
|
|
if (value >= Length) return;
|
|
|
|
|
var currentBlock = value / BlockSize;
|
|
|
|
|
var blockPos = value % BlockSize;
|
|
|
|
|
CurrentMapEntry = Map[currentBlock];
|
|
|
|
|
BaseStream.Position = CurrentMapEntry.PhysicalIndex * BlockSize + blockPos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-25 00:59:45 +02:00
|
|
|
|
public static MappingEntry[] ReadMappingEntries(Stream mapTable, int count)
|
2018-07-23 03:28:31 +02:00
|
|
|
|
{
|
2018-08-25 00:59:45 +02:00
|
|
|
|
var tableReader = new BinaryReader(mapTable);
|
2018-07-23 03:28:31 +02:00
|
|
|
|
var map = new MappingEntry[count];
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
var entry = new MappingEntry
|
|
|
|
|
{
|
|
|
|
|
VirtualIndex = i,
|
2018-08-25 00:59:45 +02:00
|
|
|
|
PhysicalIndex = tableReader.ReadInt32() & 0x7FFFFFFF
|
2018-07-23 03:28:31 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
map[i] = entry;
|
|
|
|
|
tableReader.BaseStream.Position += 4;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return map;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class MappingEntry
|
|
|
|
|
{
|
|
|
|
|
public int PhysicalIndex { get; set; }
|
|
|
|
|
public int VirtualIndex { get; set; }
|
|
|
|
|
public bool UpdatedPhysical { get; set; }
|
|
|
|
|
public bool UpdatedVirtual { get; set; }
|
|
|
|
|
public bool Unassigned { get; set; }
|
|
|
|
|
}
|
|
|
|
|
}
|