2022-01-09 06:48:33 +00:00
|
|
|
|
using System;
|
2021-06-04 00:50:48 +00:00
|
|
|
|
using System.Runtime.CompilerServices;
|
2020-01-23 08:11:07 +00:00
|
|
|
|
|
|
|
|
|
namespace PKHeX.Core
|
|
|
|
|
{
|
2021-01-07 23:34:26 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Logic for Generating and Verifying Gen8 Raid Templates against PKM data.
|
|
|
|
|
/// </summary>
|
2020-01-23 08:11:07 +00:00
|
|
|
|
public static class RaidRNG
|
|
|
|
|
{
|
|
|
|
|
public static bool Verify<T>(this T raid, PK8 pk8, ulong seed) where T: EncounterStatic8Nest<T>
|
|
|
|
|
{
|
2020-12-11 04:42:30 +00:00
|
|
|
|
var pi = PersonalTable.SWSH.GetFormEntry(raid.Species, raid.Form);
|
2020-01-23 08:11:07 +00:00
|
|
|
|
var ratio = pi.Gender;
|
|
|
|
|
var abil = RemapAbilityToParam(raid.Ability);
|
2022-01-09 06:48:33 +00:00
|
|
|
|
|
|
|
|
|
Span<int> IVs = stackalloc int[6];
|
|
|
|
|
LoadIVs(raid, IVs);
|
2020-01-23 08:11:07 +00:00
|
|
|
|
return Verify(pk8, seed, IVs, raid.FlawlessIVCount, abil, ratio);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void ApplyDetailsTo<T>(this T raid, PK8 pk8, ulong seed) where T : EncounterStatic8Nest<T>
|
|
|
|
|
{
|
|
|
|
|
// Ensure the species-form is set correctly (nature)
|
|
|
|
|
pk8.Species = raid.Species;
|
2020-12-11 04:42:30 +00:00
|
|
|
|
pk8.Form = raid.Form;
|
|
|
|
|
var pi = PersonalTable.SWSH.GetFormEntry(raid.Species, raid.Form);
|
2020-01-23 08:11:07 +00:00
|
|
|
|
var ratio = pi.Gender;
|
|
|
|
|
var abil = RemapAbilityToParam(raid.Ability);
|
2022-01-09 06:48:33 +00:00
|
|
|
|
|
|
|
|
|
Span<int> IVs = stackalloc int[6];
|
|
|
|
|
LoadIVs(raid, IVs);
|
2020-01-23 08:11:07 +00:00
|
|
|
|
ApplyDetailsTo(pk8, seed, IVs, raid.FlawlessIVCount, abil, ratio);
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-09 06:48:33 +00:00
|
|
|
|
private static void LoadIVs<T>(T raid, Span<int> IVs) where T : EncounterStatic8Nest<T>
|
|
|
|
|
{
|
|
|
|
|
if (raid.IVs.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
IVs.Fill(-1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
((int[])raid.IVs).CopyTo(IVs);
|
|
|
|
|
PKX.ReorderSpeedLast(IVs);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-09 06:34:04 +00:00
|
|
|
|
private static int RemapAbilityToParam(AbilityPermission a) => a switch
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
2022-01-09 06:34:04 +00:00
|
|
|
|
AbilityPermission.Any12H => 254,
|
|
|
|
|
AbilityPermission.Any12 => 255,
|
|
|
|
|
_ => a.GetSingleValue(),
|
2021-01-02 01:08:49 +00:00
|
|
|
|
};
|
2020-01-23 08:11:07 +00:00
|
|
|
|
|
2022-01-09 06:48:33 +00:00
|
|
|
|
private static bool Verify(PKM pk, ulong seed, Span<int> ivs, int iv_count, int ability_param, int gender_ratio, sbyte nature_param = -1, Shiny shiny = Shiny.Random)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var rng = new Xoroshiro128Plus(seed);
|
|
|
|
|
var ec = (uint)rng.NextInt();
|
|
|
|
|
if (ec != pk.EncryptionConstant)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
uint pid;
|
|
|
|
|
bool isShiny;
|
|
|
|
|
if (shiny == Shiny.Random) // let's decide if it's shiny or not!
|
|
|
|
|
{
|
|
|
|
|
var trID = (uint)rng.NextInt();
|
|
|
|
|
pid = (uint)rng.NextInt();
|
|
|
|
|
isShiny = GetShinyXor(pid, trID) < 16;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// no need to calculate a fake trainer
|
|
|
|
|
pid = (uint)rng.NextInt();
|
|
|
|
|
isShiny = shiny == Shiny.Always;
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-04 00:50:48 +00:00
|
|
|
|
ForceShinyState(pk, isShiny, ref pid);
|
2020-01-23 08:11:07 +00:00
|
|
|
|
|
|
|
|
|
if (pk.PID != pid)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
const int UNSET = -1;
|
|
|
|
|
const int MAX = 31;
|
2022-01-09 06:48:33 +00:00
|
|
|
|
for (int i = ivs.Count(MAX); i < iv_count; i++)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
int index = (int)rng.NextInt(6);
|
|
|
|
|
while (ivs[index] != UNSET)
|
|
|
|
|
index = (int)rng.NextInt(6);
|
|
|
|
|
ivs[index] = MAX;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
|
{
|
|
|
|
|
if (ivs[i] == UNSET)
|
|
|
|
|
ivs[i] = (int)rng.NextInt(32);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pk.IV_HP != ivs[0])
|
|
|
|
|
return false;
|
|
|
|
|
if (pk.IV_ATK != ivs[1])
|
|
|
|
|
return false;
|
|
|
|
|
if (pk.IV_DEF != ivs[2])
|
|
|
|
|
return false;
|
|
|
|
|
if (pk.IV_SPA != ivs[3])
|
|
|
|
|
return false;
|
|
|
|
|
if (pk.IV_SPD != ivs[4])
|
|
|
|
|
return false;
|
|
|
|
|
if (pk.IV_SPE != ivs[5])
|
|
|
|
|
return false;
|
|
|
|
|
|
2021-03-14 23:16:55 +00:00
|
|
|
|
int abil = ability_param switch
|
|
|
|
|
{
|
|
|
|
|
254 => (int)rng.NextInt(3),
|
|
|
|
|
255 => (int)rng.NextInt(2),
|
2021-08-20 20:49:20 +00:00
|
|
|
|
_ => ability_param,
|
2021-03-14 23:16:55 +00:00
|
|
|
|
};
|
2020-01-23 08:11:07 +00:00
|
|
|
|
abil <<= 1; // 1/2/4
|
|
|
|
|
|
2020-10-10 19:30:57 +00:00
|
|
|
|
var current = pk.AbilityNumber;
|
2020-10-10 19:59:31 +00:00
|
|
|
|
if (abil == 4)
|
2020-10-10 19:30:57 +00:00
|
|
|
|
{
|
2020-10-10 19:59:31 +00:00
|
|
|
|
if (current != 4)
|
|
|
|
|
return false;
|
2020-10-10 19:30:57 +00:00
|
|
|
|
}
|
2020-10-10 19:59:31 +00:00
|
|
|
|
// else, for things that were made Hidden Ability, defer to Ability Checks (Ability Patch)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
|
|
|
|
|
switch (gender_ratio)
|
|
|
|
|
{
|
2021-07-13 07:25:51 +00:00
|
|
|
|
case PersonalInfo.RatioMagicGenderless when pk.Gender != 2:
|
2020-01-23 08:11:07 +00:00
|
|
|
|
if (pk.Gender != 2)
|
|
|
|
|
return false;
|
|
|
|
|
break;
|
2021-07-13 07:25:51 +00:00
|
|
|
|
case PersonalInfo.RatioMagicFemale when pk.Gender != 1:
|
2020-01-23 08:11:07 +00:00
|
|
|
|
if (pk.Gender != 1)
|
|
|
|
|
return false;
|
|
|
|
|
break;
|
2021-07-13 07:25:51 +00:00
|
|
|
|
case PersonalInfo.RatioMagicMale:
|
2020-01-23 08:11:07 +00:00
|
|
|
|
if (pk.Gender != 0)
|
|
|
|
|
return false;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
var gender = (int)rng.NextInt(252) + 1 < gender_ratio ? 1 : 0;
|
|
|
|
|
if (pk.Gender != gender)
|
|
|
|
|
return false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (nature_param == -1)
|
|
|
|
|
{
|
2020-12-11 04:42:30 +00:00
|
|
|
|
if (pk.Species == (int) Species.Toxtricity && pk.Form == 0)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var table = Nature0;
|
|
|
|
|
var choice = table[rng.NextInt((uint)table.Length)];
|
|
|
|
|
if (pk.Nature != choice)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2020-12-11 04:42:30 +00:00
|
|
|
|
else if (pk.Species == (int) Species.Toxtricity && pk.Form == 1)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var table = Nature1;
|
|
|
|
|
var choice = table[rng.NextInt((uint)table.Length)];
|
|
|
|
|
if (pk.Nature != choice)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var nature = (int)rng.NextInt(25);
|
|
|
|
|
if (pk.Nature != nature)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (pk.Nature != nature_param)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pk is IScaledSize s)
|
|
|
|
|
{
|
|
|
|
|
var height = (int)rng.NextInt(0x81) + (int)rng.NextInt(0x80);
|
|
|
|
|
if (s.HeightScalar != height)
|
|
|
|
|
return false;
|
|
|
|
|
var weight = (int)rng.NextInt(0x81) + (int)rng.NextInt(0x80);
|
|
|
|
|
if (s.WeightScalar != weight)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-04 00:50:48 +00:00
|
|
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
|
|
|
public static void ForceShinyState(PKM pk, bool isShiny, ref uint pid)
|
|
|
|
|
{
|
|
|
|
|
if (isShiny)
|
|
|
|
|
{
|
|
|
|
|
if (!GetIsShiny(pk.TID, pk.SID, pid))
|
|
|
|
|
pid = GetShinyPID(pk.TID, pk.SID, pid, 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (GetIsShiny(pk.TID, pk.SID, pid))
|
|
|
|
|
pid ^= 0x1000_0000;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-09 06:48:33 +00:00
|
|
|
|
private static bool ApplyDetailsTo(PKM pk, ulong seed, Span<int> ivs, int iv_count, int ability_param, int gender_ratio, sbyte nature_param = -1, Shiny shiny = Shiny.Random)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var rng = new Xoroshiro128Plus(seed);
|
|
|
|
|
pk.EncryptionConstant = (uint)rng.NextInt();
|
|
|
|
|
|
|
|
|
|
uint pid;
|
|
|
|
|
bool isShiny;
|
|
|
|
|
if (shiny == Shiny.Random) // let's decide if it's shiny or not!
|
|
|
|
|
{
|
|
|
|
|
var trID = (uint)rng.NextInt();
|
|
|
|
|
pid = (uint)rng.NextInt();
|
|
|
|
|
isShiny = GetShinyXor(pid, trID) < 16;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// no need to calculate a fake trainer
|
|
|
|
|
pid = (uint)rng.NextInt();
|
|
|
|
|
isShiny = shiny == Shiny.Always;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isShiny)
|
|
|
|
|
{
|
|
|
|
|
if (!GetIsShiny(pk.TID, pk.SID, pid))
|
|
|
|
|
pid = GetShinyPID(pk.TID, pk.SID, pid, 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (GetIsShiny(pk.TID, pk.SID, pid))
|
|
|
|
|
pid ^= 0x1000_0000;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pk.PID = pid;
|
|
|
|
|
|
|
|
|
|
const int UNSET = -1;
|
|
|
|
|
const int MAX = 31;
|
2022-01-09 06:48:33 +00:00
|
|
|
|
for (int i = ivs.Count(MAX); i < iv_count; i++)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
int index = (int)rng.NextInt(6);
|
|
|
|
|
while (ivs[index] != UNSET)
|
|
|
|
|
index = (int)rng.NextInt(6);
|
|
|
|
|
ivs[index] = MAX;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
|
{
|
|
|
|
|
if (ivs[i] == UNSET)
|
|
|
|
|
ivs[i] = (int)rng.NextInt(32);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pk.IV_HP = ivs[0];
|
|
|
|
|
pk.IV_ATK = ivs[1];
|
|
|
|
|
pk.IV_DEF = ivs[2];
|
|
|
|
|
pk.IV_SPA = ivs[3];
|
|
|
|
|
pk.IV_SPD = ivs[4];
|
|
|
|
|
pk.IV_SPE = ivs[5];
|
|
|
|
|
|
2021-01-07 23:34:26 +00:00
|
|
|
|
int abil = ability_param switch
|
|
|
|
|
{
|
|
|
|
|
254 => (int)rng.NextInt(3),
|
|
|
|
|
255 => (int)rng.NextInt(2),
|
2021-08-20 20:49:20 +00:00
|
|
|
|
_ => ability_param,
|
2021-01-07 23:34:26 +00:00
|
|
|
|
};
|
2020-01-23 08:11:07 +00:00
|
|
|
|
pk.RefreshAbility(abil);
|
|
|
|
|
|
2020-12-25 18:58:33 +00:00
|
|
|
|
pk.Gender = gender_ratio switch
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
2021-07-13 07:25:51 +00:00
|
|
|
|
PersonalInfo.RatioMagicGenderless => 2,
|
|
|
|
|
PersonalInfo.RatioMagicFemale => 1,
|
|
|
|
|
PersonalInfo.RatioMagicMale => 0,
|
2021-08-20 20:49:20 +00:00
|
|
|
|
_ => (int) rng.NextInt(252) + 1 < gender_ratio ? 1 : 0,
|
2020-12-25 18:58:33 +00:00
|
|
|
|
};
|
2020-01-23 08:11:07 +00:00
|
|
|
|
|
|
|
|
|
int nature;
|
|
|
|
|
if (nature_param == -1)
|
|
|
|
|
{
|
2020-12-11 04:42:30 +00:00
|
|
|
|
if (pk.Species == (int)Species.Toxtricity && pk.Form == 0)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var table = Nature0;
|
|
|
|
|
nature = table[rng.NextInt((uint)table.Length)];
|
|
|
|
|
}
|
2020-12-11 04:42:30 +00:00
|
|
|
|
else if (pk.Species == (int)Species.Toxtricity && pk.Form == 1)
|
2020-01-23 08:11:07 +00:00
|
|
|
|
{
|
|
|
|
|
var table = Nature1;
|
|
|
|
|
nature = table[rng.NextInt((uint)table.Length)];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
nature = (int)rng.NextInt(25);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
nature = nature_param;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pk.StatNature = pk.Nature = nature;
|
|
|
|
|
|
|
|
|
|
if (pk is IScaledSize s)
|
|
|
|
|
{
|
|
|
|
|
var height = (int)rng.NextInt(0x81) + (int)rng.NextInt(0x80);
|
|
|
|
|
var weight = (int)rng.NextInt(0x81) + (int)rng.NextInt(0x80);
|
2022-03-06 02:30:35 +00:00
|
|
|
|
s.HeightScalar = (byte)height;
|
|
|
|
|
s.WeightScalar = (byte)weight;
|
2020-01-23 08:11:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static uint GetShinyPID(int tid, int sid, uint pid, int type)
|
|
|
|
|
{
|
|
|
|
|
return (uint) (((tid ^ sid ^ (pid & 0xFFFF) ^ type) << 16) | (pid & 0xFFFF));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static bool GetIsShiny(int tid, int sid, uint pid)
|
|
|
|
|
{
|
|
|
|
|
return GetShinyXor(pid, (uint) ((sid << 16) | tid)) < 16;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static uint GetShinyXor(uint pid, uint oid)
|
|
|
|
|
{
|
|
|
|
|
var xor = pid ^ oid;
|
|
|
|
|
return (xor ^ (xor >> 16)) & 0xFFFF;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static readonly int[] Nature0 = {3, 4, 2, 8, 9, 19, 22, 11, 13, 14, 0, 6, 24};
|
|
|
|
|
private static readonly int[] Nature1 = {1, 5, 7, 10, 12, 15, 16, 17, 18, 20, 21, 23};
|
|
|
|
|
}
|
|
|
|
|
}
|