Add a bucket tree builder

This commit is contained in:
Alex Barney 2020-06-17 20:56:40 -07:00
parent 0c06d9e0b3
commit 9589f681a6
6 changed files with 345 additions and 10 deletions

View file

@ -60,7 +60,7 @@ Module,DescriptionStart,DescriptionEnd,Name,Summary
2,4036,,InvalidBucketTreeEntryOffset, 2,4036,,InvalidBucketTreeEntryOffset,
2,4037,,InvalidBucketTreeEntrySetOffset, 2,4037,,InvalidBucketTreeEntrySetOffset,
2,4038,,InvalidBucketTreeNodeIndex, 2,4038,,InvalidBucketTreeNodeIndex,
2,4039,,BucketTreeEntryNotFound, 2,4039,,InvalidBucketTreeVirtualOffset,
2,4241,4259,RomHostFileSystemCorrupted, 2,4241,4259,RomHostFileSystemCorrupted,
2,4242,,RomHostEntryCorrupted, 2,4242,,RomHostEntryCorrupted,

1 Module DescriptionStart DescriptionEnd Name Summary
60 2 4301 4499 SaveDataCorrupted
61 2 4302 UnsupportedSaveVersion
62 2 4303 InvalidSaveDataEntryType
63 2 4315 InvalidSaveDataHeader
64 2 4362 InvalidSaveDataIvfcMagic
65 2 4363 InvalidSaveDataIvfcHashValidationBit
66 2 4364 InvalidSaveDataIvfcHash

View file

@ -57,5 +57,34 @@ namespace LibHac.Common
{ {
return CreateReadOnlySpan(ref Unsafe.As<T, byte>(ref reference), Unsafe.SizeOf<T>()); return CreateReadOnlySpan(ref Unsafe.As<T, byte>(ref reference), Unsafe.SizeOf<T>());
} }
// All AsStruct methods do bounds checks on the input
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static ref T AsStruct<T>(Span<byte> span) where T : unmanaged
{
return ref MemoryMarshal.Cast<byte, T>(span)[0];
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static ref readonly T AsReadOnlyStruct<T>(ReadOnlySpan<byte> span) where T : unmanaged
{
return ref MemoryMarshal.Cast<byte, T>(span)[0];
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static ref TTo AsStruct<TFrom, TTo>(Span<TFrom> span)
where TFrom : unmanaged
where TTo : unmanaged
{
return ref MemoryMarshal.Cast<TFrom, TTo>(span)[0];
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static ref readonly TTo AsStruct<TFrom, TTo>(ReadOnlySpan<TFrom> span)
where TFrom : unmanaged
where TTo : unmanaged
{
return ref MemoryMarshal.Cast<TFrom, TTo>(span)[0];
}
} }
} }

View file

@ -18,5 +18,14 @@ namespace LibHac.Diag
throw new LibHacException($"Assertion failed: {message}"); throw new LibHacException($"Assertion failed: {message}");
} }
[Conditional("DEBUG")]
public static void NotNull<T>([NotNull] T item) where T : class
{
if (!(item is null))
{
throw new LibHacException("Not-null assertion failed.");
}
}
} }
} }

View file

@ -134,7 +134,7 @@ namespace LibHac.Fs
/// <summary>Error code: 2002-4038; Inner value: 0x1f8c02</summary> /// <summary>Error code: 2002-4038; Inner value: 0x1f8c02</summary>
public static Result.Base InvalidBucketTreeNodeIndex => new Result.Base(ModuleFs, 4038); public static Result.Base InvalidBucketTreeNodeIndex => new Result.Base(ModuleFs, 4038);
/// <summary>Error code: 2002-4039; Inner value: 0x1f8e02</summary> /// <summary>Error code: 2002-4039; Inner value: 0x1f8e02</summary>
public static Result.Base BucketTreeEntryNotFound => new Result.Base(ModuleFs, 4039); public static Result.Base InvalidBucketTreeVirtualOffset => new Result.Base(ModuleFs, 4039);
/// <summary>Error code: 2002-4241; Range: 4241-4259; Inner value: 0x212202</summary> /// <summary>Error code: 2002-4241; Range: 4241-4259; Inner value: 0x212202</summary>
public static Result.Base RomHostFileSystemCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4241, 4259); } public static Result.Base RomHostFileSystemCorrupted { [MethodImpl(MethodImplOptions.AggressiveInlining)] get => new Result.Base(ModuleFs, 4241, 4259); }

