mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-08 21:47:57 +00:00
kern: mostly implement KThread::Initialize
This commit is contained in:
parent
d9db723bc8
commit
b2b1129cc0
11 changed files with 383 additions and 15 deletions
|
@ -26,6 +26,11 @@ namespace ams::kern {
|
||||||
class KProcess final : public KAutoObjectWithSlabHeapAndContainer<KProcess, KSynchronizationObject> {
|
class KProcess final : public KAutoObjectWithSlabHeapAndContainer<KProcess, KSynchronizationObject> {
|
||||||
MESOSPHERE_AUTOOBJECT_TRAITS(KProcess, KSynchronizationObject);
|
MESOSPHERE_AUTOOBJECT_TRAITS(KProcess, KSynchronizationObject);
|
||||||
/* TODO: This is a placeholder definition. */
|
/* TODO: This is a placeholder definition. */
|
||||||
|
public:
|
||||||
|
constexpr ALWAYS_INLINE u64 GetCoreMask() const { /* TODO */ return 0; }
|
||||||
|
constexpr ALWAYS_INLINE u64 GetPriorityMask() const { /* TODO */ return 0; }
|
||||||
|
|
||||||
|
constexpr ALWAYS_INLINE bool Is64Bit() const { /* TODO */ return true; }
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,12 +19,19 @@
|
||||||
#include <mesosphere/kern_k_affinity_mask.hpp>
|
#include <mesosphere/kern_k_affinity_mask.hpp>
|
||||||
#include <mesosphere/kern_k_thread_context.hpp>
|
#include <mesosphere/kern_k_thread_context.hpp>
|
||||||
#include <mesosphere/kern_k_current_context.hpp>
|
#include <mesosphere/kern_k_current_context.hpp>
|
||||||
|
#include <mesosphere/kern_k_timer_task.hpp>
|
||||||
|
#include <mesosphere/kern_k_worker_task.hpp>
|
||||||
|
|
||||||
namespace ams::kern {
|
namespace ams::kern {
|
||||||
|
|
||||||
class KThread final : public KAutoObjectWithSlabHeapAndContainer<KThread, KSynchronizationObject> {
|
using KThreadFunction = void (*)(uintptr_t);
|
||||||
|
|
||||||
|
class KThread final : public KAutoObjectWithSlabHeapAndContainer<KThread, KSynchronizationObject>, public KTimerTask, public KWorkerTask {
|
||||||
MESOSPHERE_AUTOOBJECT_TRAITS(KThread, KSynchronizationObject);
|
MESOSPHERE_AUTOOBJECT_TRAITS(KThread, KSynchronizationObject);
|
||||||
public:
|
public:
|
||||||
|
static constexpr s32 MainThreadPriority = 1;
|
||||||
|
static constexpr s32 IdleThreadPriority = 64;
|
||||||
|
|
||||||
enum ThreadType : u32 {
|
enum ThreadType : u32 {
|
||||||
ThreadType_Main = 0,
|
ThreadType_Main = 0,
|
||||||
ThreadType_Kernel = 1,
|
ThreadType_Kernel = 1,
|
||||||
|
@ -36,6 +43,10 @@ namespace ams::kern {
|
||||||
SuspendType_Process = 0,
|
SuspendType_Process = 0,
|
||||||
SuspendType_Thread = 1,
|
SuspendType_Thread = 1,
|
||||||
SuspendType_Debug = 2,
|
SuspendType_Debug = 2,
|
||||||
|
SuspendType_Unk3 = 3,
|
||||||
|
SuspendType_Unk4 = 4,
|
||||||
|
|
||||||
|
SuspendType_Count,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum ThreadState : u16 {
|
enum ThreadState : u16 {
|
||||||
|
@ -50,6 +61,10 @@ namespace ams::kern {
|
||||||
ThreadState_ProcessSuspended = (1 << (SuspendType_Process + ThreadState_SuspendShift)),
|
ThreadState_ProcessSuspended = (1 << (SuspendType_Process + ThreadState_SuspendShift)),
|
||||||
ThreadState_ThreadSuspended = (1 << (SuspendType_Thread + ThreadState_SuspendShift)),
|
ThreadState_ThreadSuspended = (1 << (SuspendType_Thread + ThreadState_SuspendShift)),
|
||||||
ThreadState_DebugSuspended = (1 << (SuspendType_Debug + ThreadState_SuspendShift)),
|
ThreadState_DebugSuspended = (1 << (SuspendType_Debug + ThreadState_SuspendShift)),
|
||||||
|
ThreadState_Unk3Suspended = (1 << (SuspendType_Unk3 + ThreadState_SuspendShift)),
|
||||||
|
ThreadState_Unk4Suspended = (1 << (SuspendType_Unk4 + ThreadState_SuspendShift)),
|
||||||
|
|
||||||
|
ThreadState_SuspendFlagMask = ((1 << SuspendType_Count) - 1) << ThreadState_SuspendShift,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum DpcFlag : u32 {
|
enum DpcFlag : u32 {
|
||||||
|
@ -76,6 +91,11 @@ namespace ams::kern {
|
||||||
public:
|
public:
|
||||||
constexpr ALWAYS_INLINE QueueEntry() : prev(nullptr), next(nullptr) { /* ... */ }
|
constexpr ALWAYS_INLINE QueueEntry() : prev(nullptr), next(nullptr) { /* ... */ }
|
||||||
|
|
||||||
|
constexpr ALWAYS_INLINE void Initialize() {
|
||||||
|
this->prev = nullptr;
|
||||||
|
this->next = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
constexpr ALWAYS_INLINE KThread *GetPrev() const { return this->prev; }
|
constexpr ALWAYS_INLINE KThread *GetPrev() const { return this->prev; }
|
||||||
constexpr ALWAYS_INLINE KThread *GetNext() const { return this->next; }
|
constexpr ALWAYS_INLINE KThread *GetNext() const { return this->next; }
|
||||||
constexpr ALWAYS_INLINE void SetPrev(KThread *t) { this->prev = t; }
|
constexpr ALWAYS_INLINE void SetPrev(KThread *t) { this->prev = t; }
|
||||||
|
@ -90,6 +110,8 @@ namespace ams::kern {
|
||||||
constexpr SyncObjectBuffer() : sync_objects() { /* ... */ }
|
constexpr SyncObjectBuffer() : sync_objects() { /* ... */ }
|
||||||
};
|
};
|
||||||
static_assert(sizeof(SyncObjectBuffer::sync_objects) == sizeof(SyncObjectBuffer::handles));
|
static_assert(sizeof(SyncObjectBuffer::sync_objects) == sizeof(SyncObjectBuffer::handles));
|
||||||
|
private:
|
||||||
|
static inline std::atomic<u64> s_next_thread_id = 0;
|
||||||
private:
|
private:
|
||||||
alignas(16) KThreadContext thread_context;
|
alignas(16) KThreadContext thread_context;
|
||||||
KAffinityMask affinity_mask;
|
KAffinityMask affinity_mask;
|
||||||
|
@ -141,25 +163,27 @@ namespace ams::kern {
|
||||||
std::atomic<bool> termination_requested;
|
std::atomic<bool> termination_requested;
|
||||||
bool ipc_cancelled;
|
bool ipc_cancelled;
|
||||||
bool wait_cancelled;
|
bool wait_cancelled;
|
||||||
bool cancelable;
|
bool cancellable;
|
||||||
bool registered;
|
bool registered;
|
||||||
bool signaled;
|
bool signaled;
|
||||||
bool initialized;
|
bool initialized;
|
||||||
bool debug_attached;
|
bool debug_attached;
|
||||||
s8 priority_inheritance_count;
|
s8 priority_inheritance_count;
|
||||||
bool resource_limit_release_hint;
|
bool resource_limit_release_hint;
|
||||||
public:
|
|
||||||
static void PostDestroy(uintptr_t arg);
|
|
||||||
public:
|
public:
|
||||||
explicit KThread() /* TODO: : ? */ { MESOSPHERE_ASSERT_THIS(); }
|
explicit KThread() /* TODO: : ? */ { MESOSPHERE_ASSERT_THIS(); }
|
||||||
|
virtual ~KThread() { /* ... */ }
|
||||||
/* TODO: Is a constexpr KThread() possible? */
|
/* TODO: Is a constexpr KThread() possible? */
|
||||||
|
|
||||||
|
Result Initialize(KThreadFunction func, uintptr_t arg, void *kern_stack_top, KProcessAddress user_stack_top, s32 prio, s32 core, KProcess *owner, ThreadType type);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
StackParameters &GetStackParameters() {
|
StackParameters &GetStackParameters() {
|
||||||
return *(reinterpret_cast<StackParameters *>(this->kernel_stack_top) - 1);
|
return *(reinterpret_cast<StackParameters *>(this->kernel_stack_top) - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
const StackParameters &GetStackParameters() const {
|
const StackParameters &GetStackParameters() const {
|
||||||
return *(reinterpret_cast<StackParameters *>(this->kernel_stack_top) - 1);
|
return *(reinterpret_cast<const StackParameters *>(this->kernel_stack_top) - 1);
|
||||||
}
|
}
|
||||||
public:
|
public:
|
||||||
ALWAYS_INLINE s32 GetDisableDispatchCount() const {
|
ALWAYS_INLINE s32 GetDisableDispatchCount() const {
|
||||||
|
@ -178,13 +202,33 @@ namespace ams::kern {
|
||||||
MESOSPHERE_ASSERT(GetCurrentThread().GetDisableDispatchCount() > 0);
|
MESOSPHERE_ASSERT(GetCurrentThread().GetDisableDispatchCount() > 0);
|
||||||
GetStackParameters().disable_count--;
|
GetStackParameters().disable_count--;
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
constexpr ALWAYS_INLINE const KAffinityMask &GetAffinityMask() const { return this->affinity_mask; }
|
constexpr ALWAYS_INLINE const KAffinityMask &GetAffinityMask() const { return this->affinity_mask; }
|
||||||
|
|
||||||
|
ALWAYS_INLINE void *GetStackTop() const { return reinterpret_cast<StackParameters *>(this->kernel_stack_top) - 1; }
|
||||||
|
ALWAYS_INLINE void *GetKernelStackTop() const { return this->kernel_stack_top; }
|
||||||
|
|
||||||
|
/* TODO: This is kind of a placeholder definition. */
|
||||||
|
|
||||||
/* TODO: This is a placeholder definition. */
|
ALWAYS_INLINE bool IsInExceptionHandler() const {
|
||||||
|
return GetStackParameters().is_in_exception_handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
ALWAYS_INLINE void SetInExceptionHandler() {
|
||||||
|
GetStackParameters().is_in_exception_handler = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
/* Overridden parent functions. */
|
||||||
|
virtual bool IsInitialized() const override { return this->initialized; }
|
||||||
|
virtual uintptr_t GetPostDestroyArgument() const override { return reinterpret_cast<uintptr_t>(this->parent) | (this->resource_limit_release_hint ? 1 : 0); }
|
||||||
|
|
||||||
|
static void PostDestroy(uintptr_t arg);
|
||||||
|
|
||||||
|
virtual void Finalize() override;
|
||||||
|
virtual bool IsSignaled() const override;
|
||||||
|
virtual void OnTimer() override;
|
||||||
|
virtual void DoWorkerTask() override;
|
||||||
public:
|
public:
|
||||||
static constexpr bool IsWaiterListValid() {
|
static constexpr bool IsWaiterListValid() {
|
||||||
return WaiterListTraits::IsValid();
|
return WaiterListTraits::IsValid();
|
||||||
|
|
|
@ -0,0 +1,33 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include <mesosphere/kern_common.hpp>
|
||||||
|
|
||||||
|
namespace ams::kern {
|
||||||
|
|
||||||
|
class KWorkerTask {
|
||||||
|
private:
|
||||||
|
KWorkerTask *next_task;
|
||||||
|
public:
|
||||||
|
constexpr ALWAYS_INLINE KWorkerTask() : next_task(nullptr) { /* ... */ }
|
||||||
|
|
||||||
|
constexpr ALWAYS_INLINE KWorkerTask *GetNextTask() const { return this->next_task; }
|
||||||
|
constexpr ALWAYS_INLINE void SetNextTask(KWorkerTask *task) { this->next_task = task; }
|
||||||
|
|
||||||
|
virtual void DoWorkerTask() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -14,10 +14,13 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include <mesosphere/kern_common.hpp>
|
||||||
#include <mesosphere/kern_k_typed_address.hpp>
|
#include <mesosphere/kern_k_typed_address.hpp>
|
||||||
|
#include <mesosphere/kern_select_cpu.hpp>
|
||||||
#include <mesosphere/kern_k_memory_layout.hpp>
|
#include <mesosphere/kern_k_memory_layout.hpp>
|
||||||
#include <mesosphere/kern_k_memory_manager.hpp>
|
#include <mesosphere/kern_k_memory_manager.hpp>
|
||||||
#include <mesosphere/kern_k_core_local_region.hpp>
|
#include <mesosphere/kern_k_core_local_region.hpp>
|
||||||
|
#include <mesosphere/kern_k_thread.hpp>
|
||||||
|
|
||||||
namespace ams::kern {
|
namespace ams::kern {
|
||||||
|
|
||||||
|
@ -30,12 +33,22 @@ namespace ams::kern {
|
||||||
};
|
};
|
||||||
private:
|
private:
|
||||||
static inline State s_state = State::Invalid;
|
static inline State s_state = State::Invalid;
|
||||||
|
static inline KThread s_main_threads[cpu::NumCores];
|
||||||
|
static inline KThread s_idle_threads[cpu::NumCores];
|
||||||
public:
|
public:
|
||||||
static NOINLINE void Initialize(s32 core_id);
|
static NOINLINE void InitializeCoreLocalRegion(s32 core_id);
|
||||||
static NOINLINE void InitializeCoreThreads(s32 core_id);
|
static NOINLINE void InitializeMainAndIdleThreads(s32 core_id);
|
||||||
|
|
||||||
static ALWAYS_INLINE State GetState() { return s_state; }
|
static ALWAYS_INLINE State GetState() { return s_state; }
|
||||||
static ALWAYS_INLINE void SetState(State state) { s_state = state; }
|
static ALWAYS_INLINE void SetState(State state) { s_state = state; }
|
||||||
|
|
||||||
|
static ALWAYS_INLINE KThread &GetMainThread(s32 core_id) {
|
||||||
|
return s_main_threads[core_id];
|
||||||
|
}
|
||||||
|
|
||||||
|
static ALWAYS_INLINE KThread &GetIdleThread(s32 core_id) {
|
||||||
|
return s_idle_threads[core_id];
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,8 @@ namespace ams::kern::svc {
|
||||||
|
|
||||||
/* 33 */ using ::ams::svc::ResultNotImplemented;
|
/* 33 */ using ::ams::svc::ResultNotImplemented;
|
||||||
|
|
||||||
|
/* 57 */ using ::ams::svc::ResultNoSynchronizationObject;
|
||||||
|
|
||||||
/* 59 */ using ::ams::svc::ResultThreadTerminating;
|
/* 59 */ using ::ams::svc::ResultThreadTerminating;
|
||||||
|
|
||||||
/* 70 */ using ::ams::svc::ResultNoEvent;
|
/* 70 */ using ::ams::svc::ResultNoEvent;
|
||||||
|
|
205
libraries/libmesosphere/source/kern_k_thread.cpp
Normal file
205
libraries/libmesosphere/source/kern_k_thread.cpp
Normal file
|
@ -0,0 +1,205 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
#include <mesosphere.hpp>
|
||||||
|
|
||||||
|
namespace ams::kern {
|
||||||
|
|
||||||
|
Result KThread::Initialize(KThreadFunction func, uintptr_t arg, void *kern_stack_top, KProcessAddress user_stack_top, s32 prio, s32 core, KProcess *owner, ThreadType type) {
|
||||||
|
/* Assert parameters are valid. */
|
||||||
|
MESOSPHERE_ASSERT_THIS();
|
||||||
|
MESOSPHERE_ASSERT(kern_stack_top != nullptr);
|
||||||
|
MESOSPHERE_ASSERT((type == ThreadType_Main) || (ams::svc::HighestThreadPriority <= prio && prio <= ams::svc::LowestThreadPriority));
|
||||||
|
MESOSPHERE_ASSERT((owner != nullptr) || (type != ThreadType_User));
|
||||||
|
MESOSPHERE_ASSERT(0 <= core && core < static_cast<s32>(cpu::NumCores));
|
||||||
|
|
||||||
|
/* First, clear the TLS address. */
|
||||||
|
this->tls_address = Null<KProcessAddress>;
|
||||||
|
|
||||||
|
const uintptr_t kern_stack_top_address = reinterpret_cast<uintptr_t>(kern_stack_top);
|
||||||
|
|
||||||
|
/* Next, assert things based on the type. */
|
||||||
|
switch (type) {
|
||||||
|
case ThreadType_Main:
|
||||||
|
{
|
||||||
|
MESOSPHERE_ASSERT(arg == 0);
|
||||||
|
}
|
||||||
|
[[fallthrough]];
|
||||||
|
case ThreadType_HighPriority:
|
||||||
|
{
|
||||||
|
MESOSPHERE_ASSERT(core == GetCurrentCoreId());
|
||||||
|
}
|
||||||
|
[[fallthrough]];
|
||||||
|
case ThreadType_Kernel:
|
||||||
|
{
|
||||||
|
MESOSPHERE_ASSERT(user_stack_top == 0);
|
||||||
|
MESOSPHERE_ASSERT(util::IsAligned(kern_stack_top_address, PageSize));
|
||||||
|
}
|
||||||
|
[[fallthrough]];
|
||||||
|
case ThreadType_User:
|
||||||
|
{
|
||||||
|
MESOSPHERE_ASSERT((owner == nullptr) || (owner->GetCoreMask() | (1ul << core)) == owner->GetCoreMask());
|
||||||
|
MESOSPHERE_ASSERT((owner == nullptr) || (owner->GetPriorityMask() | (1ul << prio)) == owner->GetPriorityMask());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
MESOSPHERE_PANIC("KThread::Initialize: Unknown ThreadType %u", static_cast<u32>(type));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set the ideal core ID and affinity mask. */
|
||||||
|
this->ideal_core_id = core;
|
||||||
|
this->affinity_mask.SetAffinity(core, true);
|
||||||
|
|
||||||
|
/* Set the thread state. */
|
||||||
|
this->thread_state = (type == ThreadType_Main) ? ThreadState_Runnable : ThreadState_Initialized;
|
||||||
|
|
||||||
|
/* Set TLS address and TLS heap address. */
|
||||||
|
/* NOTE: Nintendo wrote TLS address above already, but official code really does write tls address twice. */
|
||||||
|
this->tls_address = 0;
|
||||||
|
this->tls_heap_address = 0;
|
||||||
|
|
||||||
|
/* Set parent and condvar tree. */
|
||||||
|
this->parent = nullptr;
|
||||||
|
this->cond_var_tree = nullptr;
|
||||||
|
|
||||||
|
/* Set sync booleans. */
|
||||||
|
this->signaled = false;
|
||||||
|
this->ipc_cancelled = false;
|
||||||
|
this->termination_requested = false;
|
||||||
|
this->wait_cancelled = false;
|
||||||
|
this->cancellable = false;
|
||||||
|
|
||||||
|
/* Set core ID and wait result. */
|
||||||
|
this->core_id = this->ideal_core_id;
|
||||||
|
this->wait_result = svc::ResultNoSynchronizationObject();
|
||||||
|
|
||||||
|
/* Set the stack top. */
|
||||||
|
this->kernel_stack_top = kern_stack_top;
|
||||||
|
|
||||||
|
/* Set priorities. */
|
||||||
|
this->priority = prio;
|
||||||
|
this->base_priority = prio;
|
||||||
|
|
||||||
|
/* Set sync object and waiting lock to null. */
|
||||||
|
this->synced_object = nullptr;
|
||||||
|
this->waiting_lock = nullptr;
|
||||||
|
|
||||||
|
/* Initialize sleeping queue. */
|
||||||
|
this->sleeping_queue_entry.Initialize();
|
||||||
|
this->sleeping_queue = nullptr;
|
||||||
|
|
||||||
|
/* Set suspend flags. */
|
||||||
|
this->suspend_request_flags = 0;
|
||||||
|
this->suspend_allowed_flags = ThreadState_SuspendFlagMask;
|
||||||
|
|
||||||
|
/* We're neither debug attached, nor are we nesting our priority inheritance. */
|
||||||
|
this->debug_attached = false;
|
||||||
|
this->priority_inheritance_count = 0;
|
||||||
|
|
||||||
|
/* We haven't been scheduled, and we have done no light IPC. */
|
||||||
|
this->schedule_count = -1;
|
||||||
|
this->last_scheduled_tick = 0;
|
||||||
|
this->light_ipc_data = nullptr;
|
||||||
|
|
||||||
|
/* We're not waiting for a lock, and we haven't disabled migration. */
|
||||||
|
this->lock_owner = nullptr;
|
||||||
|
this->num_core_migration_disables = 0;
|
||||||
|
|
||||||
|
/* We have no waiters, but we do have an entrypoint. */
|
||||||
|
this->num_kernel_waiters = 0;
|
||||||
|
this->entrypoint = reinterpret_cast<uintptr_t>(func);
|
||||||
|
|
||||||
|
/* We don't need a release (probably), and we've spent no time on the cpu. */
|
||||||
|
this->resource_limit_release_hint = 0;
|
||||||
|
this->cpu_time = 0;
|
||||||
|
|
||||||
|
/* Clear our stack parameters. */
|
||||||
|
std::memset(static_cast<void *>(std::addressof(this->GetStackParameters())), 0, sizeof(StackParameters));
|
||||||
|
|
||||||
|
/* Setup the TLS, if needed. */
|
||||||
|
if (type == ThreadType_User) {
|
||||||
|
/* TODO: R_TRY(owner->CreateThreadLocalRegion(&this->tls_address)); */
|
||||||
|
/* TODO: this->tls_heap_address = owner->GetThreadLocalRegionAddress(this->tls_address); */
|
||||||
|
std::memset(this->tls_heap_address, 0, ams::svc::ThreadLocalRegionSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set parent, if relevant. */
|
||||||
|
if (owner != nullptr) {
|
||||||
|
this->parent = owner;
|
||||||
|
this->parent->Open();
|
||||||
|
/* TODO: this->parent->IncrementThreadCount(); */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Initialize thread context. */
|
||||||
|
constexpr bool IsDefault64Bit = sizeof(uintptr_t) == sizeof(u64);
|
||||||
|
const bool is_64_bit = this->parent ? this->parent->Is64Bit() : IsDefault64Bit;
|
||||||
|
const bool is_user = (type == ThreadType_User);
|
||||||
|
const bool is_main = (type == ThreadType_Main);
|
||||||
|
this->thread_context.Initialize(this->entrypoint, reinterpret_cast<uintptr_t>(this->GetStackTop()), GetInteger(user_stack_top), arg, is_user, is_64_bit, is_main);
|
||||||
|
|
||||||
|
/* Setup the stack parameters. */
|
||||||
|
StackParameters &sp = this->GetStackParameters();
|
||||||
|
if (this->parent != nullptr) {
|
||||||
|
/* TODO: this->parent->CopySvcPermissionTo(pos.svc_permission); */
|
||||||
|
}
|
||||||
|
sp.context = std::addressof(this->thread_context);
|
||||||
|
sp.disable_count = 1;
|
||||||
|
this->SetInExceptionHandler();
|
||||||
|
|
||||||
|
/* Set thread ID. */
|
||||||
|
this->thread_id = s_next_thread_id++;
|
||||||
|
|
||||||
|
/* We initialized! */
|
||||||
|
this->initialized = true;
|
||||||
|
|
||||||
|
/* Register ourselves with our parent process. */
|
||||||
|
if (this->parent != nullptr) {
|
||||||
|
/* TODO: this->parent->RegisterThread(this); */
|
||||||
|
/* TODO: if (this->parent->IsSuspended()) { this->RequestSuspend(SuspendType_Process); } */
|
||||||
|
}
|
||||||
|
|
||||||
|
return ResultSuccess();
|
||||||
|
}
|
||||||
|
|
||||||
|
void KThread::PostDestroy(uintptr_t arg) {
|
||||||
|
KProcess *owner = reinterpret_cast<KProcess *>(arg & ~1ul);
|
||||||
|
const bool resource_limit_release_hint = (arg & 1);
|
||||||
|
if (owner != nullptr) {
|
||||||
|
/* TODO: Release from owner resource limit. */
|
||||||
|
(void)(resource_limit_release_hint);
|
||||||
|
owner->Close();
|
||||||
|
} else {
|
||||||
|
/* TODO: Release from system resource limit. */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void KThread::Finalize() {
|
||||||
|
/* TODO */
|
||||||
|
}
|
||||||
|
|
||||||
|
bool KThread::IsSignaled() const {
|
||||||
|
return this->signaled;
|
||||||
|
}
|
||||||
|
|
||||||
|
void KThread::OnTimer() {
|
||||||
|
/* TODO */
|
||||||
|
}
|
||||||
|
|
||||||
|
void KThread::DoWorkerTask() {
|
||||||
|
/* TODO */
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -17,7 +17,7 @@
|
||||||
|
|
||||||
namespace ams::kern {
|
namespace ams::kern {
|
||||||
|
|
||||||
void Kernel::Initialize(s32 core_id) {
|
void Kernel::InitializeCoreLocalRegion(s32 core_id) {
|
||||||
/* Construct the core local region object in place. */
|
/* Construct the core local region object in place. */
|
||||||
KCoreLocalContext *clc = GetPointer<KCoreLocalContext>(KMemoryLayout::GetCoreLocalRegionAddress());
|
KCoreLocalContext *clc = GetPointer<KCoreLocalContext>(KMemoryLayout::GetCoreLocalRegionAddress());
|
||||||
new (clc) KCoreLocalContext;
|
new (clc) KCoreLocalContext;
|
||||||
|
@ -46,9 +46,22 @@ namespace ams::kern {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Kernel::InitializeCoreThreads(s32 core_id) {
|
void Kernel::InitializeMainAndIdleThreads(s32 core_id) {
|
||||||
/* TODO: This function wants to setup the main thread and the idle thread. */
|
/* This function wants to setup the main thread and the idle thread. */
|
||||||
/* It also wants to initialize the scheduler/interrupt manager/hardware timer. */
|
KThread *main_thread = std::addressof(Kernel::GetMainThread(core_id));
|
||||||
|
void *main_thread_stack = GetVoidPointer(KMemoryLayout::GetMainStackTopAddress(core_id));
|
||||||
|
KThread *idle_thread = std::addressof(Kernel::GetIdleThread(core_id));
|
||||||
|
void *idle_thread_stack = GetVoidPointer(KMemoryLayout::GetIdleStackTopAddress(core_id));
|
||||||
|
KAutoObject::Create(main_thread);
|
||||||
|
KAutoObject::Create(idle_thread);
|
||||||
|
main_thread->Initialize(nullptr, 0, main_thread_stack, 0, KThread::MainThreadPriority, core_id, nullptr, KThread::ThreadType_Main);
|
||||||
|
idle_thread->Initialize(nullptr, 0, idle_thread_stack, 0, KThread::IdleThreadPriority, core_id, nullptr, KThread::ThreadType_Main);
|
||||||
|
|
||||||
|
/* Set the current thread to be the main thread, and we have no processes running yet. */
|
||||||
|
SetCurrentThread(main_thread);
|
||||||
|
SetCurrentProcess(nullptr);
|
||||||
|
|
||||||
|
/* TODO: We also want to initialize the scheduler/interrupt manager/hardware timer. */
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@ namespace ams::kern {
|
||||||
|
|
||||||
NORETURN void HorizonKernelMain(s32 core_id) {
|
NORETURN void HorizonKernelMain(s32 core_id) {
|
||||||
/* Setup the Core Local Region, and note that we're initializing. */
|
/* Setup the Core Local Region, and note that we're initializing. */
|
||||||
Kernel::Initialize(core_id);
|
Kernel::InitializeCoreLocalRegion(core_id);
|
||||||
Kernel::SetState(Kernel::State::Initializing);
|
Kernel::SetState(Kernel::State::Initializing);
|
||||||
|
|
||||||
/* Ensure that all cores get to this point before proceeding. */
|
/* Ensure that all cores get to this point before proceeding. */
|
||||||
|
@ -29,7 +29,7 @@ namespace ams::kern {
|
||||||
/* Synchronize after each init to ensure the cores go in order. */
|
/* Synchronize after each init to ensure the cores go in order. */
|
||||||
for (size_t i = 0; i < cpu::NumCores; i++) {
|
for (size_t i = 0; i < cpu::NumCores; i++) {
|
||||||
if (static_cast<s32>(i) == core_id) {
|
if (static_cast<s32>(i) == core_id) {
|
||||||
Kernel::InitializeCoreThreads(core_id);
|
Kernel::InitializeMainAndIdleThreads(core_id);
|
||||||
}
|
}
|
||||||
cpu::SynchronizeAllCores();
|
cpu::SynchronizeAllCores();
|
||||||
}
|
}
|
||||||
|
|
27
libraries/libmesosphere/source/libc/kern_cxx.c
Normal file
27
libraries/libmesosphere/source/libc/kern_cxx.c
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void __dso_handle() { /* ... */ }
|
||||||
|
void __cxa_atexit() { /* ... */ }
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
} /* extern "C" */
|
||||||
|
#endif
|
24
libraries/libmesosphere/source/libc/kern_new.cpp
Normal file
24
libraries/libmesosphere/source/libc/kern_new.cpp
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope 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 for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
#include <mesosphere.hpp>
|
||||||
|
|
||||||
|
void operator delete (void *deleted) throw() {
|
||||||
|
MESOSPHERE_PANIC("operator delete(void *) was called: %p", deleted);
|
||||||
|
}
|
||||||
|
|
||||||
|
void operator delete (void *deleted, size_t size) throw() {
|
||||||
|
MESOSPHERE_PANIC("operator delete(void *, size_t) was called: %p %zu", deleted, size);
|
||||||
|
}
|
|
@ -27,6 +27,8 @@ namespace ams::svc {
|
||||||
|
|
||||||
R_DEFINE_ERROR_RESULT(NotImplemented, 33);
|
R_DEFINE_ERROR_RESULT(NotImplemented, 33);
|
||||||
|
|
||||||
|
R_DEFINE_ERROR_RESULT(NoSynchronizationObject, 57);
|
||||||
|
|
||||||
R_DEFINE_ERROR_RESULT(ThreadTerminating, 59);
|
R_DEFINE_ERROR_RESULT(ThreadTerminating, 59);
|
||||||
|
|
||||||
R_DEFINE_ERROR_RESULT(NoEvent, 70);
|
R_DEFINE_ERROR_RESULT(NoEvent, 70);
|
||||||
|
|
Loading…
Reference in a new issue