mirror of
				https://git.suyu.dev/suyu/suyu.git
				synced 2025-10-25 03:46:43 +08:00 
			
		
		
		
	kernel: refactor dummy thread wakeups
This commit is contained in:
		
							parent
							
								
									0860fffd78
								
							
						
					
					
						commit
						1a378a7769
					
				| @ -49,4 +49,26 @@ bool GlobalSchedulerContext::IsLocked() const { | |||||||
|     return scheduler_lock.IsLockedByCurrentThread(); |     return scheduler_lock.IsLockedByCurrentThread(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void GlobalSchedulerContext::RegisterDummyThreadForWakeup(KThread* thread) { | ||||||
|  |     ASSERT(IsLocked()); | ||||||
|  | 
 | ||||||
|  |     woken_dummy_threads.insert(thread); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void GlobalSchedulerContext::UnregisterDummyThreadForWakeup(KThread* thread) { | ||||||
|  |     ASSERT(IsLocked()); | ||||||
|  | 
 | ||||||
|  |     woken_dummy_threads.erase(thread); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void GlobalSchedulerContext::WakeupWaitingDummyThreads() { | ||||||
|  |     ASSERT(IsLocked()); | ||||||
|  | 
 | ||||||
|  |     for (auto* thread : woken_dummy_threads) { | ||||||
|  |         thread->DummyThreadEndWait(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     woken_dummy_threads.clear(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| } // namespace Kernel
 | } // namespace Kernel
 | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ | |||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <atomic> | #include <atomic> | ||||||
|  | #include <set> | ||||||
| #include <vector> | #include <vector> | ||||||
| 
 | 
 | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
| @ -58,6 +59,10 @@ public: | |||||||
|     /// Returns true if the global scheduler lock is acquired
 |     /// Returns true if the global scheduler lock is acquired
 | ||||||
|     bool IsLocked() const; |     bool IsLocked() const; | ||||||
| 
 | 
 | ||||||
|  |     void UnregisterDummyThreadForWakeup(KThread* thread); | ||||||
|  |     void RegisterDummyThreadForWakeup(KThread* thread); | ||||||
|  |     void WakeupWaitingDummyThreads(); | ||||||
|  | 
 | ||||||
|     [[nodiscard]] LockType& SchedulerLock() { |     [[nodiscard]] LockType& SchedulerLock() { | ||||||
|         return scheduler_lock; |         return scheduler_lock; | ||||||
|     } |     } | ||||||
| @ -76,6 +81,9 @@ private: | |||||||
|     KSchedulerPriorityQueue priority_queue; |     KSchedulerPriorityQueue priority_queue; | ||||||
|     LockType scheduler_lock; |     LockType scheduler_lock; | ||||||
| 
 | 
 | ||||||
|  |     /// Lists dummy threads pending wakeup on lock release
 | ||||||
|  |     std::set<KThread*> woken_dummy_threads; | ||||||
|  | 
 | ||||||
|     /// Lists all thread ids that aren't deleted/etc.
 |     /// Lists all thread ids that aren't deleted/etc.
 | ||||||
|     std::vector<KThread*> thread_list; |     std::vector<KThread*> thread_list; | ||||||
|     std::mutex global_list_guard; |     std::mutex global_list_guard; | ||||||
|  | |||||||
| @ -81,8 +81,8 @@ void KScheduler::RescheduleCurrentHLEThread(KernelCore& kernel) { | |||||||
|     // HACK: we cannot schedule from this thread, it is not a core thread
 |     // HACK: we cannot schedule from this thread, it is not a core thread
 | ||||||
|     ASSERT(GetCurrentThread(kernel).GetDisableDispatchCount() == 1); |     ASSERT(GetCurrentThread(kernel).GetDisableDispatchCount() == 1); | ||||||
| 
 | 
 | ||||||
|     // Special case to ensure dummy threads that are waiting block
 |     // Ensure dummy threads that are waiting block.
 | ||||||
|     GetCurrentThread(kernel).IfDummyThreadTryWait(); |     GetCurrentThread(kernel).DummyThreadBeginWait(); | ||||||
| 
 | 
 | ||||||
|     ASSERT(GetCurrentThread(kernel).GetState() != ThreadState::Waiting); |     ASSERT(GetCurrentThread(kernel).GetState() != ThreadState::Waiting); | ||||||
|     GetCurrentThread(kernel).EnableDispatch(); |     GetCurrentThread(kernel).EnableDispatch(); | ||||||
| @ -314,6 +314,16 @@ u64 KScheduler::UpdateHighestPriorityThreadsImpl(KernelCore& kernel) { | |||||||
|         idle_cores &= ~(1ULL << core_id); |         idle_cores &= ~(1ULL << core_id); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     // HACK: any waiting dummy threads can wake up now.
 | ||||||
|  |     kernel.GlobalSchedulerContext().WakeupWaitingDummyThreads(); | ||||||
|  | 
 | ||||||
|  |     // HACK: if we are a dummy thread, and we need to go sleep, indicate
 | ||||||
|  |     // that for when the lock is released.
 | ||||||
|  |     KThread* const cur_thread = GetCurrentThreadPointer(kernel); | ||||||
|  |     if (cur_thread->IsDummyThread() && cur_thread->GetState() != ThreadState::Runnable) { | ||||||
|  |         cur_thread->RequestDummyThreadWait(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return cores_needing_scheduling; |     return cores_needing_scheduling; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -531,11 +541,23 @@ void KScheduler::OnThreadStateChanged(KernelCore& kernel, KThread* thread, Threa | |||||||
|         GetPriorityQueue(kernel).Remove(thread); |         GetPriorityQueue(kernel).Remove(thread); | ||||||
|         IncrementScheduledCount(thread); |         IncrementScheduledCount(thread); | ||||||
|         SetSchedulerUpdateNeeded(kernel); |         SetSchedulerUpdateNeeded(kernel); | ||||||
|  | 
 | ||||||
|  |         if (thread->IsDummyThread()) { | ||||||
|  |             // HACK: if this is a dummy thread, it should no longer wake up when the
 | ||||||
|  |             // scheduler lock is released.
 | ||||||
|  |             kernel.GlobalSchedulerContext().UnregisterDummyThreadForWakeup(thread); | ||||||
|  |         } | ||||||
|     } else if (cur_state == ThreadState::Runnable) { |     } else if (cur_state == ThreadState::Runnable) { | ||||||
|         // If we're now runnable, then we weren't previously, and we should add.
 |         // If we're now runnable, then we weren't previously, and we should add.
 | ||||||
|         GetPriorityQueue(kernel).PushBack(thread); |         GetPriorityQueue(kernel).PushBack(thread); | ||||||
|         IncrementScheduledCount(thread); |         IncrementScheduledCount(thread); | ||||||
|         SetSchedulerUpdateNeeded(kernel); |         SetSchedulerUpdateNeeded(kernel); | ||||||
|  | 
 | ||||||
|  |         if (thread->IsDummyThread()) { | ||||||
|  |             // HACK: if this is a dummy thread, it should wake up when the scheduler
 | ||||||
|  |             // lock is released.
 | ||||||
|  |             kernel.GlobalSchedulerContext().RegisterDummyThreadForWakeup(thread); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -148,7 +148,9 @@ Result KThread::Initialize(KThreadFunction func, uintptr_t arg, VAddr user_stack | |||||||
|     physical_affinity_mask.SetAffinity(phys_core, true); |     physical_affinity_mask.SetAffinity(phys_core, true); | ||||||
| 
 | 
 | ||||||
|     // Set the thread state.
 |     // Set the thread state.
 | ||||||
|     thread_state = (type == ThreadType::Main) ? ThreadState::Runnable : ThreadState::Initialized; |     thread_state = (type == ThreadType::Main || type == ThreadType::Dummy) | ||||||
|  |                        ? ThreadState::Runnable | ||||||
|  |                        : ThreadState::Initialized; | ||||||
| 
 | 
 | ||||||
|     // Set TLS address.
 |     // Set TLS address.
 | ||||||
|     tls_address = 0; |     tls_address = 0; | ||||||
| @ -1174,30 +1176,29 @@ Result KThread::Sleep(s64 timeout) { | |||||||
|     R_SUCCEED(); |     R_SUCCEED(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void KThread::IfDummyThreadTryWait() { | void KThread::RequestDummyThreadWait() { | ||||||
|     if (!IsDummyThread()) { |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(kernel)); | ||||||
|         return; |     ASSERT(this->IsDummyThread()); | ||||||
|     } | 
 | ||||||
| 
 |     // We will block when the scheduler lock is released.
 | ||||||
|     if (GetState() != ThreadState::Waiting) { |     dummy_thread_runnable.store(false); | ||||||
|         return; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void KThread::DummyThreadBeginWait() { | ||||||
|  |     ASSERT(this->IsDummyThread()); | ||||||
|     ASSERT(!kernel.IsPhantomModeForSingleCore()); |     ASSERT(!kernel.IsPhantomModeForSingleCore()); | ||||||
| 
 | 
 | ||||||
|     // Block until we are no longer waiting.
 |     // Block until runnable is no longer false.
 | ||||||
|     std::unique_lock lk(dummy_wait_lock); |     dummy_thread_runnable.wait(false); | ||||||
|     dummy_wait_cv.wait( |  | ||||||
|         lk, [&] { return GetState() != ThreadState::Waiting || kernel.IsShuttingDown(); }); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void KThread::IfDummyThreadEndWait() { | void KThread::DummyThreadEndWait() { | ||||||
|     if (!IsDummyThread()) { |     ASSERT(KScheduler::IsSchedulerLockedByCurrentThread(kernel)); | ||||||
|         return; |     ASSERT(this->IsDummyThread()); | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     // Wake up the waiting thread.
 |     // Wake up the waiting thread.
 | ||||||
|     dummy_wait_cv.notify_one(); |     dummy_thread_runnable.store(true); | ||||||
|  |     dummy_thread_runnable.notify_one(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void KThread::BeginWait(KThreadQueue* queue) { | void KThread::BeginWait(KThreadQueue* queue) { | ||||||
| @ -1231,9 +1232,6 @@ void KThread::EndWait(Result wait_result_) { | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         wait_queue->EndWait(this, wait_result_); |         wait_queue->EndWait(this, wait_result_); | ||||||
| 
 |  | ||||||
|         // Special case for dummy threads to wakeup if necessary.
 |  | ||||||
|         IfDummyThreadEndWait(); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -643,8 +643,9 @@ public: | |||||||
|     // therefore will not block on guest kernel synchronization primitives. These methods handle
 |     // therefore will not block on guest kernel synchronization primitives. These methods handle
 | ||||||
|     // blocking as needed.
 |     // blocking as needed.
 | ||||||
| 
 | 
 | ||||||
|     void IfDummyThreadTryWait(); |     void RequestDummyThreadWait(); | ||||||
|     void IfDummyThreadEndWait(); |     void DummyThreadBeginWait(); | ||||||
|  |     void DummyThreadEndWait(); | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] uintptr_t GetArgument() const { |     [[nodiscard]] uintptr_t GetArgument() const { | ||||||
|         return argument; |         return argument; | ||||||
| @ -777,8 +778,7 @@ private: | |||||||
|     bool is_single_core{}; |     bool is_single_core{}; | ||||||
|     ThreadType thread_type{}; |     ThreadType thread_type{}; | ||||||
|     StepState step_state{}; |     StepState step_state{}; | ||||||
|     std::mutex dummy_wait_lock; |     std::atomic<bool> dummy_thread_runnable{true}; | ||||||
|     std::condition_variable dummy_wait_cv; |  | ||||||
| 
 | 
 | ||||||
|     // For debugging
 |     // For debugging
 | ||||||
|     std::vector<KSynchronizationObject*> wait_objects_for_debugging; |     std::vector<KSynchronizationObject*> wait_objects_for_debugging; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user