mirror of
https://github.com/libretro/ppsspp.git
synced 2025-04-02 19:51:44 +00:00

This reverts commit a00b1855cb6427f8eb36341137dbb81004ff29fd. Conflicts: Core/HLE/FunctionWrappers.h Core/HLE/sceCtrl.cpp Core/HLE/sceKernelModule.cpp
1106 lines
28 KiB
C++
1106 lines
28 KiB
C++
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
// Official git repository and contact information can be found at
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
#include "HLE.h"
|
|
#include "HLETables.h"
|
|
#include "../MIPS/MIPSInt.h"
|
|
#include "../MIPS/MIPSCodeUtils.h"
|
|
#include "../MIPS/MIPS.h"
|
|
#include "../../Core/CoreTiming.h"
|
|
|
|
#include "sceAudio.h"
|
|
#include "sceKernel.h"
|
|
#include "sceKernelMemory.h"
|
|
#include "sceKernelThread.h"
|
|
#include "sceKernelModule.h"
|
|
#include "sceKernelInterrupt.h"
|
|
#include "sceKernelCallback.h"
|
|
|
|
#include <queue>
|
|
|
|
enum ThreadStatus
|
|
{
|
|
THREADSTATUS_RUNNING = 1,
|
|
THREADSTATUS_READY = 2,
|
|
THREADSTATUS_WAIT = 4,
|
|
THREADSTATUS_SUSPEND = 8,
|
|
THREADSTATUS_DORMANT = 16,
|
|
THREADSTATUS_DEAD = 32,
|
|
};
|
|
|
|
enum {
|
|
ERROR_KERNEL_THREAD_ALREADY_DORMANT = 0x800201a2,
|
|
ERROR_KERNEL_THREAD_ALREADY_SUSPEND = 0x800201a3,
|
|
ERROR_KERNEL_THREAD_IS_NOT_DORMANT = 0x800201a4,
|
|
ERROR_KERNEL_THREAD_IS_NOT_SUSPEND = 0x800201a5,
|
|
ERROR_KERNEL_THREAD_IS_NOT_WAIT = 0x800201a6,
|
|
};
|
|
|
|
enum
|
|
{
|
|
PSP_THREAD_ATTR_USER = 0x80000000,
|
|
PSP_THREAD_ATTR_USBWLAN = 0xa0000000,
|
|
PSP_THREAD_ATTR_VSH = 0xc0000000,
|
|
PSP_THREAD_ATTR_KERNEL = 0x00001000,
|
|
PSP_THREAD_ATTR_VFPU = 0x00004000, // TODO: Should not bother saving VFPU context except when switching between two thread that has this attribute
|
|
PSP_THREAD_ATTR_SCRATCH_SRAM = 0x00008000, // Save/restore scratch as part of context???
|
|
PSP_THREAD_ATTR_NO_FILLSTACK = 0x00100000, // TODO: No filling of 0xff
|
|
PSP_THREAD_ATTR_CLEAR_STACK = 0x00200000, // TODO: Clear thread stack when deleted
|
|
};
|
|
|
|
const char *waitTypeStrings[] =
|
|
{
|
|
"NONE",
|
|
"Sleep",
|
|
"Delay",
|
|
"Sema",
|
|
"EventFlag",
|
|
"Mbx",
|
|
"Vpl",
|
|
"Fpl",
|
|
"",
|
|
"ThreadEnd",
|
|
"AudioChannel",
|
|
"Umd",
|
|
"Vblank",
|
|
"Mutex",
|
|
};
|
|
|
|
struct SceKernelSysClock {
|
|
u32 low;
|
|
u32 hi;
|
|
};
|
|
|
|
struct NativeThread
|
|
{
|
|
u32 nativeSize;
|
|
char name[KERNELOBJECT_MAX_NAME_LENGTH+1];
|
|
|
|
// Threading stuff
|
|
u32 attr;
|
|
u32 status;
|
|
u32 entrypoint;
|
|
u32 initialStack;
|
|
u32 stackSize;
|
|
u32 gpreg;
|
|
|
|
int initialPriority;
|
|
int currentPriority;
|
|
WaitType waitType;
|
|
SceUID waitID;
|
|
int wakeupCount;
|
|
int exitStatus;
|
|
SceKernelSysClock runForClocks;
|
|
int numInterruptPreempts;
|
|
int numThreadPreempts;
|
|
int numReleases;
|
|
};
|
|
|
|
class Thread : public KernelObject
|
|
{
|
|
public:
|
|
const char *GetName() {return nt.name;}
|
|
const char *GetTypeName() {return "Thread";}
|
|
void GetQuickInfo(char *ptr, int size)
|
|
{
|
|
sprintf(ptr, "pc= %08x sp= %08x %s %s %s %s %s %s (wt=%i wid=%i wv= %08x )",
|
|
context.pc, context.r[MIPS_REG_SP],
|
|
(nt.status & THREADSTATUS_RUNNING) ? "RUN" : "",
|
|
(nt.status & THREADSTATUS_READY) ? "READY" : "",
|
|
(nt.status & THREADSTATUS_WAIT) ? "WAIT" : "",
|
|
(nt.status & THREADSTATUS_SUSPEND) ? "SUSPEND" : "",
|
|
(nt.status & THREADSTATUS_DORMANT) ? "DORMANT" : "",
|
|
(nt.status & THREADSTATUS_DEAD) ? "DEAD" : "",
|
|
nt.waitType,
|
|
nt.waitID,
|
|
waitValue);
|
|
}
|
|
static u32 GetMissingErrorCode() { return SCE_KERNEL_ERROR_UNKNOWN_THID; }
|
|
int GetIDType() const { return SCE_KERNEL_TMID_Thread; }
|
|
bool GrabStack(u32 &stackSize)
|
|
{
|
|
if (nt.attr & PSP_THREAD_ATTR_KERNEL)
|
|
{
|
|
// Allocate stacks for kernel threads (idle) in kernel RAM
|
|
stackBlock = kernelMemory.Alloc(stackSize, true, "stack");
|
|
}
|
|
else
|
|
{
|
|
stackBlock = userMemory.Alloc(stackSize, true, "stack");
|
|
}
|
|
if (stackBlock == (u32)-1)
|
|
{
|
|
ERROR_LOG(HLE, "Failed to allocate stack for thread");
|
|
return false;
|
|
}
|
|
// Fill the stack.
|
|
Memory::Memset(stackBlock, 0xFF, stackSize);
|
|
context.r[MIPS_REG_SP] = stackBlock + stackSize;
|
|
nt.initialStack = context.r[MIPS_REG_SP];
|
|
nt.stackSize = stackSize;
|
|
// What's this 512?
|
|
context.r[MIPS_REG_K0] = context.r[MIPS_REG_SP] - 512;
|
|
context.r[MIPS_REG_SP] -= 512;
|
|
return true;
|
|
}
|
|
~Thread()
|
|
{
|
|
userMemory.Free(stackBlock);
|
|
}
|
|
|
|
NativeThread nt;
|
|
|
|
u32 waitValue;
|
|
bool sleeping;
|
|
|
|
bool isProcessingCallbacks;
|
|
|
|
ThreadContext context;
|
|
|
|
std::vector<CallbackNotification> callbacks;
|
|
|
|
u32 stackBlock;
|
|
};
|
|
|
|
|
|
Thread *__KernelCreateThread(SceUID &id, SceUID moduleID, const char *name, u32 entryPoint, u32 priority, int stacksize, u32 attr);
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//STATE BEGIN
|
|
//////////////////////////////////////////////////////////////////////////
|
|
Thread *currentThread;
|
|
u32 idleThreadHackAddr;
|
|
u32 threadReturnHackAddr;
|
|
u32 cbReturnHackAddr;
|
|
u32 intReturnHackAddr;
|
|
std::vector<Thread *> threadqueue; //Change to SceUID
|
|
|
|
SceUID threadIdleID[2];
|
|
|
|
int eventScheduledWakeup;
|
|
|
|
// This seems nasty
|
|
SceUID curModule;
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//STATE END
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
u32 __KernelCallbackReturnAddress()
|
|
{
|
|
return cbReturnHackAddr;
|
|
}
|
|
|
|
u32 __KernelInterruptReturnAddress()
|
|
{
|
|
return intReturnHackAddr;
|
|
}
|
|
|
|
void hleScheduledWakeup(u64 userdata, int cyclesLate);
|
|
|
|
void __KernelThreadingInit()
|
|
{
|
|
u32 blockSize = 4 * 4 + 4 * 2 * 3; // One 16-byte thread plus 3 8-byte "hacks"
|
|
|
|
idleThreadHackAddr = kernelMemory.Alloc(blockSize, false, "threadrethack");
|
|
// Make sure it got allocated where we expect it... at the very start of kernel RAM
|
|
//CHECK_EQ(idleThreadHackAddr & 0x3FFFFFFF, 0x08000000);
|
|
|
|
// Yeah, this is straight out of JPCSP, I should be ashamed.
|
|
Memory::Write_U32(MIPS_MAKE_ADDIU(MIPS_REG_A0, MIPS_REG_ZERO, 0), idleThreadHackAddr);
|
|
Memory::Write_U32(MIPS_MAKE_LUI(MIPS_REG_RA, 0x0800), idleThreadHackAddr + 4);
|
|
Memory::Write_U32(MIPS_MAKE_JR_RA(), idleThreadHackAddr + 8);
|
|
//Memory::Write_U32(MIPS_MAKE_SYSCALL("ThreadManForUser", "sceKernelDelayThread"), idleThreadHackAddr + 12);
|
|
Memory::Write_U32(MIPS_MAKE_SYSCALL("FakeSysCalls", "_sceKernelIdle"), idleThreadHackAddr + 12);
|
|
Memory::Write_U32(MIPS_MAKE_BREAK(), idleThreadHackAddr + 16);
|
|
|
|
threadReturnHackAddr = idleThreadHackAddr + 20;
|
|
WriteSyscall("FakeSysCalls", NID_THREADRETURN, threadReturnHackAddr);
|
|
|
|
cbReturnHackAddr = threadReturnHackAddr + 8;
|
|
WriteSyscall("FakeSysCalls", NID_CALLBACKRETURN, cbReturnHackAddr);
|
|
|
|
intReturnHackAddr = cbReturnHackAddr + 8;
|
|
WriteSyscall("FakeSysCalls", NID_INTERRUPTRETURN, intReturnHackAddr);
|
|
|
|
eventScheduledWakeup = CoreTiming::RegisterEvent("ScheduledWakeup", &hleScheduledWakeup);
|
|
|
|
// Create the two idle threads, as well. With the absolute minimal possible priority.
|
|
// Zero stack size. Hm, if callbacks are ever to run on these threads, that's not a good idea.
|
|
__KernelCreateThread(threadIdleID[0], 0, "idle0", idleThreadHackAddr, 0x7f, 0, PSP_THREAD_ATTR_KERNEL);
|
|
__KernelCreateThread(threadIdleID[1], 0, "idle1", idleThreadHackAddr, 0x7f, 0, PSP_THREAD_ATTR_KERNEL);
|
|
// These idle threads are later started in LoadExec, which calls __KernelStartIdleThreads below.
|
|
}
|
|
|
|
void __KernelStartIdleThreads()
|
|
{
|
|
for (int i = 0; i < 2; i++)
|
|
{
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadIdleID[i], error);
|
|
t->nt.gpreg = __KernelGetModuleGP(curModule);
|
|
t->context.r[MIPS_REG_GP] = t->nt.gpreg;
|
|
//t->context.pc += 4; // ADJUSTPC
|
|
t->nt.status = THREADSTATUS_READY;
|
|
}
|
|
}
|
|
|
|
void _sceKernelIdle()
|
|
{
|
|
CoreTiming::Idle();
|
|
// Advance must happen between Idle and Reschedule, so that threads that were waiting for something
|
|
// that was triggered at the end of the Idle period must get a chance to be scheduled.
|
|
// get a chance to be rescheduled.
|
|
CoreTiming::Advance();
|
|
|
|
// In Advance, we might trigger an interrupt such as vblank.
|
|
// If we end up in an interrupt, we don't want to reschedule.
|
|
// However, we have to reschedule... damn.
|
|
__KernelReSchedule("idle");
|
|
}
|
|
|
|
void __KernelThreadingShutdown()
|
|
{
|
|
kernelMemory.Free(threadReturnHackAddr);
|
|
threadReturnHackAddr = 0;
|
|
cbReturnHackAddr = 0;
|
|
currentThread = 0;
|
|
threadqueue.clear();
|
|
}
|
|
|
|
u32 __KernelGetWaitValue(SceUID threadID, u32 &error)
|
|
{
|
|
Thread *t = kernelObjects.Get<Thread>(threadID, error);
|
|
if (t)
|
|
{
|
|
return t->waitValue;
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE, "__KernelGetWaitValue ERROR: thread %i", threadID);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// TODO: If cbid == -1, notify the callback ID on all threads that have it.
|
|
void __KernelNotifyCallback(SceUID threadID, SceUID cbid, u32 notifyArg)
|
|
{
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadID, error);
|
|
if (t)
|
|
{
|
|
for (size_t i = 0; i < t->callbacks.size(); t++)
|
|
{
|
|
if (t->callbacks[i].cbid == cbid)
|
|
{
|
|
t->callbacks[i].arg = notifyArg;
|
|
t->callbacks[i].count++;
|
|
return;
|
|
}
|
|
}
|
|
CallbackNotification cb;
|
|
cb.cbid = cbid;
|
|
cb.arg = notifyArg;
|
|
cb.count = 1;
|
|
}
|
|
// TODO: error checking
|
|
}
|
|
|
|
CallbackNotification *__KernelGetCallbackNotification(SceUID cbid)
|
|
{
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(__KernelGetCurThread(), error);
|
|
for (size_t i = 0; i < t->callbacks.size(); t++)
|
|
{
|
|
if (t->callbacks[i].cbid == cbid)
|
|
{
|
|
return &t->callbacks[i];
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void sceKernelReferThreadStatus()
|
|
{
|
|
SceUID threadID = PARAM(0);
|
|
if (threadID == 0)
|
|
threadID = __KernelGetCurThread();
|
|
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadID, error);
|
|
if (t)
|
|
{
|
|
DEBUG_LOG(HLE,"sceKernelReferThreadStatus(%i, %08x)", threadID, PARAM(1));
|
|
void *outptr = (void*)Memory::GetPointer(PARAM(1));
|
|
int sz = sizeof(NativeThread);
|
|
t->nt.nativeSize = sz;
|
|
memcpy(outptr, &(t->nt), sz);
|
|
RETURN(0);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelReferThreadStatus Error %08x", error);
|
|
RETURN(error);
|
|
}
|
|
}
|
|
|
|
void sceKernelGetThreadExitStatus()
|
|
{
|
|
SceUID threadID = PARAM(0);
|
|
if (threadID == 0)
|
|
threadID = __KernelGetCurThread();
|
|
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadID, error);
|
|
if (t)
|
|
{
|
|
if (t->nt.status == THREADSTATUS_DORMANT) // TODO: can be dormant before starting, too, need to avoid that
|
|
{
|
|
DEBUG_LOG(HLE,"sceKernelGetThreadExitStatus(%i)", threadID);
|
|
RETURN(t->nt.exitStatus);
|
|
}
|
|
else
|
|
{
|
|
RETURN(SCE_KERNEL_ERROR_NOT_DORMANT);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelGetThreadExitStatus Error %08x", error);
|
|
RETURN(SCE_KERNEL_ERROR_UNKNOWN_THID);
|
|
}
|
|
}
|
|
|
|
void sceKernelGetThreadmanIdType()
|
|
{
|
|
SceUID uid = PARAM(0);
|
|
int type;
|
|
if (kernelObjects.GetIDType(uid, &type))
|
|
{
|
|
RETURN(type);
|
|
DEBUG_LOG(HLE, "%i=sceKernelGetThreadmanIdType(%i)", type, uid);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE, "sceKernelGetThreadmanIdType(%i) - FAILED", uid);
|
|
RETURN(SCE_KERNEL_ERROR_ILLEGAL_ARGUMENT);
|
|
}
|
|
}
|
|
|
|
// Saves the current CPU context
|
|
void __KernelSaveContext(ThreadContext *ctx)
|
|
{
|
|
for (int i=0; i<32; i++)
|
|
{
|
|
ctx->r[i] = currentMIPS->r[i];
|
|
ctx->f[i] = currentMIPS->f[i];
|
|
}
|
|
for (int i=0; i<128; i++)
|
|
{
|
|
ctx->v[i] = currentMIPS->v[i];
|
|
}
|
|
for (int i=0; i<16; i++)
|
|
{
|
|
ctx->vfpuCtrl[i] = currentMIPS->vfpuCtrl[i];
|
|
}
|
|
ctx->hi = currentMIPS->hi;
|
|
ctx->lo = currentMIPS->lo;
|
|
ctx->pc = currentMIPS->pc;
|
|
ctx->fpcond = currentMIPS->fpcond;
|
|
// ctx->fcr0 = currentMIPS->fcr0;
|
|
// ctx->fcr31 = currentMIPS->fcr31;
|
|
|
|
// TODO: Make VFPU saving optional/delayed, only necessary between VFPU-attr-marked threads
|
|
}
|
|
|
|
// Loads a CPU context
|
|
void __KernelLoadContext(ThreadContext *ctx)
|
|
{
|
|
for (int i=0; i<32; i++)
|
|
{
|
|
currentMIPS->r[i] = ctx->r[i];
|
|
currentMIPS->f[i] = ctx->f[i];
|
|
}
|
|
for (int i=0; i<128; i++)
|
|
{
|
|
currentMIPS->v[i] = ctx->v[i];
|
|
}
|
|
for (int i=0; i<15; i++)
|
|
{
|
|
currentMIPS->vfpuCtrl[i] = ctx->vfpuCtrl[i];
|
|
}
|
|
currentMIPS->hi = ctx->hi;
|
|
currentMIPS->lo = ctx->lo;
|
|
currentMIPS->pc = ctx->pc;
|
|
currentMIPS->fpcond = ctx->fpcond;
|
|
// currentMIPS->fcr0 = ctx->fcr0;
|
|
// currentMIPS->fcr31 = ctx->fcr31;
|
|
}
|
|
|
|
// DANGEROUS
|
|
// Only run when you can safely accept a context switch
|
|
// Triggers a waitable event, that is, it wakes up all threads that waits for it
|
|
// If any changes were made, it will context switch
|
|
bool __KernelTriggerWait(WaitType type, int id, bool dontSwitch)
|
|
{
|
|
bool doneAnything = false;
|
|
|
|
for (std::vector<Thread *>::iterator iter = threadqueue.begin(); iter != threadqueue.end(); iter++)
|
|
{
|
|
Thread *t = *iter;
|
|
if (t->nt.status & THREADSTATUS_WAIT)
|
|
{
|
|
if (t->nt.waitType == type && t->nt.waitID == id)
|
|
{
|
|
// This threads is waiting for the triggered object
|
|
t->nt.status &= ~THREADSTATUS_WAIT;
|
|
if (t->nt.status == 0)
|
|
{
|
|
t->nt.status = THREADSTATUS_READY;
|
|
}
|
|
doneAnything = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// if (doneAnything) // lumines?
|
|
{
|
|
if (!dontSwitch)
|
|
{
|
|
// TODO: time waster
|
|
char temp[256];
|
|
sprintf(temp, "resumed from wait %s", waitTypeStrings[(int)type]);
|
|
__KernelReSchedule(temp);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
u32 __KernelResumeThread(SceUID threadID)
|
|
{
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadID, error);
|
|
if (t)
|
|
{
|
|
t->nt.status &= ~THREADSTATUS_WAIT;
|
|
if (!(t->nt.status & (THREADSTATUS_SUSPEND | THREADSTATUS_WAIT)))
|
|
t->nt.status |= THREADSTATUS_READY;
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE, "__KernelResumeThread(%d): bad thread: %08x", threadID, error);
|
|
return error;
|
|
}
|
|
}
|
|
|
|
// makes the current thread wait for an event
|
|
void __KernelWaitCurThread(WaitType type, SceUID waitID, u32 waitValue, int timeout, bool processCallbacks)
|
|
{
|
|
currentThread->nt.status = THREADSTATUS_WAIT;
|
|
currentThread->nt.waitID = waitID;
|
|
currentThread->nt.waitType = type;
|
|
currentThread->nt.numReleases++;
|
|
currentThread->waitValue = waitValue;
|
|
currentThread->isProcessingCallbacks = processCallbacks;
|
|
if (timeout)
|
|
{
|
|
// TODO:
|
|
}
|
|
|
|
RETURN(0); //pretend all went OK
|
|
|
|
// TODO: time waster
|
|
char temp[256];
|
|
sprintf(temp, "started wait %s", waitTypeStrings[(int)type]);
|
|
__KernelReSchedule(temp);
|
|
// TODO: Remove thread from Ready queue?
|
|
}
|
|
|
|
void hleScheduledWakeup(u64 userdata, int cyclesLate)
|
|
{
|
|
SceUID threadID = (SceUID)userdata;
|
|
__KernelTriggerWait(WAITTYPE_DELAY, threadID, true);
|
|
}
|
|
|
|
void __KernelScheduleWakeup(SceUID threadID, int usFromNow)
|
|
{
|
|
CoreTiming::ScheduleEvent(usToCycles(usFromNow), eventScheduledWakeup, threadID);
|
|
}
|
|
|
|
void __KernelRemoveFromThreadQueue(Thread *t)
|
|
{
|
|
for (int i = 0; i < threadqueue.size(); i++)
|
|
{
|
|
if (threadqueue[i] == t)
|
|
{
|
|
DEBUG_LOG(HLE, "Deleted thread %p from thread queue", t);
|
|
threadqueue.erase(threadqueue.begin() + i);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void __KernelReSchedule(const char *reason)
|
|
{
|
|
// cancel rescheduling when in interrupt or callback, otherwise everything will be fucked up
|
|
if (__IsInInterrupt() || __KernelInCallback())
|
|
{
|
|
reason = "WTF";
|
|
return;
|
|
}
|
|
// round-robin scheduler
|
|
// seems to work ?
|
|
// not accurate!
|
|
retry:
|
|
int bestthread = -1;
|
|
int prio=0xffffff;
|
|
|
|
int next = 0;
|
|
for (size_t i=0; i<threadqueue.size(); i++)
|
|
{
|
|
if (currentThread == threadqueue[i])
|
|
{
|
|
next = (int)i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (size_t i=0; i<threadqueue.size(); i++)
|
|
{
|
|
next = (next + 1) % threadqueue.size();
|
|
|
|
Thread *t = threadqueue[next];
|
|
if (t->nt.currentPriority < prio)
|
|
{
|
|
if (t->nt.status & THREADSTATUS_READY)
|
|
{
|
|
bestthread = next;
|
|
prio = t->nt.currentPriority;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bestthread != -1)
|
|
{
|
|
if (currentThread) // It might just have been deleted.
|
|
{
|
|
__KernelSaveContext(¤tThread->context);
|
|
DEBUG_LOG(HLE,"Context saved (%s): %i - %s - pc: %08x", reason, currentThread->GetUID(), currentThread->GetName(), currentMIPS->pc);
|
|
}
|
|
currentThread = threadqueue[bestthread];
|
|
__KernelLoadContext(¤tThread->context);
|
|
DEBUG_LOG(HLE,"Context loaded (%s): %i - %s - pc: %08x", reason, currentThread->GetUID(), currentThread->GetName(), currentMIPS->pc);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
_dbg_assert_msg_(HLE,0,"No threads available to schedule! There should be at least one idle thread available.");
|
|
// This shouldn't happen anymore now that we have idle threads.
|
|
|
|
// No threads want to run : increase timers, skip time in general
|
|
// MessageBox(0,"Error: no thread to transition to",0,0);
|
|
|
|
// DEBUG_LOG(HLE,"No thread to transition to, idling");
|
|
CoreTiming::Idle();
|
|
goto retry;
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Thread Management
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void sceKernelCheckThreadStack()
|
|
{
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(__KernelGetCurThread(), error);
|
|
u32 diff = abs((s64)t->stackBlock - (s64)currentMIPS->r[MIPS_REG_SP]);
|
|
ERROR_LOG(HLE, "%i=sceKernelCheckThreadStack()", diff);
|
|
RETURN(diff); //Blatant lie
|
|
}
|
|
|
|
void ThreadContext::reset()
|
|
{
|
|
for (int i = 0; i<32; i++)
|
|
{
|
|
r[i] = 0;
|
|
f[i] = 0.0f;
|
|
}
|
|
for (int i = 0; i<128; i++)
|
|
{
|
|
v[i] = 0.0f;
|
|
}
|
|
for (int i = 0; i<15; i++)
|
|
{
|
|
vfpuCtrl[i] = 0x00000000;
|
|
}
|
|
vfpuCtrl[VFPU_CTRL_SPREFIX] = 0xe4; // neutral
|
|
vfpuCtrl[VFPU_CTRL_TPREFIX] = 0xe4; // neutral
|
|
vfpuCtrl[VFPU_CTRL_DPREFIX] = 0x0;
|
|
vfpuCtrl[VFPU_CTRL_CC] = 0x3f;
|
|
vfpuCtrl[VFPU_CTRL_INF4] = 0;
|
|
vfpuCtrl[VFPU_CTRL_RCX0] = 0x3f800001;
|
|
vfpuCtrl[VFPU_CTRL_RCX1] = 0x3f800002;
|
|
vfpuCtrl[VFPU_CTRL_RCX2] = 0x3f800004;
|
|
vfpuCtrl[VFPU_CTRL_RCX3] = 0x3f800008;
|
|
vfpuCtrl[VFPU_CTRL_RCX4] = 0x3f800000;
|
|
vfpuCtrl[VFPU_CTRL_RCX5] = 0x3f800000;
|
|
vfpuCtrl[VFPU_CTRL_RCX6] = 0x3f800000;
|
|
vfpuCtrl[VFPU_CTRL_RCX7] = 0x3f800000;
|
|
fpcond = 0;
|
|
fcr0 = 0;
|
|
fcr31 = 0;
|
|
hi = 0;
|
|
lo = 0;
|
|
}
|
|
|
|
Thread *__KernelCreateThread(SceUID &id, SceUID moduleID, const char *name, u32 entryPoint, u32 priority, int stacksize, u32 attr)
|
|
{
|
|
Thread *t = new Thread;
|
|
id = kernelObjects.Create(t);
|
|
|
|
threadqueue.push_back(t);
|
|
|
|
t->context.reset();
|
|
|
|
t->context.hi = 0;
|
|
t->context.lo = 0;
|
|
t->context.pc = entryPoint;
|
|
memset(&t->nt, 0xCD, sizeof(t->nt));
|
|
|
|
t->nt.entrypoint = entryPoint;
|
|
t->nt.nativeSize = sizeof(t->nt);
|
|
t->nt.attr = attr;
|
|
t->nt.initialPriority = t->nt.currentPriority = priority;
|
|
t->nt.stackSize = stacksize;
|
|
t->nt.status = THREADSTATUS_DORMANT;
|
|
t->nt.waitType = WAITTYPE_NONE;
|
|
t->nt.waitID = 0;
|
|
t->waitValue = 0;
|
|
t->nt.exitStatus = 0;
|
|
t->nt.numInterruptPreempts = 0;
|
|
t->nt.numReleases = 0;
|
|
t->nt.numThreadPreempts = 0;
|
|
t->nt.runForClocks.low = 0;
|
|
t->nt.runForClocks.hi = 0;
|
|
t->nt.wakeupCount = 0;
|
|
if (moduleID)
|
|
t->nt.gpreg = __KernelGetModuleGP(moduleID);
|
|
else
|
|
t->nt.gpreg = 0; // sceKernelStartThread will take care of this.
|
|
|
|
strncpy(t->nt.name, name, 32);
|
|
t->context.r[MIPS_REG_RA] = threadReturnHackAddr; //hack! TODO fix
|
|
t->GrabStack(t->nt.stackSize); // can change the stacksize!
|
|
return t;
|
|
}
|
|
|
|
void __KernelSetupRootThread(SceUID moduleID, int args, const char *argp, int prio, int stacksize, int attr)
|
|
{
|
|
curModule = moduleID;
|
|
//grab mips regs
|
|
SceUID id;
|
|
currentThread = __KernelCreateThread(id, moduleID, "root", currentMIPS->pc, prio, stacksize, attr);
|
|
currentThread->nt.status = THREADSTATUS_READY; // do not schedule
|
|
|
|
strcpy(currentThread->nt.name, "root");
|
|
|
|
__KernelLoadContext(¤tThread->context);
|
|
mipsr4k.r[MIPS_REG_A0] = args;
|
|
mipsr4k.r[MIPS_REG_SP] -= 256;
|
|
u32 location = mipsr4k.r[MIPS_REG_SP];
|
|
mipsr4k.r[MIPS_REG_A1] = location;
|
|
for (int i=0; i<args; i++)
|
|
Memory::Write_U8(argp[i], location+i);
|
|
}
|
|
|
|
|
|
void sceKernelCreateThread()
|
|
{
|
|
u32 nameAddr = PARAM(0);
|
|
const char *threadName = Memory::GetCharPointer(nameAddr);
|
|
u32 entry = PARAM(1);
|
|
u32 prio = PARAM(2);
|
|
int stacksize = PARAM(3);
|
|
u32 attr = PARAM(4);
|
|
//ignore PARAM(5)
|
|
|
|
// HACK! Kill super big stacks.
|
|
// if (stacksize > 0x4000) stacksize = 0x4000;
|
|
|
|
SceUID id;
|
|
__KernelCreateThread(id, curModule, threadName, entry, prio, stacksize, attr);
|
|
INFO_LOG(HLE,"%i = sceKernelCreateThread(name=\"%s\", entry= %08x, stacksize=%i )", id, threadName, entry, stacksize);
|
|
RETURN(id);
|
|
}
|
|
|
|
|
|
u32 sceKernelStartThread()
|
|
{
|
|
int threadToStartID = PARAM(0);
|
|
u32 argSize = PARAM(1);
|
|
u32 argBlockPtr = PARAM(2);
|
|
|
|
if (threadToStartID != currentThread->GetUID())
|
|
{
|
|
u32 error;
|
|
Thread *startThread = kernelObjects.Get<Thread>(threadToStartID, error);
|
|
if (startThread == 0)
|
|
{
|
|
ERROR_LOG(HLE,"%08x=sceKernelStartThread(thread=%i, argSize=%i, argPtr= %08x): thread does not exist!",
|
|
error,threadToStartID,argSize,argBlockPtr)
|
|
return error;
|
|
}
|
|
|
|
if (startThread->nt.status != THREADSTATUS_DORMANT)
|
|
{
|
|
//Not dormant, WTF?
|
|
return ERROR_KERNEL_THREAD_IS_NOT_DORMANT;
|
|
}
|
|
|
|
INFO_LOG(HLE,"sceKernelStartThread(thread=%i, argSize=%i, argPtr= %08x )",
|
|
threadToStartID,argSize,argBlockPtr);
|
|
|
|
startThread->nt.status = THREADSTATUS_READY;
|
|
u32 sp = startThread->context.r[MIPS_REG_SP];
|
|
if (argBlockPtr)
|
|
{
|
|
startThread->context.r[MIPS_REG_A0] = argSize;
|
|
startThread->context.r[MIPS_REG_A1] = sp;
|
|
}
|
|
else
|
|
{
|
|
startThread->context.r[MIPS_REG_A0] = 0;
|
|
startThread->context.r[MIPS_REG_A1] = 0;
|
|
}
|
|
startThread->context.r[MIPS_REG_GP] = startThread->nt.gpreg;
|
|
|
|
//now copy argument to stack
|
|
for (int i = 0; i < (int)argSize; i++)
|
|
Memory::Write_U8(Memory::Read_U8(argBlockPtr + i), sp + i);
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"thread %i trying to start itself", threadToStartID);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
void sceKernelGetThreadStackFreeSize()
|
|
{
|
|
SceUID threadID = PARAM(0);
|
|
INFO_LOG(HLE,"sceKernelGetThreadStackFreeSize(%i)", threadID);
|
|
u32 error;
|
|
Thread *thread = kernelObjects.Get<Thread>(threadID, error);
|
|
|
|
// Scan the stack for 0xFF
|
|
int sz = 0;
|
|
for (u32 addr = thread->stackBlock; addr < thread->stackBlock + thread->nt.stackSize; addr++)
|
|
{
|
|
if (Memory::Read_U8(addr) != 0xFF)
|
|
break;
|
|
sz++;
|
|
}
|
|
|
|
RETURN(sz & ~3);
|
|
}
|
|
|
|
|
|
void _sceKernelReturnFromThread()
|
|
{
|
|
INFO_LOG(HLE,"_sceKernelReturnFromThread : %s", currentThread->GetName());
|
|
currentThread->nt.exitStatus = currentThread->context.r[2];
|
|
currentThread->nt.status = THREADSTATUS_DORMANT;
|
|
|
|
// TODO: Need to remove the thread from any ready queues.
|
|
|
|
// Find threads that waited for me
|
|
// Wake them
|
|
if (!__KernelTriggerWait(WAITTYPE_THREADEND, __KernelGetCurThread()))
|
|
__KernelReSchedule("return from thread");
|
|
}
|
|
|
|
void sceKernelExitThread()
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelExitThread FAKED");
|
|
currentThread->nt.status = THREADSTATUS_DORMANT;
|
|
currentThread->nt.exitStatus = PARAM(0);
|
|
//Find threads that waited for me
|
|
// Wake them
|
|
if (!__KernelTriggerWait(WAITTYPE_THREADEND, __KernelGetCurThread()))
|
|
__KernelReSchedule("exited thread");
|
|
}
|
|
|
|
void _sceKernelExitThread()
|
|
{
|
|
ERROR_LOG(HLE,"_sceKernelExitThread FAKED");
|
|
currentThread->nt.status = THREADSTATUS_DORMANT;
|
|
currentThread->nt.exitStatus = PARAM(0);
|
|
//Find threads that waited for this one
|
|
// Wake them
|
|
if (!__KernelTriggerWait(WAITTYPE_THREADEND, __KernelGetCurThread()))
|
|
__KernelReSchedule("exit-deleted thread");
|
|
}
|
|
|
|
void sceKernelExitDeleteThread()
|
|
{
|
|
int threadHandle = __KernelGetCurThread();
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadHandle, error);
|
|
if (t)
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelExitDeleteThread()");
|
|
currentThread->nt.status = THREADSTATUS_DORMANT;
|
|
currentThread->nt.exitStatus = PARAM(0);
|
|
|
|
__KernelRemoveFromThreadQueue(t);
|
|
currentThread = 0;
|
|
|
|
RETURN(kernelObjects.Destroy<Thread>(threadHandle));
|
|
|
|
__KernelTriggerWait(WAITTYPE_THREADEND, threadHandle);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelExitDeleteThread() ERROR - could not find myself!");
|
|
RETURN(error);
|
|
}
|
|
}
|
|
|
|
|
|
void sceKernelRotateThreadReadyQueue()
|
|
{
|
|
DEBUG_LOG(HLE,"sceKernelRotateThreadReadyQueue : rescheduling");
|
|
__KernelReSchedule("rotatethreadreadyqueue");
|
|
}
|
|
|
|
void sceKernelDeleteThread()
|
|
{
|
|
int threadHandle = PARAM(0);
|
|
if (threadHandle != currentThread->GetUID())
|
|
{
|
|
//TODO: remove from threadqueue!
|
|
DEBUG_LOG(HLE,"sceKernelDeleteThread(%i)",threadHandle);
|
|
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(threadHandle, error);
|
|
if (t)
|
|
{
|
|
__KernelRemoveFromThreadQueue(t);
|
|
|
|
RETURN(kernelObjects.Destroy<Thread>(threadHandle));
|
|
|
|
__KernelTriggerWait(WAITTYPE_THREADEND, threadHandle);
|
|
|
|
//TODO: should we really reschedule here?
|
|
//if (!__KernelTriggerWait(WAITTYPE_THREADEND, threadHandle))
|
|
// __KernelReSchedule("thread deleted");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE, "Thread \"%s\" tries to delete itself! :(",currentThread->GetName());
|
|
RETURN(-1);
|
|
}
|
|
}
|
|
|
|
void sceKernelTerminateDeleteThread()
|
|
{
|
|
int threadno = PARAM(0);
|
|
if (threadno != currentThread->GetUID())
|
|
{
|
|
//TODO: remove from threadqueue!
|
|
INFO_LOG(HLE,"sceKernelTerminateDeleteThread(%i)",threadno);
|
|
RETURN(0); //kernelObjects.Destroy<Thread>(threadno));
|
|
|
|
//TODO: should we really reschedule here?
|
|
if (!__KernelTriggerWait(WAITTYPE_THREADEND, threadno))
|
|
__KernelReSchedule("termdeletethread");
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE, "Thread \"%s\" tries to delete itself! :(",currentThread->GetName());
|
|
RETURN(-1);
|
|
}
|
|
}
|
|
|
|
SceUID __KernelGetCurThread()
|
|
{
|
|
return currentThread->GetUID();
|
|
}
|
|
|
|
void sceKernelGetThreadId()
|
|
{
|
|
u32 retVal = currentThread->GetUID();
|
|
DEBUG_LOG(HLE,"%i = sceKernelGetThreadId()", retVal);
|
|
RETURN(retVal);
|
|
}
|
|
|
|
void sceKernelChangeCurrentThreadAttr()
|
|
{
|
|
int clearAttr = PARAM(0);
|
|
int setAttr = PARAM(1);
|
|
DEBUG_LOG(HLE,"0 = sceKernelChangeCurrentThreadAttr(clear = %08x, set = %08x", clearAttr, setAttr);
|
|
currentThread->nt.attr = (currentThread->nt.attr & ~clearAttr) | setAttr;
|
|
RETURN(0);
|
|
}
|
|
|
|
void sceKernelChangeThreadPriority()
|
|
{
|
|
int id = PARAM(0);
|
|
if (id == 0) id = currentThread->GetUID(); //special
|
|
|
|
u32 error;
|
|
Thread *thread = kernelObjects.Get<Thread>(id, error);
|
|
if (thread)
|
|
{
|
|
DEBUG_LOG(HLE,"sceKernelChangeThreadPriority(%i, %i)", id, PARAM(1));
|
|
thread->nt.currentPriority = PARAM(1);
|
|
RETURN(0);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"%08x=sceKernelChangeThreadPriority(%i, %i) failed", error, id, PARAM(1));
|
|
RETURN(error);
|
|
}
|
|
}
|
|
|
|
void sceKernelDelayThreadCB()
|
|
{
|
|
u32 usec = PARAM(0);
|
|
if (usec < 200) usec = 200;
|
|
DEBUG_LOG(HLE,"sceKernelDelayThreadCB(%i usec)",usec);
|
|
|
|
SceUID curThread = __KernelGetCurThread();
|
|
__KernelScheduleWakeup(curThread, usec);
|
|
__KernelWaitCurThread(WAITTYPE_DELAY, curThread, 0, 0, true);
|
|
}
|
|
|
|
void sceKernelDelayThread()
|
|
{
|
|
u32 usec = PARAM(0);
|
|
if (usec < 200) usec = 200;
|
|
DEBUG_LOG(HLE,"sceKernelDelayThread(%i usec)",usec);
|
|
SceUID curThread = __KernelGetCurThread();
|
|
__KernelScheduleWakeup(curThread, usec);
|
|
__KernelWaitCurThread(WAITTYPE_DELAY, curThread, 0, 0, false);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// WAIT/SLEEP ETC
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void sceKernelWakeupThread()
|
|
{
|
|
SceUID id = PARAM(0);
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(id, error);
|
|
if (t)
|
|
{
|
|
t->nt.wakeupCount++;
|
|
DEBUG_LOG(HLE,"sceKernelWakeupThread(%i) - wakeupCount incremented to %i",id,t->nt.wakeupCount);
|
|
if (t->nt.waitType == WAITTYPE_SLEEP && t->nt.wakeupCount>=0)
|
|
{
|
|
__KernelResumeThread(id);
|
|
}
|
|
}
|
|
RETURN(0);
|
|
}
|
|
|
|
void sceKernelSleepThread()
|
|
{
|
|
currentThread->nt.wakeupCount--;
|
|
DEBUG_LOG(HLE,"sceKernelSleepThread() - wakeupCount decremented to %i", currentThread->nt.wakeupCount);
|
|
if (currentThread->nt.wakeupCount < 0)
|
|
__KernelWaitCurThread(WAITTYPE_SLEEP, 0, 0, 0, false);
|
|
else
|
|
{
|
|
RETURN(0);
|
|
}
|
|
}
|
|
|
|
//the homebrew PollCallbacks
|
|
void sceKernelSleepThreadCB()
|
|
{
|
|
DEBUG_LOG(HLE,"sceKernelSleepThreadCB()");
|
|
//set it to waiting
|
|
currentThread->nt.wakeupCount--;
|
|
DEBUG_LOG(HLE,"sceKernelSleepThread() - wakeupCount decremented to %i", currentThread->nt.wakeupCount);
|
|
if (currentThread->nt.wakeupCount < 0)
|
|
__KernelWaitCurThread(WAITTYPE_SLEEP, 0, 0, 0, true);
|
|
else
|
|
{
|
|
RETURN(0);
|
|
}
|
|
}
|
|
|
|
void sceKernelWaitThreadEnd()
|
|
{
|
|
SceUID id = PARAM(0);
|
|
DEBUG_LOG(HLE,"sceKernelWaitThreadEnd(%i)",id);
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(id, error);
|
|
if (t)
|
|
{
|
|
if (t->nt.status != THREADSTATUS_DORMANT)
|
|
{
|
|
__KernelWaitCurThread(WAITTYPE_THREADEND, id, 0, 0, false);
|
|
return;
|
|
}
|
|
DEBUG_LOG(HLE,"sceKernelWaitThreadEnd - thread %i already ended. Doing nothing.", id);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelWaitThreadEnd - bad thread %i", id);
|
|
}
|
|
RETURN(0);
|
|
}
|
|
|
|
void sceKernelWaitThreadEndCB()
|
|
{
|
|
SceUID id = PARAM(0);
|
|
DEBUG_LOG(HLE,"sceKernelWaitThreadEnd(%i)",id);
|
|
u32 error;
|
|
Thread *t = kernelObjects.Get<Thread>(id, error);
|
|
if (t)
|
|
{
|
|
if (t->nt.status != THREADSTATUS_DORMANT)
|
|
{
|
|
__KernelWaitCurThread(WAITTYPE_THREADEND, id, 0, 0, true);
|
|
return;
|
|
}
|
|
DEBUG_LOG(HLE,"sceKernelWaitThreadEnd - thread %i already ended. Doing nothing.", id);
|
|
}
|
|
else
|
|
{
|
|
ERROR_LOG(HLE,"sceKernelWaitThreadEnd - bad thread %i", id);
|
|
}
|
|
RETURN(0);
|
|
}
|
|
|
|
void sceKernelSuspendThread()
|
|
{
|
|
DEBUG_LOG(HLE,"UNIMPL sceKernelSuspendThread");
|
|
RETURN(0);
|
|
}
|
|
|
|
void sceKernelResumeThread()
|
|
{
|
|
DEBUG_LOG(HLE,"UNIMPL sceKernelResumeThread");
|
|
RETURN(0);
|
|
}
|