2023-01-22 04:02:33 +00:00
|
|
|
using System;
|
2023-12-04 04:13:20 +00:00
|
|
|
using System.ComponentModel.DataAnnotations;
|
2022-05-07 18:47:01 +00:00
|
|
|
using System.Diagnostics;
|
2022-02-05 01:20:56 +00:00
|
|
|
using static System.Buffers.Binary.BinaryPrimitives;
|
|
|
|
|
|
|
|
namespace PKHeX.Core;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Unpacks a BinLinkerAccessor generated file container into individual arrays.
|
|
|
|
/// </summary>
|
2022-05-07 18:47:01 +00:00
|
|
|
[DebuggerDisplay($"{{{nameof(Identifier)},nq}}[{{{nameof(Length)},nq}}]")]
|
2022-02-05 01:20:56 +00:00
|
|
|
public readonly ref struct BinLinkerAccessor
|
|
|
|
{
|
|
|
|
/// <summary> Backing data object </summary>
|
|
|
|
private readonly ReadOnlySpan<byte> Data;
|
|
|
|
|
|
|
|
/// <summary> Total count of files available for accessing. </summary>
|
2024-01-04 07:06:09 +00:00
|
|
|
public ushort Length => ReadUInt16LittleEndian(Data[2..]);
|
2022-02-05 01:20:56 +00:00
|
|
|
|
|
|
|
/// <summary> Magic identifier for the file. </summary>
|
2023-12-18 00:38:53 +00:00
|
|
|
public string Identifier => new([(char)Data[0], (char)Data[1]]);
|
2022-02-05 01:20:56 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Retrieves a view of the entry at the requested <see cref="index"/>.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="index">Entry to retrieve.</param>
|
|
|
|
public ReadOnlySpan<byte> this[int index] => GetEntry(index);
|
|
|
|
|
|
|
|
private BinLinkerAccessor(ReadOnlySpan<byte> data) => Data = data;
|
|
|
|
|
|
|
|
private ReadOnlySpan<byte> GetEntry(int index)
|
|
|
|
{
|
|
|
|
int offset = 4 + (index * sizeof(int));
|
2024-01-04 07:06:09 +00:00
|
|
|
// Start and End are both 32-bit integers, sequentially.
|
|
|
|
// Read them in one shot a 64-bit integer and decompose.
|
|
|
|
var startEnd = ReadUInt64LittleEndian(Data[offset..]);
|
|
|
|
int start = (int)startEnd;
|
|
|
|
int end = (int)(startEnd >> 32);
|
2022-02-05 01:20:56 +00:00
|
|
|
return Data[start..end];
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Sanity checks the input <see cref="data"/> only in DEBUG builds, and returns a new wrapper.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="data">Data reference</param>
|
|
|
|
/// <param name="identifier">Expected identifier (debug verification only)</param>
|
2023-12-04 04:13:20 +00:00
|
|
|
public static BinLinkerAccessor Get(ReadOnlySpan<byte> data, [Length(2, 2)] ReadOnlySpan<byte> identifier)
|
2022-02-05 01:20:56 +00:00
|
|
|
{
|
2023-01-22 04:02:33 +00:00
|
|
|
SanityCheckIdentifier(data, identifier);
|
|
|
|
return new BinLinkerAccessor(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
[Conditional("DEBUG")]
|
2023-12-04 04:13:20 +00:00
|
|
|
private static void SanityCheckIdentifier(ReadOnlySpan<byte> data, [Length(2, 2)] ReadOnlySpan<byte> identifier)
|
2023-01-22 04:02:33 +00:00
|
|
|
{
|
2022-05-07 18:47:01 +00:00
|
|
|
Debug.Assert(data.Length > 4);
|
|
|
|
Debug.Assert(identifier[0] == data[0] && identifier[1] == data[1]);
|
2022-02-05 01:20:56 +00:00
|
|
|
}
|
|
|
|
}
|