1939 lines
61 KiB
C++
1939 lines
61 KiB
C++
#include "stdafx.h"
|
|
|
|
#if defined(__i386__) || defined(_M_IX86)
|
|
#include <Project64-core/N64System/N64System.h>
|
|
#include <Project64-core/N64System/Recompiler/Recompiler.h>
|
|
#include <Project64-core/N64System/Recompiler/x86/x86RegInfo.h>
|
|
#include <Project64-core/N64System/SystemGlobals.h>
|
|
|
|
#include <float.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
uint32_t CX86RegInfo::m_fpuControl = 0;
|
|
|
|
const char * Format_Name[] = {"Unknown", "dword", "qword", "float", "double"};
|
|
|
|
x86RegIndex GetIndexFromX86Reg(const asmjit::x86::Gp & Reg)
|
|
{
|
|
if (Reg == asmjit::x86::eax)
|
|
{
|
|
return x86RegIndex_EAX;
|
|
}
|
|
if (Reg == asmjit::x86::ebx)
|
|
{
|
|
return x86RegIndex_EBX;
|
|
}
|
|
if (Reg == asmjit::x86::ecx)
|
|
{
|
|
return x86RegIndex_ECX;
|
|
}
|
|
if (Reg == asmjit::x86::edx)
|
|
{
|
|
return x86RegIndex_EDX;
|
|
}
|
|
if (Reg == asmjit::x86::esi)
|
|
{
|
|
return x86RegIndex_ESI;
|
|
}
|
|
if (Reg == asmjit::x86::edi)
|
|
{
|
|
return x86RegIndex_EDI;
|
|
}
|
|
if (Reg == asmjit::x86::ebp)
|
|
{
|
|
return x86RegIndex_EBP;
|
|
}
|
|
if (Reg == asmjit::x86::esp)
|
|
{
|
|
return x86RegIndex_ESP;
|
|
}
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86RegIndex_EAX;
|
|
}
|
|
|
|
asmjit::x86::Gp GetX86RegFromIndex(x86RegIndex Index)
|
|
{
|
|
switch (Index)
|
|
{
|
|
case x86RegIndex_EAX: return asmjit::x86::eax;
|
|
case x86RegIndex_EBX: return asmjit::x86::ebx;
|
|
case x86RegIndex_ECX: return asmjit::x86::ecx;
|
|
case x86RegIndex_EDX: return asmjit::x86::edx;
|
|
case x86RegIndex_ESP: return asmjit::x86::esp;
|
|
case x86RegIndex_EBP: return asmjit::x86::ebp;
|
|
case x86RegIndex_ESI: return asmjit::x86::esi;
|
|
case x86RegIndex_EDI: return asmjit::x86::edi;
|
|
}
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
x86RegFpuIndex GetIndexFromX86FpuReg(const asmjit::x86::St & Reg)
|
|
{
|
|
if (Reg == asmjit::x86::st0)
|
|
{
|
|
return x86RegFpuIndex_ST0;
|
|
}
|
|
if (Reg == asmjit::x86::st1)
|
|
{
|
|
return x86RegFpuIndex_ST1;
|
|
}
|
|
if (Reg == asmjit::x86::st2)
|
|
{
|
|
return x86RegFpuIndex_ST2;
|
|
}
|
|
if (Reg == asmjit::x86::st3)
|
|
{
|
|
return x86RegFpuIndex_ST3;
|
|
}
|
|
if (Reg == asmjit::x86::st4)
|
|
{
|
|
return x86RegFpuIndex_ST4;
|
|
}
|
|
if (Reg == asmjit::x86::st5)
|
|
{
|
|
return x86RegFpuIndex_ST5;
|
|
}
|
|
if (Reg == asmjit::x86::st6)
|
|
{
|
|
return x86RegFpuIndex_ST6;
|
|
}
|
|
if (Reg == asmjit::x86::st7)
|
|
{
|
|
return x86RegFpuIndex_ST7;
|
|
}
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86RegFpuIndex_ST0;
|
|
}
|
|
|
|
asmjit::x86::St GetX86FpuRegFromIndex(x86RegFpuIndex Index)
|
|
{
|
|
switch (Index)
|
|
{
|
|
case x86RegFpuIndex_ST0: return asmjit::x86::st0;
|
|
case x86RegFpuIndex_ST1: return asmjit::x86::st1;
|
|
case x86RegFpuIndex_ST2: return asmjit::x86::st2;
|
|
case x86RegFpuIndex_ST3: return asmjit::x86::st3;
|
|
case x86RegFpuIndex_ST4: return asmjit::x86::st4;
|
|
case x86RegFpuIndex_ST5: return asmjit::x86::st5;
|
|
case x86RegFpuIndex_ST6: return asmjit::x86::st6;
|
|
case x86RegFpuIndex_ST7: return asmjit::x86::st7;
|
|
}
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return asmjit::x86::St();
|
|
}
|
|
|
|
CX86RegInfo::CX86RegInfo(CCodeBlock & CodeBlock, CX86Ops & Assembler) :
|
|
m_Reg(CodeBlock.Registers()),
|
|
m_CodeBlock(CodeBlock),
|
|
m_Assembler(Assembler),
|
|
m_Stack_TopPos(0),
|
|
m_InBeforeCallDirect(false)
|
|
{
|
|
for (int32_t i = 0; i < 32; i++)
|
|
{
|
|
m_RegMapLo[i] = x86Reg_Unknown;
|
|
m_RegMapHi[i] = x86Reg_Unknown;
|
|
}
|
|
for (int32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
m_x86reg_MappedTo[i] = NotMapped;
|
|
m_x86reg_Protected[i] = false;
|
|
m_x86reg_MapOrder[i] = 0;
|
|
}
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
m_x86fpu_MappedTo[i] = -1;
|
|
m_x86fpu_State[i] = FPU_Unknown;
|
|
m_x86fpu_StateChanged[i] = false;
|
|
m_x86fpu_RoundingModel[i] = RoundDefault;
|
|
}
|
|
}
|
|
|
|
CX86RegInfo::CX86RegInfo(const CX86RegInfo & rhs) :
|
|
m_Reg(rhs.m_Reg),
|
|
m_CodeBlock(rhs.m_CodeBlock),
|
|
m_Assembler(rhs.m_CodeBlock.RecompilerOps()->Assembler())
|
|
{
|
|
*this = rhs;
|
|
}
|
|
|
|
CX86RegInfo::~CX86RegInfo()
|
|
{
|
|
}
|
|
|
|
CX86RegInfo & CX86RegInfo::operator=(const CX86RegInfo & right)
|
|
{
|
|
CRegBase::operator=(right);
|
|
m_Stack_TopPos = right.m_Stack_TopPos;
|
|
m_InBeforeCallDirect = right.m_InBeforeCallDirect;
|
|
|
|
memcpy(&m_RegMapLo, &right.m_RegMapLo, sizeof(m_RegMapLo));
|
|
memcpy(&m_RegMapHi, &right.m_RegMapHi, sizeof(m_RegMapHi));
|
|
memcpy(&m_x86reg_MappedTo, &right.m_x86reg_MappedTo, sizeof(m_x86reg_MappedTo));
|
|
memcpy(&m_x86reg_Protected, &right.m_x86reg_Protected, sizeof(m_x86reg_Protected));
|
|
memcpy(&m_x86reg_MapOrder, &right.m_x86reg_MapOrder, sizeof(m_x86reg_MapOrder));
|
|
|
|
memcpy(&m_x86fpu_MappedTo, &right.m_x86fpu_MappedTo, sizeof(m_x86fpu_MappedTo));
|
|
memcpy(&m_x86fpu_State, &right.m_x86fpu_State, sizeof(m_x86fpu_State));
|
|
memcpy(&m_x86fpu_StateChanged, &right.m_x86fpu_StateChanged, sizeof(m_x86fpu_StateChanged));
|
|
memcpy(&m_x86fpu_RoundingModel, &right.m_x86fpu_RoundingModel, sizeof(m_x86fpu_RoundingModel));
|
|
|
|
#ifdef _DEBUG
|
|
if (*this != right)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
#endif
|
|
return *this;
|
|
}
|
|
|
|
bool CX86RegInfo::operator==(const CX86RegInfo & right) const
|
|
{
|
|
if (!CRegBase::operator==(right))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
int32_t count;
|
|
|
|
for (count = 0; count < 10; count++)
|
|
{
|
|
if (m_x86reg_MappedTo[count] != right.m_x86reg_MappedTo[count])
|
|
{
|
|
return false;
|
|
}
|
|
if (m_x86reg_Protected[count] != right.m_x86reg_Protected[count])
|
|
{
|
|
return false;
|
|
}
|
|
if (m_x86reg_MapOrder[count] != right.m_x86reg_MapOrder[count])
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
if (m_Stack_TopPos != right.m_Stack_TopPos)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
for (count = 0; count < x86RegFpuIndex_Size; count++)
|
|
{
|
|
if (m_x86fpu_MappedTo[count] != right.m_x86fpu_MappedTo[count])
|
|
{
|
|
return false;
|
|
}
|
|
if (m_x86fpu_State[count] != right.m_x86fpu_State[count])
|
|
{
|
|
return false;
|
|
}
|
|
if (m_x86fpu_RoundingModel[count] != right.m_x86fpu_RoundingModel[count])
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CX86RegInfo::operator!=(const CX86RegInfo & right) const
|
|
{
|
|
return !(right == *this);
|
|
}
|
|
|
|
CX86RegInfo::REG_STATE CX86RegInfo::ConstantsType(int64_t Value)
|
|
{
|
|
if (((Value >> 32) == -1) && ((Value & 0x80000000) != 0))
|
|
{
|
|
return STATE_CONST_32_SIGN;
|
|
}
|
|
if (((Value >> 32) == 0) && ((Value & 0x80000000) == 0))
|
|
{
|
|
return STATE_CONST_32_SIGN;
|
|
}
|
|
return STATE_CONST_64;
|
|
}
|
|
|
|
void CX86RegInfo::BeforeCallDirect(void)
|
|
{
|
|
if (m_InBeforeCallDirect)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_InBeforeCallDirect = true;
|
|
m_Assembler.pushad();
|
|
}
|
|
|
|
void CX86RegInfo::AfterCallDirect(void)
|
|
{
|
|
if (!m_InBeforeCallDirect)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_InBeforeCallDirect = false;
|
|
m_Assembler.popad();
|
|
SetRoundingModel(CRegInfo::RoundUnknown);
|
|
}
|
|
|
|
void CX86RegInfo::FixRoundModel(FPU_ROUND RoundMethod)
|
|
{
|
|
if (GetRoundingModel() == RoundMethod)
|
|
{
|
|
return;
|
|
}
|
|
m_CodeBlock.Log(" FixRoundModel: CurrentRoundingModel: %s targetRoundModel: %s", RoundingModelName(GetRoundingModel()), RoundingModelName(RoundMethod));
|
|
|
|
m_fpuControl = 0;
|
|
m_Assembler.fpuStoreControl(&m_fpuControl, "m_fpuControl");
|
|
asmjit::x86::Gp reg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(reg, &m_fpuControl, "m_fpuControl");
|
|
m_Assembler.and_(reg, 0xF3FF);
|
|
|
|
if (RoundMethod == RoundDefault)
|
|
{
|
|
asmjit::x86::Gp RoundReg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.OrVariableToX86Reg(reg, &CX86RecompilerOps::m_RoundingModeValue, "m_RoundingModeValue");
|
|
SetX86Protected(GetIndexFromX86Reg(RoundReg), false);
|
|
}
|
|
else
|
|
{
|
|
switch (RoundMethod)
|
|
{
|
|
case RoundTruncate: m_Assembler.or_(reg, 0x0C00); break;
|
|
case RoundNearest: m_Assembler.or_(reg, 0x0000); break;
|
|
case RoundDown: m_Assembler.or_(reg, 0x0400); break;
|
|
case RoundUp: m_Assembler.or_(reg, 0x0800); break;
|
|
default:
|
|
g_Notify->DisplayError("Unknown rounding model");
|
|
}
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_fpuControl, "m_fpuControl", reg);
|
|
SetX86Protected(GetIndexFromX86Reg(reg), false);
|
|
m_Assembler.fpuLoadControl(&m_fpuControl, "m_fpuControl");
|
|
SetRoundingModel(RoundMethod);
|
|
}
|
|
|
|
void CX86RegInfo::ChangeFPURegFormat(int32_t Reg, FPU_STATE OldFormat, FPU_STATE NewFormat, FPU_ROUND RoundingModel)
|
|
{
|
|
for (uint32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] != OldFormat || m_x86fpu_StateChanged[i])
|
|
{
|
|
UnMap_FPR(Reg, true);
|
|
Load_FPR_ToTop(Reg, Reg, OldFormat);
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log(" regcache: Changed format of ST(%d) from %s to %s", (i - StackTopPos() + 8) & 7, Format_Name[OldFormat], Format_Name[NewFormat]);
|
|
}
|
|
FpuRoundingModel(i) = RoundingModel;
|
|
m_x86fpu_State[i] = NewFormat;
|
|
m_x86fpu_StateChanged[i] = true;
|
|
return;
|
|
}
|
|
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError("ChangeFormat: Register not on stack!");
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::FPRValuePointer(int32_t Reg, FPU_STATE Format)
|
|
{
|
|
if (Reg < 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
if (RegInStack(Reg, Format))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp TempReg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
if (!TempReg.isValid())
|
|
{
|
|
return TempReg;
|
|
}
|
|
switch (Format)
|
|
{
|
|
case FPU_Dword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[Reg], stdstr_f("m_FPR_S[%d]", Reg).c_str());
|
|
break;
|
|
case FPU_Qword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[Reg], stdstr_f("m_FPR_D[%d]", Reg).c_str());
|
|
break;
|
|
case FPU_Float:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[Reg], stdstr_f("m_FPR_S[%d]", Reg).c_str());
|
|
break;
|
|
case FPU_Double:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[Reg], stdstr_f("m_FPR_D[%d]", Reg).c_str());
|
|
break;
|
|
default:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return TempReg;
|
|
}
|
|
|
|
void CX86RegInfo::Load_FPR_ToTop(int32_t Reg, int32_t RegToLoad, FPU_STATE Format)
|
|
{
|
|
if (GetRoundingModel() != RoundDefault)
|
|
{
|
|
FixRoundModel(RoundDefault);
|
|
}
|
|
m_CodeBlock.Log("CurrentRoundingModel: %s FpuRoundingModel(StackTopPos()): %s", RoundingModelName(GetRoundingModel()), RoundingModelName(FpuRoundingModel(StackTopPos())));
|
|
|
|
if (RegToLoad < 0)
|
|
{
|
|
g_Notify->DisplayError("Load_FPR_ToTop\nRegToLoad < 0 ???");
|
|
return;
|
|
}
|
|
if (Reg < 0)
|
|
{
|
|
g_Notify->DisplayError("Load_FPR_ToTop\nReg < 0 ???");
|
|
return;
|
|
}
|
|
|
|
if (Format == FPU_Double || Format == FPU_Qword)
|
|
{
|
|
UnMap_FPR(Reg + 1, true);
|
|
UnMap_FPR(RegToLoad + 1, true);
|
|
}
|
|
else
|
|
{
|
|
if ((Reg & 1) != 0)
|
|
{
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != (Reg - 1))
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] == FPU_Double || m_x86fpu_State[i] == FPU_Qword)
|
|
{
|
|
UnMap_FPR(Reg, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if ((RegToLoad & 1) != 0)
|
|
{
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != (RegToLoad - 1))
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] == FPU_Double || m_x86fpu_State[i] == FPU_Qword)
|
|
{
|
|
UnMap_FPR(RegToLoad, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (Reg == RegToLoad)
|
|
{
|
|
// If different format then unmap original register from stack
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] != Format)
|
|
{
|
|
UnMap_FPR(Reg, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If different format then unmap original register from stack
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
UnMap_FPR(Reg, m_x86fpu_State[i] != Format);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (RegInStack(RegToLoad, Format))
|
|
{
|
|
if (Reg != RegToLoad)
|
|
{
|
|
if (m_x86fpu_MappedTo[(StackTopPos() - 1) & 7] != RegToLoad)
|
|
{
|
|
UnMap_FPR(m_x86fpu_MappedTo[(StackTopPos() - 1) & 7], true);
|
|
m_CodeBlock.Log(" regcache: allocate ST(0) to %s", CRegName::FPR[Reg]);
|
|
m_Assembler.fpuLoadReg(StackTopPos(), StackPosition(RegToLoad));
|
|
FpuRoundingModel(StackTopPos()) = RoundDefault;
|
|
m_x86fpu_MappedTo[StackTopPos()] = Reg;
|
|
m_x86fpu_State[StackTopPos()] = Format;
|
|
m_x86fpu_StateChanged[StackTopPos()] = false;
|
|
}
|
|
else
|
|
{
|
|
UnMap_FPR(m_x86fpu_MappedTo[(StackTopPos() - 1) & 7], true);
|
|
Load_FPR_ToTop(Reg, RegToLoad, Format);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int32_t RegPos = -1;
|
|
for (uint32_t z = 0; z < x86RegFpuIndex_Size; z++)
|
|
{
|
|
if (m_x86fpu_MappedTo[z] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
RegPos = z;
|
|
break;
|
|
}
|
|
|
|
if (RegPos == StackTopPos())
|
|
{
|
|
return;
|
|
}
|
|
asmjit::x86::St StackPos = StackPosition(Reg);
|
|
|
|
FpuRoundingModel(RegPos) = FpuRoundingModel(StackTopPos());
|
|
m_x86fpu_MappedTo[RegPos] = m_x86fpu_MappedTo[StackTopPos()];
|
|
m_x86fpu_State[RegPos] = m_x86fpu_State[StackTopPos()];
|
|
m_x86fpu_StateChanged[RegPos] = m_x86fpu_StateChanged[StackTopPos()];
|
|
//m_CodeBlock.Log(" regcache: allocate ST(%d) to %s", StackPos, CRegName::FPR[m_x86fpu_MappedTo[RegPos]]);
|
|
m_CodeBlock.Log(" regcache: allocate ST(0) to %s", CRegName::FPR[Reg]);
|
|
|
|
m_Assembler.fxch(StackPos);
|
|
|
|
FpuRoundingModel(StackTopPos()) = RoundDefault;
|
|
m_x86fpu_MappedTo[StackTopPos()] = Reg;
|
|
m_x86fpu_State[StackTopPos()] = Format;
|
|
m_x86fpu_StateChanged[StackTopPos()] = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
UnMap_FPR(m_x86fpu_MappedTo[(StackTopPos() - 1) & 7], true);
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] == RegToLoad)
|
|
{
|
|
UnMap_FPR(RegToLoad, true);
|
|
i = 8;
|
|
}
|
|
}
|
|
m_CodeBlock.Log(" regcache: allocate ST(0) to %s", CRegName::FPR[Reg]);
|
|
asmjit::x86::Gp TempReg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
switch (Format)
|
|
{
|
|
case FPU_Dword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[RegToLoad], stdstr_f("m_FPR_S[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadIntegerDwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Qword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[RegToLoad], stdstr_f("m_FPR_D[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadIntegerQwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Float:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[RegToLoad], stdstr_f("m_FPR_S[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadDwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Double:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[RegToLoad], stdstr_f("m_FPR_D[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadQwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
default:
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("Load_FPR_ToTop\nUnkown format to load %d", Format).c_str());
|
|
}
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
FpuRoundingModel(StackTopPos()) = RoundDefault;
|
|
m_x86fpu_MappedTo[StackTopPos()] = Reg;
|
|
m_x86fpu_State[StackTopPos()] = Format;
|
|
m_x86fpu_StateChanged[StackTopPos()] = false;
|
|
}
|
|
}
|
|
|
|
const asmjit::x86::St & CX86RegInfo::StackPosition(int32_t Reg)
|
|
{
|
|
static const asmjit::x86::St StRegs[] = {
|
|
asmjit::x86::st0,
|
|
asmjit::x86::st1,
|
|
asmjit::x86::st2,
|
|
asmjit::x86::st3,
|
|
asmjit::x86::st4,
|
|
asmjit::x86::st5,
|
|
asmjit::x86::st6,
|
|
asmjit::x86::st7,
|
|
};
|
|
for (int32_t i = 0, n = sizeof(StRegs) / sizeof(StRegs[0]); i < n; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] == Reg)
|
|
{
|
|
return StRegs[((i - StackTopPos()) & 7)];
|
|
}
|
|
}
|
|
static asmjit::x86::St Unknown;
|
|
return Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::FreeX86Reg()
|
|
{
|
|
if (GetX86Mapped(x86RegIndex_EDI) == NotMapped && !GetX86Protected(x86RegIndex_EDI))
|
|
{
|
|
return asmjit::x86::edi;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_ESI) == NotMapped && !GetX86Protected(x86RegIndex_ESI))
|
|
{
|
|
return asmjit::x86::esi;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_EBX) == NotMapped && !GetX86Protected(x86RegIndex_EBX))
|
|
{
|
|
return asmjit::x86::ebx;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_EAX) == NotMapped && !GetX86Protected(x86RegIndex_EAX))
|
|
{
|
|
return asmjit::x86::eax;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_EDX) == NotMapped && !GetX86Protected(x86RegIndex_EDX))
|
|
{
|
|
return asmjit::x86::edx;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_ECX) == NotMapped && !GetX86Protected(x86RegIndex_ECX))
|
|
{
|
|
return asmjit::x86::ecx;
|
|
}
|
|
|
|
asmjit::x86::Gp Reg = UnMap_TempReg();
|
|
if (Reg.isValid())
|
|
{
|
|
return Reg;
|
|
}
|
|
|
|
uint32_t MapCount[x86RegIndex_Size];
|
|
x86RegIndex MapReg[x86RegIndex_Size];
|
|
|
|
for (int i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
MapCount[i] = GetX86MapOrder((x86RegIndex)i);
|
|
MapReg[i] = (x86RegIndex)i;
|
|
}
|
|
for (int i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
for (int32_t z = 0; z < x86RegIndex_Size - 1; z++)
|
|
{
|
|
if (MapCount[z] < MapCount[z + 1])
|
|
{
|
|
uint32_t TempCount = MapCount[z];
|
|
MapCount[z] = MapCount[z + 1];
|
|
MapCount[z + 1] = TempCount;
|
|
x86RegIndex tempReg = MapReg[z];
|
|
MapReg[z] = MapReg[z + 1];
|
|
MapReg[z + 1] = tempReg;
|
|
}
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp StackReg;
|
|
for (int i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
if (MapCount[i] > 0 && GetX86Mapped(MapReg[i]) != Stack_Mapped)
|
|
{
|
|
Reg = GetX86RegFromIndex(MapReg[i]);
|
|
if (UnMap_X86reg(Reg))
|
|
{
|
|
return Reg;
|
|
}
|
|
}
|
|
if (GetX86Mapped(MapReg[i]) == Stack_Mapped)
|
|
{
|
|
StackReg = GetX86RegFromIndex(MapReg[i]);
|
|
}
|
|
}
|
|
if (StackReg.isValid())
|
|
{
|
|
UnMap_X86reg(StackReg);
|
|
return StackReg;
|
|
}
|
|
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::Free8BitX86Reg()
|
|
{
|
|
if (GetX86Mapped(x86RegIndex_EBX) == NotMapped && !GetX86Protected(x86RegIndex_EBX))
|
|
{
|
|
return asmjit::x86::ebx;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_EAX) == NotMapped && !GetX86Protected(x86RegIndex_EAX))
|
|
{
|
|
return asmjit::x86::eax;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_EDX) == NotMapped && !GetX86Protected(x86RegIndex_EDX))
|
|
{
|
|
return asmjit::x86::edx;
|
|
}
|
|
if (GetX86Mapped(x86RegIndex_ECX) == NotMapped && !GetX86Protected(x86RegIndex_ECX))
|
|
{
|
|
return asmjit::x86::ecx;
|
|
}
|
|
|
|
asmjit::x86::Gp Reg = UnMap_8BitTempReg();
|
|
if (Reg.isValid())
|
|
{
|
|
return Reg;
|
|
}
|
|
|
|
uint32_t MapCount[x86RegIndex_Size];
|
|
x86RegIndex MapReg[x86RegIndex_Size];
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
MapCount[i] = GetX86MapOrder((x86RegIndex)i);
|
|
MapReg[i] = (x86RegIndex)i;
|
|
}
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
for (uint32_t z = 0; z < x86RegIndex_Size - 1; z++)
|
|
{
|
|
if (MapCount[z] < MapCount[z + 1])
|
|
{
|
|
uint32_t TempCount = MapCount[z];
|
|
MapCount[z] = MapCount[z + 1];
|
|
MapCount[z + 1] = TempCount;
|
|
x86RegIndex TempIndex = MapReg[z];
|
|
MapReg[z] = MapReg[z + 1];
|
|
MapReg[z + 1] = TempIndex;
|
|
}
|
|
}
|
|
}
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
if (MapCount[i] > 0)
|
|
{
|
|
if (!CX86Ops::Is8BitReg(GetX86RegFromIndex((x86RegIndex)i)))
|
|
{
|
|
continue;
|
|
}
|
|
if (UnMap_X86reg(GetX86RegFromIndex((x86RegIndex)i)))
|
|
{
|
|
return GetX86RegFromIndex((x86RegIndex)i);
|
|
}
|
|
}
|
|
}
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::UnMap_8BitTempReg()
|
|
{
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
if (!CX86Ops::Is8BitReg(GetX86RegFromIndex((x86RegIndex)i)))
|
|
{
|
|
continue;
|
|
}
|
|
if (GetX86Mapped((x86RegIndex)i) == Temp_Mapped)
|
|
{
|
|
if (GetX86Protected((x86RegIndex)i) == false)
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from temp storage", CX86Ops::x86_Name(GetX86RegFromIndex((x86RegIndex)i)));
|
|
SetX86Mapped((x86RegIndex)i, CX86RegInfo::NotMapped);
|
|
return GetX86RegFromIndex((x86RegIndex)i);
|
|
}
|
|
}
|
|
}
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::Map_FPStatusReg()
|
|
{
|
|
for (int32_t i = 0, n = x86RegIndex_Size; i < n; i++)
|
|
{
|
|
if (GetX86Mapped((x86RegIndex)i) == FPStatusReg_Mapped)
|
|
{
|
|
return GetX86RegFromIndex((x86RegIndex)i);
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp Reg = FreeX86Reg();
|
|
if (!Reg.isValid())
|
|
{
|
|
g_Notify->DisplayError("Map_MemoryStack\n\nOut of registers");
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return Reg;
|
|
}
|
|
SetX86Mapped(GetIndexFromX86Reg(Reg), CX86RegInfo::FPStatusReg_Mapped);
|
|
m_CodeBlock.Log(" regcache: allocate %s as FP Status Reg", CX86Ops::x86_Name(Reg));
|
|
m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->m_FPCR[31], "FPCR[31]");
|
|
return Reg;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::Get_MemoryStack() const
|
|
{
|
|
for (int32_t i = 0, n = x86RegIndex_Size; i < n; i++)
|
|
{
|
|
if (GetX86Mapped((x86RegIndex)i) == Stack_Mapped)
|
|
{
|
|
return GetX86RegFromIndex((x86RegIndex)i);
|
|
}
|
|
}
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::Map_MemoryStack(asmjit::x86::Gp Reg, bool bMapRegister, bool LoadValue)
|
|
{
|
|
asmjit::x86::Gp CurrentMap = Get_MemoryStack();
|
|
if (!bMapRegister)
|
|
{
|
|
// If not mapping then just return what the current mapping is
|
|
return CurrentMap;
|
|
}
|
|
|
|
if (CurrentMap.isValid() && CurrentMap == Reg)
|
|
{
|
|
// Already mapped to correct register
|
|
return CurrentMap;
|
|
}
|
|
// Map a register
|
|
if (!Reg.isValid())
|
|
{
|
|
if (CurrentMap.isValid())
|
|
{
|
|
return CurrentMap;
|
|
}
|
|
Reg = FreeX86Reg();
|
|
if (!Reg.isValid())
|
|
{
|
|
g_Notify->DisplayError("Map_MemoryStack\n\nOut of registers");
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
SetX86Mapped(GetIndexFromX86Reg(Reg), CX86RegInfo::Stack_Mapped);
|
|
m_CodeBlock.Log(" regcache: allocate %s as Memory Stack", CX86Ops::x86_Name(Reg));
|
|
if (LoadValue)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, &g_Recompiler->MemoryStackPos(), "MemoryStack");
|
|
}
|
|
return Reg;
|
|
}
|
|
|
|
// Move to a register/allocate register
|
|
UnMap_X86reg(Reg);
|
|
if (CurrentMap.isValid())
|
|
{
|
|
m_CodeBlock.Log(" regcache: change allocation of memory stack from %s to %s", CX86Ops::x86_Name(CurrentMap), CX86Ops::x86_Name(Reg));
|
|
SetX86Mapped(GetIndexFromX86Reg(Reg), CX86RegInfo::Stack_Mapped);
|
|
SetX86Mapped(GetIndexFromX86Reg(CurrentMap), CX86RegInfo::NotMapped);
|
|
m_Assembler.mov(Reg, CurrentMap);
|
|
}
|
|
else
|
|
{
|
|
SetX86Mapped(GetIndexFromX86Reg(Reg), CX86RegInfo::Stack_Mapped);
|
|
m_CodeBlock.Log(" regcache: allocate %s as memory stack", CX86Ops::x86_Name(Reg));
|
|
if (LoadValue)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, &g_Recompiler->MemoryStackPos(), "MemoryStack");
|
|
}
|
|
}
|
|
return Reg;
|
|
}
|
|
|
|
void CX86RegInfo::Map_GPR_32bit(int32_t MipsReg, bool SignValue, int32_t MipsRegToLoad)
|
|
{
|
|
asmjit::x86::Gp Reg;
|
|
if (MipsReg == 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
|
|
if (IsUnknown(MipsReg) || IsConst(MipsReg))
|
|
{
|
|
Reg = FreeX86Reg();
|
|
if (Reg.isNone())
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError("Map_GPR_32bit\n\nOut of registers");
|
|
}
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
m_CodeBlock.Log(" regcache: allocate %s to %s", CX86Ops::x86_Name(Reg), CRegName::GPR[MipsReg]);
|
|
}
|
|
else
|
|
{
|
|
if (Is64Bit(MipsReg))
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from high 32-bit of %s", CX86Ops::x86_Name(GetMipsRegMapHi(MipsReg)), CRegName::GPR_Hi[MipsReg]);
|
|
x86RegIndex RegIndex = GetIndexFromX86Reg(GetMipsRegMapHi(MipsReg));
|
|
SetX86MapOrder(RegIndex, 0);
|
|
SetX86Mapped(RegIndex, NotMapped);
|
|
SetX86Protected(RegIndex, false);
|
|
SetMipsRegHi(MipsReg, 0);
|
|
}
|
|
Reg = GetMipsRegMapLo(MipsReg);
|
|
}
|
|
for (int i = 0; i < sizeof(m_x86reg_MapOrder) / sizeof(m_x86reg_MapOrder[0]); i++)
|
|
{
|
|
if (m_x86reg_MapOrder[i] > 0)
|
|
{
|
|
m_x86reg_MapOrder[i] += 1;
|
|
}
|
|
}
|
|
x86RegIndex RegIndex = GetIndexFromX86Reg(Reg);
|
|
SetX86MapOrder(RegIndex, 1);
|
|
|
|
if (MipsRegToLoad > 0)
|
|
{
|
|
if (IsUnknown(MipsRegToLoad))
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[MipsRegToLoad].UW[0], CRegName::GPR_Lo[MipsRegToLoad]);
|
|
}
|
|
else if (IsMapped(MipsRegToLoad))
|
|
{
|
|
if (MipsReg != MipsRegToLoad)
|
|
{
|
|
m_Assembler.mov(Reg, GetMipsRegMapLo(MipsRegToLoad));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, GetMipsRegLo(MipsRegToLoad));
|
|
}
|
|
}
|
|
else if (MipsRegToLoad == 0)
|
|
{
|
|
m_Assembler.xor_(Reg, Reg);
|
|
}
|
|
SetX86Mapped(RegIndex, GPR_Mapped);
|
|
SetX86Protected(RegIndex, true);
|
|
SetMipsRegMapLo(MipsReg, Reg);
|
|
SetMipsRegState(MipsReg, SignValue ? STATE_MAPPED_32_SIGN : STATE_MAPPED_32_ZERO);
|
|
}
|
|
|
|
void CX86RegInfo::Map_GPR_64bit(int32_t MipsReg, int32_t MipsRegToLoad)
|
|
{
|
|
asmjit::x86::Gp x86Hi, x86lo;
|
|
if (MipsReg == 0)
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError("Map_GPR_32bit\n\nWhy are you trying to map register 0?");
|
|
}
|
|
return;
|
|
}
|
|
|
|
ProtectGPR(MipsReg);
|
|
if (IsUnknown(MipsReg) || IsConst(MipsReg))
|
|
{
|
|
x86Hi = FreeX86Reg();
|
|
if (!x86Hi.isValid())
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError("Map_GPR_64bit\n\nOut of registers");
|
|
}
|
|
return;
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(x86Hi), true);
|
|
|
|
x86lo = FreeX86Reg();
|
|
if (!x86lo.isValid())
|
|
{
|
|
g_Notify->DisplayError("Map_GPR_64bit\n\nOut of registers");
|
|
return;
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(x86lo), true);
|
|
|
|
m_CodeBlock.Log(" regcache: allocate %s to hi word of %s", CX86Ops::x86_Name(x86Hi), CRegName::GPR[MipsReg]);
|
|
m_CodeBlock.Log(" regcache: allocate %s to low word of %s", CX86Ops::x86_Name(x86lo), CRegName::GPR[MipsReg]);
|
|
}
|
|
else
|
|
{
|
|
x86lo = GetMipsRegMapLo(MipsReg);
|
|
if (Is32Bit(MipsReg))
|
|
{
|
|
SetX86Protected(GetIndexFromX86Reg(x86lo), true);
|
|
x86Hi = FreeX86Reg();
|
|
if (!x86Hi.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(x86Hi), true);
|
|
|
|
m_CodeBlock.Log(" regcache: allocate %s to hi word of %s", CX86Ops::x86_Name(x86Hi), CRegName::GPR[MipsReg]);
|
|
}
|
|
else
|
|
{
|
|
x86Hi = GetMipsRegMapHi(MipsReg);
|
|
}
|
|
}
|
|
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
uint32_t MapOrder = GetX86MapOrder((x86RegIndex)i);
|
|
if (MapOrder > 0)
|
|
{
|
|
SetX86MapOrder((x86RegIndex)i, MapOrder + 1);
|
|
}
|
|
}
|
|
|
|
SetX86MapOrder(GetIndexFromX86Reg(x86Hi), 1);
|
|
SetX86MapOrder(GetIndexFromX86Reg(x86lo), 1);
|
|
if (MipsRegToLoad > 0)
|
|
{
|
|
if (IsUnknown(MipsRegToLoad))
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(x86Hi, &m_Reg.m_GPR[MipsRegToLoad].UW[1], CRegName::GPR_Hi[MipsRegToLoad]);
|
|
m_Assembler.MoveVariableToX86reg(x86lo, &m_Reg.m_GPR[MipsRegToLoad].UW[0], CRegName::GPR_Lo[MipsRegToLoad]);
|
|
}
|
|
else if (IsMapped(MipsRegToLoad))
|
|
{
|
|
if (Is32Bit(MipsRegToLoad))
|
|
{
|
|
if (IsSigned(MipsRegToLoad))
|
|
{
|
|
m_Assembler.mov(x86Hi, GetMipsRegMapLo(MipsRegToLoad));
|
|
m_Assembler.sar(x86Hi, 31);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.xor_(x86Hi, x86Hi);
|
|
}
|
|
if (MipsReg != MipsRegToLoad)
|
|
{
|
|
m_Assembler.mov(x86lo, GetMipsRegMapLo(MipsRegToLoad));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (MipsReg != MipsRegToLoad)
|
|
{
|
|
m_Assembler.mov(x86Hi, GetMipsRegMapHi(MipsRegToLoad));
|
|
m_Assembler.mov(x86lo, GetMipsRegMapLo(MipsRegToLoad));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log("Map_GPR_64bit 11");
|
|
if (Is32Bit(MipsRegToLoad))
|
|
{
|
|
if (IsSigned(MipsRegToLoad))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(x86Hi, GetMipsRegLo_S(MipsRegToLoad) >> 31);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(x86Hi, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(x86Hi, GetMipsRegHi(MipsRegToLoad));
|
|
}
|
|
m_Assembler.MoveConstToX86reg(x86lo, GetMipsRegLo(MipsRegToLoad));
|
|
}
|
|
}
|
|
else if (MipsRegToLoad == 0)
|
|
{
|
|
m_Assembler.xor_(x86Hi, x86Hi);
|
|
m_Assembler.xor_(x86lo, x86lo);
|
|
}
|
|
SetX86Mapped(GetIndexFromX86Reg(x86Hi), GPR_Mapped);
|
|
SetX86Mapped(GetIndexFromX86Reg(x86lo), GPR_Mapped);
|
|
SetMipsRegMapHi(MipsReg, x86Hi);
|
|
SetMipsRegMapLo(MipsReg, x86lo);
|
|
SetMipsRegState(MipsReg, STATE_MAPPED_64);
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::Map_TempReg(asmjit::x86::Gp Reg, int32_t MipsReg, bool LoadHiWord, bool Reg8Bit)
|
|
{
|
|
if (!Reg8Bit && !Reg.isValid())
|
|
{
|
|
if (GetX86Mapped(x86RegIndex_EAX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EAX))
|
|
{
|
|
Reg = asmjit::x86::eax;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EBX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EBX))
|
|
{
|
|
Reg = asmjit::x86::ebx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ECX) == Temp_Mapped && !GetX86Protected(x86RegIndex_ECX))
|
|
{
|
|
Reg = asmjit::x86::ecx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EDX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EDX))
|
|
{
|
|
Reg = asmjit::x86::edx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ESI) == Temp_Mapped && !GetX86Protected(x86RegIndex_ESI))
|
|
{
|
|
Reg = asmjit::x86::esi;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EDI) == Temp_Mapped && !GetX86Protected(x86RegIndex_EDI))
|
|
{
|
|
Reg = asmjit::x86::edi;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EBP) == Temp_Mapped && !GetX86Protected(x86RegIndex_EBP))
|
|
{
|
|
Reg = asmjit::x86::ebp;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ESP) == Temp_Mapped && !GetX86Protected(x86RegIndex_ESP))
|
|
{
|
|
Reg = asmjit::x86::esp;
|
|
}
|
|
|
|
if (!Reg.isValid())
|
|
{
|
|
Reg = FreeX86Reg();
|
|
if (!Reg.isValid())
|
|
{
|
|
WriteTrace(TraceRegisterCache, TraceError, "Failed to find a free register");
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
}
|
|
}
|
|
else if (Reg8Bit && !Reg.isValid())
|
|
{
|
|
if (GetX86Mapped(x86RegIndex_EAX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EAX))
|
|
{
|
|
Reg = asmjit::x86::eax;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EBX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EBX))
|
|
{
|
|
Reg = asmjit::x86::ebx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ECX) == Temp_Mapped && !GetX86Protected(x86RegIndex_ECX))
|
|
{
|
|
Reg = asmjit::x86::ecx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EDX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EDX))
|
|
{
|
|
Reg = asmjit::x86::edx;
|
|
}
|
|
|
|
if (!Reg.isValid())
|
|
{
|
|
Reg = Free8BitX86Reg();
|
|
if (!Reg.isValid())
|
|
{
|
|
WriteTrace(TraceRegisterCache, TraceError, "Failed to find a free 8-bit register");
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
}
|
|
}
|
|
else if (GetX86Mapped(GetIndexFromX86Reg(Reg)) == GPR_Mapped)
|
|
{
|
|
if (GetX86Protected(GetIndexFromX86Reg(Reg)))
|
|
{
|
|
WriteTrace(TraceRegisterCache, TraceError, "Register is protected");
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return x86Reg_Unknown;
|
|
}
|
|
|
|
SetX86Protected(GetIndexFromX86Reg(Reg), true);
|
|
asmjit::x86::Gp NewReg = FreeX86Reg();
|
|
for (uint32_t i = 1; i < 32; i++)
|
|
{
|
|
if (!IsMapped(i))
|
|
{
|
|
continue;
|
|
}
|
|
if (GetMipsRegMapLo(i) == Reg)
|
|
{
|
|
if (!NewReg.isValid())
|
|
{
|
|
UnMap_GPR(i, true);
|
|
break;
|
|
}
|
|
m_CodeBlock.Log(" regcache: change allocation of %s from %s to %s", CRegName::GPR[i], CX86Ops::x86_Name(Reg), CX86Ops::x86_Name(NewReg));
|
|
SetX86Mapped(GetIndexFromX86Reg(NewReg), GPR_Mapped);
|
|
SetX86MapOrder(GetIndexFromX86Reg(NewReg), GetX86MapOrder(GetIndexFromX86Reg(Reg)));
|
|
SetMipsRegMapLo(i, NewReg);
|
|
m_Assembler.mov(NewReg, Reg);
|
|
if (MipsReg == (int32_t)i && !LoadHiWord)
|
|
{
|
|
MipsReg = -1;
|
|
}
|
|
break;
|
|
}
|
|
if (Is64Bit(i) && GetMipsRegMapHi(i) == Reg)
|
|
{
|
|
if (!NewReg.isValid())
|
|
{
|
|
UnMap_GPR(i, true);
|
|
break;
|
|
}
|
|
m_CodeBlock.Log(" regcache: change allocation of %s from %s to %s", CRegName::GPR_Hi[i], CX86Ops::x86_Name(Reg), CX86Ops::x86_Name(NewReg));
|
|
SetX86Mapped(GetIndexFromX86Reg(NewReg), GPR_Mapped);
|
|
SetX86MapOrder(GetIndexFromX86Reg(NewReg), GetX86MapOrder(GetIndexFromX86Reg(Reg)));
|
|
SetMipsRegMapHi(i, NewReg);
|
|
m_Assembler.mov(NewReg, Reg);
|
|
if (MipsReg == (int32_t)i && LoadHiWord)
|
|
{
|
|
MipsReg = -1;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if (GetX86Mapped(GetIndexFromX86Reg(Reg)) == Stack_Mapped)
|
|
{
|
|
UnMap_X86reg(Reg);
|
|
}
|
|
m_CodeBlock.Log(" regcache: allocate %s as temp storage", CX86Ops::x86_Name(Reg));
|
|
|
|
if (MipsReg >= 0)
|
|
{
|
|
if (LoadHiWord)
|
|
{
|
|
if (IsUnknown(MipsReg))
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[MipsReg].UW[1], CRegName::GPR_Hi[MipsReg]);
|
|
}
|
|
else if (IsMapped(MipsReg))
|
|
{
|
|
if (Is64Bit(MipsReg))
|
|
{
|
|
m_Assembler.mov(Reg, GetMipsRegMapHi(MipsReg));
|
|
}
|
|
else if (IsSigned(MipsReg))
|
|
{
|
|
m_Assembler.mov(Reg, GetMipsRegMapLo(MipsReg));
|
|
m_Assembler.sar(Reg, 31);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (Is64Bit(MipsReg))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, GetMipsRegHi(MipsReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, GetMipsRegLo_S(MipsReg) >> 31);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (IsUnknown(MipsReg))
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[MipsReg].UW[0], CRegName::GPR_Lo[MipsReg]);
|
|
}
|
|
else if (IsMapped(MipsReg))
|
|
{
|
|
m_Assembler.mov(Reg, GetMipsRegMapLo(MipsReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, GetMipsRegLo(MipsReg));
|
|
}
|
|
}
|
|
}
|
|
x86RegIndex RegIndex = GetIndexFromX86Reg(Reg);
|
|
SetX86Mapped(RegIndex, Temp_Mapped);
|
|
SetX86Protected(RegIndex, true);
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
int32_t MapOrder = GetX86MapOrder((x86RegIndex)i);
|
|
if (MapOrder > 0)
|
|
{
|
|
SetX86MapOrder((x86RegIndex)i, MapOrder + 1);
|
|
}
|
|
}
|
|
SetX86MapOrder(RegIndex, 1);
|
|
return Reg;
|
|
}
|
|
|
|
void CX86RegInfo::ProtectGPR(uint32_t MipsReg)
|
|
{
|
|
if (IsUnknown(MipsReg) || IsConst(MipsReg))
|
|
{
|
|
return;
|
|
}
|
|
if (Is64Bit(MipsReg))
|
|
{
|
|
SetX86Protected(GetIndexFromX86Reg(GetMipsRegMapHi(MipsReg)), true);
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(GetMipsRegMapLo(MipsReg)), true);
|
|
}
|
|
|
|
void CX86RegInfo::UnProtectGPR(uint32_t Reg)
|
|
{
|
|
if (IsUnknown(Reg) || IsConst(Reg))
|
|
{
|
|
return;
|
|
}
|
|
if (Is64Bit(Reg))
|
|
{
|
|
SetX86Protected(GetIndexFromX86Reg(GetMipsRegMapHi(Reg)), false);
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(GetMipsRegMapLo(Reg)), false);
|
|
}
|
|
|
|
void CX86RegInfo::ResetX86Protection()
|
|
{
|
|
for (int32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
SetX86Protected((x86RegIndex)i, false);
|
|
}
|
|
}
|
|
|
|
void CX86RegInfo::PrepareFPTopToBe(int32_t Reg, int32_t RegToLoad, FPU_STATE Format)
|
|
{
|
|
if (RegToLoad < 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
if (Reg < 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
|
|
if (Format == FPU_Double || Format == FPU_Qword)
|
|
{
|
|
UnMap_FPR(Reg + 1, true);
|
|
UnMap_FPR(RegToLoad + 1, true);
|
|
}
|
|
else
|
|
{
|
|
if ((Reg & 1) != 0)
|
|
{
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != (Reg - 1))
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] == FPU_Double || m_x86fpu_State[i] == FPU_Qword)
|
|
{
|
|
UnMap_FPR(Reg, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if ((RegToLoad & 1) != 0)
|
|
{
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != (RegToLoad - 1))
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] == FPU_Double || m_x86fpu_State[i] == FPU_Qword)
|
|
{
|
|
UnMap_FPR(RegToLoad, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (Reg == RegToLoad)
|
|
{
|
|
// If different format then unmap original register from stack
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
if (m_x86fpu_State[i] != Format)
|
|
{
|
|
UnMap_FPR(Reg, true);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If different format then unmap original register from stack
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
UnMap_FPR(Reg, m_x86fpu_State[i] != Format);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (RegInStack(RegToLoad, Format))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
UnMap_FPR(m_x86fpu_MappedTo[(StackTopPos() - 1) & 7], true);
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] == RegToLoad)
|
|
{
|
|
UnMap_FPR(RegToLoad, true);
|
|
i = 8;
|
|
}
|
|
}
|
|
m_CodeBlock.Log(" regcache: setup ST(0) to be %s", CRegName::FPR[Reg]);
|
|
asmjit::x86::Gp TempReg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
switch (Format)
|
|
{
|
|
case FPU_Dword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[RegToLoad], stdstr_f("m_FPR_S[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadIntegerDwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Qword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[RegToLoad], stdstr_f("m_FPR_D[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadIntegerQwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Float:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_S[RegToLoad], stdstr_f("m_FPR_S[%d]", RegToLoad).c_str());
|
|
//CompileCheckFPUInput32(TempReg);
|
|
m_Assembler.fpuLoadDwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
case FPU_Double:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &g_Reg->m_FPR_D[RegToLoad], stdstr_f("m_FPR_D[%d]", RegToLoad).c_str());
|
|
m_Assembler.fpuLoadQwordFromX86Reg(StackTopPos(), TempReg);
|
|
break;
|
|
default:
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("Load_FPR_ToTop\nUnkown format to load %d", Format).c_str());
|
|
}
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
FpuRoundingModel(StackTopPos()) = RoundDefault;
|
|
m_x86fpu_MappedTo[StackTopPos()] = -1;
|
|
m_x86fpu_State[StackTopPos()] = Format;
|
|
m_x86fpu_StateChanged[StackTopPos()] = false;
|
|
}
|
|
}
|
|
|
|
void CX86RegInfo::SetFPTopAs(int32_t Reg)
|
|
{
|
|
if (m_x86fpu_State[StackTopPos()] == FPU_Unknown || m_x86fpu_MappedTo[StackTopPos()] != -1)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
m_x86fpu_MappedTo[StackTopPos()] = Reg;
|
|
}
|
|
|
|
bool CX86RegInfo::RegInStack(int32_t Reg, FPU_STATE Format)
|
|
{
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] == Reg)
|
|
{
|
|
if (m_x86fpu_State[i] == Format || Format == FPU_Any)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CX86RegInfo::UnMap_AllFPRs()
|
|
{
|
|
for (;;)
|
|
{
|
|
int32_t StackPos = StackTopPos();
|
|
if (m_x86fpu_MappedTo[StackPos] != -1)
|
|
{
|
|
UnMap_FPR(m_x86fpu_MappedTo[StackPos], true);
|
|
continue;
|
|
}
|
|
// See if any more registers mapped
|
|
int32_t StartPos = StackTopPos();
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[(StartPos + i) & 7] != -1)
|
|
{
|
|
m_Assembler.fpuIncStack(StackTopPos());
|
|
}
|
|
}
|
|
if (StackPos != StackTopPos())
|
|
{
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CX86RegInfo::UnMap_FPR(int32_t Reg, bool WriteBackValue)
|
|
{
|
|
if (Reg < 0)
|
|
{
|
|
return;
|
|
}
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (m_x86fpu_MappedTo[i] != Reg)
|
|
{
|
|
continue;
|
|
}
|
|
m_CodeBlock.Log(" regcache: unallocate %s from ST(%d)", CRegName::FPR[Reg], (i - StackTopPos() + 8) & 7);
|
|
if (WriteBackValue)
|
|
{
|
|
int32_t RegPos;
|
|
|
|
if (((i - StackTopPos() + 8) & 7) != 0)
|
|
{
|
|
if (m_x86fpu_MappedTo[StackTopPos()] == -1 && m_x86fpu_MappedTo[(StackTopPos() + 1) & 7] == Reg)
|
|
{
|
|
m_Assembler.fpuIncStack(StackTopPos());
|
|
}
|
|
else
|
|
{
|
|
CX86RegInfo::FPU_ROUND RoundingModel = FpuRoundingModel(StackTopPos());
|
|
FPU_STATE RegState = m_x86fpu_State[StackTopPos()];
|
|
bool Changed = m_x86fpu_StateChanged[StackTopPos()];
|
|
uint32_t MappedTo = m_x86fpu_MappedTo[StackTopPos()];
|
|
FpuRoundingModel(StackTopPos()) = FpuRoundingModel(i);
|
|
m_x86fpu_MappedTo[StackTopPos()] = m_x86fpu_MappedTo[i];
|
|
m_x86fpu_State[StackTopPos()] = m_x86fpu_State[i];
|
|
m_x86fpu_StateChanged[StackTopPos()] = m_x86fpu_StateChanged[i];
|
|
FpuRoundingModel(i) = RoundingModel;
|
|
m_x86fpu_MappedTo[i] = MappedTo;
|
|
m_x86fpu_State[i] = RegState;
|
|
m_x86fpu_StateChanged[i] = Changed;
|
|
m_Assembler.fxch(GetX86FpuRegFromIndex((x86RegFpuIndex)((i - StackTopPos()) & 7)));
|
|
}
|
|
}
|
|
|
|
FixRoundModel(FpuRoundingModel(i));
|
|
|
|
RegPos = StackTopPos();
|
|
asmjit::x86::Gp TempReg = Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
switch (m_x86fpu_State[StackTopPos()])
|
|
{
|
|
case FPU_Dword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_Reg.m_FPR_S[m_x86fpu_MappedTo[StackTopPos()]], stdstr_f("_FPR_S[%d]", m_x86fpu_MappedTo[StackTopPos()]).c_str());
|
|
m_Assembler.fpuStoreIntegerDwordFromX86Reg(StackTopPos(), TempReg, true);
|
|
break;
|
|
case FPU_Qword:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_Reg.m_FPR_D[m_x86fpu_MappedTo[StackTopPos()]], stdstr_f("_FPR_D[%d]", m_x86fpu_MappedTo[StackTopPos()]).c_str());
|
|
m_Assembler.fpuStoreIntegerQwordFromX86Reg(StackTopPos(), TempReg, true);
|
|
break;
|
|
case FPU_Float:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_Reg.m_FPR_S[m_x86fpu_MappedTo[StackTopPos()]], stdstr_f("_FPR_S[%d]", m_x86fpu_MappedTo[StackTopPos()]).c_str());
|
|
m_Assembler.fpuStoreDwordFromX86Reg(StackTopPos(), TempReg, true);
|
|
break;
|
|
case FPU_Double:
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_Reg.m_FPR_D[m_x86fpu_MappedTo[StackTopPos()]], stdstr_f("_FPR_D[%d]", m_x86fpu_MappedTo[StackTopPos()]).c_str());
|
|
m_Assembler.fpuStoreQwordFromX86Reg(StackTopPos(), TempReg, true);
|
|
break;
|
|
default:
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("%s\nUnknown format to load %d", __FUNCTION__, m_x86fpu_State[StackTopPos()]).c_str());
|
|
}
|
|
}
|
|
SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
FpuRoundingModel(RegPos) = RoundDefault;
|
|
m_x86fpu_MappedTo[RegPos] = -1;
|
|
m_x86fpu_State[RegPos] = FPU_Unknown;
|
|
m_x86fpu_StateChanged[RegPos] = false;
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.ffree(GetX86FpuRegFromIndex((x86RegFpuIndex)((i - StackTopPos()) & 7)));
|
|
FpuRoundingModel(i) = RoundDefault;
|
|
m_x86fpu_MappedTo[i] = -1;
|
|
m_x86fpu_State[i] = FPU_Unknown;
|
|
m_x86fpu_StateChanged[i] = false;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
void CX86RegInfo::UnMap_FPStatusReg()
|
|
{
|
|
for (int32_t i = 0, n = x86RegIndex_Size; i < n; i++)
|
|
{
|
|
if (GetX86Mapped((x86RegIndex)i) != CX86RegInfo::FPStatusReg_Mapped)
|
|
{
|
|
continue;
|
|
}
|
|
UnMap_X86reg(GetX86RegFromIndex((x86RegIndex)i));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CX86RegInfo::UnMap_GPR(uint32_t Reg, bool WriteBackValue)
|
|
{
|
|
if (Reg == 0)
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("%s\n\nWhy are you trying to unmap register 0?", __FUNCTION__).c_str());
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (IsUnknown(Reg))
|
|
{
|
|
return;
|
|
}
|
|
//m_CodeBlock.Log("UnMap_GPR: State: %X\tReg: %s\tWriteBack: %s",State,CRegName::GPR[Reg],WriteBackValue?"true":"false");
|
|
if (IsConst(Reg))
|
|
{
|
|
if (!WriteBackValue)
|
|
{
|
|
SetMipsRegState(Reg, STATE_UNKNOWN);
|
|
return;
|
|
}
|
|
if (Is64Bit(Reg))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], GetMipsRegHi(Reg));
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[0], CRegName::GPR_Lo[Reg], GetMipsRegLo(Reg));
|
|
SetMipsRegState(Reg, STATE_UNKNOWN);
|
|
return;
|
|
}
|
|
if ((GetMipsRegLo(Reg) & 0x80000000) != 0)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], 0xFFFFFFFF);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], 0);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[0], CRegName::GPR_Lo[Reg], GetMipsRegLo(Reg));
|
|
SetMipsRegState(Reg, STATE_UNKNOWN);
|
|
return;
|
|
}
|
|
if (Is64Bit(Reg))
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from %s", CX86Ops::x86_Name(GetMipsRegMapHi(Reg)), CRegName::GPR_Hi[Reg]);
|
|
SetX86Mapped(GetIndexFromX86Reg(GetMipsRegMapHi(Reg)), NotMapped);
|
|
SetX86Protected(GetIndexFromX86Reg(GetMipsRegMapHi(Reg)), false);
|
|
}
|
|
m_CodeBlock.Log(" regcache: unallocate %s from %s", CX86Ops::x86_Name(GetMipsRegMapLo(Reg)), CRegName::GPR_Lo[Reg]);
|
|
x86RegIndex RegIndex = GetIndexFromX86Reg(GetMipsRegMapLo(Reg));
|
|
SetX86Mapped(RegIndex, NotMapped);
|
|
SetX86Protected(RegIndex, false);
|
|
if (!WriteBackValue)
|
|
{
|
|
SetMipsRegState(Reg, STATE_UNKNOWN);
|
|
return;
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[Reg].UW[0], CRegName::GPR_Lo[Reg], GetMipsRegMapLo(Reg));
|
|
if (Is64Bit(Reg))
|
|
{
|
|
SetMipsRegMapLo(Reg, x86Reg_Unknown);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], GetMipsRegMapHi(Reg));
|
|
SetMipsRegMapHi(Reg, x86Reg_Unknown);
|
|
}
|
|
else
|
|
{
|
|
if (!g_System->b32BitCore())
|
|
{
|
|
if (IsSigned(Reg))
|
|
{
|
|
m_Assembler.sar(GetMipsRegMapLo(Reg), 31);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], GetMipsRegMapLo(Reg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[Reg].UW[1], CRegName::GPR_Hi[Reg], 0);
|
|
}
|
|
}
|
|
SetMipsRegMapLo(Reg, x86Reg_Unknown);
|
|
}
|
|
SetMipsRegState(Reg, STATE_UNKNOWN);
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RegInfo::UnMap_TempReg()
|
|
{
|
|
asmjit::x86::Gp Reg;
|
|
|
|
if (GetX86Mapped(x86RegIndex_EAX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EAX))
|
|
{
|
|
Reg = asmjit::x86::eax;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EBX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EBX))
|
|
{
|
|
Reg = asmjit::x86::ebx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ECX) == Temp_Mapped && !GetX86Protected(x86RegIndex_ECX))
|
|
{
|
|
Reg = asmjit::x86::ecx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EDX) == Temp_Mapped && !GetX86Protected(x86RegIndex_EDX))
|
|
{
|
|
Reg = asmjit::x86::edx;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ESI) == Temp_Mapped && !GetX86Protected(x86RegIndex_ESI))
|
|
{
|
|
Reg = asmjit::x86::esi;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EDI) == Temp_Mapped && !GetX86Protected(x86RegIndex_EDI))
|
|
{
|
|
Reg = asmjit::x86::edi;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_EBP) == Temp_Mapped && !GetX86Protected(x86RegIndex_EBP))
|
|
{
|
|
Reg = asmjit::x86::ebp;
|
|
}
|
|
else if (GetX86Mapped(x86RegIndex_ESP) == Temp_Mapped && !GetX86Protected(x86RegIndex_ESP))
|
|
{
|
|
Reg = asmjit::x86::esp;
|
|
}
|
|
|
|
if (Reg.isValid())
|
|
{
|
|
if (GetX86Mapped(GetIndexFromX86Reg(Reg)) == Temp_Mapped)
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from temp storage", CX86Ops::x86_Name(Reg));
|
|
}
|
|
SetX86Mapped(GetIndexFromX86Reg(Reg), NotMapped);
|
|
}
|
|
return Reg;
|
|
}
|
|
|
|
bool CX86RegInfo::UnMap_X86reg(const asmjit::x86::Gp & Reg)
|
|
{
|
|
x86RegIndex RegIndex = GetIndexFromX86Reg(Reg);
|
|
if (GetX86Mapped(RegIndex) == NotMapped)
|
|
{
|
|
if (!GetX86Protected(RegIndex))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
else if (GetX86Mapped(RegIndex) == CX86RegInfo::GPR_Mapped)
|
|
{
|
|
for (int i = 1; i < 32; i++)
|
|
{
|
|
if (!IsMapped(i))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (Is64Bit(i) && GetMipsRegMapHi(i) == Reg)
|
|
{
|
|
if (!GetX86Protected(RegIndex))
|
|
{
|
|
UnMap_GPR(i, true);
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
if (GetMipsRegMapLo(i) == Reg)
|
|
{
|
|
if (!GetX86Protected(RegIndex))
|
|
{
|
|
UnMap_GPR(i, true);
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if (GetX86Mapped(RegIndex) == CX86RegInfo::Temp_Mapped)
|
|
{
|
|
if (!GetX86Protected(RegIndex))
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from temp storage", CX86Ops::x86_Name(Reg));
|
|
SetX86Mapped(RegIndex, NotMapped);
|
|
return true;
|
|
}
|
|
}
|
|
else if (GetX86Mapped(RegIndex) == CX86RegInfo::Stack_Mapped)
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from memory stack", CX86Ops::x86_Name(Reg));
|
|
m_Assembler.MoveX86regToVariable(&(g_Recompiler->MemoryStackPos()), "MemoryStack", Reg);
|
|
SetX86Mapped(RegIndex, NotMapped);
|
|
return true;
|
|
}
|
|
else if (GetX86Mapped(RegIndex) == CX86RegInfo::FPStatusReg_Mapped)
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from FP Status Reg", CX86Ops::x86_Name(Reg));
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_FPCR[31], "FPCR[31]", Reg);
|
|
SetX86Mapped(RegIndex, NotMapped);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CX86RegInfo::WriteBackRegisters()
|
|
{
|
|
UnMap_AllFPRs();
|
|
|
|
int32_t count;
|
|
bool bEdiZero = false;
|
|
bool bEsiSign = false;
|
|
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
SetX86Protected((x86RegIndex)i, false);
|
|
}
|
|
for (uint32_t i = 0; i < x86RegIndex_Size; i++)
|
|
{
|
|
UnMap_X86reg(GetX86RegFromIndex((x86RegIndex)i));
|
|
}
|
|
|
|
for (count = 1; count < 32; count++)
|
|
{
|
|
switch (GetMipsRegState(count))
|
|
{
|
|
case CX86RegInfo::STATE_UNKNOWN: break;
|
|
case CX86RegInfo::STATE_CONST_32_SIGN:
|
|
if (!g_System->b32BitCore())
|
|
{
|
|
if (!bEdiZero && (!GetMipsRegLo(count) || !(GetMipsRegLo(count) & 0x80000000)))
|
|
{
|
|
m_Assembler.xor_(asmjit::x86::edi, asmjit::x86::edi);
|
|
bEdiZero = true;
|
|
}
|
|
if (!bEsiSign && (GetMipsRegLo(count) & 0x80000000))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::esi, 0xFFFFFFFF);
|
|
bEsiSign = true;
|
|
}
|
|
if ((GetMipsRegLo(count) & 0x80000000) != 0)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], asmjit::x86::esi);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], asmjit::x86::edi);
|
|
}
|
|
}
|
|
|
|
if (GetMipsRegLo(count) == 0)
|
|
{
|
|
if (g_System->b32BitCore())
|
|
{
|
|
if (!bEdiZero)
|
|
{
|
|
m_Assembler.xor_(asmjit::x86::edi, asmjit::x86::edi);
|
|
bEdiZero = true;
|
|
}
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], asmjit::x86::edi);
|
|
}
|
|
else if (GetMipsRegLo(count) == 0xFFFFFFFF)
|
|
{
|
|
if (g_System->b32BitCore())
|
|
{
|
|
if (!bEsiSign)
|
|
{
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::esi, 0xFFFFFFFF);
|
|
bEsiSign = true;
|
|
}
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], asmjit::x86::esi);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], GetMipsRegLo(count));
|
|
}
|
|
|
|
SetMipsRegState(count, CX86RegInfo::STATE_UNKNOWN);
|
|
break;
|
|
case CX86RegInfo::STATE_CONST_32_ZERO:
|
|
if (!g_System->b32BitCore())
|
|
{
|
|
if (!bEdiZero)
|
|
{
|
|
m_Assembler.xor_(asmjit::x86::edi, asmjit::x86::edi);
|
|
bEdiZero = true;
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], asmjit::x86::edi);
|
|
}
|
|
|
|
if (GetMipsRegLo(count) == 0)
|
|
{
|
|
if (g_System->b32BitCore())
|
|
{
|
|
if (!bEdiZero)
|
|
{
|
|
m_Assembler.xor_(asmjit::x86::edi, asmjit::x86::edi);
|
|
bEdiZero = true;
|
|
}
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], asmjit::x86::edi);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], GetMipsRegLo(count));
|
|
}
|
|
SetMipsRegState(count, CX86RegInfo::STATE_UNKNOWN);
|
|
break;
|
|
case CX86RegInfo::STATE_CONST_64:
|
|
if (GetMipsRegLo(count) == 0 || GetMipsRegHi(count) == 0)
|
|
{
|
|
m_Assembler.xor_(asmjit::x86::edi, asmjit::x86::edi);
|
|
bEdiZero = true;
|
|
}
|
|
if (GetMipsRegLo(count) == 0xFFFFFFFF || GetMipsRegHi(count) == 0xFFFFFFFF)
|
|
{
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::esi, 0xFFFFFFFF);
|
|
bEsiSign = true;
|
|
}
|
|
|
|
if (GetMipsRegHi(count) == 0)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], asmjit::x86::edi);
|
|
}
|
|
else if (GetMipsRegLo(count) == 0xFFFFFFFF)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], asmjit::x86::esi);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[count].UW[1], CRegName::GPR_Hi[count], GetMipsRegHi(count));
|
|
}
|
|
|
|
if (GetMipsRegLo(count) == 0)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], asmjit::x86::edi);
|
|
}
|
|
else if (GetMipsRegLo(count) == 0xFFFFFFFF)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], asmjit::x86::esi);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[count].UW[0], CRegName::GPR_Lo[count], GetMipsRegLo(count));
|
|
}
|
|
SetMipsRegState(count, CX86RegInfo::STATE_UNKNOWN);
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log("%s: Unknown State: %d reg %d (%s)", __FUNCTION__, GetMipsRegState(count), count, CRegName::GPR[count]);
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif |