mirror of
https://github.com/RPCSX/rpcsx.git
synced 2026-01-03 15:20:27 +01:00
move module initialization into a module manager, still has some issues like stopping not working and debug crashing add #idef 0 to modules that aren't in the windows project don't double initialize and don't de-initialize for now, since many modules don't expect it and it leads to many errors remove duplicate module lists for empty modules and implemented ones, make Module non-copyable but movable add secondary project, no real use for it now add some memleak config to the emucore and add asmjit path to rpcs3 small rebase error fixed to get it to compile again add filters for emucore re-add the module manager and static file WIP commit, linker errors abound some more abstraction layer stuff fix the remaining linker errors, re-enable platform specific mouse, pad and keyboard handlers rebasing fix memset undefined and re() usage of se_t before declaration Add wxGUI define by default for cmake builds fix copy constructors of Datetime header fix copy constructors of other wx interface classes remove static declarations of global variables make wxGLCanvas constructor non-ambiguous even with wx2.8. compat mode, fix wrong std::exception constructor calls remove duplicate definition for FromUTF8 and ToUTF8 temp changes
200 lines
4.9 KiB
C++
200 lines
4.9 KiB
C++
#include "stdafx.h"
|
|
#include "Emu/ConLog.h"
|
|
#include "Emu/Memory/Memory.h"
|
|
#include "Emu/System.h"
|
|
#include "Emu/Cell/PPUThread.h"
|
|
#include "Emu/SysCalls/SC_FUNC.h"
|
|
#include "Emu/SysCalls/Modules.h"
|
|
#include "Emu/SysCalls/SysCalls.h"
|
|
|
|
extern Module *sysPrxForUser;
|
|
|
|
static const u32 PPU_THREAD_ID_INVALID = 0xFFFFFFFFU;
|
|
enum
|
|
{
|
|
SYS_PPU_THREAD_ONCE_INIT,
|
|
SYS_PPU_THREAD_DONE_INIT,
|
|
};
|
|
|
|
void sys_ppu_thread_exit(u64 errorcode)
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_exit(0x%llx)", errorcode);
|
|
|
|
PPUThread& thr = GetCurrentPPUThread();
|
|
u32 tid = thr.GetId();
|
|
|
|
if (thr.owned_mutexes)
|
|
{
|
|
ConLog.Error("Owned mutexes found (%d)", thr.owned_mutexes);
|
|
thr.owned_mutexes = 0;
|
|
}
|
|
|
|
thr.SetExitStatus(errorcode);
|
|
thr.Stop();
|
|
}
|
|
|
|
int sys_ppu_thread_yield()
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_yield()");
|
|
Sleep(1);
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_join(u32 thread_id, mem64_t vptr)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_join(thread_id=%d, vptr_addr=0x%x)", thread_id, vptr.GetAddr());
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
|
|
while (thr->IsAlive())
|
|
{
|
|
if (Emu.IsStopped())
|
|
{
|
|
ConLog.Warning("sys_ppu_thread_join(%d) aborted", thread_id);
|
|
return CELL_OK;
|
|
}
|
|
Sleep(1);
|
|
}
|
|
|
|
vptr = thr->GetExitStatus();
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_detach(u32 thread_id)
|
|
{
|
|
sysPrxForUser->Error("sys_ppu_thread_detach(thread_id=%d)", thread_id);
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
|
|
if(!thr->IsJoinable())
|
|
return CELL_EINVAL;
|
|
thr->SetJoinable(false);
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
void sys_ppu_thread_get_join_state(u32 isjoinable_addr)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_get_join_state(isjoinable_addr=0x%x)", isjoinable_addr);
|
|
Memory.Write32(isjoinable_addr, GetCurrentPPUThread().IsJoinable());
|
|
}
|
|
|
|
int sys_ppu_thread_set_priority(u32 thread_id, int prio)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_set_priority(thread_id=%d, prio=%d)", thread_id, prio);
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
|
|
thr->SetPrio(prio);
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_get_priority(u32 thread_id, u32 prio_addr)
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_get_priority(thread_id=%d, prio_addr=0x%x)", thread_id, prio_addr);
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
if(!Memory.IsGoodAddr(prio_addr)) return CELL_EFAULT;
|
|
|
|
Memory.Write32(prio_addr, thr->GetPrio());
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_get_stack_information(u32 info_addr)
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_get_stack_information(info_addr=0x%x)", info_addr);
|
|
|
|
if(!Memory.IsGoodAddr(info_addr)) return CELL_EFAULT;
|
|
|
|
declCPU();
|
|
|
|
Memory.Write32(info_addr, CPU.GetStackAddr());
|
|
Memory.Write32(info_addr+4, CPU.GetStackSize());
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_stop(u32 thread_id)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_stop(thread_id=%d)", thread_id);
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
|
|
thr->Stop();
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_restart(u32 thread_id)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_restart(thread_id=%d)", thread_id);
|
|
|
|
CPUThread* thr = Emu.GetCPU().GetThread(thread_id);
|
|
if(!thr) return CELL_ESRCH;
|
|
|
|
thr->Stop();
|
|
thr->Run();
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
int sys_ppu_thread_create(u32 thread_id_addr, u32 entry, u64 arg, int prio, u32 stacksize, u64 flags, u32 threadname_addr)
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_create(thread_id_addr=0x%x, entry=0x%x, arg=0x%x, prio=%d, stacksize=0x%x, flags=0x%llx, threadname_addr=0x%x('%s'))",
|
|
thread_id_addr, entry, arg, prio, stacksize, flags, threadname_addr, Memory.ReadString(threadname_addr).c_str());
|
|
|
|
if(!Memory.IsGoodAddr(entry) || !Memory.IsGoodAddr(thread_id_addr) || !Memory.IsGoodAddr(threadname_addr))
|
|
{
|
|
return CELL_EFAULT;
|
|
}
|
|
|
|
CPUThread& new_thread = Emu.GetCPU().AddThread(CPU_THREAD_PPU);
|
|
|
|
Memory.Write64(thread_id_addr, new_thread.GetId());
|
|
new_thread.SetEntry(entry);
|
|
new_thread.SetArg(0, arg);
|
|
new_thread.SetPrio(prio);
|
|
new_thread.SetStackSize(stacksize);
|
|
//new_thread.flags = flags;
|
|
new_thread.SetName(Memory.ReadString(threadname_addr));
|
|
|
|
ConLog.Write("*** New PPU Thread [%s] (): id = %d", new_thread.GetName().c_str(), new_thread.GetId());
|
|
|
|
new_thread.Run();
|
|
new_thread.Exec();
|
|
|
|
return CELL_OK;
|
|
}
|
|
|
|
void sys_ppu_thread_once(u32 once_ctrl_addr, u32 entry)
|
|
{
|
|
sysPrxForUser->Warning("sys_ppu_thread_once(once_ctrl_addr=0x%x, entry=0x%x)", once_ctrl_addr, entry);
|
|
|
|
if(Memory.IsGoodAddr(once_ctrl_addr, 4) && Memory.Read32(once_ctrl_addr) == SYS_PPU_THREAD_ONCE_INIT)
|
|
{
|
|
Memory.Write32(once_ctrl_addr, SYS_PPU_THREAD_DONE_INIT);
|
|
|
|
CPUThread& new_thread = Emu.GetCPU().AddThread(CPU_THREAD_PPU);
|
|
new_thread.SetEntry(entry);
|
|
new_thread.Run();
|
|
new_thread.Exec();
|
|
|
|
//GetCurrentPPUThread().Wait(new_thread);
|
|
}
|
|
}
|
|
|
|
int sys_ppu_thread_get_id(const u32 id_addr)
|
|
{
|
|
sysPrxForUser->Log("sys_ppu_thread_get_id(id_addr=0x%x)", id_addr);
|
|
|
|
Memory.Write64(id_addr, GetCurrentPPUThread().GetId());
|
|
return CELL_OK;
|
|
}
|