2024-02-27 22:10:34 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
2024-07-14 04:38:20 +00:00
|
|
|
#include <mutex>
|
2024-09-09 10:23:16 +00:00
|
|
|
#include "common/arch.h"
|
2024-02-27 22:10:34 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/types.h"
|
2024-11-21 20:59:38 +00:00
|
|
|
#include "core/cpu_patches.h"
|
|
|
|
#include "core/libraries/kernel/threads/pthread.h"
|
2024-02-27 22:10:34 +00:00
|
|
|
#include "core/tls.h"
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include <windows.h>
|
2024-09-09 10:23:16 +00:00
|
|
|
#elif defined(__APPLE__) && defined(ARCH_X86_64)
|
2024-09-01 09:22:42 +00:00
|
|
|
#include <architecture/i386/table.h>
|
|
|
|
#include <boost/icl/interval_set.hpp>
|
|
|
|
#include <i386/user_ldt.h>
|
|
|
|
#include <sys/mman.h>
|
2024-09-09 10:23:16 +00:00
|
|
|
#elif !defined(ARCH_X86_64)
|
|
|
|
#include <pthread.h>
|
2024-02-27 22:10:34 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace Core {
|
|
|
|
|
2024-05-01 10:38:41 +00:00
|
|
|
#ifdef _WIN32
|
2024-07-14 04:38:20 +00:00
|
|
|
|
2024-09-09 10:23:16 +00:00
|
|
|
// Windows
|
|
|
|
|
2024-05-01 10:38:41 +00:00
|
|
|
static DWORD slot = 0;
|
2024-09-01 09:22:42 +00:00
|
|
|
static std::once_flag slot_alloc_flag;
|
2024-02-27 22:10:34 +00:00
|
|
|
|
2024-07-14 04:38:20 +00:00
|
|
|
static void AllocTcbKey() {
|
|
|
|
slot = TlsAlloc();
|
|
|
|
}
|
|
|
|
|
2024-09-01 09:22:42 +00:00
|
|
|
u32 GetTcbKey() {
|
|
|
|
std::call_once(slot_alloc_flag, &AllocTcbKey);
|
|
|
|
return slot;
|
|
|
|
}
|
|
|
|
|
2024-06-05 19:08:18 +00:00
|
|
|
void SetTcbBase(void* image_address) {
|
2024-07-14 04:38:20 +00:00
|
|
|
const BOOL result = TlsSetValue(GetTcbKey(), image_address);
|
2024-05-01 10:38:41 +00:00
|
|
|
ASSERT(result != 0);
|
2024-02-27 22:10:34 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 19:08:18 +00:00
|
|
|
Tcb* GetTcbBase() {
|
2024-07-14 04:38:20 +00:00
|
|
|
return reinterpret_cast<Tcb*>(TlsGetValue(GetTcbKey()));
|
2024-06-13 21:58:57 +00:00
|
|
|
}
|
|
|
|
|
2024-09-09 10:23:16 +00:00
|
|
|
#elif defined(__APPLE__) && defined(ARCH_X86_64)
|
|
|
|
|
|
|
|
// Apple x86_64
|
2024-07-09 09:18:34 +00:00
|
|
|
|
2024-09-01 09:22:42 +00:00
|
|
|
// Reserve space in the 32-bit address range for allocating TCB pages.
|
|
|
|
asm(".zerofill TCB_SPACE,TCB_SPACE,__guest_system,0x3FC000");
|
2024-07-09 09:18:34 +00:00
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
struct LdtPage {
|
|
|
|
void* tcb;
|
|
|
|
u16 index;
|
|
|
|
};
|
|
|
|
|
|
|
|
static constexpr uintptr_t ldt_region_base = 0x4000;
|
|
|
|
static constexpr size_t ldt_region_size = 0x3FC000;
|
2024-09-01 09:22:42 +00:00
|
|
|
static constexpr u16 ldt_block_size = 0x1000;
|
|
|
|
static constexpr u16 ldt_index_base = 8;
|
|
|
|
static constexpr u16 ldt_index_total = (ldt_region_size - ldt_region_base) / ldt_block_size;
|
|
|
|
|
|
|
|
static boost::icl::interval_set<u16> free_ldts{};
|
|
|
|
static std::mutex free_ldts_lock;
|
|
|
|
static std::once_flag ldt_region_init_flag;
|
2024-11-21 20:59:38 +00:00
|
|
|
static pthread_key_t ldt_page_slot = 0;
|
|
|
|
|
|
|
|
static void FreeLdtPage(void* raw) {
|
|
|
|
const auto* ldt_page = static_cast<LdtPage*>(raw);
|
2024-09-01 09:22:42 +00:00
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
std::unique_lock lock{free_ldts_lock};
|
|
|
|
free_ldts += ldt_page->index;
|
2024-09-01 09:22:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void InitLdtRegion() {
|
|
|
|
const void* result =
|
|
|
|
mmap(reinterpret_cast<void*>(ldt_region_base), ldt_region_size, PROT_READ | PROT_WRITE,
|
|
|
|
MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED, -1, 0);
|
|
|
|
ASSERT_MSG(result != MAP_FAILED, "Failed to map memory region for LDT entries.");
|
|
|
|
|
|
|
|
free_ldts +=
|
|
|
|
boost::icl::interval<u16>::right_open(ldt_index_base, ldt_index_base + ldt_index_total);
|
2024-11-21 20:59:38 +00:00
|
|
|
ASSERT_MSG(pthread_key_create(&ldt_page_slot, FreeLdtPage) == 0,
|
|
|
|
"Failed to create thread LDT page key: {}", errno);
|
2024-09-01 09:22:42 +00:00
|
|
|
}
|
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
void SetTcbBase(void* image_address) {
|
2024-09-01 09:22:42 +00:00
|
|
|
std::call_once(ldt_region_init_flag, InitLdtRegion);
|
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
auto* ldt_page = static_cast<LdtPage*>(pthread_getspecific(ldt_page_slot));
|
|
|
|
if (ldt_page != nullptr) {
|
|
|
|
// Update TCB pointer in existing page.
|
|
|
|
ldt_page->tcb = image_address;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-09-01 09:22:42 +00:00
|
|
|
// Allocate a new LDT index for the current thread.
|
|
|
|
u16 ldt_index;
|
|
|
|
{
|
|
|
|
std::unique_lock lock{free_ldts_lock};
|
|
|
|
ASSERT_MSG(!free_ldts.empty(), "Out of LDT space.");
|
|
|
|
ldt_index = first(*free_ldts.begin());
|
|
|
|
free_ldts -= ldt_index;
|
|
|
|
}
|
2024-11-21 20:59:38 +00:00
|
|
|
|
|
|
|
const uintptr_t addr = ldt_region_base + (ldt_index - ldt_index_base) * ldt_block_size;
|
2024-09-01 09:22:42 +00:00
|
|
|
|
|
|
|
// Create an LDT entry for the TCB.
|
2024-11-21 20:59:38 +00:00
|
|
|
ldt_entry ldt{};
|
|
|
|
ldt.data = {
|
2024-09-01 09:22:42 +00:00
|
|
|
.base00 = static_cast<u16>(addr),
|
|
|
|
.base16 = static_cast<u8>(addr >> 16),
|
|
|
|
.base24 = static_cast<u8>(addr >> 24),
|
|
|
|
.limit00 = static_cast<u16>(ldt_block_size - 1),
|
|
|
|
.limit16 = 0,
|
|
|
|
.type = DESC_DATA_WRITE,
|
|
|
|
.dpl = 3, // User accessible
|
|
|
|
.present = 1, // Segment present
|
|
|
|
.stksz = DESC_DATA_32B,
|
|
|
|
.granular = DESC_GRAN_BYTE,
|
2024-11-21 20:59:38 +00:00
|
|
|
};
|
2024-09-01 09:22:42 +00:00
|
|
|
int ret = i386_set_ldt(ldt_index, &ldt, 1);
|
|
|
|
ASSERT_MSG(ret == ldt_index,
|
2024-11-21 20:59:38 +00:00
|
|
|
"Failed to set LDT {} at {:#x} for TLS area: syscall returned {}, errno {}",
|
|
|
|
ldt_index, addr, ret, errno);
|
2024-09-01 09:22:42 +00:00
|
|
|
|
|
|
|
// Set the FS segment to the created LDT.
|
2024-11-21 20:59:38 +00:00
|
|
|
const sel_t new_selector{
|
2024-09-01 09:22:42 +00:00
|
|
|
.rpl = USER_PRIV,
|
|
|
|
.ti = SEL_LDT,
|
|
|
|
.index = ldt_index,
|
|
|
|
};
|
2024-11-21 20:59:38 +00:00
|
|
|
asm volatile("mov %0, %%fs" ::"r"(new_selector));
|
2024-09-01 09:22:42 +00:00
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
// Store the TCB base pointer and index in the created LDT area.
|
|
|
|
ldt_page = reinterpret_cast<LdtPage*>(addr);
|
|
|
|
ldt_page->tcb = image_address;
|
|
|
|
ldt_page->index = ldt_index;
|
2024-07-09 09:18:34 +00:00
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
ASSERT_MSG(pthread_setspecific(ldt_page_slot, ldt_page) == 0,
|
|
|
|
"Failed to store thread LDT page pointer: {}", errno);
|
2024-07-09 09:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Tcb* GetTcbBase() {
|
2024-09-01 09:22:42 +00:00
|
|
|
Tcb* tcb;
|
|
|
|
asm volatile("mov %%fs:0x0, %0" : "=r"(tcb));
|
|
|
|
return tcb;
|
2024-07-09 09:18:34 +00:00
|
|
|
}
|
|
|
|
|
2024-09-09 10:23:16 +00:00
|
|
|
#elif defined(ARCH_X86_64)
|
|
|
|
|
|
|
|
// Other POSIX x86_64
|
2024-06-13 21:58:57 +00:00
|
|
|
|
|
|
|
void SetTcbBase(void* image_address) {
|
2024-07-15 00:47:10 +00:00
|
|
|
asm volatile("wrgsbase %0" ::"r"(image_address) : "memory");
|
2024-06-13 21:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Tcb* GetTcbBase() {
|
2024-07-14 22:48:57 +00:00
|
|
|
Tcb* tcb;
|
2024-07-15 00:47:10 +00:00
|
|
|
asm volatile("rdgsbase %0" : "=r"(tcb)::"memory");
|
2024-07-14 22:48:57 +00:00
|
|
|
return tcb;
|
2024-06-13 21:58:57 +00:00
|
|
|
}
|
2024-02-27 22:10:34 +00:00
|
|
|
|
2024-09-09 10:23:16 +00:00
|
|
|
#else
|
|
|
|
|
|
|
|
// POSIX non-x86_64
|
|
|
|
// Just sets up a simple thread-local variable to store it, then instruction translation can point
|
|
|
|
// code to it.
|
|
|
|
|
|
|
|
static pthread_key_t slot = 0;
|
|
|
|
static std::once_flag slot_alloc_flag;
|
|
|
|
|
|
|
|
static void AllocTcbKey() {
|
|
|
|
ASSERT(pthread_key_create(&slot, nullptr) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_key_t GetTcbKey() {
|
|
|
|
std::call_once(slot_alloc_flag, &AllocTcbKey);
|
|
|
|
return slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetTcbBase(void* image_address) {
|
|
|
|
ASSERT(pthread_setspecific(GetTcbKey(), image_address) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Tcb* GetTcbBase() {
|
|
|
|
return static_cast<Tcb*>(pthread_getspecific(GetTcbKey()));
|
|
|
|
}
|
|
|
|
|
2024-06-13 21:58:57 +00:00
|
|
|
#endif
|
|
|
|
|
2024-11-21 20:59:38 +00:00
|
|
|
thread_local std::once_flag init_tls_flag;
|
|
|
|
|
|
|
|
void EnsureThreadInitialized() {
|
|
|
|
std::call_once(init_tls_flag, [] {
|
|
|
|
#ifdef ARCH_X86_64
|
|
|
|
InitializeThreadPatchStack();
|
|
|
|
#endif
|
|
|
|
SetTcbBase(Libraries::Kernel::g_curthread->tcb);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-02-27 22:10:34 +00:00
|
|
|
} // namespace Core
|