View file

@ -8,7 +8,7 @@ using LibHac.Fs;
namespace LibHac.FsSystem namespace LibHac.FsSystem
{ {
public class BucketTree2 public partial class BucketTree2
{ {
private const uint ExpectedMagic = 0x52544B42; // BKTR private const uint ExpectedMagic = 0x52544B42; // BKTR
private const int MaxVersion = 1; private const int MaxVersion = 1;
@ -103,6 +103,10 @@ namespace LibHac.FsSystem
public bool IsInitialized() => NodeSize > 0; public bool IsInitialized() => NodeSize > 0;
public bool IsEmpty() => EntrySize == 0; public bool IsEmpty() => EntrySize == 0;
public long GetStart() => StartOffset;
public long GetEnd() => EndOffset;
public long GetSize() => EndOffset - StartOffset;
public Result Find(ref Visitor visitor, long virtualAddress) public Result Find(ref Visitor visitor, long virtualAddress)
{ {
Assert.AssertTrue(IsInitialized()); Assert.AssertTrue(IsInitialized());
@ -165,7 +169,7 @@ namespace LibHac.FsSystem
return Util.DivideByRoundUp(entryCount, entryCountPerNode); return Util.DivideByRoundUp(entryCount, entryCountPerNode);
} }
private static int GetNodeL2Count(long nodeSize, long entrySize, int entryCount) public static int GetNodeL2Count(long nodeSize, long entrySize, int entryCount)
{ {
int offsetCountPerNode = GetOffsetCount(nodeSize); int offsetCountPerNode = GetOffsetCount(nodeSize);
int entrySetCount = GetEntrySetCount(nodeSize, entrySize, entryCount); int entrySetCount = GetEntrySetCount(nodeSize, entrySize, entryCount);
@ -312,9 +316,9 @@ namespace LibHac.FsSystem
public readonly ref struct BucketTreeNode<TEntry> where TEntry : unmanaged public readonly ref struct BucketTreeNode<TEntry> where TEntry : unmanaged
{ {
private readonly ReadOnlySpan<byte> _buffer; private readonly Span<byte> _buffer;
public BucketTreeNode(ReadOnlySpan<byte> buffer) public BucketTreeNode(Span<byte> buffer)
{ {
_buffer = buffer; _buffer = buffer;
@ -324,7 +328,8 @@ namespace LibHac.FsSystem
public int GetCount() => GetHeader().Count; public int GetCount() => GetHeader().Count;
public ReadOnlySpan<TEntry> GetArray() => GetArray<TEntry>(); public ReadOnlySpan<TEntry> GetArray() => GetWritableArray();
internal Span<TEntry> GetWritableArray() => GetWritableArray<TEntry>();
public long GetBeginOffset() => GetArray<long>()[0]; public long GetBeginOffset() => GetArray<long>()[0];
public long GetEndOffset() => GetHeader().Offset; public long GetEndOffset() => GetHeader().Offset;
@ -332,12 +337,18 @@ namespace LibHac.FsSystem
[MethodImpl(MethodImplOptions.AggressiveInlining)] [MethodImpl(MethodImplOptions.AggressiveInlining)]
public ReadOnlySpan<TElement> GetArray<TElement>() where TElement : unmanaged public ReadOnlySpan<TElement> GetArray<TElement>() where TElement : unmanaged
{
return GetWritableArray<TElement>();
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private Span<TElement> GetWritableArray<TElement>() where TElement : unmanaged
{ {
return MemoryMarshal.Cast<byte, TElement>(_buffer.Slice(Unsafe.SizeOf<NodeHeader>())); return MemoryMarshal.Cast<byte, TElement>(_buffer.Slice(Unsafe.SizeOf<NodeHeader>()));
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)] [MethodImpl(MethodImplOptions.AggressiveInlining)]
private ref NodeHeader GetHeader() internal ref NodeHeader GetHeader()
{ {
return ref Unsafe.As<byte, NodeHeader>(ref MemoryMarshal.GetReference(_buffer)); return ref Unsafe.As<byte, NodeHeader>(ref MemoryMarshal.GetReference(_buffer));
} }
@ -594,7 +605,7 @@ namespace LibHac.FsSystem
node.Find(buffer, virtualAddress); node.Find(buffer, virtualAddress);
if (node.GetIndex() < 0) if (node.GetIndex() < 0)
return ResultFs.BucketTreeEntryNotFound.Log(); return ResultFs.InvalidBucketTreeVirtualOffset.Log();
// Return the index. // Return the index.
outIndex = (int)Tree.GetEntrySetIndex(header.Index, node.GetIndex()); outIndex = (int)Tree.GetEntrySetIndex(header.Index, node.GetIndex());
@ -633,7 +644,7 @@ namespace LibHac.FsSystem
node.Find(buffer, virtualAddress); node.Find(buffer, virtualAddress);
if (node.GetIndex() < 0) if (node.GetIndex() < 0)
return ResultFs.BucketTreeEntryNotFound.Log(); return ResultFs.InvalidBucketTreeVirtualOffset.Log();
// Copy the data into entry. // Copy the data into entry.
int entryIndex = node.GetIndex(); int entryIndex = node.GetIndex();

View file

@ -0,0 +1,286 @@
using System;
using System.Buffers.Binary;
using System.Runtime.CompilerServices;
using LibHac.Common;
using LibHac.Diag;
using LibHac.Fs;
namespace LibHac.FsSystem
{
public partial class BucketTree2
{
public ref struct Builder
{
private Span<byte> NodeBuffer { get; set; }
private Span<byte> EntryBuffer { get; set; }
private int NodeSize { get; set; }
private int EntrySize { get; set; }
private int EntryCount { get; set; }
private int EntriesPerEntrySet { get; set; }
private int OffsetsPerNode { get; set; }
private int CurrentL2OffsetIndex { get; set; }
private int CurrentEntryIndex { get; set; }
private long CurrentOffset { get; set; }
/// <summary>
/// Initializes the bucket tree builder.
/// </summary>
/// <param name="headerBuffer">The buffer for the tree's header. Must be at least the size in bytes returned by <see cref="QueryHeaderStorageSize"/>.</param>
/// <param name="nodeBuffer">The buffer for the tree's nodes. Must be at least the size in bytes returned by <see cref="QueryNodeStorageSize"/>.</param>
/// <param name="entryBuffer">The buffer for the tree's entries. Must be at least the size in bytes returned by <see cref="QueryEntryStorageSize"/>.</param>
/// <param name="nodeSize">The size of each node in the bucket tree.</param>
/// <param name="entrySize">The size of each entry that will be stored in the bucket tree.</param>
/// <param name="entryCount">The exact number of entries that will be added to the bucket tree.</param>
/// <returns>The <see cref="Result"/> of the operation.</returns>
public Result Initialize(Span<byte> headerBuffer, Span<byte> nodeBuffer, Span<byte> entryBuffer,
int nodeSize, int entrySize, int entryCount)
{
Assert.AssertTrue(entrySize >= sizeof(long));
Assert.AssertTrue(nodeSize >= entrySize + Unsafe.SizeOf<NodeHeader>());
Assert.AssertTrue(NodeSizeMin <= nodeSize && nodeSize <= NodeSizeMax);
Assert.AssertTrue(Util.IsPowerOfTwo(nodeSize));
NodeSize = nodeSize;
EntrySize = entrySize;
EntryCount = entryCount;
EntriesPerEntrySet = GetEntryCount(nodeSize, entrySize);
OffsetsPerNode = GetOffsetCount(nodeSize);
CurrentL2OffsetIndex = GetNodeL2Count(nodeSize, entrySize, entryCount);
// Verify the provided buffers are large enough
int nodeStorageSize = (int)QueryNodeStorageSize(nodeSize, entrySize, entryCount);
int entryStorageSize = (int)QueryEntryStorageSize(nodeSize, entrySize, entryCount);
if (headerBuffer.Length < QueryHeaderStorageSize() ||
nodeBuffer.Length < nodeStorageSize ||
entryBuffer.Length < entryStorageSize)
{
return ResultFs.InvalidSize.Log();
}
// Set and clear the buffers
NodeBuffer = nodeBuffer.Slice(0, nodeStorageSize);
EntryBuffer = entryBuffer.Slice(0, entryStorageSize);
nodeBuffer.Clear();
entryBuffer.Clear();
// Format the tree header
ref Header header = ref SpanHelpers.AsStruct<Header>(headerBuffer);
header.Format(entryCount);
// Set the initial position
CurrentEntryIndex = 0;
CurrentOffset = -1;
return Result.Success;
}
/// <summary>
/// Adds a new entry to the bucket tree.
/// </summary>
/// <typeparam name="T">The type of the entry to add. Added entries should all be the same type.</typeparam>
/// <param name="entry">The entry to add.</param>
/// <returns>The <see cref="Result"/> of the operation.</returns>
public Result Add<T>(ref T entry) where T : unmanaged
{
Assert.AssertTrue(Unsafe.SizeOf<T>() == EntrySize);
if (CurrentEntryIndex >= EntryCount)
return ResultFs.OutOfRange.Log();
long entryOffset = BinaryPrimitives.ReadInt64LittleEndian(SpanHelpers.AsByteSpan(ref entry));
if (entryOffset <= CurrentOffset)
return ResultFs.InvalidOffset.Log();
FinalizePreviousEntrySet(entryOffset);
AddEntryOffset(entryOffset);
int entrySetIndex = CurrentEntryIndex / EntriesPerEntrySet;
int indexInEntrySet = CurrentEntryIndex % EntriesPerEntrySet;
GetEntrySet<T>(entrySetIndex).GetWritableArray()[indexInEntrySet] = entry;
CurrentOffset = entryOffset;
CurrentEntryIndex++;
return Result.Success;
}
/// <summary>
/// Checks if a new entry set is being started. If so, sets the end offset of the previous entry set.
/// </summary>
/// <param name="endOffset">The end offset of the previous entry.</param>
private void FinalizePreviousEntrySet(long endOffset)
{
int prevEntrySetIndex = CurrentEntryIndex / EntriesPerEntrySet - 1;
int indexInEntrySet = CurrentEntryIndex % EntriesPerEntrySet;
// If the previous Add finished an entry set
if (CurrentEntryIndex > 0 && indexInEntrySet == 0)
{
// Set the end offset of that entry set
BucketTreeNode<long> prevEntrySet = GetEntrySet<long>(prevEntrySetIndex);
prevEntrySet.GetHeader().Index = prevEntrySetIndex;
prevEntrySet.GetHeader().Count = EntriesPerEntrySet;
prevEntrySet.GetHeader().Offset = endOffset;
// Check if we're writing in L2 nodes
if (CurrentL2OffsetIndex > OffsetsPerNode)
{
int prevL2NodeIndex = CurrentL2OffsetIndex / OffsetsPerNode - 2;
int indexInL2Node = CurrentL2OffsetIndex % OffsetsPerNode;
// If the previous Add finished an L2 node
if (indexInL2Node == 0)
{
// Set the end offset of that node
BucketTreeNode<long> prevL2Node = GetL2Node(prevL2NodeIndex);
prevL2Node.GetHeader().Index = prevL2NodeIndex;
prevL2Node.GetHeader().Count = OffsetsPerNode;
prevL2Node.GetHeader().Offset = endOffset;
}
}
}
}
/// <summary>
/// If needed, adds a new entry set's start offset to the L1 or L2 nodes.
/// </summary>
/// <param name="entryOffset">The start offset of the entry being added.</param>
private void AddEntryOffset(long entryOffset)
{
int entrySetIndex = CurrentEntryIndex / EntriesPerEntrySet;
int indexInEntrySet = CurrentEntryIndex % EntriesPerEntrySet;
// If we're starting a new entry set we need to add its start offset to the L1/L2 nodes
if (indexInEntrySet == 0)
{
BucketTreeNode<long> l1Node = GetL1Node();
if (CurrentL2OffsetIndex == 0)
{
// There are no L2 nodes. Write the entry set end offset directly to L1
l1Node.GetWritableArray()[entrySetIndex] = entryOffset;
}
else
{
if (CurrentL2OffsetIndex < OffsetsPerNode)
{
// The current L2 offset is stored in the L1 node
l1Node.GetWritableArray()[CurrentL2OffsetIndex] = entryOffset;
}
else
{
// Write the entry set offset to the current L2 node
int l2NodeIndex = CurrentL2OffsetIndex / OffsetsPerNode;
int indexInL2Node = CurrentL2OffsetIndex % OffsetsPerNode;
BucketTreeNode<long> l2Node = GetL2Node(l2NodeIndex - 1);
l2Node.GetWritableArray()[indexInL2Node] = entryOffset;
// If we're starting a new L2 node we need to add its start offset to the L1 node
if (indexInL2Node == 0)
{
l1Node.GetWritableArray()[l2NodeIndex - 1] = entryOffset;
}
}
CurrentL2OffsetIndex++;
}
}
}
/// <summary>
/// Finalizes the bucket tree. Must be called after all entries are added.
/// </summary>
/// <param name="endOffset">The end offset of the bucket tree.</param>
/// <returns>The <see cref="Result"/> of the operation.</returns>
public Result Finalize(long endOffset)
{
// Finalize must only be called after all entries are added
if (EntryCount != CurrentEntryIndex)
return ResultFs.OutOfRange.Log();
if (endOffset <= CurrentOffset)
return ResultFs.InvalidOffset.Log();
FinalizePreviousEntrySet(endOffset);
int entrySetIndex = CurrentEntryIndex / EntriesPerEntrySet;
int indexInEntrySet = CurrentEntryIndex % EntriesPerEntrySet;
// Finalize the current entry set if needed
if (indexInEntrySet != 0)
{
ref NodeHeader entrySetHeader = ref GetEntrySetHeader(entrySetIndex);
entrySetHeader.Index = entrySetIndex;
entrySetHeader.Count = indexInEntrySet;
entrySetHeader.Offset = endOffset;
}
int l2NodeIndex = Util.DivideByRoundUp(CurrentL2OffsetIndex, OffsetsPerNode) - 2;
int indexInL2Node = CurrentL2OffsetIndex % OffsetsPerNode;
// Finalize the current L2 node if needed
if (CurrentL2OffsetIndex > OffsetsPerNode && (indexInEntrySet != 0 || indexInL2Node != 0))
{
ref NodeHeader l2NodeHeader = ref GetL2Node(l2NodeIndex).GetHeader();
l2NodeHeader.Index = l2NodeIndex;
l2NodeHeader.Count = indexInL2Node != 0 ? indexInL2Node : OffsetsPerNode;
l2NodeHeader.Offset = endOffset;
}
// Finalize the L1 node
ref NodeHeader l1Header = ref GetL1Node().GetHeader();
l1Header.Index = 0;
l1Header.Offset = endOffset;
// L1 count depends on the existence or absence of L2 nodes
if (CurrentL2OffsetIndex == 0)
{
l1Header.Count = Util.DivideByRoundUp(CurrentEntryIndex, EntriesPerEntrySet);
}
else
{
l1Header.Count = l2NodeIndex + 1;
}
return Result.Success;
}
private ref NodeHeader GetEntrySetHeader(int index)
{
BucketTreeNode<long> entrySetNode = GetEntrySet<long>(index);
return ref entrySetNode.GetHeader();
}
private BucketTreeNode<T> GetEntrySet<T>(int index) where T : unmanaged
{
Span<byte> entrySetBuffer = EntryBuffer.Slice(NodeSize * index, NodeSize);
return new BucketTreeNode<T>(entrySetBuffer);
}
private BucketTreeNode<long> GetL1Node()
{
Span<byte> l1NodeBuffer = NodeBuffer.Slice(0, NodeSize);
return new BucketTreeNode<long>(l1NodeBuffer);
}
private BucketTreeNode<long> GetL2Node(int index)
{
Span<byte> l2NodeBuffer = NodeBuffer.Slice(NodeSize * (index + 1), NodeSize);
return new BucketTreeNode<long>(l2NodeBuffer);
}
}
}
}