2013-07-12 14:42:17 +02:00
|
|
|
#include "stdafx.h"
|
2014-06-17 17:44:03 +02:00
|
|
|
#include "Utilities/Log.h"
|
2014-06-02 19:27:24 +02:00
|
|
|
#include "Emu/Memory/Memory.h"
|
|
|
|
|
#include "Emu/System.h"
|
2014-06-25 00:38:34 +02:00
|
|
|
#include "Emu/SysCalls/lv2/sys_lwmutex.h"
|
|
|
|
|
#include "Emu/SysCalls/lv2/sys_event.h"
|
2014-06-02 19:27:24 +02:00
|
|
|
|
2013-07-12 14:42:17 +02:00
|
|
|
#include "Emu/Cell/RawSPUThread.h"
|
|
|
|
|
|
2014-07-16 14:06:58 +02:00
|
|
|
RawSPUThread::RawSPUThread(CPUThreadType type)
|
2013-11-03 20:23:16 +01:00
|
|
|
: SPUThread(type)
|
2014-01-19 04:14:11 +01:00
|
|
|
, MemoryBlock()
|
2013-07-12 14:42:17 +02:00
|
|
|
{
|
2014-07-16 14:06:58 +02:00
|
|
|
m_index = Memory.InitRawSPU(this);
|
2013-07-12 14:42:17 +02:00
|
|
|
Reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RawSPUThread::~RawSPUThread()
|
|
|
|
|
{
|
2014-07-07 19:22:36 +02:00
|
|
|
Memory.CloseRawSPU(this, m_index);
|
2013-11-03 20:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool RawSPUThread::Read32(const u64 addr, u32* value)
|
|
|
|
|
{
|
|
|
|
|
if(addr < GetStartAddr() + RAW_SPU_PROB_OFFSET)
|
|
|
|
|
{
|
|
|
|
|
return MemoryBlock::Read32(addr, value);
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-10 02:13:04 +02:00
|
|
|
const u32 offset = addr - GetStartAddr() - RAW_SPU_PROB_OFFSET;
|
|
|
|
|
|
2013-11-03 20:23:16 +01:00
|
|
|
switch(offset)
|
|
|
|
|
{
|
2014-07-10 02:13:04 +02:00
|
|
|
case MFC_LSA_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(MFC_LSA)", m_index);
|
|
|
|
|
*value = MFC2.LSA.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_EAH_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(MFC_EAH)", m_index);
|
|
|
|
|
*value = MFC2.EAH.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_EAL_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(MFC_EAL)", m_index);
|
|
|
|
|
*value = MFC2.EAL.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_Size_Tag_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(MFC_Size_Tag)", m_index);
|
|
|
|
|
*value = MFC2.Size_Tag.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_CMDStatus_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(MFC_CMDStatus)", m_index);
|
|
|
|
|
*value = MFC2.CMDStatus.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 15:25:38 +02:00
|
|
|
case MFC_QStatus_offs:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Read32(MFC_QStatus)", m_index);
|
2014-07-10 02:13:04 +02:00
|
|
|
*value = MFC2.QStatus.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case Prxy_QueryType_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Read32(Prxy_QueryType)", m_index);
|
|
|
|
|
*value = Prxy.QueryType.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case Prxy_QueryMask_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Read32(Prxy_QueryMask)", m_index);
|
|
|
|
|
*value = Prxy.QueryMask.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case Prxy_TagStatus_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Read32(Prxy_TagStatus)", m_index);
|
|
|
|
|
*value = Prxy.TagStatus.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 15:25:38 +02:00
|
|
|
case SPU_Out_MBox_offs:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
|
|
|
|
// if Out_MBox is empty, the result is undefined
|
|
|
|
|
SPU.Out_MBox.PopUncond(*value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_In_MBox_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(SPU_In_MBox)", m_index);
|
2014-07-12 09:02:39 +02:00
|
|
|
while (!SPU.In_MBox.Pop(*value) && !Emu.IsStopped())
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
2014-07-10 02:13:04 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_MBox_Status_offs:
|
|
|
|
|
{
|
|
|
|
|
//LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Read32(SPU_MBox_Status)", m_index);
|
2013-12-07 23:52:41 +01:00
|
|
|
//SPU.MBox_Status.SetValue(SPU.Out_MBox.GetCount() ? SPU.MBox_Status.GetValue() | 1 : SPU.MBox_Status.GetValue() & ~1);
|
|
|
|
|
SPU.MBox_Status.SetValue((SPU.Out_MBox.GetCount() & 0xff) | (SPU.In_MBox.GetFreeCount() << 8));
|
2013-11-23 19:50:54 +01:00
|
|
|
*value = SPU.MBox_Status.GetValue();
|
|
|
|
|
break;
|
2014-07-10 02:13:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_RunCntl_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(SPU_RunCntl)", m_index);
|
|
|
|
|
*value = (u32)IsRunning();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-23 03:03:16 +02:00
|
|
|
case SPU_Status_offs:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
2014-06-23 03:03:16 +02:00
|
|
|
*value = SPU.Status.GetValue();
|
|
|
|
|
break;
|
2014-07-10 02:13:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_NPC_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(SPU_NPC)", m_index);
|
|
|
|
|
*value = SPU.NPC.GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_RdSigNotify1_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(SPU_RdSigNotify1)", m_index);
|
|
|
|
|
*value = SPU.SNR[0].GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_RdSigNotify2_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(SPU_RdSigNotify2)", m_index);
|
|
|
|
|
*value = SPU.SNR[1].GetValue();
|
|
|
|
|
break;
|
|
|
|
|
}
|
2013-11-03 20:23:16 +01:00
|
|
|
|
|
|
|
|
default:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Read32(0x%x)", m_index, offset);
|
2013-11-03 20:23:16 +01:00
|
|
|
Emu.Pause();
|
2014-07-10 02:13:04 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
2013-11-03 20:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool RawSPUThread::Write32(const u64 addr, const u32 value)
|
|
|
|
|
{
|
|
|
|
|
if(addr < GetStartAddr() + RAW_SPU_PROB_OFFSET)
|
|
|
|
|
{
|
|
|
|
|
return MemoryBlock::Write32(addr, value);
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-10 02:13:04 +02:00
|
|
|
const u32 offset = addr - GetStartAddr() - RAW_SPU_PROB_OFFSET;
|
2013-11-03 20:23:16 +01:00
|
|
|
|
|
|
|
|
switch(offset)
|
|
|
|
|
{
|
2014-07-10 02:13:04 +02:00
|
|
|
case MFC_LSA_offs:
|
|
|
|
|
{
|
|
|
|
|
MFC2.LSA.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_EAH_offs:
|
|
|
|
|
{
|
|
|
|
|
MFC2.EAH.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_EAL_offs:
|
|
|
|
|
{
|
|
|
|
|
MFC2.EAL.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_Size_Tag_offs:
|
|
|
|
|
{
|
|
|
|
|
MFC2.Size_Tag.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-03 20:23:16 +01:00
|
|
|
case MFC_CMDStatus_offs:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
2013-12-10 23:58:11 +01:00
|
|
|
MFC2.CMDStatus.SetValue(value);
|
|
|
|
|
EnqMfcCmd(MFC2);
|
2014-07-10 02:13:04 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case MFC_QStatus_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Write32(MFC_QStatus, 0x%x)", m_index, value);
|
|
|
|
|
//MFC2.QStatus.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-03 20:23:16 +01:00
|
|
|
case Prxy_QueryType_offs:
|
|
|
|
|
{
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(Prxy_QueryType, 0x%x)", m_index, value);
|
2013-11-03 20:23:16 +01:00
|
|
|
Prxy.QueryType.SetValue(value);
|
|
|
|
|
|
|
|
|
|
switch(value)
|
|
|
|
|
{
|
|
|
|
|
case 2:
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Prxy Query Immediate.", m_index);
|
2013-11-03 20:23:16 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Unknown Prxy Query Type. (prxy_query=0x%x)", m_index, value);
|
2013-11-03 20:23:16 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Prxy.QueryType.SetValue(0);
|
2013-12-10 23:58:11 +01:00
|
|
|
MFC2.QStatus.SetValue(Prxy.QueryMask.GetValue());
|
2014-07-10 02:13:04 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case Prxy_QueryMask_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(Prxy_QueryMask, 0x%x)", m_index, value);
|
|
|
|
|
Prxy.QueryMask.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case Prxy_TagStatus_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(Prxy_TagStatus, 0x%x)", m_index, value);
|
|
|
|
|
Prxy.TagStatus.SetValue(value);
|
|
|
|
|
break;
|
2013-11-03 20:23:16 +01:00
|
|
|
}
|
2014-07-10 02:13:04 +02:00
|
|
|
|
|
|
|
|
case SPU_Out_MBox_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_Out_MBox, 0x%x)", m_index, value);
|
2014-07-12 09:02:39 +02:00
|
|
|
while (!SPU.Out_MBox.Push(value) && !Emu.IsStopped())
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
2014-07-10 02:13:04 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 15:25:38 +02:00
|
|
|
case SPU_In_MBox_offs:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
|
|
|
|
// if In_MBox is already full, the last message is overwritten
|
|
|
|
|
SPU.In_MBox.PushUncond(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_MBox_Status_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_MBox_Status, 0x%x)", m_index, value);
|
|
|
|
|
SPU.MBox_Status.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-07 19:22:36 +02:00
|
|
|
case SPU_RunCntl_offs:
|
|
|
|
|
{
|
|
|
|
|
if (value == SPU_RUNCNTL_RUNNABLE)
|
|
|
|
|
{
|
|
|
|
|
SPU.Status.SetValue(SPU_STATUS_RUNNING);
|
|
|
|
|
Exec();
|
|
|
|
|
}
|
|
|
|
|
else if (value == SPU_RUNCNTL_STOP)
|
|
|
|
|
{
|
|
|
|
|
SPU.Status.SetValue(SPU_STATUS_STOPPED);
|
|
|
|
|
Stop();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Write32(SPU_RunCtrl, 0x%x): unknown value", m_index, value);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2014-07-10 02:13:04 +02:00
|
|
|
|
|
|
|
|
case SPU_Status_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_Status, 0x%x)", m_index, value);
|
|
|
|
|
SPU.Status.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_NPC_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_NPC, 0x%x)", m_index, value);
|
|
|
|
|
SPU.NPC.SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_RdSigNotify1_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_RdSigNotify1, 0x%x)", m_index, value);
|
|
|
|
|
SPU.SNR[0].SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case SPU_RdSigNotify2_offs:
|
|
|
|
|
{
|
|
|
|
|
LOG_WARNING(Log::SPU, "RawSPUThread[%d]: Write32(SPU_RdSigNotify2, 0x%x)", m_index, value);
|
|
|
|
|
SPU.SNR[1].SetValue(value);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2013-11-03 20:23:16 +01:00
|
|
|
|
|
|
|
|
default:
|
2014-07-10 02:13:04 +02:00
|
|
|
{
|
2014-06-27 15:26:46 +02:00
|
|
|
LOG_ERROR(Log::SPU, "RawSPUThread[%d]: Write32(0x%x, 0x%x)", m_index, offset, value);
|
2013-11-03 20:23:16 +01:00
|
|
|
Emu.Pause();
|
2014-07-10 02:13:04 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2013-11-03 20:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2013-07-12 14:42:17 +02:00
|
|
|
void RawSPUThread::InitRegs()
|
|
|
|
|
{
|
2013-11-23 19:50:54 +01:00
|
|
|
dmac.ls_offset = m_offset = GetStartAddr() + RAW_SPU_LS_OFFSET;
|
2013-11-03 20:23:16 +01:00
|
|
|
SPUThread::InitRegs();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u32 RawSPUThread::GetIndex() const
|
|
|
|
|
{
|
|
|
|
|
return m_index;
|
2013-07-12 14:42:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RawSPUThread::Task()
|
|
|
|
|
{
|
2014-07-07 19:22:36 +02:00
|
|
|
PC = SPU.NPC.GetValue();
|
2013-07-12 14:42:17 +02:00
|
|
|
|
2014-07-16 14:06:58 +02:00
|
|
|
SPUThread::Task();
|
2013-07-12 14:42:17 +02:00
|
|
|
|
2014-07-07 19:22:36 +02:00
|
|
|
SPU.NPC.SetValue(PC);
|
2013-11-03 20:23:16 +01:00
|
|
|
}
|