Merge pull request #11400 from Pokechu22/better-ppc-tables

Use C++20 features to create opcode tables at compile time
This commit is contained in:
Admiral H. Curtiss 2023-03-17 04:13:25 +01:00 committed by GitHub
commit 49b495f756
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 1240 additions and 735 deletions

View File

@ -83,4 +83,17 @@ static_assert(!IsNOf<int, 1, int, int>::value);
static_assert(IsNOf<int, 2, int, int>::value); static_assert(IsNOf<int, 2, int, int>::value);
static_assert(IsNOf<int, 2, int, short>::value); // Type conversions ARE allowed static_assert(IsNOf<int, 2, int, short>::value); // Type conversions ARE allowed
static_assert(!IsNOf<int, 2, int, char*>::value); static_assert(!IsNOf<int, 2, int, char*>::value);
// TODO: This can be replaced with std::array's fill() once C++20 is fully supported.
// Prior to C++20, std::array's fill() function is, unfortunately, not constexpr.
// Ditto for <algorithm>'s std::fill. Although Dolphin targets C++20, Android doesn't
// seem to properly support constexpr fill(), so we need this for now.
template <typename T1, size_t N, typename T2>
constexpr void Fill(std::array<T1, N>& array, const T2& value)
{
for (auto& entry : array)
{
entry = value;
}
}
} // namespace Common } // namespace Common

View File

@ -270,7 +270,7 @@ void CachedInterpreter::Jit(u32 address)
{ {
PPCAnalyst::CodeOp& op = m_code_buffer[i]; PPCAnalyst::CodeOp& op = m_code_buffer[i];
js.downcountAmount += op.opinfo->numCycles; js.downcountAmount += op.opinfo->num_cycles;
if (op.opinfo->flags & FL_LOADSTORE) if (op.opinfo->flags & FL_LOADSTORE)
++js.numLoadStoreInst; ++js.numLoadStoreInst;
if (op.opinfo->flags & FL_USE_FPU) if (op.opinfo->flags & FL_USE_FPU)
@ -301,7 +301,7 @@ void CachedInterpreter::Jit(u32 address)
js.firstFPInstructionFound = true; js.firstFPInstructionFound = true;
} }
m_code.emplace_back(PPCTables::GetInterpreterOp(op.inst), op.inst); m_code.emplace_back(Interpreter::GetInterpreterOp(op.inst), op.inst);
if (memcheck) if (memcheck)
m_code.emplace_back(CheckDSI, js.downcountAmount); m_code.emplace_back(CheckDSI, js.downcountAmount);
if (check_program_exception) if (check_program_exception)

View File

