Rewrite IndirectStorage

This commit is contained in:
Alex Barney 2020-06-25 14:01:45 -07:00
parent c2247e583f
commit 33af34cefc
8 changed files with 332 additions and 83 deletions

View file

@ -50,7 +50,14 @@ Module,DescriptionStart,DescriptionEnd,Name,Summary
2,4000,4999,DataCorrupted, 2,4000,4999,DataCorrupted,
2,4001,4299,RomCorrupted, 2,4001,4299,RomCorrupted,
2,4023,,InvalidIndirectStorageSource,
2,4021,4029,IndirectStorageCorrupted,
2,4022,,InvalidIndirectEntryOffset,
2,4023,,InvalidIndirectEntryStorageIndex,
2,4024,,InvalidIndirectStorageSize,
2,4025,,InvalidIndirectVirtualOffset,
2,4026,,InvalidIndirectPhysicalOffset,
2,4027,,InvalidIndirectStorageIndex,
2,4031,4039,BucketTreeCorrupted, 2,4031,4039,BucketTreeCorrupted,
2,4032,,InvalidBucketTreeSignature, 2,4032,,InvalidBucketTreeSignature,

1 Module DescriptionStart DescriptionEnd Name Summary
50 2 4035 4026 InvalidBucketTreeNodeOffset InvalidIndirectPhysicalOffset
51 2 4036 4027 InvalidBucketTreeEntryOffset InvalidIndirectStorageIndex
52 2 4037 4031 4039 InvalidBucketTreeEntrySetOffset BucketTreeCorrupted
53 2 4038 4032 InvalidBucketTreeNodeIndex InvalidBucketTreeSignature
54 2 4033 InvalidBucketTreeEntryCount
55 2 4034 InvalidBucketTreeNodeEntryCount
56 2 4035 InvalidBucketTreeNodeOffset
57 2 4036 InvalidBucketTreeEntryOffset
58 2 4037 InvalidBucketTreeEntrySetOffset
59 2 4038 InvalidBucketTreeNodeIndex
60 2 4039 InvalidBucketTreeVirtualOffset
61 2 4039 4241 4259 InvalidBucketTreeVirtualOffset RomHostFileSystemCorrupted
62 2 4241 4242 4259 RomHostFileSystemCorrupted RomHostEntryCorrupted
63 2 4242 4243 RomHostEntryCorrupted RomHostFileDataCorrupted

View file

@ -27,5 +27,20 @@ namespace LibHac.Diag
throw new LibHacException("Not-null assertion failed."); throw new LibHacException("Not-null assertion failed.");
} }
} }
[Conditional("DEBUG")]
public static void InRange(int value, int lowerInclusive, int upperExclusive)
{
InRange((long)value, lowerInclusive, upperExclusive);
}
[Conditional("DEBUG")]
public static void InRange(long value, long lowerInclusive, long upperExclusive)
{
if (value < lowerInclusive || value >= upperExclusive)
{
throw new LibHacException($"Value {value} is not in the range {lowerInclusive} to {upperExclusive}");
}
}
} }
} }

View file

@ -132,8 +132,8 @@ namespace LibHac.Fs
protected abstract Result DoRead(long offset, Span<byte> destination); protected abstract Result DoRead(long offset, Span<byte> destination);
protected abstract Result DoWrite(long offset, ReadOnlySpan<byte> source); protected abstract Result DoWrite(long offset, ReadOnlySpan<byte> source);
protected abstract Result DoFlush(); protected abstract Result DoFlush();
protected abstract Result DoGetSize(out long size);
protected abstract Result DoSetSize(long size); protected abstract Result DoSetSize(long size);
protected abstract Result DoGetSize(out long size);
protected virtual Result DoOperateRange(Span<byte> outBuffer, OperationId operationId, long offset, long size, protected virtual Result DoOperateRange(Span<byte> outBuffer, OperationId operationId, long offset, long size,
ReadOnlySpan<byte> inBuffer) ReadOnlySpan<byte> inBuffer)

View file

