mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2024-12-23 04:41:12 +00:00
kern: SvcGetThreadList
This commit is contained in:
parent
51084c0837
commit
96c3dfee14
6 changed files with 100 additions and 20 deletions
|
@ -38,6 +38,8 @@ namespace ams::kern {
|
||||||
void Initialize();
|
void Initialize();
|
||||||
Result Attach(KProcess *process);
|
Result Attach(KProcess *process);
|
||||||
|
|
||||||
|
KScopedAutoObject<KProcess> GetProcess();
|
||||||
|
|
||||||
Result QueryMemoryInfo(ams::svc::MemoryInfo *out_memory_info, ams::svc::PageInfo *out_page_info, KProcessAddress address);
|
Result QueryMemoryInfo(ams::svc::MemoryInfo *out_memory_info, ams::svc::PageInfo *out_page_info, KProcessAddress address);
|
||||||
Result ReadMemory(KProcessAddress buffer, KProcessAddress address, size_t size);
|
Result ReadMemory(KProcessAddress buffer, KProcessAddress address, size_t size);
|
||||||
Result WriteMemory(KProcessAddress buffer, KProcessAddress address, size_t size);
|
Result WriteMemory(KProcessAddress buffer, KProcessAddress address, size_t size);
|
||||||
|
@ -49,8 +51,6 @@ namespace ams::kern {
|
||||||
|
|
||||||
/* TODO: This is a placeholder definition. */
|
/* TODO: This is a placeholder definition. */
|
||||||
private:
|
private:
|
||||||
KScopedAutoObject<KProcess> GetProcess();
|
|
||||||
|
|
||||||
void PushDebugEvent(ams::svc::DebugEvent event, uintptr_t param0 = 0, uintptr_t param1 = 0, uintptr_t param2 = 0, uintptr_t param3 = 0, uintptr_t param4 = 0);
|
void PushDebugEvent(ams::svc::DebugEvent event, uintptr_t param0 = 0, uintptr_t param1 = 0, uintptr_t param2 = 0, uintptr_t param3 = 0, uintptr_t param4 = 0);
|
||||||
void EnqueueDebugEventInfo(KEventInfo *info);
|
void EnqueueDebugEventInfo(KEventInfo *info);
|
||||||
public:
|
public:
|
||||||
|
|
|
@ -324,6 +324,8 @@ namespace ams::kern {
|
||||||
return this->address_arbiter.WaitForAddress(address, arb_type, value, timeout);
|
return this->address_arbiter.WaitForAddress(address, arb_type, value, timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result GetThreadList(s32 *out_num_threads, ams::kern::svc::KUserPointer<u64 *> out_thread_ids, s32 max_out_count);
|
||||||
|
|
||||||
static KProcess *GetProcessFromId(u64 process_id);
|
static KProcess *GetProcessFromId(u64 process_id);
|
||||||
static Result GetProcessList(s32 *out_num_processes, ams::kern::svc::KUserPointer<u64 *> out_process_ids, s32 max_out_count);
|
static Result GetProcessList(s32 *out_num_processes, ams::kern::svc::KUserPointer<u64 *> out_process_ids, s32 max_out_count);
|
||||||
|
|
||||||
|
|
|
@ -515,6 +515,8 @@ namespace ams::kern {
|
||||||
return ConditionVariableThreadTreeTraits::IsValid();
|
return ConditionVariableThreadTreeTraits::IsValid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Result GetThreadList(s32 *out_num_threads, ams::kern::svc::KUserPointer<u64 *> out_thread_ids, s32 max_out_count);
|
||||||
|
|
||||||
using ConditionVariableThreadTreeType = ConditionVariableThreadTree;
|
using ConditionVariableThreadTreeType = ConditionVariableThreadTree;
|
||||||
};
|
};
|
||||||
static_assert(alignof(KThread) == 0x10);
|
static_assert(alignof(KThread) == 0x10);
|
||||||
|
|
|
@ -924,6 +924,33 @@ namespace ams::kern {
|
||||||
return ResultSuccess();
|
return ResultSuccess();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result KProcess::GetThreadList(s32 *out_num_threads, ams::kern::svc::KUserPointer<u64 *> out_thread_ids, s32 max_out_count) {
|
||||||
|
/* Lock the list. */
|
||||||
|
KScopedLightLock lk(this->list_lock);
|
||||||
|
|
||||||
|
/* Iterate over the list. */
|
||||||
|
s32 count = 0;
|
||||||
|
auto end = this->GetThreadList().end();
|
||||||
|
for (auto it = this->GetThreadList().begin(); it != end; ++it) {
|
||||||
|
/* If we're within array bounds, write the id. */
|
||||||
|
if (count < max_out_count) {
|
||||||
|
/* Get the thread id. */
|
||||||
|
KThread *thread = std::addressof(*it);
|
||||||
|
const u64 id = thread->GetId();
|
||||||
|
|
||||||
|
/* Copy the id to userland. */
|
||||||
|
R_TRY(out_thread_ids.CopyArrayElementFrom(std::addressof(id), count));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Increment the count. */
|
||||||
|
++count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We successfully iterated the list. */
|
||||||
|
*out_num_threads = count;
|
||||||
|
return ResultSuccess();
|
||||||
|
}
|
||||||
|
|
||||||
KProcess::State KProcess::SetDebugObject(void *debug_object) {
|
KProcess::State KProcess::SetDebugObject(void *debug_object) {
|
||||||
/* Attaching should only happen to non-null objects while the scheduler is locked. */
|
/* Attaching should only happen to non-null objects while the scheduler is locked. */
|
||||||
MESOSPHERE_ASSERT(KScheduler::IsSchedulerLockedByCurrentThread());
|
MESOSPHERE_ASSERT(KScheduler::IsSchedulerLockedByCurrentThread());
|
||||||
|
|
|
@ -1189,4 +1189,31 @@ namespace ams::kern {
|
||||||
return std::addressof(this->GetContext());
|
return std::addressof(this->GetContext());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result KThread::GetThreadList(s32 *out_num_threads, ams::kern::svc::KUserPointer<u64 *> out_thread_ids, s32 max_out_count) {
|
||||||
|
/* Lock the list. */
|
||||||
|
KThread::ListAccessor accessor;
|
||||||
|
const auto end = accessor.end();
|
||||||
|
|
||||||
|
/* Iterate over the list. */
|
||||||
|
s32 count = 0;
|
||||||
|
for (auto it = accessor.begin(); it != end; ++it) {
|
||||||
|
/* If we're within array bounds, write the id. */
|
||||||
|
if (count < max_out_count) {
|
||||||
|
/* Get the thread id. */
|
||||||
|
KThread *thread = static_cast<KThread *>(std::addressof(*it));
|
||||||
|
const u64 id = thread->GetId();
|
||||||
|
|
||||||
|
/* Copy the id to userland. */
|
||||||
|
R_TRY(out_thread_ids.CopyArrayElementFrom(std::addressof(id), count));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Increment the count. */
|
||||||
|
++count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We successfully iterated the list. */
|
||||||
|
*out_num_threads = count;
|
||||||
|
return ResultSuccess();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -214,6 +214,44 @@ namespace ams::kern::svc {
|
||||||
return ResultSuccess();
|
return ResultSuccess();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result GetThreadList(int32_t *out_num_threads, KUserPointer<uint64_t *> out_thread_ids, int32_t max_out_count, ams::svc::Handle debug_handle) {
|
||||||
|
/* Validate that the out count is valid. */
|
||||||
|
R_UNLESS((0 <= max_out_count && max_out_count <= static_cast<int32_t>(std::numeric_limits<int32_t>::max() / sizeof(u64))), svc::ResultOutOfRange());
|
||||||
|
|
||||||
|
/* Validate that the pointer is in range. */
|
||||||
|
if (max_out_count > 0) {
|
||||||
|
R_UNLESS(GetCurrentProcess().GetPageTable().Contains(KProcessAddress(out_thread_ids.GetUnsafePointer()), max_out_count * sizeof(u64)), svc::ResultInvalidCurrentMemory());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (debug_handle == ams::svc::InvalidHandle) {
|
||||||
|
/* If passed invalid handle, we should return the global thread list. */
|
||||||
|
return KThread::GetThreadList(out_num_threads, out_thread_ids, max_out_count);
|
||||||
|
} else {
|
||||||
|
/* Get the handle table. */
|
||||||
|
auto &handle_table = GetCurrentProcess().GetHandleTable();
|
||||||
|
|
||||||
|
/* Try to get as a debug object. */
|
||||||
|
KScopedAutoObject debug = handle_table.GetObject<KDebug>(debug_handle);
|
||||||
|
if (debug.IsNotNull()) {
|
||||||
|
/* Get the debug object's process. */
|
||||||
|
KScopedAutoObject process = debug->GetProcess();
|
||||||
|
R_UNLESS(process.IsNotNull(), svc::ResultProcessTerminated());
|
||||||
|
|
||||||
|
/* Get the thread list. */
|
||||||
|
R_TRY(process->GetThreadList(out_num_threads, out_thread_ids, max_out_count));
|
||||||
|
} else {
|
||||||
|
/* Try to get as a process. */
|
||||||
|
KScopedAutoObject process = handle_table.GetObjectWithoutPseudoHandle<KProcess>(debug_handle);
|
||||||
|
R_UNLESS(process.IsNotNull(), svc::ResultInvalidHandle());
|
||||||
|
|
||||||
|
/* Get the thread list. */
|
||||||
|
R_TRY(process->GetThreadList(out_num_threads, out_thread_ids, max_out_count));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ResultSuccess();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ============================= 64 ABI ============================= */
|
/* ============================= 64 ABI ============================= */
|
||||||
|
@ -254,20 +292,12 @@ namespace ams::kern::svc {
|
||||||
return GetThreadId(out_thread_id, thread_handle);
|
return GetThreadId(out_thread_id, thread_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
Result GetDebugFutureThreadInfo64(ams::svc::lp64::LastThreadContext *out_context, uint64_t *thread_id, ams::svc::Handle debug_handle, int64_t ns) {
|
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetDebugFutureThreadInfo64 was called.");
|
|
||||||
}
|
|
||||||
|
|
||||||
Result GetLastThreadInfo64(ams::svc::lp64::LastThreadContext *out_context, ams::svc::Address *out_tls_address, uint32_t *out_flags) {
|
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetLastThreadInfo64 was called.");
|
|
||||||
}
|
|
||||||
|
|
||||||
Result GetThreadContext364(KUserPointer<ams::svc::ThreadContext *> out_context, ams::svc::Handle thread_handle) {
|
Result GetThreadContext364(KUserPointer<ams::svc::ThreadContext *> out_context, ams::svc::Handle thread_handle) {
|
||||||
return GetThreadContext3(out_context, thread_handle);
|
return GetThreadContext3(out_context, thread_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
Result GetThreadList64(int32_t *out_num_threads, KUserPointer<uint64_t *> out_thread_ids, int32_t max_out_count, ams::svc::Handle debug_handle) {
|
Result GetThreadList64(int32_t *out_num_threads, KUserPointer<uint64_t *> out_thread_ids, int32_t max_out_count, ams::svc::Handle debug_handle) {
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetThreadList64 was called.");
|
return GetThreadList(out_num_threads, out_thread_ids, max_out_count, debug_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ============================= 64From32 ABI ============================= */
|
/* ============================= 64From32 ABI ============================= */
|
||||||
|
@ -308,20 +338,12 @@ namespace ams::kern::svc {
|
||||||
return GetThreadId(out_thread_id, thread_handle);
|
return GetThreadId(out_thread_id, thread_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
Result GetDebugFutureThreadInfo64From32(ams::svc::ilp32::LastThreadContext *out_context, uint64_t *thread_id, ams::svc::Handle debug_handle, int64_t ns) {
|
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetDebugFutureThreadInfo64From32 was called.");
|
|
||||||
}
|
|
||||||
|
|
||||||
Result GetLastThreadInfo64From32(ams::svc::ilp32::LastThreadContext *out_context, ams::svc::Address *out_tls_address, uint32_t *out_flags) {
|
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetLastThreadInfo64From32 was called.");
|
|
||||||
}
|
|
||||||
|
|
||||||
Result GetThreadContext364From32(KUserPointer<ams::svc::ThreadContext *> out_context, ams::svc::Handle thread_handle) {
|
Result GetThreadContext364From32(KUserPointer<ams::svc::ThreadContext *> out_context, ams::svc::Handle thread_handle) {
|
||||||
return GetThreadContext3(out_context, thread_handle);
|
return GetThreadContext3(out_context, thread_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
Result GetThreadList64From32(int32_t *out_num_threads, KUserPointer<uint64_t *> out_thread_ids, int32_t max_out_count, ams::svc::Handle debug_handle) {
|
Result GetThreadList64From32(int32_t *out_num_threads, KUserPointer<uint64_t *> out_thread_ids, int32_t max_out_count, ams::svc::Handle debug_handle) {
|
||||||
MESOSPHERE_PANIC("Stubbed SvcGetThreadList64From32 was called.");
|
return GetThreadList(out_num_threads, out_thread_ids, max_out_count, debug_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue