Play-/Source/iop/Iop_Thbase.cpp
Jean-Philip Desjardins 0af06179bb Add stub for GetThreadmanIdList.
Prevent issues with undefined values. This is used by Tekken 5, but the results are unused (most likely left-over debugging code).
2021-09-01 09:31:31 -04:00

467 lines
13 KiB
C++

#include "Iop_Thbase.h"
#include "IopBios.h"
#include "../Log.h"
using namespace Iop;
#define LOG_NAME ("iop_thbase")
#define FUNCTION_CREATETHREAD "CreateThread"
#define FUNCTION_STARTTHREAD "StartThread"
#define FUNCTION_STARTTHREADARGS "StartThreadArgs"
#define FUNCTION_DELETETHREAD "DeleteThread"
#define FUNCTION_EXITTHREAD "ExitThread"
#define FUNCTION_TERMINATETHREAD "TerminateThread"
#define FUNCTION_CHANGETHREADPRIORITY "ChangeThreadPriority"
#define FUNCTION_ROTATETHREADREADYQUEUE "RotateThreadReadyQueue"
#define FUNCTION_RELEASEWAITTHREAD "ReleaseWaitThread"
#define FUNCTION_IRELEASEWAITTHREAD "iReleaseWaitThread"
#define FUNCTION_GETTHREADID "GetThreadId"
#define FUNCTION_REFERTHREADSTATUS "ReferThreadStatus"
#define FUNCTION_IREFERTHREADSTATUS "iReferThreadStatus"
#define FUNCTION_SLEEPTHREAD "SleepThread"
#define FUNCTION_WAKEUPTHREAD "WakeupThread"
#define FUNCTION_IWAKEUPTHREAD "iWakeupThread"
#define FUNCTION_CANCELWAKEUPTHREAD "CancelWakeupThread"
#define FUNCTION_ICANCELWAKEUPTHREAD "iCancelWakeupThread"
#define FUNCTION_DELAYTHREAD "DelayThread"
#define FUNCTION_GETSYSTEMTIME "GetSystemTime"
#define FUNCTION_GETSYSTEMTIMELOW "GetSystemTimeLow"
#define FUNCTION_SETALARM "SetAlarm"
#define FUNCTION_CANCELALARM "CancelAlarm"
#define FUNCTION_USECTOSYSCLOCK "USecToSysClock"
#define FUNCTION_SYSCLOCKTOUSEC "SysClockToUSec"
#define FUNCTION_GETCURRENTTHREADPRIORITY "GetCurrentThreadPriority"
#define FUNCTION_GETTHREADMANIDLIST "GetThreadManIdList"
CThbase::CThbase(CIopBios& bios, uint8* ram)
: m_ram(ram)
, m_bios(bios)
{
}
std::string CThbase::GetId() const
{
return "thbase";
}
std::string CThbase::GetFunctionName(unsigned int functionId) const
{
switch(functionId)
{
case 4:
return FUNCTION_CREATETHREAD;
break;
case 5:
return FUNCTION_DELETETHREAD;
break;
case 6:
return FUNCTION_STARTTHREAD;
break;
case 7:
return FUNCTION_STARTTHREADARGS;
break;
case 8:
return FUNCTION_EXITTHREAD;
break;
case 10:
return FUNCTION_TERMINATETHREAD;
break;
case 14:
return FUNCTION_CHANGETHREADPRIORITY;
break;
case 16:
return FUNCTION_ROTATETHREADREADYQUEUE;
break;
case 18:
return FUNCTION_RELEASEWAITTHREAD;
break;
case 19:
return FUNCTION_IRELEASEWAITTHREAD;
break;
case 20:
return FUNCTION_GETTHREADID;
break;
case 22:
return FUNCTION_REFERTHREADSTATUS;
break;
case 23:
return FUNCTION_IREFERTHREADSTATUS;
break;
case 24:
return FUNCTION_SLEEPTHREAD;
break;
case 25:
return FUNCTION_WAKEUPTHREAD;
break;
case 26:
return FUNCTION_IWAKEUPTHREAD;
break;
case 27:
return FUNCTION_CANCELWAKEUPTHREAD;
break;
case 28:
return FUNCTION_ICANCELWAKEUPTHREAD;
break;
case 33:
return FUNCTION_DELAYTHREAD;
break;
case 34:
return FUNCTION_GETSYSTEMTIME;
break;
case 35:
return FUNCTION_SETALARM;
break;
case 37:
return FUNCTION_CANCELALARM;
break;
case 39:
return FUNCTION_USECTOSYSCLOCK;
break;
case 40:
return FUNCTION_SYSCLOCKTOUSEC;
break;
case 42:
return FUNCTION_GETCURRENTTHREADPRIORITY;
break;
case 43:
return FUNCTION_GETSYSTEMTIMELOW;
break;
case 47:
return FUNCTION_GETTHREADMANIDLIST;
break;
default:
return "unknown";
break;
}
}
void CThbase::Invoke(CMIPS& context, unsigned int functionId)
{
switch(functionId)
{
case 4:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(CreateThread(
reinterpret_cast<THREAD*>(&m_ram[context.m_State.nGPR[CMIPS::A0].nV0])));
break;
case 5:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(DeleteThread(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 6:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(StartThread(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0));
break;
case 7:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(StartThreadArgs(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0,
context.m_State.nGPR[CMIPS::A2].nV0));
break;
case 8:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(ExitThread());
break;
case 10:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(TerminateThread(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 14:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(ChangeThreadPriority(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0));
break;
case 16:
context.m_State.nGPR[CMIPS::V0].nD0 = RotateThreadReadyQueue(
context.m_State.nGPR[CMIPS::A0].nV0);
break;
case 18:
context.m_State.nGPR[CMIPS::V0].nD0 = ReleaseWaitThread(
context.m_State.nGPR[CMIPS::A0].nV0);
break;
case 19:
context.m_State.nGPR[CMIPS::V0].nD0 = iReleaseWaitThread(
context.m_State.nGPR[CMIPS::A0].nV0);
break;
case 20:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(GetThreadId());
break;
case 22:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(ReferThreadStatus(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0));
break;
case 23:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(iReferThreadStatus(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0));
break;
case 24:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(SleepThread());
break;
case 25:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(WakeupThread(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 26:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(iWakeupThread(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 27:
context.m_State.nGPR[CMIPS::V0].nD0 = CancelWakeupThread(
context.m_State.nGPR[CMIPS::A0].nV0);
break;
case 28:
context.m_State.nGPR[CMIPS::V0].nD0 = iCancelWakeupThread(
context.m_State.nGPR[CMIPS::A0].nV0);
break;
case 33:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(DelayThread(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 34:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(GetSystemTime(
context.m_State.nGPR[CMIPS::A0].nV0));
break;
case 35:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(SetAlarm(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0,
context.m_State.nGPR[CMIPS::A2].nV0));
break;
case 37:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(CancelAlarm(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0));
break;
case 39:
USecToSysClock(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0);
break;
case 40:
SysClockToUSec(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0,
context.m_State.nGPR[CMIPS::A2].nV0);
break;
case 42:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(GetCurrentThreadPriority());
break;
case 43:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(GetSystemTimeLow());
break;
case 47:
context.m_State.nGPR[CMIPS::V0].nD0 = static_cast<int32>(GetThreadmanIdList(
context.m_State.nGPR[CMIPS::A0].nV0,
context.m_State.nGPR[CMIPS::A1].nV0,
context.m_State.nGPR[CMIPS::A2].nV0,
context.m_State.nGPR[CMIPS::A3].nV0));
break;
default:
CLog::GetInstance().Warn(LOG_NAME, "Unknown function (%d) called at (%08X).\r\n", functionId, context.m_State.nPC);
break;
}
}
uint32 CThbase::CreateThread(const THREAD* thread)
{
return m_bios.CreateThread(thread->threadProc, thread->priority, thread->stackSize, thread->options, thread->attributes);
}
uint32 CThbase::DeleteThread(uint32 threadId)
{
return m_bios.DeleteThread(threadId);
}
int32 CThbase::StartThread(uint32 threadId, uint32 param)
{
return m_bios.StartThread(threadId, param);
}
int32 CThbase::StartThreadArgs(uint32 threadId, uint32 args, uint32 argpPtr)
{
return m_bios.StartThreadArgs(threadId, args, argpPtr);
}
uint32 CThbase::ExitThread()
{
m_bios.ExitThread();
return 0;
}
uint32 CThbase::TerminateThread(uint32 threadId)
{
return m_bios.TerminateThread(threadId);
}
uint32 CThbase::ChangeThreadPriority(uint32 threadId, uint32 newPrio)
{
return m_bios.ChangeThreadPriority(threadId, newPrio);
}
int32 CThbase::RotateThreadReadyQueue(uint32 priority)
{
return m_bios.RotateThreadReadyQueue(priority);
}
int32 CThbase::ReleaseWaitThread(uint32 threadId)
{
return m_bios.ReleaseWaitThread(threadId, false);
}
int32 CThbase::iReleaseWaitThread(uint32 threadId)
{
return m_bios.ReleaseWaitThread(threadId, true);
}
uint32 CThbase::GetThreadId()
{
return m_bios.GetCurrentThreadId();
}
uint32 CThbase::ReferThreadStatus(uint32 threadId, uint32 statusPtr)
{
return m_bios.ReferThreadStatus(threadId, statusPtr, false);
}
uint32 CThbase::iReferThreadStatus(uint32 threadId, uint32 statusPtr)
{
return m_bios.ReferThreadStatus(threadId, statusPtr, true);
}
uint32 CThbase::SleepThread()
{
return m_bios.SleepThread();
}
uint32 CThbase::WakeupThread(uint32 threadId)
{
return m_bios.WakeupThread(threadId, false);
}
uint32 CThbase::iWakeupThread(uint32 threadId)
{
return m_bios.WakeupThread(threadId, true);
}
int32 CThbase::CancelWakeupThread(uint32 threadId)
{
return m_bios.CancelWakeupThread(threadId, false);
}
int32 CThbase::iCancelWakeupThread(uint32 threadId)
{
return m_bios.CancelWakeupThread(threadId, true);
}
uint32 CThbase::DelayThread(uint32 delay)
{
return m_bios.DelayThread(delay);
}
uint32 CThbase::GetSystemTime(uint32 resultPtr)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : " FUNCTION_GETSYSTEMTIME "(resultPtr = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), resultPtr);
#endif
uint64* result = nullptr;
if(resultPtr != 0)
{
result = reinterpret_cast<uint64*>(&m_ram[resultPtr]);
}
if(result)
{
(*result) = m_bios.GetCurrentTime();
}
return 1;
}
uint32 CThbase::GetSystemTimeLow()
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : " FUNCTION_GETSYSTEMTIMELOW "();\r\n",
m_bios.GetCurrentThreadIdRaw());
#endif
return static_cast<uint32>(m_bios.GetCurrentTime());
}
uint32 CThbase::SetAlarm(uint32 timePtr, uint32 alarmFunction, uint32 param)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : SetAlarm(timePtr = 0x%08X, alarmFunction = 0x%08X, param = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), timePtr, alarmFunction, param);
#endif
return m_bios.SetAlarm(timePtr, alarmFunction, param);
}
uint32 CThbase::CancelAlarm(uint32 alarmFunction, uint32 param)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : CancelAlarm(alarmFunction = 0x%08X, param = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), alarmFunction, param);
#endif
return m_bios.CancelAlarm(alarmFunction, param);
}
void CThbase::USecToSysClock(uint32 usec, uint32 timePtr)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : USecToSysClock(usec = 0x%08X, timePtr = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), usec, timePtr);
#endif
uint64* time = (timePtr != 0) ? reinterpret_cast<uint64*>(&m_ram[timePtr]) : NULL;
if(time != NULL)
{
(*time) = m_bios.MicroSecToClock(usec);
}
}
void CThbase::SysClockToUSec(uint32 timePtr, uint32 secPtr, uint32 usecPtr)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : SysClockToUSec(time = 0x%08X, sec = 0x%08X, usec = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), timePtr, secPtr, usecPtr);
#endif
uint64* time = (timePtr != 0) ? reinterpret_cast<uint64*>(&m_ram[timePtr]) : NULL;
uint32* sec = (secPtr != 0) ? reinterpret_cast<uint32*>(&m_ram[secPtr]) : NULL;
uint32* usec = (usecPtr != 0) ? reinterpret_cast<uint32*>(&m_ram[usecPtr]) : NULL;
if(time != NULL)
{
uint64 totalusec = m_bios.ClockToMicroSec(*time);
if(sec != NULL)
{
*sec = static_cast<uint32>(totalusec / 1000000);
}
if(usec != NULL)
{
*usec = static_cast<uint32>(totalusec % 1000000);
}
}
}
uint32 CThbase::GetCurrentThreadPriority()
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : " FUNCTION_GETCURRENTTHREADPRIORITY "();\r\n",
m_bios.GetCurrentThreadIdRaw());
#endif
CIopBios::THREAD* currentThread = m_bios.GetThread(m_bios.GetCurrentThreadIdRaw());
if(currentThread == NULL) return -1;
return currentThread->priority;
}
int32 CThbase::GetThreadmanIdList(uint32 type, uint32 bufferPtr, uint32 bufferSize, uint32 countPtr)
{
#ifdef _DEBUG
CLog::GetInstance().Print(LOG_NAME, "%d : " FUNCTION_GETTHREADMANIDLIST "(type = %d, bufferPtr = 0x%08X, bufferSize = %d, countPtr = 0x%08X);\r\n",
m_bios.GetCurrentThreadIdRaw(), type, bufferPtr, bufferSize, countPtr);
#endif
CLog::GetInstance().Warn(LOG_NAME, "Using GetThreadmanIdList which is not fully implemented.\r\n");
if(countPtr != 0)
{
*reinterpret_cast<uint32*>(m_ram + countPtr) = 0;
}
return 0;
}