Merge pull request #11695 from AdmiralCurtiss/deglobal-jitarm
JitArm64: Avoid System::GetInstance() and ppcState.
This commit is contained in:
commit
db40348c86
|
@ -50,8 +50,7 @@ void JitArm64::Init()
|
||||||
AllocCodeSpace(CODE_SIZE + child_code_size);
|
AllocCodeSpace(CODE_SIZE + child_code_size);
|
||||||
AddChildCodeSpace(&m_far_code, child_code_size);
|
AddChildCodeSpace(&m_far_code, child_code_size);
|
||||||
|
|
||||||
auto& system = Core::System::GetInstance();
|
auto& memory = m_system.GetMemory();
|
||||||
auto& memory = system.GetMemory();
|
|
||||||
|
|
||||||
jo.fastmem_arena = m_fastmem_enabled && memory.InitFastmemArena();
|
jo.fastmem_arena = m_fastmem_enabled && memory.InitFastmemArena();
|
||||||
jo.optimizeGatherPipe = true;
|
jo.optimizeGatherPipe = true;
|
||||||
|
@ -115,21 +114,18 @@ bool JitArm64::HandleFault(uintptr_t access_address, SContext* ctx)
|
||||||
// If the fault is in JIT code space, look for fastmem areas.
|
// If the fault is in JIT code space, look for fastmem areas.
|
||||||
if (!success && IsInSpace(reinterpret_cast<u8*>(ctx->CTX_PC)))
|
if (!success && IsInSpace(reinterpret_cast<u8*>(ctx->CTX_PC)))
|
||||||
{
|
{
|
||||||
auto& system = Core::System::GetInstance();
|
auto& memory = m_system.GetMemory();
|
||||||
auto& memory = system.GetMemory();
|
|
||||||
|
|
||||||
if (memory.IsAddressInFastmemArea(reinterpret_cast<u8*>(access_address)))
|
if (memory.IsAddressInFastmemArea(reinterpret_cast<u8*>(access_address)))
|
||||||
{
|
{
|
||||||
auto& ppc_state = system.GetPPCState();
|
|
||||||
const uintptr_t memory_base = reinterpret_cast<uintptr_t>(
|
const uintptr_t memory_base = reinterpret_cast<uintptr_t>(
|
||||||
ppc_state.msr.DR ? memory.GetLogicalBase() : memory.GetPhysicalBase());
|
m_ppc_state.msr.DR ? memory.GetLogicalBase() : memory.GetPhysicalBase());
|
||||||
|
|
||||||
if (access_address < memory_base || access_address >= memory_base + 0x1'0000'0000)
|
if (access_address < memory_base || access_address >= memory_base + 0x1'0000'0000)
|
||||||
{
|
{
|
||||||
ERROR_LOG_FMT(DYNA_REC,
|
ERROR_LOG_FMT(DYNA_REC,
|
||||||
"JitArm64 address calculation overflowed. This should never happen! "
|
"JitArm64 address calculation overflowed. This should never happen! "
|
||||||
"PC {:#018x}, access address {:#018x}, memory base {:#018x}, MSR.DR {}",
|
"PC {:#018x}, access address {:#018x}, memory base {:#018x}, MSR.DR {}",
|
||||||
ctx->CTX_PC, access_address, memory_base, ppc_state.msr.DR);
|
ctx->CTX_PC, access_address, memory_base, m_ppc_state.msr.DR);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -173,8 +169,7 @@ void JitArm64::ResetFreeMemoryRanges()
|
||||||
|
|
||||||
void JitArm64::Shutdown()
|
void JitArm64::Shutdown()
|
||||||
{
|
{
|
||||||
auto& system = Core::System::GetInstance();
|
auto& memory = m_system.GetMemory();
|
||||||
auto& memory = system.GetMemory();
|
|
||||||
memory.ShutdownFastmemArena();
|
memory.ShutdownFastmemArena();
|
||||||
FreeCodeSpace();
|
FreeCodeSpace();
|
||||||
blocks.Shutdown();
|
blocks.Shutdown();
|
||||||
|
@ -199,7 +194,7 @@ void JitArm64::FallBackToInterpreter(UGeckoInstruction inst)
|
||||||
|
|
||||||
Interpreter::Instruction instr = Interpreter::GetInterpreterOp(inst);
|
Interpreter::Instruction instr = Interpreter::GetInterpreterOp(inst);
|
||||||
MOVP2R(ARM64Reg::X8, instr);
|
MOVP2R(ARM64Reg::X8, instr);
|
||||||
MOVP2R(ARM64Reg::X0, &Core::System::GetInstance().GetInterpreter());
|
MOVP2R(ARM64Reg::X0, &m_system.GetInterpreter());
|
||||||
MOVI2R(ARM64Reg::W1, inst.hex);
|
MOVI2R(ARM64Reg::W1, inst.hex);
|
||||||
BLR(ARM64Reg::X8);
|
BLR(ARM64Reg::X8);
|
||||||
|
|
||||||
|
@ -277,19 +272,19 @@ void JitArm64::Cleanup()
|
||||||
CMP(ARM64Reg::X0, GPFifo::GATHER_PIPE_SIZE);
|
CMP(ARM64Reg::X0, GPFifo::GATHER_PIPE_SIZE);
|
||||||
FixupBranch exit = B(CC_LT);
|
FixupBranch exit = B(CC_LT);
|
||||||
MOVP2R(ARM64Reg::X1, &GPFifo::UpdateGatherPipe);
|
MOVP2R(ARM64Reg::X1, &GPFifo::UpdateGatherPipe);
|
||||||
MOVP2R(ARM64Reg::X0, &Core::System::GetInstance().GetGPFifo());
|
MOVP2R(ARM64Reg::X0, &m_system.GetGPFifo());
|
||||||
BLR(ARM64Reg::X1);
|
BLR(ARM64Reg::X1);
|
||||||
SetJumpTarget(exit);
|
SetJumpTarget(exit);
|
||||||
}
|
}
|
||||||
|
|
||||||
// SPEED HACK: MMCR0/MMCR1 should be checked at run-time, not at compile time.
|
// SPEED HACK: MMCR0/MMCR1 should be checked at run-time, not at compile time.
|
||||||
if (MMCR0(PowerPC::ppcState).Hex || MMCR1(PowerPC::ppcState).Hex)
|
if (MMCR0(m_ppc_state).Hex || MMCR1(m_ppc_state).Hex)
|
||||||
{
|
{
|
||||||
MOVP2R(ARM64Reg::X8, &PowerPC::UpdatePerformanceMonitor);
|
MOVP2R(ARM64Reg::X8, &PowerPC::UpdatePerformanceMonitor);
|
||||||
MOVI2R(ARM64Reg::X0, js.downcountAmount);
|
MOVI2R(ARM64Reg::X0, js.downcountAmount);
|
||||||
MOVI2R(ARM64Reg::X1, js.numLoadStoreInst);
|
MOVI2R(ARM64Reg::X1, js.numLoadStoreInst);
|
||||||
MOVI2R(ARM64Reg::X2, js.numFloatingPointInst);
|
MOVI2R(ARM64Reg::X2, js.numFloatingPointInst);
|
||||||
MOVP2R(ARM64Reg::X3, &PowerPC::ppcState);
|
MOVP2R(ARM64Reg::X3, &m_ppc_state);
|
||||||
BLR(ARM64Reg::X8);
|
BLR(ARM64Reg::X8);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -321,7 +316,7 @@ void JitArm64::IntializeSpeculativeConstants()
|
||||||
const u8* fail = nullptr;
|
const u8* fail = nullptr;
|
||||||
for (auto i : code_block.m_gpr_inputs)
|
for (auto i : code_block.m_gpr_inputs)
|
||||||
{
|
{
|
||||||
u32 compile_time_value = PowerPC::ppcState.gpr[i];
|
u32 compile_time_value = m_ppc_state.gpr[i];
|
||||||
if (PowerPC::IsOptimizableGatherPipeWrite(compile_time_value) ||
|
if (PowerPC::IsOptimizableGatherPipeWrite(compile_time_value) ||
|
||||||
PowerPC::IsOptimizableGatherPipeWrite(compile_time_value - 0x8000) ||
|
PowerPC::IsOptimizableGatherPipeWrite(compile_time_value - 0x8000) ||
|
||||||
compile_time_value == 0xCC000000)
|
compile_time_value == 0xCC000000)
|
||||||
|
@ -659,25 +654,24 @@ void JitArm64::Trace()
|
||||||
std::string fregs;
|
std::string fregs;
|
||||||
|
|
||||||
#ifdef JIT_LOG_GPR
|
#ifdef JIT_LOG_GPR
|
||||||
for (size_t i = 0; i < std::size(PowerPC::ppcState.gpr); i++)
|
for (size_t i = 0; i < std::size(m_ppc_state.gpr); i++)
|
||||||
{
|
{
|
||||||
regs += fmt::format("r{:02d}: {:08x} ", i, PowerPC::ppcState.gpr[i]);
|
regs += fmt::format("r{:02d}: {:08x} ", i, m_ppc_state.gpr[i]);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef JIT_LOG_FPR
|
#ifdef JIT_LOG_FPR
|
||||||
for (size_t i = 0; i < std::size(PowerPC::ppcState.ps); i++)
|
for (size_t i = 0; i < std::size(m_ppc_state.ps); i++)
|
||||||
{
|
{
|
||||||
fregs += fmt::format("f{:02d}: {:016x} ", i, PowerPC::ppcState.ps[i].PS0AsU64());
|
fregs += fmt::format("f{:02d}: {:016x} ", i, m_ppc_state.ps[i].PS0AsU64());
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
DEBUG_LOG_FMT(DYNA_REC,
|
DEBUG_LOG_FMT(DYNA_REC,
|
||||||
"JitArm64 PC: {:08x} SRR0: {:08x} SRR1: {:08x} FPSCR: {:08x} "
|
"JitArm64 PC: {:08x} SRR0: {:08x} SRR1: {:08x} FPSCR: {:08x} "
|
||||||
"MSR: {:08x} LR: {:08x} {} {}",
|
"MSR: {:08x} LR: {:08x} {} {}",
|
||||||
PowerPC::ppcState.pc, SRR0(PowerPC::ppcState), SRR1(PowerPC::ppcState),
|
m_ppc_state.pc, SRR0(m_ppc_state), SRR1(m_ppc_state), m_ppc_state.fpscr.Hex,
|
||||||
PowerPC::ppcState.fpscr.Hex, PowerPC::ppcState.msr.Hex, PowerPC::ppcState.spr[8],
|
m_ppc_state.msr.Hex, m_ppc_state.spr[8], regs, fregs);
|
||||||
regs, fregs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JitArm64::Jit(u32 em_address)
|
void JitArm64::Jit(u32 em_address)
|
||||||
|
@ -715,8 +709,7 @@ void JitArm64::Jit(u32 em_address, bool clear_cache_and_retry_on_failure)
|
||||||
|
|
||||||
std::size_t block_size = m_code_buffer.size();
|
std::size_t block_size = m_code_buffer.size();
|
||||||
|
|
||||||
auto& system = Core::System::GetInstance();
|
auto& cpu = m_system.GetCPU();
|
||||||
auto& cpu = system.GetCPU();
|
|
||||||
|
|
||||||
if (m_enable_debugging)
|
if (m_enable_debugging)
|
||||||
{
|
{
|
||||||
|
@ -746,8 +739,8 @@ void JitArm64::Jit(u32 em_address, bool clear_cache_and_retry_on_failure)
|
||||||
if (code_block.m_memory_exception)
|
if (code_block.m_memory_exception)
|
||||||
{
|
{
|
||||||
// Address of instruction could not be translated
|
// Address of instruction could not be translated
|
||||||
PowerPC::ppcState.npc = nextPC;
|
m_ppc_state.npc = nextPC;
|
||||||
PowerPC::ppcState.Exceptions |= EXCEPTION_ISI;
|
m_ppc_state.Exceptions |= EXCEPTION_ISI;
|
||||||
PowerPC::CheckExceptions();
|
PowerPC::CheckExceptions();
|
||||||
WARN_LOG_FMT(POWERPC, "ISI exception at {:#010x}", nextPC);
|
WARN_LOG_FMT(POWERPC, "ISI exception at {:#010x}", nextPC);
|
||||||
return;
|
return;
|
||||||
|
@ -824,8 +817,7 @@ bool JitArm64::SetEmitterStateToFreeCodeRegion()
|
||||||
|
|
||||||
bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
{
|
{
|
||||||
auto& system = Core::System::GetInstance();
|
auto& cpu = m_system.GetCPU();
|
||||||
auto& cpu = system.GetCPU();
|
|
||||||
|
|
||||||
js.isLastInstruction = false;
|
js.isLastInstruction = false;
|
||||||
js.firstFPInstructionFound = false;
|
js.firstFPInstructionFound = false;
|
||||||
|
@ -865,7 +857,7 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
js.pairedQuantizeAddresses.find(js.blockStart) == js.pairedQuantizeAddresses.end())
|
js.pairedQuantizeAddresses.find(js.blockStart) == js.pairedQuantizeAddresses.end())
|
||||||
{
|
{
|
||||||
int gqr = *code_block.m_gqr_used.begin();
|
int gqr = *code_block.m_gqr_used.begin();
|
||||||
if (!code_block.m_gqr_modified[gqr] && !GQR(PowerPC::ppcState, gqr))
|
if (!code_block.m_gqr_modified[gqr] && !GQR(m_ppc_state, gqr))
|
||||||
{
|
{
|
||||||
LDR(IndexType::Unsigned, ARM64Reg::W0, PPC_REG, PPCSTATE_OFF_SPR(SPR_GQR0 + gqr));
|
LDR(IndexType::Unsigned, ARM64Reg::W0, PPC_REG, PPCSTATE_OFF_SPR(SPR_GQR0 + gqr));
|
||||||
FixupBranch no_fail = CBZ(ARM64Reg::W0);
|
FixupBranch no_fail = CBZ(ARM64Reg::W0);
|
||||||
|
@ -936,7 +928,7 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
ABI_PushRegisters(regs_in_use);
|
ABI_PushRegisters(regs_in_use);
|
||||||
m_float_emit.ABI_PushRegisters(fprs_in_use, ARM64Reg::X30);
|
m_float_emit.ABI_PushRegisters(fprs_in_use, ARM64Reg::X30);
|
||||||
MOVP2R(ARM64Reg::X8, &GPFifo::FastCheckGatherPipe);
|
MOVP2R(ARM64Reg::X8, &GPFifo::FastCheckGatherPipe);
|
||||||
MOVP2R(ARM64Reg::X0, &Core::System::GetInstance().GetGPFifo());
|
MOVP2R(ARM64Reg::X0, &m_system.GetGPFifo());
|
||||||
BLR(ARM64Reg::X8);
|
BLR(ARM64Reg::X8);
|
||||||
m_float_emit.ABI_PopRegisters(fprs_in_use, ARM64Reg::X30);
|
m_float_emit.ABI_PopRegisters(fprs_in_use, ARM64Reg::X30);
|
||||||
ABI_PopRegisters(regs_in_use);
|
ABI_PopRegisters(regs_in_use);
|
||||||
|
@ -952,7 +944,7 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
LDR(IndexType::Unsigned, ARM64Reg::W30, PPC_REG, PPCSTATE_OFF(msr));
|
LDR(IndexType::Unsigned, ARM64Reg::W30, PPC_REG, PPCSTATE_OFF(msr));
|
||||||
TBZ(ARM64Reg::W30, 15, done_here); // MSR.EE
|
TBZ(ARM64Reg::W30, 15, done_here); // MSR.EE
|
||||||
LDR(IndexType::Unsigned, ARM64Reg::W30, ARM64Reg::X30,
|
LDR(IndexType::Unsigned, ARM64Reg::W30, ARM64Reg::X30,
|
||||||
MOVPage2R(ARM64Reg::X30, &system.GetProcessorInterface().m_interrupt_cause));
|
MOVPage2R(ARM64Reg::X30, &m_system.GetProcessorInterface().m_interrupt_cause));
|
||||||
constexpr u32 cause_mask = ProcessorInterface::INT_CAUSE_CP |
|
constexpr u32 cause_mask = ProcessorInterface::INT_CAUSE_CP |
|
||||||
ProcessorInterface::INT_CAUSE_PE_TOKEN |
|
ProcessorInterface::INT_CAUSE_PE_TOKEN |
|
||||||
ProcessorInterface::INT_CAUSE_PE_FINISH;
|
ProcessorInterface::INT_CAUSE_PE_FINISH;
|
||||||
|
@ -988,7 +980,7 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
LDR(IndexType::Unsigned, WA, PPC_REG, PPCSTATE_OFF(msr));
|
LDR(IndexType::Unsigned, WA, PPC_REG, PPCSTATE_OFF(msr));
|
||||||
TBZ(WA, 15, done_here); // MSR.EE
|
TBZ(WA, 15, done_here); // MSR.EE
|
||||||
LDR(IndexType::Unsigned, WA, XA,
|
LDR(IndexType::Unsigned, WA, XA,
|
||||||
MOVPage2R(XA, &system.GetProcessorInterface().m_interrupt_cause));
|
MOVPage2R(XA, &m_system.GetProcessorInterface().m_interrupt_cause));
|
||||||
constexpr u32 cause_mask = ProcessorInterface::INT_CAUSE_CP |
|
constexpr u32 cause_mask = ProcessorInterface::INT_CAUSE_CP |
|
||||||
ProcessorInterface::INT_CAUSE_PE_TOKEN |
|
ProcessorInterface::INT_CAUSE_PE_TOKEN |
|
||||||
ProcessorInterface::INT_CAUSE_PE_FINISH;
|
ProcessorInterface::INT_CAUSE_PE_FINISH;
|
||||||
|
|
|
@ -145,10 +145,8 @@ void JitArm64::SafeLoadToReg(u32 dest, s32 addr, s32 offsetReg, u32 flags, s32 o
|
||||||
regs_in_use[DecodeReg(ARM64Reg::W0)] = 0;
|
regs_in_use[DecodeReg(ARM64Reg::W0)] = 0;
|
||||||
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
|
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
|
||||||
regs_in_use[DecodeReg(dest_reg)] = 0;
|
regs_in_use[DecodeReg(dest_reg)] = 0;
|
||||||
auto& system = Core::System::GetInstance();
|
MMIOLoadToReg(m_system, m_system.GetMemory().GetMMIOMapping(), this, &m_float_emit, regs_in_use,
|
||||||
auto& memory = system.GetMemory();
|
fprs_in_use, dest_reg, mmio_address, flags);
|
||||||
MMIOLoadToReg(memory.GetMMIOMapping(), this, &m_float_emit, regs_in_use, fprs_in_use, dest_reg,
|
|
||||||
mmio_address, flags);
|
|
||||||
addr_reg_set = false;
|
addr_reg_set = false;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -319,10 +317,8 @@ void JitArm64::SafeStoreFromReg(s32 dest, u32 value, s32 regOffset, u32 flags, s
|
||||||
regs_in_use[DecodeReg(ARM64Reg::W1)] = 0;
|
regs_in_use[DecodeReg(ARM64Reg::W1)] = 0;
|
||||||
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
|
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
|
||||||
regs_in_use[DecodeReg(RS)] = 0;
|
regs_in_use[DecodeReg(RS)] = 0;
|
||||||
auto& system = Core::System::GetInstance();
|
MMIOWriteRegToAddr(m_system, m_system.GetMemory().GetMMIOMapping(), this, &m_float_emit,
|
||||||
auto& memory = system.GetMemory();
|
regs_in_use, fprs_in_use, RS, mmio_address, flags);
|
||||||
MMIOWriteRegToAddr(memory.GetMMIOMapping(), this, &m_float_emit, regs_in_use, fprs_in_use, RS,
|
|
||||||
mmio_address, flags);
|
|
||||||
addr_reg_set = false;
|
addr_reg_set = false;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -731,7 +727,7 @@ void JitArm64::dcbx(UGeckoInstruction inst)
|
||||||
// Translate effective address to physical address.
|
// Translate effective address to physical address.
|
||||||
const u8* loop_start = GetCodePtr();
|
const u8* loop_start = GetCodePtr();
|
||||||
FixupBranch bat_lookup_failed;
|
FixupBranch bat_lookup_failed;
|
||||||
if (PowerPC::ppcState.msr.IR)
|
if (m_ppc_state.msr.IR)
|
||||||
{
|
{
|
||||||
bat_lookup_failed =
|
bat_lookup_failed =
|
||||||
BATAddressLookup(physical_addr, effective_addr, WA, PowerPC::ibat_table.data());
|
BATAddressLookup(physical_addr, effective_addr, WA, PowerPC::ibat_table.data());
|
||||||
|
@ -760,7 +756,7 @@ void JitArm64::dcbx(UGeckoInstruction inst)
|
||||||
|
|
||||||
SwitchToFarCode();
|
SwitchToFarCode();
|
||||||
SetJumpTarget(invalidate_needed);
|
SetJumpTarget(invalidate_needed);
|
||||||
if (PowerPC::ppcState.msr.IR)
|
if (m_ppc_state.msr.IR)
|
||||||
SetJumpTarget(bat_lookup_failed);
|
SetJumpTarget(bat_lookup_failed);
|
||||||
|
|
||||||
BitSet32 gprs_to_push = gpr.GetCallerSavedUsed();
|
BitSet32 gprs_to_push = gpr.GetCallerSavedUsed();
|
||||||
|
|
|
@ -23,7 +23,7 @@ void JitArm64::psq_lXX(UGeckoInstruction inst)
|
||||||
JITDISABLE(bJITLoadStorePairedOff);
|
JITDISABLE(bJITLoadStorePairedOff);
|
||||||
|
|
||||||
// If we have a fastmem arena, the asm routines assume address translation is on.
|
// If we have a fastmem arena, the asm routines assume address translation is on.
|
||||||
FALLBACK_IF(!js.assumeNoPairedQuantize && jo.fastmem_arena && !PowerPC::ppcState.msr.DR);
|
FALLBACK_IF(!js.assumeNoPairedQuantize && jo.fastmem_arena && !m_ppc_state.msr.DR);
|
||||||
|
|
||||||
// X30 is LR
|
// X30 is LR
|
||||||
// X0 is the address
|
// X0 is the address
|
||||||
|
@ -148,7 +148,7 @@ void JitArm64::psq_stXX(UGeckoInstruction inst)
|
||||||
JITDISABLE(bJITLoadStorePairedOff);
|
JITDISABLE(bJITLoadStorePairedOff);
|
||||||
|
|
||||||
// If we have a fastmem arena, the asm routines assume address translation is on.
|
// If we have a fastmem arena, the asm routines assume address translation is on.
|
||||||
FALLBACK_IF(!js.assumeNoPairedQuantize && jo.fastmem_arena && !PowerPC::ppcState.msr.DR);
|
FALLBACK_IF(!js.assumeNoPairedQuantize && jo.fastmem_arena && !m_ppc_state.msr.DR);
|
||||||
|
|
||||||
// X30 is LR
|
// X30 is LR
|
||||||
// X0 contains the scale
|
// X0 contains the scale
|
||||||
|
|
|
@ -307,7 +307,7 @@ void JitArm64::mfspr(UGeckoInstruction inst)
|
||||||
// An inline implementation of CoreTiming::GetFakeTimeBase, since in timer-heavy games the
|
// An inline implementation of CoreTiming::GetFakeTimeBase, since in timer-heavy games the
|
||||||
// cost of calling out to C for this is actually significant.
|
// cost of calling out to C for this is actually significant.
|
||||||
|
|
||||||
auto& core_timing_globals = Core::System::GetInstance().GetCoreTiming().GetGlobals();
|
auto& core_timing_globals = m_system.GetCoreTiming().GetGlobals();
|
||||||
MOVP2R(Xg, &core_timing_globals);
|
MOVP2R(Xg, &core_timing_globals);
|
||||||
|
|
||||||
LDR(IndexType::Unsigned, WA, PPC_REG, PPCSTATE_OFF(downcount));
|
LDR(IndexType::Unsigned, WA, PPC_REG, PPCSTATE_OFF(downcount));
|
||||||
|
|
|
@ -43,7 +43,7 @@ void JitArm64::GenerateAsm()
|
||||||
ABI_PushRegisters(regs_to_save);
|
ABI_PushRegisters(regs_to_save);
|
||||||
m_float_emit.ABI_PushRegisters(regs_to_save_fpr, ARM64Reg::X30);
|
m_float_emit.ABI_PushRegisters(regs_to_save_fpr, ARM64Reg::X30);
|
||||||
|
|
||||||
MOVP2R(PPC_REG, &PowerPC::ppcState);
|
MOVP2R(PPC_REG, &m_ppc_state);
|
||||||
|
|
||||||
// Store the stack pointer, so we can reset it if the BLR optimization fails.
|
// Store the stack pointer, so we can reset it if the BLR optimization fails.
|
||||||
ADD(ARM64Reg::X0, ARM64Reg::SP, 0);
|
ADD(ARM64Reg::X0, ARM64Reg::SP, 0);
|
||||||
|
@ -67,7 +67,7 @@ void JitArm64::GenerateAsm()
|
||||||
// dispatcher_no_check:
|
// dispatcher_no_check:
|
||||||
// ExecuteBlock(JitBase::Dispatch());
|
// ExecuteBlock(JitBase::Dispatch());
|
||||||
// dispatcher:
|
// dispatcher:
|
||||||
// } while (PowerPC::ppcState.downcount > 0);
|
// } while (m_ppc_state.downcount > 0);
|
||||||
// do_timing:
|
// do_timing:
|
||||||
// NPC = PC = DISPATCHER_PC;
|
// NPC = PC = DISPATCHER_PC;
|
||||||
// } while (CPU::GetState() == CPU::State::Running);
|
// } while (CPU::GetState() == CPU::State::Running);
|
||||||
|
@ -81,8 +81,7 @@ void JitArm64::GenerateAsm()
|
||||||
|
|
||||||
dispatcher_no_timing_check = GetCodePtr();
|
dispatcher_no_timing_check = GetCodePtr();
|
||||||
|
|
||||||
auto& system = Core::System::GetInstance();
|
auto& cpu = m_system.GetCPU();
|
||||||
auto& cpu = system.GetCPU();
|
|
||||||
|
|
||||||
FixupBranch debug_exit;
|
FixupBranch debug_exit;
|
||||||
if (enable_debugging)
|
if (enable_debugging)
|
||||||
|
@ -96,7 +95,7 @@ void JitArm64::GenerateAsm()
|
||||||
|
|
||||||
bool assembly_dispatcher = true;
|
bool assembly_dispatcher = true;
|
||||||
|
|
||||||
auto& memory = system.GetMemory();
|
auto& memory = m_system.GetMemory();
|
||||||
|
|
||||||
if (assembly_dispatcher)
|
if (assembly_dispatcher)
|
||||||
{
|
{
|
||||||
|
|
|
@ -16,10 +16,10 @@ template <typename T>
|
||||||
class MMIOWriteCodeGenerator : public MMIO::WriteHandlingMethodVisitor<T>
|
class MMIOWriteCodeGenerator : public MMIO::WriteHandlingMethodVisitor<T>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
MMIOWriteCodeGenerator(ARM64XEmitter* emit, BitSet32 gprs_in_use, BitSet32 fprs_in_use,
|
MMIOWriteCodeGenerator(Core::System* system, ARM64XEmitter* emit, BitSet32 gprs_in_use,
|
||||||
ARM64Reg src_reg, u32 address)
|
BitSet32 fprs_in_use, ARM64Reg src_reg, u32 address)
|
||||||
: m_emit(emit), m_gprs_in_use(gprs_in_use), m_fprs_in_use(fprs_in_use), m_src_reg(src_reg),
|
: m_system(system), m_emit(emit), m_gprs_in_use(gprs_in_use), m_fprs_in_use(fprs_in_use),
|
||||||
m_address(address)
|
m_src_reg(src_reg), m_address(address)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,7 +78,7 @@ private:
|
||||||
|
|
||||||
m_emit->ABI_PushRegisters(m_gprs_in_use);
|
m_emit->ABI_PushRegisters(m_gprs_in_use);
|
||||||
float_emit.ABI_PushRegisters(m_fprs_in_use, ARM64Reg::X1);
|
float_emit.ABI_PushRegisters(m_fprs_in_use, ARM64Reg::X1);
|
||||||
m_emit->MOVP2R(ARM64Reg::X1, &Core::System::GetInstance());
|
m_emit->MOVP2R(ARM64Reg::X1, m_system);
|
||||||
m_emit->MOVI2R(ARM64Reg::W2, m_address);
|
m_emit->MOVI2R(ARM64Reg::W2, m_address);
|
||||||
m_emit->MOV(ARM64Reg::W3, m_src_reg);
|
m_emit->MOV(ARM64Reg::W3, m_src_reg);
|
||||||
m_emit->BLR(m_emit->ABI_SetupLambda(lambda));
|
m_emit->BLR(m_emit->ABI_SetupLambda(lambda));
|
||||||
|
@ -87,6 +87,7 @@ private:
|
||||||
m_emit->ABI_PopRegisters(m_gprs_in_use);
|
m_emit->ABI_PopRegisters(m_gprs_in_use);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Core::System* m_system;
|
||||||
ARM64XEmitter* m_emit;
|
ARM64XEmitter* m_emit;
|
||||||
BitSet32 m_gprs_in_use;
|
BitSet32 m_gprs_in_use;
|
||||||
BitSet32 m_fprs_in_use;
|
BitSet32 m_fprs_in_use;
|
||||||
|
@ -98,10 +99,10 @@ template <typename T>
|
||||||
class MMIOReadCodeGenerator : public MMIO::ReadHandlingMethodVisitor<T>
|
class MMIOReadCodeGenerator : public MMIO::ReadHandlingMethodVisitor<T>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
MMIOReadCodeGenerator(ARM64XEmitter* emit, BitSet32 gprs_in_use, BitSet32 fprs_in_use,
|
MMIOReadCodeGenerator(Core::System* system, ARM64XEmitter* emit, BitSet32 gprs_in_use,
|
||||||
ARM64Reg dst_reg, u32 address, bool sign_extend)
|
BitSet32 fprs_in_use, ARM64Reg dst_reg, u32 address, bool sign_extend)
|
||||||
: m_emit(emit), m_gprs_in_use(gprs_in_use), m_fprs_in_use(fprs_in_use), m_dst_reg(dst_reg),
|
: m_system(system), m_emit(emit), m_gprs_in_use(gprs_in_use), m_fprs_in_use(fprs_in_use),
|
||||||
m_address(address), m_sign_extend(sign_extend)
|
m_dst_reg(dst_reg), m_address(address), m_sign_extend(sign_extend)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -175,7 +176,7 @@ private:
|
||||||
|
|
||||||
m_emit->ABI_PushRegisters(m_gprs_in_use);
|
m_emit->ABI_PushRegisters(m_gprs_in_use);
|
||||||
float_emit.ABI_PushRegisters(m_fprs_in_use, ARM64Reg::X1);
|
float_emit.ABI_PushRegisters(m_fprs_in_use, ARM64Reg::X1);
|
||||||
m_emit->MOVP2R(ARM64Reg::X1, &Core::System::GetInstance());
|
m_emit->MOVP2R(ARM64Reg::X1, m_system);
|
||||||
m_emit->MOVI2R(ARM64Reg::W2, m_address);
|
m_emit->MOVI2R(ARM64Reg::W2, m_address);
|
||||||
m_emit->BLR(m_emit->ABI_SetupLambda(lambda));
|
m_emit->BLR(m_emit->ABI_SetupLambda(lambda));
|
||||||
if (m_sign_extend)
|
if (m_sign_extend)
|
||||||
|
@ -187,6 +188,7 @@ private:
|
||||||
m_emit->ABI_PopRegisters(m_gprs_in_use);
|
m_emit->ABI_PopRegisters(m_gprs_in_use);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Core::System* m_system;
|
||||||
ARM64XEmitter* m_emit;
|
ARM64XEmitter* m_emit;
|
||||||
BitSet32 m_gprs_in_use;
|
BitSet32 m_gprs_in_use;
|
||||||
BitSet32 m_fprs_in_use;
|
BitSet32 m_fprs_in_use;
|
||||||
|
@ -293,27 +295,27 @@ ARM64Reg ByteswapBeforeStore(ARM64XEmitter* emit, ARM64FloatEmitter* float_emit,
|
||||||
return dst_reg;
|
return dst_reg;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MMIOLoadToReg(MMIO::Mapping* mmio, ARM64XEmitter* emit, ARM64FloatEmitter* float_emit,
|
void MMIOLoadToReg(Core::System& system, MMIO::Mapping* mmio, ARM64XEmitter* emit,
|
||||||
BitSet32 gprs_in_use, BitSet32 fprs_in_use, ARM64Reg dst_reg, u32 address,
|
ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use, BitSet32 fprs_in_use,
|
||||||
u32 flags)
|
ARM64Reg dst_reg, u32 address, u32 flags)
|
||||||
{
|
{
|
||||||
ASSERT(!(flags & BackPatchInfo::FLAG_FLOAT));
|
ASSERT(!(flags & BackPatchInfo::FLAG_FLOAT));
|
||||||
|
|
||||||
if (flags & BackPatchInfo::FLAG_SIZE_8)
|
if (flags & BackPatchInfo::FLAG_SIZE_8)
|
||||||
{
|
{
|
||||||
MMIOReadCodeGenerator<u8> gen(emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
MMIOReadCodeGenerator<u8> gen(&system, emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
||||||
flags & BackPatchInfo::FLAG_EXTEND);
|
flags & BackPatchInfo::FLAG_EXTEND);
|
||||||
mmio->GetHandlerForRead<u8>(address).Visit(gen);
|
mmio->GetHandlerForRead<u8>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
else if (flags & BackPatchInfo::FLAG_SIZE_16)
|
else if (flags & BackPatchInfo::FLAG_SIZE_16)
|
||||||
{
|
{
|
||||||
MMIOReadCodeGenerator<u16> gen(emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
MMIOReadCodeGenerator<u16> gen(&system, emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
||||||
flags & BackPatchInfo::FLAG_EXTEND);
|
flags & BackPatchInfo::FLAG_EXTEND);
|
||||||
mmio->GetHandlerForRead<u16>(address).Visit(gen);
|
mmio->GetHandlerForRead<u16>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
else if (flags & BackPatchInfo::FLAG_SIZE_32)
|
else if (flags & BackPatchInfo::FLAG_SIZE_32)
|
||||||
{
|
{
|
||||||
MMIOReadCodeGenerator<u32> gen(emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
MMIOReadCodeGenerator<u32> gen(&system, emit, gprs_in_use, fprs_in_use, dst_reg, address,
|
||||||
flags & BackPatchInfo::FLAG_EXTEND);
|
flags & BackPatchInfo::FLAG_EXTEND);
|
||||||
mmio->GetHandlerForRead<u32>(address).Visit(gen);
|
mmio->GetHandlerForRead<u32>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
|
@ -321,9 +323,9 @@ void MMIOLoadToReg(MMIO::Mapping* mmio, ARM64XEmitter* emit, ARM64FloatEmitter*
|
||||||
ByteswapAfterLoad(emit, float_emit, dst_reg, dst_reg, flags, false, true);
|
ByteswapAfterLoad(emit, float_emit, dst_reg, dst_reg, flags, false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MMIOWriteRegToAddr(MMIO::Mapping* mmio, ARM64XEmitter* emit, ARM64FloatEmitter* float_emit,
|
void MMIOWriteRegToAddr(Core::System& system, MMIO::Mapping* mmio, ARM64XEmitter* emit,
|
||||||
BitSet32 gprs_in_use, BitSet32 fprs_in_use, ARM64Reg src_reg, u32 address,
|
ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use, BitSet32 fprs_in_use,
|
||||||
u32 flags)
|
ARM64Reg src_reg, u32 address, u32 flags)
|
||||||
{
|
{
|
||||||
ASSERT(!(flags & BackPatchInfo::FLAG_FLOAT));
|
ASSERT(!(flags & BackPatchInfo::FLAG_FLOAT));
|
||||||
|
|
||||||
|
@ -331,17 +333,17 @@ void MMIOWriteRegToAddr(MMIO::Mapping* mmio, ARM64XEmitter* emit, ARM64FloatEmit
|
||||||
|
|
||||||
if (flags & BackPatchInfo::FLAG_SIZE_8)
|
if (flags & BackPatchInfo::FLAG_SIZE_8)
|
||||||
{
|
{
|
||||||
MMIOWriteCodeGenerator<u8> gen(emit, gprs_in_use, fprs_in_use, src_reg, address);
|
MMIOWriteCodeGenerator<u8> gen(&system, emit, gprs_in_use, fprs_in_use, src_reg, address);
|
||||||
mmio->GetHandlerForWrite<u8>(address).Visit(gen);
|
mmio->GetHandlerForWrite<u8>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
else if (flags & BackPatchInfo::FLAG_SIZE_16)
|
else if (flags & BackPatchInfo::FLAG_SIZE_16)
|
||||||
{
|
{
|
||||||
MMIOWriteCodeGenerator<u16> gen(emit, gprs_in_use, fprs_in_use, src_reg, address);
|
MMIOWriteCodeGenerator<u16> gen(&system, emit, gprs_in_use, fprs_in_use, src_reg, address);
|
||||||
mmio->GetHandlerForWrite<u16>(address).Visit(gen);
|
mmio->GetHandlerForWrite<u16>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
else if (flags & BackPatchInfo::FLAG_SIZE_32)
|
else if (flags & BackPatchInfo::FLAG_SIZE_32)
|
||||||
{
|
{
|
||||||
MMIOWriteCodeGenerator<u32> gen(emit, gprs_in_use, fprs_in_use, src_reg, address);
|
MMIOWriteCodeGenerator<u32> gen(&system, emit, gprs_in_use, fprs_in_use, src_reg, address);
|
||||||
mmio->GetHandlerForWrite<u32>(address).Visit(gen);
|
mmio->GetHandlerForWrite<u32>(address).Visit(gen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,11 @@
|
||||||
|
|
||||||
#include "Core/HW/MMIO.h"
|
#include "Core/HW/MMIO.h"
|
||||||
|
|
||||||
|
namespace Core
|
||||||
|
{
|
||||||
|
class System;
|
||||||
|
}
|
||||||
|
|
||||||
void SwapPairs(Arm64Gen::ARM64XEmitter* emit, Arm64Gen::ARM64Reg dst_reg,
|
void SwapPairs(Arm64Gen::ARM64XEmitter* emit, Arm64Gen::ARM64Reg dst_reg,
|
||||||
Arm64Gen::ARM64Reg src_reg, u32 flags);
|
Arm64Gen::ARM64Reg src_reg, u32 flags);
|
||||||
|
|
||||||
|
@ -20,10 +25,10 @@ Arm64Gen::ARM64Reg ByteswapBeforeStore(Arm64Gen::ARM64XEmitter* emit,
|
||||||
Arm64Gen::ARM64Reg tmp_reg, Arm64Gen::ARM64Reg src_reg,
|
Arm64Gen::ARM64Reg tmp_reg, Arm64Gen::ARM64Reg src_reg,
|
||||||
u32 flags, bool want_reversed);
|
u32 flags, bool want_reversed);
|
||||||
|
|
||||||
void MMIOLoadToReg(MMIO::Mapping* mmio, Arm64Gen::ARM64XEmitter* emit,
|
void MMIOLoadToReg(Core::System& system, MMIO::Mapping* mmio, Arm64Gen::ARM64XEmitter* emit,
|
||||||
Arm64Gen::ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use,
|
Arm64Gen::ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use,
|
||||||
BitSet32 fprs_in_use, Arm64Gen::ARM64Reg dst_reg, u32 address, u32 flags);
|
BitSet32 fprs_in_use, Arm64Gen::ARM64Reg dst_reg, u32 address, u32 flags);
|
||||||
|
|
||||||
void MMIOWriteRegToAddr(MMIO::Mapping* mmio, Arm64Gen::ARM64XEmitter* emit,
|
void MMIOWriteRegToAddr(Core::System& system, MMIO::Mapping* mmio, Arm64Gen::ARM64XEmitter* emit,
|
||||||
Arm64Gen::ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use,
|
Arm64Gen::ARM64FloatEmitter* float_emit, BitSet32 gprs_in_use,
|
||||||
BitSet32 fprs_in_use, Arm64Gen::ARM64Reg src_reg, u32 address, u32 flags);
|
BitSet32 fprs_in_use, Arm64Gen::ARM64Reg src_reg, u32 address, u32 flags);
|
||||||
|
|
|
@ -96,7 +96,7 @@ JitBlock* JitBaseBlockCache::AllocateBlock(u32 em_address)
|
||||||
JitBlock& b = block_map.emplace(physical_address, JitBlock())->second;
|
JitBlock& b = block_map.emplace(physical_address, JitBlock())->second;
|
||||||
b.effectiveAddress = em_address;
|
b.effectiveAddress = em_address;
|
||||||
b.physicalAddress = physical_address;
|
b.physicalAddress = physical_address;
|
||||||
b.msrBits = PowerPC::ppcState.msr.Hex & JIT_CACHE_MSR_MASK;
|
b.msrBits = m_jit.m_ppc_state.msr.Hex & JIT_CACHE_MSR_MASK;
|
||||||
b.linkData.clear();
|
b.linkData.clear();
|
||||||
b.fast_block_map_index = 0;
|
b.fast_block_map_index = 0;
|
||||||
return &b;
|
return &b;
|
||||||
|
@ -168,13 +168,13 @@ JitBlock* JitBaseBlockCache::GetBlockFromStartAddress(u32 addr, u32 msr)
|
||||||
|
|
||||||
const u8* JitBaseBlockCache::Dispatch()
|
const u8* JitBaseBlockCache::Dispatch()
|
||||||
{
|
{
|
||||||
JitBlock* block = fast_block_map[FastLookupIndexForAddress(PowerPC::ppcState.pc)];
|
const auto& ppc_state = m_jit.m_ppc_state;
|
||||||
|
JitBlock* block = fast_block_map[FastLookupIndexForAddress(ppc_state.pc)];
|
||||||
|
|
||||||
if (!block || block->effectiveAddress != PowerPC::ppcState.pc ||
|
if (!block || block->effectiveAddress != ppc_state.pc ||
|
||||||
block->msrBits != (PowerPC::ppcState.msr.Hex & JIT_CACHE_MSR_MASK))
|
block->msrBits != (ppc_state.msr.Hex & JIT_CACHE_MSR_MASK))
|
||||||
{
|
{
|
||||||
block = MoveBlockIntoFastCache(PowerPC::ppcState.pc,
|
block = MoveBlockIntoFastCache(ppc_state.pc, ppc_state.msr.Hex & JIT_CACHE_MSR_MASK);
|
||||||
PowerPC::ppcState.msr.Hex & JIT_CACHE_MSR_MASK);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!block)
|
if (!block)
|
||||||
|
|
Loading…
Reference in New Issue