mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-10-31 05:40:04 +00:00 
			
		
		
		
	Kernel/Thread: Allow specifying which process a thread belongs to when creating it.
Don't automatically assume that Thread::Create will only be called when the parent process is currently scheduled. This assumption will be broken when applets or system modules are loaded.
This commit is contained in:
		
							parent
							
								
									a8d2f5787f
								
							
						
					
					
						commit
						3165466b66
					
				
					 4 changed files with 22 additions and 17 deletions
				
			
		|  | @ -147,7 +147,7 @@ void Process::Run(s32 main_thread_priority, u32 stack_size) { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     vm_manager.LogLayout(Log::Level::Debug); |     vm_manager.LogLayout(Log::Level::Debug); | ||||||
|     Kernel::SetupMainThread(codeset->entrypoint, main_thread_priority); |     Kernel::SetupMainThread(codeset->entrypoint, main_thread_priority, this); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VAddr Process::GetLinearHeapAreaAddress() const { | VAddr Process::GetLinearHeapAreaAddress() const { | ||||||
|  |  | ||||||
|  | @ -361,7 +361,8 @@ static void ResetThreadContext(ARM_Interface::ThreadContext& context, u32 stack_ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ResultVal<SharedPtr<Thread>> Thread::Create(std::string name, VAddr entry_point, u32 priority, | ResultVal<SharedPtr<Thread>> Thread::Create(std::string name, VAddr entry_point, u32 priority, | ||||||
|                                             u32 arg, s32 processor_id, VAddr stack_top) { |                                             u32 arg, s32 processor_id, VAddr stack_top, | ||||||
|  |                                             SharedPtr<Process> owner_process) { | ||||||
|     // Check if priority is in ranged. Lowest priority -> highest priority id.
 |     // Check if priority is in ranged. Lowest priority -> highest priority id.
 | ||||||
|     if (priority > THREADPRIO_LOWEST) { |     if (priority > THREADPRIO_LOWEST) { | ||||||
|         LOG_ERROR(Kernel_SVC, "Invalid thread priority: %d", priority); |         LOG_ERROR(Kernel_SVC, "Invalid thread priority: %d", priority); | ||||||
|  | @ -375,7 +376,7 @@ ResultVal<SharedPtr<Thread>> Thread::Create(std::string name, VAddr entry_point, | ||||||
| 
 | 
 | ||||||
|     // TODO(yuriks): Other checks, returning 0xD9001BEA
 |     // TODO(yuriks): Other checks, returning 0xD9001BEA
 | ||||||
| 
 | 
 | ||||||
|     if (!Memory::IsValidVirtualAddress(entry_point)) { |     if (!Memory::IsValidVirtualAddress(*owner_process, entry_point)) { | ||||||
|         LOG_ERROR(Kernel_SVC, "(name=%s): invalid entry %08x", name.c_str(), entry_point); |         LOG_ERROR(Kernel_SVC, "(name=%s): invalid entry %08x", name.c_str(), entry_point); | ||||||
|         // TODO: Verify error
 |         // TODO: Verify error
 | ||||||
|         return ResultCode(ErrorDescription::InvalidAddress, ErrorModule::Kernel, |         return ResultCode(ErrorDescription::InvalidAddress, ErrorModule::Kernel, | ||||||
|  | @ -399,10 +400,10 @@ ResultVal<SharedPtr<Thread>> Thread::Create(std::string name, VAddr entry_point, | ||||||
|     thread->wait_address = 0; |     thread->wait_address = 0; | ||||||
|     thread->name = std::move(name); |     thread->name = std::move(name); | ||||||
|     thread->callback_handle = wakeup_callback_handle_table.Create(thread).Unwrap(); |     thread->callback_handle = wakeup_callback_handle_table.Create(thread).Unwrap(); | ||||||
|     thread->owner_process = g_current_process; |     thread->owner_process = owner_process; | ||||||
| 
 | 
 | ||||||
|     // Find the next available TLS index, and mark it as used
 |     // Find the next available TLS index, and mark it as used
 | ||||||
|     auto& tls_slots = Kernel::g_current_process->tls_slots; |     auto& tls_slots = owner_process->tls_slots; | ||||||
|     bool needs_allocation = true; |     bool needs_allocation = true; | ||||||
|     u32 available_page; // Which allocated page has free space
 |     u32 available_page; // Which allocated page has free space
 | ||||||
|     u32 available_slot; // Which slot within the page is free
 |     u32 available_slot; // Which slot within the page is free
 | ||||||
|  | @ -426,13 +427,13 @@ ResultVal<SharedPtr<Thread>> Thread::Create(std::string name, VAddr entry_point, | ||||||
|         // Allocate some memory from the end of the linear heap for this region.
 |         // Allocate some memory from the end of the linear heap for this region.
 | ||||||
|         linheap_memory->insert(linheap_memory->end(), Memory::PAGE_SIZE, 0); |         linheap_memory->insert(linheap_memory->end(), Memory::PAGE_SIZE, 0); | ||||||
|         memory_region->used += Memory::PAGE_SIZE; |         memory_region->used += Memory::PAGE_SIZE; | ||||||
|         Kernel::g_current_process->linear_heap_used += Memory::PAGE_SIZE; |         owner_process->linear_heap_used += Memory::PAGE_SIZE; | ||||||
| 
 | 
 | ||||||
|         tls_slots.emplace_back(0); // The page is completely available at the start
 |         tls_slots.emplace_back(0); // The page is completely available at the start
 | ||||||
|         available_page = tls_slots.size() - 1; |         available_page = tls_slots.size() - 1; | ||||||
|         available_slot = 0; // Use the first slot in the new page
 |         available_slot = 0; // Use the first slot in the new page
 | ||||||
| 
 | 
 | ||||||
|         auto& vm_manager = Kernel::g_current_process->vm_manager; |         auto& vm_manager = owner_process->vm_manager; | ||||||
|         vm_manager.RefreshMemoryBlockMappings(linheap_memory.get()); |         vm_manager.RefreshMemoryBlockMappings(linheap_memory.get()); | ||||||
| 
 | 
 | ||||||
|         // Map the page to the current process' address space.
 |         // Map the page to the current process' address space.
 | ||||||
|  | @ -486,10 +487,10 @@ void Thread::BoostPriority(s32 priority) { | ||||||
|     current_priority = priority; |     current_priority = priority; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| SharedPtr<Thread> SetupMainThread(u32 entry_point, s32 priority) { | SharedPtr<Thread> SetupMainThread(u32 entry_point, s32 priority, SharedPtr<Process> owner_process) { | ||||||
|     // Initialize new "main" thread
 |     // Initialize new "main" thread
 | ||||||
|     auto thread_res = Thread::Create("main", entry_point, priority, 0, THREADPROCESSORID_0, |     auto thread_res = Thread::Create("main", entry_point, priority, 0, THREADPROCESSORID_0, | ||||||
|                                      Memory::HEAP_VADDR_END); |                                      Memory::HEAP_VADDR_END, owner_process); | ||||||
| 
 | 
 | ||||||
|     SharedPtr<Thread> thread = std::move(thread_res).Unwrap(); |     SharedPtr<Thread> thread = std::move(thread_res).Unwrap(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -56,10 +56,12 @@ public: | ||||||
|      * @param arg User data to pass to the thread |      * @param arg User data to pass to the thread | ||||||
|      * @param processor_id The ID(s) of the processors on which the thread is desired to be run |      * @param processor_id The ID(s) of the processors on which the thread is desired to be run | ||||||
|      * @param stack_top The address of the thread's stack top |      * @param stack_top The address of the thread's stack top | ||||||
|  |      * @param owner_process The parent process for the thread | ||||||
|      * @return A shared pointer to the newly created thread |      * @return A shared pointer to the newly created thread | ||||||
|      */ |      */ | ||||||
|     static ResultVal<SharedPtr<Thread>> Create(std::string name, VAddr entry_point, u32 priority, |     static ResultVal<SharedPtr<Thread>> Create(std::string name, VAddr entry_point, u32 priority, | ||||||
|                                                u32 arg, s32 processor_id, VAddr stack_top); |                                                u32 arg, s32 processor_id, VAddr stack_top, | ||||||
|  |                                                SharedPtr<Process> owner_process); | ||||||
| 
 | 
 | ||||||
|     std::string GetName() const override { |     std::string GetName() const override { | ||||||
|         return name; |         return name; | ||||||
|  | @ -116,9 +118,9 @@ public: | ||||||
|     void ResumeFromWait(); |     void ResumeFromWait(); | ||||||
| 
 | 
 | ||||||
|     /**
 |     /**
 | ||||||
|     * Schedules an event to wake up the specified thread after the specified delay |      * Schedules an event to wake up the specified thread after the specified delay | ||||||
|     * @param nanoseconds The time this thread will be allowed to sleep for |      * @param nanoseconds The time this thread will be allowed to sleep for | ||||||
|     */ |      */ | ||||||
|     void WakeAfterDelay(s64 nanoseconds); |     void WakeAfterDelay(s64 nanoseconds); | ||||||
| 
 | 
 | ||||||
|     /**
 |     /**
 | ||||||
|  | @ -214,9 +216,10 @@ private: | ||||||
|  * Sets up the primary application thread |  * Sets up the primary application thread | ||||||
|  * @param entry_point The address at which the thread should start execution |  * @param entry_point The address at which the thread should start execution | ||||||
|  * @param priority The priority to give the main thread |  * @param priority The priority to give the main thread | ||||||
|  |  * @param owner_process The parent process for the main thread | ||||||
|  * @return A shared pointer to the main thread |  * @return A shared pointer to the main thread | ||||||
|  */ |  */ | ||||||
| SharedPtr<Thread> SetupMainThread(u32 entry_point, s32 priority); | SharedPtr<Thread> SetupMainThread(u32 entry_point, s32 priority, SharedPtr<Process> owner_process); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Returns whether there are any threads that are ready to run. |  * Returns whether there are any threads that are ready to run. | ||||||
|  | @ -276,4 +279,4 @@ void ThreadingShutdown(); | ||||||
|  */ |  */ | ||||||
| const std::vector<SharedPtr<Thread>>& GetThreadList(); | const std::vector<SharedPtr<Thread>>& GetThreadList(); | ||||||
| 
 | 
 | ||||||
| } // namespace
 | } // namespace Kernel
 | ||||||
|  |  | ||||||
|  | @ -656,8 +656,9 @@ static ResultCode CreateThread(Kernel::Handle* out_handle, u32 priority, u32 ent | ||||||
|                   "Newly created thread must run in the SysCore (Core1), unimplemented."); |                   "Newly created thread must run in the SysCore (Core1), unimplemented."); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     CASCADE_RESULT(SharedPtr<Thread> thread, Kernel::Thread::Create(name, entry_point, priority, |     CASCADE_RESULT(SharedPtr<Thread> thread, | ||||||
|                                                                     arg, processor_id, stack_top)); |                    Kernel::Thread::Create(name, entry_point, priority, arg, processor_id, stack_top, | ||||||
|  |                                           Kernel::g_current_process)); | ||||||
| 
 | 
 | ||||||
|     thread->context.fpscr = |     thread->context.fpscr = | ||||||
|         FPSCR_DEFAULT_NAN | FPSCR_FLUSH_TO_ZERO | FPSCR_ROUND_TOZERO; // 0x03C00000
 |         FPSCR_DEFAULT_NAN | FPSCR_FLUSH_TO_ZERO | FPSCR_ROUND_TOZERO; // 0x03C00000
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue