[JIT]: Specialize register write handlers
This commit is contained in:
5
.vscode/settings.json
vendored
Normal file
5
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"githubPullRequests.ignoredPullRequestBranches": [
|
||||||
|
"dev"
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
#include <jit/helpers.hpp>
|
#include <jit/helpers.hpp>
|
||||||
|
|
||||||
namespace n64 {
|
namespace n64 {
|
||||||
JIT::JIT(ParallelRDP ¶llel) : mem(regs, parallel) {}
|
JIT::JIT(ParallelRDP ¶llel) : regs(this), mem(regs, parallel) {}
|
||||||
|
|
||||||
bool JIT::ShouldServiceInterrupt() const {
|
bool JIT::ShouldServiceInterrupt() const {
|
||||||
const bool interrupts_pending = (regs.cop0.status.im & regs.cop0.cause.interruptPending) != 0;
|
const bool interrupts_pending = (regs.cop0.status.im & regs.cop0.cause.interruptPending) != 0;
|
||||||
|
|||||||
@@ -40,6 +40,18 @@ private:
|
|||||||
u64 cop2Latch{};
|
u64 cop2Latch{};
|
||||||
friend struct Cop1;
|
friend struct Cop1;
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
AddressFrame GPR(size_t index) {
|
||||||
|
if constexpr(sizeof(T) == 1) {
|
||||||
|
return code.byte[offsetof(JIT, regs) + offsetof(Registers, gpr) + 8 * x];
|
||||||
|
} else if constexpr(sizeof(T) == 2) {
|
||||||
|
return code.word[offsetof(JIT, regs) + offsetof(Registers, gpr) + 8 * x];
|
||||||
|
} else if constexpr(sizeof(T) == 4) {
|
||||||
|
return code.dword[offsetof(JIT, regs) + offsetof(Registers, gpr) + 8 * x];
|
||||||
|
} else if constexpr(sizeof(T) == 8) {
|
||||||
|
return code.qword[offsetof(JIT, regs) + offsetof(Registers, gpr) + 8 * x];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Credits to PCSX-Redux: https://github.com/grumpycoders/pcsx-redux
|
// Credits to PCSX-Redux: https://github.com/grumpycoders/pcsx-redux
|
||||||
// Sets dest to "pointer"
|
// Sets dest to "pointer"
|
||||||
|
|||||||
@@ -42,6 +42,5 @@ static bool InstrEndsBlock(const u32 instr) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#define REG(acc, x) code.acc[offsetof(JIT, regs) + offsetof(Registers, x)]
|
#define REG(acc, x) code.acc[offsetof(JIT, regs) + offsetof(Registers, x)]
|
||||||
#define GPR(x) code.qword[offsetof(JIT, regs) + offsetof(Registers, gpr) + 8 * x]
|
|
||||||
#define GPR_constant_marker(x) code.byte[offsetof(JIT, regs) + offsetof(Registers, gprIsConstant) + x]
|
#define GPR_constant_marker(x) code.byte[offsetof(JIT, regs) + offsetof(Registers, gprIsConstant) + x]
|
||||||
} // namespace n64
|
} // namespace n64
|
||||||
|
|||||||
@@ -76,8 +76,13 @@ void JIT::addu(u32 instr) {
|
|||||||
code.mov(code.eax, result);
|
code.mov(code.eax, result);
|
||||||
code.movsxd(code.rax, code.eax);
|
code.movsxd(code.rax, code.eax);
|
||||||
code.mov(GPR(RD(instr)), code.rax);
|
code.mov(GPR(RD(instr)), code.rax);
|
||||||
} else {
|
code.mov(REG(byte, gprIsConstant), 1);
|
||||||
Util::panic("[JIT]: Implement non constant ADDI");
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (regs.IsRegConstant(RS(instr))) {
|
||||||
|
const s32 rs = regs.Read<s32>(RS(instr));
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
#include <core/registers/Registers.hpp>
|
#include <core/registers/Registers.hpp>
|
||||||
|
#include <core/JIT.hpp>
|
||||||
|
|
||||||
namespace n64 {
|
namespace n64 {
|
||||||
Registers::Registers() : cop0(*this), cop1(*this) { Reset(); }
|
Registers::Registers(JIT* jit) : jit(jit), cop0(*this), cop1(*this) { Reset(); }
|
||||||
|
|
||||||
void Registers::Reset() {
|
void Registers::Reset() {
|
||||||
hi = 0;
|
hi = 0;
|
||||||
@@ -74,6 +75,15 @@ template <>
|
|||||||
void Registers::Write<bool>(size_t idx, bool v) {
|
void Registers::Write<bool>(size_t idx, bool v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.rax, v);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = v;
|
gpr[idx] = v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -82,6 +92,15 @@ template <>
|
|||||||
void Registers::Write<u64>(size_t idx, u64 v) {
|
void Registers::Write<u64>(size_t idx, u64 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.rax, v);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = v;
|
gpr[idx] = v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -95,6 +114,15 @@ template <>
|
|||||||
void Registers::Write<u32>(size_t idx, u32 v) {
|
void Registers::Write<u32>(size_t idx, u32 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.rax, v);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = (u32)v;
|
gpr[idx] = (u32)v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -103,6 +131,16 @@ template <>
|
|||||||
void Registers::Write<s32>(size_t idx, s32 v) {
|
void Registers::Write<s32>(size_t idx, s32 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.eax, v);
|
||||||
|
jit->code.movsxd(jit->code.rax, jit->code.eax);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = v;
|
gpr[idx] = v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -111,6 +149,15 @@ template <>
|
|||||||
void Registers::Write<u16>(size_t idx, u16 v) {
|
void Registers::Write<u16>(size_t idx, u16 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.rax, v);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = (u16)v;
|
gpr[idx] = (u16)v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -119,6 +166,16 @@ template <>
|
|||||||
void Registers::Write<s16>(size_t idx, s16 v) {
|
void Registers::Write<s16>(size_t idx, s16 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.ax, v);
|
||||||
|
jit->code.movsx(jit->code.rax, jit->code.ax);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = v;
|
gpr[idx] = v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -127,6 +184,15 @@ template <>
|
|||||||
void Registers::Write<u8>(size_t idx, u8 v) {
|
void Registers::Write<u8>(size_t idx, u8 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.rax, v);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = (u8)v;
|
gpr[idx] = (u8)v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
@@ -135,6 +201,16 @@ template <>
|
|||||||
void Registers::Write<s8>(size_t idx, s8 v) {
|
void Registers::Write<s8>(size_t idx, s8 v) {
|
||||||
if (idx == 0)
|
if (idx == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(jit) {
|
||||||
|
jit->code.mov(jit->code.al, v);
|
||||||
|
jit->code.movsx(jit->code.rax, jit->code.al);
|
||||||
|
jit->code.mov(GPR(idx), jit->code.rax);
|
||||||
|
jit->code.mov(jit->code.rax, 1);
|
||||||
|
jit->code.mov(GPR_constant_marker(idx), jit->code.rax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
gpr[idx] = v;
|
gpr[idx] = v;
|
||||||
gprIsConstant[idx] = true;
|
gprIsConstant[idx] = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,9 @@
|
|||||||
#include <backend/core/registers/Cop1.hpp>
|
#include <backend/core/registers/Cop1.hpp>
|
||||||
|
|
||||||
namespace n64 {
|
namespace n64 {
|
||||||
|
struct JIT;
|
||||||
struct Registers {
|
struct Registers {
|
||||||
Registers();
|
Registers(JIT* jit = nullptr);
|
||||||
void Reset();
|
void Reset();
|
||||||
void SetPC64(s64);
|
void SetPC64(s64);
|
||||||
void SetPC32(s32);
|
void SetPC32(s32);
|
||||||
@@ -19,6 +20,8 @@ struct Registers {
|
|||||||
return IsRegConstant(first) && IsRegConstant(second);
|
return IsRegConstant(first) && IsRegConstant(second);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JIT* jit = nullptr;
|
||||||
|
|
||||||
std::array<bool, 32> gprIsConstant{};
|
std::array<bool, 32> gprIsConstant{};
|
||||||
bool loIsConstant = false, hiIsConstant = false;
|
bool loIsConstant = false, hiIsConstant = false;
|
||||||
Cop0 cop0;
|
Cop0 cop0;
|
||||||
@@ -41,6 +44,10 @@ struct Registers {
|
|||||||
T Read(size_t);
|
T Read(size_t);
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void Write(size_t, T);
|
void Write(size_t, T);
|
||||||
|
|
||||||
std::array<s64, 32> gpr{};
|
std::array<s64, 32> gpr{};
|
||||||
|
private:
|
||||||
|
template <typename T>
|
||||||
|
void WriteJIT(size_t, T);
|
||||||
};
|
};
|
||||||
} // namespace n64
|
} // namespace n64
|
||||||
|
|||||||
Reference in New Issue
Block a user