Play-/Source/iop/Iop_Spu2.cpp
2018-04-30 21:01:23 +01:00

125 lines
3.0 KiB
C++

#include <assert.h>
#include "make_unique.h"
#include "Iop_Spu2.h"
#include "../Log.h"
#include "placeholder_def.h"
#define LOG_NAME ("iop_spu2")
using namespace Iop;
using namespace Iop::Spu2;
CSpu2::CSpu2(CSpuBase& spuBase0, CSpuBase& spuBase1)
{
for(unsigned int i = 0; i < CORE_NUM; i++)
{
CSpuBase& base(i == 0 ? spuBase0 : spuBase1);
m_core[i] = std::make_unique<CCore>(i, base);
}
m_readDispatchInfo.global = std::bind(&CSpu2::ReadRegisterImpl, this, std::placeholders::_1, std::placeholders::_2);
m_writeDispatchInfo.global = std::bind(&CSpu2::WriteRegisterImpl, this, std::placeholders::_1, std::placeholders::_2);
for(unsigned int i = 0; i < CORE_NUM; i++)
{
m_readDispatchInfo.core[i] = std::bind(&CCore::ReadRegister, m_core[i].get(), std::placeholders::_1, std::placeholders::_2);
m_writeDispatchInfo.core[i] = std::bind(&CCore::WriteRegister, m_core[i].get(), std::placeholders::_1, std::placeholders::_2);
}
}
CSpu2::~CSpu2()
{
}
void CSpu2::Reset()
{
for(unsigned int i = 0; i < CORE_NUM; i++)
{
m_core[i]->Reset();
}
}
CCore* CSpu2::GetCore(unsigned int coreId)
{
assert(coreId < CORE_NUM);
if(coreId >= CORE_NUM) return nullptr;
return m_core[coreId].get();
}
uint32 CSpu2::ReadRegister(uint32 address)
{
return ProcessRegisterAccess(m_readDispatchInfo, address, 0);
}
uint32 CSpu2::WriteRegister(uint32 address, uint32 value)
{
return ProcessRegisterAccess(m_writeDispatchInfo, address, value);
}
uint32 CSpu2::ProcessRegisterAccess(const REGISTER_DISPATCH_INFO& dispatchInfo, uint32 address, uint32 value)
{
uint32 tmpAddress = address - REGS_BEGIN;
if(tmpAddress < 0x760)
{
unsigned int coreId = (tmpAddress & 0x400) ? 1 : 0;
address &= ~0x400;
return dispatchInfo.core[coreId](address, value);
}
else if(tmpAddress < 0x7B0)
{
unsigned int coreId = (tmpAddress - 0x760) / 40;
address -= coreId * 40;
return dispatchInfo.core[coreId](address, value);
}
return dispatchInfo.global(address, value);
}
uint32 CSpu2::ReadRegisterImpl(uint32 address, uint32 value)
{
uint32 result = 0;
switch(address)
{
case C_IRQINFO:
for(unsigned int i = 0; i < CORE_NUM; i++)
{
auto& core = m_core[i];
if(core->GetSpuBase().GetIrqPending())
{
result |= (1 << (i + 2));
core->GetSpuBase().ClearIrqPending();
}
}
break;
}
LogRead(address);
return result;
}
uint32 CSpu2::WriteRegisterImpl(uint32 address, uint32 value)
{
LogWrite(address, value);
return 0;
}
void CSpu2::LogRead(uint32 address)
{
switch(address)
{
case C_IRQINFO:
CLog::GetInstance().Print(LOG_NAME, " = C_IRQINFO\r\n");
break;
default:
CLog::GetInstance().Print(LOG_NAME, "Read an unknown register 0x%08X.\r\n", address);
break;
}
}
void CSpu2::LogWrite(uint32 address, uint32 value)
{
switch(address)
{
default:
CLog::GetInstance().Print(LOG_NAME, "Wrote 0x%08X to unknown register 0x%08X.\r\n", value, address);
break;
}
}