#pragma once
#include "Core/MIPS/MIPS.h"
#include "ArmEmitter.h"
namespace ArmJitConstants {
const ArmGen::ARMReg JITBASEREG = ArmGen::R9;
const ArmGen::ARMReg CTXREG = ArmGen::R10;
const ArmGen::ARMReg MEMBASEREG = ArmGen::R11;
const ArmGen::ARMReg SCRATCHREG1 = ArmGen::R0;
const ArmGen::ARMReg SCRATCHREG2 = ArmGen::R14;
const ArmGen::ARMReg DOWNCOUNTREG = ArmGen::R7;
enum {
TOTAL_MAPPABLE_MIPSREGS = 36,
};
enum RegMIPSLoc {
ML_IMM,
ML_ARMREG,
ML_ARMREG_AS_PTR,
ML_ARMREG_IMM,
ML_MEM,
};
enum {
MAP_DIRTY = 1,
MAP_NOINIT = 2 | MAP_DIRTY,
};
}
namespace MIPSAnalyst {
struct AnalysisResults;
};
typedef int MIPSReg;
struct RegARM {
MIPSGPReg mipsReg;
bool isDirty;
};
struct RegMIPS {
ArmJitConstants::RegMIPSLoc loc;
u32 imm;
ArmGen::ARMReg reg;
bool spillLock;
};
namespace MIPSComp {
struct JitOptions;
struct JitState;
}
class ArmRegCache {
public:
ArmRegCache(MIPSState *mipsState, MIPSComp::JitState *js, MIPSComp::JitOptions *jo);
~ArmRegCache() {}
void Init(ArmGen::ARMXEmitter *emitter);
void Start(MIPSAnalyst::AnalysisResults &stats);
void SpillLock(MIPSGPReg reg, MIPSGPReg reg2 = MIPS_REG_INVALID, MIPSGPReg reg3 = MIPS_REG_INVALID, MIPSGPReg reg4 = MIPS_REG_INVALID);
void ReleaseSpillLock(MIPSGPReg reg);
void ReleaseSpillLocks();
void SetImm(MIPSGPReg reg, u32 immVal);
bool IsImm(MIPSGPReg reg) const;
u32 GetImm(MIPSGPReg reg) const;
void SetRegImm(ArmGen::ARMReg reg, u32 imm);
ArmGen::ARMReg MapReg(MIPSGPReg reg, int mapFlags = 0);
ArmGen::ARMReg MapRegAsPointer(MIPSGPReg reg);
bool IsMapped(MIPSGPReg reg);
bool IsMappedAsPointer(MIPSGPReg reg);
void MapInIn(MIPSGPReg rd, MIPSGPReg rs);
void MapDirtyIn(MIPSGPReg rd, MIPSGPReg rs, bool avoidLoad = true);
void MapDirtyInIn(MIPSGPReg rd, MIPSGPReg rs, MIPSGPReg rt, bool avoidLoad = true);
void MapDirtyDirtyIn(MIPSGPReg rd1, MIPSGPReg rd2, MIPSGPReg rs, bool avoidLoad = true);
void MapDirtyDirtyInIn(MIPSGPReg rd1, MIPSGPReg rd2, MIPSGPReg rs, MIPSGPReg rt, bool avoidLoad = true);
void FlushArmReg(ArmGen::ARMReg r);
void FlushR(MIPSGPReg r);
void FlushBeforeCall();
void FlushAll();
void DiscardR(MIPSGPReg r);
ArmGen::ARMReg R(MIPSGPReg preg);
ArmGen::ARMReg RPtr(MIPSGPReg preg);
void SetEmitter(ArmGen::ARMXEmitter *emitter) { emit_ = emitter; }
void SetCompilerPC(u32 compilerPC) { compilerPC_ = compilerPC; }
int GetMipsRegOffset(MIPSGPReg r);
private:
const ArmGen::ARMReg *GetMIPSAllocationOrder(int &count);
void MapRegTo(ArmGen::ARMReg reg, MIPSGPReg mipsReg, int mapFlags);
int FlushGetSequential(MIPSGPReg startMipsReg, bool allowFlushImm);
ArmGen::ARMReg FindBestToSpill(bool unusedOnly, bool *clobbered);
MIPSState *mips_;
ArmGen::ARMXEmitter *emit_;
MIPSComp::JitState *js_;
MIPSComp::JitOptions *jo_;
u32 compilerPC_;
enum {
NUM_ARMREG = 16,
NUM_MIPSREG = ArmJitConstants::TOTAL_MAPPABLE_MIPSREGS,
};
RegARM ar[NUM_ARMREG];
RegMIPS mr[NUM_MIPSREG];
};