Play-/Source/BasicBlock.cpp
jpd002 6d375ec8fd Re-enabled Q register pipeline emulation.
Added MAC flags pipeline emulation.
Removed MAC flags checking patches for Atelier Iris.


git-svn-id: http://svn.purei.org/purei/trunk@746 b36208d7-6611-0410-8bec-b1987f11c4a2
2011-04-03 05:25:07 +00:00

128 lines
2.3 KiB
C++

#include "BasicBlock.h"
#include "MemStream.h"
#include "offsetof_def.h"
#include "MipsJitter.h"
#include "Jitter_CodeGenFactory.h"
using namespace Framework;
CBasicBlock::CBasicBlock(CMIPS& context, uint32 begin, uint32 end) :
m_begin(begin),
m_end(end),
m_context(context),
m_function(NULL),
m_selfLoopCount(0),
m_refCount(0),
m_branchHint(NULL)
{
assert(m_end >= m_begin);
}
CBasicBlock::~CBasicBlock()
{
if(m_function != NULL)
{
delete m_function;
m_function = NULL;
}
}
void CBasicBlock::Compile()
{
CMemStream stream;
{
static CMipsJitter* jitter = NULL;
if(jitter == NULL)
{
Jitter::CCodeGen* codeGen = Jitter::CreateCodeGen();
jitter = new CMipsJitter(codeGen);
for(unsigned int i = 0; i < 4; i++)
{
jitter->SetVariableAsConstant(
offsetof(CMIPS, m_State.nGPR[CMIPS::R0].nV[i]),
0
);
}
}
jitter->SetStream(&stream);
jitter->Begin();
CompileRange(jitter);
// codeGen.DumpVariables(0);
// codeGen.EndQuota();
jitter->End();
}
m_function = new CMemoryFunction(stream.GetBuffer(), stream.GetSize());
}
void CBasicBlock::CompileRange(CMipsJitter* jitter)
{
for(uint32 address = m_begin; address <= m_end; address += 4)
{
m_context.m_pArch->CompileInstruction(
address,
jitter,
&m_context);
//Sanity check
assert(jitter->IsStackEmpty());
}
}
unsigned int CBasicBlock::Execute()
{
(*m_function)(&m_context);
if((m_context.m_State.nGPR[CMIPS::RA].nV0 & 3) != 0)
{
assert(0);
}
if(m_context.m_State.nDelayedJumpAddr != MIPS_INVALID_PC)
{
m_context.m_State.nPC = m_context.m_State.nDelayedJumpAddr;
m_context.m_State.nDelayedJumpAddr = MIPS_INVALID_PC;
}
else
{
m_context.m_State.nPC = m_end + 4;
}
return ((m_end - m_begin) / 4) + 1;
}
uint32 CBasicBlock::GetBeginAddress() const
{
return m_begin;
}
uint32 CBasicBlock::GetEndAddress() const
{
return m_end;
}
bool CBasicBlock::IsCompiled() const
{
return m_function != NULL;
}
unsigned int CBasicBlock::GetSelfLoopCount() const
{
return m_selfLoopCount;
}
void CBasicBlock::SetSelfLoopCount(unsigned int selfLoopCount)
{
m_selfLoopCount = selfLoopCount;
}
BasicBlockPtr CBasicBlock::GetBranchHint() const
{
return m_branchHint;
}
void CBasicBlock::SetBranchHint(const BasicBlockPtr& branchHint)
{
m_branchHint = branchHint;
}