namespace PKHeX.Core
{
///
/// Self-mutating value that returns a crypto value to be xor-ed with another (unaligned) byte stream.
///
/// This implementation allows for yielding crypto bytes on demand.
///
///
[System.Diagnostics.CodeAnalysis.SuppressMessage("Performance", "CA1815:Override equals and operator equals on value types", Justification = "Unused")]
public ref struct SCXorShift32
{
private int Counter;
private uint Seed;
public SCXorShift32(uint seed)
{
var pop_count = PopCount(seed);
for (var i = 0; i < pop_count; i++)
XorshiftAdvance(ref seed);
Counter = 0;
Seed = seed;
}
///
/// Gets a from the current state.
///
public uint Next()
{
var c = Counter;
var val = (Seed >> (c << 3)) & 0xFF;
if (c == 3)
{
XorshiftAdvance(ref Seed);
Counter = 0;
}
else
{
++Counter;
}
return val;
}
///
/// Gets a from the current state.
///
public uint Next32()
{
return Next() | (Next() << 8) | (Next() << 16) | (Next() << 24);
}
private static void XorshiftAdvance(ref uint key)
{
key ^= key << 2;
key ^= key >> 15;
key ^= key << 13;
}
///
/// Count of bits set in value
///
private static uint PopCount(uint x)
{
x -= ((x >> 1) & 0x55555555u);
x = (x & 0x33333333u) + ((x >> 2) & 0x33333333u);
x = (x + (x >> 4)) & 0x0F0F0F0Fu;
x += (x >> 8);
x += (x >> 16);
return x & 0x0000003Fu;
}
}
}