@ -114,8 +114,20 @@ namespace LibHac.Fs
public static Result.Base DataCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4000, 4999); } public static Result.Base DataCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4000, 4999); }
/// <summary>Error code: 2002-4001; Range: 4001-4299; Inner value: 0x1f4202</summary> /// <summary>Error code: 2002-4001; Range: 4001-4299; Inner value: 0x1f4202</summary>
public static Result.Base RomCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4001, 4299); } public static Result.Base RomCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4001, 4299); }
/// <summary>Error code: 2002-4021; Range: 4021-4029; Inner value: 0x1f6a02</summary>
public static Result.Base IndirectStorageCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4021, 4029); }
/// <summary>Error code: 2002-4022; Inner value: 0x1f6c02</summary>
public static Result.Base InvalidIndirectEntryOffset => new Result.Base(ModuleFs, 4022);
/// <summary>Error code: 2002-4023; Inner value: 0x1f6e02</summary> /// <summary>Error code: 2002-4023; Inner value: 0x1f6e02</summary>
public static Result.Base InvalidIndirectStorageSource => new Result.Base(ModuleFs, 4023); public static Result.Base InvalidIndirectEntryStorageIndex => new Result.Base(ModuleFs, 4023);
/// <summary>Error code: 2002-4024; Inner value: 0x1f7002</summary>
public static Result.Base InvalidIndirectStorageSize => new Result.Base(ModuleFs, 4024);
/// <summary>Error code: 2002-4025; Inner value: 0x1f7202</summary>
public static Result.Base InvalidIndirectVirtualOffset => new Result.Base(ModuleFs, 4025);
/// <summary>Error code: 2002-4026; Inner value: 0x1f7402</summary>
public static Result.Base InvalidIndirectPhysicalOffset => new Result.Base(ModuleFs, 4026);
/// <summary>Error code: 2002-4027; Inner value: 0x1f7602</summary>
public static Result.Base InvalidIndirectStorageIndex => new Result.Base(ModuleFs, 4027);
/// <summary>Error code: 2002-4031; Range: 4031-4039; Inner value: 0x1f7e02</summary> /// <summary>Error code: 2002-4031; Range: 4031-4039; Inner value: 0x1f7e02</summary>
public static Result.Base BucketTreeCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4031, 4039); } public static Result.Base BucketTreeCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4031, 4039); }

View file

@ -105,6 +105,16 @@ namespace LibHac.FsSystem
public long GetEnd() => EndOffset; public long GetEnd() => EndOffset;
public long GetSize() => EndOffset - StartOffset; public long GetSize() => EndOffset - StartOffset;
public bool Includes(long offset)
{
return StartOffset <= offset && offset < EndOffset;
}
public bool Includes(long offset, long size)
{
return size > 0 && StartOffset <= offset && size <= EndOffset - offset;
}
public Result Find(ref Visitor visitor, long virtualAddress) public Result Find(ref Visitor visitor, long virtualAddress)
{ {
Assert.AssertTrue(IsInitialized()); Assert.AssertTrue(IsInitialized());
@ -382,6 +392,11 @@ namespace LibHac.FsSystem
return Result.Success; return Result.Success;
} }
public void Dispose()
{
// todo: try using shared arrays
}
public bool IsValid() => EntryIndex >= 0; public bool IsValid() => EntryIndex >= 0;
public bool CanMoveNext() public bool CanMoveNext()

View file

