2024-02-23 21:32:32 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2023-11-11 10:13:43 +00:00
|
|
|
|
2024-05-12 16:36:40 +00:00
|
|
|
#include <chrono>
|
2024-04-29 12:16:42 +00:00
|
|
|
#include <thread>
|
2024-02-27 22:10:34 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/logging/log.h"
|
2024-04-13 21:35:48 +00:00
|
|
|
#include "common/singleton.h"
|
|
|
|
#include "core/libraries/error_codes.h"
|
|
|
|
#include "core/libraries/kernel/cpu_management.h"
|
|
|
|
#include "core/libraries/kernel/event_queues.h"
|
|
|
|
#include "core/libraries/kernel/file_system.h"
|
|
|
|
#include "core/libraries/kernel/libkernel.h"
|
|
|
|
#include "core/libraries/kernel/memory_management.h"
|
|
|
|
#include "core/libraries/kernel/thread_management.h"
|
|
|
|
#include "core/libraries/kernel/time_management.h"
|
|
|
|
#include "core/libraries/libs.h"
|
|
|
|
#include "core/linker.h"
|
2024-05-16 12:55:50 +00:00
|
|
|
#include "core/memory.h"
|
2023-11-05 23:11:54 +00:00
|
|
|
#ifdef _WIN64
|
2023-11-19 10:55:07 +00:00
|
|
|
#include <io.h>
|
2024-02-23 20:57:57 +00:00
|
|
|
#include <windows.h>
|
2024-02-14 22:52:57 +00:00
|
|
|
#else
|
|
|
|
#include <sys/mman.h>
|
2023-11-05 23:11:54 +00:00
|
|
|
#endif
|
2024-05-16 20:24:51 +00:00
|
|
|
#include <core/file_format/psf.h>
|
2023-07-13 09:56:36 +00:00
|
|
|
|
2024-04-13 21:35:48 +00:00
|
|
|
namespace Libraries::Kernel {
|
2023-07-13 09:56:36 +00:00
|
|
|
|
2024-02-23 20:57:57 +00:00
|
|
|
static u64 g_stack_chk_guard = 0xDEADBEEF54321ABC; // dummy return
|
2023-07-13 09:56:36 +00:00
|
|
|
|
2024-03-26 16:48:26 +00:00
|
|
|
static void* PS4_SYSV_ABI sceKernelGetProcParam() {
|
|
|
|
auto* linker = Common::Singleton<Core::Linker>::Instance();
|
|
|
|
return reinterpret_cast<void*>(linker->GetProcParam());
|
|
|
|
}
|
|
|
|
|
2023-10-06 18:49:53 +00:00
|
|
|
int32_t PS4_SYSV_ABI sceKernelReleaseDirectMemory(off_t start, size_t len) {
|
2024-02-27 22:10:34 +00:00
|
|
|
UNREACHABLE();
|
2023-10-06 18:49:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2023-07-14 11:29:13 +00:00
|
|
|
|
2024-02-23 20:57:57 +00:00
|
|
|
static PS4_SYSV_ABI void stack_chk_fail() {
|
2024-02-27 22:10:34 +00:00
|
|
|
UNREACHABLE();
|
2024-02-23 20:57:57 +00:00
|
|
|
}
|
2023-11-05 23:11:54 +00:00
|
|
|
|
2024-02-23 20:57:57 +00:00
|
|
|
int PS4_SYSV_ABI sceKernelMunmap(void* addr, size_t len) {
|
2024-05-16 12:55:50 +00:00
|
|
|
LOG_INFO(Kernel_Vmm, "addr = {}, len = {:#x}", fmt::ptr(addr), len);
|
|
|
|
auto* memory = Core::Memory::Instance();
|
|
|
|
memory->UnmapMemory(std::bit_cast<VAddr>(addr), len);
|
2024-05-05 09:59:26 +00:00
|
|
|
return SCE_OK;
|
2024-02-23 20:57:57 +00:00
|
|
|
}
|
2023-11-11 10:13:43 +00:00
|
|
|
|
2024-02-23 20:57:57 +00:00
|
|
|
void PS4_SYSV_ABI sceKernelUsleep(unsigned int microseconds) {
|
|
|
|
std::this_thread::sleep_for(std::chrono::microseconds(microseconds));
|
|
|
|
}
|
2023-11-19 10:55:07 +00:00
|
|
|
|
|
|
|
struct iovec {
|
|
|
|
void* iov_base; /* Base address. */
|
|
|
|
size_t iov_len; /* Length. */
|
|
|
|
};
|
|
|
|
|
|
|
|
size_t PS4_SYSV_ABI _writev(int fd, const struct iovec* iov, int iovcn) {
|
2024-02-23 20:57:57 +00:00
|
|
|
// weird it gives fd ==0 and writes to stdout , i am not sure if it that is valid (found in
|
|
|
|
// openorbis)
|
2023-11-19 10:55:07 +00:00
|
|
|
size_t total_written = 0;
|
|
|
|
for (int i = 0; i < iovcn; i++) {
|
|
|
|
total_written += ::fwrite(iov[i].iov_base, 1, iov[i].iov_len, stdout);
|
|
|
|
}
|
|
|
|
return total_written;
|
|
|
|
}
|
|
|
|
|
2023-11-11 10:13:43 +00:00
|
|
|
static thread_local int libc_error;
|
2024-02-23 20:57:57 +00:00
|
|
|
int* PS4_SYSV_ABI __Error() {
|
|
|
|
return &libc_error;
|
|
|
|
}
|
2023-10-30 18:22:25 +00:00
|
|
|
|
2023-11-19 10:55:07 +00:00
|
|
|
#define PROT_READ 0x1
|
|
|
|
#define PROT_WRITE 0x2
|
|
|
|
|
2024-02-23 20:57:57 +00:00
|
|
|
int PS4_SYSV_ABI sceKernelMmap(void* addr, u64 len, int prot, int flags, int fd, off_t offset,
|
|
|
|
void** res) {
|
2024-02-14 22:52:57 +00:00
|
|
|
#ifdef _WIN64
|
2024-02-27 22:10:34 +00:00
|
|
|
LOG_INFO(Kernel_Vmm, "called");
|
|
|
|
if (prot > 3) {
|
|
|
|
LOG_ERROR(Kernel_Vmm, "prot = {} not supported", prot);
|
2023-11-19 10:55:07 +00:00
|
|
|
}
|
|
|
|
DWORD flProtect;
|
|
|
|
if (prot & PROT_WRITE) {
|
|
|
|
flProtect = PAGE_READWRITE;
|
|
|
|
}
|
|
|
|
off_t end = len + offset;
|
|
|
|
HANDLE mmap_fd, h;
|
|
|
|
if (fd == -1)
|
|
|
|
mmap_fd = INVALID_HANDLE_VALUE;
|
|
|
|
else
|
|
|
|
mmap_fd = (HANDLE)_get_osfhandle(fd);
|
|
|
|
h = CreateFileMapping(mmap_fd, NULL, flProtect, 0, end, NULL);
|
|
|
|
int k = GetLastError();
|
2024-02-23 20:57:57 +00:00
|
|
|
if (NULL == h)
|
|
|
|
return -1;
|
2023-11-19 10:55:07 +00:00
|
|
|
DWORD dwDesiredAccess;
|
|
|
|
if (prot & PROT_WRITE)
|
|
|
|
dwDesiredAccess = FILE_MAP_WRITE;
|
|
|
|
else
|
|
|
|
dwDesiredAccess = FILE_MAP_READ;
|
|
|
|
void* ret = MapViewOfFile(h, dwDesiredAccess, 0, offset, len);
|
|
|
|
if (ret == NULL) {
|
|
|
|
CloseHandle(h);
|
|
|
|
ret = nullptr;
|
|
|
|
}
|
|
|
|
*res = ret;
|
|
|
|
return 0;
|
2024-02-14 22:52:57 +00:00
|
|
|
#else
|
|
|
|
void* result = mmap(addr, len, prot, flags, fd, offset);
|
|
|
|
if (result != MAP_FAILED) {
|
|
|
|
*res = result;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
std::abort();
|
|
|
|
#endif
|
2023-11-19 10:55:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PS4_SYSV_ABI void* posix_mmap(void* addr, u64 len, int prot, int flags, int fd, u64 offset) {
|
|
|
|
void* ptr;
|
2024-02-27 22:10:34 +00:00
|
|
|
LOG_INFO(Kernel_Vmm, "posix mmap redirect to sceKernelMmap\n");
|
2024-02-23 20:57:57 +00:00
|
|
|
// posix call the difference is that there is a different behaviour when it doesn't return 0 or
|
|
|
|
// SCE_OK
|
2023-11-19 10:55:07 +00:00
|
|
|
int result = sceKernelMmap(addr, len, prot, flags, fd, offset, &ptr);
|
2024-02-27 22:10:34 +00:00
|
|
|
ASSERT(result == 0);
|
2023-11-19 10:55:07 +00:00
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2024-03-26 17:19:52 +00:00
|
|
|
static uint64_t g_mspace_atomic_id_mask = 0;
|
|
|
|
static uint64_t g_mstate_table[64] = {0};
|
|
|
|
|
|
|
|
struct HeapInfoInfo {
|
|
|
|
uint64_t size = sizeof(HeapInfoInfo);
|
|
|
|
uint32_t flag;
|
|
|
|
uint32_t getSegmentInfo;
|
|
|
|
uint64_t* mspace_atomic_id_mask;
|
|
|
|
uint64_t* mstate_table;
|
|
|
|
};
|
|
|
|
|
|
|
|
void PS4_SYSV_ABI sceLibcHeapGetTraceInfo(HeapInfoInfo* info) {
|
|
|
|
info->mspace_atomic_id_mask = &g_mspace_atomic_id_mask;
|
|
|
|
info->mstate_table = g_mstate_table;
|
|
|
|
info->getSegmentInfo = 0;
|
|
|
|
}
|
|
|
|
|
2024-04-13 21:35:48 +00:00
|
|
|
s64 PS4_SYSV_ABI ps4__write(int d, const void* buf, std::size_t nbytes) {
|
|
|
|
if (d <= 2) { // stdin,stdout,stderr
|
|
|
|
char* str = strdup((const char*)buf);
|
|
|
|
if (str[nbytes - 1] == '\n')
|
|
|
|
str[nbytes - 1] = 0;
|
|
|
|
LOG_INFO(Tty, "{}", str);
|
|
|
|
free(str);
|
|
|
|
return nbytes;
|
|
|
|
}
|
|
|
|
LOG_ERROR(Kernel, "(STUBBED) called d = {} nbytes = {} ", d, nbytes);
|
|
|
|
UNREACHABLE(); // normal write , is it a posix call??
|
|
|
|
return ORBIS_OK;
|
|
|
|
}
|
|
|
|
|
2024-05-12 16:03:51 +00:00
|
|
|
int PS4_SYSV_ABI sceKernelConvertUtcToLocaltime(time_t time, time_t* local_time,
|
|
|
|
struct OrbisTimesec* st, unsigned long* dst_sec) {
|
|
|
|
LOG_TRACE(Kernel, "Called");
|
|
|
|
const auto* time_zone = std::chrono::current_zone();
|
|
|
|
auto info = time_zone->get_info(std::chrono::system_clock::now());
|
|
|
|
|
|
|
|
*local_time = info.offset.count() + info.save.count() * 60 + time;
|
|
|
|
|
|
|
|
if (st != nullptr) {
|
|
|
|
st->t = time;
|
|
|
|
st->west_sec = info.offset.count() * 60;
|
|
|
|
st->dst_sec = info.save.count() * 60;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dst_sec != nullptr) {
|
|
|
|
*dst_sec = info.save.count() * 60;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ORBIS_OK;
|
|
|
|
}
|
|
|
|
|
2024-05-17 05:01:02 +00:00
|
|
|
int PS4_SYSV_ABI sceKernelGetCompiledSdkVersion(int* ver) {
|
2024-05-16 20:24:51 +00:00
|
|
|
auto* param_sfo = Common::Singleton<PSF>::Instance();
|
|
|
|
int version = param_sfo->GetInteger("SYSTEM_VER");
|
|
|
|
LOG_INFO(Kernel, "returned system version = {:#x}", version);
|
2024-05-17 05:01:02 +00:00
|
|
|
*ver = version;
|
|
|
|
return ORBIS_OK;
|
2024-05-16 20:24:51 +00:00
|
|
|
}
|
2024-04-13 21:35:48 +00:00
|
|
|
void LibKernel_Register(Core::Loader::SymbolsResolver* sym) {
|
2023-10-06 18:49:53 +00:00
|
|
|
// obj
|
2023-11-05 23:11:54 +00:00
|
|
|
LIB_OBJ("f7uOxY9mM1U", "libkernel", 1, "libkernel", 1, 1, &g_stack_chk_guard);
|
2023-10-06 18:49:53 +00:00
|
|
|
// memory
|
2024-04-13 21:35:48 +00:00
|
|
|
LIB_FUNCTION("rTXw65xmLIA", "libkernel", 1, "libkernel", 1, 1, sceKernelAllocateDirectMemory);
|
|
|
|
LIB_FUNCTION("pO96TwzOm5E", "libkernel", 1, "libkernel", 1, 1, sceKernelGetDirectMemorySize);
|
|
|
|
LIB_FUNCTION("L-Q3LEjIbgA", "libkernel", 1, "libkernel", 1, 1, sceKernelMapDirectMemory);
|
2023-10-06 18:49:53 +00:00
|
|
|
LIB_FUNCTION("MBuItvba6z8", "libkernel", 1, "libkernel", 1, 1, sceKernelReleaseDirectMemory);
|
|
|
|
LIB_FUNCTION("cQke9UuBQOk", "libkernel", 1, "libkernel", 1, 1, sceKernelMunmap);
|
2024-05-16 12:55:50 +00:00
|
|
|
LIB_FUNCTION("mL8NDH86iQI", "libkernel", 1, "libkernel", 1, 1, sceKernelMapNamedFlexibleMemory);
|
|
|
|
LIB_FUNCTION("IWIBBdTHit4", "libkernel", 1, "libkernel", 1, 1, sceKernelMapFlexibleMemory);
|
2023-10-06 18:49:53 +00:00
|
|
|
// equeue
|
2024-04-13 21:35:48 +00:00
|
|
|
LIB_FUNCTION("D0OdFMjp46I", "libkernel", 1, "libkernel", 1, 1, sceKernelCreateEqueue);
|
2024-05-10 20:04:41 +00:00
|
|
|
LIB_FUNCTION("jpFjmgAC5AE", "libkernel", 1, "libkernel", 1, 1, sceKernelDeleteEqueue);
|
2024-04-13 21:35:48 +00:00
|
|
|
LIB_FUNCTION("fzyMKs9kim0", "libkernel", 1, "libkernel", 1, 1, sceKernelWaitEqueue);
|
2024-05-17 21:32:15 +00:00
|
|
|
LIB_FUNCTION("vz+pg2zdopI", "libkernel", 1, "libkernel", 1, 1, sceKernelGetEventUserData);
|
|
|
|
LIB_FUNCTION("4R6-OvI2cEA", "libkernel", 1, "libkernel", 1, 1, sceKernelAddUserEvent);
|
2023-10-06 18:49:53 +00:00
|
|
|
// misc
|
2024-04-13 21:35:48 +00:00
|
|
|
LIB_FUNCTION("WslcK1FQcGI", "libkernel", 1, "libkernel", 1, 1, sceKernelIsNeoMode);
|
2023-10-06 18:49:53 +00:00
|
|
|
LIB_FUNCTION("Ou3iL1abvng", "libkernel", 1, "libkernel", 1, 1, stack_chk_fail);
|
2023-11-11 10:13:43 +00:00
|
|
|
LIB_FUNCTION("9BcDykPmo1I", "libkernel", 1, "libkernel", 1, 1, __Error);
|
2023-11-19 10:55:07 +00:00
|
|
|
LIB_FUNCTION("BPE9s9vQQXo", "libkernel", 1, "libkernel", 1, 1, posix_mmap);
|
|
|
|
LIB_FUNCTION("1jfXLRVzisc", "libkernel", 1, "libkernel", 1, 1, sceKernelUsleep);
|
|
|
|
LIB_FUNCTION("YSHRBRLn2pI", "libkernel", 1, "libkernel", 1, 1, _writev);
|
2024-03-26 16:48:26 +00:00
|
|
|
LIB_FUNCTION("959qrazPIrg", "libkernel", 1, "libkernel", 1, 1, sceKernelGetProcParam);
|
2024-05-12 16:03:51 +00:00
|
|
|
LIB_FUNCTION("-o5uEDpN+oY", "libkernel", 1, "libkernel", 1, 1, sceKernelConvertUtcToLocaltime);
|
2024-05-16 20:24:51 +00:00
|
|
|
LIB_FUNCTION("WB66evu8bsU", "libkernel", 1, "libkernel", 1, 1, sceKernelGetCompiledSdkVersion);
|
2023-11-11 10:13:43 +00:00
|
|
|
|
2024-04-13 21:35:48 +00:00
|
|
|
Libraries::Kernel::fileSystemSymbolsRegister(sym);
|
|
|
|
Libraries::Kernel::timeSymbolsRegister(sym);
|
|
|
|
Libraries::Kernel::pthreadSymbolsRegister(sym);
|
2024-03-26 17:19:52 +00:00
|
|
|
|
|
|
|
// temp
|
|
|
|
LIB_FUNCTION("NWtTN10cJzE", "libSceLibcInternalExt", 1, "libSceLibcInternal", 1, 1,
|
|
|
|
sceLibcHeapGetTraceInfo);
|
2024-04-13 21:35:48 +00:00
|
|
|
LIB_FUNCTION("FxVZqBAA7ks", "libkernel", 1, "libkernel", 1, 1, ps4__write);
|
2023-10-06 18:49:53 +00:00
|
|
|
}
|
2023-07-13 09:56:36 +00:00
|
|
|
|
2024-04-13 21:35:48 +00:00
|
|
|
} // namespace Libraries::Kernel
|