@ -34,14 +34,6 @@ u32 last_pc;
bool Interpreter::m_end_block; bool Interpreter::m_end_block;
// function tables
std::array<Interpreter::Instruction, 64> Interpreter::m_op_table;
std::array<Interpreter::Instruction, 1024> Interpreter::m_op_table4;
std::array<Interpreter::Instruction, 1024> Interpreter::m_op_table19;
std::array<Interpreter::Instruction, 1024> Interpreter::m_op_table31;
std::array<Interpreter::Instruction, 32> Interpreter::m_op_table59;
std::array<Interpreter::Instruction, 1024> Interpreter::m_op_table63;
namespace namespace
{ {
// Determines whether or not the given instruction is one where its execution // Determines whether or not the given instruction is one where its execution
@ -79,30 +71,8 @@ void UpdatePC()
} }
} // Anonymous namespace } // Anonymous namespace
void Interpreter::RunTable4(UGeckoInstruction inst)
{
m_op_table4[inst.SUBOP10](inst);
}
void Interpreter::RunTable19(UGeckoInstruction inst)
{
m_op_table19[inst.SUBOP10](inst);
}
void Interpreter::RunTable31(UGeckoInstruction inst)
{
m_op_table31[inst.SUBOP10](inst);
}
void Interpreter::RunTable59(UGeckoInstruction inst)
{
m_op_table59[inst.SUBOP5](inst);
}
void Interpreter::RunTable63(UGeckoInstruction inst)
{
m_op_table63[inst.SUBOP10](inst);
}
void Interpreter::Init() void Interpreter::Init()
{ {
InitializeInstructionTables();
m_end_block = false; m_end_block = false;
} }
@ -150,12 +120,17 @@ int Interpreter::SingleStepInner()
if (HandleFunctionHooking(PowerPC::ppcState.pc)) if (HandleFunctionHooking(PowerPC::ppcState.pc))
{ {
UpdatePC(); UpdatePC();
return PPCTables::GetOpInfo(m_prev_inst)->numCycles; // TODO: Does it make sense to use m_prev_inst here?
// It seems like we should use the num_cycles for the instruction at PC instead
// (m_prev_inst has not yet been updated)
return PPCTables::GetOpInfo(m_prev_inst)->num_cycles;
} }
PowerPC::ppcState.npc = PowerPC::ppcState.pc + sizeof(UGeckoInstruction); PowerPC::ppcState.npc = PowerPC::ppcState.pc + sizeof(UGeckoInstruction);
m_prev_inst.hex = PowerPC::Read_Opcode(PowerPC::ppcState.pc); m_prev_inst.hex = PowerPC::Read_Opcode(PowerPC::ppcState.pc);
const GekkoOPInfo* opinfo = PPCTables::GetOpInfo(m_prev_inst);
// Uncomment to trace the interpreter // Uncomment to trace the interpreter
// if ((PowerPC::ppcState.pc & 0x00FFFFFF) >= 0x000AB54C && // if ((PowerPC::ppcState.pc & 0x00FFFFFF) >= 0x000AB54C &&
// (PowerPC::ppcState.pc & 0x00FFFFFF) <= 0x000AB624) // (PowerPC::ppcState.pc & 0x00FFFFFF) <= 0x000AB624)
@ -181,7 +156,7 @@ int Interpreter::SingleStepInner()
} }
else if (PowerPC::ppcState.msr.FP) else if (PowerPC::ppcState.msr.FP)
{ {
m_op_table[m_prev_inst.OPCD](m_prev_inst); RunInterpreterOp(m_prev_inst);
if ((PowerPC::ppcState.Exceptions & EXCEPTION_DSI) != 0) if ((PowerPC::ppcState.Exceptions & EXCEPTION_DSI) != 0)
{ {
CheckExceptions(); CheckExceptions();
@ -190,14 +165,14 @@ int Interpreter::SingleStepInner()
else else
{ {
// check if we have to generate a FPU unavailable exception or a program exception. // check if we have to generate a FPU unavailable exception or a program exception.
if (PPCTables::UsesFPU(m_prev_inst)) if ((opinfo->flags & FL_USE_FPU) != 0)
{ {
PowerPC::ppcState.Exceptions |= EXCEPTION_FPU_UNAVAILABLE; PowerPC::ppcState.Exceptions |= EXCEPTION_FPU_UNAVAILABLE;
CheckExceptions(); CheckExceptions();
} }
else else
{ {
m_op_table[m_prev_inst.OPCD](m_prev_inst); RunInterpreterOp(m_prev_inst);
if ((PowerPC::ppcState.Exceptions & EXCEPTION_DSI) != 0) if ((PowerPC::ppcState.Exceptions & EXCEPTION_DSI) != 0)
{ {
CheckExceptions(); CheckExceptions();
@ -213,10 +188,9 @@ int Interpreter::SingleStepInner()
UpdatePC(); UpdatePC();
const GekkoOPInfo* opinfo = PPCTables::GetOpInfo(m_prev_inst); PowerPC::UpdatePerformanceMonitor(opinfo->num_cycles, (opinfo->flags & FL_LOADSTORE) != 0,
PowerPC::UpdatePerformanceMonitor(opinfo->numCycles, (opinfo->flags & FL_LOADSTORE) != 0,
(opinfo->flags & FL_USE_FPU) != 0, PowerPC::ppcState); (opinfo->flags & FL_USE_FPU) != 0, PowerPC::ppcState);
return opinfo->numCycles; return opinfo->num_cycles;
} }
void Interpreter::SingleStep() void Interpreter::SingleStep()

View File

@ -262,12 +262,9 @@ public:
static void isync(UGeckoInstruction inst); static void isync(UGeckoInstruction inst);
using Instruction = void (*)(UGeckoInstruction inst); using Instruction = void (*)(UGeckoInstruction inst);
static std::array<Instruction, 64> m_op_table;
static std::array<Instruction, 1024> m_op_table4; static Instruction GetInterpreterOp(UGeckoInstruction inst);
static std::array<Instruction, 1024> m_op_table19; static void RunInterpreterOp(UGeckoInstruction inst);
static std::array<Instruction, 1024> m_op_table31;
static std::array<Instruction, 32> m_op_table59;
static std::array<Instruction, 1024> m_op_table63;
// singleton // singleton
static Interpreter* getInstance(); static Interpreter* getInstance();
@ -283,8 +280,6 @@ public:
private: private:
void CheckExceptions(); void CheckExceptions();
static void InitializeInstructionTables();
static bool HandleFunctionHooking(u32 address); static bool HandleFunctionHooking(u32 address);
// flag helper // flag helper

View File

@ -5,492 +5,492 @@
#include <array> #include <array>
#include "Common/Assert.h"
#include "Common/TypeUtils.h"
#include "Core/PowerPC/Gekko.h" #include "Core/PowerPC/Gekko.h"
#include "Core/PowerPC/PPCTables.h"
namespace namespace
{ {
struct GekkoOPTemplate struct InterpreterOpTemplate
{ {
int opcode; u32 opcode;
Interpreter::Instruction Inst; Interpreter::Instruction fn;
GekkoOPInfo opinfo;
}; };
} // namespace } // namespace
// clang-format off constexpr std::array<InterpreterOpTemplate, 54> s_primary_table{{
static GekkoOPInfo unknownopinfo = { "unknown_instruction", OpType::Unknown, FL_ENDBLOCK, 0, 0, 0, 0 }; {4, Interpreter::RunTable4}, // RunTable4
{19, Interpreter::RunTable19}, // RunTable19
{31, Interpreter::RunTable31}, // RunTable31
{59, Interpreter::RunTable59}, // RunTable59
{63, Interpreter::RunTable63}, // RunTable63
static std::array<GekkoOPTemplate, 54> primarytable = {16, Interpreter::bcx}, // bcx
{{ {18, Interpreter::bx}, // bx
{4, Interpreter::RunTable4, {"RunTable4", OpType::Subtable, 0, 0, 0, 0, 0}},
{19, Interpreter::RunTable19, {"RunTable19", OpType::Subtable, 0, 0, 0, 0, 0}},
{31, Interpreter::RunTable31, {"RunTable31", OpType::Subtable, 0, 0, 0, 0, 0}},
{59, Interpreter::RunTable59, {"RunTable59", OpType::Subtable, 0, 0, 0, 0, 0}},
{63, Interpreter::RunTable63, {"RunTable63", OpType::Subtable, 0, 0, 0, 0, 0}},
{16, Interpreter::bcx, {"bcx", OpType::Branch, FL_ENDBLOCK | FL_READ_CR_BI, 1, 0, 0, 0}}, {3, Interpreter::twi}, // twi
{18, Interpreter::bx, {"bx", OpType::Branch, FL_ENDBLOCK, 1, 0, 0, 0}}, {17, Interpreter::sc}, // sc
{3, Interpreter::twi, {"twi", OpType::System, FL_IN_A | FL_ENDBLOCK, 1, 0, 0, 0}}, {7, Interpreter::mulli}, // mulli
{17, Interpreter::sc, {"sc", OpType::System, FL_ENDBLOCK, 2, 0, 0, 0}}, {8, Interpreter::subfic}, // subfic
{10, Interpreter::cmpli}, // cmpli
{11, Interpreter::cmpi}, // cmpi
{12, Interpreter::addic}, // addic
{13, Interpreter::addic_rc}, // addic_rc
{14, Interpreter::addi}, // addi
{15, Interpreter::addis}, // addis
{7, Interpreter::mulli, {"mulli", OpType::Integer, FL_OUT_D | FL_IN_A, 3, 0, 0, 0}}, {20, Interpreter::rlwimix}, // rlwimix
{8, Interpreter::subfic, {"subfic", OpType::Integer, FL_OUT_D | FL_IN_A | FL_SET_CA, 1, 0, 0, 0}}, {21, Interpreter::rlwinmx}, // rlwinmx
{10, Interpreter::cmpli, {"cmpli", OpType::Integer, FL_IN_A | FL_SET_CRn, 1, 0, 0, 0}}, {23, Interpreter::rlwnmx}, // rlwnmx
{11, Interpreter::cmpi, {"cmpi", OpType::Integer, FL_IN_A | FL_SET_CRn, 1, 0, 0, 0}},
{12, Interpreter::addic, {"addic", OpType::Integer, FL_OUT_D | FL_IN_A | FL_SET_CA, 1, 0, 0, 0}},
{13, Interpreter::addic_rc, {"addic_rc", OpType::Integer, FL_OUT_D | FL_IN_A | FL_SET_CA | FL_SET_CR0, 1, 0, 0, 0}},
{14, Interpreter::addi, {"addi", OpType::Integer, FL_OUT_D | FL_IN_A0, 1, 0, 0, 0}},
{15, Interpreter::addis, {"addis", OpType::Integer, FL_OUT_D | FL_IN_A0, 1, 0, 0, 0}},
{20, Interpreter::rlwimix, {"rlwimix", OpType::Integer, FL_OUT_A | FL_IN_A | FL_IN_S | FL_RC_BIT, 1, 0, 0, 0}}, {24, Interpreter::ori}, // ori
{21, Interpreter::rlwinmx, {"rlwinmx", OpType::Integer, FL_OUT_A | FL_IN_S | FL_RC_BIT, 1, 0, 0, 0}}, {25, Interpreter::oris}, // oris
{23, Interpreter::rlwnmx, {"rlwnmx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {26, Interpreter::xori}, // xori
{27, Interpreter::xoris}, // xoris
{28, Interpreter::andi_rc}, // andi_rc
{29, Interpreter::andis_rc}, // andis_rc
{24, Interpreter::ori, {"ori", OpType::Integer, FL_OUT_A | FL_IN_S, 1, 0, 0, 0}}, {32, Interpreter::lwz}, // lwz
{25, Interpreter::oris, {"oris", OpType::Integer, FL_OUT_A | FL_IN_S, 1, 0, 0, 0}}, {33, Interpreter::lwzu}, // lwzu
{26, Interpreter::xori, {"xori", OpType::Integer, FL_OUT_A | FL_IN_S, 1, 0, 0, 0}}, {34, Interpreter::lbz}, // lbz
{27, Interpreter::xoris, {"xoris", OpType::Integer, FL_OUT_A | FL_IN_S, 1, 0, 0, 0}}, {35, Interpreter::lbzu}, // lbzu
{28, Interpreter::andi_rc, {"andi_rc", OpType::Integer, FL_OUT_A | FL_IN_S | FL_SET_CR0, 1, 0, 0, 0}}, {40, Interpreter::lhz}, // lhz
{29, Interpreter::andis_rc, {"andis_rc", OpType::Integer, FL_OUT_A | FL_IN_S | FL_SET_CR0, 1, 0, 0, 0}}, {41, Interpreter::lhzu}, // lhzu
{32, Interpreter::lwz, {"lwz", OpType::Load, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE, 1, 0, 0, 0}}, {42, Interpreter::lha}, // lha
{33, Interpreter::lwzu, {"lwzu", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE, 1, 0, 0, 0}}, {43, Interpreter::lhau}, // lhau
{34, Interpreter::lbz, {"lbz", OpType::Load, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE, 1, 0, 0, 0}},
{35, Interpreter::lbzu, {"lbzu", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE, 1, 0, 0, 0}},
{40, Interpreter::lhz, {"lhz", OpType::Load, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE, 1, 0, 0, 0}},
{41, Interpreter::lhzu, {"lhzu", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE, 1, 0, 0, 0}},
{42, Interpreter::lha, {"lha", OpType::Load, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE, 1, 0, 0, 0}}, {44, Interpreter::sth}, // sth
{43, Interpreter::lhau, {"lhau", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE, 1, 0, 0, 0}}, {45, Interpreter::sthu}, // sthu
{36, Interpreter::stw}, // stw
{37, Interpreter::stwu}, // stwu
{38, Interpreter::stb}, // stb
{39, Interpreter::stbu}, // stbu
{44, Interpreter::sth, {"sth", OpType::Store, FL_IN_A0 | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}}, {46, Interpreter::lmw}, // lmw
{45, Interpreter::sthu, {"sthu", OpType::Store, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}}, {47, Interpreter::stmw}, // stmw
{36, Interpreter::stw, {"stw", OpType::Store, FL_IN_A0 | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}},
{37, Interpreter::stwu, {"stwu", OpType::Store, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}},
{38, Interpreter::stb, {"stb", OpType::Store, FL_IN_A0 | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}},
{39, Interpreter::stbu, {"stbu", OpType::Store, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}},
{46, Interpreter::lmw, {"lmw", OpType::System, FL_EVIL | FL_IN_A0 | FL_LOADSTORE, 11, 0, 0, 0}}, {48, Interpreter::lfs}, // lfs
{47, Interpreter::stmw, {"stmw", OpType::System, FL_EVIL | FL_IN_A0 | FL_LOADSTORE, 11, 0, 0, 0}}, {49, Interpreter::lfsu}, // lfsu
{50, Interpreter::lfd}, // lfd
{51, Interpreter::lfdu}, // lfdu
{48, Interpreter::lfs, {"lfs", OpType::LoadFP, FL_OUT_FLOAT_D | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {52, Interpreter::stfs}, // stfs
{49, Interpreter::lfsu, {"lfsu", OpType::LoadFP, FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {53, Interpreter::stfsu}, // stfsu
{50, Interpreter::lfd, {"lfd", OpType::LoadFP, FL_INOUT_FLOAT_D | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {54, Interpreter::stfd}, // stfd
{51, Interpreter::lfdu, {"lfdu", OpType::LoadFP, FL_INOUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {55, Interpreter::stfdu}, // stfdu
{52, Interpreter::stfs, {"stfs", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {56, Interpreter::psq_l}, // psq_l
{53, Interpreter::stfsu, {"stfsu", OpType::StoreFP, FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {57, Interpreter::psq_lu}, // psq_lu
{54, Interpreter::stfd, {"stfd", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {60, Interpreter::psq_st}, // psq_st
{55, Interpreter::stfdu, {"stfdu", OpType::StoreFP, FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {61, Interpreter::psq_stu}, // psq_stu
{56, Interpreter::psq_l, {"psq_l", OpType::LoadPS, FL_OUT_FLOAT_D | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, // missing: 0, 1, 2, 5, 6, 9, 22, 30, 62, 58
{57, Interpreter::psq_lu, {"psq_lu", OpType::LoadPS, FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}},
{60, Interpreter::psq_st, {"psq_st", OpType::StorePS, FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}},
{61, Interpreter::psq_stu, {"psq_stu", OpType::StorePS, FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}},
//missing: 0, 1, 2, 5, 6, 9, 22, 30, 62, 58
}}; }};
static std::array<GekkoOPTemplate, 13> table4 = constexpr std::array<InterpreterOpTemplate, 13> s_table4{{
{{ //SUBOP10 // SUBOP10
{0, Interpreter::ps_cmpu0, {"ps_cmpu0", OpType::PS, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {0, Interpreter::ps_cmpu0}, // ps_cmpu0
{32, Interpreter::ps_cmpo0, {"ps_cmpo0", OpType::PS, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {32, Interpreter::ps_cmpo0}, // ps_cmpo0
{40, Interpreter::ps_neg, {"ps_neg", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {40, Interpreter::ps_neg}, // ps_neg
{136, Interpreter::ps_nabs, {"ps_nabs", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {136, Interpreter::ps_nabs}, // ps_nabs
{264, Interpreter::ps_abs, {"ps_abs", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {264, Interpreter::ps_abs}, // ps_abs
{64, Interpreter::ps_cmpu1, {"ps_cmpu1", OpType::PS, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {64, Interpreter::ps_cmpu1}, // ps_cmpu1
{72, Interpreter::ps_mr, {"ps_mr", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {72, Interpreter::ps_mr}, // ps_mr
{96, Interpreter::ps_cmpo1, {"ps_cmpo1", OpType::PS, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {96, Interpreter::ps_cmpo1}, // ps_cmpo1
{528, Interpreter::ps_merge00, {"ps_merge00", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {528, Interpreter::ps_merge00}, // ps_merge00
{560, Interpreter::ps_merge01, {"ps_merge01", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {560, Interpreter::ps_merge01}, // ps_merge01
{592, Interpreter::ps_merge10, {"ps_merge10", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {592, Interpreter::ps_merge10}, // ps_merge10
{624, Interpreter::ps_merge11, {"ps_merge11", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {624, Interpreter::ps_merge11}, // ps_merge11
{1014, Interpreter::dcbz_l, {"dcbz_l", OpType::System, FL_IN_A0B | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {1014, Interpreter::dcbz_l}, // dcbz_l
}}; }};
static std::array<GekkoOPTemplate, 17> table4_2 = constexpr std::array<InterpreterOpTemplate, 17> s_table4_2{{
{{ {10, Interpreter::ps_sum0}, // ps_sum0
{10, Interpreter::ps_sum0, {"ps_sum0", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {11, Interpreter::ps_sum1}, // ps_sum1
{11, Interpreter::ps_sum1, {"ps_sum1", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {12, Interpreter::ps_muls0}, // ps_muls0
{12, Interpreter::ps_muls0, {"ps_muls0", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {13, Interpreter::ps_muls1}, // ps_muls1
{13, Interpreter::ps_muls1, {"ps_muls1", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {14, Interpreter::ps_madds0}, // ps_madds0
{14, Interpreter::ps_madds0, {"ps_madds0", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {15, Interpreter::ps_madds1}, // ps_madds1
{15, Interpreter::ps_madds1, {"ps_madds1", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {18, Interpreter::ps_div}, // ps_div
{18, Interpreter::ps_div, {"ps_div", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 17, 0, 0, 0}}, {20, Interpreter::ps_sub}, // ps_sub
{20, Interpreter::ps_sub, {"ps_sub", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {21, Interpreter::ps_add}, // ps_add
{21, Interpreter::ps_add, {"ps_add", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {23, Interpreter::ps_sel}, // ps_sel
{23, Interpreter::ps_sel, {"ps_sel", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_IN_FLOAT_BC_BITEXACT | FL_RC_BIT_F | FL_USE_FPU | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {24, Interpreter::ps_res}, // ps_res
{24, Interpreter::ps_res, {"ps_res", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 1, 0, 0, 0}}, {25, Interpreter::ps_mul}, // ps_mul
{25, Interpreter::ps_mul, {"ps_mul", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {26, Interpreter::ps_rsqrte}, // ps_rsqrte
{26, Interpreter::ps_rsqrte, {"ps_rsqrte", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 2, 0, 0, 0}}, {28, Interpreter::ps_msub}, // ps_msub
{28, Interpreter::ps_msub, {"ps_msub", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {29, Interpreter::ps_madd}, // ps_madd
{29, Interpreter::ps_madd, {"ps_madd", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {30, Interpreter::ps_nmsub}, // ps_nmsub
{30, Interpreter::ps_nmsub, {"ps_nmsub", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {31, Interpreter::ps_nmadd}, // ps_nmadd
{31, Interpreter::ps_nmadd, {"ps_nmadd", OpType::PS, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
}}; }};
constexpr std::array<InterpreterOpTemplate, 4> s_table4_3{{
static std::array<GekkoOPTemplate, 4> table4_3 = {6, Interpreter::psq_lx}, // psq_lx
{{ {7, Interpreter::psq_stx}, // psq_stx
{6, Interpreter::psq_lx, {"psq_lx", OpType::LoadPS, FL_OUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {38, Interpreter::psq_lux}, // psq_lux
{7, Interpreter::psq_stx, {"psq_stx", OpType::StorePS, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {39, Interpreter::psq_stux}, // psq_stux
{38, Interpreter::psq_lux, {"psq_lux", OpType::LoadPS, FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_AB | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}},
{39, Interpreter::psq_stux, {"psq_stux", OpType::StorePS, FL_IN_FLOAT_S | FL_OUT_A | FL_IN_AB | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}},
}}; }};
static std::array<GekkoOPTemplate, 13> table19 = constexpr std::array<InterpreterOpTemplate, 13> s_table19{{
{{ {528, Interpreter::bcctrx}, // bcctrx
{528, Interpreter::bcctrx, {"bcctrx", OpType::Branch, FL_ENDBLOCK | FL_READ_CR_BI, 1, 0, 0, 0}}, {16, Interpreter::bclrx}, // bclrx
{16, Interpreter::bclrx, {"bclrx", OpType::Branch, FL_ENDBLOCK | FL_READ_CR_BI, 1, 0, 0, 0}}, {257, Interpreter::crand}, // crand
{257, Interpreter::crand, {"crand", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {129, Interpreter::crandc}, // crandc
{129, Interpreter::crandc, {"crandc", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {289, Interpreter::creqv}, // creqv
{289, Interpreter::creqv, {"creqv", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {225, Interpreter::crnand}, // crnand
{225, Interpreter::crnand, {"crnand", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {33, Interpreter::crnor}, // crnor
{33, Interpreter::crnor, {"crnor", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {449, Interpreter::cror}, // cror
{449, Interpreter::cror, {"cror", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {417, Interpreter::crorc}, // crorc
{417, Interpreter::crorc, {"crorc", OpType::CR, FL_EVIL, 1, 0, 0, 0}}, {193, Interpreter::crxor}, // crxor
{193, Interpreter::crxor, {"crxor", OpType::CR, FL_EVIL, 1, 0, 0, 0}},
{150, Interpreter::isync, {"isync", OpType::InstructionCache, FL_EVIL, 1, 0, 0, 0}}, {150, Interpreter::isync}, // isync
{0, Interpreter::mcrf, {"mcrf", OpType::System, FL_EVIL | FL_SET_CRn | FL_READ_CRn, 1, 0, 0, 0}}, {0, Interpreter::mcrf}, // mcrf
{50, Interpreter::rfi, {"rfi", OpType::System, FL_ENDBLOCK | FL_CHECKEXCEPTIONS | FL_PROGRAMEXCEPTION, 2, 0, 0, 0}}, {50, Interpreter::rfi}, // rfi
}}; }};
static std::array<GekkoOPTemplate, 107> table31 = constexpr std::array<InterpreterOpTemplate, 107> s_table31{{
{{ {266, Interpreter::addx}, // addx
{266, Interpreter::addx, {"addx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 1, 0, 0, 0}}, {778, Interpreter::addx}, // addox
{778, Interpreter::addx, {"addox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {10, Interpreter::addcx}, // addcx
{10, Interpreter::addcx, {"addcx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {522, Interpreter::addcx}, // addcox
{522, Interpreter::addcx, {"addcox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {138, Interpreter::addex}, // addex
{138, Interpreter::addex, {"addex", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {650, Interpreter::addex}, // addeox
{650, Interpreter::addex, {"addeox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {234, Interpreter::addmex}, // addmex
{234, Interpreter::addmex, {"addmex", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {746, Interpreter::addmex}, // addmeox
{746, Interpreter::addmex, {"addmeox", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {202, Interpreter::addzex}, // addzex
{202, Interpreter::addzex, {"addzex", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {714, Interpreter::addzex}, // addzeox
{714, Interpreter::addzex, {"addzeox", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {491, Interpreter::divwx}, // divwx
{491, Interpreter::divwx, {"divwx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 40, 0, 0, 0}}, {1003, Interpreter::divwx}, // divwox
{1003, Interpreter::divwx, {"divwox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE, 40, 0, 0, 0}}, {459, Interpreter::divwux}, // divwux
{459, Interpreter::divwux, {"divwux", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 40, 0, 0, 0}}, {971, Interpreter::divwux}, // divwuox
{971, Interpreter::divwux, {"divwuox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE, 40, 0, 0, 0}}, {75, Interpreter::mulhwx}, // mulhwx
{75, Interpreter::mulhwx, {"mulhwx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 5, 0, 0, 0}}, {11, Interpreter::mulhwux}, // mulhwux
{11, Interpreter::mulhwux, {"mulhwux", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 5, 0, 0, 0}}, {235, Interpreter::mullwx}, // mullwx
{235, Interpreter::mullwx, {"mullwx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 5, 0, 0, 0}}, {747, Interpreter::mullwx}, // mullwox
{747, Interpreter::mullwx, {"mullwox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE, 5, 0, 0, 0}}, {104, Interpreter::negx}, // negx
{104, Interpreter::negx, {"negx", OpType::Integer, FL_OUT_D | FL_IN_A | FL_RC_BIT, 1, 0, 0, 0}}, {616, Interpreter::negx}, // negox
{616, Interpreter::negx, {"negox", OpType::Integer, FL_OUT_D | FL_IN_A | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {40, Interpreter::subfx}, // subfx
{40, Interpreter::subfx, {"subfx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT, 1, 0, 0, 0}}, {552, Interpreter::subfx}, // subfox
{552, Interpreter::subfx, {"subfox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {8, Interpreter::subfcx}, // subfcx
{8, Interpreter::subfcx, {"subfcx", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {520, Interpreter::subfcx}, // subfcox
{520, Interpreter::subfcx, {"subfcox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {136, Interpreter::subfex}, // subfex
{136, Interpreter::subfex, {"subfex", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {648, Interpreter::subfex}, // subfeox
{648, Interpreter::subfex, {"subfeox", OpType::Integer, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {232, Interpreter::subfmex}, // subfmex
{232, Interpreter::subfmex, {"subfmex", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {744, Interpreter::subfmex}, // subfmeox
{744, Interpreter::subfmex, {"subfmeox",OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}}, {200, Interpreter::subfzex}, // subfzex
{200, Interpreter::subfzex, {"subfzex", OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {712, Interpreter::subfzex}, // subfzeox
{712, Interpreter::subfzex, {"subfzeox",OpType::Integer, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE, 1, 0, 0, 0}},
{28, Interpreter::andx, {"andx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {28, Interpreter::andx}, // andx
{60, Interpreter::andcx, {"andcx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {60, Interpreter::andcx}, // andcx
{444, Interpreter::orx, {"orx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {444, Interpreter::orx}, // orx
{124, Interpreter::norx, {"norx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {124, Interpreter::norx}, // norx
{316, Interpreter::xorx, {"xorx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {316, Interpreter::xorx}, // xorx
{412, Interpreter::orcx, {"orcx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {412, Interpreter::orcx}, // orcx
{476, Interpreter::nandx, {"nandx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {476, Interpreter::nandx}, // nandx
{284, Interpreter::eqvx, {"eqvx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {284, Interpreter::eqvx}, // eqvx
{0, Interpreter::cmp, {"cmp", OpType::Integer, FL_IN_AB | FL_SET_CRn, 1, 0, 0, 0}}, {0, Interpreter::cmp}, // cmp
{32, Interpreter::cmpl, {"cmpl", OpType::Integer, FL_IN_AB | FL_SET_CRn, 1, 0, 0, 0}}, {32, Interpreter::cmpl}, // cmpl
{26, Interpreter::cntlzwx, {"cntlzwx",OpType::Integer, FL_OUT_A | FL_IN_S | FL_RC_BIT, 1, 0, 0, 0}}, {26, Interpreter::cntlzwx}, // cntlzwx
{922, Interpreter::extshx, {"extshx", OpType::Integer, FL_OUT_A | FL_IN_S | FL_RC_BIT, 1, 0, 0, 0}}, {922, Interpreter::extshx}, // extshx
{954, Interpreter::extsbx, {"extsbx", OpType::Integer, FL_OUT_A | FL_IN_S | FL_RC_BIT, 1, 0, 0, 0}}, {954, Interpreter::extsbx}, // extsbx
{536, Interpreter::srwx, {"srwx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {536, Interpreter::srwx}, // srwx
{792, Interpreter::srawx, {"srawx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {792, Interpreter::srawx}, // srawx
{824, Interpreter::srawix, {"srawix", OpType::Integer, FL_OUT_A | FL_IN_S | FL_SET_CA | FL_RC_BIT, 1, 0, 0, 0}}, {824, Interpreter::srawix}, // srawix
{24, Interpreter::slwx, {"slwx", OpType::Integer, FL_OUT_A | FL_IN_SB | FL_RC_BIT, 1, 0, 0, 0}}, {24, Interpreter::slwx}, // slwx
{54, Interpreter::dcbst, {"dcbst", OpType::DataCache, FL_IN_A0B | FL_LOADSTORE, 5, 0, 0, 0}}, {54, Interpreter::dcbst}, // dcbst
{86, Interpreter::dcbf, {"dcbf", OpType::DataCache, FL_IN_A0B | FL_LOADSTORE, 5, 0, 0, 0}}, {86, Interpreter::dcbf}, // dcbf
{246, Interpreter::dcbtst, {"dcbtst", OpType::DataCache, 0, 2, 0, 0, 0}}, {246, Interpreter::dcbtst}, // dcbtst
{278, Interpreter::dcbt, {"dcbt", OpType::DataCache, 0, 2, 0, 0, 0}}, {278, Interpreter::dcbt}, // dcbt
{470, Interpreter::dcbi, {"dcbi", OpType::DataCache, FL_IN_A0B | FL_LOADSTORE | FL_PROGRAMEXCEPTION, 5, 0, 0, 0}}, {470, Interpreter::dcbi}, // dcbi
{758, Interpreter::dcba, {"dcba", OpType::DataCache, 0, 5, 0, 0, 0}}, {758, Interpreter::dcba}, // dcba
{1014, Interpreter::dcbz, {"dcbz", OpType::DataCache, FL_IN_A0B | FL_LOADSTORE, 5, 0, 0, 0}}, {1014, Interpreter::dcbz}, // dcbz
//load word // load word
{23, Interpreter::lwzx, {"lwzx", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {23, Interpreter::lwzx}, // lwzx
{55, Interpreter::lwzux, {"lwzux", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {55, Interpreter::lwzux}, // lwzux
//load halfword // load halfword
{279, Interpreter::lhzx, {"lhzx", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {279, Interpreter::lhzx}, // lhzx
{311, Interpreter::lhzux, {"lhzux", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {311, Interpreter::lhzux}, // lhzux
//load halfword signextend // load halfword signextend
{343, Interpreter::lhax, {"lhax", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {343, Interpreter::lhax}, // lhax
{375, Interpreter::lhaux, {"lhaux", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {375, Interpreter::lhaux}, // lhaux
//load byte // load byte
{87, Interpreter::lbzx, {"lbzx", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {87, Interpreter::lbzx}, // lbzx
{119, Interpreter::lbzux, {"lbzux", OpType::Load, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {119, Interpreter::lbzux}, // lbzux
//load byte reverse // load byte reverse
{534, Interpreter::lwbrx, {"lwbrx", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {534, Interpreter::lwbrx}, // lwbrx
{790, Interpreter::lhbrx, {"lhbrx", OpType::Load, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {790, Interpreter::lhbrx}, // lhbrx
// Conditional load/store (Wii SMP) // Conditional load/store (Wii SMP)
{150, Interpreter::stwcxd, {"stwcxd", OpType::Store, FL_EVIL | FL_IN_S | FL_IN_A0B | FL_SET_CR0 | FL_LOADSTORE, 1, 0, 0, 0}}, {150, Interpreter::stwcxd}, // stwcxd
{20, Interpreter::lwarx, {"lwarx", OpType::Load, FL_EVIL | FL_OUT_D | FL_IN_A0B | FL_SET_CR0 | FL_LOADSTORE, 1, 0, 0, 0}}, {20, Interpreter::lwarx}, // lwarx
//load string (Inst these) // load string (Inst these)
{533, Interpreter::lswx, {"lswx", OpType::Load, FL_EVIL | FL_IN_A0B | FL_OUT_D | FL_LOADSTORE, 1, 0, 0, 0}}, {533, Interpreter::lswx}, // lswx
{597, Interpreter::lswi, {"lswi", OpType::Load, FL_EVIL | FL_IN_A0 | FL_OUT_D | FL_LOADSTORE, 1, 0, 0, 0}}, {597, Interpreter::lswi}, // lswi
//store word // store word
{151, Interpreter::stwx, {"stwx", OpType::Store, FL_IN_S | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {151, Interpreter::stwx}, // stwx
{183, Interpreter::stwux, {"stwux", OpType::Store, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {183, Interpreter::stwux}, // stwux
//store halfword // store halfword
{407, Interpreter::sthx, {"sthx", OpType::Store, FL_IN_S | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {407, Interpreter::sthx}, // sthx
{439, Interpreter::sthux, {"sthux", OpType::Store, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {439, Interpreter::sthux}, // sthux
//store byte // store byte
{215, Interpreter::stbx, {"stbx", OpType::Store, FL_IN_S | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {215, Interpreter::stbx}, // stbx
{247, Interpreter::stbux, {"stbux", OpType::Store, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE, 1, 0, 0, 0}}, {247, Interpreter::stbux}, // stbux
//store bytereverse // store bytereverse
{662, Interpreter::stwbrx, {"stwbrx", OpType::Store, FL_IN_S | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {662, Interpreter::stwbrx}, // stwbrx
{918, Interpreter::sthbrx, {"sthbrx", OpType::Store, FL_IN_S | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {918, Interpreter::sthbrx}, // sthbrx
{661, Interpreter::stswx, {"stswx", OpType::Store, FL_EVIL | FL_IN_A0B | FL_LOADSTORE, 1, 0, 0, 0}}, {661, Interpreter::stswx}, // stswx
{725, Interpreter::stswi, {"stswi", OpType::Store, FL_EVIL | FL_IN_A0 | FL_LOADSTORE, 1, 0, 0, 0}}, {725, Interpreter::stswi}, // stswi
// fp load/store // fp load/store
{535, Interpreter::lfsx, {"lfsx", OpType::LoadFP, FL_OUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {535, Interpreter::lfsx}, // lfsx
{567, Interpreter::lfsux, {"lfsux", OpType::LoadFP, FL_OUT_FLOAT_D | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {567, Interpreter::lfsux}, // lfsux
{599, Interpreter::lfdx, {"lfdx", OpType::LoadFP, FL_INOUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {599, Interpreter::lfdx}, // lfdx
{631, Interpreter::lfdux, {"lfdux", OpType::LoadFP, FL_INOUT_FLOAT_D | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {631, Interpreter::lfdux}, // lfdux
{663, Interpreter::stfsx, {"stfsx", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {663, Interpreter::stfsx}, // stfsx
{695, Interpreter::stfsux, {"stfsux", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {695, Interpreter::stfsux}, // stfsux
{727, Interpreter::stfdx, {"stfdx", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {727, Interpreter::stfdx}, // stfdx
{759, Interpreter::stfdux, {"stfdux", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {759, Interpreter::stfdux}, // stfdux
{983, Interpreter::stfiwx, {"stfiwx", OpType::StoreFP, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE, 1, 0, 0, 0}}, {983, Interpreter::stfiwx}, // stfiwx
{19, Interpreter::mfcr, {"mfcr", OpType::System, FL_OUT_D | FL_READ_ALL_CR, 1, 0, 0, 0}}, {19, Interpreter::mfcr}, // mfcr
{83, Interpreter::mfmsr, {"mfmsr", OpType::System, FL_OUT_D | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {83, Interpreter::mfmsr}, // mfmsr
{144, Interpreter::mtcrf, {"mtcrf", OpType::System, FL_IN_S | FL_SET_ALL_CR | FL_READ_ALL_CR, 1, 0, 0, 0}}, {144, Interpreter::mtcrf}, // mtcrf
{146, Interpreter::mtmsr, {"mtmsr", OpType::System, FL_IN_S | FL_ENDBLOCK | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {146, Interpreter::mtmsr}, // mtmsr
{210, Interpreter::mtsr, {"mtsr", OpType::System, FL_IN_S | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {210, Interpreter::mtsr}, // mtsr
{242, Interpreter::mtsrin, {"mtsrin", OpType::System, FL_IN_SB | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {242, Interpreter::mtsrin}, // mtsrin
{339, Interpreter::mfspr, {"mfspr", OpType::SPR, FL_OUT_D | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {339, Interpreter::mfspr}, // mfspr
{467, Interpreter::mtspr, {"mtspr", OpType::SPR, FL_IN_S | FL_PROGRAMEXCEPTION, 2, 0, 0, 0}}, {467, Interpreter::mtspr}, // mtspr
{371, Interpreter::mftb, {"mftb", OpType::System, FL_OUT_D | FL_TIMER | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {371, Interpreter::mftb}, // mftb
{512, Interpreter::mcrxr, {"mcrxr", OpType::System, FL_SET_CRn | FL_READ_CA | FL_SET_CA, 1, 0, 0, 0}}, {512, Interpreter::mcrxr}, // mcrxr
{595, Interpreter::mfsr, {"mfsr", OpType::System, FL_OUT_D | FL_PROGRAMEXCEPTION, 3, 0, 0, 0}}, {595, Interpreter::mfsr}, // mfsr
{659, Interpreter::mfsrin, {"mfsrin", OpType::System, FL_OUT_D | FL_IN_B | FL_PROGRAMEXCEPTION, 3, 0, 0, 0}}, {659, Interpreter::mfsrin}, // mfsrin
{4, Interpreter::tw, {"tw", OpType::System, FL_IN_AB | FL_ENDBLOCK, 2, 0, 0, 0}}, {4, Interpreter::tw}, // tw
{598, Interpreter::sync, {"sync", OpType::System, 0, 3, 0, 0, 0}}, {598, Interpreter::sync}, // sync
{982, Interpreter::icbi, {"icbi", OpType::System, FL_IN_A0B | FL_ENDBLOCK | FL_LOADSTORE, 4, 0, 0, 0}}, {982, Interpreter::icbi}, // icbi
// Unused instructions on GC // Unused instructions on GC
{310, Interpreter::eciwx, {"eciwx", OpType::System, FL_IN_A0B | FL_OUT_D | FL_LOADSTORE, 1, 0, 0, 0}}, {310, Interpreter::eciwx}, // eciwx
{438, Interpreter::ecowx, {"ecowx", OpType::System, FL_IN_A0B | FL_IN_S | FL_LOADSTORE, 1, 0, 0, 0}}, {438, Interpreter::ecowx}, // ecowx
{854, Interpreter::eieio, {"eieio", OpType::System, 0, 1, 0, 0, 0}}, {854, Interpreter::eieio}, // eieio
{306, Interpreter::tlbie, {"tlbie", OpType::System, FL_IN_B | FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {306, Interpreter::tlbie}, // tlbie
{566, Interpreter::tlbsync, {"tlbsync", OpType::System, FL_PROGRAMEXCEPTION, 1, 0, 0, 0}}, {566, Interpreter::tlbsync}, // tlbsync
}}; }};
static std::array<GekkoOPTemplate, 9> table59 = constexpr std::array<InterpreterOpTemplate, 9> s_table59{{
{{ {18, Interpreter::fdivsx}, // fdivsx // TODO
{18, Interpreter::fdivsx, {"fdivsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 17, 0, 0, 0}}, // TODO {20, Interpreter::fsubsx}, // fsubsx
{20, Interpreter::fsubsx, {"fsubsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {21, Interpreter::faddsx}, // faddsx
{21, Interpreter::faddsx, {"faddsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {24, Interpreter::fresx}, // fresx
{24, Interpreter::fresx, {"fresx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 1, 0, 0, 0}}, {25, Interpreter::fmulsx}, // fmulsx
{25, Interpreter::fmulsx, {"fmulsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {28, Interpreter::fmsubsx}, // fmsubsx
{28, Interpreter::fmsubsx, {"fmsubsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {29, Interpreter::fmaddsx}, // fmaddsx
{29, Interpreter::fmaddsx, {"fmaddsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {30, Interpreter::fnmsubsx}, // fnmsubsx
{30, Interpreter::fnmsubsx, {"fnmsubsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {31, Interpreter::fnmaddsx}, // fnmaddsx
{31, Interpreter::fnmaddsx, {"fnmaddsx", OpType::SingleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
}}; }};
static std::array<GekkoOPTemplate, 15> table63 = constexpr std::array<InterpreterOpTemplate, 15> s_table63{{
{{ {264, Interpreter::fabsx}, // fabsx
{264, Interpreter::fabsx, {"fabsx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU, 1, 0, 0, 0}}, {32, Interpreter::fcmpo}, // fcmpo
{0, Interpreter::fcmpu}, // fcmpu
{14, Interpreter::fctiwx}, // fctiwx
{15, Interpreter::fctiwzx}, // fctiwzx
{72, Interpreter::fmrx}, // fmrx
{136, Interpreter::fnabsx}, // fnabsx
{40, Interpreter::fnegx}, // fnegx
{12, Interpreter::frspx}, // frspx
// FIXME: fcmp modifies the FPRF flags, but if the flags are clobbered later, {64, Interpreter::mcrfs}, // mcrfs
// we don't actually need to calculate or store them here. So FL_READ_FPRF and FL_SET_FPRF is not {583, Interpreter::mffsx}, // mffsx
// an ideal representation of fcmp's effect on FPRF flags and might result in {70, Interpreter::mtfsb0x}, // mtfsb0x
// slightly sub-optimal code. {38, Interpreter::mtfsb1x}, // mtfsb1x
{32, Interpreter::fcmpo, {"fcmpo", OpType::DoubleFP, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {134, Interpreter::mtfsfix}, // mtfsfix
{0, Interpreter::fcmpu, {"fcmpu", OpType::DoubleFP, FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {711, Interpreter::mtfsfx}, // mtfsfx
{14, Interpreter::fctiwx, {"fctiwx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
{15, Interpreter::fctiwzx, {"fctiwzx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
{72, Interpreter::fmrx, {"fmrx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU, 1, 0, 0, 0}},
{136, Interpreter::fnabsx, {"fnabsx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_IN_FLOAT_B_BITEXACT | FL_USE_FPU, 1, 0, 0, 0}},
{40, Interpreter::fnegx, {"fnegx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_IN_FLOAT_B_BITEXACT | FL_USE_FPU, 1, 0, 0, 0}},
{12, Interpreter::frspx, {"frspx", OpType::DoubleFP, FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
{64, Interpreter::mcrfs, {"mcrfs", OpType::SystemFP, FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF, 1, 0, 0, 0}},
{583, Interpreter::mffsx, {"mffsx", OpType::SystemFP, FL_RC_BIT_F | FL_INOUT_FLOAT_D | FL_USE_FPU | FL_READ_FPRF, 1, 0, 0, 0}},
{70, Interpreter::mtfsb0x, {"mtfsb0x", OpType::SystemFP, FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF, 3, 0, 0, 0}},
{38, Interpreter::mtfsb1x, {"mtfsb1x", OpType::SystemFP, FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 3, 0, 0, 0}},
{134, Interpreter::mtfsfix, {"mtfsfix", OpType::SystemFP, FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 3, 0, 0, 0}},
{711, Interpreter::mtfsfx, {"mtfsfx", OpType::SystemFP, FL_RC_BIT_F | FL_IN_FLOAT_B | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 3, 0, 0, 0}},
}}; }};
static std::array<GekkoOPTemplate, 10> table63_2 = constexpr std::array<InterpreterOpTemplate, 10> s_table63_2{{
{{ {18, Interpreter::fdivx}, // fdivx
{18, Interpreter::fdivx, {"fdivx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 31, 0, 0, 0}}, {20, Interpreter::fsubx}, // fsubx
{20, Interpreter::fsubx, {"fsubx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {21, Interpreter::faddx}, // faddx
{21, Interpreter::faddx, {"faddx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {23, Interpreter::fselx}, // fselx
{23, Interpreter::fselx, {"fselx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_IN_FLOAT_BC_BITEXACT | FL_RC_BIT_F | FL_USE_FPU, 1, 0, 0, 0}}, {25, Interpreter::fmulx}, // fmulx
{25, Interpreter::fmulx, {"fmulx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {26, Interpreter::frsqrtex}, // frsqrtex
{26, Interpreter::frsqrtex, {"frsqrtex", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV, 1, 0, 0, 0}}, {28, Interpreter::fmsubx}, // fmsubx
{28, Interpreter::fmsubx, {"fmsubx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {29, Interpreter::fmaddx}, // fmaddx
{29, Interpreter::fmaddx, {"fmaddx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {30, Interpreter::fnmsubx}, // fnmsubx
{30, Interpreter::fnmsubx, {"fnmsubx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}}, {31, Interpreter::fnmaddx}, // fnmaddx
{31, Interpreter::fnmaddx, {"fnmaddx", OpType::DoubleFP, FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION, 1, 0, 0, 0}},
}}; }};
// clang-format on
constexpr size_t TotalInstructionFunctionCount() constexpr std::array<Interpreter::Instruction, 64> s_interpreter_op_table = []() consteval
{ {
return primarytable.size() + table4_2.size() + table4_3.size() + table4.size() + table31.size() + std::array<Interpreter::Instruction, 64> table{};
table19.size() + table59.size() + table63.size() + table63_2.size(); Common::Fill(table, Interpreter::unknown_instruction);
for (auto& tpl : s_primary_table)
{
ASSERT(table[tpl.opcode] == Interpreter::unknown_instruction);
table[tpl.opcode] = tpl.fn;
};
return table;
}
();
constexpr std::array<Interpreter::Instruction, 1024> s_interpreter_op_table4 = []() consteval
{
std::array<Interpreter::Instruction, 1024> table{};
Common::Fill(table, Interpreter::unknown_instruction);
for (u32 i = 0; i < 32; i++)
{
const u32 fill = i << 5;
for (const auto& tpl : s_table4_2)
{
const u32 op = fill + tpl.opcode;
ASSERT(table[op] == Interpreter::unknown_instruction);
table[op] = tpl.fn;
}
}
for (u32 i = 0; i < 16; i++)
{
const u32 fill = i << 6;
for (const auto& tpl : s_table4_3)
{
const u32 op = fill + tpl.opcode;
ASSERT(table[op] == Interpreter::unknown_instruction);
table[op] = tpl.fn;
}
}
for (const auto& tpl : s_table4)
{
const u32 op = tpl.opcode;
ASSERT(table[op] == Interpreter::unknown_instruction);
table[op] = tpl.fn;
}
return table;
}
();
constexpr std::array<Interpreter::Instruction, 1024> s_interpreter_op_table19 = []() consteval
{
std::array<Interpreter::Instruction, 1024> table{};
Common::Fill(table, Interpreter::unknown_instruction);
for (auto& tpl : s_table19)
{
ASSERT(table[tpl.opcode] == Interpreter::unknown_instruction);
table[tpl.opcode] = tpl.fn;
};
return table;
}
();
constexpr std::array<Interpreter::Instruction, 1024> s_interpreter_op_table31 = []() consteval
{
std::array<Interpreter::Instruction, 1024> table{};
Common::Fill(table, Interpreter::unknown_instruction);
for (auto& tpl : s_table31)
{
ASSERT(table[tpl.opcode] == Interpreter::unknown_instruction);
table[tpl.opcode] = tpl.fn;
};
return table;
}
();
constexpr std::array<Interpreter::Instruction, 32> s_interpreter_op_table59 = []() consteval
{
std::array<Interpreter::Instruction, 32> table{};
Common::Fill(table, Interpreter::unknown_instruction);
for (auto& tpl : s_table59)
{
ASSERT(table[tpl.opcode] == Interpreter::unknown_instruction);
table[tpl.opcode] = tpl.fn;
};
return table;
}
();
constexpr std::array<Interpreter::Instruction, 1024> s_interpreter_op_table63 = []() consteval
{
std::array<Interpreter::Instruction, 1024> table{};
Common::Fill(table, Interpreter::unknown_instruction);
for (auto& tpl : s_table63)
{
ASSERT(table[tpl.opcode] == Interpreter::unknown_instruction);
table[tpl.opcode] = tpl.fn;
};
for (u32 i = 0; i < 32; i++)
{
const u32 fill = i << 5;
for (const auto& tpl : s_table63_2)
{
const u32 op = fill + tpl.opcode;
ASSERT(table[op] == Interpreter::unknown_instruction);
table[op] = tpl.fn;
}
}
return table;
}
();
Interpreter::Instruction Interpreter::GetInterpreterOp(UGeckoInstruction inst)
{
// Check for the appropriate subtable ahead of time.
// (This is used by the cached interpreter and JIT, and called once per instruction, so spending a
// bit of extra time to optimise is worthwhile)
Interpreter::Instruction result = s_interpreter_op_table[inst.OPCD];
if (result == Interpreter::RunTable4)
return s_interpreter_op_table4[inst.SUBOP10];
else if (result == Interpreter::RunTable19)
return s_interpreter_op_table19[inst.SUBOP10];
else if (result == Interpreter::RunTable31)
return s_interpreter_op_table31[inst.SUBOP10];
else if (result == Interpreter::RunTable59)
return s_interpreter_op_table59[inst.SUBOP5];
else if (result == Interpreter::RunTable63)
return s_interpreter_op_table63[inst.SUBOP10];
else
return result;
} }
static_assert(TotalInstructionFunctionCount() < m_allInstructions.size(), void Interpreter::RunInterpreterOp(UGeckoInstruction inst)
"m_allInstructions is too small");
void Interpreter::InitializeInstructionTables()
{ {
// once initialized, tables are read-only // Will handle subtables using RunTable4 etc.
static bool initialized = false; s_interpreter_op_table[inst.OPCD](inst);
if (initialized) }
return;
void Interpreter::RunTable4(UGeckoInstruction inst)
// clear {
for (int i = 0; i < 64; i++) s_interpreter_op_table4[inst.SUBOP10](inst);
{ }
m_op_table[i] = Interpreter::unknown_instruction; void Interpreter::RunTable19(UGeckoInstruction inst)
m_infoTable[i] = &unknownopinfo; {
} s_interpreter_op_table19[inst.SUBOP10](inst);
}
for (int i = 0; i < 32; i++) void Interpreter::RunTable31(UGeckoInstruction inst)
{ {
m_op_table59[i] = Interpreter::unknown_instruction; s_interpreter_op_table31[inst.SUBOP10](inst);
m_infoTable59[i] = &unknownopinfo; }
} void Interpreter::RunTable59(UGeckoInstruction inst)
{
for (int i = 0; i < 1024; i++) s_interpreter_op_table59[inst.SUBOP5](inst);
{ }
m_op_table4[i] = Interpreter::unknown_instruction; void Interpreter::RunTable63(UGeckoInstruction inst)
m_op_table19[i] = Interpreter::unknown_instruction; {
m_op_table31[i] = Interpreter::unknown_instruction; s_interpreter_op_table63[inst.SUBOP10](inst);
m_op_table63[i] = Interpreter::unknown_instruction;
m_infoTable4[i] = &unknownopinfo;
m_infoTable19[i] = &unknownopinfo;
m_infoTable31[i] = &unknownopinfo;
m_infoTable63[i] = &unknownopinfo;
}
for (auto& tpl : primarytable)
{
m_op_table[tpl.opcode] = tpl.Inst;
m_infoTable[tpl.opcode] = &tpl.opinfo;
}
for (int i = 0; i < 32; i++)
{
int fill = i << 5;
for (auto& tpl : table4_2)
{
int op = fill + tpl.opcode;
m_op_table4[op] = tpl.Inst;
m_infoTable4[op] = &tpl.opinfo;
}
}
for (int i = 0; i < 16; i++)
{
int fill = i << 6;
for (auto& tpl : table4_3)
{
int op = fill + tpl.opcode;
m_op_table4[op] = tpl.Inst;
m_infoTable4[op] = &tpl.opinfo;
}
}
for (auto& tpl : table4)
{
int op = tpl.opcode;
m_op_table4[op] = tpl.Inst;
m_infoTable4[op] = &tpl.opinfo;
}
for (auto& tpl : table31)
{
int op = tpl.opcode;
m_op_table31[op] = tpl.Inst;
m_infoTable31[op] = &tpl.opinfo;
}
for (auto& tpl : table19)
{
int op = tpl.opcode;
m_op_table19[op] = tpl.Inst;
m_infoTable19[op] = &tpl.opinfo;
}
for (auto& tpl : table59)
{
int op = tpl.opcode;
m_op_table59[op] = tpl.Inst;
m_infoTable59[op] = &tpl.opinfo;
}
for (auto& tpl : table63)
{
int op = tpl.opcode;
m_op_table63[op] = tpl.Inst;
m_infoTable63[op] = &tpl.opinfo;
}
for (int i = 0; i < 32; i++)
{
int fill = i << 5;
for (auto& tpl : table63_2)
{
int op = fill + tpl.opcode;
m_op_table63[op] = tpl.Inst;
m_infoTable63[op] = &tpl.opinfo;
}
}
m_numInstructions = 0;
for (auto& tpl : primarytable)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table4_2)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table4_3)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table4)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table31)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table19)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table59)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table63)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
for (auto& tpl : table63_2)
m_allInstructions[m_numInstructions++] = &tpl.opinfo;
initialized = true;
} }

View File

@ -342,7 +342,7 @@ void Jit64::FallBackToInterpreter(UGeckoInstruction inst)
MOV(32, PPCSTATE(npc), Imm32(js.compilerPC + 4)); MOV(32, PPCSTATE(npc), Imm32(js.compilerPC + 4));
} }
Interpreter::Instruction instr = PPCTables::GetInterpreterOp(inst); Interpreter::Instruction instr = Interpreter::GetInterpreterOp(inst);
ABI_PushRegistersAndAdjustStack({}, 0); ABI_PushRegistersAndAdjustStack({}, 0);
ABI_CallFunctionC(instr, inst.hex); ABI_CallFunctionC(instr, inst.hex);
ABI_PopRegistersAndAdjustStack({}, 0); ABI_PopRegistersAndAdjustStack({}, 0);
@ -921,7 +921,7 @@ bool Jit64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
js.instructionNumber = i; js.instructionNumber = i;
js.instructionsLeft = (code_block.m_num_instructions - 1) - i; js.instructionsLeft = (code_block.m_num_instructions - 1) - i;
const GekkoOPInfo* opinfo = op.opinfo; const GekkoOPInfo* opinfo = op.opinfo;
js.downcountAmount += opinfo->numCycles; js.downcountAmount += opinfo->num_cycles;
js.fastmemLoadStore = nullptr; js.fastmemLoadStore = nullptr;
js.fixupExceptionHandler = false; js.fixupExceptionHandler = false;

View File

@ -5,17 +5,19 @@
#include <array> #include <array>
#include "Common/Assert.h"
#include "Common/TypeUtils.h"
#include "Core/PowerPC/Gekko.h" #include "Core/PowerPC/Gekko.h"
namespace namespace
{ {
struct GekkoOPTemplate struct Jit64OpTemplate
{ {
u32 opcode; u32 opcode;
Jit64::Instruction fn; Jit64::Instruction fn;
}; };
constexpr std::array<GekkoOPTemplate, 54> s_primary_table{{ constexpr std::array<Jit64OpTemplate, 54> s_primary_table{{
{4, &Jit64::DynaRunTable4}, // RunTable4 {4, &Jit64::DynaRunTable4}, // RunTable4
{19, &Jit64::DynaRunTable19}, // RunTable19 {19, &Jit64::DynaRunTable19}, // RunTable19
{31, &Jit64::DynaRunTable31}, // RunTable31 {31, &Jit64::DynaRunTable31}, // RunTable31
@ -85,7 +87,7 @@ constexpr std::array<GekkoOPTemplate, 54> s_primary_table{{
// missing: 0, 1, 2, 5, 6, 9, 22, 30, 62, 58 // missing: 0, 1, 2, 5, 6, 9, 22, 30, 62, 58
}}; }};
constexpr std::array<GekkoOPTemplate, 13> s_table4{{ constexpr std::array<Jit64OpTemplate, 13> s_table4{{
// SUBOP10 // SUBOP10
{0, &Jit64::ps_cmpXX}, // ps_cmpu0 {0, &Jit64::ps_cmpXX}, // ps_cmpu0
{32, &Jit64::ps_cmpXX}, // ps_cmpo0 {32, &Jit64::ps_cmpXX}, // ps_cmpo0
@ -103,7 +105,7 @@ constexpr std::array<GekkoOPTemplate, 13> s_table4{{
{1014, &Jit64::FallBackToInterpreter}, // dcbz_l {1014, &Jit64::FallBackToInterpreter}, // dcbz_l
}}; }};
constexpr std::array<GekkoOPTemplate, 17> s_table4_2{{ constexpr std::array<Jit64OpTemplate, 17> s_table4_2{{
{10, &Jit64::ps_sum}, // ps_sum0 {10, &Jit64::ps_sum}, // ps_sum0
{11, &Jit64::ps_sum}, // ps_sum1 {11, &Jit64::ps_sum}, // ps_sum1
{12, &Jit64::ps_muls}, // ps_muls0 {12, &Jit64::ps_muls}, // ps_muls0
@ -123,14 +125,14 @@ constexpr std::array<GekkoOPTemplate, 17> s_table4_2{{
{31, &Jit64::fmaddXX}, // ps_nmadd {31, &Jit64::fmaddXX}, // ps_nmadd
}}; }};
constexpr std::array<GekkoOPTemplate, 4> s_table4_3{{ constexpr std::array<Jit64OpTemplate, 4> s_table4_3{{
{6, &Jit64::psq_lXX}, // psq_lx {6, &Jit64::psq_lXX}, // psq_lx
{7, &Jit64::psq_stXX}, // psq_stx {7, &Jit64::psq_stXX}, // psq_stx
{38, &Jit64::psq_lXX}, // psq_lux {38, &Jit64::psq_lXX}, // psq_lux
{39, &Jit64::psq_stXX}, // psq_stux {39, &Jit64::psq_stXX}, // psq_stux
}}; }};
constexpr std::array<GekkoOPTemplate, 13> s_table19{{ constexpr std::array<Jit64OpTemplate, 13> s_table19{{
{528, &Jit64::bcctrx}, // bcctrx {528, &Jit64::bcctrx}, // bcctrx
{16, &Jit64::bclrx}, // bclrx {16, &Jit64::bclrx}, // bclrx
{257, &Jit64::crXXX}, // crand {257, &Jit64::crXXX}, // crand
@ -148,7 +150,7 @@ constexpr std::array<GekkoOPTemplate, 13> s_table19{{
{50, &Jit64::rfi}, // rfi {50, &Jit64::rfi}, // rfi
}}; }};
constexpr std::array<GekkoOPTemplate, 107> s_table31{{ constexpr std::array<Jit64OpTemplate, 107> s_table31{{
{266, &Jit64::addx}, // addx {266, &Jit64::addx}, // addx
{778, &Jit64::addx}, // addox {778, &Jit64::addx}, // addox
{10, &Jit64::addx}, // addcx {10, &Jit64::addx}, // addcx
@ -290,7 +292,7 @@ constexpr std::array<GekkoOPTemplate, 107> s_table31{{
{566, &Jit64::DoNothing}, // tlbsync {566, &Jit64::DoNothing}, // tlbsync
}}; }};
constexpr std::array<GekkoOPTemplate, 9> s_table59{{ constexpr std::array<Jit64OpTemplate, 9> s_table59{{
{18, &Jit64::fp_arith}, // fdivsx {18, &Jit64::fp_arith}, // fdivsx
{20, &Jit64::fp_arith}, // fsubsx {20, &Jit64::fp_arith}, // fsubsx
{21, &Jit64::fp_arith}, // faddsx {21, &Jit64::fp_arith}, // faddsx
@ -302,7 +304,7 @@ constexpr std::array<GekkoOPTemplate, 9> s_table59{{
{31, &Jit64::fmaddXX}, // fnmaddsx {31, &Jit64::fmaddXX}, // fnmaddsx
}}; }};
constexpr std::array<GekkoOPTemplate, 15> s_table63{{ constexpr std::array<Jit64OpTemplate, 15> s_table63{{
{264, &Jit64::fsign}, // fabsx {264, &Jit64::fsign}, // fabsx
{32, &Jit64::fcmpX}, // fcmpo {32, &Jit64::fcmpX}, // fcmpo
{0, &Jit64::fcmpX}, // fcmpu {0, &Jit64::fcmpX}, // fcmpu
@ -321,7 +323,7 @@ constexpr std::array<GekkoOPTemplate, 15> s_table63{{
{711, &Jit64::mtfsfx}, // mtfsfx {711, &Jit64::mtfsfx}, // mtfsfx
}}; }};
constexpr std::array<GekkoOPTemplate, 10> s_table63_2{{ constexpr std::array<Jit64OpTemplate, 10> s_table63_2{{
{18, &Jit64::fp_arith}, // fdivx {18, &Jit64::fp_arith}, // fdivx
{20, &Jit64::fp_arith}, // fsubx {20, &Jit64::fp_arith}, // fsubx
{21, &Jit64::fp_arith}, // faddx {21, &Jit64::fp_arith}, // faddx
@ -334,38 +336,25 @@ constexpr std::array<GekkoOPTemplate, 10> s_table63_2{{
{31, &Jit64::fmaddXX}, // fnmaddx {31, &Jit64::fmaddXX}, // fnmaddx
}}; }};
// TODO: This can be replaced with: constexpr std::array<Jit64::Instruction, 64> s_dyna_op_table = []() consteval
//
// table.fill(&Jit64::FallbackToInterpreter);
//
// whenever we end up migrating to C++20. Prior to C++20,
// std::array's fill() function is, unfortunately, not constexpr.
// Ditto for <algorithm>'s std::fill. Thus, this function exists
// to bridge the gap.
template <size_t N>
constexpr void FillWithFallbacks(std::array<Jit64::Instruction, N>& table)
{ {
for (auto& entry : table)
{
entry = &Jit64::FallBackToInterpreter;
}
}
constexpr std::array<Jit64::Instruction, 64> s_dyna_op_table = [] {
std::array<Jit64::Instruction, 64> table{}; std::array<Jit64::Instruction, 64> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (auto& tpl : s_primary_table) for (auto& tpl : s_primary_table)
{ {
ASSERT(table[tpl.opcode] == &Jit64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table4 = [] { constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table4 = []() consteval
{
std::array<Jit64::Instruction, 1024> table{}; std::array<Jit64::Instruction, 1024> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (u32 i = 0; i < 32; i++) for (u32 i = 0; i < 32; i++)
{ {
@ -373,6 +362,7 @@ constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table4 = [] {
for (const auto& tpl : s_table4_2) for (const auto& tpl : s_table4_2)
{ {
const u32 op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
@ -383,6 +373,7 @@ constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table4 = [] {
for (const auto& tpl : s_table4_3) for (const auto& tpl : s_table4_3)
{ {
const u32 op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
@ -390,59 +381,68 @@ constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table4 = [] {
for (const auto& tpl : s_table4) for (const auto& tpl : s_table4)
{ {
const u32 op = tpl.opcode; const u32 op = tpl.opcode;
ASSERT(table[op] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table19 = [] { constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table19 = []() consteval
{
std::array<Jit64::Instruction, 1024> table{}; std::array<Jit64::Instruction, 1024> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (const auto& tpl : s_table19) for (const auto& tpl : s_table19)
{ {
const u32 op = tpl.opcode; ASSERT(table[tpl.opcode] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table31 = [] { constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table31 = []() consteval
{
std::array<Jit64::Instruction, 1024> table{}; std::array<Jit64::Instruction, 1024> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (const auto& tpl : s_table31) for (const auto& tpl : s_table31)
{ {
const u32 op = tpl.opcode; ASSERT(table[tpl.opcode] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<Jit64::Instruction, 32> s_dyna_op_table59 = [] { constexpr std::array<Jit64::Instruction, 32> s_dyna_op_table59 = []() consteval
{
std::array<Jit64::Instruction, 32> table{}; std::array<Jit64::Instruction, 32> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (const auto& tpl : s_table59) for (const auto& tpl : s_table59)
{ {
const u32 op = tpl.opcode; ASSERT(table[tpl.opcode] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table63 = [] { constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table63 = []() consteval
{
std::array<Jit64::Instruction, 1024> table{}; std::array<Jit64::Instruction, 1024> table{};
FillWithFallbacks(table); Common::Fill(table, &Jit64::FallBackToInterpreter);
for (const auto& tpl : s_table63) for (const auto& tpl : s_table63)
{ {
const u32 op = tpl.opcode; ASSERT(table[tpl.opcode] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
for (u32 i = 0; i < 32; i++) for (u32 i = 0; i < 32; i++)
@ -451,12 +451,14 @@ constexpr std::array<Jit64::Instruction, 1024> s_dyna_op_table63 = [] {
for (const auto& tpl : s_table63_2) for (const auto& tpl : s_table63_2)
{ {
const u32 op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &Jit64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
return table; return table;
}(); }
();
} // Anonymous namespace } // Anonymous namespace
@ -489,16 +491,5 @@ void Jit64::CompileInstruction(PPCAnalyst::CodeOp& op)
{ {
(this->*s_dyna_op_table[op.inst.OPCD])(op.inst); (this->*s_dyna_op_table[op.inst.OPCD])(op.inst);
GekkoOPInfo* info = op.opinfo; PPCTables::CountInstructionCompile(op.opinfo, js.compilerPC);
if (info)
{
#ifdef OPLOG
if (!strcmp(info->opname, OP_TO_LOG)) // "mcrfs"
{
rsplocations.push_back(js.compilerPC);
}
#endif
info->compileCount++;
info->lastUse = js.compilerPC;
}
} }

View File

@ -267,7 +267,7 @@ void Jit64::dcbx(UGeckoInstruction inst)
// Alright, now figure out how many loops we want to do. // Alright, now figure out how many loops we want to do.
const u8 cycle_count_per_loop = const u8 cycle_count_per_loop =
js.op[0].opinfo->numCycles + js.op[1].opinfo->numCycles + js.op[2].opinfo->numCycles; js.op[0].opinfo->num_cycles + js.op[1].opinfo->num_cycles + js.op[2].opinfo->num_cycles;
// This is both setting the adjusted loop count to 0 for the downcount <= 0 case and clearing // This is both setting the adjusted loop count to 0 for the downcount <= 0 case and clearing
// the upper bits for the DIV instruction in the downcount > 0 case. // the upper bits for the DIV instruction in the downcount > 0 case.

View File

@ -197,7 +197,7 @@ void JitArm64::FallBackToInterpreter(UGeckoInstruction inst)
gpr.Unlock(WA); gpr.Unlock(WA);
} }
Interpreter::Instruction instr = PPCTables::GetInterpreterOp(inst); Interpreter::Instruction instr = Interpreter::GetInterpreterOp(inst);
MOVP2R(ARM64Reg::X8, instr); MOVP2R(ARM64Reg::X8, instr);
MOVI2R(ARM64Reg::W0, inst.hex); MOVI2R(ARM64Reg::W0, inst.hex);
BLR(ARM64Reg::X8); BLR(ARM64Reg::X8);
@ -903,7 +903,7 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
js.instructionNumber = i; js.instructionNumber = i;
js.instructionsLeft = (code_block.m_num_instructions - 1) - i; js.instructionsLeft = (code_block.m_num_instructions - 1) - i;
const GekkoOPInfo* opinfo = op.opinfo; const GekkoOPInfo* opinfo = op.opinfo;
js.downcountAmount += opinfo->numCycles; js.downcountAmount += opinfo->num_cycles;
js.isLastInstruction = i == (code_block.m_num_instructions - 1); js.isLastInstruction = i == (code_block.m_num_instructions - 1);
if (!m_enable_debugging) if (!m_enable_debugging)

View File

@ -677,7 +677,7 @@ void JitArm64::dcbx(UGeckoInstruction inst)
// Figure out how many loops we want to do. // Figure out how many loops we want to do.
const u8 cycle_count_per_loop = const u8 cycle_count_per_loop =
js.op[0].opinfo->numCycles + js.op[1].opinfo->numCycles + js.op[2].opinfo->numCycles; js.op[0].opinfo->num_cycles + js.op[1].opinfo->num_cycles + js.op[2].opinfo->num_cycles;
LDR(IndexType::Unsigned, reg_downcount, PPC_REG, PPCSTATE_OFF(downcount)); LDR(IndexType::Unsigned, reg_downcount, PPC_REG, PPCSTATE_OFF(downcount));
MOVI2R(WA, 0); MOVI2R(WA, 0);

View File

@ -5,19 +5,19 @@
#include <array> #include <array>
#include "Common/Assert.h"
#include "Common/TypeUtils.h"
#include "Core/PowerPC/Gekko.h" #include "Core/PowerPC/Gekko.h"
#include "Core/PowerPC/PPCTables.h"
namespace namespace
{ {
struct GekkoOPTemplate struct JitArm64OpTemplate
{ {
int opcode; u32 opcode;
JitArm64::Instruction fn; JitArm64::Instruction fn;
// GekkoOPInfo opinfo; // Doesn't need opinfo, Interpreter fills it out
}; };
constexpr std::array<GekkoOPTemplate, 54> primarytable{{ constexpr std::array<JitArm64OpTemplate, 54> s_primary_table{{
{4, &JitArm64::DynaRunTable4}, // RunTable4 {4, &JitArm64::DynaRunTable4}, // RunTable4
{19, &JitArm64::DynaRunTable19}, // RunTable19 {19, &JitArm64::DynaRunTable19}, // RunTable19
{31, &JitArm64::DynaRunTable31}, // RunTable31 {31, &JitArm64::DynaRunTable31}, // RunTable31
@ -87,7 +87,7 @@ constexpr std::array<GekkoOPTemplate, 54> primarytable{{
// missing: 0, 1, 2, 5, 6, 9, 22, 30, 58, 62 // missing: 0, 1, 2, 5, 6, 9, 22, 30, 58, 62
}}; }};
constexpr std::array<GekkoOPTemplate, 13> table4{{ constexpr std::array<JitArm64OpTemplate, 13> s_table4{{
// SUBOP10 // SUBOP10
{0, &JitArm64::ps_cmpXX}, // ps_cmpu0 {0, &JitArm64::ps_cmpXX}, // ps_cmpu0
{32, &JitArm64::ps_cmpXX}, // ps_cmpo0 {32, &JitArm64::ps_cmpXX}, // ps_cmpo0
@ -105,7 +105,7 @@ constexpr std::array<GekkoOPTemplate, 13> table4{{
{1014, &JitArm64::FallBackToInterpreter}, // dcbz_l {1014, &JitArm64::FallBackToInterpreter}, // dcbz_l
}}; }};
constexpr std::array<GekkoOPTemplate, 17> table4_2{{ constexpr std::array<JitArm64OpTemplate, 17> s_table4_2{{
{10, &JitArm64::ps_sumX}, // ps_sum0 {10, &JitArm64::ps_sumX}, // ps_sum0
{11, &JitArm64::ps_sumX}, // ps_sum1 {11, &JitArm64::ps_sumX}, // ps_sum1
{12, &JitArm64::ps_arith}, // ps_muls0 {12, &JitArm64::ps_arith}, // ps_muls0
@ -125,14 +125,14 @@ constexpr std::array<GekkoOPTemplate, 17> table4_2{{
{31, &JitArm64::ps_arith}, // ps_nmadd {31, &JitArm64::ps_arith}, // ps_nmadd
}}; }};
constexpr std::array<GekkoOPTemplate, 4> table4_3{{ constexpr std::array<JitArm64OpTemplate, 4> s_table4_3{{
{6, &JitArm64::psq_lXX}, // psq_lx {6, &JitArm64::psq_lXX}, // psq_lx
{7, &JitArm64::psq_stXX}, // psq_stx {7, &JitArm64::psq_stXX}, // psq_stx
{38, &JitArm64::psq_lXX}, // psq_lux {38, &JitArm64::psq_lXX}, // psq_lux
{39, &JitArm64::psq_stXX}, // psq_stux {39, &JitArm64::psq_stXX}, // psq_stux
}}; }};
constexpr std::array<GekkoOPTemplate, 13> table19{{ constexpr std::array<JitArm64OpTemplate, 13> s_table19{{
{528, &JitArm64::bcctrx}, // bcctrx {528, &JitArm64::bcctrx}, // bcctrx
{16, &JitArm64::bclrx}, // bclrx {16, &JitArm64::bclrx}, // bclrx
{257, &JitArm64::crXXX}, // crand {257, &JitArm64::crXXX}, // crand
@ -150,7 +150,7 @@ constexpr std::array<GekkoOPTemplate, 13> table19{{
{50, &JitArm64::rfi}, // rfi {50, &JitArm64::rfi}, // rfi
}}; }};
constexpr std::array<GekkoOPTemplate, 107> table31{{ constexpr std::array<JitArm64OpTemplate, 107> s_table31{{
{266, &JitArm64::addx}, // addx {266, &JitArm64::addx}, // addx
{778, &JitArm64::addx}, // addox {778, &JitArm64::addx}, // addox
{10, &JitArm64::addcx}, // addcx {10, &JitArm64::addcx}, // addcx
@ -292,7 +292,7 @@ constexpr std::array<GekkoOPTemplate, 107> table31{{
{566, &JitArm64::DoNothing}, // tlbsync {566, &JitArm64::DoNothing}, // tlbsync
}}; }};
constexpr std::array<GekkoOPTemplate, 9> table59{{ constexpr std::array<JitArm64OpTemplate, 9> s_table59{{
{18, &JitArm64::fp_arith}, // fdivsx {18, &JitArm64::fp_arith}, // fdivsx
{20, &JitArm64::fp_arith}, // fsubsx {20, &JitArm64::fp_arith}, // fsubsx
{21, &JitArm64::fp_arith}, // faddsx {21, &JitArm64::fp_arith}, // faddsx
@ -304,7 +304,7 @@ constexpr std::array<GekkoOPTemplate, 9> table59{{
{31, &JitArm64::fp_arith}, // fnmaddsx {31, &JitArm64::fp_arith}, // fnmaddsx
}}; }};
constexpr std::array<GekkoOPTemplate, 15> table63{{ constexpr std::array<JitArm64OpTemplate, 15> s_table63{{
{264, &JitArm64::fp_logic}, // fabsx {264, &JitArm64::fp_logic}, // fabsx
{32, &JitArm64::fcmpX}, // fcmpo {32, &JitArm64::fcmpX}, // fcmpo
{0, &JitArm64::fcmpX}, // fcmpu {0, &JitArm64::fcmpX}, // fcmpu
@ -323,7 +323,7 @@ constexpr std::array<GekkoOPTemplate, 15> table63{{
{711, &JitArm64::mtfsfx}, // mtfsfx {711, &JitArm64::mtfsfx}, // mtfsfx
}}; }};
constexpr std::array<GekkoOPTemplate, 10> table63_2{{ constexpr std::array<JitArm64OpTemplate, 10> s_table63_2{{
{18, &JitArm64::fp_arith}, // fdivx {18, &JitArm64::fp_arith}, // fdivx
{20, &JitArm64::fp_arith}, // fsubx {20, &JitArm64::fp_arith}, // fsubx
{21, &JitArm64::fp_arith}, // faddx {21, &JitArm64::fp_arith}, // faddx
@ -336,172 +336,160 @@ constexpr std::array<GekkoOPTemplate, 10> table63_2{{
{31, &JitArm64::fp_arith}, // fnmaddx {31, &JitArm64::fp_arith}, // fnmaddx
}}; }};
constexpr std::array<JitArm64::Instruction, 64> dynaOpTable = [] { constexpr std::array<JitArm64::Instruction, 64> s_dyna_op_table = []() consteval
{
std::array<JitArm64::Instruction, 64> table{}; std::array<JitArm64::Instruction, 64> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& tpl : table) for (auto& tpl : s_primary_table)
{
tpl = &JitArm64::FallBackToInterpreter;
}
for (const auto& tpl : primarytable)
{ {
ASSERT(table[tpl.opcode] == &JitArm64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<JitArm64::Instruction, 1024> dynaOpTable4 = [] { constexpr std::array<JitArm64::Instruction, 1024> s_dyna_op_table4 = []() consteval
{
std::array<JitArm64::Instruction, 1024> table{}; std::array<JitArm64::Instruction, 1024> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& entry : table) for (u32 i = 0; i < 32; i++)
{ {
entry = &JitArm64::FallBackToInterpreter; const u32 fill = i << 5;
} for (const auto& tpl : s_table4_2)
for (int i = 0; i < 32; i++)
{
const int fill = i << 5;
for (const auto& tpl : table4_2)
{ {
const int op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &JitArm64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
for (int i = 0; i < 16; i++) for (u32 i = 0; i < 16; i++)
{ {
const int fill = i << 6; const u32 fill = i << 6;
for (const auto& tpl : table4_3) for (const auto& tpl : s_table4_3)
{ {
const int op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &JitArm64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
for (const auto& tpl : table4) for (const auto& tpl : s_table4)
{ {
table[tpl.opcode] = tpl.fn; const u32 op = tpl.opcode;
ASSERT(table[op] == &JitArm64::FallBackToInterpreter);
table[op] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<JitArm64::Instruction, 1024> dynaOpTable19 = [] { constexpr std::array<JitArm64::Instruction, 1024> s_dyna_op_table19 = []() consteval
{
std::array<JitArm64::Instruction, 1024> table{}; std::array<JitArm64::Instruction, 1024> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& entry : table) for (const auto& tpl : s_table19)
{
entry = &JitArm64::FallBackToInterpreter;
}
for (const auto& tpl : table19)
{ {
ASSERT(table[tpl.opcode] == &JitArm64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<JitArm64::Instruction, 1024> dynaOpTable31 = [] { constexpr std::array<JitArm64::Instruction, 1024> s_dyna_op_table31 = []() consteval
{
std::array<JitArm64::Instruction, 1024> table{}; std::array<JitArm64::Instruction, 1024> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& entry : table) for (const auto& tpl : s_table31)
{
entry = &JitArm64::FallBackToInterpreter;
}
for (const auto& tpl : table31)
{ {
ASSERT(table[tpl.opcode] == &JitArm64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<JitArm64::Instruction, 32> dynaOpTable59 = [] { constexpr std::array<JitArm64::Instruction, 32> s_dyna_op_table59 = []() consteval
{
std::array<JitArm64::Instruction, 32> table{}; std::array<JitArm64::Instruction, 32> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& entry : table) for (const auto& tpl : s_table59)
{
entry = &JitArm64::FallBackToInterpreter;
}
for (const auto& tpl : table59)
{ {
ASSERT(table[tpl.opcode] == &JitArm64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
return table; return table;
}(); }
();
constexpr std::array<JitArm64::Instruction, 1024> dynaOpTable63 = [] { constexpr std::array<JitArm64::Instruction, 1024> s_dyna_op_table63 = []() consteval
{
std::array<JitArm64::Instruction, 1024> table{}; std::array<JitArm64::Instruction, 1024> table{};
Common::Fill(table, &JitArm64::FallBackToInterpreter);
for (auto& entry : table) for (const auto& tpl : s_table63)
{
entry = &JitArm64::FallBackToInterpreter;
}
for (const auto& tpl : table63)
{ {
ASSERT(table[tpl.opcode] == &JitArm64::FallBackToInterpreter);
table[tpl.opcode] = tpl.fn; table[tpl.opcode] = tpl.fn;
} }
for (int i = 0; i < 32; i++) for (u32 i = 0; i < 32; i++)
{ {
const int fill = i << 5; const u32 fill = i << 5;
for (const auto& tpl : table63_2) for (const auto& tpl : s_table63_2)
{ {
const int op = fill + tpl.opcode; const u32 op = fill + tpl.opcode;
ASSERT(table[op] == &JitArm64::FallBackToInterpreter);
table[op] = tpl.fn; table[op] = tpl.fn;
} }
} }
return table; return table;
}(); }
();
} // Anonymous namespace } // Anonymous namespace
void JitArm64::DynaRunTable4(UGeckoInstruction inst) void JitArm64::DynaRunTable4(UGeckoInstruction inst)
{ {
(this->*dynaOpTable4[inst.SUBOP10])(inst); (this->*s_dyna_op_table4[inst.SUBOP10])(inst);
} }
void JitArm64::DynaRunTable19(UGeckoInstruction inst) void JitArm64::DynaRunTable19(UGeckoInstruction inst)
{ {
(this->*dynaOpTable19[inst.SUBOP10])(inst); (this->*s_dyna_op_table19[inst.SUBOP10])(inst);
} }
void JitArm64::DynaRunTable31(UGeckoInstruction inst) void JitArm64::DynaRunTable31(UGeckoInstruction inst)
{ {
(this->*dynaOpTable31[inst.SUBOP10])(inst); (this->*s_dyna_op_table31[inst.SUBOP10])(inst);
} }
void JitArm64::DynaRunTable59(UGeckoInstruction inst) void JitArm64::DynaRunTable59(UGeckoInstruction inst)
{ {
(this->*dynaOpTable59[inst.SUBOP5])(inst); (this->*s_dyna_op_table59[inst.SUBOP5])(inst);
} }
void JitArm64::DynaRunTable63(UGeckoInstruction inst) void JitArm64::DynaRunTable63(UGeckoInstruction inst)
{ {
(this->*dynaOpTable63[inst.SUBOP10])(inst); (this->*s_dyna_op_table63[inst.SUBOP10])(inst);
} }
void JitArm64::CompileInstruction(PPCAnalyst::CodeOp& op) void JitArm64::CompileInstruction(PPCAnalyst::CodeOp& op)
{ {
(this->*dynaOpTable[op.inst.OPCD])(op.inst); (this->*s_dyna_op_table[op.inst.OPCD])(op.inst);
GekkoOPInfo* info = op.opinfo; PPCTables::CountInstructionCompile(op.opinfo, js.compilerPC);
if (info)
{
#ifdef OPLOG
if (!strcmp(info->opname, OP_TO_LOG))
{ ///"mcrfs"
rsplocations.push_back(js.compilerPC);
}
#endif
info->compileCount++;
info->lastUse = js.compilerPC;
}
} }

View File

@ -770,13 +770,13 @@ u32 PPCAnalyzer::Analyze(u32 address, CodeBlock* block, CodeBuffer* buffer,
num_inst++; num_inst++;
const UGeckoInstruction inst = result.hex; const UGeckoInstruction inst = result.hex;
GekkoOPInfo* opinfo = PPCTables::GetOpInfo(inst); const GekkoOPInfo* opinfo = PPCTables::GetOpInfo(inst);
code[i] = {}; code[i] = {};
code[i].opinfo = opinfo; code[i].opinfo = opinfo;
code[i].address = address; code[i].address = address;
code[i].inst = inst; code[i].inst = inst;
code[i].skip = false; code[i].skip = false;
block->m_stats->numCycles += opinfo->numCycles; block->m_stats->numCycles += opinfo->num_cycles;
block->m_physical_addresses.insert(result.physical_address); block->m_physical_addresses.insert(result.physical_address);
SetInstructionStats(block, &code[i], opinfo); SetInstructionStats(block, &code[i], opinfo);

View File

@ -29,7 +29,7 @@ namespace PPCAnalyst
struct CodeOp // 16B struct CodeOp // 16B
{ {
UGeckoInstruction inst; UGeckoInstruction inst;
GekkoOPInfo* opinfo = nullptr; const GekkoOPInfo* opinfo = nullptr;
u32 address = 0; u32 address = 0;
u32 branchTo = 0; // if UINT32_MAX, not a branch u32 branchTo = 0; // if UINT32_MAX, not a branch
BitSet32 regsOut; BitSet32 regsOut;

View File

@ -17,43 +17,623 @@
#include "Common/IOFile.h" #include "Common/IOFile.h"
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Common/StringUtil.h" #include "Common/StringUtil.h"
#include "Common/TypeUtils.h"
#include "Core/PowerPC/Interpreter/Interpreter.h" #include "Core/PowerPC/Interpreter/Interpreter.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
std::array<GekkoOPInfo*, 64> m_infoTable;
std::array<GekkoOPInfo*, 1024> m_infoTable4;
std::array<GekkoOPInfo*, 1024> m_infoTable19;
std::array<GekkoOPInfo*, 1024> m_infoTable31;
std::array<GekkoOPInfo*, 32> m_infoTable59;
std::array<GekkoOPInfo*, 1024> m_infoTable63;
std::array<GekkoOPInfo*, 512> m_allInstructions;
size_t m_numInstructions;
namespace PPCTables namespace PPCTables
{ {
GekkoOPInfo* GetOpInfo(UGeckoInstruction inst) namespace
{ {
const GekkoOPInfo* info = m_infoTable[inst.OPCD]; struct GekkoOPTemplate
{
u32 opcode;
const char* opname;
OpType type;
u32 num_cycles;
u64 flags;
};
constexpr GekkoOPTemplate s_unknown_op_info = {0, "unknown_instruction", OpType::Unknown, 0,
FL_ENDBLOCK};
constexpr std::array<GekkoOPTemplate, 54> s_primary_table{{
{4, "RunTable4", OpType::Subtable, 0, 0},
{19, "RunTable19", OpType::Subtable, 0, 0},
{31, "RunTable31", OpType::Subtable, 0, 0},
{59, "RunTable59", OpType::Subtable, 0, 0},
{63, "RunTable63", OpType::Subtable, 0, 0},
{16, "bcx", OpType::Branch, 1, FL_ENDBLOCK | FL_READ_CR_BI},
{18, "bx", OpType::Branch, 1, FL_ENDBLOCK},
{3, "twi", OpType::System, 1, FL_IN_A | FL_ENDBLOCK},
{17, "sc", OpType::System, 2, FL_ENDBLOCK},
{7, "mulli", OpType::Integer, 3, FL_OUT_D | FL_IN_A},
{8, "subfic", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_SET_CA},
{10, "cmpli", OpType::Integer, 1, FL_IN_A | FL_SET_CRn},
{11, "cmpi", OpType::Integer, 1, FL_IN_A | FL_SET_CRn},
{12, "addic", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_SET_CA},
{13, "addic_rc", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_SET_CA | FL_SET_CR0},
{14, "addi", OpType::Integer, 1, FL_OUT_D | FL_IN_A0},
{15, "addis", OpType::Integer, 1, FL_OUT_D | FL_IN_A0},
{20, "rlwimix", OpType::Integer, 1, FL_OUT_A | FL_IN_A | FL_IN_S | FL_RC_BIT},
{21, "rlwinmx", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_RC_BIT},
{23, "rlwnmx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{24, "ori", OpType::Integer, 1, FL_OUT_A | FL_IN_S},
{25, "oris", OpType::Integer, 1, FL_OUT_A | FL_IN_S},
{26, "xori", OpType::Integer, 1, FL_OUT_A | FL_IN_S},
{27, "xoris", OpType::Integer, 1, FL_OUT_A | FL_IN_S},
{28, "andi_rc", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_SET_CR0},
{29, "andis_rc", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_SET_CR0},
{32, "lwz", OpType::Load, 1, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE},
{33, "lwzu", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE},
{34, "lbz", OpType::Load, 1, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE},
{35, "lbzu", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE},
{40, "lhz", OpType::Load, 1, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE},
{41, "lhzu", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE},
{42, "lha", OpType::Load, 1, FL_OUT_D | FL_IN_A0 | FL_LOADSTORE},
{43, "lhau", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_A | FL_LOADSTORE},
{44, "sth", OpType::Store, 1, FL_IN_A0 | FL_IN_S | FL_LOADSTORE},
{45, "sthu", OpType::Store, 1, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE},
{36, "stw", OpType::Store, 1, FL_IN_A0 | FL_IN_S | FL_LOADSTORE},
{37, "stwu", OpType::Store, 1, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE},
{38, "stb", OpType::Store, 1, FL_IN_A0 | FL_IN_S | FL_LOADSTORE},
{39, "stbu", OpType::Store, 1, FL_OUT_A | FL_IN_A | FL_IN_S | FL_LOADSTORE},
{46, "lmw", OpType::System, 11, FL_EVIL | FL_IN_A0 | FL_LOADSTORE},
{47, "stmw", OpType::System, 11, FL_EVIL | FL_IN_A0 | FL_LOADSTORE},
{48, "lfs", OpType::LoadFP, 1, FL_OUT_FLOAT_D | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{49, "lfsu", OpType::LoadFP, 1,
FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{50, "lfd", OpType::LoadFP, 1, FL_INOUT_FLOAT_D | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{51, "lfdu", OpType::LoadFP, 1,
FL_INOUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{52, "stfs", OpType::StoreFP, 1, FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE},
{53, "stfsu", OpType::StoreFP, 1,
FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{54, "stfd", OpType::StoreFP, 1, FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE},
{55, "stfdu", OpType::StoreFP, 1,
FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE},
{56, "psq_l", OpType::LoadPS, 1,
FL_OUT_FLOAT_D | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
{57, "psq_lu", OpType::LoadPS, 1,
FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
{60, "psq_st", OpType::StorePS, 1,
FL_IN_FLOAT_S | FL_IN_A0 | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
{61, "psq_stu", OpType::StorePS, 1,
FL_IN_FLOAT_S | FL_OUT_A | FL_IN_A | FL_USE_FPU | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
// missing: 0, 1, 2, 5, 6, 9, 22, 30, 62, 58
}};
constexpr std::array<GekkoOPTemplate, 13> s_table4{{
// SUBOP10
{0, "ps_cmpu0", OpType::PS, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION},
{32, "ps_cmpo0", OpType::PS, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION},
{40, "ps_neg", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{136, "ps_nabs", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{264, "ps_abs", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{64, "ps_cmpu1", OpType::PS, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION},
{72, "ps_mr", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{96, "ps_cmpo1", OpType::PS, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION},
{528, "ps_merge00", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{560, "ps_merge01", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{592, "ps_merge10", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{624, "ps_merge11", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_IN_FLOAT_AB_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{1014, "dcbz_l", OpType::System, 1, FL_IN_A0B | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 17> s_table4_2{{
{10, "ps_sum0", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{11, "ps_sum1", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{12, "ps_muls0", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{13, "ps_muls1", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{14, "ps_madds0", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{15, "ps_madds1", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{18, "ps_div", OpType::PS, 17,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION | FL_FLOAT_DIV},
{20, "ps_sub", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{21, "ps_add", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{23, "ps_sel", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_IN_FLOAT_BC_BITEXACT | FL_RC_BIT_F | FL_USE_FPU |
FL_PROGRAMEXCEPTION},
{24, "ps_res", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION | FL_FLOAT_DIV},
{25, "ps_mul", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{26, "ps_rsqrte", OpType::PS, 2,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_PROGRAMEXCEPTION |
FL_FLOAT_EXCEPTION | FL_FLOAT_DIV},
{28, "ps_msub", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{29, "ps_madd", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{30, "ps_nmsub", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{31, "ps_nmadd", OpType::PS, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 4> s_table4_3{{
{6, "psq_lx", OpType::LoadPS, 1, FL_OUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{7, "psq_stx", OpType::StorePS, 1, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{38, "psq_lux", OpType::LoadPS, 1,
FL_OUT_FLOAT_D | FL_OUT_A | FL_IN_AB | FL_USE_FPU | FL_LOADSTORE},
{39, "psq_stux", OpType::StorePS, 1,
FL_IN_FLOAT_S | FL_OUT_A | FL_IN_AB | FL_USE_FPU | FL_LOADSTORE},
}};
constexpr std::array<GekkoOPTemplate, 13> s_table19{{
{528, "bcctrx", OpType::Branch, 1, FL_ENDBLOCK | FL_READ_CR_BI},
{16, "bclrx", OpType::Branch, 1, FL_ENDBLOCK | FL_READ_CR_BI},
{257, "crand", OpType::CR, 1, FL_EVIL},
{129, "crandc", OpType::CR, 1, FL_EVIL},
{289, "creqv", OpType::CR, 1, FL_EVIL},
{225, "crnand", OpType::CR, 1, FL_EVIL},
{33, "crnor", OpType::CR, 1, FL_EVIL},
{449, "cror", OpType::CR, 1, FL_EVIL},
{417, "crorc", OpType::CR, 1, FL_EVIL},
{193, "crxor", OpType::CR, 1, FL_EVIL},
{150, "isync", OpType::InstructionCache, 1, FL_EVIL},
{0, "mcrf", OpType::System, 1, FL_EVIL | FL_SET_CRn | FL_READ_CRn},
{50, "rfi", OpType::System, 2, FL_ENDBLOCK | FL_CHECKEXCEPTIONS | FL_PROGRAMEXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 107> s_table31{{
{266, "addx", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{778, "addox", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE},
{10, "addcx", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT},
{522, "addcox", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{138, "addex", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{650, "addeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{234, "addmex", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{746, "addmeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{202, "addzex", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{714, "addzeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{491, "divwx", OpType::Integer, 40, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{1003, "divwox", OpType::Integer, 40, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE},
{459, "divwux", OpType::Integer, 40, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{971, "divwuox", OpType::Integer, 40, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE},
{75, "mulhwx", OpType::Integer, 5, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{11, "mulhwux", OpType::Integer, 5, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{235, "mullwx", OpType::Integer, 5, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{747, "mullwox", OpType::Integer, 5, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE},
{104, "negx", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_RC_BIT},
{616, "negox", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_RC_BIT | FL_SET_OE},
{40, "subfx", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_RC_BIT},
{552, "subfox", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_RC_BIT | FL_SET_OE},
{8, "subfcx", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT},
{520, "subfcox", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{136, "subfex", OpType::Integer, 1, FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{648, "subfeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_AB | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{232, "subfmex", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{744, "subfmeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{200, "subfzex", OpType::Integer, 1, FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT},
{712, "subfzeox", OpType::Integer, 1,
FL_OUT_D | FL_IN_A | FL_READ_CA | FL_SET_CA | FL_RC_BIT | FL_SET_OE},
{28, "andx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{60, "andcx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{444, "orx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{124, "norx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{316, "xorx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{412, "orcx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{476, "nandx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{284, "eqvx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{0, "cmp", OpType::Integer, 1, FL_IN_AB | FL_SET_CRn},
{32, "cmpl", OpType::Integer, 1, FL_IN_AB | FL_SET_CRn},
{26, "cntlzwx", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_RC_BIT},
{922, "extshx", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_RC_BIT},
{954, "extsbx", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_RC_BIT},
{536, "srwx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{792, "srawx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_SET_CA | FL_RC_BIT},
{824, "srawix", OpType::Integer, 1, FL_OUT_A | FL_IN_S | FL_SET_CA | FL_RC_BIT},
{24, "slwx", OpType::Integer, 1, FL_OUT_A | FL_IN_SB | FL_RC_BIT},
{54, "dcbst", OpType::DataCache, 5, FL_IN_A0B | FL_LOADSTORE},
{86, "dcbf", OpType::DataCache, 5, FL_IN_A0B | FL_LOADSTORE},
{246, "dcbtst", OpType::DataCache, 2, 0},
{278, "dcbt", OpType::DataCache, 2, 0},
{470, "dcbi", OpType::DataCache, 5, FL_IN_A0B | FL_LOADSTORE | FL_PROGRAMEXCEPTION},
{758, "dcba", OpType::DataCache, 5, 0},
{1014, "dcbz", OpType::DataCache, 5, FL_IN_A0B | FL_LOADSTORE},
// load word
{23, "lwzx", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
{55, "lwzux", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// load halfword
{279, "lhzx", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
{311, "lhzux", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// load halfword signextend
{343, "lhax", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
{375, "lhaux", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// load byte
{87, "lbzx", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
{119, "lbzux", OpType::Load, 1, FL_OUT_D | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// load byte reverse
{534, "lwbrx", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
{790, "lhbrx", OpType::Load, 1, FL_OUT_D | FL_IN_A0B | FL_LOADSTORE},
// Conditional load/store (Wii SMP)
{150, "stwcxd", OpType::Store, 1, FL_EVIL | FL_IN_S | FL_IN_A0B | FL_SET_CR0 | FL_LOADSTORE},
{20, "lwarx", OpType::Load, 1, FL_EVIL | FL_OUT_D | FL_IN_A0B | FL_SET_CR0 | FL_LOADSTORE},
// load string (Inst these)
{533, "lswx", OpType::Load, 1, FL_EVIL | FL_IN_A0B | FL_OUT_D | FL_LOADSTORE},
{597, "lswi", OpType::Load, 1, FL_EVIL | FL_IN_A0 | FL_OUT_D | FL_LOADSTORE},
// store word
{151, "stwx", OpType::Store, 1, FL_IN_S | FL_IN_A0B | FL_LOADSTORE},
{183, "stwux", OpType::Store, 1, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// store halfword
{407, "sthx", OpType::Store, 1, FL_IN_S | FL_IN_A0B | FL_LOADSTORE},
{439, "sthux", OpType::Store, 1, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// store byte
{215, "stbx", OpType::Store, 1, FL_IN_S | FL_IN_A0B | FL_LOADSTORE},
{247, "stbux", OpType::Store, 1, FL_IN_S | FL_OUT_A | FL_IN_AB | FL_LOADSTORE},
// store bytereverse
{662, "stwbrx", OpType::Store, 1, FL_IN_S | FL_IN_A0B | FL_LOADSTORE},
{918, "sthbrx", OpType::Store, 1, FL_IN_S | FL_IN_A0B | FL_LOADSTORE},
{661, "stswx", OpType::Store, 1, FL_EVIL | FL_IN_A0B | FL_LOADSTORE},
{725, "stswi", OpType::Store, 1, FL_EVIL | FL_IN_A0 | FL_LOADSTORE},
// fp load/store
{535, "lfsx", OpType::LoadFP, 1, FL_OUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{567, "lfsux", OpType::LoadFP, 1,
FL_OUT_FLOAT_D | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE},
{599, "lfdx", OpType::LoadFP, 1, FL_INOUT_FLOAT_D | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{631, "lfdux", OpType::LoadFP, 1,
FL_INOUT_FLOAT_D | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE},
{663, "stfsx", OpType::StoreFP, 1, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{695, "stfsux", OpType::StoreFP, 1,
FL_IN_FLOAT_S | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE},
{727, "stfdx", OpType::StoreFP, 1, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{759, "stfdux", OpType::StoreFP, 1,
FL_IN_FLOAT_S | FL_IN_AB | FL_OUT_A | FL_USE_FPU | FL_LOADSTORE},
{983, "stfiwx", OpType::StoreFP, 1, FL_IN_FLOAT_S | FL_IN_A0B | FL_USE_FPU | FL_LOADSTORE},
{19, "mfcr", OpType::System, 1, FL_OUT_D | FL_READ_ALL_CR},
{83, "mfmsr", OpType::System, 1, FL_OUT_D | FL_PROGRAMEXCEPTION},
{144, "mtcrf", OpType::System, 1, FL_IN_S | FL_SET_ALL_CR | FL_READ_ALL_CR},
{146, "mtmsr", OpType::System, 1,
FL_IN_S | FL_ENDBLOCK | FL_PROGRAMEXCEPTION | FL_FLOAT_EXCEPTION},
{210, "mtsr", OpType::System, 1, FL_IN_S | FL_PROGRAMEXCEPTION},
{242, "mtsrin", OpType::System, 1, FL_IN_SB | FL_PROGRAMEXCEPTION},
{339, "mfspr", OpType::SPR, 1, FL_OUT_D | FL_PROGRAMEXCEPTION},
{467, "mtspr", OpType::SPR, 2, FL_IN_S | FL_PROGRAMEXCEPTION},
{371, "mftb", OpType::System, 1, FL_OUT_D | FL_TIMER | FL_PROGRAMEXCEPTION},
{512, "mcrxr", OpType::System, 1, FL_SET_CRn | FL_READ_CA | FL_SET_CA},
{595, "mfsr", OpType::System, 3, FL_OUT_D | FL_PROGRAMEXCEPTION},
{659, "mfsrin", OpType::System, 3, FL_OUT_D | FL_IN_B | FL_PROGRAMEXCEPTION},
{4, "tw", OpType::System, 2, FL_IN_AB | FL_ENDBLOCK},
{598, "sync", OpType::System, 3, 0},
{982, "icbi", OpType::System, 4, FL_IN_A0B | FL_ENDBLOCK | FL_LOADSTORE},
// Unused instructions on GC
{310, "eciwx", OpType::System, 1, FL_IN_A0B | FL_OUT_D | FL_LOADSTORE},
{438, "ecowx", OpType::System, 1, FL_IN_A0B | FL_IN_S | FL_LOADSTORE},
{854, "eieio", OpType::System, 1, 0},
{306, "tlbie", OpType::System, 1, FL_IN_B | FL_PROGRAMEXCEPTION},
{566, "tlbsync", OpType::System, 1, FL_PROGRAMEXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 9> s_table59{{
{18, "fdivsx", OpType::SingleFP, 17,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION |
FL_FLOAT_DIV}, // TODO
{20, "fsubsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{21, "faddsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{24, "fresx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION |
FL_FLOAT_DIV},
{25, "fmulsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{28, "fmsubsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{29, "fmaddsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{30, "fnmsubsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{31, "fnmaddsx", OpType::SingleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 15> s_table63{{
{264, "fabsx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU},
// FIXME: fcmp modifies the FPRF flags, but if the flags are clobbered later,
// we don't actually need to calculate or store them here. So FL_READ_FPRF and FL_SET_FPRF is
// not an ideal representation of fcmp's effect on FPRF flags and might result in slightly
// sub-optimal code.
{32, "fcmpo", OpType::DoubleFP, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{0, "fcmpu", OpType::DoubleFP, 1,
FL_IN_FLOAT_AB | FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{14, "fctiwx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_FLOAT_EXCEPTION},
{15, "fctiwzx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_FLOAT_EXCEPTION},
{72, "fmrx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_IN_FLOAT_B_BITEXACT | FL_RC_BIT_F | FL_USE_FPU},
{136, "fnabsx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_IN_FLOAT_B_BITEXACT | FL_USE_FPU},
{40, "fnegx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_IN_FLOAT_B_BITEXACT | FL_USE_FPU},
{12, "frspx", OpType::DoubleFP, 1,
FL_OUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{64, "mcrfs", OpType::SystemFP, 1, FL_SET_CRn | FL_USE_FPU | FL_READ_FPRF},
{583, "mffsx", OpType::SystemFP, 1, FL_RC_BIT_F | FL_INOUT_FLOAT_D | FL_USE_FPU | FL_READ_FPRF},
{70, "mtfsb0x", OpType::SystemFP, 3, FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF},
{38, "mtfsb1x", OpType::SystemFP, 3,
FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{134, "mtfsfix", OpType::SystemFP, 3,
FL_RC_BIT_F | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
{711, "mtfsfx", OpType::SystemFP, 3,
FL_RC_BIT_F | FL_IN_FLOAT_B | FL_USE_FPU | FL_READ_FPRF | FL_SET_FPRF | FL_FLOAT_EXCEPTION},
}};
constexpr std::array<GekkoOPTemplate, 10> s_table63_2{{
{18, "fdivx", OpType::DoubleFP, 31,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION | FL_FLOAT_DIV},
{20, "fsubx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{21, "faddx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_AB | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{23, "fselx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_IN_FLOAT_BC_BITEXACT | FL_RC_BIT_F | FL_USE_FPU},
{25, "fmulx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_AC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{26, "frsqrtex", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_B | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION | FL_FLOAT_DIV},
{28, "fmsubx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{29, "fmaddx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{30, "fnmsubx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
{31, "fnmaddx", OpType::DoubleFP, 1,
FL_INOUT_FLOAT_D | FL_IN_FLOAT_ABC | FL_RC_BIT_F | FL_USE_FPU | FL_SET_FPRF |
FL_FLOAT_EXCEPTION},
}};
constexpr size_t TOTAL_INSTRUCTION_COUNT =
1 + s_primary_table.size() + s_table4_2.size() + s_table4_3.size() + s_table4.size() +
s_table31.size() + s_table19.size() + s_table59.size() + s_table63.size() + s_table63_2.size();
struct Tables
{
std::array<GekkoOPInfo, TOTAL_INSTRUCTION_COUNT> all_instructions{};
u32 unknown_op_info;
std::array<u32, 64> primary_table{};
std::array<u32, 1024> table4{};
std::array<u32, 1024> table19{};
std::array<u32, 1024> table31{};
std::array<u32, 32> table59{};
std::array<u32, 1024> table63{};
};
} // namespace
static std::array<GekkoOPStats, TOTAL_INSTRUCTION_COUNT> s_all_instructions_stats;
constexpr Tables s_tables = []() consteval
{
Tables tables{};
u32 counter = 0;
auto make_info = [&](const GekkoOPTemplate& inst) consteval->u32
{
ASSERT(counter < TOTAL_INSTRUCTION_COUNT);
GekkoOPInfo* info = &tables.all_instructions[counter];
info->opname = inst.opname;
info->flags = inst.flags;
info->type = inst.type;
info->num_cycles = inst.num_cycles;
info->stats = &s_all_instructions_stats[counter];
return counter++;
};
u32 unknown_op_info = make_info(s_unknown_op_info);
tables.unknown_op_info = unknown_op_info;
Common::Fill(tables.primary_table, unknown_op_info);
for (auto& tpl : s_primary_table)
{
ASSERT(tables.primary_table[tpl.opcode] == unknown_op_info);
tables.primary_table[tpl.opcode] = make_info(tpl);
};
Common::Fill(tables.table4, unknown_op_info);
for (const auto& tpl : s_table4_2)
{
u32 info = make_info(tpl);
for (u32 i = 0; i < 32; i++)
{
const u32 fill = i << 5;
const u32 op = fill + tpl.opcode;
ASSERT(tables.table4[op] == unknown_op_info);
tables.table4[op] = info;
}
}
for (const auto& tpl : s_table4_3)
{
u32 info = make_info(tpl);
for (u32 i = 0; i < 16; i++)
{
const u32 fill = i << 6;
const u32 op = fill + tpl.opcode;
ASSERT(tables.table4[op] == unknown_op_info);
tables.table4[op] = info;
}
}
for (const auto& tpl : s_table4)
{
const u32 op = tpl.opcode;
ASSERT(tables.table4[op] == unknown_op_info);
tables.table4[op] = make_info(tpl);
}
Common::Fill(tables.table19, unknown_op_info);
for (auto& tpl : s_table19)
{
ASSERT(tables.table19[tpl.opcode] == unknown_op_info);
tables.table19[tpl.opcode] = make_info(tpl);
};
Common::Fill(tables.table31, unknown_op_info);
for (auto& tpl : s_table31)
{
ASSERT(tables.table31[tpl.opcode] == unknown_op_info);
tables.table31[tpl.opcode] = make_info(tpl);
};
Common::Fill(tables.table59, unknown_op_info);
for (auto& tpl : s_table59)
{
ASSERT(tables.table59[tpl.opcode] == unknown_op_info);
tables.table59[tpl.opcode] = make_info(tpl);
};
Common::Fill(tables.table63, unknown_op_info);
for (auto& tpl : s_table63)
{
ASSERT(tables.table63[tpl.opcode] == unknown_op_info);
tables.table63[tpl.opcode] = make_info(tpl);
};
for (const auto& tpl : s_table63_2)
{
u32 info = make_info(tpl);
for (u32 i = 0; i < 32; i++)
{
const u32 fill = i << 5;
const u32 op = fill + tpl.opcode;
ASSERT(tables.table63[op] == unknown_op_info);
tables.table63[op] = info;
}
}
ASSERT(counter == TOTAL_INSTRUCTION_COUNT);
return tables;
}
();
const GekkoOPInfo* GetOpInfo(UGeckoInstruction inst)
{
const GekkoOPInfo* info = &s_tables.all_instructions[s_tables.primary_table[inst.OPCD]];
if (info->type == OpType::Subtable) if (info->type == OpType::Subtable)
{ {
switch (inst.OPCD) switch (inst.OPCD)
{ {
case 4: case 4:
return m_infoTable4[inst.SUBOP10]; return &s_tables.all_instructions[s_tables.table4[inst.SUBOP10]];
case 19: case 19:
return m_infoTable19[inst.SUBOP10]; return &s_tables.all_instructions[s_tables.table19[inst.SUBOP10]];
case 31: case 31:
return m_infoTable31[inst.SUBOP10]; return &s_tables.all_instructions[s_tables.table31[inst.SUBOP10]];
case 59: case 59:
return m_infoTable59[inst.SUBOP5]; return &s_tables.all_instructions[s_tables.table59[inst.SUBOP5]];
case 63: case 63:
return m_infoTable63[inst.SUBOP10]; return &s_tables.all_instructions[s_tables.table63[inst.SUBOP10]];
default: default:
ASSERT_MSG(POWERPC, 0, "GetOpInfo - invalid subtable op {:08x} @ {:08x}", inst.hex, ASSERT_MSG(POWERPC, 0, "GetOpInfo - invalid subtable op {:08x} @ {:08x}", inst.hex,
PowerPC::ppcState.pc); PowerPC::ppcState.pc);
return nullptr; return &s_tables.all_instructions[s_tables.unknown_op_info];
} }
} }
else else
@ -62,56 +642,14 @@ GekkoOPInfo* GetOpInfo(UGeckoInstruction inst)
{ {
ASSERT_MSG(POWERPC, 0, "GetOpInfo - invalid op {:08x} @ {:08x}", inst.hex, ASSERT_MSG(POWERPC, 0, "GetOpInfo - invalid op {:08x} @ {:08x}", inst.hex,
PowerPC::ppcState.pc); PowerPC::ppcState.pc);
return nullptr; return &s_tables.all_instructions[s_tables.unknown_op_info];
} }
return m_infoTable[inst.OPCD]; return info;
} }
} }
Interpreter::Instruction GetInterpreterOp(UGeckoInstruction inst) // #define OPLOG
{ // #define OP_TO_LOG "mtfsb0x"
const GekkoOPInfo* info = m_infoTable[inst.OPCD];
if (info->type == OpType::Subtable)
{
switch (inst.OPCD)
{
case 4:
return Interpreter::m_op_table4[inst.SUBOP10];
case 19:
return Interpreter::m_op_table19[inst.SUBOP10];
case 31:
return Interpreter::m_op_table31[inst.SUBOP10];
case 59:
return Interpreter::m_op_table59[inst.SUBOP5];
case 63:
return Interpreter::m_op_table63[inst.SUBOP10];
default:
ASSERT_MSG(POWERPC, 0, "GetInterpreterOp - invalid subtable op {:08x} @ {:08x}", inst.hex,
PowerPC::ppcState.pc);
return nullptr;
}
}
else
{
if (info->type == OpType::Invalid)
{
ASSERT_MSG(POWERPC, 0, "GetInterpreterOp - invalid op {:08x} @ {:08x}", inst.hex,
PowerPC::ppcState.pc);
return nullptr;
}
return Interpreter::m_op_table[inst.OPCD];
}
}
bool UsesFPU(UGeckoInstruction inst)
{
GekkoOPInfo* const info = GetOpInfo(inst);
return (info->flags & FL_USE_FPU) != 0;
}
#define OPLOG
#define OP_TO_LOG "mtfsb0x"
#ifdef OPLOG #ifdef OPLOG
namespace namespace
@ -123,33 +661,42 @@ std::vector<u32> rsplocations;
const char* GetInstructionName(UGeckoInstruction inst) const char* GetInstructionName(UGeckoInstruction inst)
{ {
const GekkoOPInfo* info = GetOpInfo(inst); const GekkoOPInfo* info = GetOpInfo(inst);
return info ? info->opname : nullptr; return info->opname;
} }
bool IsValidInstruction(UGeckoInstruction inst) bool IsValidInstruction(UGeckoInstruction inst)
{ {
const GekkoOPInfo* info = GetOpInfo(inst); const GekkoOPInfo* info = GetOpInfo(inst);
return info != nullptr && info->type != OpType::Unknown; return info->type != OpType::Invalid && info->type != OpType::Unknown;
} }
void CountInstruction(UGeckoInstruction inst) void CountInstruction(UGeckoInstruction inst)
{ {
GekkoOPInfo* info = GetOpInfo(inst); const GekkoOPInfo* info = GetOpInfo(inst);
if (info) info->stats->run_count++;
}
void CountInstructionCompile(const GekkoOPInfo* info, u32 pc)
{
info->stats->compile_count++;
info->stats->last_use = pc;
#ifdef OPLOG
if (!strcmp(info->opname, OP_TO_LOG))
{ {
info->runCount++; rsplocations.push_back(pc);
} }
#endif
} }
void PrintInstructionRunCounts() void PrintInstructionRunCounts()
{ {
typedef std::pair<const char*, u64> OpInfo; typedef std::pair<const char*, u64> OpInfo;
std::vector<OpInfo> temp; std::array<OpInfo, TOTAL_INSTRUCTION_COUNT> temp;
temp.reserve(m_numInstructions); for (size_t i = 0; i < TOTAL_INSTRUCTION_COUNT; i++)
for (size_t i = 0; i < m_numInstructions; ++i)
{ {
GekkoOPInfo* pInst = m_allInstructions[i]; const GekkoOPInfo& info = s_tables.all_instructions[i];
temp.emplace_back(pInst->opname, pInst->runCount); temp[i] = std::make_pair(info.opname, info.stats->run_count);
} }
std::sort(temp.begin(), temp.end(), std::sort(temp.begin(), temp.end(),
[](const OpInfo& a, const OpInfo& b) { return a.second > b.second; }); [](const OpInfo& a, const OpInfo& b) { return a.second > b.second; });
@ -159,7 +706,7 @@ void PrintInstructionRunCounts()
if (inst.second == 0) if (inst.second == 0)
break; break;
DEBUG_LOG_FMT(POWERPC, "{} : {}", inst.first, inst.second); INFO_LOG_FMT(POWERPC, "{} : {}", inst.first, inst.second);
} }
} }
@ -168,24 +715,24 @@ void LogCompiledInstructions()
static unsigned int time = 0; static unsigned int time = 0;
File::IOFile f(fmt::format("{}inst_log{}.txt", File::GetUserPath(D_LOGS_IDX), time), "w"); File::IOFile f(fmt::format("{}inst_log{}.txt", File::GetUserPath(D_LOGS_IDX), time), "w");
for (size_t i = 0; i < m_numInstructions; i++) for (size_t i = 0; i < TOTAL_INSTRUCTION_COUNT; i++)
{ {
GekkoOPInfo* pInst = m_allInstructions[i]; const GekkoOPInfo& info = s_tables.all_instructions[i];
if (pInst->compileCount > 0) if (info.stats->compile_count > 0)
{ {
f.WriteString(fmt::format("{0}\t{1}\t{2}\t{3:08x}\n", pInst->opname, pInst->compileCount, f.WriteString(fmt::format("{0}\t{1}\t{2}\t{3:08x}\n", info.opname, info.stats->compile_count,
pInst->runCount, pInst->lastUse)); info.stats->run_count, info.stats->last_use));
} }
} }
f.Open(fmt::format("{}inst_not{}.txt", File::GetUserPath(D_LOGS_IDX), time), "w"); f.Open(fmt::format("{}inst_not{}.txt", File::GetUserPath(D_LOGS_IDX), time), "w");
for (size_t i = 0; i < m_numInstructions; i++) for (size_t i = 0; i < TOTAL_INSTRUCTION_COUNT; i++)
{ {
GekkoOPInfo* pInst = m_allInstructions[i]; const GekkoOPInfo& info = s_tables.all_instructions[i];
if (pInst->compileCount == 0) if (info.stats->compile_count == 0)
{ {
f.WriteString( f.WriteString(fmt::format("{0}\t{1}\t{2}\n", info.opname, info.stats->compile_count,
fmt::format("{0}\t{1}\t{2}\n", pInst->opname, pInst->compileCount, pInst->runCount)); info.stats->run_count));
} }
} }

View File

@ -5,6 +5,7 @@
#include <array> #include <array>
#include <cstddef> #include <cstddef>
#include <utility>
#include "Common/CommonTypes.h" #include "Common/CommonTypes.h"
#include "Core/PowerPC/Gekko.h" #include "Core/PowerPC/Gekko.h"
@ -98,35 +99,31 @@ enum class OpType
Unknown, Unknown,
}; };
struct GekkoOPStats
{
u64 run_count;
u32 compile_count;
u32 last_use;
};
struct GekkoOPInfo struct GekkoOPInfo
{ {
const char* opname; const char* opname;
OpType type; OpType type;
u32 num_cycles;
u64 flags; u64 flags;
int numCycles; // Mutable
u64 runCount; GekkoOPStats* stats;
int compileCount;
u32 lastUse;
}; };
extern std::array<GekkoOPInfo*, 64> m_infoTable;
extern std::array<GekkoOPInfo*, 1024> m_infoTable4;
extern std::array<GekkoOPInfo*, 1024> m_infoTable19;
extern std::array<GekkoOPInfo*, 1024> m_infoTable31;
extern std::array<GekkoOPInfo*, 32> m_infoTable59;
extern std::array<GekkoOPInfo*, 1024> m_infoTable63;
extern std::array<GekkoOPInfo*, 512> m_allInstructions;
extern size_t m_numInstructions;
namespace PPCTables namespace PPCTables
{ {
GekkoOPInfo* GetOpInfo(UGeckoInstruction inst); const GekkoOPInfo* GetOpInfo(UGeckoInstruction inst);
Interpreter::Instruction GetInterpreterOp(UGeckoInstruction inst);
bool IsValidInstruction(UGeckoInstruction inst); bool IsValidInstruction(UGeckoInstruction inst);
bool UsesFPU(UGeckoInstruction inst);
void CountInstruction(UGeckoInstruction inst); void CountInstruction(UGeckoInstruction inst);
void CountInstructionCompile(const GekkoOPInfo* info, u32 pc);
void PrintInstructionRunCounts(); void PrintInstructionRunCounts();
void LogCompiledInstructions(); void LogCompiledInstructions();
const char* GetInstructionName(UGeckoInstruction inst); const char* GetInstructionName(UGeckoInstruction inst);