JitArm64: Add flush/discard support for condition registers
By flushing the condition registers as soon as we no longer need them, we reduce the register pressure.
This commit is contained in:
parent
6cc4f593e5
commit
8e9609df6e
|
@ -205,6 +205,7 @@ void JitArm64::FallBackToInterpreter(UGeckoInstruction inst)
|
||||||
// we must mark them as no longer discarded
|
// we must mark them as no longer discarded
|
||||||
gpr.ResetRegisters(js.op->regsOut);
|
gpr.ResetRegisters(js.op->regsOut);
|
||||||
fpr.ResetRegisters(js.op->GetFregsOut());
|
fpr.ResetRegisters(js.op->GetFregsOut());
|
||||||
|
gpr.ResetCRRegisters(js.op->crOut);
|
||||||
|
|
||||||
if (js.op->opinfo->flags & FL_ENDBLOCK)
|
if (js.op->opinfo->flags & FL_ENDBLOCK)
|
||||||
{
|
{
|
||||||
|
@ -1199,9 +1200,11 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
||||||
{
|
{
|
||||||
gpr.DiscardRegisters(op.gprDiscardable);
|
gpr.DiscardRegisters(op.gprDiscardable);
|
||||||
fpr.DiscardRegisters(op.fprDiscardable);
|
fpr.DiscardRegisters(op.fprDiscardable);
|
||||||
|
gpr.DiscardCRRegisters(op.crDiscardable);
|
||||||
}
|
}
|
||||||
gpr.StoreRegisters(~op.gprInUse & (op.regsIn | op.regsOut));
|
gpr.StoreRegisters(~op.gprInUse & (op.regsIn | op.regsOut));
|
||||||
fpr.StoreRegisters(~op.fprInUse & (op.fregsIn | op.GetFregsOut()));
|
fpr.StoreRegisters(~op.fprInUse & (op.fregsIn | op.GetFregsOut()));
|
||||||
|
gpr.StoreCRRegisters(~op.crInUse & (op.crIn | op.crOut));
|
||||||
|
|
||||||
if (opinfo->flags & FL_LOADSTORE)
|
if (opinfo->flags & FL_LOADSTORE)
|
||||||
++js.numLoadStoreInst;
|
++js.numLoadStoreInst;
|
||||||
|
|
|
@ -129,6 +129,8 @@ void Arm64RegCache::DiscardRegister(size_t preg)
|
||||||
{
|
{
|
||||||
OpArg& reg = m_guest_registers[preg];
|
OpArg& reg = m_guest_registers[preg];
|
||||||
ARM64Reg host_reg = reg.GetReg();
|
ARM64Reg host_reg = reg.GetReg();
|
||||||
|
if (!IsVector(host_reg))
|
||||||
|
host_reg = EncodeRegTo32(host_reg);
|
||||||
|
|
||||||
reg.Discard();
|
reg.Discard();
|
||||||
if (host_reg != ARM64Reg::INVALID_REG)
|
if (host_reg != ARM64Reg::INVALID_REG)
|
||||||
|
@ -288,6 +290,25 @@ void Arm64GPRCache::FlushCRRegisters(BitSet8 regs, bool maintain_state, ARM64Reg
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Arm64GPRCache::DiscardCRRegisters(BitSet8 regs)
|
||||||
|
{
|
||||||
|
for (int i : regs)
|
||||||
|
DiscardRegister(GUEST_CR_OFFSET + i);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Arm64GPRCache::ResetCRRegisters(BitSet8 regs)
|
||||||
|
{
|
||||||
|
for (int i : regs)
|
||||||
|
{
|
||||||
|
OpArg& reg = m_guest_registers[GUEST_CR_OFFSET + i];
|
||||||
|
ARM64Reg host_reg = reg.GetReg();
|
||||||
|
|
||||||
|
ASSERT_MSG(DYNA_REC, host_reg == ARM64Reg::INVALID_REG,
|
||||||
|
"Attempted to reset a loaded register (did you mean to flush it?)");
|
||||||
|
reg.Flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void Arm64GPRCache::Flush(FlushMode mode, ARM64Reg tmp_reg)
|
void Arm64GPRCache::Flush(FlushMode mode, ARM64Reg tmp_reg)
|
||||||
{
|
{
|
||||||
FlushRegisters(BitSet32(0xFFFFFFFF), mode == FlushMode::MaintainState, tmp_reg);
|
FlushRegisters(BitSet32(0xFFFFFFFF), mode == FlushMode::MaintainState, tmp_reg);
|
||||||
|
|
|
@ -335,6 +335,9 @@ public:
|
||||||
FlushCRRegisters(regs, false, tmp_reg);
|
FlushCRRegisters(regs, false, tmp_reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DiscardCRRegisters(BitSet8 regs);
|
||||||
|
void ResetCRRegisters(BitSet8 regs);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// Get the order of the host registers
|
// Get the order of the host registers
|
||||||
void GetAllocationOrder() override;
|
void GetAllocationOrder() override;
|
||||||
|
|
Loading…
Reference in New Issue