@ -1,72 +1,183 @@
using System; using System;
using System.Collections.Generic; using System.Runtime.CompilerServices;
using System.Linq; using System.Runtime.InteropServices;
using LibHac.Common;
using LibHac.Diag;
using LibHac.Fs; using LibHac.Fs;
namespace LibHac.FsSystem namespace LibHac.FsSystem
{ {
public class IndirectStorage : IStorage public class IndirectStorage : IStorage
{ {
private List<RelocationEntry> RelocationEntries { get; } public static readonly int StorageCount = 2;
private List<long> RelocationOffsets { get; } public static readonly int NodeSize = 1024 * 16;
private List<IStorage> Sources { get; } = new List<IStorage>(); private BucketTree2 Table { get; } = new BucketTree2();
private BucketTree<RelocationEntry> BucketTree { get; } private SubStorage2[] DataStorage { get; } = new SubStorage2[StorageCount];
private long Length { get; }
private bool LeaveOpen { get; }
public IndirectStorage(IStorage bucketTreeData, bool leaveOpen, params IStorage[] sources) [StructLayout(LayoutKind.Sequential, Size = 0x14)]
public struct Entry
{ {
Sources.AddRange(sources); private long VirtualOffset;
private long PhysicalOffset;
public int StorageIndex;
LeaveOpen = leaveOpen; public void SetVirtualOffset(long offset) => VirtualOffset = offset;
public long GetVirtualOffset() => VirtualOffset;
BucketTree = new BucketTree<RelocationEntry>(bucketTreeData); public void SetPhysicalOffset(long offset) => PhysicalOffset = offset;
public long GetPhysicalOffset() => PhysicalOffset;
RelocationEntries = BucketTree.GetEntryList();
RelocationOffsets = RelocationEntries.Select(x => x.Offset).ToList();
Length = BucketTree.BucketOffsets.OffsetEnd;
} }
protected override Result DoRead(long offset, Span<byte> destination) public static long QueryHeaderStorageSize() => BucketTree2.QueryHeaderStorageSize();
public static long QueryNodeStorageSize(int entryCount) =>
BucketTree2.QueryNodeStorageSize(NodeSize, Unsafe.SizeOf<Entry>(), entryCount);
public static long QueryEntryStorageSize(int entryCount) =>
BucketTree2.QueryEntryStorageSize(NodeSize, Unsafe.SizeOf<Entry>(), entryCount);
public bool IsInitialized() => Table.IsInitialized();
public Result Initialize(SubStorage2 tableStorage)
{ {
RelocationEntry entry = GetRelocationEntry(offset); // Read and verify the bucket tree header.
// note: skip init
var header = new BucketTree2.Header();
if (entry.SourceIndex > Sources.Count) Result rc = tableStorage.Read(0, SpanHelpers.AsByteSpan(ref header));
{
return ResultFs.InvalidIndirectStorageSource.Log();
}
long inPos = offset;
int outPos = 0;
int remaining = destination.Length;
while (remaining > 0)
{
long entryPos = inPos - entry.Offset;
int bytesToRead = (int)Math.Min(entry.OffsetEnd - inPos, remaining);
Result rc = Sources[entry.SourceIndex].Read(entry.SourceOffset + entryPos, destination.Slice(outPos, bytesToRead));
if (rc.IsFailure()) return rc; if (rc.IsFailure()) return rc;
outPos += bytesToRead; rc = header.Verify();
inPos += bytesToRead; if (rc.IsFailure()) return rc;
remaining -= bytesToRead;
if (inPos >= entry.OffsetEnd) // Determine extents.
long nodeStorageSize = QueryNodeStorageSize(header.EntryCount);
long entryStorageSize = QueryEntryStorageSize(header.EntryCount);
long nodeStorageOffset = QueryHeaderStorageSize();
long entryStorageOffset = nodeStorageOffset + nodeStorageSize;
// Initialize.
var nodeStorage = new SubStorage2(tableStorage, nodeStorageOffset, nodeStorageSize);
var entryStorage = new SubStorage2(tableStorage, entryStorageOffset, entryStorageSize);
return Initialize(nodeStorage, entryStorage, header.EntryCount);
}
public Result Initialize(SubStorage2 nodeStorage, SubStorage2 entryStorage, int entryCount)
{ {
entry = entry.Next; return Table.Initialize(nodeStorage, entryStorage, NodeSize, Unsafe.SizeOf<Entry>(), entryCount);
}
public void SetStorage(int index, SubStorage2 storage)
{
Assert.InRange(index, 0, StorageCount);
DataStorage[index] = storage;
}
public void SetStorage(int index, IStorage storage, long offset, long size)
{
Assert.InRange(index, 0, StorageCount);
DataStorage[index] = new SubStorage2(storage, offset, size);
}
public Result GetEntryList(Span<Entry> entryBuffer, out int outputEntryCount, long offset, long size)
{
// Validate pre-conditions
Assert.AssertTrue(offset >= 0);
Assert.AssertTrue(size >= 0);
Assert.AssertTrue(IsInitialized());
// Clear the out count
outputEntryCount = 0;
// Succeed if there's no range
if (size == 0)
return Result.Success;
// Check that our range is valid
if (!Table.Includes(offset, size))
return ResultFs.OutOfRange.Log();
// Find the offset in our tree
var visitor = new BucketTree2.Visitor();
Result rc = Table.Find(ref visitor, offset);
if (rc.IsFailure()) return rc;
long entryOffset = visitor.Get<Entry>().GetVirtualOffset();
if (entryOffset > 0 || !Table.Includes(entryOffset))
return ResultFs.InvalidIndirectEntryOffset.Log();
// Prepare to loop over entries
long endOffset = offset + size;
int count = 0;
ref Entry currentEntry = ref visitor.Get<Entry>();
while (currentEntry.GetVirtualOffset() < endOffset)
{
// Try to write the entry to the out list
if (entryBuffer.Length != 0)
{
if (count >= entryBuffer.Length)
break;
entryBuffer[count] = currentEntry;
}
count++;
// Advance
if (visitor.CanMoveNext())
{
rc = visitor.MoveNext();
if (rc.IsFailure()) return rc;
currentEntry = ref visitor.Get<Entry>();
}
else
{
break;
} }
} }
// Write the entry count
outputEntryCount = count;
return Result.Success; return Result.Success;
} }
protected override unsafe Result DoRead(long offset, Span<byte> destination)
{
// Validate pre-conditions
Assert.AssertTrue(offset >= 0);
Assert.AssertTrue(IsInitialized());
// Succeed if there's nothing to read
if (destination.Length == 0)
return Result.Success;
// Pin and recreate the span because C# can't use byref-like types in a closure
int bufferSize = destination.Length;
fixed (byte* pBuffer = destination)
{
// Copy the pointer to workaround CS1764.
// OperatePerEntry won't store the delegate anywhere, so it should be safe
byte* pBuffer2 = pBuffer;
Result Operate(IStorage storage, long dataOffset, long currentOffset, long currentSize)
{
var buffer = new Span<byte>(pBuffer2, bufferSize);
return storage.Read(dataOffset,
buffer.Slice((int)(currentOffset - offset), (int)currentSize));
}
return OperatePerEntry(offset, destination.Length, Operate);
}
}
protected override Result DoWrite(long offset, ReadOnlySpan<byte> source) protected override Result DoWrite(long offset, ReadOnlySpan<byte> source)
{ {
return ResultFs.UnsupportedOperationInIndirectStorageSetSize.Log(); return ResultFs.UnsupportedOperationInIndirectStorageWrite.Log();
} }
protected override Result DoFlush() protected override Result DoFlush()
@ -74,36 +185,117 @@ namespace LibHac.FsSystem
return Result.Success; return Result.Success;
} }
protected override Result DoGetSize(out long size)
{
size = Length;
return Result.Success;
}
protected override Result DoSetSize(long size) protected override Result DoSetSize(long size)
{ {
return ResultFs.UnsupportedOperationInIndirectStorageSetSize.Log(); return ResultFs.UnsupportedOperationInIndirectStorageSetSize.Log();
} }
protected override void Dispose(bool disposing) protected override Result DoGetSize(out long size)
{ {
if (disposing) size = Table.GetEnd();
{ return Result.Success;
if (!LeaveOpen && Sources != null)
{
foreach (IStorage storage in Sources)
{
storage?.Dispose();
}
}
}
} }
private RelocationEntry GetRelocationEntry(long offset) private delegate Result OperateFunc(IStorage storage, long dataOffset, long currentOffset, long currentSize);
private Result OperatePerEntry(long offset, long size, OperateFunc func)
{ {
int index = RelocationOffsets.BinarySearch(offset); // Validate preconditions
if (index < 0) index = ~index - 1; Assert.AssertTrue(offset >= 0);
return RelocationEntries[index]; Assert.AssertTrue(size >= 0);
Assert.AssertTrue(IsInitialized());
// Succeed if there's nothing to operate on
if (size == 0)
return Result.Success;
// Validate arguments
if (!Table.Includes(offset, size))
return ResultFs.OutOfRange.Log();
// Find the offset in our tree
var visitor = new BucketTree2.Visitor();
Result rc = Table.Find(ref visitor, offset);
if (rc.IsFailure()) return rc;
long entryOffset = visitor.Get<Entry>().GetVirtualOffset();
if (entryOffset < 0 || !Table.Includes(entryOffset))
return ResultFs.InvalidIndirectEntryStorageIndex.Log();
// Prepare to operate in chunks
long currentOffset = offset;
long endOffset = offset + size;
while (currentOffset < endOffset)
{
// Get the current entry
var currentEntry = visitor.Get<Entry>();
// Get and validate the entry's offset
long currentEntryOffset = currentEntry.GetVirtualOffset();
if (currentEntryOffset > currentOffset)
return ResultFs.InvalidIndirectEntryOffset.Log();
// Validate the storage index
if (currentEntry.StorageIndex < 0 || currentEntry.StorageIndex >= StorageCount)
return ResultFs.InvalidIndirectEntryStorageIndex.Log();
// todo: Implement continuous reading
// Get and validate the next entry offset
long nextEntryOffset;
if (visitor.CanMoveNext())
{
rc = visitor.MoveNext();
if (rc.IsFailure()) return rc;
nextEntryOffset = visitor.Get<Entry>().GetVirtualOffset();
if (!Table.Includes(nextEntryOffset))
return ResultFs.InvalidIndirectEntryOffset.Log();
}
else
{
nextEntryOffset = Table.GetEnd();
}
if (currentOffset >= nextEntryOffset)
return ResultFs.InvalidIndirectEntryOffset.Log();
// Get the offset of the entry in the data we read
long dataOffset = currentOffset - currentEntryOffset;
long dataSize = nextEntryOffset - currentEntryOffset - dataOffset;
Assert.AssertTrue(dataSize > 0);
// Determine how much is left
long remainingSize = endOffset - currentOffset;
long currentSize = Math.Min(remainingSize, dataSize);
Assert.AssertTrue(currentSize <= size);
{
SubStorage2 currentStorage = DataStorage[currentEntry.StorageIndex];
// Get the current data storage's size.
rc = currentStorage.GetSize(out long currentDataStorageSize);
if (rc.IsFailure()) return rc;
// Ensure that we remain within range.
long currentEntryPhysicalOffset = currentEntry.GetPhysicalOffset();
if (currentEntryPhysicalOffset < 0 || currentEntryPhysicalOffset > currentDataStorageSize)
return ResultFs.IndirectStorageCorrupted.Log();
if (currentDataStorageSize < currentEntryPhysicalOffset + dataOffset + currentSize)
return ResultFs.IndirectStorageCorrupted.Log();
rc = func(currentStorage, currentEntryPhysicalOffset + dataOffset, currentOffset, currentSize);
if (rc.IsFailure()) return rc;
}
currentOffset += currentSize;
}
return Result.Success;
} }
} }
} }

