11584 lines
484 KiB
C++
11584 lines
484 KiB
C++
#include "stdafx.h"
|
|
|
|
#if defined(__i386__) || defined(_M_IX86)
|
|
|
|
#include <Project64-core/Debugger.h>
|
|
#include <Project64-core/N64System/Interpreter/InterpreterOps.h>
|
|
#include <Project64-core/N64System/Mips/Disk.h>
|
|
#include <Project64-core/N64System/Mips/MemoryVirtualMem.h>
|
|
#include <Project64-core/N64System/Mips/R4300iInstruction.h>
|
|
#include <Project64-core/N64System/N64Rom.h>
|
|
#include <Project64-core/N64System/N64System.h>
|
|
#include <Project64-core/N64System/Recompiler/CodeSection.h>
|
|
#include <Project64-core/N64System/Recompiler/LoopAnalysis.h>
|
|
#include <Project64-core/N64System/Recompiler/Recompiler.h>
|
|
#include <Project64-core/N64System/Recompiler/SectionInfo.h>
|
|
#include <Project64-core/N64System/Recompiler/x86/x86RecompilerOps.h>
|
|
#include <Project64-core/N64System/SystemGlobals.h>
|
|
#include <fenv.h>
|
|
#include <stdio.h>
|
|
|
|
uint32_t CX86RecompilerOps::m_RoundingModeValue = 0;
|
|
bool CX86RecompilerOps::m_TempMemoryUsed = false;
|
|
uint32_t CX86RecompilerOps::m_TempValue32 = 0;
|
|
uint64_t CX86RecompilerOps::m_TempValue64 = 0;
|
|
uint32_t CX86RecompilerOps::m_BranchCompare = 0;
|
|
|
|
/*int TestValue = 0;
|
|
void TestFunc()
|
|
{
|
|
TestValue += 1;
|
|
if (TestValue >= 4)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}*/
|
|
|
|
void CX86RecompilerOps::x86CompilerBreakPoint()
|
|
{
|
|
g_Settings->SaveBool(Debugger_SteppingOps, true);
|
|
do
|
|
{
|
|
g_Debugger->WaitForStep();
|
|
if (CDebugSettings::SkipOp())
|
|
{
|
|
// Skip command if instructed by the debugger
|
|
g_Settings->SaveBool(Debugger_SkipOp, false);
|
|
g_Reg->m_PROGRAM_COUNTER += 4;
|
|
|
|
uint32_t OpcodeValue;
|
|
if (!g_MMU->MemoryValue32((uint32_t)g_Reg->m_PROGRAM_COUNTER, OpcodeValue))
|
|
{
|
|
g_Reg->TriggerAddressException(g_Reg->m_PROGRAM_COUNTER, EXC_RMISS);
|
|
g_Reg->m_PROGRAM_COUNTER = g_System->JumpToLocation();
|
|
g_System->m_PipelineStage = PIPELINE_STAGE_NORMAL;
|
|
continue;
|
|
}
|
|
continue;
|
|
}
|
|
g_System->m_OpCodes.ExecuteOps(g_System->CountPerOp());
|
|
if (g_SyncSystem)
|
|
{
|
|
g_System->UpdateSyncCPU(g_System->CountPerOp());
|
|
g_System->SyncSystem();
|
|
}
|
|
|
|
} while (CDebugSettings::isStepping());
|
|
|
|
if (g_System->PipelineStage() != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
g_System->m_OpCodes.ExecuteOps(g_System->CountPerOp());
|
|
if (g_SyncSystem)
|
|
{
|
|
g_System->UpdateSyncCPU(g_System->CountPerOp());
|
|
g_System->SyncSystem();
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86BreakPointDelaySlot()
|
|
{
|
|
g_System->m_OpCodes.ExecuteOps(g_System->CountPerOp());
|
|
if (g_SyncSystem)
|
|
{
|
|
g_System->UpdateSyncCPU(g_System->CountPerOp());
|
|
g_System->SyncSystem();
|
|
}
|
|
if (g_Debugger->ExecutionBP((uint32_t)g_Reg->m_PROGRAM_COUNTER))
|
|
{
|
|
x86CompilerBreakPoint();
|
|
}
|
|
if (g_System->PipelineStage() != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
g_System->m_OpCodes.ExecuteOps(g_System->CountPerOp());
|
|
if (g_SyncSystem)
|
|
{
|
|
g_System->UpdateSyncCPU(g_System->CountPerOp());
|
|
g_System->SyncSystem();
|
|
}
|
|
}
|
|
}
|
|
|
|
static uint32_t memory_access_address;
|
|
static uint32_t memory_write_in_delayslot;
|
|
static uint32_t memory_breakpoint_found = 0;
|
|
|
|
void CX86RecompilerOps::x86MemoryBreakPoint()
|
|
{
|
|
memory_breakpoint_found = 1;
|
|
if (memory_write_in_delayslot)
|
|
{
|
|
g_Reg->m_PROGRAM_COUNTER -= 4;
|
|
*g_NextTimer += g_System->CountPerOp();
|
|
g_System->m_OpCodes.ExecuteOps(g_System->CountPerOp());
|
|
}
|
|
x86CompilerBreakPoint();
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestReadBreakPoint8()
|
|
{
|
|
if (g_Debugger->ReadBP8(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestReadBreakPoint16()
|
|
{
|
|
if (g_Debugger->ReadBP16(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestReadBreakPoint32()
|
|
{
|
|
if (g_Debugger->ReadBP32(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestReadBreakPoint64()
|
|
{
|
|
if (g_Debugger->ReadBP64(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestWriteBreakPoint8()
|
|
{
|
|
if (g_Debugger->WriteBP8(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestWriteBreakPoint16()
|
|
{
|
|
if (g_Debugger->WriteBP16(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestWriteBreakPoint32()
|
|
{
|
|
if (g_Debugger->WriteBP32(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::x86TestWriteBreakPoint64()
|
|
{
|
|
if (g_Debugger->WriteBP64(memory_access_address))
|
|
{
|
|
x86MemoryBreakPoint();
|
|
}
|
|
}
|
|
|
|
CX86RecompilerOps::CX86RecompilerOps(CN64System & m_System, CCodeBlock & CodeBlock) :
|
|
CRecompilerOpsBase(m_System, CodeBlock),
|
|
m_Assembler(CodeBlock),
|
|
m_PipelineStage(PIPELINE_STAGE_NORMAL),
|
|
m_RegWorkingSet(CodeBlock, m_Assembler),
|
|
m_CompilePC(m_Instruction.Address()),
|
|
m_RegBeforeDelay(CodeBlock, m_Assembler),
|
|
m_EffectDelaySlot(false)
|
|
{
|
|
}
|
|
|
|
CX86RecompilerOps::~CX86RecompilerOps()
|
|
{
|
|
}
|
|
|
|
void CX86RecompilerOps::PreCompileOpcode(void)
|
|
{
|
|
if (m_PipelineStage != PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
m_CodeBlock.Log(" %X %s", (uint32_t)m_CompilePC, m_Instruction.NameAndParam().c_str());
|
|
}
|
|
/*if ((uint32_t)m_CompilePC == 0x803275F4 && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.X86BreakPoint(__FILE__, __LINE__);
|
|
}*/
|
|
|
|
/*if ((uint32_t)m_CompilePC >= 0x80000000 && (uint32_t)m_CompilePC <= 0x80400000 && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem) {
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}*/
|
|
|
|
/*if (((uint32_t)m_CompilePC == 0x8031C0E4 || (uint32_t)m_CompilePC == 0x8031C118 ||
|
|
(uint32_t)m_CompilePC == 0x8031CD88 || (uint32_t)m_CompilePC == 0x8031CE24 ||
|
|
(uint32_t)m_CompilePC == 0x8031CE30 || (uint32_t)m_CompilePC == 0x8031CE40) && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}*/
|
|
|
|
/*if ((uint32_t)m_CompilePC == 0x801C1B88)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(TestFunc), "TestFunc");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}*/
|
|
|
|
/*if (((uint32_t)m_CompilePC == 0x80263900) && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.X86BreakPoint(__FILEW__,__LINE__);
|
|
}*/
|
|
|
|
/*if (((uint32_t)m_CompilePC >= 0x80325D80 && (uint32_t)m_CompilePC <= 0x80325DF0) && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
CX86RegInfo TestRegSet = m_RegWorkingSet;
|
|
m_Assembler.pushad();
|
|
TestRegSet.WriteBackRegisters();
|
|
UpdateCounters(TestRegSet,false,true);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER,"PROGRAM_COUNTER",m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(0);
|
|
m_Assembler.popad();
|
|
}*/
|
|
/*if (((uint32_t)m_CompilePC == 0x80324E14) && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.X86BreakPoint(__FILEW__,__LINE__);
|
|
}*/
|
|
|
|
/*if ((uint32_t)m_CompilePC == 0x80324E18 && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet,false,true);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER,"PROGRAM_COUNTER",m_CompilePC);
|
|
if (g_SyncSystem) {
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}*/
|
|
/*if ((uint32_t)m_CompilePC >= 0x80324E00 && (uint32_t)m_CompilePC <= 0x80324E18 && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet,false,true);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER,"PROGRAM_COUNTER",m_CompilePC);
|
|
if (g_SyncSystem) {
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}*/
|
|
/* if ((uint32_t)m_CompilePC == 0x803245CC && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
//m_RegWorkingSet.UnMap_AllFPRs();
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
//X86HardBreakPoint();
|
|
//m_Assembler.X86BreakPoint(__FILEW__,__LINE__);
|
|
//m_RegWorkingSet.UnMap_AllFPRs();
|
|
}*/
|
|
/*if ((uint32_t)m_CompilePC >= 0x80179DC4 && (uint32_t)m_CompilePC <= 0x80179DF0 && m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.UnMap_AllFPRs();
|
|
}*/
|
|
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
}
|
|
|
|
void CX86RecompilerOps::PostCompileOpcode(void)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
if (!g_System->bRegCaching())
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
}
|
|
if (!g_System->bFPURegCaching())
|
|
{
|
|
m_RegWorkingSet.UnMap_AllFPRs();
|
|
if (m_RegWorkingSet.StackTopPos() != 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
/*if ((uint32_t)m_CompilePC >= 0x800933B4 && (uint32_t)m_CompilePC <= 0x80093414 && (m_PipelineStage == PIPELINE_STAGE_NORMAL || m_PipelineStage == PIPELINE_STAGE_DO_DELAY_SLOT))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC + 4);
|
|
UpdateSyncCPU(m_RegWorkingSet, m_RegWorkingSet.GetBlockCycleCount());
|
|
m_Assembler.SubConstFromVariable(m_RegWorkingSet.GetBlockCycleCount(), g_NextTimer, "g_NextTimer"); // Updates compare flag
|
|
m_RegWorkingSet.SetBlockCycleCount(0);
|
|
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::ecx, (uint32_t)g_BaseSystem);
|
|
m_Assembler.CallFunc(AddressOf(&CN64System::SyncSystemPC), "CN64System::SyncSystemPC");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
}*/
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileReadTLBMiss(uint32_t VirtualAddress, const asmjit::x86::Gp & LookUpReg)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_System.m_TLBLoadAddress, "m_TLBLoadAddress", VirtualAddress);
|
|
m_Assembler.CompConstToX86reg(LookUpReg, (uint32_t)-1);
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_TLBReadMiss, false, &CX86Ops::JeLabel);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileReadTLBMiss(const asmjit::x86::Gp & AddressReg, const asmjit::x86::Gp & LookUpReg)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_System.m_TLBLoadAddress, "m_TLBLoadAddress", AddressReg);
|
|
m_Assembler.CompConstToX86reg(LookUpReg, (uint32_t)-1);
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_TLBReadMiss, false, &CX86Ops::JeLabel);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileWriteTLBMiss(const asmjit::x86::Gp & AddressReg, const asmjit::x86::Gp & LookUpReg)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_System.m_TLBStoreAddress, "m_TLBStoreAddress", AddressReg);
|
|
m_Assembler.CompConstToX86reg(LookUpReg, (uint32_t)-1);
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_TLBWriteMiss, false, &CX86Ops::JeLabel);
|
|
}
|
|
|
|
// Trap functions
|
|
void CX86RecompilerOps::Compile_TrapCompare(RecompilerTrapCompare CompareType)
|
|
{
|
|
uint32_t FunctAddress = 0;
|
|
const char * FunctName = nullptr;
|
|
switch (CompareType)
|
|
{
|
|
case RecompilerTrapCompare_TEQ:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TEQ);
|
|
FunctName = "R4300iOp::SPECIAL_TEQ";
|
|
break;
|
|
case RecompilerTrapCompare_TNE:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TNE);
|
|
FunctName = "R4300iOp::SPECIAL_TNE";
|
|
break;
|
|
case RecompilerTrapCompare_TGE:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TGE);
|
|
FunctName = "R4300iOp::SPECIAL_TGE";
|
|
break;
|
|
case RecompilerTrapCompare_TGEU:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TGEU);
|
|
FunctName = "R4300iOp::SPECIAL_TGEU";
|
|
break;
|
|
case RecompilerTrapCompare_TLT:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TLT);
|
|
FunctName = "R4300iOp::SPECIAL_TLT";
|
|
break;
|
|
case RecompilerTrapCompare_TLTU:
|
|
FunctAddress = AddressOf(&R4300iOp::SPECIAL_TLTU);
|
|
FunctName = "R4300iOp::SPECIAL_TLTU";
|
|
break;
|
|
case RecompilerTrapCompare_TEQI:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TEQI);
|
|
FunctName = "R4300iOp::REGIMM_TEQI";
|
|
break;
|
|
case RecompilerTrapCompare_TNEI:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TNEI);
|
|
FunctName = "R4300iOp::REGIMM_TNEI";
|
|
break;
|
|
case RecompilerTrapCompare_TGEI:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TGEI);
|
|
FunctName = "R4300iOp::REGIMM_TGEI";
|
|
break;
|
|
case RecompilerTrapCompare_TGEIU:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TGEIU);
|
|
FunctName = "R4300iOp::REGIMM_TGEIU";
|
|
break;
|
|
case RecompilerTrapCompare_TLTI:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TLTI);
|
|
FunctName = "R4300iOp::REGIMM_TLTI";
|
|
break;
|
|
case RecompilerTrapCompare_TLTIU:
|
|
FunctAddress = AddressOf(&R4300iOp::REGIMM_TLTIU);
|
|
FunctName = "R4300iOp::REGIMM_TLTIU";
|
|
break;
|
|
default:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
if (FunctName != nullptr && FunctAddress != 0)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rs, true);
|
|
}
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, FunctAddress, FunctName, 4);
|
|
m_Assembler.CallFunc(FunctAddress, FunctName);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
// Branch functions
|
|
void CX86RecompilerOps::Compile_BranchCompare(RecompilerBranchCompare CompareType)
|
|
{
|
|
switch (CompareType)
|
|
{
|
|
case RecompilerBranchCompare_BEQ: BEQ_Compare(); break;
|
|
case RecompilerBranchCompare_BNE: BNE_Compare(); break;
|
|
case RecompilerBranchCompare_BLTZ: BLTZ_Compare(); break;
|
|
case RecompilerBranchCompare_BLEZ: BLEZ_Compare(); break;
|
|
case RecompilerBranchCompare_BGTZ: BGTZ_Compare(); break;
|
|
case RecompilerBranchCompare_BGEZ: BGEZ_Compare(); break;
|
|
case RecompilerBranchCompare_COP1BCF: COP1_BCF_Compare(); break;
|
|
case RecompilerBranchCompare_COP1BCT: COP1_BCT_Compare(); break;
|
|
default:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::Compile_Branch(RecompilerBranchCompare CompareType, bool Link)
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
Compile_BranchCompare(CompareType);
|
|
if (m_Section->m_Jump.FallThrough || (!m_Section->m_Cont.FallThrough && m_Section->m_Jump.LinkLocation.isValid()))
|
|
{
|
|
LinkJump(m_Section->m_Jump);
|
|
m_Assembler.MoveConst64ToVariable(&g_System->m_JumpDelayLocation, "System::m_JumpDelayLocation", m_CompilePC + ((int16_t)m_Opcode.offset << 2) + 8);
|
|
if (m_Section->m_Cont.LinkLocation.isValid())
|
|
{
|
|
// jump to link
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
LinkJump(m_Section->m_Cont);
|
|
m_Assembler.MoveConst64ToVariable(&g_System->m_JumpDelayLocation, "System::m_JumpDelayLocation", m_CompilePC + 8);
|
|
if (m_Section->m_Jump.LinkLocation.isValid())
|
|
{
|
|
// jump to link
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
if (m_Section->m_Jump.LinkLocation.isValid() || m_Section->m_Cont.LinkLocation.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (Link)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(31, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(31), &g_System->m_JumpToLocation, "System::m_JumpToLocation");
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(31), 4);
|
|
}
|
|
OverflowDelaySlot(false);
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
if (CompareType == RecompilerBranchCompare_COP1BCF || CompareType == RecompilerBranchCompare_COP1BCT)
|
|
{
|
|
CompileCop1Test();
|
|
asmjit::x86::Gp StatusReg = m_RegWorkingSet.Map_FPStatusReg();
|
|
m_Assembler.and_(StatusReg, (uint32_t)(~0x0003F000));
|
|
}
|
|
if (m_CompilePC + ((int16_t)m_Opcode.offset << 2) + 4 == m_CompilePC + 8 && (m_CompilePC & 0xFFC) != 0xFFC)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
return;
|
|
}
|
|
|
|
if ((m_CompilePC & 0xFFC) != 0xFFC)
|
|
{
|
|
R4300iOpcode DelaySlot;
|
|
m_EffectDelaySlot = g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value) && R4300iInstruction(m_CompilePC, m_Opcode.Value).DelaySlotEffectsCompare(DelaySlot.Value);
|
|
}
|
|
else
|
|
{
|
|
m_EffectDelaySlot = true;
|
|
}
|
|
m_Section->m_Jump.JumpPC = (uint32_t)m_CompilePC;
|
|
m_Section->m_Jump.TargetPC = (uint32_t)(m_CompilePC + ((int16_t)m_Opcode.offset << 2) + 4);
|
|
if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Section->m_Jump.TargetPC += 4;
|
|
m_EffectDelaySlot = true;
|
|
}
|
|
if (m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Section_%d", m_Section->m_JumpSection->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Exit_%X_jump_%X", m_Section->m_EnterPC, m_Section->m_Jump.TargetPC);
|
|
}
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_Section->m_Jump.DoneDelaySlot = false;
|
|
m_Section->m_Cont.JumpPC = (uint32_t)m_CompilePC;
|
|
m_Section->m_Cont.TargetPC = (uint32_t)(m_CompilePC + 8);
|
|
if (m_Section->m_ContinueSection != nullptr)
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("Section_%d", m_Section->m_ContinueSection->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("Exit_%X_continue_%X", m_Section->m_EnterPC, m_Section->m_Cont.TargetPC);
|
|
}
|
|
m_Section->m_Cont.LinkLocation = asmjit::Label();
|
|
m_Section->m_Cont.LinkLocation2 = asmjit::Label();
|
|
m_Section->m_Cont.DoneDelaySlot = false;
|
|
if (m_Section->m_Jump.TargetPC < m_Section->m_Cont.TargetPC)
|
|
{
|
|
m_Section->m_Cont.FallThrough = false;
|
|
m_Section->m_Jump.FallThrough = true;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.FallThrough = true;
|
|
m_Section->m_Jump.FallThrough = false;
|
|
}
|
|
|
|
if (Link)
|
|
{
|
|
R4300iInstruction Instruction(m_CompilePC, m_Opcode.Value);
|
|
uint32_t ReadReg1, ReadReg2;
|
|
Instruction.ReadsGPR(ReadReg1, ReadReg2);
|
|
|
|
if (ReadReg1 != 31 && ReadReg2 != 31)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(31, false);
|
|
m_RegWorkingSet.SetMipsRegLo(31, (uint32_t)m_CompilePC + 8);
|
|
m_RegWorkingSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkAddress = (uint32_t)(m_CompilePC + 8);
|
|
m_Section->m_Jump.LinkAddress = (uint32_t)(m_CompilePC + 8);
|
|
}
|
|
}
|
|
if (m_EffectDelaySlot)
|
|
{
|
|
if ((m_CompilePC & 0xFFC) != 0xFFC)
|
|
{
|
|
m_Section->m_Cont.BranchLabel = m_Section->m_ContinueSection != nullptr ? "Continue" : stdstr_f("ExitBlock_%X_Continue", m_Section->m_EnterPC);
|
|
m_Section->m_Jump.BranchLabel = m_Section->m_JumpSection != nullptr ? "Jump" : stdstr_f("ExitBlock_%X_Jump", m_Section->m_EnterPC);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.BranchLabel = "Continue";
|
|
m_Section->m_Jump.BranchLabel = "Jump";
|
|
}
|
|
if (m_Section->m_Jump.TargetPC != m_Section->m_Cont.TargetPC)
|
|
{
|
|
Compile_BranchCompare(CompareType);
|
|
}
|
|
if (!m_Section->m_Jump.FallThrough && !m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (m_Section->m_Jump.LinkLocation.isValid())
|
|
{
|
|
LinkJump(m_Section->m_Jump);
|
|
m_Section->m_Jump.FallThrough = true;
|
|
}
|
|
else if (m_Section->m_Cont.LinkLocation.isValid())
|
|
{
|
|
LinkJump(m_Section->m_Cont);
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
asmjit::Label DelayLinkLocation;
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
if (m_Section->m_Jump.LinkLocation.isValid() || m_Section->m_Jump.LinkLocation2.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_Section->m_Jump.TargetPC);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (m_Section->m_Cont.LinkLocation.isValid() || m_Section->m_Cont.LinkLocation2.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_Section->m_Cont.TargetPC);
|
|
}
|
|
|
|
if (m_Section->m_Jump.LinkLocation.isValid() || m_Section->m_Jump.LinkLocation2.isValid())
|
|
{
|
|
if (DelayLinkLocation.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
DelayLinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("DoDelaySlot", DelayLinkLocation);
|
|
|
|
LinkJump(m_Section->m_Jump);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_Section->m_Jump.TargetPC);
|
|
}
|
|
if (m_Section->m_Cont.LinkLocation.isValid() || m_Section->m_Cont.LinkLocation2.isValid())
|
|
{
|
|
if (DelayLinkLocation.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
DelayLinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("DoDelaySlot", DelayLinkLocation);
|
|
|
|
LinkJump(m_Section->m_Cont);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_Section->m_Cont.TargetPC);
|
|
}
|
|
if (DelayLinkLocation.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(DelayLinkLocation);
|
|
}
|
|
OverflowDelaySlot(false);
|
|
return;
|
|
}
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
m_RegBeforeDelay = m_RegWorkingSet;
|
|
}
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
m_Section->m_Cont.RegSet = m_RegWorkingSet;
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
if (m_CompilePC + ((int16_t)m_Opcode.offset << 2) + 4 == m_CompilePC + 8)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_NORMAL;
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
SetCurrentPC((uint32_t)(GetCurrentPC() + 4));
|
|
return;
|
|
}
|
|
if (m_EffectDelaySlot)
|
|
{
|
|
CJumpInfo * FallInfo = m_Section->m_Jump.FallThrough ? &m_Section->m_Jump : &m_Section->m_Cont;
|
|
CJumpInfo * JumpInfo = m_Section->m_Jump.FallThrough ? &m_Section->m_Cont : &m_Section->m_Jump;
|
|
|
|
if (FallInfo->FallThrough && !FallInfo->DoneDelaySlot)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
FallInfo->RegSet = m_RegWorkingSet;
|
|
if (FallInfo == &m_Section->m_Jump)
|
|
{
|
|
if (m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Section_%d", m_Section->m_JumpSection->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.BranchLabel = "ExitBlock";
|
|
}
|
|
if (FallInfo->TargetPC <= (uint32_t)m_CompilePC)
|
|
{
|
|
UpdateCounters(m_Section->m_Jump.RegSet, true, true, true);
|
|
m_CodeBlock.Log("CompileSystemCheck 12");
|
|
CompileSystemCheck(FallInfo->TargetPC, m_Section->m_Jump.RegSet);
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
FallInfo->Reason = ExitReason_NormalNoSysCheck;
|
|
FallInfo->JumpPC = (uint32_t)-1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_Section->m_ContinueSection != nullptr)
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("Section_%d", m_Section->m_ContinueSection->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("ExitBlock_%X_Continue", m_Section->m_EnterPC);
|
|
}
|
|
}
|
|
FallInfo->DoneDelaySlot = true;
|
|
if (!JumpInfo->DoneDelaySlot)
|
|
{
|
|
FallInfo->FallThrough = false;
|
|
FallInfo->LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(FallInfo->BranchLabel.c_str(), FallInfo->LinkLocation);
|
|
|
|
if (JumpInfo->LinkLocation.isValid())
|
|
{
|
|
LinkJump(*JumpInfo);
|
|
JumpInfo->FallThrough = true;
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
m_RegWorkingSet = m_RegBeforeDelay;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_Section->m_Jump.TargetPC != m_Section->m_Cont.TargetPC)
|
|
{
|
|
Compile_BranchCompare(CompareType);
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
m_Section->m_Cont.RegSet = m_RegWorkingSet;
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
if (m_Section->m_Cont.LinkAddress != (uint32_t)-1)
|
|
{
|
|
m_Section->m_Cont.RegSet.UnMap_GPR(31, false);
|
|
m_Section->m_Cont.RegSet.SetMipsRegLo(31, m_Section->m_Cont.LinkAddress);
|
|
m_Section->m_Cont.RegSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_Section->m_Cont.LinkAddress = (uint32_t)-1;
|
|
}
|
|
if (m_Section->m_Jump.LinkAddress != (uint32_t)-1)
|
|
{
|
|
m_Section->m_Jump.RegSet.UnMap_GPR(31, false);
|
|
m_Section->m_Jump.RegSet.SetMipsRegLo(31, m_Section->m_Jump.LinkAddress);
|
|
m_Section->m_Jump.RegSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_Section->m_Jump.LinkAddress = (uint32_t)-1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
m_Section->m_Cont.RegSet = m_RegWorkingSet;
|
|
if (m_Section->m_ContinueSection == nullptr && m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_ContinueSection = m_Section->m_JumpSection;
|
|
m_Section->m_JumpSection = nullptr;
|
|
}
|
|
if (m_Section->m_ContinueSection != nullptr)
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("Section_%d", m_Section->m_ContinueSection->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.BranchLabel = "ExitBlock";
|
|
}
|
|
}
|
|
}
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("WTF\n\nBranch\nNextInstruction = %X", m_PipelineStage).c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::Compile_BranchLikely(RecompilerBranchCompare CompareType, bool Link)
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
if (CompareType == RecompilerBranchCompare_COP1BCF || CompareType == RecompilerBranchCompare_COP1BCT)
|
|
{
|
|
CompileCop1Test();
|
|
asmjit::x86::Gp StatusReg = m_RegWorkingSet.Map_FPStatusReg();
|
|
m_Assembler.and_(StatusReg, (uint32_t)(~0x0003F000));
|
|
}
|
|
if (!g_System->bLinkBlocks() || (m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
m_Section->m_Jump.JumpPC = (uint32_t)m_CompilePC;
|
|
m_Section->m_Jump.TargetPC = (uint32_t)(m_CompilePC + ((int16_t)m_Opcode.offset << 2) + 4);
|
|
m_Section->m_Cont.JumpPC = (uint32_t)m_CompilePC;
|
|
m_Section->m_Cont.TargetPC = (uint32_t)(m_CompilePC + 8);
|
|
}
|
|
else
|
|
{
|
|
if (m_Section->m_Jump.JumpPC != (uint32_t)m_CompilePC)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (m_Section->m_Cont.JumpPC != (uint32_t)m_CompilePC)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (m_Section->m_Cont.TargetPC != (uint32_t)(m_CompilePC + 8))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
if (m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Section_%d", ((CCodeSection *)m_Section->m_JumpSection)->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.BranchLabel = "ExitBlock";
|
|
}
|
|
|
|
if (m_Section->m_ContinueSection != nullptr)
|
|
{
|
|
m_Section->m_Cont.BranchLabel = stdstr_f("Section_%d", ((CCodeSection *)m_Section->m_ContinueSection)->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.BranchLabel = "ExitBlock";
|
|
}
|
|
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_Section->m_Cont.FallThrough = false;
|
|
m_Section->m_Cont.LinkLocation = asmjit::Label();
|
|
m_Section->m_Cont.LinkLocation2 = asmjit::Label();
|
|
if (Link)
|
|
{
|
|
R4300iInstruction Instruction(m_CompilePC, m_Opcode.Value);
|
|
uint32_t ReadReg1, ReadReg2;
|
|
Instruction.ReadsGPR(ReadReg1, ReadReg2);
|
|
|
|
if (ReadReg1 != 31 && ReadReg2 != 31)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(31, false);
|
|
m_RegWorkingSet.SetMipsRegLo(31, (uint32_t)(m_CompilePC + 8));
|
|
m_RegWorkingSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkAddress = (uint32_t)(m_CompilePC + 8);
|
|
m_Section->m_Jump.LinkAddress = (uint32_t)(m_CompilePC + 8);
|
|
}
|
|
}
|
|
|
|
Compile_BranchCompare(CompareType);
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
|
|
m_Section->m_Cont.RegSet = m_RegWorkingSet;
|
|
m_Section->m_Cont.RegSet.SetBlockCycleCount(m_Section->m_Cont.RegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
if (m_Section->m_Cont.LinkAddress != (uint32_t)-1)
|
|
{
|
|
m_Section->m_Cont.RegSet.UnMap_GPR(31, false);
|
|
m_Section->m_Cont.RegSet.SetMipsRegLo(31, m_Section->m_Cont.LinkAddress);
|
|
m_Section->m_Cont.RegSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_Section->m_Cont.LinkAddress = (uint32_t)-1;
|
|
}
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (m_Section->m_Jump.LinkLocation.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
if (m_Section->m_Jump.LinkLocation.isValid() || m_Section->m_Jump.FallThrough)
|
|
{
|
|
LinkJump(m_Section->m_Jump);
|
|
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_Section->m_Jump.TargetPC);
|
|
OverflowDelaySlot(false);
|
|
m_CodeBlock.Log(" ");
|
|
m_CodeBlock.Log(" %s:", m_Section->m_Cont.BranchLabel.c_str());
|
|
}
|
|
else if (!m_Section->m_Cont.FallThrough)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
LinkJump(m_Section->m_Cont);
|
|
CompileExit(m_CompilePC, m_CompilePC + 8, m_Section->m_Cont.RegSet, ExitReason_Normal, true, nullptr);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
|
|
if (g_System->bLinkBlocks())
|
|
{
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
m_Section->m_Jump.RegSet.SetBlockCycleCount(m_Section->m_Jump.RegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
if (m_Section->m_Jump.LinkAddress != (uint32_t)-1)
|
|
{
|
|
m_Section->m_Jump.RegSet.UnMap_GPR(31, false);
|
|
m_Section->m_Jump.RegSet.SetMipsRegLo(31, m_Section->m_Jump.LinkAddress);
|
|
m_Section->m_Jump.RegSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_Section->m_Jump.LinkAddress = (uint32_t)-1;
|
|
}
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else
|
|
{
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (m_Section->m_Jump.LinkLocation.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
}
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
m_Section->m_Jump.RegSet.SetBlockCycleCount(m_Section->m_Jump.RegSet.GetBlockCycleCount());
|
|
if (m_Section->m_Jump.LinkAddress != (uint32_t)-1)
|
|
{
|
|
m_Section->m_Jump.RegSet.UnMap_GPR(31, false);
|
|
m_Section->m_Jump.RegSet.SetMipsRegLo(31, m_Section->m_Jump.LinkAddress);
|
|
m_Section->m_Jump.RegSet.SetMipsRegState(31, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_Section->m_Jump.LinkAddress = (uint32_t)-1;
|
|
}
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("WTF\n%s\nNextInstruction = %X", __FUNCTION__, m_PipelineStage).c_str());
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BNE_Compare()
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsKnown(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs) || m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) != m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt))
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs) || m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
m_Assembler.cmp(
|
|
m_RegWorkingSet.Is32Bit(m_Opcode.rs) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false) : m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs),
|
|
m_RegWorkingSet.Is32Bit(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false) : m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg) || m_RegWorkingSet.Is64Bit(MappedReg))
|
|
{
|
|
if (m_RegWorkingSet.Is32Bit(ConstReg) || m_RegWorkingSet.Is32Bit(MappedReg))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(MappedReg);
|
|
if (m_RegWorkingSet.Is32Bit(MappedReg))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, MappedReg, true, false), m_RegWorkingSet.GetMipsRegHi(ConstReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(MappedReg), m_RegWorkingSet.GetMipsRegLo_S(ConstReg) >> 31);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(MappedReg), m_RegWorkingSet.GetMipsRegHi(ConstReg));
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), m_RegWorkingSet.GetMipsRegLo(ConstReg));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), m_RegWorkingSet.GetMipsRegLo(ConstReg));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rs) || m_RegWorkingSet.IsKnown(m_Opcode.rt))
|
|
{
|
|
uint32_t KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (!b32BitCore())
|
|
{
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], m_RegWorkingSet.GetMipsRegHi(KnownReg));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], (m_RegWorkingSet.GetMipsRegLo_S(KnownReg) >> 31));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapHi(KnownReg), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(KnownReg))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, KnownReg, true, false), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], 0);
|
|
}
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], m_RegWorkingSet.GetMipsRegLo(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
|
|
if (Jump.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg;
|
|
|
|
if (!b32BitCore())
|
|
{
|
|
Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs]);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
|
|
Reg = m_RegWorkingSet.Map_TempReg(Reg, m_Opcode.rt, false, false);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
if (Jump.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BEQ_Compare()
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsKnown(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs) || m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) == m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt))
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
if ((m_RegWorkingSet.Is64Bit(m_Opcode.rs) || m_RegWorkingSet.Is64Bit(m_Opcode.rt)) && !b32BitCore())
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
|
|
m_Assembler.cmp(
|
|
m_RegWorkingSet.Is32Bit(m_Opcode.rs) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false) : m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs),
|
|
m_RegWorkingSet.Is32Bit(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false) : m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg) || m_RegWorkingSet.Is64Bit(MappedReg))
|
|
{
|
|
if (m_RegWorkingSet.Is32Bit(ConstReg) || m_RegWorkingSet.Is32Bit(MappedReg))
|
|
{
|
|
if (m_RegWorkingSet.Is32Bit(MappedReg))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(MappedReg);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, MappedReg, true, false), m_RegWorkingSet.GetMipsRegHi(ConstReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(MappedReg), m_RegWorkingSet.GetMipsRegLo_S(ConstReg) >> 31);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(MappedReg), m_RegWorkingSet.GetMipsRegHi(ConstReg));
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), m_RegWorkingSet.GetMipsRegLo(ConstReg));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), m_RegWorkingSet.GetMipsRegLo(ConstReg));
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rs) || m_RegWorkingSet.IsKnown(m_Opcode.rt))
|
|
{
|
|
uint32_t KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (!b32BitCore())
|
|
{
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], m_RegWorkingSet.GetMipsRegHi(KnownReg));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], m_RegWorkingSet.GetMipsRegLo_S(KnownReg) >> 31);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapHi(KnownReg), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(KnownReg))
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, KnownReg, true, false), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], 0);
|
|
}
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
}
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], m_RegWorkingSet.GetMipsRegLo(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
if (Jump.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg;
|
|
if (!b32BitCore())
|
|
{
|
|
Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("Continue", Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
}
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(Reg, m_Opcode.rs, false, false), &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
if (Jump.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BGTZ_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsReg_S(m_Opcode.rs) > 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) > 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.Is32Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsUnknown(m_Opcode.rs) && b32BitCore())
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel("Continue", Jump);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel("Continue", Jump);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BLEZ_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsReg_S(m_Opcode.rs) <= 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) <= 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) == 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is32Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel("Continue", Jump);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel("Continue", Jump);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("BranchToJump", m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (!b32BitCore())
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel("Continue", Jump);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel("Continue", Jump);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
}
|
|
if (Jump.isValid())
|
|
{
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation2);
|
|
if (Jump.isValid())
|
|
{
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation2);
|
|
m_Section->m_Jump.LinkLocation2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("BranchToJump", m_Section->m_Jump.LinkLocation2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JleLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BLTZ_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsReg_S(m_Opcode.rs) < 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsUnknown(m_Opcode.rs))
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::BGEZ_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) >= 0)
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Cont.FallThrough = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), 0);
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), 0);
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Cont.FallThrough = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], 0);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], 0);
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JlLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_BCF_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.test(m_RegWorkingSet.Map_FPStatusReg(), FPCSR_C);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.TestVariable(&m_Reg.m_FPCR[31], "_FPCR[31]", FPCSR_C);
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_BCT_Compare()
|
|
{
|
|
if (m_RegWorkingSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.test(m_RegWorkingSet.Map_FPStatusReg(), FPCSR_C);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.TestVariable(&m_Reg.m_FPCR[31], "_FPCR[31]", FPCSR_C);
|
|
}
|
|
if (m_Section->m_Cont.FallThrough)
|
|
{
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
else if (m_Section->m_Jump.FallThrough)
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Cont.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel(m_Section->m_Cont.BranchLabel.c_str(), m_Section->m_Cont.LinkLocation);
|
|
m_Section->m_Jump.LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(m_Section->m_Jump.BranchLabel.c_str(), m_Section->m_Jump.LinkLocation);
|
|
}
|
|
}
|
|
|
|
// Opcode functions
|
|
void CX86RecompilerOps::J()
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", (m_CompilePC & 0xF0000000) + (m_Opcode.target << 2));
|
|
OverflowDelaySlot(false);
|
|
return;
|
|
}
|
|
R4300iOpcode DelaySlot;
|
|
g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value);
|
|
if (R4300iInstruction(m_CompilePC + 4, DelaySlot.Value).HasDelaySlot())
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", (m_CompilePC & 0xF0000000) + (m_Opcode.target << 2));
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
return;
|
|
}
|
|
|
|
m_Section->m_Jump.TargetPC = (m_CompilePC & 0xF0000000) + (m_Opcode.target << 2);
|
|
m_Section->m_Jump.JumpPC = (uint32_t)(m_CompilePC);
|
|
if (m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Section_%d", ((CCodeSection *)m_Section->m_JumpSection)->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.BranchLabel = "ExitBlock";
|
|
}
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
m_Section->GenerateSectionLinkage();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("WTF\n\nJ\nNextInstruction = %X", m_PipelineStage).c_str());
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::JAL()
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(31, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(31), &m_Reg.m_PROGRAM_COUNTER, "_PROGRAM_COUNTER");
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(31), 0xF0000000);
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(31), (m_CompilePC + 8) & ~0xF0000000);
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", (m_CompilePC & 0xF0000000) + (m_Opcode.target << 2));
|
|
OverflowDelaySlot(false);
|
|
return;
|
|
}
|
|
m_Section->m_Jump.TargetPC = (m_CompilePC & 0xF0000000) + (m_Opcode.target << 2);
|
|
m_Section->m_Jump.JumpPC = (uint32_t)m_CompilePC;
|
|
if (m_Section->m_JumpSection != nullptr)
|
|
{
|
|
m_Section->m_Jump.BranchLabel = stdstr_f("Section_%d", ((CCodeSection *)m_Section->m_JumpSection)->m_SectionID);
|
|
}
|
|
else
|
|
{
|
|
m_Section->m_Jump.BranchLabel = "ExitBlock";
|
|
}
|
|
m_Section->m_Jump.FallThrough = true;
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
if (m_Section->m_JumpSection)
|
|
{
|
|
m_Section->m_Jump.RegSet = m_RegWorkingSet;
|
|
m_Section->GenerateSectionLinkage();
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
|
|
asmjit::x86::Gp PCReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(PCReg, &m_Reg.m_PROGRAM_COUNTER, "_PROGRAM_COUNTER");
|
|
m_Assembler.and_(PCReg, 0xF0000000);
|
|
m_Assembler.AddConstToX86Reg(PCReg, (m_Opcode.target << 2));
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "_PROGRAM_COUNTER", PCReg);
|
|
|
|
uint64_t TargetPC = (m_CompilePC & 0xFFFFFFFFF0000000) + (m_Opcode.target << 2);
|
|
bool bCheck = TargetPC <= m_CompilePC;
|
|
UpdateCounters(m_RegWorkingSet, bCheck, true);
|
|
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, bCheck ? ExitReason_Normal : ExitReason_NormalNoSysCheck, true, nullptr);
|
|
}
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::ADDI()
|
|
{
|
|
if (g_System->bFastSP() && m_Opcode.rs == 29 && m_Opcode.rt == 29)
|
|
{
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true), (int16_t)m_Opcode.immediate);
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int32_t rs = m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs);
|
|
int32_t imm = (int16_t)m_Opcode.immediate;
|
|
int32_t sum = rs + imm;
|
|
if ((~(rs ^ imm) & (rs ^ sum)) & 0x80000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (m_Opcode.rt != 0)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, sum);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_Assembler.AddConstToX86Reg(Reg, (int16_t)m_Opcode.immediate);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Reg);
|
|
}
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29 && m_Opcode.rs != 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::ADDIU()
|
|
{
|
|
if (m_Opcode.rt == 0 || (m_Opcode.immediate == 0 && m_Opcode.rs == m_Opcode.rt))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (g_System->bFastSP())
|
|
{
|
|
if (m_Opcode.rs == 29 && m_Opcode.rt == 29)
|
|
{
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true), (int16_t)m_Opcode.immediate);
|
|
}
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) + (int16_t)m_Opcode.immediate);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rs);
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), (int16_t)m_Opcode.immediate);
|
|
}
|
|
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29 && m_Opcode.rs != 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SLTIU()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Result = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) < ((unsigned)((int64_t)((int16_t)m_Opcode.immediate))) ? 1 : 0 : m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) < ((unsigned)((int16_t)m_Opcode.immediate)) ? 1
|
|
: 0;
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, Result);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), ((int16_t)m_Opcode.immediate >> 31));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), (int16_t)m_Opcode.immediate);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), (int16_t)m_Opcode.immediate);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], (int16_t)m_Opcode.immediate);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], ((int16_t)m_Opcode.immediate >> 31));
|
|
asmjit::Label Jump = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("CompareSet", Jump);
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], (int16_t)m_Opcode.immediate);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SLTI()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Result = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? ((int64_t)m_RegWorkingSet.GetMipsReg(m_Opcode.rs) < (int64_t)((int16_t)m_Opcode.immediate) ? 1 : 0) : (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < (int16_t)m_Opcode.immediate ? 1 : 0);
|
|
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, Result);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs), ((int16_t)m_Opcode.immediate >> 31));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), (int16_t)m_Opcode.immediate);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), (int16_t)m_Opcode.immediate);
|
|
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::ebx)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), 1);
|
|
}
|
|
}
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], (int16_t)m_Opcode.immediate);
|
|
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::ebx)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), 1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump[2];
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs], ((int16_t)m_Opcode.immediate >> 31));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs], (int16_t)m_Opcode.immediate);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
if (Jump[1].isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::ANDI()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & m_Opcode.immediate);
|
|
}
|
|
else if (m_Opcode.immediate != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, m_Opcode.rs);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_Opcode.immediate);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, 0);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::ORI()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (g_System->bFastSP() && m_Opcode.rs == 29 && m_Opcode.rt == 29)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true), m_Opcode.immediate);
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, m_RegWorkingSet.GetMipsRegState(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegHi(m_Opcode.rt, m_RegWorkingSet.GetMipsRegHi(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) | m_Opcode.immediate);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rs);
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, m_Opcode.rs);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, m_RegWorkingSet.IsSigned(m_Opcode.rs), m_Opcode.rs);
|
|
}
|
|
}
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_Opcode.immediate);
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rs);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, m_Opcode.rs);
|
|
}
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_Opcode.immediate);
|
|
}
|
|
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29 && m_Opcode.rs != 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::XORI()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_Opcode.rs != m_Opcode.rt)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, m_RegWorkingSet.GetMipsRegState(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegHi(m_Opcode.rt, m_RegWorkingSet.GetMipsRegHi(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) ^ m_Opcode.immediate);
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.Is32Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, m_RegWorkingSet.IsSigned(m_Opcode.rs), m_Opcode.rs);
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rs);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, m_Opcode.rs);
|
|
}
|
|
if (m_Opcode.immediate != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_Opcode.immediate);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LUI()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29)
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true, false);
|
|
uint32_t Address;
|
|
|
|
m_MMU.VAddrToPAddr(((int16_t)m_Opcode.offset << 16), Address);
|
|
if (!Reg.isValid())
|
|
{
|
|
m_Assembler.MoveConstToVariable(&(g_Recompiler->MemoryStackPos()), "MemoryStack", (uint32_t)(Address + g_MMU->Rdram()));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, (uint32_t)(Address + g_MMU->Rdram()));
|
|
}
|
|
}
|
|
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rt, ((int16_t)m_Opcode.offset << 16));
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
|
|
void CX86RecompilerOps::DADDI()
|
|
{
|
|
int64_t imm = (int64_t)((int16_t)m_Opcode.immediate);
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int64_t rs = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs);
|
|
int64_t sum = rs + imm;
|
|
if ((~(rs ^ imm) & (rs ^ sum)) & 0x8000000000000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (m_Opcode.rt != 0)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rt, sum);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rt) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rt) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
asmjit::x86::Gp RegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
asmjit::x86::Gp RegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
|
|
m_Assembler.add(RegLo, (uint32_t)imm);
|
|
m_Assembler.adc(RegHi, (uint32_t)(imm >> 32));
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(m_Opcode.rs);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), RegLo);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), RegHi);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::DADDIU()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
int64_t imm = (int64_t)((int16_t)m_Opcode.immediate);
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int64_t rs = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rt, rs + imm);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rt) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rt) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rt, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
asmjit::x86::Gp RegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
asmjit::x86::Gp RegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
|
|
m_Assembler.add(RegLo, (uint32_t)imm);
|
|
m_Assembler.adc(RegHi, (uint32_t)(imm >> 32));
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(m_Opcode.rs);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), RegLo);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), RegHi);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CACHE()
|
|
{
|
|
if (g_Settings->LoadDword(Game_SMM_Cache) == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch (m_Opcode.rt)
|
|
{
|
|
case 0:
|
|
case 16:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("CRecompiler::Remove_Cache", CRecompiler::Remove_Cache);
|
|
m_Assembler.push(0x20);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
m_Assembler.PushImm32("Address", Address);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.base))
|
|
{
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.base), (int16_t)m_Opcode.offset);
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.base));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::eax, &m_Reg.m_GPR[m_Opcode.base].UW[0], CRegName::GPR_Lo[m_Opcode.base]);
|
|
m_Assembler.AddConstToX86Reg(asmjit::x86::eax, (int16_t)m_Opcode.offset);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
}
|
|
m_Assembler.CallThis((uint32_t)g_Recompiler, AddressOf(&CRecompiler::ClearRecompCode_Virt), "CRecompiler::ClearRecompCode_Virt", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 1:
|
|
case 3:
|
|
case 13:
|
|
case 5:
|
|
case 8:
|
|
case 9:
|
|
case 17:
|
|
case 21:
|
|
case 25:
|
|
break;
|
|
default:
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("cache: %d", m_Opcode.rt).c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LDL()
|
|
{
|
|
if (m_Opcode.base != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.base, true);
|
|
}
|
|
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::LDL), "R4300iOp::LDL", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::LDR()
|
|
{
|
|
if (m_Opcode.base != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.base, true);
|
|
}
|
|
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::LDR), "R4300iOp::LDR", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::LB_KnownAddress(const asmjit::x86::Gp & Reg, uint32_t VAddr, bool SignExtend)
|
|
{
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileLoadMemoryValue(AddressReg, Reg, x86Reg_Unknown, 8, SignExtend);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
m_CodeBlock.Log("%s\nFailed to translate address %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.MoveSxVariableToX86regByte(Reg, (PAddr ^ 3) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 3)", PAddr).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveZxVariableToX86regByte(Reg, (PAddr ^ 3) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 3)", PAddr).c_str());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
break;
|
|
case 0x04000000:
|
|
if (PAddr < 0x04001000)
|
|
{
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.MoveSxVariableToX86regByte(Reg, ((PAddr ^ 3) - 0x04000000) + g_MMU->Dmem(), stdstr_f("Dmem + (%X ^ 3)", (PAddr - 0x04000000)).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveZxVariableToX86regByte(Reg, ((PAddr ^ 3) - 0x04000000) + g_MMU->Dmem(), stdstr_f("Dmem + (%X ^ 3)", (PAddr - 0x04000000)).c_str());
|
|
}
|
|
}
|
|
else if (PAddr < 0x04002000)
|
|
{
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.MoveSxVariableToX86regByte(Reg, ((PAddr ^ 3) - 0x04001000) + g_MMU->Imem(), stdstr_f("Imem + (%X ^ 3)", (PAddr - 0x04001000)).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveZxVariableToX86regByte(Reg, ((PAddr ^ 3) - 0x04001000) + g_MMU->Imem(), stdstr_f("Imem + (%X ^ 3)", (PAddr - 0x04001000)).c_str());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x10000000:
|
|
if ((PAddr - 0x10000000) < g_Rom->GetRomSize())
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(((PAddr + 2) & ~0x3) & 0x1FFFFFFC);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][0], "RomMemoryHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
uint8_t Shift = (((PAddr & 1) ^ 3) << 3);
|
|
if (Shift == 0x10)
|
|
{
|
|
m_Assembler.shl(Reg, 0x8);
|
|
}
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.sar(Reg, 0x18);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.shr(Reg, 0x18);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LH_KnownAddress(const asmjit::x86::Gp & Reg, uint32_t VAddr, bool SignExtend)
|
|
{
|
|
uint32_t PAddr;
|
|
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileLoadMemoryValue(AddressReg, Reg, x86Reg_Unknown, 16, SignExtend);
|
|
}
|
|
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
m_CodeBlock.Log("%s\nFailed to translate address %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.MoveSxVariableToX86regHalf(Reg, (PAddr ^ 2) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 2)", PAddr).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveZxVariableToX86regHalf(Reg, (PAddr ^ 2) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 2)", PAddr).c_str());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
break;
|
|
case 0x10000000:
|
|
if ((PAddr - 0x10000000) < g_Rom->GetRomSize())
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(((PAddr + 2) & ~0x3) & 0x1FFFFFFC);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][0], "RomMemoryHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.sar(Reg, 16);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.shr(Reg, 16);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::RESERVED31()
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_IllegalInstruction, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
void CX86RecompilerOps::LB()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveReadBP() && g_Debugger->ReadBP8(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
LB_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address, true);
|
|
return;
|
|
}
|
|
PreReadInstruction();
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.base == m_Opcode.rt ? m_Opcode.rt : -1);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), x86Reg_Unknown, 8, true);
|
|
}
|
|
|
|
void CX86RecompilerOps::LH()
|
|
{
|
|
if (m_Opcode.rt == 0) return;
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveReadBP() && g_Debugger->ReadBP16(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
LH_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address, true);
|
|
return;
|
|
}
|
|
PreReadInstruction();
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, 16, true);
|
|
}
|
|
|
|
void CX86RecompilerOps::LWL()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
uint32_t Offset = Address & 3;
|
|
if (HaveReadBP() && g_Debugger->ReadBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rt);
|
|
asmjit::x86::Gp Value = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(Value, (Address & ~3));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), R4300iOp::LWL_MASK[Offset]);
|
|
m_Assembler.shl(Value, (uint8_t)R4300iOp::LWL_SHIFT[Offset]);
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Value);
|
|
}
|
|
else
|
|
{
|
|
PreReadInstruction();
|
|
asmjit::x86::Gp shift = m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, -1, false, false);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
asmjit::x86::Gp AddressReg = BaseOffsetAddress(false);
|
|
asmjit::x86::Gp OffsetReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(OffsetReg, AddressReg);
|
|
m_Assembler.and_(OffsetReg, 3);
|
|
m_Assembler.and_(AddressReg, (uint32_t)~3);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint32, "x86TestReadBreakpoint32");
|
|
CompileLoadMemoryValue(AddressReg, AddressReg, x86Reg_Unknown, 32, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rt);
|
|
m_Assembler.AndVariableDispToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), (void *)R4300iOp::LWL_MASK, "LWL_MASK", OffsetReg, CX86Ops::Multip_x4);
|
|
m_Assembler.MoveVariableDispToX86Reg(shift, (void *)R4300iOp::LWL_SHIFT, "LWL_SHIFT", OffsetReg, CX86Ops::Multip_x4);
|
|
m_Assembler.shl(AddressReg, asmjit::x86::cl);
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), AddressReg);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LW()
|
|
{
|
|
LW(true, false);
|
|
}
|
|
|
|
void CX86RecompilerOps::LW(bool ResultSigned, bool bRecordLLBit)
|
|
{
|
|
if (!HaveReadBP() && m_Opcode.base == 29 && g_System->bFastSP() && m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, ResultSigned, -1);
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true);
|
|
m_Assembler.MoveVariableDispToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), (void *)((uint32_t)(int16_t)m_Opcode.offset), stdstr_f("%Xh", (int16_t)m_Opcode.offset).c_str(), TempReg1, CX86Ops::Multip_x1);
|
|
if (bRecordLLBit)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsConst(m_Opcode.base) && m_RegWorkingSet.Is32Bit(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if ((Address & 3) != 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
//m_Reg.TriggerAddressException(VAddr, EXC_RADE);
|
|
return;
|
|
}
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
if (HaveReadBP() && g_Debugger->ReadBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, ResultSigned, -1);
|
|
LW_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address);
|
|
if (bRecordLLBit)
|
|
{
|
|
RecordLLAddress((int64_t)((int32_t)Address));
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
PreReadInstruction();
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, 32, false);
|
|
if (bRecordLLBit)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(AddressReg), true);
|
|
RecordLLAddress(AddressReg);
|
|
}
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LW_KnownAddress(const asmjit::x86::Gp & Reg, uint32_t VAddr)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(Reg), true);
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileLoadMemoryValue(AddressReg, Reg, x86Reg_Unknown, 32, true);
|
|
}
|
|
else
|
|
{
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, PAddr + g_MMU->Rdram(), stdstr_f("RDRAM + 0x%X", PAddr).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
}
|
|
break;
|
|
case 0x04000000:
|
|
if (PAddr < 0x04001000)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, (PAddr - 0x04000000) + g_MMU->Dmem(), stdstr_f("Dmem + 0x%X", (PAddr - 0x04000000)).c_str());
|
|
}
|
|
else if (PAddr < 0x04002000)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, (PAddr - 0x04001000) + g_MMU->Imem(), stdstr_f("Imem + 0x%X", (PAddr - 0x04001000)).c_str());
|
|
}
|
|
else
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x04040010: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SP_STATUS_REG, "SP_STATUS_REG"); break;
|
|
case 0x04040014: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SP_DMA_FULL_REG, "SP_DMA_FULL_REG"); break;
|
|
case 0x04040018: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SP_DMA_BUSY_REG, "SP_DMA_BUSY_REG"); break;
|
|
case 0x0404001C:
|
|
m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SP_SEMAPHORE_REG, "SP_SEMAPHORE_REG");
|
|
m_Assembler.MoveConstToVariable(&g_Reg->SP_SEMAPHORE_REG, "SP_SEMAPHORE_REG", 1);
|
|
break;
|
|
case 0x04080000: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SP_PC_REG, "SP_PC_REG"); break;
|
|
default:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(g_MMU), AddressOf(&CMipsMemoryVM::LW_PhysicalAddress), "CMipsMemoryVM::LW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 0x04100000:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(g_MMU), AddressOf(&CMipsMemoryVM::LW_PhysicalAddress), "CMipsMemoryVM::LW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
break;
|
|
case 0x04300000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04300000: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->MI_MODE_REG, "MI_MODE_REG"); break;
|
|
case 0x04300004: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->MI_VERSION_REG, "MI_VERSION_REG"); break;
|
|
case 0x04300008: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->MI_INTR_REG, "MI_INTR_REG"); break;
|
|
case 0x0430000C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->MI_INTR_MASK_REG, "MI_INTR_MASK_REG"); break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04400000:
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler)[0][0], "VideoInterfaceHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
break;
|
|
case 0x04500000:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler)[0][0], "AudioInterfaceHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
break;
|
|
case 0x04600000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04600000: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_DRAM_ADDR_REG, "PI_DRAM_ADDR_REG"); break;
|
|
case 0x04600004: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_CART_ADDR_REG, "PI_CART_ADDR_REG"); break;
|
|
case 0x04600008: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_RD_LEN_REG, "PI_RD_LEN_REG"); break;
|
|
case 0x0460000C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_WR_LEN_REG, "PI_WR_LEN_REG"); break;
|
|
case 0x04600010: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_STATUS_REG, "PI_STATUS_REG"); break;
|
|
case 0x04600014: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_DOMAIN1_REG, "PI_DOMAIN1_REG"); break;
|
|
case 0x04600018: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM1_PWD_REG, "PI_BSD_DOM1_PWD_REG"); break;
|
|
case 0x0460001C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM1_PGS_REG, "PI_BSD_DOM1_PGS_REG"); break;
|
|
case 0x04600020: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM1_RLS_REG, "PI_BSD_DOM1_RLS_REG"); break;
|
|
case 0x04600024: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_DOMAIN2_REG, "PI_DOMAIN2_REG"); break;
|
|
case 0x04600028: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM2_PWD_REG, "PI_BSD_DOM2_PWD_REG"); break;
|
|
case 0x0460002C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM2_PGS_REG, "PI_BSD_DOM2_PGS_REG"); break;
|
|
case 0x04600030: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->PI_BSD_DOM2_RLS_REG, "PI_BSD_DOM2_RLS_REG"); break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04700000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x0470000C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->RI_SELECT_REG, "RI_SELECT_REG"); break;
|
|
case 0x04700010: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->RI_REFRESH_REG, "RI_REFRESH_REG"); break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04800000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04800018: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->SI_STATUS_REG, "SI_STATUS_REG"); break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x05000000:
|
|
// 64DD registers
|
|
if (EnableDisk())
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x05000500: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_DATA, "ASIC_DATA"); break;
|
|
case 0x05000504: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_MISC_REG, "ASIC_MISC_REG"); break;
|
|
case 0x05000508:
|
|
m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_STATUS, "ASIC_STATUS");
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(&DiskGapSectorCheck), "DiskGapSectorCheck");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x0500050C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_CUR_TK, "ASIC_CUR_TK"); break;
|
|
case 0x05000510: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_BM_STATUS, "ASIC_BM_STATUS"); break;
|
|
case 0x05000514: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_ERR_SECTOR, "ASIC_ERR_SECTOR"); break;
|
|
case 0x05000518: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_SEQ_STATUS, "ASIC_SEQ_STATUS"); break;
|
|
case 0x0500051C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_CUR_SECTOR, "ASIC_CUR_SECTOR"); break;
|
|
case 0x05000520: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_HARD_RESET, "ASIC_HARD_RESET"); break;
|
|
case 0x05000524: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_C1_S0, "ASIC_C1_S0"); break;
|
|
case 0x05000528: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_HOST_SECBYTE, "ASIC_HOST_SECBYTE"); break;
|
|
case 0x0500052C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_C1_S2, "ASIC_C1_S2"); break;
|
|
case 0x05000530: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_SEC_BYTE, "ASIC_SEC_BYTE"); break;
|
|
case 0x05000534: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_C1_S4, "ASIC_C1_S4"); break;
|
|
case 0x05000538: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_C1_S6, "ASIC_C1_S6"); break;
|
|
case 0x0500053C: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_CUR_ADDR, "ASIC_CUR_ADDR"); break;
|
|
case 0x05000540: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_ID_REG, "ASIC_ID_REG"); break;
|
|
case 0x05000544: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_TEST_REG, "ASIC_TEST_REG"); break;
|
|
case 0x05000548: m_Assembler.MoveVariableToX86reg(Reg, &g_Reg->ASIC_TEST_PIN_SEL, "ASIC_TEST_PIN_SEL"); break;
|
|
default:
|
|
m_Assembler.MoveConstToX86reg(Reg, 0);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, (uint32_t)((PAddr & 0xFFFF) << 16) | (PAddr & 0xFFFF));
|
|
}
|
|
break;
|
|
case 0x1FC00000:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_PifRamHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_PifRamHandler)[0][0], "PifRamHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
break;
|
|
default:
|
|
if ((PAddr & 0xF0000000) == 0x10000000 && (PAddr - 0x10000000) < g_Rom->GetRomSize())
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][0], "RomMemoryHandler::Read32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
}
|
|
else if (g_DDRom != nullptr && ((PAddr & 0xFF000000) == 0x06000000 && (PAddr - 0x06000000) < g_DDRom->GetRomSize()))
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(Reg, g_DDRom->GetRomAddress() + (PAddr - 0x06000000), stdstr_f("DDRom + %X", (PAddr - 0x06000000)).c_str());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToX86reg(Reg, ((PAddr & 0xFFFF) << 16) | (PAddr & 0xFFFF));
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LBU()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveReadBP() && g_Debugger->ReadBP8(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
LB_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address, false);
|
|
return;
|
|
}
|
|
PreReadInstruction();
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.base == m_Opcode.rt ? m_Opcode.rt : -1);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), x86Reg_Unknown, 8, false);
|
|
}
|
|
|
|
void CX86RecompilerOps::LHU()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveReadBP() && g_Debugger->ReadBP16(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
LH_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address, false);
|
|
return;
|
|
}
|
|
PreReadInstruction();
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, m_Opcode.base == m_Opcode.rt ? m_Opcode.rt : -1);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), x86Reg_Unknown, 16, false);
|
|
}
|
|
|
|
void CX86RecompilerOps::LWR()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
uint32_t Offset = Address & 3;
|
|
if (HaveReadBP() && g_Debugger->ReadBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rt);
|
|
asmjit::x86::Gp Value = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(Value, (Address & ~3));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), R4300iOp::LWR_MASK[Offset]);
|
|
m_Assembler.shr(Value, (uint8_t)R4300iOp::LWR_SHIFT[Offset]);
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Value);
|
|
}
|
|
else
|
|
{
|
|
PreReadInstruction();
|
|
asmjit::x86::Gp shift = m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, -1, false, false);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
asmjit::x86::Gp AddressReg = BaseOffsetAddress(false);
|
|
asmjit::x86::Gp OffsetReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(OffsetReg, AddressReg);
|
|
m_Assembler.and_(OffsetReg, 3);
|
|
m_Assembler.and_(AddressReg, (uint32_t)~3);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint32, "x86TestReadBreakpoint32");
|
|
CompileLoadMemoryValue(AddressReg, AddressReg, x86Reg_Unknown, 32, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, m_Opcode.rt);
|
|
m_Assembler.AndVariableDispToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), (void *)R4300iOp::LWR_MASK, "LWR_MASK", OffsetReg, CX86Ops::Multip_x4);
|
|
m_Assembler.MoveVariableDispToX86Reg(shift, (void *)R4300iOp::LWR_SHIFT, "LWR_SHIFT", OffsetReg, CX86Ops::Multip_x4);
|
|
m_Assembler.shr(AddressReg, asmjit::x86::cl);
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), AddressReg);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LWU()
|
|
{
|
|
LW(false, false);
|
|
}
|
|
|
|
void CX86RecompilerOps::SB()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveWriteBP() && g_Debugger->WriteBP8(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
SB_Const(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), Address);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_Assembler.Is8BitReg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt)))
|
|
{
|
|
SB_Register(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address);
|
|
}
|
|
else
|
|
{
|
|
SB_Register(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, true), Address);
|
|
}
|
|
return;
|
|
}
|
|
PreWriteInstruction();
|
|
|
|
asmjit::x86::Gp ValueReg;
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
ValueReg = m_RegWorkingSet.IsUnknown(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, true) : m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && !m_Assembler.Is8BitReg(ValueReg))
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(m_Opcode.rt);
|
|
ValueReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, true);
|
|
}
|
|
}
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueReg, x86Reg_Unknown, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), 8);
|
|
}
|
|
|
|
void CX86RecompilerOps::SH()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset);
|
|
if (HaveWriteBP() && g_Debugger->WriteBP16(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
SH_Const(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), Address);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
SH_Register(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address);
|
|
}
|
|
else
|
|
{
|
|
SH_Register(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false), Address);
|
|
}
|
|
return;
|
|
}
|
|
|
|
PreWriteInstruction();
|
|
|
|
asmjit::x86::Gp ValueReg;
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
ValueReg = m_RegWorkingSet.IsUnknown(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false) : m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueReg, x86Reg_Unknown, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), 16);
|
|
}
|
|
|
|
void CX86RecompilerOps::SWL()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address;
|
|
|
|
Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
uint32_t Offset = Address & 3;
|
|
|
|
asmjit::x86::Gp Value = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(Value, (Address & ~3));
|
|
m_Assembler.and_(Value, R4300iOp::SWL_MASK[Offset]);
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_Assembler.shr(TempReg1, (uint8_t)R4300iOp::SWL_SHIFT[Offset]);
|
|
m_Assembler.add(Value, TempReg1);
|
|
SW_Register(Value, (Address & ~3));
|
|
return;
|
|
}
|
|
PreWriteInstruction();
|
|
asmjit::x86::Gp shift = m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, -1, false, false), AddressReg = BaseOffsetAddress(false);
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint32, "x86TestWriteBreakpoint32");
|
|
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp OffsetReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp ValueReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(TempReg2, AddressReg);
|
|
m_Assembler.shr(TempReg2, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg2, g_MMU->m_MemoryReadMap, "MMU->m_MemoryReadMap", TempReg2, CX86Ops::Multip_x4);
|
|
m_Assembler.CompConstToX86reg(TempReg2, (uint32_t)-1);
|
|
asmjit::Label JumpFound = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("MemoryReadMap_%X_Found", m_CompilePC).c_str(), JumpFound);
|
|
m_Assembler.mov(TempReg2, AddressReg);
|
|
m_Assembler.shr(TempReg2, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg2, g_MMU->m_TLB_ReadMap, "MMU->TLB_ReadMap", TempReg2, CX86Ops::Multip_x4);
|
|
CompileReadTLBMiss(AddressReg, TempReg2);
|
|
m_Assembler.AddConstToX86Reg(TempReg2, (uint32_t)m_MMU.Rdram());
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpFound);
|
|
m_Assembler.mov(OffsetReg, AddressReg);
|
|
m_Assembler.and_(OffsetReg, 3);
|
|
m_Assembler.and_(AddressReg, (uint32_t)~3);
|
|
m_Assembler.mov(ValueReg, asmjit::x86::dword_ptr(AddressReg, TempReg2));
|
|
|
|
m_Assembler.AndVariableDispToX86Reg(ValueReg, (void *)R4300iOp::SWL_MASK, "R4300iOp::SWL_MASK", OffsetReg, CX86Ops::Multip_x4);
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt) || m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) != 0)
|
|
{
|
|
m_Assembler.MoveVariableDispToX86Reg(shift, (void *)R4300iOp::SWL_SHIFT, "R4300iOp::SWL_SHIFT", OffsetReg, CX86Ops::Multip_x4);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(OffsetReg, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(OffsetReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(OffsetReg, &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
}
|
|
m_Assembler.shr(OffsetReg, asmjit::x86::cl);
|
|
m_Assembler.add(ValueReg, OffsetReg);
|
|
}
|
|
|
|
CompileStoreMemoryValue(AddressReg, ValueReg, x86Reg_Unknown, 0, 32);
|
|
}
|
|
|
|
void CX86RecompilerOps::SW()
|
|
{
|
|
SW(false);
|
|
}
|
|
|
|
void CX86RecompilerOps::SW(bool bCheckLLbit)
|
|
{
|
|
if (!HaveWriteBP() && m_Opcode.base == 29 && g_System->bFastSP())
|
|
{
|
|
if (bCheckLLbit)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true);
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg1, (uint32_t)((int16_t)m_Opcode.offset)), m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg1, (uint32_t)((int16_t)m_Opcode.offset)), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg1, (uint32_t)((int16_t)m_Opcode.offset)), TempReg2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base) && m_RegWorkingSet.Is32Bit(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if ((Address & 3) != 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
if (bCheckLLbit)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
SW_Const(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), Address);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
SW_Register(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address);
|
|
}
|
|
else
|
|
{
|
|
SW_Register(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false), Address);
|
|
}
|
|
return;
|
|
}
|
|
else if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
PreWriteInstruction();
|
|
asmjit::Label JumpLLBit;
|
|
if (bCheckLLbit)
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_LLBit, "_LLBit", 1);
|
|
JumpLLBit = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("LLBit_Continue", JumpLLBit);
|
|
}
|
|
|
|
asmjit::x86::Gp ValueReg;
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
ValueReg = m_RegWorkingSet.IsUnknown(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false) : m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueReg, x86Reg_Unknown, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), 32);
|
|
if (bCheckLLbit)
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpLLBit);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_Reg.m_LLBit, "_LLBit");
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SWR()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
uint32_t Offset = Address & 3;
|
|
|
|
asmjit::x86::Gp Value = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(Value, (Address & ~3));
|
|
m_Assembler.and_(Value, R4300iOp::SWR_MASK[Offset]);
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_Assembler.shl(TempReg, (uint8_t)R4300iOp::SWR_SHIFT[Offset]);
|
|
m_Assembler.add(Value, TempReg);
|
|
SW_Register(Value, (Address & ~3));
|
|
return;
|
|
}
|
|
PreWriteInstruction();
|
|
asmjit::x86::Gp shift = m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, -1, false, false);
|
|
asmjit::x86::Gp AddressReg = BaseOffsetAddress(false);
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint32, "x86TestWriteBreakpoint32");
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp OffsetReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp ValueReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(TempReg2, AddressReg);
|
|
m_Assembler.shr(TempReg2, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg2, g_MMU->m_MemoryReadMap, "MMU->m_MemoryReadMap", TempReg2, CX86Ops::Multip_x4);
|
|
m_Assembler.CompConstToX86reg(TempReg2, (uint32_t)-1);
|
|
asmjit::Label JumpFound = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("MemoryReadMap_%X_Found", m_CompilePC).c_str(), JumpFound);
|
|
m_Assembler.mov(TempReg2, AddressReg);
|
|
m_Assembler.shr(TempReg2, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg2, g_MMU->m_TLB_ReadMap, "MMU->TLB_ReadMap", TempReg2, CX86Ops::Multip_x4);
|
|
CompileReadTLBMiss(AddressReg, TempReg2);
|
|
m_Assembler.AddConstToX86Reg(TempReg2, (uint32_t)m_MMU.Rdram());
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpFound);
|
|
|
|
m_Assembler.mov(OffsetReg, AddressReg);
|
|
m_Assembler.and_(OffsetReg, 3);
|
|
m_Assembler.and_(AddressReg, (uint32_t)~3);
|
|
|
|
m_Assembler.mov(ValueReg, asmjit::x86::dword_ptr(AddressReg, TempReg2));
|
|
|
|
m_Assembler.AndVariableDispToX86Reg(ValueReg, (void *)R4300iOp::SWR_MASK, "R4300iOp::SWR_MASK", OffsetReg, CX86Ops::Multip_x4);
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt) || m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) != 0)
|
|
{
|
|
m_Assembler.MoveVariableDispToX86Reg(shift, (void *)R4300iOp::SWR_SHIFT, "R4300iOp::SWR_SHIFT", OffsetReg, CX86Ops::Multip_x4);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(OffsetReg, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(OffsetReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(OffsetReg, &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
}
|
|
m_Assembler.shl(OffsetReg, asmjit::x86::cl);
|
|
m_Assembler.add(ValueReg, OffsetReg);
|
|
}
|
|
|
|
CompileStoreMemoryValue(AddressReg, ValueReg, x86Reg_Unknown, 0, 32);
|
|
}
|
|
|
|
void CX86RecompilerOps::SDL()
|
|
{
|
|
if (m_Opcode.base != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.base, true);
|
|
}
|
|
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SDL), "R4300iOp::SDL", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::SDR()
|
|
{
|
|
if (m_Opcode.base != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.base, true);
|
|
}
|
|
|
|
if (m_Opcode.rt != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SDR), "R4300iOp::SDR", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::LL()
|
|
{
|
|
LW(true, true);
|
|
}
|
|
|
|
void CX86RecompilerOps::LWC1()
|
|
{
|
|
CompileCop1Test();
|
|
if ((m_Opcode.ft & 1) != 0)
|
|
{
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.ft - 1, CRegInfo::FPU_Double) || m_RegWorkingSet.RegInStack(m_Opcode.ft - 1, CRegInfo::FPU_Qword))
|
|
{
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft - 1, true);
|
|
}
|
|
}
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.ft, CRegInfo::FPU_Double) || m_RegWorkingSet.RegInStack(m_Opcode.ft, CRegInfo::FPU_Qword))
|
|
{
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, true);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, false);
|
|
}
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveReadBP() && g_Debugger->ReadBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(TempReg1, Address);
|
|
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg2, &m_Reg.m_FPR_S[m_Opcode.ft], stdstr_f("m_FPR_S[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg2), TempReg1);
|
|
return;
|
|
}
|
|
PreReadInstruction();
|
|
asmjit::x86::Gp ValueReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, ValueReg, x86Reg_Unknown, 32, false);
|
|
asmjit::x86::Gp FPR_SPtr = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(FPR_SPtr, &m_Reg.m_FPR_S[m_Opcode.ft], stdstr_f("m_FPR_S[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FPR_SPtr), ValueReg);
|
|
}
|
|
|
|
void CX86RecompilerOps::LDC1()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, false);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveReadBP() && g_Debugger->ReadBP64(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
LW_KnownAddress(TempReg1, Address);
|
|
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg2, &m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("_FPR_D[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.AddConstToX86Reg(TempReg2, 4);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg2), TempReg1);
|
|
|
|
LW_KnownAddress(TempReg1, Address + 4);
|
|
m_Assembler.MoveVariableToX86reg(TempReg2, &m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("m_FPR_S[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg2), TempReg1);
|
|
}
|
|
else
|
|
{
|
|
PreReadInstruction();
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, true);
|
|
|
|
asmjit::x86::Gp ValueRegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false), ValueRegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, ValueRegLo, ValueRegHi, 64, false);
|
|
|
|
asmjit::x86::Gp FPR_DPtr = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(FPR_DPtr, &m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("_FPR_D[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FPR_DPtr), ValueRegLo);
|
|
m_Assembler.AddConstToX86Reg(FPR_DPtr, 4);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FPR_DPtr), ValueRegHi);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::LD()
|
|
{
|
|
if (m_Opcode.rt == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!HaveReadBP() && m_Opcode.base == 29 && g_System->bFastSP())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
asmjit::x86::Gp StackReg = m_RegWorkingSet.Map_MemoryStack(x86Reg_Unknown, true);
|
|
m_Assembler.MoveVariableDispToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), (void *)((uint32_t)(int16_t)m_Opcode.offset), stdstr_f("%Xh", (int16_t)m_Opcode.offset).c_str(), StackReg, CX86Ops::Multip_x1);
|
|
m_Assembler.MoveVariableDispToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), (void *)((uint32_t)(int16_t)m_Opcode.offset + 4), stdstr_f("%Xh", (int16_t)m_Opcode.offset + 4).c_str(), StackReg, CX86Ops::Multip_x1);
|
|
}
|
|
else if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveReadBP() && g_Debugger->ReadBP64(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
LW_KnownAddress(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), Address);
|
|
LW_KnownAddress(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address + 4);
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29)
|
|
{
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PreReadInstruction();
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, m_Opcode.rt == m_Opcode.base ? m_Opcode.base : -1);
|
|
asmjit::x86::Gp AddressReg = x86Reg_Unknown;
|
|
CompileLoadMemoryValue(AddressReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), 64, false);
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rt == 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SC()
|
|
{
|
|
SW(true);
|
|
}
|
|
|
|
void CX86RecompilerOps::SWC1()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, true);
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg1, &m_Reg.m_FPR_S[m_Opcode.ft], stdstr_f("m_FPR_S[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(TempReg1, asmjit::x86::dword_ptr(TempReg1));
|
|
SW_Register(TempReg1, Address);
|
|
return;
|
|
}
|
|
PreWriteInstruction();
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, true);
|
|
asmjit::x86::Gp ValueReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(ValueReg, &m_Reg.m_FPR_S[m_Opcode.ft], stdstr_f("m_FPR_S[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(ValueReg, asmjit::x86::dword_ptr(ValueReg));
|
|
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueReg, x86Reg_Unknown, 0, 32);
|
|
}
|
|
|
|
void CX86RecompilerOps::SDC1()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg1, (uint8_t *)&m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("_FPR_D[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.AddConstToX86Reg(TempReg1, 4);
|
|
m_Assembler.mov(TempReg1, asmjit::x86::dword_ptr(TempReg1));
|
|
SW_Register(TempReg1, Address);
|
|
|
|
m_Assembler.MoveVariableToX86reg(TempReg1, &m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("_FPR_D[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(TempReg1, asmjit::x86::dword_ptr(TempReg1));
|
|
SW_Register(TempReg1, Address + 4);
|
|
return;
|
|
}
|
|
PreWriteInstruction();
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.ft, true);
|
|
asmjit::x86::Gp ValueRegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false), ValueRegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(ValueRegHi, (uint8_t *)&m_Reg.m_FPR_D[m_Opcode.ft], stdstr_f("_FPR_D[%d]", m_Opcode.ft).c_str());
|
|
m_Assembler.mov(ValueRegLo, ValueRegHi);
|
|
m_Assembler.AddConstToX86Reg(ValueRegHi, 4);
|
|
m_Assembler.mov(ValueRegHi, asmjit::x86::dword_ptr(ValueRegHi));
|
|
m_Assembler.mov(ValueRegLo, asmjit::x86::dword_ptr(ValueRegLo));
|
|
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueRegLo, ValueRegHi, 0, 64);
|
|
}
|
|
|
|
void CX86RecompilerOps::SD()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
uint32_t Address = m_RegWorkingSet.GetMipsRegLo(m_Opcode.base) + (int16_t)m_Opcode.offset;
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(Address))
|
|
{
|
|
FoundMemoryBreakpoint();
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
SW_Const(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt) : (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> 31), Address);
|
|
SW_Const(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt), Address + 4);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
SW_Register(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false), Address);
|
|
SW_Register(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), Address + 4);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp TempReg1 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
SW_Register(TempReg1, Address);
|
|
SW_Register(m_RegWorkingSet.Map_TempReg(TempReg1, m_Opcode.rt, false, false), Address + 4);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PreWriteInstruction();
|
|
asmjit::x86::Gp ValueReg;
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
ValueReg = m_RegWorkingSet.IsUnknown(m_Opcode.rt) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false) : m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
uint64_t RtValue = 0;
|
|
asmjit::x86::Gp ValueRegHi, ValueRegLo;
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
RtValue = ((uint64_t)(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt) : (uint32_t)(m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> 31)) << 32) | m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt);
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
ValueRegHi = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
ValueRegLo = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
else
|
|
{
|
|
ValueRegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
ValueRegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
}
|
|
CompileStoreMemoryValue(x86Reg_Unknown, ValueReg, ValueRegHi, RtValue, 64);
|
|
}
|
|
}
|
|
|
|
// R4300i opcodes: Special
|
|
void CX86RecompilerOps::SPECIAL_SLL()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) << m_Opcode.sa);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
if (m_Opcode.rd != m_Opcode.rt && m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
switch (m_Opcode.sa)
|
|
{
|
|
case 0:
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
break;
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.lea(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::ptr(0, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_Opcode.sa));
|
|
break;
|
|
default:
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SRL()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) >> m_Opcode.sa);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SRA()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> m_Opcode.sa);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), reg, (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SLLV()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Shift = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x1F);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) << Shift);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)Shift);
|
|
}
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SRLV()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Shift = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x1F);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) >> Shift);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)Shift);
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SRAV()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Shift = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x1F);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> Shift);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)Shift);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Reg, (uint8_t)Shift);
|
|
}
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
if (b32BitCore())
|
|
{
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Reg, asmjit::x86::cl);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_JR()
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
m_Assembler.MoveX86regToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
OverflowDelaySlot(true);
|
|
return;
|
|
}
|
|
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_Section->m_Cont.FallThrough = false;
|
|
m_Section->m_Cont.LinkLocation = asmjit::Label();
|
|
m_Section->m_Cont.LinkLocation2 = asmjit::Label();
|
|
|
|
R4300iOpcode DelaySlot;
|
|
if (g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value) &&
|
|
R4300iInstruction(m_CompilePC, m_Opcode.Value).DelaySlotEffectsCompare(DelaySlot.Value))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
}
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
R4300iOpcode DelaySlot;
|
|
if (g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value) && R4300iInstruction(m_CompilePC, m_Opcode.Value).DelaySlotEffectsCompare(DelaySlot.Value))
|
|
{
|
|
CompileExit(m_CompilePC, (uint64_t)-1, m_RegWorkingSet, ExitReason_CheckPCAlignment, true, nullptr);
|
|
}
|
|
else
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, ExitReason_CheckPCAlignment, true, nullptr);
|
|
if (m_Section->m_JumpSection)
|
|
{
|
|
m_Section->GenerateSectionLinkage();
|
|
}
|
|
}
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("WTF\n\nBranch\nNextInstruction = %X", m_PipelineStage).c_str());
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_JALR()
|
|
{
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
R4300iOpcode DelaySlot;
|
|
if (g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value) &&
|
|
R4300iInstruction(m_CompilePC, m_Opcode.Value).DelaySlotEffectsCompare(DelaySlot.Value) && (m_CompilePC & 0xFFC) != 0xFFC)
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
}
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, (uint32_t)(m_CompilePC + 8));
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
if ((m_CompilePC & 0xFFC) == 0xFFC)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
m_Assembler.MoveX86regToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
OverflowDelaySlot(true);
|
|
return;
|
|
}
|
|
|
|
m_Section->m_Jump.FallThrough = false;
|
|
m_Section->m_Jump.LinkLocation = asmjit::Label();
|
|
m_Section->m_Jump.LinkLocation2 = asmjit::Label();
|
|
m_Section->m_Cont.FallThrough = false;
|
|
m_Section->m_Cont.LinkLocation = asmjit::Label();
|
|
m_Section->m_Cont.LinkLocation2 = asmjit::Label();
|
|
|
|
m_PipelineStage = PIPELINE_STAGE_DO_DELAY_SLOT;
|
|
}
|
|
else if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT_DONE)
|
|
{
|
|
R4300iOpcode DelaySlot;
|
|
if (g_MMU->MemoryValue32((uint32_t)(m_CompilePC + 4), DelaySlot.Value) &&
|
|
R4300iInstruction(m_CompilePC, m_Opcode.Value).DelaySlotEffectsCompare(DelaySlot.Value))
|
|
{
|
|
CompileExit(m_CompilePC, (uint64_t)-1, m_RegWorkingSet, ExitReason_CheckPCAlignment, true, nullptr);
|
|
}
|
|
else
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false));
|
|
}
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, ExitReason_CheckPCAlignment, true, nullptr);
|
|
if (m_Section->m_JumpSection)
|
|
{
|
|
m_Section->GenerateSectionLinkage();
|
|
}
|
|
}
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError(stdstr_f("WTF\n\nBranch\nNextInstruction = %X", m_PipelineStage).c_str());
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SYSCALL()
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_DoSysCall, true, nullptr);
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_BREAK()
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_Break, true, nullptr);
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SYNC()
|
|
{
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MFLO()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_LO.UW[0], "_RegLO->UW[0]");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_LO.UW[1], "_RegLO->UW[1]");
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MTLO()
|
|
{
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", m_RegWorkingSet.GetMipsRegHi(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs) && ((m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x80000000) != 0))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", reg);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", m_RegWorkingSet.Map_TempReg(reg, m_Opcode.rs, false, false));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MFHI()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_HI.UW[0], "_RegHI->UW[0]");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_HI.UW[1], "_RegHI->UW[1]");
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MTHI()
|
|
{
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", m_RegWorkingSet.GetMipsRegHi(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs) && ((m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x80000000) != 0))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0xFFFFFFFF);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rs) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs));
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", reg);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", m_RegWorkingSet.Map_TempReg(reg, m_Opcode.rs, false, false));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSLLV()
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x3F);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.CompConstToX86reg(asmjit::x86::ecx, 0x20);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JaeLabel("MORE32", Jump[0]);
|
|
m_Assembler.shld(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
// MORE32:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
|
|
// Continue:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRLV()
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t Shift = (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) & 0x3F);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt));
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, m_RegWorkingSet.GetMipsReg(m_Opcode.rd) >> Shift);
|
|
if ((m_RegWorkingSet.GetMipsRegHi(m_Opcode.rd) == 0) && (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rd) & 0x80000000) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if ((m_RegWorkingSet.GetMipsRegHi(m_Opcode.rd) == 0xFFFFFFFF) && (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rd) & 0x80000000) != 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::ecx, Shift);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
if ((Shift & 0x20) == 0x20)
|
|
{
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x3F);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.CompConstToX86reg(asmjit::x86::ecx, 0x20);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JaeLabel("MORE32", Jump[0]);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
// MORE32:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
|
|
// Continue:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRAV()
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::ecx, m_Opcode.rs, false, false);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x3F);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.CompConstToX86reg(asmjit::x86::ecx, 0x20);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JaeLabel("MORE32", Jump[0]);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), asmjit::x86::cl);
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
// MORE32:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), 0x1F);
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), asmjit::x86::cl);
|
|
|
|
// Continue:
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MULT()
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, true);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, false);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::edx, m_Opcode.rt, false, false);
|
|
|
|
m_Assembler.imul(asmjit::x86::edx);
|
|
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31); // Paired
|
|
m_Assembler.sar(asmjit::x86::edx, 31);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", asmjit::x86::edx);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_MULTU()
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, true);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, false);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::edx, m_Opcode.rt, false, false);
|
|
|
|
m_Assembler.mul(asmjit::x86::edx);
|
|
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31); // Paired
|
|
m_Assembler.sar(asmjit::x86::edx, 31);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", asmjit::x86::edx);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DIV()
|
|
{
|
|
asmjit::x86::Gp RegRs, RegRsHi, DivReg;
|
|
asmjit::Label JumpNotDiv0;
|
|
asmjit::Label JumpEnd;
|
|
asmjit::Label JumpEnd2;
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) == 0)
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < 0 ? 0x00000001 : 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < 0 ? 0x00000000 : 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < 0 ? 0xFFFFFFFF : 0x00000000);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_Assembler.CompConstToX86reg(Reg, 0);
|
|
asmjit::Label JumpPositive = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(stdstr_f("RsPositive_%08X", m_CompilePC).c_str(), JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x00000001);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0x00000000);
|
|
asmjit::Label JumpLoSet = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("LoSet_%08X", m_CompilePC).c_str(), JumpLoSet);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpLoSet);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", Reg);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.sar(Reg, 31);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", Reg);
|
|
}
|
|
return;
|
|
}
|
|
else if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) == -1)
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs) && m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) == 0x80000000)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x80000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", 0x00000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0x00000000);
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::edx);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::eax);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
RegRs = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(RegRs), true);
|
|
RegRsHi = m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_Opcode.rs : -1, true, false);
|
|
DivReg = m_RegWorkingSet.IsMapped(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(RegRs, 0x80000000);
|
|
asmjit::Label JumpValid = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("ValidDiv_%08X", m_CompilePC).c_str(), JumpValid);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x80000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", 0x00000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0x00000000);
|
|
JumpEnd = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("EndDiv_%08X", m_CompilePC).c_str(), JumpEnd);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpValid);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::edx);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::eax);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
RegRs = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(RegRs), true);
|
|
RegRsHi = m_RegWorkingSet.IsMapped(m_Opcode.rs) && m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_Opcode.rs : -1, true, false);
|
|
DivReg = m_RegWorkingSet.IsMapped(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.CompConstToX86reg(DivReg, 0);
|
|
JumpNotDiv0 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("NotDiv0_%08X", m_CompilePC).c_str(), JumpNotDiv0);
|
|
|
|
m_Assembler.CompConstToX86reg(RegRs, 0);
|
|
asmjit::Label JumpPositive = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(stdstr_f("RsPositive_%08X", m_CompilePC).c_str(), JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x00000001);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0x00000000);
|
|
asmjit::Label JumpLoSet = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("LoSet_%08X", m_CompilePC).c_str(), JumpLoSet);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpLoSet);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", RegRs);
|
|
if (!m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_Assembler.sar(RegRsHi, 31);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", RegRsHi);
|
|
JumpEnd = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("EndDiv_%08X", m_CompilePC).c_str(), JumpEnd);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpNotDiv0);
|
|
m_Assembler.CompConstToX86reg(DivReg, (uint32_t)-1);
|
|
asmjit::Label JumpValidDiv0 = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("ValidDiv0_%08X", m_CompilePC).c_str(), JumpValidDiv0);
|
|
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x80000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", 0x00000000);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", 0x00000000);
|
|
JumpEnd2 = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("EndDiv_%08X", m_CompilePC).c_str(), JumpEnd2);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpValidDiv0);
|
|
}
|
|
}
|
|
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::edx);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::edx, -1, false, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, false);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, m_Opcode.rs, false, false);
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToX86reg(asmjit::x86::edx, m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) >> 31);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::edx, asmjit::x86::eax);
|
|
m_Assembler.sar(asmjit::x86::edx, 31);
|
|
}
|
|
|
|
m_Assembler.idiv(DivReg);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31);
|
|
m_Assembler.sar(asmjit::x86::edx, 31);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", asmjit::x86::edx);
|
|
|
|
if (JumpEnd.isValid() || JumpEnd2.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
if (JumpEnd.isValid())
|
|
{
|
|
m_Assembler.bind(JumpEnd);
|
|
}
|
|
if (JumpEnd2.isValid())
|
|
{
|
|
m_Assembler.bind(JumpEnd2);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DIVU()
|
|
{
|
|
asmjit::Label JumpEndDivu;
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) == 0)
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < 0 ? 0xFFFFFFFF : 0x00000000);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp RegRs = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_Assembler.CompConstToX86reg(RegRs, 0);
|
|
asmjit::Label JumpPositive = m_Assembler.newLabel();
|
|
m_Assembler.JgeLabel(stdstr_f("RsPositive_%08X", m_CompilePC).c_str(), JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0x00000001);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0x00000000);
|
|
asmjit::Label JumpLoSet = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("LoSet_%08X", m_CompilePC).c_str(), JumpLoSet);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpPositive);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpLoSet);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", RegRs);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
RegRs = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.sar(RegRs, 31);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", RegRs);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::edx);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, true);
|
|
m_RegWorkingSet.UnMap_X86reg(asmjit::x86::eax);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
asmjit::x86::Gp RegRsLo = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
asmjit::x86::Gp RegRsHi = m_RegWorkingSet.IsMapped(m_Opcode.rs) ? m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_RegWorkingSet.IsMapped(m_Opcode.rs), true, false) : x86Reg_Unknown;
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, true);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EDX, false);
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::edx, 0, false, false);
|
|
m_RegWorkingSet.SetX86Protected(x86RegIndex_EAX, false);
|
|
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, m_Opcode.rs, false, false);
|
|
asmjit::x86::Gp DivReg = m_RegWorkingSet.IsMapped(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
|
|
if (!m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.CompConstToX86reg(DivReg, 0);
|
|
asmjit::Label JumpNoExcept = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("NoExcept", JumpNoExcept);
|
|
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", 0xFFFFFFFF);
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", 0xFFFFFFFF);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", RegRsLo);
|
|
if (!m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
RegRsHi = RegRsLo;
|
|
m_Assembler.sar(RegRsHi, 31);
|
|
}
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", RegRsHi);
|
|
|
|
JumpEndDivu = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("EndDivu", JumpEndDivu);
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpNoExcept);
|
|
}
|
|
m_Assembler.div(DivReg);
|
|
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[0], "_RegLO->UW[0]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[0], "_RegHI->UW[0]", asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31);
|
|
m_Assembler.sar(asmjit::x86::edx, 31);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_LO.UW[1], "_RegLO->UW[1]", asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_HI.UW[1], "_RegHI->UW[1]", asmjit::x86::edx);
|
|
|
|
if (JumpEndDivu.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpEndDivu);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DMULT()
|
|
{
|
|
if (m_Opcode.rs != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rs, true);
|
|
}
|
|
|
|
if (m_Opcode.rs != 0)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
}
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SPECIAL_DMULT), "R4300iOp::SPECIAL_DMULT", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DMULTU()
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rs, true);
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SPECIAL_DMULTU), "R4300iOp::SPECIAL_DMULTU", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DDIV()
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rs, true);
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SPECIAL_DDIV), "R4300iOp::SPECIAL_DDIV", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DDIVU()
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rs, true);
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rt, true);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::SPECIAL_DDIVU), "R4300iOp::SPECIAL_DDIVU", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_ADD()
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(source1) && m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
int32_t Val1 = m_RegWorkingSet.GetMipsRegLo(source1);
|
|
int32_t Val2 = m_RegWorkingSet.GetMipsRegLo(source2);
|
|
int32_t Sum = Val1 + Val2;
|
|
if ((~(Val1 ^ Val2) & (Val1 ^ Sum)) & 0x80000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (m_Opcode.rd != 0)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, Sum);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rd);
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source1, false, false);
|
|
if (m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
m_Assembler.AddConstToX86Reg(Reg, m_RegWorkingSet.GetMipsRegLo(source2));
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(source2) && m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
m_Assembler.add(Reg, m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AddVariableToX86reg(Reg, &m_Reg.m_GPR[source2].W[0], CRegName::GPR_Lo[source2]);
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rd == 29)
|
|
{
|
|
ResetMemoryStack();
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rd != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Reg);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_ADDU()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(source1) && m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
uint32_t temp = m_RegWorkingSet.GetMipsRegLo(source1) + m_RegWorkingSet.GetMipsRegLo(source2);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, temp);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, source1);
|
|
if (m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegLo(source2));
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(source2) && m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AddVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[source2].W[0], CRegName::GPR_Lo[source2]);
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rd == 29)
|
|
{
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SUB()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int32_t rs = m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs);
|
|
int32_t rt = m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt);
|
|
int32_t sub = rs - rt;
|
|
|
|
if (((rs ^ rt) & (rs ^ sub)) & 0x80000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else if (m_Opcode.rd != 0)
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, sub);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rd);
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.sub(Reg, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.sub(Reg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SubVariableFromX86reg(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rd != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Reg);
|
|
}
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rd == 29)
|
|
{
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SUBU()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
uint32_t temp = m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) - m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt);
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, temp);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
if (m_Opcode.rd == m_Opcode.rt)
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rs);
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Reg);
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rs);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SubVariableFromX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
}
|
|
}
|
|
|
|
if (g_System->bFastSP() && m_Opcode.rd == 29)
|
|
{
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_AND()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd,
|
|
(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt)) &
|
|
(m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs)));
|
|
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) & m_RegWorkingSet.GetMipsReg(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(source1);
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
if (m_RegWorkingSet.Is32Bit(source1) && m_RegWorkingSet.Is32Bit(source2))
|
|
{
|
|
bool Sign = (m_RegWorkingSet.IsSigned(m_Opcode.rt) && m_RegWorkingSet.IsSigned(m_Opcode.rs));
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, Sign, source1);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else if (m_RegWorkingSet.Is32Bit(source1) || m_RegWorkingSet.Is32Bit(source2))
|
|
{
|
|
if (m_RegWorkingSet.IsUnsigned(m_RegWorkingSet.Is32Bit(source1) ? source1 : source2))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, source1);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
if (m_RegWorkingSet.Is32Bit(source2))
|
|
{
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(source2));
|
|
}
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(source2));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
int MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg))
|
|
{
|
|
if (m_RegWorkingSet.Is32Bit(MappedReg) && m_RegWorkingSet.IsUnsigned(MappedReg))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegLo(ConstReg) == 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, 0);
|
|
}
|
|
else
|
|
{
|
|
uint32_t Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, MappedReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int64_t Value = m_RegWorkingSet.GetMipsReg(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, MappedReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint32_t)(Value & 0xFFFFFFFF));
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.Is64Bit(MappedReg))
|
|
{
|
|
uint32_t Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
if (Value != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(ConstReg), MappedReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(ConstReg), 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
bool Sign = false;
|
|
|
|
if (m_RegWorkingSet.IsSigned(ConstReg) && m_RegWorkingSet.IsSigned(MappedReg))
|
|
{
|
|
Sign = true;
|
|
}
|
|
|
|
if (Value != 0)
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, Sign, MappedReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
uint32_t KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
uint64_t Value = m_RegWorkingSet.GetMipsReg(KnownReg);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, UnknownReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint32_t)(Value & 0xFFFFFFFF));
|
|
}
|
|
else
|
|
{
|
|
uint32_t Value = m_RegWorkingSet.GetMipsRegLo(KnownReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(KnownReg), UnknownReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
if (KnownReg == m_Opcode.rd)
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg) || !b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, KnownReg);
|
|
m_Assembler.AndVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
m_Assembler.AndVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(KnownReg), KnownReg);
|
|
m_Assembler.AndVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, UnknownReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(KnownReg));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(KnownReg), UnknownReg);
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(KnownReg));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.AndVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs]);
|
|
}
|
|
m_Assembler.AndVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_OR()
|
|
{
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd,
|
|
(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt)) |
|
|
(m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs)));
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) | m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
if (m_RegWorkingSet.Is64Bit(source2))
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(source2));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, source1);
|
|
}
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
uint64_t Value;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg))
|
|
{
|
|
Value = m_RegWorkingSet.GetMipsReg(ConstReg);
|
|
}
|
|
else
|
|
{
|
|
Value = m_RegWorkingSet.IsSigned(ConstReg) ? (int64_t)m_RegWorkingSet.GetMipsRegLo_S(ConstReg) : m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, MappedReg);
|
|
if ((Value >> 32) != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
}
|
|
uint32_t dwValue = (uint32_t)(Value & 0xFFFFFFFF);
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, MappedReg);
|
|
if (Value != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
int KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
int UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
uint64_t Value = m_RegWorkingSet.Is64Bit(KnownReg) ? m_RegWorkingSet.GetMipsReg(KnownReg) : m_RegWorkingSet.GetMipsRegLo_S(KnownReg);
|
|
uint32_t dwValue = (uint32_t)(Value & 0xFFFFFFFF);
|
|
|
|
if (b32BitCore() && m_RegWorkingSet.Is32Bit(KnownReg))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, UnknownReg);
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, UnknownReg);
|
|
if ((Value >> 32) != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
}
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, KnownReg);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, KnownReg);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs]);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
}
|
|
if (g_System->bFastSP() && m_Opcode.rd == 29)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
ResetMemoryStack();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_XOR()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_Opcode.rt == m_Opcode.rs)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 0);
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
if (HaveDebugger())
|
|
{
|
|
g_Notify->DisplayError("XOR 1");
|
|
}
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) ^ m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs));
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(source1);
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
if (m_RegWorkingSet.Is64Bit(source2))
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(source2));
|
|
}
|
|
else if (m_RegWorkingSet.IsSigned(source2))
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false));
|
|
}
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsSigned(m_Opcode.rt) != m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, source1);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(m_Opcode.rt), source1);
|
|
}
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
uint32_t ConstHi, ConstLo;
|
|
|
|
ConstHi = m_RegWorkingSet.Is32Bit(ConstReg) ? (uint32_t)(m_RegWorkingSet.GetMipsRegLo_S(ConstReg) >> 31) : m_RegWorkingSet.GetMipsRegHi(ConstReg);
|
|
ConstLo = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, MappedReg);
|
|
if (ConstHi != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), ConstHi);
|
|
}
|
|
if (ConstLo != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), ConstLo);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
if (m_RegWorkingSet.IsSigned(m_Opcode.rt) != m_RegWorkingSet.IsSigned(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, MappedReg);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, m_RegWorkingSet.IsSigned(MappedReg), MappedReg);
|
|
}
|
|
if (Value != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
int KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
int UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
uint64_t Value;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
Value = m_RegWorkingSet.GetMipsReg(KnownReg);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, UnknownReg);
|
|
if ((Value >> 32) != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, UnknownReg);
|
|
if (m_RegWorkingSet.IsSigned(KnownReg))
|
|
{
|
|
Value = (int)m_RegWorkingSet.GetMipsRegLo(KnownReg);
|
|
}
|
|
else
|
|
{
|
|
Value = m_RegWorkingSet.GetMipsRegLo(KnownReg);
|
|
}
|
|
}
|
|
uint32_t dwValue = (uint32_t)(Value & 0xFFFFFFFF);
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, KnownReg);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, KnownReg);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
}
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs]);
|
|
m_Assembler.XorVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_NOR()
|
|
{
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd,
|
|
~((m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt)) |
|
|
(m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs))));
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, ~(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) | m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs)));
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
if (m_RegWorkingSet.Is64Bit(source2))
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(source2));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, source1);
|
|
}
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
uint64_t Value;
|
|
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg))
|
|
{
|
|
Value = m_RegWorkingSet.GetMipsReg(ConstReg);
|
|
}
|
|
else
|
|
{
|
|
Value = m_RegWorkingSet.IsSigned(ConstReg) ? (int64_t)m_RegWorkingSet.GetMipsRegLo_S(ConstReg) : m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, MappedReg);
|
|
if ((Value >> 32) != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
}
|
|
uint32_t dwValue = (uint32_t)(Value & 0xFFFFFFFF);
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int Value = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, MappedReg);
|
|
if (Value != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), Value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
int KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
int UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
uint64_t Value = m_RegWorkingSet.Is64Bit(KnownReg) ? m_RegWorkingSet.GetMipsReg(KnownReg) : m_RegWorkingSet.GetMipsRegLo_S(KnownReg);
|
|
uint32_t dwValue = (uint32_t)(Value & 0xFFFFFFFF);
|
|
|
|
if (b32BitCore() && m_RegWorkingSet.Is32Bit(KnownReg))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, UnknownReg);
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, UnknownReg);
|
|
if ((Value >> 32) != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint32_t)(Value >> 32));
|
|
}
|
|
if (dwValue != 0)
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), dwValue);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, KnownReg);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, KnownReg);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, m_Opcode.rt);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[1], CRegName::GPR_Hi[m_Opcode.rs]);
|
|
m_Assembler.OrVariableToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rs].W[0], CRegName::GPR_Lo[m_Opcode.rs]);
|
|
}
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rd))
|
|
{
|
|
m_Assembler.not_(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd));
|
|
}
|
|
m_Assembler.not_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SLT()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
g_Notify->DisplayError("1");
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) < m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 1);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 0);
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
if ((m_RegWorkingSet.Is64Bit(m_Opcode.rt) && m_RegWorkingSet.Is64Bit(m_Opcode.rs)) ||
|
|
(!b32BitCore() && (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))))
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
m_Assembler.cmp(
|
|
m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false),
|
|
m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::ebx)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), 1);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rs) ? m_Opcode.rs : m_Opcode.rt;
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rs) ? m_Opcode.rt : m_Opcode.rs;
|
|
|
|
m_RegWorkingSet.ProtectGPR(MappedReg);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
m_Assembler.CompConstToX86reg(
|
|
m_RegWorkingSet.Is64Bit(MappedReg) ? m_RegWorkingSet.GetMipsRegMapHi(MappedReg) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, MappedReg, true, false),
|
|
m_RegWorkingSet.Is64Bit(ConstReg) ? m_RegWorkingSet.GetMipsRegHi(ConstReg) : (m_RegWorkingSet.GetMipsRegLo_S(ConstReg) >> 31));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetgVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), m_RegWorkingSet.GetMipsRegLo(ConstReg));
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
uint32_t Constant = m_RegWorkingSet.GetMipsRegLo(ConstReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), Constant);
|
|
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::ebx)
|
|
{
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetgVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
uint32_t KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
asmjit::Label Jump[2];
|
|
|
|
if (!b32BitCore())
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], m_RegWorkingSet.GetMipsRegHi(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapHi(KnownReg), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], (m_RegWorkingSet.GetMipsRegLo_S(KnownReg) >> 31));
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, KnownReg, true, false), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
}
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
if (KnownReg == (m_RegWorkingSet.IsConst(KnownReg) ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.SetgVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], m_RegWorkingSet.GetMipsRegLo(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (KnownReg == (m_RegWorkingSet.IsConst(KnownReg) ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(KnownReg))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
}
|
|
bool bConstant = m_RegWorkingSet.IsConst(KnownReg);
|
|
uint32_t Value = m_RegWorkingSet.IsConst(KnownReg) ? m_RegWorkingSet.GetMipsRegLo(KnownReg) : 0;
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
if (bConstant)
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], Value);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::ebx)
|
|
{
|
|
if (KnownReg == (bConstant ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.SetgVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
if (KnownReg == (bConstant ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.setg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), 1);
|
|
}
|
|
}
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, -1);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::eax && m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd) != asmjit::x86::ebx)
|
|
{
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.setl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
m_Assembler.and_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), 1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetlVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(Reg, m_Opcode.rs, false, false), &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
if (Jump[1].isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_SLTU()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsKnown(m_Opcode.rt) && m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
g_Notify->DisplayError("1");
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
if (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) < m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 1);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 0);
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt) && m_RegWorkingSet.IsMapped(m_Opcode.rs))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rs);
|
|
if ((m_RegWorkingSet.Is64Bit(m_Opcode.rt) && m_RegWorkingSet.Is64Bit(m_Opcode.rs)) ||
|
|
(!b32BitCore() && (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))))
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
m_Assembler.cmp(
|
|
m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rs) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false),
|
|
m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false));
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.cmp(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rs), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt) || m_RegWorkingSet.Is64Bit(m_Opcode.rs))
|
|
{
|
|
uint32_t ConstHi, ConstLo, ConstReg, MappedReg;
|
|
asmjit::x86::Gp MappedRegHi, MappedRegLo;
|
|
asmjit::Label Jump[2];
|
|
|
|
ConstReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
ConstLo = m_RegWorkingSet.GetMipsRegLo_S(ConstReg);
|
|
ConstHi = m_RegWorkingSet.GetMipsRegLo_S(ConstReg) >> 31;
|
|
if (m_RegWorkingSet.Is64Bit(ConstReg))
|
|
{
|
|
ConstHi = m_RegWorkingSet.GetMipsRegHi(ConstReg);
|
|
}
|
|
|
|
m_RegWorkingSet.ProtectGPR(MappedReg);
|
|
MappedRegLo = m_RegWorkingSet.GetMipsRegMapLo(MappedReg);
|
|
MappedRegHi = m_RegWorkingSet.GetMipsRegMapHi(MappedReg);
|
|
if (m_RegWorkingSet.Is32Bit(MappedReg))
|
|
{
|
|
MappedRegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, MappedReg, true, false);
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.CompConstToX86reg(MappedRegHi, ConstHi);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompConstToX86reg(MappedRegLo, ConstLo);
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
uint32_t Const = m_RegWorkingSet.IsConst(m_Opcode.rs) ? m_RegWorkingSet.GetMipsRegLo(m_Opcode.rs) : m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt);
|
|
uint32_t MappedReg = m_RegWorkingSet.IsConst(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_Assembler.CompConstToX86reg(m_RegWorkingSet.GetMipsRegMapLo(MappedReg), Const);
|
|
if (MappedReg == m_Opcode.rs)
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsKnown(m_Opcode.rt) || m_RegWorkingSet.IsKnown(m_Opcode.rs))
|
|
{
|
|
uint32_t KnownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rt : m_Opcode.rs;
|
|
uint32_t UnknownReg = m_RegWorkingSet.IsKnown(m_Opcode.rt) ? m_Opcode.rs : m_Opcode.rt;
|
|
asmjit::Label Jump[2];
|
|
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
if (b32BitCore())
|
|
{
|
|
uint32_t TestReg = m_RegWorkingSet.IsConst(KnownReg) ? m_Opcode.rs : m_Opcode.rt;
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
uint32_t Value = m_RegWorkingSet.GetMipsRegLo(KnownReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], Value);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (KnownReg == TestReg)
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], m_RegWorkingSet.GetMipsRegHi(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg], (m_RegWorkingSet.GetMipsRegLo_S(KnownReg) >> 31));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(KnownReg))
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapHi(KnownReg), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(KnownReg);
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, KnownReg, true, false), &m_Reg.m_GPR[UnknownReg].W[1], CRegName::GPR_Hi[UnknownReg]);
|
|
}
|
|
}
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
|
|
if (KnownReg == (m_RegWorkingSet.IsConst(KnownReg) ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
if (m_RegWorkingSet.IsConst(KnownReg))
|
|
{
|
|
m_Assembler.CompConstToVariable(&m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg], m_RegWorkingSet.GetMipsRegLo(KnownReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.GetMipsRegMapLo(KnownReg), &m_Reg.m_GPR[UnknownReg].W[0], CRegName::GPR_Lo[UnknownReg]);
|
|
}
|
|
if (KnownReg == (m_RegWorkingSet.IsConst(KnownReg) ? m_Opcode.rs : m_Opcode.rt))
|
|
{
|
|
m_Assembler.SetaVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
if (Jump[1].isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else if (b32BitCore())
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, false, false);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, -1);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
else
|
|
{
|
|
asmjit::Label Jump[2];
|
|
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rs, true, false);
|
|
m_Assembler.CompX86regToVariable(Reg, &m_Reg.m_GPR[m_Opcode.rt].W[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
Jump[0] = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Low Compare", Jump[0]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
Jump[1] = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("Continue", Jump[1]);
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[0]);
|
|
m_Assembler.CompX86regToVariable(m_RegWorkingSet.Map_TempReg(Reg, m_Opcode.rs, false, false), &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.SetbVariable(&m_BranchCompare, "m_BranchCompare");
|
|
if (Jump[1].isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump[1]);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_BranchCompare, "m_BranchCompare");
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DADD()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int64_t rs = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs);
|
|
int64_t rt = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
int64_t sum = rs + rt;
|
|
if ((~(rs ^ rt) & (rs ^ sum)) & 0x8000000000000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, sum);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(source1);
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
asmjit::x86::Gp RegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source1, false, false);
|
|
asmjit::x86::Gp RegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source1, true, false);
|
|
|
|
if (m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
m_Assembler.AddConstToX86Reg(RegLo, m_RegWorkingSet.GetMipsRegLo(source2));
|
|
m_Assembler.adc(RegHi, m_RegWorkingSet.GetMipsRegHi(source2));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.Is64Bit(source2) ? m_RegWorkingSet.GetMipsRegMapHi(source2) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false);
|
|
m_Assembler.add(RegLo, m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
m_Assembler.adc(RegHi, HiReg);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AddVariableToX86reg(RegLo, &m_Reg.m_GPR[source2].W[0], CRegName::GPR_Lo[source2]);
|
|
m_Assembler.AdcVariableToX86reg(RegHi, &m_Reg.m_GPR[source2].W[1], CRegName::GPR_Hi[source2]);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rd != 0)
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(source1);
|
|
m_RegWorkingSet.UnProtectGPR(source2);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), RegLo);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), RegHi);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DADDU()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int64_t ValRs = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs);
|
|
int64_t ValRt = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, ValRs + ValRt);
|
|
if ((m_RegWorkingSet.GetMipsRegHi(m_Opcode.rd) == 0) && (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rd) & 0x80000000) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if ((m_RegWorkingSet.GetMipsRegHi(m_Opcode.rd) == 0xFFFFFFFF) && (m_RegWorkingSet.GetMipsRegLo(m_Opcode.rd) & 0x80000000) != 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
if (m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
if (m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
uint32_t LoReg = m_RegWorkingSet.GetMipsRegLo(source2);
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), LoReg);
|
|
if (LoReg != 0)
|
|
{
|
|
m_Assembler.adc(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegHi(source2));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AddConstToX86Reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegHi(source2));
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.Is64Bit(source2) ? m_RegWorkingSet.GetMipsRegMapHi(source2) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false);
|
|
m_Assembler.add(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
m_Assembler.adc(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), HiReg);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AddVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[source2].W[0], CRegName::GPR_Lo[source2]);
|
|
m_Assembler.AdcVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[source2].W[1], CRegName::GPR_Hi[source2]);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSUB()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
int64_t rs = m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs);
|
|
int64_t rt = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
int64_t sub = rs - rt;
|
|
|
|
if (((rs ^ rt) & (rs ^ sub)) & 0x8000000000000000)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
else
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, sub);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int source1 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rt : m_Opcode.rs;
|
|
int source2 = m_Opcode.rd == m_Opcode.rt ? m_Opcode.rs : m_Opcode.rt;
|
|
|
|
m_RegWorkingSet.ProtectGPR(source1);
|
|
m_RegWorkingSet.ProtectGPR(source2);
|
|
asmjit::x86::Gp RegLo = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source1, false, false);
|
|
asmjit::x86::Gp RegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source1, true, false);
|
|
|
|
if (m_RegWorkingSet.IsConst(source2))
|
|
{
|
|
m_Assembler.sub(RegLo, m_RegWorkingSet.GetMipsRegLo(source2));
|
|
m_Assembler.sbb(RegHi, m_RegWorkingSet.GetMipsRegHi(source2));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(source2))
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.Is64Bit(source2) ? m_RegWorkingSet.GetMipsRegMapHi(source2) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, source2, true, false);
|
|
m_Assembler.sub(RegLo, m_RegWorkingSet.GetMipsRegMapLo(source2));
|
|
m_Assembler.sbb(RegHi, HiReg);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SubVariableFromX86reg(RegLo, &m_Reg.m_GPR[source2].W[0], CRegName::GPR_Lo[source2]);
|
|
m_Assembler.SbbVariableFromX86reg(RegHi, &m_Reg.m_GPR[source2].W[1], CRegName::GPR_Hi[source2]);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_ExceptionOverflow, false, &CX86Ops::JoLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
if (m_Opcode.rd != 0)
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(source1);
|
|
m_RegWorkingSet.UnProtectGPR(source2);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, source1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), RegLo);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), RegHi);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSUBU()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt) && m_RegWorkingSet.IsConst(m_Opcode.rs))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd,
|
|
m_RegWorkingSet.Is64Bit(m_Opcode.rs) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rs) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rs) - m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg(m_Opcode.rt)
|
|
: (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt));
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (m_Opcode.rd == m_Opcode.rt)
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
asmjit::x86::Gp LoReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rs);
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), LoReg);
|
|
m_Assembler.sbb(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), HiReg);
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rs);
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
m_Assembler.sbb(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt) : m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false);
|
|
m_Assembler.sub(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.sbb(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), HiReg);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.SubVariableFromX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt].W[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.SbbVariableFromX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt].W[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSLL()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
int64_t Value = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg_S(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, Value << m_Opcode.sa);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.shld(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRL()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
int64_t Value = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg_S(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, Value >> m_Opcode.sa);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
return;
|
|
}
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRA()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.rd))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
int64_t Value = m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsReg_S(m_Opcode.rt) : (int64_t)m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsReg_S(m_Opcode.rd, Value >> m_Opcode.sa);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, m_Opcode.rt);
|
|
m_Assembler.shrd(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSLL32()
|
|
{
|
|
if (m_Opcode.rd == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_Opcode.rt != m_Opcode.rd)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegHi(m_Opcode.rd, m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) << m_Opcode.sa);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, 0);
|
|
if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) < 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == -1)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else if (m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rd) >= 0 && m_RegWorkingSet.GetMipsRegHi_S(m_Opcode.rd) == 0)
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, -1);
|
|
if (m_Opcode.rt != m_Opcode.rd)
|
|
{
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log(" regcache: switch hi (%s) with lo (%s) for %s", CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt)), CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt)), CRegName::GPR[m_Opcode.rt]);
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsRegMapHi(m_Opcode.rt, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_RegWorkingSet.SetMipsRegMapLo(m_Opcode.rt, HiReg);
|
|
}
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rd, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.shl(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
m_Assembler.xor_(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRL32()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_Opcode.rt != m_Opcode.rd)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_64);
|
|
m_RegWorkingSet.SetMipsReg(m_Opcode.rd, (uint32_t)(m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt) >> m_Opcode.sa));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
if (m_Opcode.rt == m_Opcode.rd)
|
|
{
|
|
m_CodeBlock.Log(" regcache: switch hi (%s) with lo (%s) for %s", CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt)), CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt)), CRegName::GPR[m_Opcode.rt]);
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsRegMapHi(m_Opcode.rt, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_RegWorkingSet.SetMipsRegMapLo(m_Opcode.rt, HiReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, -1);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
}
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, false, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt].UW[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.shr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SPECIAL_DSRA32()
|
|
{
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
if (m_Opcode.rt != m_Opcode.rd)
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(m_Opcode.rd, false);
|
|
}
|
|
|
|
m_RegWorkingSet.SetMipsRegState(m_Opcode.rd, CRegInfo::STATE_CONST_32_SIGN);
|
|
m_RegWorkingSet.SetMipsRegLo(m_Opcode.rd, (uint32_t)(m_RegWorkingSet.GetMipsReg_S(m_Opcode.rt) >> (m_Opcode.sa + 32)));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.rt);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
if (m_Opcode.rt == m_Opcode.rd)
|
|
{
|
|
m_CodeBlock.Log(" regcache: switch hi (%s) with lo (%s) for %s", CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt)), CX86Ops::x86_Name(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt)), CRegName::GPR[m_Opcode.rt]);
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt);
|
|
m_RegWorkingSet.SetMipsRegMapHi(m_Opcode.rt, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_RegWorkingSet.SetMipsRegMapLo(m_Opcode.rt, HiReg);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
}
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CX86RecompilerOps::UnknownOpcode();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rd, true, -1);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), &m_Reg.m_GPR[m_Opcode.rt].UW[1], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
if ((uint8_t)m_Opcode.sa != 0)
|
|
{
|
|
m_Assembler.sar(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rd), (uint8_t)m_Opcode.sa);
|
|
}
|
|
}
|
|
}
|
|
|
|
// COP0 functions
|
|
void CX86RecompilerOps::COP0_MF()
|
|
{
|
|
if (m_Opcode.rd == CRegisters::COP0Reg_Count)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(m_Opcode.rd);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::Cop0_MF), "CRegisters::Cop0_MF", 8);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt], asmjit::x86::eax);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_DMF()
|
|
{
|
|
if (m_Opcode.rd == CRegisters::COP0Reg_Count)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(m_Opcode.rd);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::Cop0_MF), "CRegisters::Cop0_MF", 8);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt], asmjit::x86::eax);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[m_Opcode.rt].UW[1], CRegName::GPR_Hi[m_Opcode.rt], asmjit::x86::edx);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), &m_Reg.m_GPR[m_Opcode.rt].UW[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_MT()
|
|
{
|
|
if (m_Opcode.rd == 6 || m_Opcode.rd == 11)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> 31);
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::edx ? asmjit::x86::edx : asmjit::x86::eax;
|
|
m_Assembler.mov(HiReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.sar(HiReg, 0x1F);
|
|
m_Assembler.push(HiReg);
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::eax, &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.mov(asmjit::x86::edx, asmjit::x86::eax);
|
|
m_Assembler.sar(asmjit::x86::edx, 0x1F);
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
}
|
|
m_Assembler.push(m_Opcode.rd);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::Cop0_MT), "CRegisters::Cop0_MT", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_DMT()
|
|
{
|
|
if (m_Opcode.rd == 6)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.push(m_RegWorkingSet.Is64Bit(m_Opcode.rt) ? m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt) : m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> 31);
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp HiReg = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt) != asmjit::x86::edx ? asmjit::x86::edx : asmjit::x86::eax;
|
|
m_Assembler.mov(HiReg, m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.sar(HiReg, 0x1F);
|
|
m_Assembler.push(HiReg);
|
|
}
|
|
m_Assembler.push(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::eax, &m_Reg.m_GPR[m_Opcode.rt].UW[0], CRegName::GPR_Lo[m_Opcode.rt]);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &m_Reg.m_GPR[m_Opcode.rt].UW[1], CRegName::GPR_Hi[m_Opcode.rt]);
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
}
|
|
m_Assembler.push(m_Opcode.rd);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::Cop0_MT), "CRegisters::Cop0_MT", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
// COP0 CO functions
|
|
void CX86RecompilerOps::COP0_CO_TLBR(void)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::ReadEntry), "CTLB::ReadEntry", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_CO_TLBWI(void)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("false", 0);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::ecx, &g_Reg->INDEX_REGISTER, "INDEX_REGISTER");
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.push(asmjit::x86::ecx);
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::WriteEntry), "CTLB::WriteEntry", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_CO_TLBWR(void)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_SystemTimer, AddressOf(&CSystemTimer::UpdateTimers), "CSystemTimer::UpdateTimers", 4);
|
|
m_Assembler.PushImm32("true", true);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::ecx, &g_Reg->RANDOM_REGISTER, "RANDOM_REGISTER");
|
|
m_Assembler.and_(asmjit::x86::ecx, 0x1F);
|
|
m_Assembler.push(asmjit::x86::ecx);
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::WriteEntry), "CTLB::WriteEntry", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_CO_TLBP(void)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::Probe), "CTLB::TLB_Probe", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void x86_compiler_COP0_CO_ERET()
|
|
{
|
|
if (g_Reg->STATUS_REGISTER.ErrorLevel != 0)
|
|
{
|
|
g_Reg->m_PROGRAM_COUNTER = g_Reg->ERROREPC_REGISTER;
|
|
g_Reg->STATUS_REGISTER.ErrorLevel = 0;
|
|
}
|
|
else
|
|
{
|
|
g_Reg->m_PROGRAM_COUNTER = g_Reg->EPC_REGISTER;
|
|
g_Reg->STATUS_REGISTER.ExceptionLevel = 0;
|
|
}
|
|
g_Reg->m_LLBit = 0;
|
|
g_Reg->CheckInterrupts();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP0_CO_ERET(void)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
m_Assembler.CallFunc((uint32_t)x86_compiler_COP0_CO_ERET, "x86_compiler_COP0_CO_ERET");
|
|
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
CompileExit(m_CompilePC, (uint64_t)-1, m_RegWorkingSet, ExitReason_Normal, true, nullptr);
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
// FPU options
|
|
void CX86RecompilerOps::ChangeDefaultRoundingModel()
|
|
{
|
|
switch ((g_Reg->m_FPCR[31] & 3))
|
|
{
|
|
case 0: m_RoundingModeValue = 0x0000; break;
|
|
case 1: m_RoundingModeValue = 0x0C00; break;
|
|
case 2: m_RoundingModeValue = 0x0800; break;
|
|
case 3: m_RoundingModeValue = 0x0400; break;
|
|
}
|
|
}
|
|
|
|
// COP1 functions
|
|
void CX86RecompilerOps::COP1_MF()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs, true);
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg, (uint8_t *)&m_Reg.m_FPR_S[m_Opcode.fs], stdstr_f("m_FPR_S[%d]", m_Opcode.fs).c_str());
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), asmjit::x86::dword_ptr(TempReg));
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_DMF()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs, true);
|
|
m_RegWorkingSet.Map_GPR_64bit(m_Opcode.rt, -1);
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg, (uint8_t *)&m_Reg.m_FPR_D[m_Opcode.fs], stdstr_f("_FPR_D[%d]", m_Opcode.fs).c_str());
|
|
m_Assembler.AddConstToX86Reg(TempReg, 4);
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt), asmjit::x86::dword_ptr(TempReg));
|
|
m_Assembler.MoveVariableToX86reg(TempReg, (uint8_t *)&m_Reg.m_FPR_D[m_Opcode.fs], stdstr_f("_FPR_D[%d]", m_Opcode.fs).c_str());
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), asmjit::x86::dword_ptr(TempReg));
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_CF()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if (m_Opcode.fs != 31 && m_Opcode.fs != 0)
|
|
{
|
|
UnknownOpcode();
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
if (m_Opcode.fs == 31 && m_RegWorkingSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.mov(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), m_RegWorkingSet.Map_FPStatusReg());
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt), &m_Reg.m_FPCR[m_Opcode.fs], CRegName::FPR_Ctrl[m_Opcode.fs]);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_MT()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if ((m_Opcode.fs & 1) != 0)
|
|
{
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs - 1, CRegInfo::FPU_Double) || m_RegWorkingSet.RegInStack(m_Opcode.fs - 1, CRegInfo::FPU_Qword))
|
|
{
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs - 1, true);
|
|
}
|
|
}
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs, true);
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg, (uint8_t *)&m_Reg.m_FPR_S[m_Opcode.fs], stdstr_f("m_FPR_S[%d]", m_Opcode.fs).c_str());
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_DMT()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if ((m_Opcode.fs & 1) == 0)
|
|
{
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs + 1, CRegInfo::FPU_Float) || m_RegWorkingSet.RegInStack(m_Opcode.fs + 1, CRegInfo::FPU_Dword))
|
|
{
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs + 1, true);
|
|
}
|
|
}
|
|
m_RegWorkingSet.UnMap_FPR(m_Opcode.fs, true);
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempReg, (uint8_t *)&m_Reg.m_FPR_D[m_Opcode.fs], stdstr_f("_FPR_D[%d]", m_Opcode.fs).c_str());
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt));
|
|
m_Assembler.AddConstToX86Reg(TempReg, 4);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegHi(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegLo_S(m_Opcode.rt) >> 31);
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt));
|
|
m_Assembler.AddConstToX86Reg(TempReg, 4);
|
|
if (m_RegWorkingSet.Is64Bit(m_Opcode.rt))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.rt));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, true, false));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), Reg);
|
|
m_Assembler.AddConstToX86Reg(TempReg, 4);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempReg), m_RegWorkingSet.Map_TempReg(Reg, m_Opcode.rt, true, false));
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_CT()
|
|
{
|
|
CompileCop1Test();
|
|
|
|
if (m_Opcode.fs != 31)
|
|
{
|
|
UnknownOpcode();
|
|
return;
|
|
}
|
|
|
|
m_RegWorkingSet.UnMap_FPStatusReg();
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.rt))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_FPCR[m_Opcode.fs], CRegName::FPR_Ctrl[m_Opcode.fs], m_RegWorkingSet.GetMipsRegLo(m_Opcode.rt) & 0x183FFFF);
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.rt, false, false);
|
|
m_Assembler.and_(TempReg, 0x183FFFF);
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_FPCR[m_Opcode.fs], CRegName::FPR_Ctrl[m_Opcode.fs], TempReg);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)ChangeDefaultRoundingModel, "ChangeDefaultRoundingModel");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_RegWorkingSet.SetRoundingModel(CRegInfo::RoundUnknown);
|
|
}
|
|
|
|
// COP1: S functions
|
|
void CX86RecompilerOps::COP1_S_ADD()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fadd);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_SUB()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fsub);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_MUL()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fmul);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_DIV()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fdiv);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_ABS()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fabs);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_NEG()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fchs);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_SQRT()
|
|
{
|
|
COP1_S_Opcode(&CX86Ops::Fsqrt);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_MOV()
|
|
{
|
|
CompileCop1Test();
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) || m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempRegValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp FsPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Double);
|
|
asmjit::x86::Gp FdPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fd, CRegInfo::FPU_UnsignedDoubleWord);
|
|
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr));
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FdPtr), TempRegValue);
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr, 4));
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FdPtr, 4), TempRegValue);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_ROUND_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundNearest, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_TRUNC_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundTruncate, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CEIL_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundUp, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_FLOOR_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDown, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_ROUND_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundNearest, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_TRUNC_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundTruncate, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CEIL_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundUp, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_FLOOR_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDown, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CVT_D()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Double);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CVT_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CVT_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_FloatLow, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CMP()
|
|
{
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, 0, false, false);
|
|
CompileInitFpuOperation(CRegBase::RoundUnknown);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) || m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.and_(m_RegWorkingSet.Map_FPStatusReg(), (uint32_t)~FPCSR_C);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AndConstToVariable(&m_Reg.m_FPCR[31], "_FPCR[31]", (uint32_t)~FPCSR_C);
|
|
}
|
|
asmjit::x86::Gp StatusReg = m_RegWorkingSet.Map_FPStatusReg();
|
|
asmjit::x86::Gp TempRegValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, true);
|
|
asmjit::x86::Gp FsPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_FloatLow);
|
|
asmjit::x86::Gp FtPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.ft, CRegInfo::FPU_Dword);
|
|
|
|
m_Assembler.fpuLoadDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), FtPtr);
|
|
m_Assembler.fpuLoadDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), FsPtr);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fstsw(asmjit::x86::ax);
|
|
m_Assembler.sahf();
|
|
asmjit::Label NotNanLabel = m_Assembler.newLabel();
|
|
m_Assembler.JnpLabel("NotNan", NotNanLabel);
|
|
|
|
//is Nan
|
|
asmjit::Label NotQuietNanLabel;
|
|
if ((m_Opcode.funct & 8) == 0)
|
|
{
|
|
//check QuietNan
|
|
NotQuietNanLabel = m_Assembler.newLabel();
|
|
asmjit::Label FsIsQuietNan = m_Assembler.newLabel();
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr));
|
|
m_Assembler.cmp(TempRegValue, 0x7FC00000);
|
|
m_Assembler.JgeLabel("FsIsQuietNan", FsIsQuietNan);
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FtPtr));
|
|
m_Assembler.cmp(TempRegValue, 0x7FC00000);
|
|
m_Assembler.JlLabel("FsIsQuietNan", NotQuietNanLabel);
|
|
m_Assembler.bind(FsIsQuietNan);
|
|
}
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_CV);
|
|
m_Assembler.test(StatusReg, FPCSR_EV);
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_ExceptionFloatingPoint, false, &CX86Ops::JnzLabel);
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_FV);
|
|
if ((m_Opcode.funct & 8) == 0)
|
|
{
|
|
m_Assembler.bind(NotQuietNanLabel);
|
|
}
|
|
if ((m_Opcode.funct & 1) != 0)
|
|
{
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_C);
|
|
}
|
|
uint32_t cmp = 0;
|
|
if ((m_Opcode.funct & 2) != 0)
|
|
{
|
|
cmp |= 0x4000;
|
|
}
|
|
if ((m_Opcode.funct & 4) != 0)
|
|
{
|
|
cmp |= 0x0100;
|
|
}
|
|
asmjit::Label IsNanSkipCompare = m_Assembler.newLabel();
|
|
if (cmp != 0)
|
|
{
|
|
m_Assembler.JmpLabel("IsNanSkipCompare", IsNanSkipCompare);
|
|
}
|
|
m_Assembler.bind(NotNanLabel);
|
|
if (cmp != 0)
|
|
{
|
|
m_Assembler.xor_(TempRegValue, TempRegValue);
|
|
m_Assembler.test(asmjit::x86::eax, cmp);
|
|
m_Assembler.setnz(TempRegValue);
|
|
m_Assembler.shl(TempRegValue, 23);
|
|
m_Assembler.or_(StatusReg, TempRegValue);
|
|
m_Assembler.bind(IsNanSkipCompare);
|
|
}
|
|
}
|
|
|
|
// COP1: D functions
|
|
void CX86RecompilerOps::COP1_D_ADD()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fadd);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_SUB()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fsub);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_MUL()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fmul);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_DIV()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fdiv);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_ABS()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fabs);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_NEG()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fchs);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_SQRT()
|
|
{
|
|
COP1_D_Opcode(&CX86Ops::Fsqrt);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_MOV()
|
|
{
|
|
CompileCop1Test();
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) || m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempRegValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
asmjit::x86::Gp FsPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Double);
|
|
asmjit::x86::Gp FdPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fd, CRegInfo::FPU_UnsignedDoubleWord);
|
|
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr));
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FdPtr), TempRegValue);
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr, 4));
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(FdPtr, 4), TempRegValue);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_ROUND_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundNearest, CRegInfo::FPU_Double, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_TRUNC_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundTruncate, CRegInfo::FPU_Double, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CEIL_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundUp, CRegInfo::FPU_Double, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_FLOOR_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDown, CRegInfo::FPU_Double, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_ROUND_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundNearest, CRegInfo::FPU_Double, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_TRUNC_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundTruncate, CRegInfo::FPU_Double, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CEIL_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundUp, CRegInfo::FPU_Double, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_FLOOR_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDown, CRegInfo::FPU_Double, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CVT_S()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Double, CRegInfo::FPU_Float);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CVT_W()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Double, CRegInfo::FPU_Dword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CVT_L()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Double, CRegInfo::FPU_Qword);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_CMP()
|
|
{
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, 0, false, false);
|
|
CompileInitFpuOperation(CRegBase::RoundUnknown);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) || m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
if (m_RegWorkingSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.and_(m_RegWorkingSet.Map_FPStatusReg(), (uint32_t)~FPCSR_C);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.AndConstToVariable(&m_Reg.m_FPCR[31], "_FPCR[31]", (uint32_t)~FPCSR_C);
|
|
}
|
|
asmjit::x86::Gp StatusReg = m_RegWorkingSet.Map_FPStatusReg();
|
|
asmjit::x86::Gp TempRegValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, true);
|
|
asmjit::x86::Gp FsPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Double);
|
|
asmjit::x86::Gp FtPtr = m_RegWorkingSet.FPRValuePointer(m_Opcode.ft, CRegInfo::FPU_UnsignedDoubleWord);
|
|
|
|
m_Assembler.fpuLoadQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), FtPtr);
|
|
m_Assembler.fpuLoadQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), FsPtr);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fstsw(asmjit::x86::ax);
|
|
m_Assembler.sahf();
|
|
asmjit::Label NotNanLabel = m_Assembler.newLabel();
|
|
m_Assembler.JnpLabel("NotNan", NotNanLabel);
|
|
|
|
//is Nan
|
|
asmjit::Label NotQuietNanLabel;
|
|
if ((m_Opcode.funct & 8) == 0)
|
|
{
|
|
//check QuietNan
|
|
NotQuietNanLabel = m_Assembler.newLabel();
|
|
asmjit::Label FsIsQuietNan = m_Assembler.newLabel();
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FsPtr));
|
|
m_Assembler.cmp(TempRegValue, 0x7FC00000);
|
|
m_Assembler.JgeLabel("FsIsQuietNan", FsIsQuietNan);
|
|
m_Assembler.mov(TempRegValue, asmjit::x86::dword_ptr(FtPtr));
|
|
m_Assembler.cmp(TempRegValue, 0x7FC00000);
|
|
m_Assembler.JlLabel("FsIsQuietNan", NotQuietNanLabel);
|
|
m_Assembler.bind(FsIsQuietNan);
|
|
}
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_CV);
|
|
m_Assembler.test(StatusReg, FPCSR_EV);
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_ExceptionFloatingPoint, false, &CX86Ops::JnzLabel);
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_FV);
|
|
if ((m_Opcode.funct & 8) == 0)
|
|
{
|
|
m_Assembler.bind(NotQuietNanLabel);
|
|
}
|
|
if ((m_Opcode.funct & 1) != 0)
|
|
{
|
|
m_Assembler.or_(StatusReg, (uint32_t)FPCSR_C);
|
|
}
|
|
uint32_t cmp = 0;
|
|
if ((m_Opcode.funct & 2) != 0)
|
|
{
|
|
cmp |= 0x4000;
|
|
}
|
|
if ((m_Opcode.funct & 4) != 0)
|
|
{
|
|
cmp |= 0x0100;
|
|
}
|
|
asmjit::Label IsNanSkipCompare = m_Assembler.newLabel();
|
|
if (cmp != 0)
|
|
{
|
|
m_Assembler.JmpLabel("IsNanSkipCompare", IsNanSkipCompare);
|
|
}
|
|
m_Assembler.bind(NotNanLabel);
|
|
if (cmp != 0)
|
|
{
|
|
m_Assembler.xor_(TempRegValue, TempRegValue);
|
|
m_Assembler.test(asmjit::x86::eax, cmp);
|
|
m_Assembler.setnz(TempRegValue);
|
|
m_Assembler.shl(TempRegValue, 23);
|
|
m_Assembler.or_(StatusReg, TempRegValue);
|
|
m_Assembler.bind(IsNanSkipCompare);
|
|
}
|
|
}
|
|
|
|
// COP1: W functions
|
|
void CX86RecompilerOps::COP1_W_CVT_S()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Dword, CRegInfo::FPU_Float);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_W_CVT_D()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Dword, CRegInfo::FPU_Double);
|
|
}
|
|
|
|
// COP1: L functions
|
|
void CX86RecompilerOps::COP1_L_CVT_S()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Qword, CRegInfo::FPU_Float);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_L_CVT_D()
|
|
{
|
|
COP1_S_CVT(CRegInfo::RoundDefault, CRegInfo::FPU_Qword, CRegInfo::FPU_Double);
|
|
}
|
|
|
|
// Other functions
|
|
void CX86RecompilerOps::UnknownOpcode()
|
|
{
|
|
m_CodeBlock.Log(" %X Unhandled opcode: %s", m_CompilePC, R4300iInstruction(m_CompilePC, m_Opcode.Value).NameAndParam().c_str());
|
|
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_Assembler.MoveConst64ToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
|
|
m_Assembler.MoveConstToVariable(&g_System->m_OpCodes.m_Opcode.Value, "R4300iOp::m_Opcode.Value", m_Opcode.Value);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::UnknownOpcode), "R4300iOp::UnknownOpcode", 4);
|
|
m_Assembler.ret();
|
|
if (m_PipelineStage == PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::RecordLLAddress(const asmjit::x86::Gp & AddressReg)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LLBit, "LLBit", 1);
|
|
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempMemoryUsed", (uint32_t)&m_TempMemoryUsed);
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.mov(TempReg, AddressReg);
|
|
m_Assembler.sar(TempReg, 31);
|
|
m_Assembler.push(TempReg);
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::VAddrToPAddr), "CTLB::VAddrToPAddr", 20);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_TempValue32, "m_TempValue32");
|
|
m_Assembler.shr(TempReg, 4);
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_CP0[17], "m_CP0[17]", TempReg);
|
|
m_Assembler.MoveConstToVariable(((uint8_t *)&g_Reg->m_CP0[17]) + 4, "m_CP0[17] + 4", 0);
|
|
}
|
|
|
|
void CX86RecompilerOps::RecordLLAddress(uint64_t Address)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_LLBit, "LLBit", 1);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempMemoryUsed", (uint32_t)&m_TempMemoryUsed);
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push((uint32_t)(Address >> 32));
|
|
m_Assembler.push((uint32_t)(Address & 0xFFFFFFFF));
|
|
m_Assembler.CallThis((uint32_t)&m_TLB, AddressOf(&CTLB::VAddrToPAddr), "CTLB::VAddrToPAddr", 20);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
asmjit::x86::Gp Reg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_TempValue32, "m_TempValue32");
|
|
m_Assembler.shr(Reg, 4);
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_CP0[17], "m_CP0[17]", Reg);
|
|
m_Assembler.MoveConstToVariable(((uint8_t *)&g_Reg->m_CP0[17]) + 4, "m_CP0[17] + 4", 0);
|
|
}
|
|
|
|
void CX86RecompilerOps::ClearCachedInstructionInfo()
|
|
{
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
m_Assembler.MoveConst64ToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::FoundMemoryBreakpoint()
|
|
{
|
|
ClearCachedInstructionInfo();
|
|
m_Assembler.MoveConstToVariable(&memory_write_in_delayslot, "memory_write_in_delayslot", (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT) ? 1 : 0);
|
|
m_Assembler.CallFunc((uint32_t)x86MemoryBreakPoint, "x86MemoryBreakPoint");
|
|
m_Assembler.MoveConstToVariable(&memory_breakpoint_found, "memory_breakpoint_found", 0);
|
|
ExitCodeBlock();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
void CX86RecompilerOps::PreReadInstruction()
|
|
{
|
|
if (!HaveReadBP())
|
|
{
|
|
return;
|
|
}
|
|
ClearCachedInstructionInfo();
|
|
}
|
|
|
|
void CX86RecompilerOps::PreWriteInstruction()
|
|
{
|
|
if (!HaveWriteBP())
|
|
{
|
|
return;
|
|
}
|
|
ClearCachedInstructionInfo();
|
|
}
|
|
|
|
void CX86RecompilerOps::TestBreakpoint(const asmjit::x86::Gp & AddressReg, uint32_t FunctAddress, const char * FunctName)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.MoveX86regToVariable(&memory_access_address, "memory_access_address", AddressReg);
|
|
m_Assembler.MoveConstToVariable(&memory_write_in_delayslot, "memory_write_in_delayslot", (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT) ? 1 : 0);
|
|
m_Assembler.CallFunc(FunctAddress, FunctName);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_Assembler.CompConstToVariable(&memory_breakpoint_found, "memory_breakpoint_found", 0);
|
|
asmjit::Label Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("NoBreakPoint", Jump);
|
|
m_Assembler.MoveConstToVariable(&memory_breakpoint_found, "memory_breakpoint_found", 0);
|
|
ExitCodeBlock();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
|
|
void CX86RecompilerOps::TestWriteBreakpoint(const asmjit::x86::Gp & AddressReg, uint32_t FunctAddress, const char * FunctName)
|
|
{
|
|
if (!HaveWriteBP())
|
|
{
|
|
return;
|
|
}
|
|
TestBreakpoint(AddressReg, FunctAddress, FunctName);
|
|
}
|
|
|
|
void CX86RecompilerOps::TestReadBreakpoint(const asmjit::x86::Gp & AddressReg, uint32_t FunctAddress, const char * FunctName)
|
|
{
|
|
if (!HaveReadBP())
|
|
{
|
|
return;
|
|
}
|
|
TestBreakpoint(AddressReg, FunctAddress, FunctName);
|
|
}
|
|
|
|
void CX86RecompilerOps::EnterCodeBlock()
|
|
{
|
|
#ifdef _DEBUG
|
|
m_Assembler.push(asmjit::x86::esi);
|
|
#else
|
|
m_Assembler.push(asmjit::x86::edi);
|
|
m_Assembler.push(asmjit::x86::esi);
|
|
m_Assembler.push(asmjit::x86::ebx);
|
|
#endif
|
|
}
|
|
|
|
void CX86RecompilerOps::ExitCodeBlock()
|
|
{
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
#ifdef _DEBUG
|
|
m_Assembler.pop(asmjit::x86::esi);
|
|
#else
|
|
m_Assembler.pop(asmjit::x86::ebx);
|
|
m_Assembler.pop(asmjit::x86::esi);
|
|
m_Assembler.pop(asmjit::x86::edi);
|
|
#endif
|
|
m_Assembler.ret();
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileExitCode()
|
|
{
|
|
for (EXIT_LIST::iterator ExitIter = m_ExitInfo.begin(); ExitIter != m_ExitInfo.end(); ExitIter++)
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(ExitIter->JumpLabel);
|
|
m_PipelineStage = ExitIter->PipelineStage;
|
|
CompileExit((uint64_t)-1, ExitIter->TargetPC, ExitIter->ExitRegSet, ExitIter->Reason, true, nullptr);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileCheckFPUInput(asmjit::x86::Gp RegPointer, FpuOpSize OpSize, bool Conv)
|
|
{
|
|
m_RegWorkingSet.UnMap_FPStatusReg();
|
|
if (Conv)
|
|
{
|
|
CX86RegInfo::REG_MAPPED EaxMapping = m_RegWorkingSet.GetX86Mapped(GetIndexFromX86Reg(asmjit::x86::eax));
|
|
if (EaxMapping == CX86RegInfo::Temp_Mapped || EaxMapping == CX86RegInfo::NotMapped)
|
|
{
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, -1, false, false);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
}
|
|
asmjit::x86::Gp TempPointerValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempPointerValue), false);
|
|
asmjit::Label PossiblySubNormal = m_Assembler.newLabel();
|
|
asmjit::Label PossiblyNanJump = m_Assembler.newLabel();
|
|
asmjit::Label InvalidValueMaxJump, InvalidMinValueJump;
|
|
if (OpSize == FpuOpSize_32bit)
|
|
{
|
|
m_Assembler.mov(TempPointerValue, asmjit::x86::dword_ptr(RegPointer));
|
|
m_Assembler.and_(TempPointerValue, 0x7F800000);
|
|
m_Assembler.JzLabel("PossiblySubNormal", PossiblySubNormal);
|
|
m_Assembler.cmp(TempPointerValue, 0x7F800000);
|
|
m_Assembler.JeLabel("PossiblyNan", PossiblyNanJump);
|
|
if (Conv)
|
|
{
|
|
InvalidValueMaxJump = m_Assembler.newLabel();
|
|
InvalidMinValueJump = m_Assembler.newLabel();
|
|
|
|
static uint32_t InvalidValueMax = 0x5a000000, InvalidMinValue = 0xda000000;
|
|
m_Assembler.fpuLoadDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), RegPointer);
|
|
m_Assembler.fpuLoadDwordFromPtr(m_RegWorkingSet.StackTopPos(), (uint64_t)&InvalidValueMax);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.test(asmjit::x86::ah, 0x41);
|
|
m_Assembler.jnp(InvalidValueMaxJump);
|
|
m_Assembler.fpuLoadDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), RegPointer);
|
|
m_Assembler.fpuLoadDwordFromPtr(m_RegWorkingSet.StackTopPos(), (uint64_t)&InvalidMinValue);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.test(asmjit::x86::ah, 0x1);
|
|
m_Assembler.je(InvalidMinValueJump);
|
|
}
|
|
}
|
|
else if (OpSize == FpuOpSize_64bit)
|
|
{
|
|
m_Assembler.mov(TempPointerValue, asmjit::x86::dword_ptr(RegPointer, 4));
|
|
m_Assembler.and_(TempPointerValue, 0x7FF00000);
|
|
m_Assembler.JzLabel("PossiblySubNormal", PossiblySubNormal);
|
|
m_Assembler.cmp(TempPointerValue, 0x7FF00000);
|
|
m_Assembler.JeLabel("PossiblyNan", PossiblyNanJump);
|
|
|
|
if (Conv)
|
|
{
|
|
InvalidValueMaxJump = m_Assembler.newLabel();
|
|
InvalidMinValueJump = m_Assembler.newLabel();
|
|
|
|
static uint64_t InvalidValueMax = 0x4340000000000000, InvalidMinValue = 0xc340000000000000;
|
|
m_Assembler.fld(asmjit::x86::qword_ptr((uint64_t)&InvalidValueMax));
|
|
m_Assembler.fcomp(asmjit::x86::qword_ptr(RegPointer));
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.test(asmjit::x86::ah, 0x41);
|
|
m_Assembler.jnp(InvalidValueMaxJump);
|
|
m_Assembler.fld(asmjit::x86::qword_ptr((uint64_t)&InvalidMinValue));
|
|
m_Assembler.fcomp(asmjit::x86::qword_ptr(RegPointer));
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.test(asmjit::x86::ah, 0x1);
|
|
m_Assembler.je(InvalidMinValueJump);
|
|
}
|
|
}
|
|
asmjit::Label ValidFpuValue = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel("ValidFpuValue", ValidFpuValue);
|
|
m_Assembler.bind(PossiblySubNormal);
|
|
m_Assembler.bind(PossiblyNanJump);
|
|
if (InvalidValueMaxJump.isValid())
|
|
{
|
|
m_Assembler.bind(InvalidValueMaxJump);
|
|
}
|
|
if (InvalidMinValueJump.isValid())
|
|
{
|
|
m_Assembler.bind(InvalidMinValueJump);
|
|
}
|
|
m_Assembler.mov(TempPointerValue, asmjit::x86::dword_ptr(RegPointer));
|
|
if (OpSize == FpuOpSize_32bit)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue32, "TempValue32", TempPointerValue);
|
|
}
|
|
else if (OpSize == FpuOpSize_64bit)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue64, "TempValue64", TempPointerValue);
|
|
m_Assembler.mov(TempPointerValue, asmjit::x86::dword_ptr(RegPointer, 4));
|
|
m_Assembler.MoveX86regToVariable((uint8_t *)&m_TempValue64 + 4, "TempValue64", TempPointerValue);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (OpSize == FpuOpSize_32bit)
|
|
{
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
if (Conv)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInput32Conv), "R4300iOp::CheckFPUInput32Conv", 8);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInput32), "R4300iOp::CheckFPUInput32", 8);
|
|
}
|
|
}
|
|
else if (OpSize == FpuOpSize_64bit)
|
|
{
|
|
m_Assembler.PushImm32("&m_TempValue64", (uint32_t)&m_TempValue64);
|
|
if (Conv)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInput64Conv), "R4300iOp::CheckFPUInput64Conv", 8);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInput64), "R4300iOp::CheckFPUInput64", 8);
|
|
}
|
|
}
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_Exception, false, &CX86Ops::JnzLabel);
|
|
m_Assembler.bind(ValidFpuValue);
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileCheckFPUResult32(int32_t DestReg)
|
|
{
|
|
m_RegWorkingSet.UnMap_FPStatusReg();
|
|
m_Assembler.fst(asmjit::x86::dword_ptr((uint64_t)&m_TempValue32));
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, -1, false, false);
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.and_(asmjit::x86::ax, 0x3D);
|
|
m_Assembler.cmp(asmjit::x86::ax, 0);
|
|
asmjit::Label FpuExceptionJump = m_Assembler.newLabel();
|
|
m_Assembler.JnzLabel("FpuException", FpuExceptionJump);
|
|
m_Assembler.AndConstToVariable(&m_TempValue32, "TempValue32", 0x7F800000);
|
|
asmjit::Label PossiblySubNormal = m_Assembler.newLabel();
|
|
m_Assembler.JzLabel("PossiblySubNormal", PossiblySubNormal);
|
|
m_Assembler.CompConstToVariable(&m_TempValue32, "TempValue32", 0x7F800000);
|
|
asmjit::Label PossiblyNanJump = m_Assembler.newLabel();
|
|
asmjit::Label DoNoModify = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("PossiblyNan", PossiblyNanJump);
|
|
m_Assembler.JmpLabel("DoNotModify", DoNoModify);
|
|
m_Assembler.bind(PossiblySubNormal);
|
|
m_Assembler.bind(PossiblyNanJump);
|
|
m_Assembler.bind(FpuExceptionJump);
|
|
m_Assembler.fst(asmjit::x86::dword_ptr((uint64_t)&m_TempValue32));
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_TempValue32, "TempValue32");
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
m_Assembler.PushImm32("Result", (uint32_t)&m_TempValue32);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUResult32), "R4300iOp::CheckFPUResult32", 8);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_Exception, false, &CX86Ops::JneLabel);
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
m_Assembler.cmp(TempReg, asmjit::x86::dword_ptr((uint64_t)&m_TempValue32));
|
|
asmjit::Label ValueSame = m_Assembler.newLabel();
|
|
m_Assembler.je(ValueSame);
|
|
ExitRegSet = m_RegWorkingSet;
|
|
for (int32_t i = 0; i < x86RegFpuIndex_Size; i++)
|
|
{
|
|
if (ExitRegSet.FpuMappedTo(i) != DestReg)
|
|
{
|
|
continue;
|
|
}
|
|
m_CodeBlock.Log(" regcache: unallocate %s from ST(%d)", CRegName::FPR[DestReg], (i - ExitRegSet.StackTopPos() + 8) & 7);
|
|
ExitRegSet.FpuRoundingModel(i) = CRegBase::RoundDefault;
|
|
ExitRegSet.FpuMappedTo(i) = -1;
|
|
ExitRegSet.FpuState(i) = CX86RegInfo::FPU_Unknown;
|
|
ExitRegSet.FpuStateChanged(i) = false;
|
|
break;
|
|
}
|
|
m_Assembler.MoveVariableToX86reg(TempReg, &m_TempValue32, "TempValue32");
|
|
asmjit::x86::Gp TempRegFPR_S = ExitRegSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempRegFPR_S, &m_Reg.m_FPR_UDW[DestReg], stdstr_f("m_FPR_UDW[%d]", DestReg).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempRegFPR_S), TempReg);
|
|
m_Assembler.and_(asmjit::x86::dword_ptr(TempRegFPR_S, 4), 0);
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC + 4, m_CompilePC + 4, ExitRegSet, ExitReason_Normal, false, &CX86Ops::JmpLabel);
|
|
|
|
m_Assembler.bind(ValueSame);
|
|
m_Assembler.bind(DoNoModify);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileCheckFPUResult64(asmjit::x86::Gp RegPointer)
|
|
{
|
|
m_RegWorkingSet.UnMap_FPStatusReg();
|
|
bool RegPointerProtect = m_RegWorkingSet.GetX86Protected(GetIndexFromX86Reg(RegPointer));
|
|
if (!RegPointerProtect)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(RegPointer), true);
|
|
}
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, -1, false, false);
|
|
if (RegPointer == TempReg)
|
|
{
|
|
asmjit::x86::Gp RegPointerValue = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(RegPointerValue, RegPointer);
|
|
RegPointer = RegPointerValue;
|
|
}
|
|
|
|
asmjit::x86::Gp TempReg2 = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.fnstsw(asmjit::x86::ax);
|
|
m_Assembler.and_(asmjit::x86::ax, 0x3D);
|
|
m_Assembler.cmp(asmjit::x86::ax, 0);
|
|
asmjit::Label FpuExceptionJump = m_Assembler.newLabel();
|
|
m_Assembler.JnzLabel("FpuException", FpuExceptionJump);
|
|
m_Assembler.mov(TempReg2, asmjit::x86::dword_ptr(RegPointer, 4));
|
|
m_Assembler.and_(TempReg2, 0x7FF00000);
|
|
asmjit::Label isSubNormal = m_Assembler.newLabel();
|
|
m_Assembler.JzLabel("isSubNormal", isSubNormal);
|
|
m_Assembler.cmp(TempReg2, 0x7FF00000);
|
|
asmjit::Label DoNoModify = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel("DoNotModify", DoNoModify);
|
|
|
|
m_Assembler.bind(isSubNormal);
|
|
m_Assembler.bind(FpuExceptionJump);
|
|
m_Assembler.mov(TempReg, asmjit::x86::dword_ptr(RegPointer, 4));
|
|
m_Assembler.mov(TempReg2, asmjit::x86::dword_ptr(RegPointer));
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
m_Assembler.push(RegPointer);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUResult64), "R4300iOp::CheckFPUResult64", 8);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_Exception, false, &CX86Ops::JneLabel);
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
|
|
asmjit::Label ValueDiffernt = m_Assembler.newLabel();
|
|
asmjit::Label ValueSame = m_Assembler.newLabel();
|
|
m_Assembler.cmp(TempReg, asmjit::x86::dword_ptr((uint64_t)((uint8_t *)&m_TempValue64 + 4)));
|
|
m_Assembler.jne(ValueDiffernt);
|
|
m_Assembler.cmp(TempReg2, asmjit::x86::dword_ptr((uint64_t)&m_TempValue64));
|
|
m_Assembler.je(ValueSame);
|
|
m_Assembler.bind(ValueDiffernt);
|
|
ExitRegSet = m_RegWorkingSet;
|
|
m_Assembler.mov(TempReg2, asmjit::x86::dword_ptr(RegPointer));
|
|
m_Assembler.mov(TempReg, asmjit::x86::dword_ptr(RegPointer, 4));
|
|
asmjit::x86::Gp TempRegFPR_D = ExitRegSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(TempRegFPR_D, &m_Reg.m_FPR_D[m_Opcode.fd], stdstr_f("m_FPR_D[%d]", m_Opcode.fd).c_str());
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempRegFPR_D), TempReg2);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(TempRegFPR_D, 4), TempReg);
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC + 4, m_CompilePC + 4, ExitRegSet, ExitReason_Normal, false, &CX86Ops::JmpLabel);
|
|
m_Assembler.bind(ValueSame);
|
|
m_Assembler.bind(DoNoModify);
|
|
|
|
if (!RegPointerProtect)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(RegPointer), false);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileCop1Test()
|
|
{
|
|
if (m_RegWorkingSet.GetFpuBeenUsed())
|
|
{
|
|
return;
|
|
}
|
|
|
|
m_Assembler.finit();
|
|
m_Assembler.TestVariable(&g_Reg->STATUS_REGISTER, "STATUS_REGISTER", STATUS_CU1);
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, ExitRegSet, ExitReason_COP1Unuseable, false, &CX86Ops::JeLabel);
|
|
m_RegWorkingSet.SetFpuBeenUsed(true);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileInitFpuOperation(CRegBase::FPU_ROUND RoundMethod)
|
|
{
|
|
static uint32_t StatusRegister = 0;
|
|
|
|
CompileCop1Test();
|
|
asmjit::x86::Gp StatusReg = m_RegWorkingSet.Map_FPStatusReg();
|
|
m_Assembler.and_(StatusReg, (uint32_t)(~0x0003F000));
|
|
if (RoundMethod != CRegBase::RoundUnknown)
|
|
{
|
|
m_RegWorkingSet.FixRoundModel(RoundMethod);
|
|
}
|
|
|
|
m_Assembler.stmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
m_Assembler.and_(asmjit::x86::dword_ptr((uint64_t)&StatusRegister), ~0x25);
|
|
m_Assembler.ldmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
m_Assembler.fclex();
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileInPermLoop(CRegInfo & RegSet, uint32_t ProgramCounter)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", ProgramCounter);
|
|
RegSet.WriteBackRegisters();
|
|
UpdateCounters(RegSet, false, true, false);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::InPermLoop), "R4300iOp::InPermLoop", 4);
|
|
m_Assembler.CallThis((uint32_t)g_SystemTimer, AddressOf(&CSystemTimer::TimerDone), "CSystemTimer::TimerDone", 4);
|
|
m_CodeBlock.Log("CompileSystemCheck 3");
|
|
CompileSystemCheck((uint32_t)-1, RegSet);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
}
|
|
|
|
bool CX86RecompilerOps::SetupRegisterForLoop(CCodeBlock & BlockInfo, const CRegInfo & RegSet)
|
|
{
|
|
CRegInfo OriginalReg = m_RegWorkingSet;
|
|
if (!LoopAnalysis(BlockInfo, m_Section).SetupRegisterForLoop())
|
|
{
|
|
return false;
|
|
}
|
|
for (int i = 1; i < 32; i++)
|
|
{
|
|
if (OriginalReg.GetMipsRegState(i) != RegSet.GetMipsRegState(i))
|
|
{
|
|
m_RegWorkingSet.UnMap_GPR(i, true);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void CX86RecompilerOps::SyncRegState(const CRegInfo & SyncTo)
|
|
{
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
|
|
bool changed = false;
|
|
m_RegWorkingSet.UnMap_AllFPRs();
|
|
if (m_RegWorkingSet.GetRoundingModel() != SyncTo.GetRoundingModel())
|
|
{
|
|
m_RegWorkingSet.SetRoundingModel(CRegInfo::RoundUnknown);
|
|
}
|
|
asmjit::x86::Gp MemStackReg = m_RegWorkingSet.Get_MemoryStack();
|
|
asmjit::x86::Gp TargetStackReg = SyncTo.Get_MemoryStack();
|
|
|
|
//m_CodeBlock.Log("MemoryStack for Original State = %s",MemStackReg > 0?CX86Ops::x86_Name(MemStackReg):"Not Mapped");
|
|
if (MemStackReg != TargetStackReg)
|
|
{
|
|
if (!TargetStackReg.isValid())
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(MemStackReg);
|
|
}
|
|
else if (!MemStackReg.isValid())
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(TargetStackReg);
|
|
m_CodeBlock.Log(" regcache: allocate %s as memory stack", CX86Ops::x86_Name(TargetStackReg));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(TargetStackReg), CRegInfo::Stack_Mapped);
|
|
m_Assembler.MoveVariableToX86reg(TargetStackReg, &g_Recompiler->MemoryStackPos(), "MemoryStack");
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(TargetStackReg);
|
|
m_CodeBlock.Log(" regcache: change allocation of memory stack from %s to %s", CX86Ops::x86_Name(MemStackReg), CX86Ops::x86_Name(TargetStackReg));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(TargetStackReg), CRegInfo::Stack_Mapped);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(MemStackReg), CRegInfo::NotMapped);
|
|
m_Assembler.mov(TargetStackReg, MemStackReg);
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp TargetFPStatusReg = SyncTo.GetFPStatusReg();
|
|
asmjit::x86::Gp FPStatusReg = m_RegWorkingSet.GetFPStatusReg();
|
|
if (FPStatusReg != TargetFPStatusReg)
|
|
{
|
|
if (!TargetFPStatusReg.isValid())
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(FPStatusReg);
|
|
}
|
|
else if (!FPStatusReg.isValid())
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(TargetFPStatusReg);
|
|
m_CodeBlock.Log(" regcache: allocate %s as fp status reg", CX86Ops::x86_Name(TargetFPStatusReg));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(TargetFPStatusReg), CRegInfo::FPStatusReg_Mapped);
|
|
m_Assembler.MoveVariableToX86reg(TargetFPStatusReg, &g_Reg->m_FPCR[31], "FPCR[31]");
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(TargetFPStatusReg);
|
|
m_CodeBlock.Log(" regcache: change allocation of fp status reg from %s to %s", CX86Ops::x86_Name(FPStatusReg), CX86Ops::x86_Name(TargetFPStatusReg));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(TargetFPStatusReg), CRegInfo::FPStatusReg_Mapped);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(FPStatusReg), CRegInfo::NotMapped);
|
|
m_Assembler.mov(TargetFPStatusReg, FPStatusReg);
|
|
}
|
|
}
|
|
|
|
for (int i = 1; i < 32; i++)
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegState(i) == SyncTo.GetMipsRegState(i) ||
|
|
(b32BitCore() && m_RegWorkingSet.GetMipsRegState(i) == CRegInfo::STATE_MAPPED_32_ZERO && SyncTo.GetMipsRegState(i) == CRegInfo::STATE_MAPPED_32_SIGN) ||
|
|
(b32BitCore() && m_RegWorkingSet.GetMipsRegState(i) == CRegInfo::STATE_MAPPED_32_SIGN && SyncTo.GetMipsRegState(i) == CRegInfo::STATE_MAPPED_32_ZERO))
|
|
{
|
|
switch (m_RegWorkingSet.GetMipsRegState(i))
|
|
{
|
|
case CRegInfo::STATE_UNKNOWN: continue;
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
if (m_RegWorkingSet.GetMipsRegMapHi(i) == SyncTo.GetMipsRegMapHi(i) &&
|
|
m_RegWorkingSet.GetMipsRegMapLo(i) == SyncTo.GetMipsRegMapLo(i))
|
|
{
|
|
continue;
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(i) == SyncTo.GetMipsRegMapLo(i))
|
|
{
|
|
continue;
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_CONST_64:
|
|
if (m_RegWorkingSet.GetMipsReg(i) != SyncTo.GetMipsReg(i))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
continue;
|
|
case CRegInfo::STATE_CONST_32_SIGN:
|
|
if (m_RegWorkingSet.GetMipsRegLo(i) != SyncTo.GetMipsRegLo(i))
|
|
{
|
|
m_CodeBlock.Log("Value of constant is different register %d (%s) Value: 0x%08X to 0x%08X", i, CRegName::GPR[i], m_RegWorkingSet.GetMipsRegLo(i), SyncTo.GetMipsRegLo(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
continue;
|
|
default:
|
|
m_CodeBlock.Log("Unhandled register state %d\nin SyncRegState", m_RegWorkingSet.GetMipsRegState(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
changed = true;
|
|
|
|
switch (SyncTo.GetMipsRegState(i))
|
|
{
|
|
case CRegInfo::STATE_UNKNOWN: m_RegWorkingSet.UnMap_GPR(i, true); break;
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
{
|
|
asmjit::x86::Gp Reg = SyncTo.GetMipsRegMapLo(i);
|
|
asmjit::x86::Gp x86RegHi = SyncTo.GetMipsRegMapHi(i);
|
|
m_RegWorkingSet.UnMap_X86reg(Reg);
|
|
m_RegWorkingSet.UnMap_X86reg(x86RegHi);
|
|
switch (m_RegWorkingSet.GetMipsRegState(i))
|
|
{
|
|
case CRegInfo::STATE_UNKNOWN:
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[i].UW[0], CRegName::GPR_Lo[i]);
|
|
m_Assembler.MoveVariableToX86reg(x86RegHi, &m_Reg.m_GPR[i].UW[1], CRegName::GPR_Hi[i]);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
m_Assembler.mov(x86RegHi, m_RegWorkingSet.GetMipsRegMapHi(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapHi(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
m_Assembler.mov(x86RegHi, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_Assembler.sar(x86RegHi, 31);
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
m_Assembler.xor_(x86RegHi, x86RegHi);
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_CONST_64:
|
|
m_Assembler.MoveConstToX86reg(x86RegHi, m_RegWorkingSet.GetMipsRegHi(i));
|
|
m_Assembler.MoveConstToX86reg(Reg, m_RegWorkingSet.GetMipsRegLo(i));
|
|
break;
|
|
case CRegInfo::STATE_CONST_32_SIGN:
|
|
m_Assembler.MoveConstToX86reg(x86RegHi, m_RegWorkingSet.GetMipsRegLo_S(i) >> 31);
|
|
m_Assembler.MoveConstToX86reg(Reg, m_RegWorkingSet.GetMipsRegLo(i));
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log("Do something with states in SyncRegState\nSTATE_MAPPED_64\n%d", m_RegWorkingSet.GetMipsRegState(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
continue;
|
|
}
|
|
m_RegWorkingSet.SetMipsRegMapLo(i, Reg);
|
|
m_RegWorkingSet.SetMipsRegMapHi(i, x86RegHi);
|
|
m_RegWorkingSet.SetMipsRegState(i, CRegInfo::STATE_MAPPED_64);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(Reg), CRegInfo::GPR_Mapped);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(x86RegHi), CRegInfo::GPR_Mapped);
|
|
m_RegWorkingSet.SetX86MapOrder(GetIndexFromX86Reg(Reg), 1);
|
|
m_RegWorkingSet.SetX86MapOrder(GetIndexFromX86Reg(x86RegHi), 1);
|
|
break;
|
|
}
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
{
|
|
asmjit::x86::Gp Reg = SyncTo.GetMipsRegMapLo(i);
|
|
m_RegWorkingSet.UnMap_X86reg(Reg);
|
|
switch (m_RegWorkingSet.GetMipsRegState(i))
|
|
{
|
|
case CRegInfo::STATE_UNKNOWN: m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[i].UW[0], CRegName::GPR_Lo[i]); break;
|
|
case CRegInfo::STATE_CONST_32_SIGN: m_Assembler.MoveConstToX86reg(Reg, m_RegWorkingSet.GetMipsRegLo(i)); break;
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(i) != Reg)
|
|
{
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapHi(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_CONST_64:
|
|
m_CodeBlock.Log("hi %X\nLo %X", m_RegWorkingSet.GetMipsRegHi(i), m_RegWorkingSet.GetMipsRegLo(i));
|
|
default:
|
|
m_CodeBlock.Log("Do something with states in SyncRegState\nSTATE_MAPPED_32_SIGN\n%d", m_RegWorkingSet.GetMipsRegState(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegMapLo(i, Reg);
|
|
m_RegWorkingSet.SetMipsRegState(i, CRegInfo::STATE_MAPPED_32_SIGN);
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(Reg), CRegInfo::GPR_Mapped);
|
|
m_RegWorkingSet.SetX86MapOrder(GetIndexFromX86Reg(Reg), 1);
|
|
break;
|
|
}
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
{
|
|
asmjit::x86::Gp Reg = SyncTo.GetMipsRegMapLo(i);
|
|
m_RegWorkingSet.UnMap_X86reg(Reg);
|
|
switch (m_RegWorkingSet.GetMipsRegState(i))
|
|
{
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
case CRegInfo::STATE_UNKNOWN:
|
|
m_Assembler.MoveVariableToX86reg(Reg, &m_Reg.m_GPR[i].UW[0], CRegName::GPR_Lo[i]);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
if (b32BitCore())
|
|
{
|
|
m_Assembler.mov(Reg, m_RegWorkingSet.GetMipsRegMapLo(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(i)), CRegInfo::NotMapped);
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log("Do something with states in SyncRegState\nSTATE_MAPPED_32_ZERO\n%d", m_RegWorkingSet.GetMipsRegState(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_CONST_32_SIGN:
|
|
if (!b32BitCore() && m_RegWorkingSet.GetMipsRegLo_S(i) < 0)
|
|
{
|
|
m_CodeBlock.Log("Sign problems in SyncRegState\nSTATE_MAPPED_32_ZERO");
|
|
m_CodeBlock.Log("%s: %X", CRegName::GPR[i], m_RegWorkingSet.GetMipsRegLo_S(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_Assembler.MoveConstToX86reg(Reg, m_RegWorkingSet.GetMipsRegLo(i));
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log("Do something with states in SyncRegState\nSTATE_MAPPED_32_ZERO\n%d", m_RegWorkingSet.GetMipsRegState(i));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_RegWorkingSet.SetMipsRegMapLo(i, Reg);
|
|
m_RegWorkingSet.SetMipsRegState(i, SyncTo.GetMipsRegState(i));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(Reg), CRegInfo::GPR_Mapped);
|
|
m_RegWorkingSet.SetX86MapOrder(GetIndexFromX86Reg(Reg), 1);
|
|
break;
|
|
}
|
|
default:
|
|
m_CodeBlock.Log("%d - %d reg: %s (%d)", SyncTo.GetMipsRegState(i), m_RegWorkingSet.GetMipsRegState(i), CRegName::GPR[i], i);
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
changed = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
CRegInfo & CX86RecompilerOps::GetRegWorkingSet(void)
|
|
{
|
|
return m_RegWorkingSet;
|
|
}
|
|
|
|
void CX86RecompilerOps::SetRegWorkingSet(const CRegInfo & RegInfo)
|
|
{
|
|
m_RegWorkingSet = RegInfo;
|
|
}
|
|
|
|
bool CX86RecompilerOps::InheritParentInfo()
|
|
{
|
|
m_Section->DisplaySectionInformation();
|
|
|
|
if (m_Section->m_ParentSection.empty())
|
|
{
|
|
SetRegWorkingSet(m_Section->m_RegEnter);
|
|
return true;
|
|
}
|
|
|
|
if (m_Section->m_ParentSection.size() == 1)
|
|
{
|
|
CCodeSection * Parent = *(m_Section->m_ParentSection.begin());
|
|
if (!Parent->m_EnterLabel.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
CJumpInfo * JumpInfo = m_Section == Parent->m_ContinueSection ? &Parent->m_Cont : &Parent->m_Jump;
|
|
|
|
m_Section->m_RegEnter = JumpInfo->RegSet;
|
|
LinkJump(*JumpInfo);
|
|
SetRegWorkingSet(m_Section->m_RegEnter);
|
|
return true;
|
|
}
|
|
|
|
// Multiple parents
|
|
BLOCK_PARENT_LIST ParentList;
|
|
CCodeSection::SECTION_LIST::iterator iter;
|
|
for (iter = m_Section->m_ParentSection.begin(); iter != m_Section->m_ParentSection.end(); iter++)
|
|
{
|
|
CCodeSection * Parent = *iter;
|
|
BLOCK_PARENT BlockParent;
|
|
|
|
if (!Parent->m_EnterLabel.isValid())
|
|
{
|
|
continue;
|
|
}
|
|
if (Parent->m_JumpSection != Parent->m_ContinueSection)
|
|
{
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = m_Section == Parent->m_ContinueSection ? &Parent->m_Cont : &Parent->m_Jump;
|
|
ParentList.push_back(BlockParent);
|
|
}
|
|
else
|
|
{
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = &Parent->m_Cont;
|
|
ParentList.push_back(BlockParent);
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = &Parent->m_Jump;
|
|
ParentList.push_back(BlockParent);
|
|
}
|
|
}
|
|
size_t NoOfCompiledParents = ParentList.size();
|
|
if (NoOfCompiledParents == 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return false;
|
|
}
|
|
|
|
// Add all the uncompiled blocks to the end of the list
|
|
for (iter = m_Section->m_ParentSection.begin(); iter != m_Section->m_ParentSection.end(); iter++)
|
|
{
|
|
CCodeSection * Parent = *iter;
|
|
BLOCK_PARENT BlockParent;
|
|
|
|
if (Parent->m_EnterLabel.isValid())
|
|
{
|
|
continue;
|
|
}
|
|
if (Parent->m_JumpSection != Parent->m_ContinueSection)
|
|
{
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = m_Section == Parent->m_ContinueSection ? &Parent->m_Cont : &Parent->m_Jump;
|
|
ParentList.push_back(BlockParent);
|
|
}
|
|
else
|
|
{
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = &Parent->m_Cont;
|
|
ParentList.push_back(BlockParent);
|
|
BlockParent.Parent = Parent;
|
|
BlockParent.JumpInfo = &Parent->m_Jump;
|
|
ParentList.push_back(BlockParent);
|
|
}
|
|
}
|
|
int FirstParent = -1;
|
|
for (size_t i = 0; i < NoOfCompiledParents; i++)
|
|
{
|
|
if (!ParentList[i].JumpInfo->FallThrough)
|
|
{
|
|
continue;
|
|
}
|
|
if (FirstParent != -1)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
FirstParent = i;
|
|
}
|
|
if (FirstParent == -1)
|
|
{
|
|
FirstParent = 0;
|
|
}
|
|
|
|
// Link first parent to start
|
|
CCodeSection * Parent = ParentList[FirstParent].Parent;
|
|
CJumpInfo * JumpInfo = ParentList[FirstParent].JumpInfo;
|
|
|
|
SetRegWorkingSet(JumpInfo->RegSet);
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
LinkJump(*JumpInfo);
|
|
|
|
if (JumpInfo->Reason == ExitReason_NormalNoSysCheck)
|
|
{
|
|
if (JumpInfo->RegSet.GetBlockCycleCount() != 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (JumpInfo->JumpPC != (uint32_t)-1)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
UpdateCounters(JumpInfo->RegSet, m_Section->m_EnterPC < JumpInfo->JumpPC, true);
|
|
if (JumpInfo->JumpPC == (uint32_t)-1)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (m_Section->m_EnterPC <= JumpInfo->JumpPC)
|
|
{
|
|
m_CodeBlock.Log("CompileSystemCheck 10");
|
|
CompileSystemCheck(m_Section->m_EnterPC, GetRegWorkingSet());
|
|
}
|
|
}
|
|
JumpInfo->FallThrough = false;
|
|
|
|
// Fix up initial state
|
|
m_RegWorkingSet.UnMap_AllFPRs();
|
|
|
|
// Determine loop registry usage
|
|
if (m_Section->m_InLoop && ParentList.size() > 1)
|
|
{
|
|
if (!SetupRegisterForLoop(m_Section->m_CodeBlock, m_Section->m_RegEnter))
|
|
{
|
|
return false;
|
|
}
|
|
m_RegWorkingSet.SetRoundingModel(CRegInfo::RoundUnknown);
|
|
}
|
|
|
|
for (size_t i = 0; i < ParentList.size(); i++)
|
|
{
|
|
asmjit::x86::Gp MemoryStackPos;
|
|
int i2;
|
|
|
|
if (i == (size_t)FirstParent)
|
|
{
|
|
continue;
|
|
}
|
|
Parent = ParentList[i].Parent;
|
|
if (!Parent->m_EnterLabel.isValid())
|
|
{
|
|
continue;
|
|
}
|
|
CRegInfo * RegSet = &ParentList[i].JumpInfo->RegSet;
|
|
|
|
if (m_RegWorkingSet.GetRoundingModel() != RegSet->GetRoundingModel())
|
|
{
|
|
m_RegWorkingSet.SetRoundingModel(CRegInfo::RoundUnknown);
|
|
}
|
|
|
|
// Find parent MapRegState
|
|
MemoryStackPos = x86Reg_Unknown;
|
|
for (i2 = 0; i2 < x86RegIndex_Size; i2++)
|
|
{
|
|
if (RegSet->GetX86Mapped((x86RegIndex)i2) == CRegInfo::Stack_Mapped)
|
|
{
|
|
MemoryStackPos = GetX86RegFromIndex((x86RegIndex)i2);
|
|
break;
|
|
}
|
|
}
|
|
if (!MemoryStackPos.isValid())
|
|
{
|
|
// If the memory stack position is not mapped then unmap it
|
|
asmjit::x86::Gp MemStackReg = m_RegWorkingSet.Get_MemoryStack();
|
|
if (MemStackReg.isValid())
|
|
{
|
|
m_RegWorkingSet.UnMap_X86reg(MemStackReg);
|
|
}
|
|
}
|
|
|
|
for (i2 = 1; i2 < 32; i2++)
|
|
{
|
|
if (m_RegWorkingSet.Is32BitMapped(i2))
|
|
{
|
|
switch (RegSet->GetMipsRegState(i2))
|
|
{
|
|
case CRegInfo::STATE_MAPPED_64: m_RegWorkingSet.Map_GPR_64bit(i2, i2); break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO: break;
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
if (m_RegWorkingSet.IsUnsigned(i2))
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(i2, CRegInfo::STATE_MAPPED_32_SIGN);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_CONST_64: m_RegWorkingSet.Map_GPR_64bit(i2, i2); break;
|
|
case CRegInfo::STATE_CONST_32_SIGN:
|
|
if ((RegSet->GetMipsRegLo_S(i2) < 0) && m_RegWorkingSet.IsUnsigned(i2))
|
|
{
|
|
m_RegWorkingSet.SetMipsRegState(i2, CRegInfo::STATE_MAPPED_32_SIGN);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_UNKNOWN:
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, true, i2);
|
|
}
|
|
else
|
|
{
|
|
//m_RegWorkingSet.Map_GPR_32bit(i2,true,i2);
|
|
m_RegWorkingSet.Map_GPR_64bit(i2, i2); //??
|
|
//m_RegWorkingSet.UnMap_GPR(Section,i2,true); ??
|
|
}
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log("Unknown CPU state(%d) in InheritParentInfo", m_RegWorkingSet.GetMipsRegState(i2));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
if (m_RegWorkingSet.IsConst(i2))
|
|
{
|
|
if (m_RegWorkingSet.GetMipsRegState(i2) != RegSet->GetMipsRegState(i2))
|
|
{
|
|
switch (RegSet->GetMipsRegState(i2))
|
|
{
|
|
case CRegInfo::STATE_CONST_64:
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
m_RegWorkingSet.Map_GPR_64bit(i2, i2);
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
if (m_RegWorkingSet.Is32Bit(i2))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, (m_RegWorkingSet.GetMipsRegLo(i2) & 0x80000000) != 0, i2);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
if (m_RegWorkingSet.Is32Bit(i2))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, true, i2);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_UNKNOWN:
|
|
if (b32BitCore())
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, true, i2);
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_64bit(i2, i2);
|
|
}
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log("Unknown CPU state(%d) in InheritParentInfo", RegSet->GetMipsRegState(i2));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
break;
|
|
}
|
|
}
|
|
else if (m_RegWorkingSet.Is32Bit(i2) && m_RegWorkingSet.GetMipsRegLo(i2) != RegSet->GetMipsRegLo(i2))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, true, i2);
|
|
}
|
|
else if (m_RegWorkingSet.Is64Bit(i2) && m_RegWorkingSet.GetMipsReg(i2) != RegSet->GetMipsReg(i2))
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(i2, true, i2);
|
|
}
|
|
}
|
|
m_RegWorkingSet.ResetX86Protection();
|
|
}
|
|
|
|
if (MemoryStackPos.isValid())
|
|
{
|
|
m_RegWorkingSet.Map_MemoryStack(MemoryStackPos, true);
|
|
}
|
|
}
|
|
m_Section->m_RegEnter = m_RegWorkingSet;
|
|
|
|
// Sync registers for different blocks
|
|
stdstr_f Label("Section_%d", m_Section->m_SectionID);
|
|
int CurrentParent = FirstParent;
|
|
bool NeedSync = false;
|
|
for (size_t i = 0; i < NoOfCompiledParents; i++)
|
|
{
|
|
CRegInfo * RegSet;
|
|
int i2;
|
|
|
|
if (i == (size_t)FirstParent)
|
|
{
|
|
continue;
|
|
}
|
|
Parent = ParentList[i].Parent;
|
|
JumpInfo = ParentList[i].JumpInfo;
|
|
RegSet = &ParentList[i].JumpInfo->RegSet;
|
|
|
|
if (JumpInfo->RegSet.GetBlockCycleCount() != 0)
|
|
{
|
|
NeedSync = true;
|
|
}
|
|
|
|
for (i2 = 0; !NeedSync && i2 < 8; i2++)
|
|
{
|
|
if (m_RegWorkingSet.FpuMappedTo(i2) == (uint32_t)-1)
|
|
{
|
|
NeedSync = true;
|
|
}
|
|
}
|
|
|
|
for (i2 = 0; !NeedSync && i2 < x86RegIndex_Size; i2++)
|
|
{
|
|
if (m_RegWorkingSet.GetX86Mapped((x86RegIndex)i2) == CRegInfo::Stack_Mapped)
|
|
{
|
|
if (m_RegWorkingSet.GetX86Mapped((x86RegIndex)i2) != RegSet->GetX86Mapped((x86RegIndex)i2))
|
|
{
|
|
NeedSync = true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
for (i2 = 0; !NeedSync && i2 < 32; i2++)
|
|
{
|
|
if (NeedSync == true)
|
|
{
|
|
break;
|
|
}
|
|
if (m_RegWorkingSet.GetMipsRegState(i2) != RegSet->GetMipsRegState(i2))
|
|
{
|
|
NeedSync = true;
|
|
continue;
|
|
}
|
|
switch (m_RegWorkingSet.GetMipsRegState(i2))
|
|
{
|
|
case CRegInfo::STATE_UNKNOWN: break;
|
|
case CRegInfo::STATE_MAPPED_64:
|
|
if (m_RegWorkingSet.GetMipsRegMapHi(i2) != RegSet->GetMipsRegMapHi(i2) ||
|
|
m_RegWorkingSet.GetMipsRegMapLo(i2) != RegSet->GetMipsRegMapLo(i2))
|
|
{
|
|
NeedSync = true;
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_MAPPED_32_ZERO:
|
|
case CRegInfo::STATE_MAPPED_32_SIGN:
|
|
if (m_RegWorkingSet.GetMipsRegMapLo(i2) != RegSet->GetMipsRegMapLo(i2))
|
|
{
|
|
//DisplayError(L"Parent: %d",Parent->SectionID);
|
|
NeedSync = true;
|
|
}
|
|
break;
|
|
case CRegInfo::STATE_CONST_32_SIGN:
|
|
if (m_RegWorkingSet.GetMipsRegLo(i2) != RegSet->GetMipsRegLo(i2))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
NeedSync = true;
|
|
}
|
|
break;
|
|
default:
|
|
WriteTrace(TraceRecompiler, TraceError, "Unhandled register state %d\nin InheritParentInfo", m_RegWorkingSet.GetMipsRegState(i2));
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
if (NeedSync == false)
|
|
{
|
|
continue;
|
|
}
|
|
Parent = ParentList[CurrentParent].Parent;
|
|
JumpInfo = ParentList[CurrentParent].JumpInfo;
|
|
JumpInfo->LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(Label.c_str(), JumpInfo->LinkLocation);
|
|
JumpInfo->LinkLocation2 = asmjit::Label();
|
|
|
|
CurrentParent = i;
|
|
Parent = ParentList[CurrentParent].Parent;
|
|
JumpInfo = ParentList[CurrentParent].JumpInfo;
|
|
m_CodeBlock.Log(" Section_%d (from %d):", m_Section->m_SectionID, Parent->m_SectionID);
|
|
if (JumpInfo->LinkLocation.isValid())
|
|
{
|
|
m_Assembler.bind(JumpInfo->LinkLocation);
|
|
JumpInfo->LinkLocation = asmjit::Label();
|
|
if (JumpInfo->LinkLocation2.isValid())
|
|
{
|
|
m_Assembler.bind(JumpInfo->LinkLocation2);
|
|
JumpInfo->LinkLocation2 = asmjit::Label();
|
|
}
|
|
}
|
|
m_RegWorkingSet = JumpInfo->RegSet;
|
|
if (m_Section->m_EnterPC < JumpInfo->JumpPC)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
m_CodeBlock.Log("CompileSystemCheck 11");
|
|
CompileSystemCheck(m_Section->m_EnterPC, m_RegWorkingSet);
|
|
}
|
|
else
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
SyncRegState(m_Section->m_RegEnter); // Sync
|
|
m_Section->m_RegEnter = m_RegWorkingSet;
|
|
}
|
|
|
|
for (size_t i = 0; i < NoOfCompiledParents; i++)
|
|
{
|
|
Parent = ParentList[i].Parent;
|
|
JumpInfo = ParentList[i].JumpInfo;
|
|
LinkJump(*JumpInfo);
|
|
}
|
|
|
|
m_CodeBlock.Log(" Section_%d:", m_Section->m_SectionID);
|
|
m_Section->m_RegEnter.SetBlockCycleCount(0);
|
|
return true;
|
|
}
|
|
|
|
void CX86RecompilerOps::LinkJump(CJumpInfo & JumpInfo)
|
|
{
|
|
if (JumpInfo.LinkLocation.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpInfo.LinkLocation);
|
|
JumpInfo.LinkLocation = asmjit::Label();
|
|
if (JumpInfo.LinkLocation2.isValid())
|
|
{
|
|
m_Assembler.bind(JumpInfo.LinkLocation2);
|
|
JumpInfo.LinkLocation2 = asmjit::Label();
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::JumpToSection(CCodeSection * Section)
|
|
{
|
|
m_Assembler.JmpLabel(stdstr_f("Section_%d", Section->m_SectionID).c_str(), Section->m_EnterLabel);
|
|
}
|
|
|
|
void CX86RecompilerOps::JumpToUnknown(CJumpInfo * JumpInfo)
|
|
{
|
|
JumpInfo->LinkLocation = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(JumpInfo->BranchLabel.c_str(), JumpInfo->LinkLocation);
|
|
}
|
|
|
|
void CX86RecompilerOps::SetCurrentPC(uint32_t ProgramCounter)
|
|
{
|
|
uint32_t Value;
|
|
if (!g_MMU->MemoryValue32(ProgramCounter, Value))
|
|
{
|
|
g_Notify->FatalError(GS(MSG_FAIL_LOAD_WORD));
|
|
}
|
|
m_Instruction = R4300iInstruction((int32_t)ProgramCounter, Value);
|
|
}
|
|
|
|
uint32_t CX86RecompilerOps::GetCurrentPC(void)
|
|
{
|
|
return (uint32_t)m_CompilePC;
|
|
}
|
|
|
|
void CX86RecompilerOps::SetCurrentSection(CCodeSection * section)
|
|
{
|
|
m_Section = section;
|
|
}
|
|
|
|
void CX86RecompilerOps::SetNextStepType(PIPELINE_STAGE StepType)
|
|
{
|
|
m_PipelineStage = StepType;
|
|
}
|
|
|
|
PIPELINE_STAGE CX86RecompilerOps::GetNextStepType(void)
|
|
{
|
|
return m_PipelineStage;
|
|
}
|
|
|
|
const R4300iOpcode & CX86RecompilerOps::GetOpcode(void) const
|
|
{
|
|
return m_Opcode;
|
|
}
|
|
|
|
const R4300iInstruction & CX86RecompilerOps::GetInstruction(void) const
|
|
{
|
|
return m_Instruction;
|
|
}
|
|
|
|
void CX86RecompilerOps::UpdateSyncCPU(CRegInfo & RegSet, uint32_t Cycles)
|
|
{
|
|
if (!g_SyncSystem)
|
|
{
|
|
return;
|
|
}
|
|
|
|
m_CodeBlock.Log("");
|
|
m_CodeBlock.Log(" // Updating sync CPU");
|
|
RegSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32(stdstr_f("%d", Cycles).c_str(), Cycles);
|
|
m_Assembler.CallThis((uint32_t)g_System, AddressOf(&CN64System::UpdateSyncCPU), "CN64System::UpdateSyncCPU", 8);
|
|
RegSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::UpdateCounters(CRegInfo & RegSet, bool CheckTimer, bool ClearValues, bool UpdateTimer)
|
|
{
|
|
if (RegSet.GetBlockCycleCount() != 0)
|
|
{
|
|
UpdateSyncCPU(RegSet, RegSet.GetBlockCycleCount());
|
|
m_CodeBlock.Log("");
|
|
m_CodeBlock.Log(" // Update counter");
|
|
m_Assembler.SubConstFromVariable(RegSet.GetBlockCycleCount(), g_NextTimer, "g_NextTimer"); // Updates compare flag
|
|
if (ClearValues)
|
|
{
|
|
RegSet.SetBlockCycleCount(0);
|
|
}
|
|
}
|
|
else if (CheckTimer)
|
|
{
|
|
m_Assembler.CompConstToVariable(g_NextTimer, "g_NextTimer", 0);
|
|
}
|
|
|
|
if (CheckTimer)
|
|
{
|
|
asmjit::Label Jump = m_Assembler.newLabel();
|
|
m_Assembler.JnsLabel("Continue_From_Timer_Test", Jump);
|
|
RegSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_SystemTimer, AddressOf(&CSystemTimer::TimerDone), "CSystemTimer::TimerDone", 4);
|
|
RegSet.AfterCallDirect();
|
|
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
|
|
if ((UpdateTimer || CGameSettings::OverClockModifier() != 1) && g_SyncSystem)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_SystemTimer, AddressOf(&CSystemTimer::UpdateTimers), "CSystemTimer::UpdateTimers", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileSystemCheck(uint32_t TargetPC, const CRegInfo & RegSet)
|
|
{
|
|
m_Assembler.CompConstByteToVariable((void *)&m_SystemEvents.DoSomething(), "m_SystemEvents.DoSomething()", 0);
|
|
asmjit::Label Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Continue_From_Interrupt_Test", Jump);
|
|
if (TargetPC != (uint32_t)-1)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", TargetPC);
|
|
}
|
|
|
|
CRegInfo RegSetCopy(RegSet);
|
|
RegSetCopy.WriteBackRegisters();
|
|
m_Assembler.CallThis((uint32_t)&m_SystemEvents, AddressOf(&CSystemEvents::ExecuteEvents), "CSystemEvents::ExecuteEvents", 4);
|
|
ExitCodeBlock();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileExecuteBP(void)
|
|
{
|
|
bool bDelay = m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT;
|
|
if (bDelay)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
m_Assembler.CallFunc((uint32_t)x86CompilerBreakPoint, "x86CompilerBreakPoint");
|
|
ExitCodeBlock();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileExecuteDelaySlotBP(void)
|
|
{
|
|
bool bDelay = m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT;
|
|
if (bDelay)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
|
|
UpdateCounters(m_RegWorkingSet, true, true);
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
m_Assembler.CallFunc((uint32_t)x86BreakPointDelaySlot, "x86BreakPointDelaySlot");
|
|
ExitCodeBlock();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
void CX86RecompilerOps::OverflowDelaySlot(bool TestTimer)
|
|
{
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
m_RegWorkingSet.WriteBackRegisters();
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
if (m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::ecx, &g_System->m_JumpToLocation, "System::m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::ecx);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::ecx, &g_System->m_JumpDelayLocation, "System::JumpDelayLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_System->m_JumpToLocation, "System::m_JumpToLocation", asmjit::x86::ecx);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC + 4);
|
|
}
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
if (g_SyncSystem)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_BaseSystem, AddressOf(&CN64System::SyncSystem), "CN64System::SyncSystem", 4);
|
|
}
|
|
|
|
if (TestTimer)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_TestTimer, "R4300iOp::m_TestTimer", TestTimer);
|
|
}
|
|
|
|
m_Assembler.PushImm32("g_System->CountPerOp()", g_System->CountPerOp());
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::ExecuteOps), "R4300iOp::ExecuteOps", 8);
|
|
|
|
if (g_System->bFastSP() && g_Recompiler)
|
|
{
|
|
m_Assembler.CallThis((uint32_t)g_Recompiler, AddressOf(&CRecompiler::ResetMemoryStackPos), "CRecompiler::ResetMemoryStackPos", 4);
|
|
}
|
|
if (g_SyncSystem)
|
|
{
|
|
UpdateSyncCPU(m_RegWorkingSet, g_System->CountPerOp());
|
|
}
|
|
|
|
ExitCodeBlock();
|
|
m_PipelineStage = PIPELINE_STAGE_END_BLOCK;
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileExit(uint64_t JumpPC, uint64_t TargetPC, CRegInfo & ExitRegSet, ExitReason reason)
|
|
{
|
|
CompileExit(JumpPC, TargetPC, ExitRegSet, reason, true, nullptr);
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileExit(uint64_t JumpPC, uint64_t TargetPC, CRegInfo & ExitRegSet, ExitReason reason, bool CompileNow, void (CX86Ops::*x86Jmp)(const char * LabelName, asmjit::Label & JumpLabel))
|
|
{
|
|
if (!CompileNow)
|
|
{
|
|
if (x86Jmp == nullptr)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
CExitInfo ExitInfo(m_CodeBlock);
|
|
stdstr_f ExitName("Exit_%08X_%d", JumpPC, m_ExitInfo.size());
|
|
(m_Assembler.*x86Jmp)(ExitName.c_str(), ExitInfo.JumpLabel);
|
|
ExitInfo.ID = m_ExitInfo.size();
|
|
ExitInfo.Name = ExitName;
|
|
ExitInfo.TargetPC = TargetPC;
|
|
ExitInfo.ExitRegSet = ExitRegSet;
|
|
ExitInfo.Reason = reason;
|
|
ExitInfo.PipelineStage = m_PipelineStage;
|
|
m_ExitInfo.push_back(ExitInfo);
|
|
return;
|
|
}
|
|
|
|
//m_CodeBlock.Log("CompileExit: %d",reason);
|
|
ExitRegSet.WriteBackRegisters();
|
|
|
|
if (TargetPC != (uint64_t)-1)
|
|
{
|
|
m_Assembler.MoveConst64ToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", TargetPC);
|
|
UpdateCounters(ExitRegSet, TargetPC <= JumpPC && JumpPC != (uint64_t)-1, reason == ExitReason_Normal);
|
|
}
|
|
else
|
|
{
|
|
UpdateCounters(ExitRegSet, false, reason == ExitReason_Normal);
|
|
}
|
|
|
|
bool InDelaySlot = m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT;
|
|
switch (reason)
|
|
{
|
|
case ExitReason_Normal:
|
|
case ExitReason_CheckPCAlignment:
|
|
case ExitReason_NormalNoSysCheck:
|
|
ExitRegSet.SetBlockCycleCount(0);
|
|
if (reason == ExitReason_Normal && (TargetPC == (uint64_t)-1 || TargetPC <= JumpPC))
|
|
{
|
|
CompileSystemCheck((uint32_t)-1, ExitRegSet);
|
|
}
|
|
if (reason == ExitReason_CheckPCAlignment)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::eax, &g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER");
|
|
m_Assembler.test(asmjit::x86::eax, 3);
|
|
asmjit::Label ValidPCJump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("", ValidPCJump);
|
|
m_Assembler.push(1);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, (void *)(((uint64_t)&g_Reg->m_PROGRAM_COUNTER) + 4), "PROGRAM_COUNTER + 4");
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::DoAddressError), "CRegisters::DoAddressError", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
m_Assembler.bind(ValidPCJump);
|
|
}
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_DoCPUAction:
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_SystemEvents, AddressOf(&CSystemEvents::ExecuteEvents), "CSystemEvents::ExecuteEvents", 4);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_DoSysCall:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.PushImm32("EXC_SYSCALL", EXC_SYSCALL);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_Break:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.PushImm32("EXC_BREAK", EXC_BREAK);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_COP1Unuseable:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(1);
|
|
m_Assembler.PushImm32("EXC_CPU", EXC_CPU);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitRegSet.SetBlockCycleCount(0);
|
|
UpdateCounters(ExitRegSet, true, false, false);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_ResetRecompCode:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_TLBReadMiss:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.PushImm32("EXC_RMISS", EXC_RMISS);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &m_System.m_TLBLoadAddress, "m_TLBLoadAddress");
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerAddressException), "CRegisters::TriggerAddressException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_TLBWriteMiss:
|
|
m_Assembler.X86BreakPoint(__FILE__, __LINE__);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_ExceptionOverflow:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.PushImm32("EXC_OV", EXC_OV);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_ExceptionFloatingPoint:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.PushImm32("EXC_FPE", EXC_FPE);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_AddressErrorExceptionRead32:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(1);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &m_TempValue32, "TempValue32");
|
|
m_Assembler.mov(asmjit::x86::eax, asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::DoAddressError), "CRegisters::DoAddressError", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_AddressErrorExceptionRead64:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(1);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &m_TempValue64, "TempValue64");
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::eax, &m_TempValue64 + 4, "TempValue64+4");
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::DoAddressError), "CRegisters::DoAddressError", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_AddressErrorExceptionWrite32:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &m_TempValue32, "TempValue32");
|
|
m_Assembler.mov(asmjit::x86::eax, asmjit::x86::edx);
|
|
m_Assembler.sar(asmjit::x86::eax, 31);
|
|
m_Assembler.push(asmjit::x86::eax);
|
|
m_Assembler.push(asmjit::x86::edx);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::DoAddressError), "CRegisters::DoAddressError", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_IllegalInstruction:
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", InDelaySlot ? PIPELINE_STAGE_JUMP : PIPELINE_STAGE_NORMAL);
|
|
m_Assembler.push(0);
|
|
m_Assembler.PushImm32("EXC_II", EXC_II);
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::TriggerException), "CRegisters::TriggerException", 12);
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
ExitCodeBlock();
|
|
break;
|
|
case ExitReason_Exception:
|
|
m_Assembler.MoveVariableToX86reg(asmjit::x86::edx, &g_System->m_JumpToLocation, "System->m_JumpToLocation");
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", asmjit::x86::edx);
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
if (TargetPC == (uint32_t)-1)
|
|
{
|
|
ExitRegSet.SetBlockCycleCount(0);
|
|
UpdateCounters(ExitRegSet, true, false, false);
|
|
}
|
|
ExitCodeBlock();
|
|
break;
|
|
default:
|
|
WriteTrace(TraceRecompiler, TraceError, "How did you want to exit on reason (%d) ???", reason);
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp CX86RecompilerOps::BaseOffsetAddress(bool UseBaseRegister)
|
|
{
|
|
asmjit::x86::Gp AddressReg;
|
|
if (m_RegWorkingSet.IsMapped(m_Opcode.base))
|
|
{
|
|
if (m_Opcode.offset != 0)
|
|
{
|
|
bool UnProtect = m_RegWorkingSet.GetX86Protected(GetIndexFromX86Reg(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.base)));
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.base);
|
|
AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.lea(AddressReg, asmjit::x86::ptr(m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.base), (int16_t)m_Opcode.offset));
|
|
if (!UnProtect)
|
|
{
|
|
m_RegWorkingSet.UnProtectGPR(m_Opcode.base);
|
|
}
|
|
}
|
|
else if (UseBaseRegister)
|
|
{
|
|
m_RegWorkingSet.ProtectGPR(m_Opcode.base);
|
|
AddressReg = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.base);
|
|
}
|
|
else
|
|
{
|
|
AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.base, false, false);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, m_Opcode.base, false, false);
|
|
m_Assembler.AddConstToX86Reg(AddressReg, (int16_t)m_Opcode.immediate);
|
|
}
|
|
|
|
if (!b32BitCore() && ((m_RegWorkingSet.IsKnown(m_Opcode.base) && m_RegWorkingSet.Is64Bit(m_Opcode.base)) || m_RegWorkingSet.IsUnknown(m_Opcode.base)))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue64, "TempValue64", AddressReg);
|
|
asmjit::x86::Gp AddressRegHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(AddressRegHi, AddressReg);
|
|
m_Assembler.sar(AddressRegHi, 31);
|
|
|
|
if (m_RegWorkingSet.IsConst(m_Opcode.base))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_TempValue64 + 4, "TempValue64 + 4", m_RegWorkingSet.GetMipsRegHi(m_Opcode.base));
|
|
m_Assembler.CompConstToX86reg(AddressRegHi, m_RegWorkingSet.GetMipsRegHi(m_Opcode.base));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(m_Opcode.base))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue64 + 4, "TempValue64 + 4", m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.base));
|
|
m_Assembler.cmp(AddressRegHi, m_RegWorkingSet.GetMipsRegMapHi(m_Opcode.base));
|
|
}
|
|
else
|
|
{
|
|
asmjit::x86::Gp AddressMemoryHi = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveVariableToX86reg(AddressMemoryHi, &m_Reg.m_GPR[m_Opcode.base].W[1], CRegName::GPR_Hi[m_Opcode.base]);
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue64 + 4, "TempValue64 + 4", AddressMemoryHi);
|
|
m_Assembler.cmp(AddressRegHi, AddressMemoryHi);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(AddressMemoryHi), false);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_AddressErrorExceptionRead64, false, &CX86Ops::JneLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(AddressRegHi), false);
|
|
}
|
|
return AddressReg;
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileLoadMemoryValue(asmjit::x86::Gp & AddressReg, const asmjit::x86::Gp & ValueReg, const asmjit::x86::Gp & ValueRegHi, uint8_t ValueSize, bool SignExtend)
|
|
{
|
|
bool UnprotectAddressReg = !AddressReg.isValid();
|
|
if (UnprotectAddressReg)
|
|
{
|
|
if (ValueSize == 8)
|
|
{
|
|
AddressReg = BaseOffsetAddress(false);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint8, "x86TestReadBreakpoint8");
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
AddressReg = BaseOffsetAddress(false);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint16, "x86TestReadBreakpoint16");
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
AddressReg = BaseOffsetAddress(true);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint32, "x86TestReadBreakpoint32");
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
AddressReg = BaseOffsetAddress(true);
|
|
TestReadBreakpoint(AddressReg, (uint32_t)x86TestReadBreakPoint64, "x86TestReadBreakpoint64");
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
if (ValueSize == 16)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue32, "TempValue32", AddressReg);
|
|
m_Assembler.test(AddressReg, 1);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_AddressErrorExceptionRead32, false, &CX86Ops::JneLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue32, "TempValue32", AddressReg);
|
|
m_Assembler.test(AddressReg, 3);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_AddressErrorExceptionRead32, false, &CX86Ops::JneLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue32, "TempValue32", AddressReg);
|
|
m_Assembler.test(AddressReg, 7);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_AddressErrorExceptionRead32, false, &CX86Ops::JneLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
}
|
|
|
|
m_Assembler.mov(TempReg, AddressReg);
|
|
m_Assembler.shr(TempReg, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg, g_MMU->m_MemoryReadMap, "MMU->m_MemoryReadMap", TempReg, CX86Ops::Multip_x4);
|
|
m_Assembler.CompConstToX86reg(TempReg, (uint32_t)-1);
|
|
asmjit::Label JumpFound = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("MemoryReadMap_%X_Found", m_CompilePC).c_str(), JumpFound);
|
|
uint32_t OpsExecuted = m_RegWorkingSet.GetBlockCycleCount();
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.SubConstFromVariable(OpsExecuted, g_NextTimer, "g_NextTimer");
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
if (ValueSize == 32)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)(&m_MMU), AddressOf(&CMipsMemoryVM::LW_VAddr32), "CMipsMemoryVM::LW_VAddr32", 12);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, ExitReason_NormalNoSysCheck, false, &CX86Ops::JeLabel);
|
|
m_Assembler.MoveConstToX86reg(TempReg, (uint32_t)&m_TempValue32);
|
|
m_Assembler.sub(TempReg, AddressReg);
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)(&m_MMU), AddressOf(&CMipsMemoryVM::LH_VAddr32), "CMipsMemoryVM::LH_VAddr32", 12);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, ExitReason_NormalNoSysCheck, false, &CX86Ops::JeLabel);
|
|
m_Assembler.MoveConstToX86reg(TempReg, (uint32_t)&m_TempValue32);
|
|
m_Assembler.sub(TempReg, AddressReg);
|
|
m_Assembler.xor_(AddressReg, 2);
|
|
}
|
|
else if (ValueSize == 8)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.PushImm32("m_TempValue32", (uint32_t)&m_TempValue32);
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_MMU, AddressOf(&CMipsMemoryVM::LB_VAddr32), "CMipsMemoryVM::LB_VAddr32", 12);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, m_RegWorkingSet, ExitReason_NormalNoSysCheck, false, &CX86Ops::JeLabel);
|
|
m_Assembler.MoveConstToX86reg(TempReg, (uint32_t)&m_TempValue32);
|
|
m_Assembler.sub(TempReg, AddressReg);
|
|
m_Assembler.xor_(AddressReg, 3);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.X86BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.AddConstToVariable(g_NextTimer, "g_NextTimer", OpsExecuted);
|
|
}
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpFound);
|
|
|
|
if (m_Instruction.WritesGPR() == 0)
|
|
{
|
|
//ignore load to R0
|
|
}
|
|
else if (ValueSize == 8)
|
|
{
|
|
m_Assembler.xor_(AddressReg, 3);
|
|
if (!ValueReg.isValid())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
else if (SignExtend)
|
|
{
|
|
m_Assembler.movsx(ValueReg, asmjit::x86::byte_ptr(AddressReg, TempReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.movzx(ValueReg, asmjit::x86::byte_ptr(AddressReg, TempReg));
|
|
}
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
m_Assembler.xor_(AddressReg, 2);
|
|
asmjit::x86::Gp MovReg;
|
|
if (ValueReg.isValid())
|
|
{
|
|
MovReg = ValueReg;
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, SignExtend, -1);
|
|
MovReg = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
|
|
if (SignExtend)
|
|
{
|
|
m_Assembler.movsx(MovReg, asmjit::x86::word_ptr(AddressReg, TempReg));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.movzx(MovReg, asmjit::x86::word_ptr(AddressReg, TempReg));
|
|
}
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
asmjit::x86::Gp MovReg;
|
|
if (ValueReg.isValid())
|
|
{
|
|
MovReg = ValueReg;
|
|
}
|
|
else
|
|
{
|
|
m_RegWorkingSet.Map_GPR_32bit(m_Opcode.rt, true, -1);
|
|
MovReg = m_RegWorkingSet.GetMipsRegMapLo(m_Opcode.rt);
|
|
}
|
|
|
|
if (MovReg.isValid())
|
|
{
|
|
m_Assembler.mov(MovReg, asmjit::x86::dword_ptr(AddressReg, TempReg));
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
if (ValueReg.isValid())
|
|
{
|
|
m_Assembler.mov(ValueRegHi, asmjit::x86::dword_ptr(AddressReg, TempReg));
|
|
m_Assembler.mov(ValueReg, asmjit::x86::dword_ptr(AddressReg, TempReg, 0, 4));
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
if (UnprotectAddressReg)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(AddressReg), false);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::CompileStoreMemoryValue(asmjit::x86::Gp AddressReg, const asmjit::x86::Gp & ValueReg, const asmjit::x86::Gp & ValueRegHi, uint64_t Value, uint8_t ValueSize)
|
|
{
|
|
asmjit::Label MemoryWriteDone;
|
|
|
|
if (!AddressReg.isValid())
|
|
{
|
|
AddressReg = BaseOffsetAddress(ValueSize == 32);
|
|
if (ValueSize == 8)
|
|
{
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint8, "x86TestWriteBreakpoint8");
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint16, "x86TestWriteBreakpoint16");
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint32, "x86TestWriteBreakpoint32");
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
TestWriteBreakpoint(AddressReg, (uint32_t)x86TestWriteBreakPoint64, "x86TestWriteBreakpoint64");
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
if (ValueSize != 8)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_TempValue32, "TempValue32", AddressReg);
|
|
if (ValueSize == 16)
|
|
{
|
|
m_Assembler.test(AddressReg, 1);
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
m_Assembler.test(AddressReg, 3);
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
m_Assembler.test(AddressReg, 7);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_AddressErrorExceptionWrite32, false, &CX86Ops::JneLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
}
|
|
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.mov(TempReg, AddressReg);
|
|
m_Assembler.shr(TempReg, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg, g_MMU->m_MemoryWriteMap, "MMU->m_MemoryWriteMap", TempReg, CX86Ops::Multip_x4);
|
|
m_Assembler.CompConstToX86reg(TempReg, (uint32_t)-1);
|
|
asmjit::Label JumpFound = m_Assembler.newLabel();
|
|
m_Assembler.JneLabel(stdstr_f("MemoryWriteMap_%X_Found", m_CompilePC).c_str(), JumpFound);
|
|
|
|
m_Assembler.MoveConst64ToVariable(&g_Reg->m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", m_PipelineStage);
|
|
}
|
|
uint32_t OpsExecuted = m_RegWorkingSet.GetBlockCycleCount();
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.SubConstFromVariable(OpsExecuted, g_NextTimer, "g_NextTimer");
|
|
}
|
|
if (ValueSize == 8)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.push((uint32_t)Value);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.push(ValueReg);
|
|
}
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_MMU, AddressOf(&CMipsMemoryVM::SB_VAddr32), "CMipsMemoryVM::SB_VAddr32", 12);
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.AddConstToVariable(g_NextTimer, "g_NextTimer", OpsExecuted);
|
|
}
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_NormalNoSysCheck, false, &CX86Ops::JeLabel);
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
MemoryWriteDone = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("MemoryWrite_%X_Done:", m_CompilePC).c_str(), MemoryWriteDone);
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.push((uint32_t)(Value & 0xFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.push(ValueReg);
|
|
}
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_MMU, AddressOf(&CMipsMemoryVM::SH_VAddr32), "CMipsMemoryVM::SH_VAddr32", 12);
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.AddConstToVariable(g_NextTimer, "g_NextTimer", OpsExecuted);
|
|
}
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_Exception, false, &CX86Ops::JeLabel);
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
MemoryWriteDone = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("MemoryWrite_%X_Done:", m_CompilePC).c_str(), MemoryWriteDone);
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.push((uint32_t)(Value & 0xFFFFFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.push(ValueReg);
|
|
}
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_MMU, AddressOf(&CMipsMemoryVM::SW_VAddr32), "CMipsMemoryVM::SW_VAddr32", 12);
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.AddConstToVariable(g_NextTimer, "g_NextTimer", OpsExecuted);
|
|
}
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_Exception, false, &CX86Ops::JeLabel);
|
|
m_RegWorkingSet.SetBlockCycleCount(m_RegWorkingSet.GetBlockCycleCount() - g_System->CountPerOp());
|
|
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
MemoryWriteDone = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("MemoryWrite_%X_Done:", m_CompilePC).c_str(), MemoryWriteDone);
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.push((uint32_t)(Value & 0xFFFFFFFF));
|
|
m_Assembler.push((uint32_t)((Value >> 32) & 0xFFFFFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.push(ValueReg);
|
|
m_Assembler.push(ValueRegHi);
|
|
}
|
|
m_Assembler.push(AddressReg);
|
|
m_Assembler.CallThis((uint32_t)&m_MMU, AddressOf(&CMipsMemoryVM::SD_VAddr32), "CMipsMemoryVM::SD_VAddr32", 12);
|
|
if (OpsExecuted != 0)
|
|
{
|
|
m_Assembler.AddConstToVariable(g_NextTimer, "g_NextTimer", OpsExecuted);
|
|
}
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CompileExit(m_CompilePC, m_CompilePC, m_RegWorkingSet, ExitReason_Exception, false, &CX86Ops::JeLabel);
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
MemoryWriteDone = m_Assembler.newLabel();
|
|
m_Assembler.JmpLabel(stdstr_f("MemoryWrite_%X_Done:", m_CompilePC).c_str(), MemoryWriteDone);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.X86BreakPoint(__FILE__, __LINE__);
|
|
m_Assembler.mov(TempReg, AddressReg);
|
|
m_Assembler.shr(TempReg, 12);
|
|
m_Assembler.MoveVariableDispToX86Reg(TempReg, g_MMU->m_TLB_WriteMap, "MMU->TLB_WriteMap", TempReg, CX86Ops::Multip_x4);
|
|
CompileWriteTLBMiss(AddressReg, TempReg);
|
|
m_Assembler.AddConstToX86Reg(TempReg, (uint32_t)m_MMU.Rdram());
|
|
if (m_PipelineStage != PIPELINE_STAGE_NORMAL)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "g_System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
}
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(JumpFound);
|
|
|
|
if (ValueSize == 8)
|
|
{
|
|
m_Assembler.xor_(AddressReg, 3);
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.mov(asmjit::x86::byte_ptr(AddressReg, TempReg), (uint8_t)(Value & 0xFF));
|
|
}
|
|
else if (m_Assembler.Is8BitReg(ValueReg))
|
|
{
|
|
m_Assembler.mov(asmjit::x86::byte_ptr(AddressReg, TempReg), ValueReg.r8Lo());
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
else if (ValueSize == 16)
|
|
{
|
|
m_Assembler.xor_(AddressReg, 2);
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.mov(asmjit::x86::word_ptr(AddressReg, TempReg), (uint16_t)(Value & 0xFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::word_ptr(AddressReg, TempReg), ValueReg.r16());
|
|
}
|
|
}
|
|
else if (ValueSize == 32)
|
|
{
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), (uint32_t)(Value & 0xFFFFFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), ValueReg);
|
|
}
|
|
}
|
|
else if (ValueSize == 64)
|
|
{
|
|
if (!ValueReg.isValid())
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), (uint32_t)(Value >> 32));
|
|
m_Assembler.AddConstToX86Reg(AddressReg, 4);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), (uint32_t)(Value & 0xFFFFFFFF));
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), ValueRegHi);
|
|
m_Assembler.AddConstToX86Reg(AddressReg, 4);
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(AddressReg, TempReg), ValueReg);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
if (MemoryWriteDone.isValid())
|
|
{
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(MemoryWriteDone);
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_Opcode(void (CX86Ops::*Instruction)(void))
|
|
{
|
|
CompileInitFpuOperation(CRegInfo::RoundDefault);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Double);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_64bit);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
m_RegWorkingSet.PrepareFPTopToBe(m_Opcode.fd, m_Opcode.fs, CRegInfo::FPU_Double);
|
|
(m_Assembler.*Instruction)();
|
|
m_Assembler.mov(TempReg, (uint64_t)&m_TempValue64);
|
|
m_Assembler.fpuStoreQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), TempReg, false);
|
|
CompileCheckFPUResult64(TempReg);
|
|
m_RegWorkingSet.SetFPTopAs(m_Opcode.fd);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_D_Opcode(void (CX86Ops::*Instruction)(const asmjit::x86::Mem &))
|
|
{
|
|
CompileInitFpuOperation(CRegInfo::RoundDefault);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.ft, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Double);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_64bit);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.ft, CRegInfo::FPU_Double);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_64bit);
|
|
m_RegWorkingSet.PrepareFPTopToBe(m_Opcode.fd, m_Opcode.fs, CRegInfo::FPU_Double);
|
|
(m_Assembler.*Instruction)(asmjit::x86::qword_ptr(TempReg));
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
m_Assembler.mov(TempReg, (uint64_t)&m_TempValue64);
|
|
m_Assembler.fpuStoreQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), TempReg, false);
|
|
CompileCheckFPUResult64(TempReg);
|
|
m_RegWorkingSet.SetFPTopAs(m_Opcode.fd);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_Opcode(void (CX86Ops::*Instruction)(void))
|
|
{
|
|
CompileInitFpuOperation(CRegInfo::RoundUnknown);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_FloatLow);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_32bit);
|
|
m_RegWorkingSet.PrepareFPTopToBe(m_Opcode.fd, m_Opcode.fs, CRegInfo::FPU_Float);
|
|
(m_Assembler.*Instruction)();
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
CompileCheckFPUResult32(m_Opcode.fd);
|
|
m_RegWorkingSet.SetFPTopAs(m_Opcode.fd);
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_Opcode(void (CX86Ops::*Instruction)(const asmjit::x86::Mem &))
|
|
{
|
|
CompileInitFpuOperation(CRegInfo::RoundDefault);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.ft, CRegInfo::FPU_Any) ||
|
|
m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, CRegInfo::FPU_Float);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_32bit);
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
TempReg = m_RegWorkingSet.FPRValuePointer(m_Opcode.ft, CRegInfo::FPU_Float);
|
|
CompileCheckFPUInput(TempReg, FpuOpSize_32bit);
|
|
m_RegWorkingSet.PrepareFPTopToBe(m_Opcode.fd, m_Opcode.fs, CRegInfo::FPU_Float);
|
|
(m_Assembler.*Instruction)(asmjit::x86::dword_ptr(TempReg));
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(TempReg), false);
|
|
CompileCheckFPUResult32(m_Opcode.fd);
|
|
m_RegWorkingSet.SetFPTopAs(m_Opcode.fd);
|
|
}
|
|
|
|
void CX86RecompilerOps::SB_Const(uint32_t Value, uint32_t VAddr)
|
|
{
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, Value, 8);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, Value, 8);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveConstByteToVariable((PAddr ^ 3) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 3)", PAddr).c_str(), (uint8_t)Value);
|
|
}
|
|
break;
|
|
case 0x04000000:
|
|
if (PAddr < 0x04001000)
|
|
{
|
|
m_Assembler.MoveConstByteToVariable(((PAddr - 0x04000000) ^ 3) + g_MMU->Dmem(), stdstr_f("DMem + (%X ^ 3)", (PAddr - 0x04000000)).c_str(), (uint8_t)Value);
|
|
}
|
|
else if (PAddr < 0x04002000)
|
|
{
|
|
m_Assembler.MoveConstByteToVariable(((PAddr - 0x04001000) ^ 3) + g_MMU->Imem(), stdstr_f("Imem + (%X ^ 3)", (PAddr - 0x04001000)).c_str(), (uint8_t)Value);
|
|
}
|
|
else
|
|
{
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
if (PAddr >= 0x10000000 && PAddr < 0x13F00000)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value << ((3 - (PAddr & 3)) * 8));
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][1], "RomMemoryHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SB_Register(const asmjit::x86::Gp & Reg, uint32_t VAddr)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(Reg), true);
|
|
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 8);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 8);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveX86regByteToVariable((PAddr ^ 3) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 3)", PAddr).c_str(), Reg);
|
|
}
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SH_Const(uint32_t Value, uint32_t VAddr)
|
|
{
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, (uint16_t)Value, 16);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, (uint16_t)Value, 16);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveConstHalfToVariable((PAddr ^ 2) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 2)", PAddr).c_str(), (uint16_t)Value);
|
|
}
|
|
break;
|
|
default:
|
|
if (PAddr >= 0x10000000 && PAddr < 0x13F00000)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value << ((2 - (PAddr & 2)) * 8));
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][1], "RomMemoryHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SH_Register(const asmjit::x86::Gp & Reg, uint32_t VAddr)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(Reg), true);
|
|
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 16);
|
|
}
|
|
else
|
|
{
|
|
uint32_t PAddr;
|
|
if (m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 16);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveX86regHalfToVariable((PAddr ^ 2) + g_MMU->Rdram(), stdstr_f("RDRAM + (%X ^ 2)", PAddr).c_str(), Reg);
|
|
}
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SW_Const(uint32_t Value, uint32_t VAddr)
|
|
{
|
|
asmjit::Label Jump;
|
|
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, Value, 32);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, x86Reg_Unknown, x86Reg_Unknown, Value, 32);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveConstToVariable(PAddr + g_MMU->Rdram(), stdstr_f("RDRAM + %X", PAddr).c_str(), Value);
|
|
}
|
|
break;
|
|
case 0x03F00000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x03F00000: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_CONFIG_REG, "RDRAM_CONFIG_REG", Value); break;
|
|
case 0x03F00004: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_DEVICE_ID_REG, "RDRAM_DEVICE_ID_REG", Value); break;
|
|
case 0x03F00008: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_DELAY_REG, "RDRAM_DELAY_REG", Value); break;
|
|
case 0x03F0000C: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_MODE_REG, "RDRAM_MODE_REG", Value); break;
|
|
case 0x03F00010: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_REF_INTERVAL_REG, "RDRAM_REF_INTERVAL_REG", Value); break;
|
|
case 0x03F00014: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_REF_ROW_REG, "RDRAM_REF_ROW_REG", Value); break;
|
|
case 0x03F00018: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_RAS_INTERVAL_REG, "RDRAM_RAS_INTERVAL_REG", Value); break;
|
|
case 0x03F0001C: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_MIN_INTERVAL_REG, "RDRAM_MIN_INTERVAL_REG", Value); break;
|
|
case 0x03F00020: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_ADDR_SELECT_REG, "RDRAM_ADDR_SELECT_REG", Value); break;
|
|
case 0x03F00024: m_Assembler.MoveConstToVariable(&g_Reg->RDRAM_DEVICE_MANUF_REG, "RDRAM_DEVICE_MANUF_REG", Value); break;
|
|
case 0x03F04004: break;
|
|
case 0x03F08004: break;
|
|
case 0x03F80004: break;
|
|
case 0x03F80008: break;
|
|
case 0x03F8000C: break;
|
|
case 0x03F80014: break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04000000:
|
|
if (PAddr < 0x04001000)
|
|
{
|
|
m_Assembler.MoveConstToVariable((PAddr - 0x04000000) + g_MMU->Dmem(), stdstr_f("DMem + %X", (PAddr - 0x04000000)).c_str(), Value);
|
|
}
|
|
else if (PAddr < 0x04002000)
|
|
{
|
|
m_Assembler.MoveConstToVariable((PAddr - 0x04001000) + g_MMU->Imem(), stdstr_f("Imem + %X", (PAddr - 0x04001000)).c_str(), Value);
|
|
}
|
|
else
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x04040000:
|
|
case 0x04040004:
|
|
case 0x04040008:
|
|
case 0x0404000C:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_SPRegistersHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_SPRegistersHandler)[0][1], "SPRegistersHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04040010:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)g_MMU, AddressOf(&CMipsMemoryVM::SW_PhysicalAddress), "CMipsMemoryVM::SW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x0404001C: m_Assembler.MoveConstToVariable(&g_Reg->SP_SEMAPHORE_REG, "SP_SEMAPHORE_REG", 0); break;
|
|
case 0x04080000: m_Assembler.MoveConstToVariable(&g_Reg->SP_PC_REG, "SP_PC_REG", Value & 0xFFC); break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 0x04100000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x0410000C:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)g_MMU, AddressOf(&CMipsMemoryVM::SW_PhysicalAddress), "CMipsMemoryVM::SW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04300000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04300000:
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_MIPSInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_MIPSInterfaceHandler)[0][1], "MIPSInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
}
|
|
case 0x0430000C:
|
|
{
|
|
uint32_t ModValue;
|
|
ModValue = 0;
|
|
if ((Value & MI_INTR_MASK_CLR_SP) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_SP;
|
|
}
|
|
if ((Value & MI_INTR_MASK_CLR_SI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_SI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_CLR_AI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_AI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_CLR_VI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_VI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_CLR_PI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_PI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_CLR_DP) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_DP;
|
|
}
|
|
if (ModValue != 0)
|
|
{
|
|
m_Assembler.AndConstToVariable(&g_Reg->MI_INTR_MASK_REG, "MI_INTR_MASK_REG", ~ModValue);
|
|
}
|
|
|
|
ModValue = 0;
|
|
if ((Value & MI_INTR_MASK_SET_SP) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_SP;
|
|
}
|
|
if ((Value & MI_INTR_MASK_SET_SI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_SI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_SET_AI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_AI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_SET_VI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_VI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_SET_PI) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_PI;
|
|
}
|
|
if ((Value & MI_INTR_MASK_SET_DP) != 0)
|
|
{
|
|
ModValue |= MI_INTR_MASK_DP;
|
|
}
|
|
if (ModValue != 0)
|
|
{
|
|
m_Assembler.OrConstToVariable(&g_Reg->MI_INTR_MASK_REG, "MI_INTR_MASK_REG", ModValue);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04400000:
|
|
if (GenerateLog() && LogVideoInterface())
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler)[0][1], "VideoInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x04400000:
|
|
if (g_Plugins->Gfx()->ViStatusChanged != nullptr)
|
|
{
|
|
m_Assembler.CompConstToVariable(&g_Reg->VI_STATUS_REG, "VI_STATUS_REG", Value);
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Continue", Jump);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->VI_STATUS_REG, "VI_STATUS_REG", Value);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)g_Plugins->Gfx()->ViStatusChanged, "ViStatusChanged");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
break;
|
|
case 0x04400004: m_Assembler.MoveConstToVariable(&g_Reg->VI_ORIGIN_REG, "VI_ORIGIN_REG", (Value & 0xFFFFFF)); break;
|
|
case 0x04400008:
|
|
if (g_Plugins->Gfx()->ViWidthChanged != nullptr)
|
|
{
|
|
m_Assembler.CompConstToVariable(&g_Reg->VI_WIDTH_REG, "VI_WIDTH_REG", Value);
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Continue", Jump);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->VI_WIDTH_REG, "VI_WIDTH_REG", Value);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)g_Plugins->Gfx()->ViWidthChanged, "ViWidthChanged");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
break;
|
|
case 0x0440000C: m_Assembler.MoveConstToVariable(&g_Reg->VI_INTR_REG, "VI_INTR_REG", Value); break;
|
|
case 0x04400010:
|
|
m_Assembler.AndConstToVariable(&g_Reg->MI_INTR_REG, "MI_INTR_REG", (uint32_t)~MI_INTR_VI);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::CheckInterrupts), "CRegisters::CheckInterrupts", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04400014: m_Assembler.MoveConstToVariable(&g_Reg->VI_BURST_REG, "VI_BURST_REG", Value); break;
|
|
case 0x04400018: m_Assembler.MoveConstToVariable(&g_Reg->VI_V_SYNC_REG, "VI_V_SYNC_REG", Value); break;
|
|
case 0x0440001C: m_Assembler.MoveConstToVariable(&g_Reg->VI_H_SYNC_REG, "VI_H_SYNC_REG", Value); break;
|
|
case 0x04400020: m_Assembler.MoveConstToVariable(&g_Reg->VI_LEAP_REG, "VI_LEAP_REG", Value); break;
|
|
case 0x04400024: m_Assembler.MoveConstToVariable(&g_Reg->VI_H_START_REG, "VI_H_START_REG", Value); break;
|
|
case 0x04400028: m_Assembler.MoveConstToVariable(&g_Reg->VI_V_START_REG, "VI_V_START_REG", Value); break;
|
|
case 0x0440002C: m_Assembler.MoveConstToVariable(&g_Reg->VI_V_BURST_REG, "VI_V_BURST_REG", Value); break;
|
|
case 0x04400030: m_Assembler.MoveConstToVariable(&g_Reg->VI_X_SCALE_REG, "VI_X_SCALE_REG", Value); break;
|
|
case 0x04400034: m_Assembler.MoveConstToVariable(&g_Reg->VI_Y_SCALE_REG, "VI_Y_SCALE_REG", Value); break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 0x04500000:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler)[0][1], "AudioInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04600000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04600000:
|
|
case 0x04600004:
|
|
case 0x04600008:
|
|
case 0x0460000C:
|
|
case 0x04600010:
|
|
if (PAddr == 0x04600008 || PAddr == 0x0460000C)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_PeripheralInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_PeripheralInterfaceHandler)[0][1], "PeripheralInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04600014: m_Assembler.MoveConstToVariable(&g_Reg->PI_DOMAIN1_REG, "PI_DOMAIN1_REG", (Value & 0xFF)); break;
|
|
case 0x04600018: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM1_PWD_REG, "PI_BSD_DOM1_PWD_REG", (Value & 0xFF)); break;
|
|
case 0x0460001C: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM1_PGS_REG, "PI_BSD_DOM1_PGS_REG", (Value & 0xFF)); break;
|
|
case 0x04600020: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM1_RLS_REG, "PI_BSD_DOM1_RLS_REG", (Value & 0xFF)); break;
|
|
case 0x04600024: m_Assembler.MoveConstToVariable(&g_Reg->PI_DOMAIN2_REG, "PI_DOMAIN2_REG", (Value & 0xFF)); break;
|
|
case 0x04600028: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM2_PWD_REG, "PI_BSD_DOM2_PWD_REG", (Value & 0xFF)); break;
|
|
case 0x0460002C: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM2_PGS_REG, "PI_BSD_DOM2_PGS_REG", (Value & 0xFF)); break;
|
|
case 0x04600030: m_Assembler.MoveConstToVariable(&g_Reg->PI_BSD_DOM2_RLS_REG, "PI_BSD_DOM2_RLS_REG", (Value & 0xFF)); break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04700000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04700000: m_Assembler.MoveConstToVariable(&g_Reg->RI_MODE_REG, "RI_MODE_REG", Value); break;
|
|
case 0x04700004: m_Assembler.MoveConstToVariable(&g_Reg->RI_CONFIG_REG, "RI_CONFIG_REG", Value); break;
|
|
case 0x04700008: m_Assembler.MoveConstToVariable(&g_Reg->RI_CURRENT_LOAD_REG, "RI_CURRENT_LOAD_REG", Value); break;
|
|
case 0x0470000C: m_Assembler.MoveConstToVariable(&g_Reg->RI_SELECT_REG, "RI_SELECT_REG", Value); break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04800000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04800000: m_Assembler.MoveConstToVariable(&g_Reg->SI_DRAM_ADDR_REG, "SI_DRAM_ADDR_REG", Value); break;
|
|
case 0x04800004:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->SI_PIF_ADDR_RD64B_REG, "SI_PIF_ADDR_RD64B_REG", Value);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)(&g_MMU->m_PifRamHandler), AddressOf(&PifRamHandler::DMA_READ), "PifRamHandler::DMA_READ", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04800010:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_Assembler.MoveConstToVariable(&g_Reg->SI_PIF_ADDR_WR64B_REG, "SI_PIF_ADDR_WR64B_REG", Value);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)(&g_MMU->m_PifRamHandler), AddressOf(&PifRamHandler::DMA_WRITE), "PifRamHandler::DMA_WRITE", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04800018:
|
|
m_Assembler.AndConstToVariable(&g_Reg->MI_INTR_REG, "MI_INTR_REG", (uint32_t)~MI_INTR_SI);
|
|
m_Assembler.AndConstToVariable(&g_Reg->SI_STATUS_REG, "SI_STATUS_REG", (uint32_t)~SI_STATUS_INTERRUPT);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::CheckInterrupts), "CRegisters::CheckInterrupts", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x05000000:
|
|
// 64DD registers
|
|
if (EnableDisk())
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x05000520:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(&DiskReset), "DiskReset");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case 0x1fc00000:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_PifRamHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_PifRamHandler)[0][1], "PifRamHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (PAddr >= 0x10000000 && PAddr < 0x13F00000)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][1], "RomMemoryHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else
|
|
{
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
UpdateCounters(m_RegWorkingSet, false, true, true);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(Value);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(g_MMU), AddressOf(&CMipsMemoryVM::SW_PhysicalAddress), "CMipsMemoryVM::SW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::SW_Register(const asmjit::x86::Gp & Reg, uint32_t VAddr)
|
|
{
|
|
m_RegWorkingSet.SetX86Protected(GetIndexFromX86Reg(Reg), true);
|
|
|
|
if (VAddr < 0x80000000 || VAddr >= 0xC0000000)
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 32);
|
|
return;
|
|
}
|
|
|
|
uint32_t PAddr;
|
|
if (!m_MMU.VAddrToPAddr(VAddr, PAddr))
|
|
{
|
|
m_CodeBlock.Log("%s\nFailed to translate address: %08X", __FUNCTION__, VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (PAddr & 0xFFF00000)
|
|
{
|
|
case 0x00000000:
|
|
case 0x00100000:
|
|
case 0x00200000:
|
|
case 0x00300000:
|
|
case 0x00400000:
|
|
case 0x00500000:
|
|
case 0x00600000:
|
|
case 0x00700000:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 32);
|
|
}
|
|
else if (PAddr < g_MMU->RdramSize())
|
|
{
|
|
m_Assembler.MoveX86regToVariable(PAddr + g_MMU->Rdram(), stdstr_f("RDRAM + %X", PAddr).c_str(), Reg);
|
|
}
|
|
break;
|
|
case 0x04000000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04040000:
|
|
case 0x04040004:
|
|
case 0x04040008:
|
|
case 0x0404000C:
|
|
case 0x04040010:
|
|
if (PAddr == 0x04040010)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_SPRegistersHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_SPRegistersHandler)[0][1], "SPRegistersHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x0404001C: m_Assembler.MoveConstToVariable(&g_Reg->SP_SEMAPHORE_REG, "SP_SEMAPHORE_REG", 0); break;
|
|
case 0x04080000:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->SP_PC_REG, "SP_PC_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->SP_PC_REG, "SP_PC_REG", 0xFFC);
|
|
break;
|
|
default:
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
|
{
|
|
asmjit::x86::Gp AddressReg = m_RegWorkingSet.Map_TempReg(x86Reg_Unknown, -1, false, false);
|
|
m_Assembler.MoveConstToX86reg(AddressReg, VAddr);
|
|
CompileStoreMemoryValue(AddressReg, Reg, x86Reg_Unknown, 0, 32);
|
|
}
|
|
else if (PAddr < 0x04001000)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(g_MMU->Dmem() + (PAddr - 0x04000000), stdstr_f("DMEM + 0x%X", (PAddr - 0x04000000)).c_str(), Reg);
|
|
}
|
|
else if (PAddr < 0x04002000)
|
|
{
|
|
m_Assembler.MoveX86regToVariable(g_MMU->Imem() + (PAddr - 0x04001000), stdstr_f("IMEM + 0x%X", (PAddr - 0x04001000)).c_str(), Reg);
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log(" should be moving %s in to %08X ?", CX86Ops::x86_Name(Reg), VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 0x04100000:
|
|
if (PAddr == 0x0410000C)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(g_MMU), AddressOf(&CMipsMemoryVM::SW_PhysicalAddress), "CMipsMemoryVM::SW_PhysicalAddress", 12);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04300000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04300000:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_MIPSInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_MIPSInterfaceHandler)[0][1], "MIPSInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x0430000C:
|
|
m_Assembler.MoveX86regToVariable(&CMipsMemoryVM::RegModValue, "CMipsMemoryVM::RegModValue", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)CMipsMemoryVM::ChangeMiIntrMask, "CMipsMemoryVM::ChangeMiIntrMask");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log(" should be moving %s in to %08X ?", CX86Ops::x86_Name(Reg), VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04400000:
|
|
if (GenerateLog() && LogVideoInterface())
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_VideoInterfaceHandler)[0][1], "VideoInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x04400000:
|
|
if (g_Plugins->Gfx()->ViStatusChanged != nullptr)
|
|
{
|
|
asmjit::Label Jump;
|
|
m_Assembler.CompX86regToVariable(Reg, &g_Reg->VI_STATUS_REG, "VI_STATUS_REG");
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Continue", Jump);
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->VI_STATUS_REG, "VI_STATUS_REG", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)g_Plugins->Gfx()->ViStatusChanged, "ViStatusChanged");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
break;
|
|
case 0x04400004:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->VI_ORIGIN_REG, "VI_ORIGIN_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->VI_ORIGIN_REG, "VI_ORIGIN_REG", 0xFFFFFF);
|
|
break;
|
|
case 0x04400008:
|
|
if (g_Plugins->Gfx()->ViWidthChanged != nullptr)
|
|
{
|
|
asmjit::Label Jump;
|
|
m_Assembler.CompX86regToVariable(Reg, &g_Reg->VI_WIDTH_REG, "VI_WIDTH_REG");
|
|
Jump = m_Assembler.newLabel();
|
|
m_Assembler.JeLabel("Continue", Jump);
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->VI_WIDTH_REG, "VI_WIDTH_REG", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc((uint32_t)g_Plugins->Gfx()->ViWidthChanged, "ViWidthChanged");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
m_CodeBlock.Log("");
|
|
m_Assembler.bind(Jump);
|
|
}
|
|
break;
|
|
case 0x0440000C: m_Assembler.MoveX86regToVariable(&g_Reg->VI_INTR_REG, "VI_INTR_REG", Reg); break;
|
|
case 0x04400010:
|
|
m_Assembler.AndConstToVariable(&g_Reg->MI_INTR_REG, "MI_INTR_REG", (uint32_t)~MI_INTR_VI);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::CheckInterrupts), "CRegisters::CheckInterrupts", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04400014: m_Assembler.MoveX86regToVariable(&g_Reg->VI_BURST_REG, "VI_BURST_REG", Reg); break;
|
|
case 0x04400018: m_Assembler.MoveX86regToVariable(&g_Reg->VI_V_SYNC_REG, "VI_V_SYNC_REG", Reg); break;
|
|
case 0x0440001C: m_Assembler.MoveX86regToVariable(&g_Reg->VI_H_SYNC_REG, "VI_H_SYNC_REG", Reg); break;
|
|
case 0x04400020: m_Assembler.MoveX86regToVariable(&g_Reg->VI_LEAP_REG, "VI_LEAP_REG", Reg); break;
|
|
case 0x04400024: m_Assembler.MoveX86regToVariable(&g_Reg->VI_H_START_REG, "VI_H_START_REG", Reg); break;
|
|
case 0x04400028: m_Assembler.MoveX86regToVariable(&g_Reg->VI_V_START_REG, "VI_V_START_REG", Reg); break;
|
|
case 0x0440002C: m_Assembler.MoveX86regToVariable(&g_Reg->VI_V_BURST_REG, "VI_V_BURST_REG", Reg); break;
|
|
case 0x04400030: m_Assembler.MoveX86regToVariable(&g_Reg->VI_X_SCALE_REG, "VI_X_SCALE_REG", Reg); break;
|
|
case 0x04400034: m_Assembler.MoveX86regToVariable(&g_Reg->VI_Y_SCALE_REG, "VI_Y_SCALE_REG", Reg); break;
|
|
default:
|
|
m_CodeBlock.Log(" should be moving %s in to %08X ?", CX86Ops::x86_Name(Reg), VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 0x04500000: // AI registers
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_AudioInterfaceHandler)[0][1], "AudioInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04600000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04600000:
|
|
case 0x04600004:
|
|
case 0x04600008:
|
|
case 0x0460000C:
|
|
case 0x04600010:
|
|
if (PAddr == 0x04600008 || PAddr == 0x0460000C)
|
|
{
|
|
UpdateCounters(m_RegWorkingSet, false, true);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_PeripheralInterfaceHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_PeripheralInterfaceHandler)[0][1], "PeripheralInterfaceHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04600014:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_DOMAIN1_REG, "PI_DOMAIN1_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_DOMAIN1_REG, "PI_DOMAIN1_REG", 0xFF);
|
|
break;
|
|
case 0x04600018:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM1_PWD_REG, "PI_BSD_DOM1_PWD_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM1_PWD_REG, "PI_BSD_DOM1_PWD_REG", 0xFF);
|
|
break;
|
|
case 0x0460001C:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM1_PGS_REG, "PI_BSD_DOM1_PGS_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM1_PGS_REG, "PI_BSD_DOM1_PGS_REG", 0xFF);
|
|
break;
|
|
case 0x04600020:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM1_RLS_REG, "PI_BSD_DOM1_RLS_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM1_RLS_REG, "PI_BSD_DOM1_RLS_REG", 0xFF);
|
|
break;
|
|
case 0x04600024:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_DOMAIN2_REG, "PI_DOMAIN2_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_DOMAIN2_REG, "PI_DOMAIN2_REG", 0xFF);
|
|
break;
|
|
case 0x04600028:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM2_PWD_REG, "PI_BSD_DOM2_PWD_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM2_PWD_REG, "PI_BSD_DOM2_PWD_REG", 0xFF);
|
|
break;
|
|
case 0x0460002C:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM2_PGS_REG, "PI_BSD_DOM2_PGS_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM2_PGS_REG, "PI_BSD_DOM2_PGS_REG", 0xFF);
|
|
break;
|
|
case 0x04600030:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->PI_BSD_DOM2_RLS_REG, "PI_BSD_DOM2_RLS_REG", Reg);
|
|
m_Assembler.AndConstToVariable(&g_Reg->PI_BSD_DOM2_RLS_REG, "PI_BSD_DOM2_RLS_REG", 0xFF);
|
|
break;
|
|
default:
|
|
m_CodeBlock.Log(" should be moving %s in to %08X ?", CX86Ops::x86_Name(Reg), VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04700000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04700000: m_Assembler.MoveX86regToVariable(&g_Reg->RI_MODE_REG, "RI_MODE_REG", Reg); break;
|
|
case 0x04700004: m_Assembler.MoveX86regToVariable(&g_Reg->RI_CONFIG_REG, "RI_CONFIG_REG", Reg); break;
|
|
case 0x0470000C: m_Assembler.MoveX86regToVariable(&g_Reg->RI_SELECT_REG, "RI_SELECT_REG", Reg); break;
|
|
case 0x04700010: m_Assembler.MoveX86regToVariable(&g_Reg->RI_REFRESH_REG, "RI_REFRESH_REG", Reg); break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x04800000:
|
|
switch (PAddr)
|
|
{
|
|
case 0x04800000: m_Assembler.MoveX86regToVariable(&g_Reg->SI_DRAM_ADDR_REG, "SI_DRAM_ADDR_REG", Reg); break;
|
|
case 0x04800004:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->SI_PIF_ADDR_RD64B_REG, "SI_PIF_ADDR_RD64B_REG", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)(&g_MMU->m_PifRamHandler), AddressOf(&PifRamHandler::DMA_READ), "PifRamHandler::DMA_READ", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04800010:
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->SI_PIF_ADDR_WR64B_REG, "SI_PIF_ADDR_WR64B_REG", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)(&g_MMU->m_PifRamHandler), AddressOf(&PifRamHandler::DMA_WRITE), "PifRamHandler::DMA_WRITE", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x04800018:
|
|
m_Assembler.AndConstToVariable(&g_Reg->MI_INTR_REG, "MI_INTR_REG", (uint32_t)~MI_INTR_SI);
|
|
m_Assembler.AndConstToVariable(&g_Reg->SI_STATUS_REG, "SI_STATUS_REG", (uint32_t)~SI_STATUS_INTERRUPT);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_Reg, AddressOf(&CRegisters::CheckInterrupts), "CRegisters::CheckInterrupts", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
break;
|
|
case 0x05000000:
|
|
// 64DD registers
|
|
if (EnableDisk())
|
|
{
|
|
switch (PAddr)
|
|
{
|
|
case 0x05000500: m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_DATA, "ASIC_DATA", Reg); break;
|
|
case 0x05000508:
|
|
{
|
|
// ASIC_CMD
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_CMD, "ASIC_CMD", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(&DiskCommand), "DiskCommand");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
}
|
|
case 0x05000510:
|
|
{
|
|
// ASIC_BM_CTL
|
|
m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_BM_CTL, "ASIC_BM_CTL", Reg);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(&DiskBMControl), "DiskBMControl");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
}
|
|
case 0x05000518:
|
|
break;
|
|
case 0x05000520:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallFunc(AddressOf(&DiskReset), "DiskReset");
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x05000528: m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_HOST_SECBYTE, "ASIC_HOST_SECBYTE", Reg); break;
|
|
case 0x05000530: m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_SEC_BYTE, "ASIC_SEC_BYTE", Reg); break;
|
|
case 0x05000548: m_Assembler.MoveX86regToVariable(&g_Reg->ASIC_TEST_PIN_SEL, "ASIC_TEST_PIN_SEL", Reg); break;
|
|
}
|
|
break;
|
|
}
|
|
case 0x13F00000:
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_ISViewerHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_ISViewerHandler)[0][1], "ISViewerHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
case 0x1FC00000:
|
|
UpdateCounters(m_RegWorkingSet, false, true, false);
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr & 0x1FFFFFFF);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_PifRamHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_PifRamHandler)[0][1], "PifRamHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
break;
|
|
default:
|
|
if (PAddr >= 0x10000000 && PAddr < 0x13F00000)
|
|
{
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.push(0xFFFFFFFF);
|
|
m_Assembler.push(Reg);
|
|
m_Assembler.push(PAddr);
|
|
m_Assembler.CallThis((uint32_t)(MemoryHandler *)&g_MMU->m_RomMemoryHandler, (uint32_t)((long **)(MemoryHandler *)&g_MMU->m_RomMemoryHandler)[0][1], "RomMemoryHandler::Write32", 16);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
else
|
|
{
|
|
m_CodeBlock.Log(" should be moving %s in to %08X ?", CX86Ops::x86_Name(Reg), VAddr);
|
|
if (BreakOnUnhandledMemory())
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CX86RecompilerOps::ResetMemoryStack()
|
|
{
|
|
int32_t MipsReg = 29;
|
|
if (m_RegWorkingSet.IsConst(MipsReg))
|
|
{
|
|
m_Assembler.MoveConstToVariable(&m_Reg.m_GPR[MipsReg].UW[0], CRegName::GPR_Lo[MipsReg], m_RegWorkingSet.GetMipsRegLo(MipsReg));
|
|
}
|
|
else if (m_RegWorkingSet.IsMapped(MipsReg))
|
|
{
|
|
m_Assembler.MoveX86regToVariable(&m_Reg.m_GPR[MipsReg].UW[0], CRegName::GPR_Lo[MipsReg], m_RegWorkingSet.GetMipsRegMapLo(MipsReg));
|
|
}
|
|
|
|
asmjit::x86::Gp MemoryStackReg = m_RegWorkingSet.Get_MemoryStack();
|
|
if (MemoryStackReg.isValid())
|
|
{
|
|
m_CodeBlock.Log(" regcache: unallocate %s from memory stack", CX86Ops::x86_Name(MemoryStackReg));
|
|
m_RegWorkingSet.SetX86Mapped(GetIndexFromX86Reg(MemoryStackReg), CRegInfo::NotMapped);
|
|
}
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
m_Assembler.CallThis((uint32_t)g_Recompiler, AddressOf(&CRecompiler::ResetMemoryStackPos), "CRecompiler::ResetMemoryStackPos", 4);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
}
|
|
|
|
void CX86RecompilerOps::COP1_S_CVT(CRegBase::FPU_ROUND RoundMethod, CRegInfo::FPU_STATE OldFormat, CRegInfo::FPU_STATE NewFormat)
|
|
{
|
|
static uint32_t StatusRegister = 0;
|
|
|
|
m_RegWorkingSet.Map_TempReg(asmjit::x86::eax, 0, false, false);
|
|
CompileInitFpuOperation(RoundMethod);
|
|
if (m_RegWorkingSet.RegInStack(m_Opcode.fs, CRegInfo::FPU_Any) || m_RegWorkingSet.RegInStack(m_Opcode.fd, CRegInfo::FPU_Any))
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
asmjit::x86::Gp fsRegPointer = m_RegWorkingSet.FPRValuePointer(m_Opcode.fs, OldFormat);
|
|
if (OldFormat == CRegInfo::FPU_Float || OldFormat == CRegInfo::FPU_FloatLow)
|
|
{
|
|
CompileCheckFPUInput(fsRegPointer, FpuOpSize_32bit, NewFormat == CRegInfo::FPU_Dword || NewFormat == CRegInfo::FPU_Qword);
|
|
m_Assembler.fpuLoadDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
}
|
|
else if (OldFormat == CRegInfo::FPU_Double)
|
|
{
|
|
CompileCheckFPUInput(fsRegPointer, FpuOpSize_64bit, (NewFormat == CRegInfo::FPU_Dword || NewFormat == CRegInfo::FPU_Qword));
|
|
m_Assembler.fpuLoadQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
}
|
|
else if (OldFormat == CRegInfo::FPU_Dword)
|
|
{
|
|
m_Assembler.fpuLoadIntegerDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
}
|
|
else if (OldFormat == CRegInfo::FPU_Qword)
|
|
{
|
|
m_Assembler.mov(asmjit::x86::eax, asmjit::x86::dword_ptr(fsRegPointer, 4));
|
|
m_Assembler.adc(asmjit::x86::eax, 0x800000);
|
|
m_Assembler.cmp(asmjit::x86::eax, 0xFFFFFF);
|
|
asmjit::Label UnimplementedOperationLabel = m_Assembler.newLabel();
|
|
asmjit::Label ValidValueLabel = m_Assembler.newLabel();
|
|
m_Assembler.JaLabel("ValidValue", UnimplementedOperationLabel);
|
|
m_Assembler.JbLabel("UnimplementedOperationLabel", ValidValueLabel);
|
|
m_Assembler.cmp(asmjit::x86::eax, 0xFFFFFFFF);
|
|
m_Assembler.JbeLabel("ValidValue", ValidValueLabel);
|
|
m_Assembler.bind(UnimplementedOperationLabel);
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
if (ExitRegSet.IsFPStatusRegMapped())
|
|
{
|
|
m_Assembler.or_(m_RegWorkingSet.Map_FPStatusReg(), FPCSR_CE);
|
|
}
|
|
else
|
|
{
|
|
m_Assembler.OrConstToVariable(&m_Reg.m_FPCR[31], "_FPCR[31]", FPCSR_CE);
|
|
}
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit(m_CompilePC, m_CompilePC, ExitRegSet, ExitReason_ExceptionFloatingPoint, false, &CX86Ops::JmpLabel);
|
|
m_Assembler.bind(ValidValueLabel);
|
|
m_Assembler.fpuLoadIntegerQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
if (NewFormat == CRegInfo::FPU_Dword)
|
|
{
|
|
m_Assembler.mov(fsRegPointer, (uint64_t)&m_TempValue32);
|
|
m_Assembler.fpuStoreIntegerDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer, false);
|
|
m_Assembler.fpuLoadIntegerDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fstsw(asmjit::x86::ax);
|
|
m_Assembler.sahf();
|
|
asmjit::Label ExactLabel = m_Assembler.newLabel();
|
|
m_Assembler.JzLabel("Exact", ExactLabel);
|
|
|
|
m_Assembler.stmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
m_Assembler.or_(asmjit::x86::dword_ptr((uint64_t)&StatusRegister), 0x20L);
|
|
m_Assembler.ldmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInvalidException), "R4300iOp::CheckFPUInvalidException", 8);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_Exception, false, &CX86Ops::JnzLabel);
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
m_Assembler.bind(ExactLabel);
|
|
}
|
|
else if (NewFormat == CRegInfo::FPU_Qword)
|
|
{
|
|
m_Assembler.mov(fsRegPointer, (uint64_t)&m_TempValue64);
|
|
m_Assembler.fpuLoadDwordFromStackReg(m_RegWorkingSet.StackTopPos(), asmjit::x86::st0);
|
|
m_Assembler.fpuStoreIntegerQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer, true);
|
|
m_Assembler.fpuLoadIntegerQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer);
|
|
m_Assembler.fpuCompp(m_RegWorkingSet.StackTopPos());
|
|
m_Assembler.fstsw(asmjit::x86::ax);
|
|
m_Assembler.sahf();
|
|
asmjit::Label ExactLabel = m_Assembler.newLabel();
|
|
m_Assembler.JzLabel("Exact", ExactLabel);
|
|
|
|
m_Assembler.stmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
m_Assembler.or_(asmjit::x86::dword_ptr((uint64_t)&StatusRegister), 0x20L);
|
|
m_Assembler.ldmxcsr(asmjit::x86::dword_ptr((uint64_t)&StatusRegister));
|
|
|
|
m_RegWorkingSet.BeforeCallDirect();
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_JUMP);
|
|
}
|
|
m_Assembler.MoveConst64ToVariable(&m_Reg.m_PROGRAM_COUNTER, "PROGRAM_COUNTER", m_CompilePC);
|
|
m_Assembler.CallThis((uint32_t)&g_System->m_OpCodes, AddressOf(&R4300iOp::CheckFPUInvalidException), "R4300iOp::CheckFPUInvalidException", 8);
|
|
m_Assembler.test(asmjit::x86::al, asmjit::x86::al);
|
|
m_RegWorkingSet.AfterCallDirect();
|
|
CRegInfo ExitRegSet = m_RegWorkingSet;
|
|
ExitRegSet.SetBlockCycleCount(ExitRegSet.GetBlockCycleCount() + g_System->CountPerOp());
|
|
CompileExit((uint64_t)-1, (uint64_t)-1, ExitRegSet, ExitReason_Exception, false, &CX86Ops::JnzLabel);
|
|
if (m_PipelineStage == PIPELINE_STAGE_JUMP || m_PipelineStage == PIPELINE_STAGE_DELAY_SLOT)
|
|
{
|
|
m_Assembler.MoveConstToVariable(&g_System->m_PipelineStage, "System->m_PipelineStage", PIPELINE_STAGE_NORMAL);
|
|
}
|
|
m_Assembler.bind(ExactLabel);
|
|
}
|
|
else if (NewFormat == CRegInfo::FPU_Float)
|
|
{
|
|
CompileCheckFPUResult32(m_Opcode.fd);
|
|
}
|
|
else if (NewFormat == CRegInfo::FPU_Double)
|
|
{
|
|
m_Assembler.mov(fsRegPointer, (uint64_t)&m_TempValue64);
|
|
m_Assembler.fpuStoreQwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fsRegPointer, true);
|
|
CompileCheckFPUResult64(fsRegPointer);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
|
|
asmjit::x86::Gp fdRegPointer = m_RegWorkingSet.FPRValuePointer(m_Opcode.fd, CRegInfo::FPU_UnsignedDoubleWord);
|
|
if (NewFormat == CRegInfo::FPU_Float)
|
|
{
|
|
m_Assembler.fpuStoreDwordFromX86Reg(m_RegWorkingSet.StackTopPos(), fdRegPointer, true);
|
|
m_Assembler.and_(asmjit::x86::dword_ptr(fdRegPointer, 4), 0);
|
|
}
|
|
else if (NewFormat == CRegInfo::FPU_Dword)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(fsRegPointer, &m_TempValue32, "m_TempValue32");
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(fdRegPointer), fsRegPointer);
|
|
m_Assembler.and_(asmjit::x86::dword_ptr(fdRegPointer, 4), 0);
|
|
}
|
|
else if (NewFormat == CRegInfo::FPU_Qword || NewFormat == CRegInfo::FPU_Double)
|
|
{
|
|
m_Assembler.MoveVariableToX86reg(fsRegPointer, &m_TempValue64, "m_TempValue64");
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(fdRegPointer), fsRegPointer);
|
|
m_Assembler.MoveVariableToX86reg(fsRegPointer, (void *)(((uint32_t)&m_TempValue64) + 4), "m_TempValue64+4");
|
|
m_Assembler.mov(asmjit::x86::dword_ptr(fdRegPointer, 4), fsRegPointer);
|
|
}
|
|
else
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return;
|
|
}
|
|
}
|
|
#endif |