// 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 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 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 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(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(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(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(__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(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(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::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(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; int.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(__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 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(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(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(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(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(threadHandle, error); if (t) { __KernelRemoveFromThreadQueue(t); RETURN(kernelObjects.Destroy(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(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(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(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(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(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); }