View file

@ -2,6 +2,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Diagnostics; using System.Diagnostics;
using System.IO; using System.IO;
using LibHac.Common;
using LibHac.Fs; using LibHac.Fs;
using LibHac.Fs.Fsa; using LibHac.Fs.Fsa;
using LibHac.FsSystem.RomFs; using LibHac.FsSystem.RomFs;
@ -214,6 +215,9 @@ namespace LibHac.FsSystem.NcaUtils
IStorage patchStorage = patchNca.OpenRawStorage(index); IStorage patchStorage = patchNca.OpenRawStorage(index);
IStorage baseStorage = SectionExists(index) ? OpenRawStorage(index) : new NullStorage(); IStorage baseStorage = SectionExists(index) ? OpenRawStorage(index) : new NullStorage();
patchStorage.GetSize(out long patchSize).ThrowIfFailure();
baseStorage.GetSize(out long baseSize).ThrowIfFailure();
NcaFsHeader header = patchNca.Header.GetFsHeader(index); NcaFsHeader header = patchNca.Header.GetFsHeader(index);
NcaFsPatchInfo patchInfo = header.GetPatchInfo(); NcaFsPatchInfo patchInfo = header.GetPatchInfo();
@ -222,9 +226,24 @@ namespace LibHac.FsSystem.NcaUtils
return patchStorage; return patchStorage;
} }
IStorage relocationTableStorage = patchStorage.Slice(patchInfo.RelocationTreeOffset, patchInfo.RelocationTreeSize); var treeHeader = new BucketTree2.Header();
patchInfo.RelocationTreeHeader.CopyTo(SpanHelpers.AsByteSpan(ref treeHeader));
long nodeStorageSize = IndirectStorage.QueryNodeStorageSize(treeHeader.EntryCount);
long entryStorageSize = IndirectStorage.QueryEntryStorageSize(treeHeader.EntryCount);
return new IndirectStorage(relocationTableStorage, true, baseStorage, patchStorage); var relocationTableStorage = new SubStorage2(patchStorage, patchInfo.RelocationTreeOffset, patchInfo.RelocationTreeSize);
var cachedTableStorage = new CachedStorage(relocationTableStorage, IndirectStorage.NodeSize, 4, true);
var tableNodeStorage = new SubStorage2(cachedTableStorage, 0, nodeStorageSize);
var tableEntryStorage = new SubStorage2(cachedTableStorage, nodeStorageSize, entryStorageSize);
var storage = new IndirectStorage();
storage.Initialize(tableNodeStorage, tableEntryStorage, treeHeader.EntryCount).ThrowIfFailure();
storage.SetStorage(0, baseStorage, 0, baseSize);
storage.SetStorage(1, patchStorage, 0, patchSize);
return storage;
} }
public IStorage OpenStorage(int index, IntegrityCheckLevel integrityCheckLevel) public IStorage OpenStorage(int index, IntegrityCheckLevel integrityCheckLevel)

View file

@ -48,20 +48,9 @@ namespace LibHac
return true; return true;
} }
public static bool SpansEqual<T>(Span<T> a1, Span<T> a2) public static bool SpansEqual<T>(Span<T> a1, Span<T> a2) where T : IEquatable<T>
{ {
if (a1 == a2) return true; return a1.SequenceEqual(a2);
if (a1.Length != a2.Length) return false;
for (int i = 0; i < a1.Length; i++)
{
if (!a1[i].Equals(a2[i]))
{
return false;
}
}
return true;
} }
public static ReadOnlySpan<byte> GetUtf8Bytes(string value) public static ReadOnlySpan<byte> GetUtf8Bytes(string value)