mirror of
				https://git.zaroz.cloud/nintendo-back-up/yuzu/yuzu-mainline.git
				synced 2025-03-21 01:53:15 +00:00 
			
		
		
		
	Merge pull request #2270 from lioncash/plist
kernel/svc: Implement svcGetProcessList and svcGetThreadList
This commit is contained in:
		
						commit
						e796351a0d
					
				| @ -191,6 +191,10 @@ const Process* KernelCore::CurrentProcess() const { | |||||||
|     return impl->current_process; |     return impl->current_process; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const std::vector<SharedPtr<Process>>& KernelCore::GetProcessList() const { | ||||||
|  |     return impl->process_list; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void KernelCore::AddNamedPort(std::string name, SharedPtr<ClientPort> port) { | void KernelCore::AddNamedPort(std::string name, SharedPtr<ClientPort> port) { | ||||||
|     impl->named_ports.emplace(std::move(name), std::move(port)); |     impl->named_ports.emplace(std::move(name), std::move(port)); | ||||||
| } | } | ||||||
|  | |||||||
| @ -72,6 +72,9 @@ public: | |||||||
|     /// Retrieves a const pointer to the current process.
 |     /// Retrieves a const pointer to the current process.
 | ||||||
|     const Process* CurrentProcess() const; |     const Process* CurrentProcess() const; | ||||||
| 
 | 
 | ||||||
|  |     /// Retrieves the list of processes.
 | ||||||
|  |     const std::vector<SharedPtr<Process>>& GetProcessList() const; | ||||||
|  | 
 | ||||||
|     /// Adds a port to the named port table
 |     /// Adds a port to the named port table
 | ||||||
|     void AddNamedPort(std::string name, SharedPtr<ClientPort> port); |     void AddNamedPort(std::string name, SharedPtr<ClientPort> port); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -80,6 +80,14 @@ u64 Process::GetTotalPhysicalMemoryUsed() const { | |||||||
|     return vm_manager.GetCurrentHeapSize() + main_thread_stack_size + code_memory_size; |     return vm_manager.GetCurrentHeapSize() + main_thread_stack_size + code_memory_size; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void Process::RegisterThread(const Thread* thread) { | ||||||
|  |     thread_list.push_back(thread); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void Process::UnregisterThread(const Thread* thread) { | ||||||
|  |     thread_list.remove(thread); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| ResultCode Process::ClearSignalState() { | ResultCode Process::ClearSignalState() { | ||||||
|     if (status == ProcessStatus::Exited) { |     if (status == ProcessStatus::Exited) { | ||||||
|         LOG_ERROR(Kernel, "called on a terminated process instance."); |         LOG_ERROR(Kernel, "called on a terminated process instance."); | ||||||
|  | |||||||
| @ -7,6 +7,7 @@ | |||||||
| #include <array> | #include <array> | ||||||
| #include <bitset> | #include <bitset> | ||||||
| #include <cstddef> | #include <cstddef> | ||||||
|  | #include <list> | ||||||
| #include <string> | #include <string> | ||||||
| #include <vector> | #include <vector> | ||||||
| #include <boost/container/static_vector.hpp> | #include <boost/container/static_vector.hpp> | ||||||
| @ -189,6 +190,19 @@ public: | |||||||
|     /// Retrieves the total physical memory used by this process in bytes.
 |     /// Retrieves the total physical memory used by this process in bytes.
 | ||||||
|     u64 GetTotalPhysicalMemoryUsed() const; |     u64 GetTotalPhysicalMemoryUsed() const; | ||||||
| 
 | 
 | ||||||
|  |     /// Gets the list of all threads created with this process as their owner.
 | ||||||
|  |     const std::list<const Thread*>& GetThreadList() const { | ||||||
|  |         return thread_list; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Registers a thread as being created under this process,
 | ||||||
|  |     /// adding it to this process' thread list.
 | ||||||
|  |     void RegisterThread(const Thread* thread); | ||||||
|  | 
 | ||||||
|  |     /// Unregisters a thread from this process, removing it
 | ||||||
|  |     /// from this process' thread list.
 | ||||||
|  |     void UnregisterThread(const Thread* thread); | ||||||
|  | 
 | ||||||
|     /// Clears the signaled state of the process if and only if it's signaled.
 |     /// Clears the signaled state of the process if and only if it's signaled.
 | ||||||
|     ///
 |     ///
 | ||||||
|     /// @pre The process must not be already terminated. If this is called on a
 |     /// @pre The process must not be already terminated. If this is called on a
 | ||||||
| @ -308,6 +322,9 @@ private: | |||||||
|     /// Random values for svcGetInfo RandomEntropy
 |     /// Random values for svcGetInfo RandomEntropy
 | ||||||
|     std::array<u64, RANDOM_ENTROPY_SIZE> random_entropy; |     std::array<u64, RANDOM_ENTROPY_SIZE> random_entropy; | ||||||
| 
 | 
 | ||||||
|  |     /// List of threads that are running with this process as their owner.
 | ||||||
|  |     std::list<const Thread*> thread_list; | ||||||
|  | 
 | ||||||
|     /// System context
 |     /// System context
 | ||||||
|     Core::System& system; |     Core::System& system; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1983,6 +1983,83 @@ static ResultCode SetResourceLimitLimitValue(Handle resource_limit, u32 resource | |||||||
|     return RESULT_SUCCESS; |     return RESULT_SUCCESS; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static ResultCode GetProcessList(u32* out_num_processes, VAddr out_process_ids, | ||||||
|  |                                  u32 out_process_ids_size) { | ||||||
|  |     LOG_DEBUG(Kernel_SVC, "called. out_process_ids=0x{:016X}, out_process_ids_size={}", | ||||||
|  |               out_process_ids, out_process_ids_size); | ||||||
|  | 
 | ||||||
|  |     // If the supplied size is negative or greater than INT32_MAX / sizeof(u64), bail.
 | ||||||
|  |     if ((out_process_ids_size & 0xF0000000) != 0) { | ||||||
|  |         LOG_ERROR(Kernel_SVC, | ||||||
|  |                   "Supplied size outside [0, 0x0FFFFFFF] range. out_process_ids_size={}", | ||||||
|  |                   out_process_ids_size); | ||||||
|  |         return ERR_OUT_OF_RANGE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const auto& kernel = Core::System::GetInstance().Kernel(); | ||||||
|  |     const auto& vm_manager = kernel.CurrentProcess()->VMManager(); | ||||||
|  |     const auto total_copy_size = out_process_ids_size * sizeof(u64); | ||||||
|  | 
 | ||||||
|  |     if (out_process_ids_size > 0 && | ||||||
|  |         !vm_manager.IsWithinAddressSpace(out_process_ids, total_copy_size)) { | ||||||
|  |         LOG_ERROR(Kernel_SVC, "Address range outside address space. begin=0x{:016X}, end=0x{:016X}", | ||||||
|  |                   out_process_ids, out_process_ids + total_copy_size); | ||||||
|  |         return ERR_INVALID_ADDRESS_STATE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const auto& process_list = kernel.GetProcessList(); | ||||||
|  |     const auto num_processes = process_list.size(); | ||||||
|  |     const auto copy_amount = std::min(std::size_t{out_process_ids_size}, num_processes); | ||||||
|  | 
 | ||||||
|  |     for (std::size_t i = 0; i < copy_amount; ++i) { | ||||||
|  |         Memory::Write64(out_process_ids, process_list[i]->GetProcessID()); | ||||||
|  |         out_process_ids += sizeof(u64); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     *out_num_processes = static_cast<u32>(num_processes); | ||||||
|  |     return RESULT_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ResultCode GetThreadList(u32* out_num_threads, VAddr out_thread_ids, u32 out_thread_ids_size, | ||||||
|  |                          Handle debug_handle) { | ||||||
|  |     // TODO: Handle this case when debug events are supported.
 | ||||||
|  |     UNIMPLEMENTED_IF(debug_handle != InvalidHandle); | ||||||
|  | 
 | ||||||
|  |     LOG_DEBUG(Kernel_SVC, "called. out_thread_ids=0x{:016X}, out_thread_ids_size={}", | ||||||
|  |               out_thread_ids, out_thread_ids_size); | ||||||
|  | 
 | ||||||
|  |     // If the size is negative or larger than INT32_MAX / sizeof(u64)
 | ||||||
|  |     if ((out_thread_ids_size & 0xF0000000) != 0) { | ||||||
|  |         LOG_ERROR(Kernel_SVC, "Supplied size outside [0, 0x0FFFFFFF] range. size={}", | ||||||
|  |                   out_thread_ids_size); | ||||||
|  |         return ERR_OUT_OF_RANGE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const auto* const current_process = Core::System::GetInstance().Kernel().CurrentProcess(); | ||||||
|  |     const auto& vm_manager = current_process->VMManager(); | ||||||
|  |     const auto total_copy_size = out_thread_ids_size * sizeof(u64); | ||||||
|  | 
 | ||||||
|  |     if (out_thread_ids_size > 0 && | ||||||
|  |         !vm_manager.IsWithinAddressSpace(out_thread_ids, total_copy_size)) { | ||||||
|  |         LOG_ERROR(Kernel_SVC, "Address range outside address space. begin=0x{:016X}, end=0x{:016X}", | ||||||
|  |                   out_thread_ids, out_thread_ids + total_copy_size); | ||||||
|  |         return ERR_INVALID_ADDRESS_STATE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const auto& thread_list = current_process->GetThreadList(); | ||||||
|  |     const auto num_threads = thread_list.size(); | ||||||
|  |     const auto copy_amount = std::min(std::size_t{out_thread_ids_size}, num_threads); | ||||||
|  | 
 | ||||||
|  |     auto list_iter = thread_list.cbegin(); | ||||||
|  |     for (std::size_t i = 0; i < copy_amount; ++i, ++list_iter) { | ||||||
|  |         Memory::Write64(out_thread_ids, (*list_iter)->GetThreadID()); | ||||||
|  |         out_thread_ids += sizeof(u64); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     *out_num_threads = static_cast<u32>(num_threads); | ||||||
|  |     return RESULT_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| namespace { | namespace { | ||||||
| struct FunctionDef { | struct FunctionDef { | ||||||
|     using Func = void(); |     using Func = void(); | ||||||
| @ -2095,8 +2172,8 @@ static const FunctionDef SVC_Table[] = { | |||||||
|     {0x62, nullptr, "TerminateDebugProcess"}, |     {0x62, nullptr, "TerminateDebugProcess"}, | ||||||
|     {0x63, nullptr, "GetDebugEvent"}, |     {0x63, nullptr, "GetDebugEvent"}, | ||||||
|     {0x64, nullptr, "ContinueDebugEvent"}, |     {0x64, nullptr, "ContinueDebugEvent"}, | ||||||
|     {0x65, nullptr, "GetProcessList"}, |     {0x65, SvcWrap<GetProcessList>, "GetProcessList"}, | ||||||
|     {0x66, nullptr, "GetThreadList"}, |     {0x66, SvcWrap<GetThreadList>, "GetThreadList"}, | ||||||
|     {0x67, nullptr, "GetDebugThreadContext"}, |     {0x67, nullptr, "GetDebugThreadContext"}, | ||||||
|     {0x68, nullptr, "SetDebugThreadContext"}, |     {0x68, nullptr, "SetDebugThreadContext"}, | ||||||
|     {0x69, nullptr, "QueryDebugProcessMemory"}, |     {0x69, nullptr, "QueryDebugProcessMemory"}, | ||||||
|  | |||||||
| @ -78,6 +78,14 @@ void SvcWrap() { | |||||||
|     FuncReturn(retval); |     FuncReturn(retval); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | template <ResultCode func(u32*, u64, u32)> | ||||||
|  | void SvcWrap() { | ||||||
|  |     u32 param_1 = 0; | ||||||
|  |     const u32 retval = func(¶m_1, Param(1), static_cast<u32>(Param(2))).raw; | ||||||
|  |     Core::CurrentArmInterface().SetReg(1, param_1); | ||||||
|  |     FuncReturn(retval); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| template <ResultCode func(u64*, u32)> | template <ResultCode func(u64*, u32)> | ||||||
| void SvcWrap() { | void SvcWrap() { | ||||||
|     u64 param_1 = 0; |     u64 param_1 = 0; | ||||||
|  | |||||||
| @ -62,6 +62,8 @@ void Thread::Stop() { | |||||||
|     } |     } | ||||||
|     wait_objects.clear(); |     wait_objects.clear(); | ||||||
| 
 | 
 | ||||||
|  |     owner_process->UnregisterThread(this); | ||||||
|  | 
 | ||||||
|     // Mark the TLS slot in the thread's page as free.
 |     // Mark the TLS slot in the thread's page as free.
 | ||||||
|     owner_process->FreeTLSSlot(tls_address); |     owner_process->FreeTLSSlot(tls_address); | ||||||
| } | } | ||||||
| @ -202,6 +204,8 @@ ResultVal<SharedPtr<Thread>> Thread::Create(KernelCore& kernel, std::string name | |||||||
|     thread->scheduler->AddThread(thread); |     thread->scheduler->AddThread(thread); | ||||||
|     thread->tls_address = thread->owner_process->MarkNextAvailableTLSSlotAsUsed(*thread); |     thread->tls_address = thread->owner_process->MarkNextAvailableTLSSlotAsUsed(*thread); | ||||||
| 
 | 
 | ||||||
|  |     thread->owner_process->RegisterThread(thread.get()); | ||||||
|  | 
 | ||||||
|     // TODO(peachum): move to ScheduleThread() when scheduler is added so selected core is used
 |     // TODO(peachum): move to ScheduleThread() when scheduler is added so selected core is used
 | ||||||
|     // to initialize the context
 |     // to initialize the context
 | ||||||
|     ResetThreadContext(thread->context, stack_top, entry_point, arg); |     ResetThreadContext(thread->context, stack_top, entry_point, arg); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 bunnei
						bunnei