forked from eden-emu/eden
		
	Kernel/Arbiters: HLE is atomic, adjust code to reflect that.
This commit is contained in:
		
							parent
							
								
									8f8fe62a19
								
							
						
					
					
						commit
						dc70a87af1
					
				
					 2 changed files with 13 additions and 37 deletions
				
			
		|  | @ -20,14 +20,14 @@ namespace Kernel { | ||||||
|         ResultCode WaitForAddress(VAddr address, s64 timeout) { |         ResultCode WaitForAddress(VAddr address, s64 timeout) { | ||||||
|             SharedPtr<Thread> current_thread = GetCurrentThread(); |             SharedPtr<Thread> current_thread = GetCurrentThread(); | ||||||
|             current_thread->arb_wait_address = address; |             current_thread->arb_wait_address = address; | ||||||
|             current_thread->arb_wait_result = RESULT_TIMEOUT; |  | ||||||
|             current_thread->status = THREADSTATUS_WAIT_ARB; |             current_thread->status = THREADSTATUS_WAIT_ARB; | ||||||
|             current_thread->wakeup_callback = nullptr; |             current_thread->wakeup_callback = nullptr; | ||||||
| 
 | 
 | ||||||
|             current_thread->WakeAfterDelay(timeout); |             current_thread->WakeAfterDelay(timeout); | ||||||
| 
 | 
 | ||||||
|             Core::System::GetInstance().CpuCore(current_thread->processor_id).PrepareReschedule(); |             Core::System::GetInstance().CpuCore(current_thread->processor_id).PrepareReschedule(); | ||||||
|             return current_thread->arb_wait_result; |             // This should never actually execute.
 | ||||||
|  |             return RESULT_SUCCESS; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // Gets the threads waiting on an address.
 |         // Gets the threads waiting on an address.
 | ||||||
|  | @ -67,7 +67,7 @@ namespace Kernel { | ||||||
|             // TODO: Rescheduling should not occur while waking threads. How can it be prevented?
 |             // TODO: Rescheduling should not occur while waking threads. How can it be prevented?
 | ||||||
|             for (size_t i = 0; i < last; i++) { |             for (size_t i = 0; i < last; i++) { | ||||||
|                 ASSERT(waiting_threads[i]->status = THREADSTATUS_WAIT_ARB); |                 ASSERT(waiting_threads[i]->status = THREADSTATUS_WAIT_ARB); | ||||||
|                 waiting_threads[i]->arb_wait_result = RESULT_SUCCESS; |                 waiting_threads[i]->SetWaitSynchronizationResult(RESULT_SUCCESS); | ||||||
|                 waiting_threads[i]->arb_wait_address = 0; |                 waiting_threads[i]->arb_wait_address = 0; | ||||||
|                 waiting_threads[i]->ResumeFromWait(); |                 waiting_threads[i]->ResumeFromWait(); | ||||||
|             } |             } | ||||||
|  | @ -91,17 +91,9 @@ namespace Kernel { | ||||||
|                 return ERR_INVALID_ADDRESS_STATE; |                 return ERR_INVALID_ADDRESS_STATE; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             s32 cur_value; |             if ((s32)Memory::Read32(address) == value) { | ||||||
|             // Get value, incrementing if equal.
 |                 Memory::Write32(address, (u32)(value + 1)); | ||||||
|             { |             } else { | ||||||
|                 // Increment if Equal must be an atomic operation.
 |  | ||||||
|                 std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); |  | ||||||
|                 cur_value = (s32)Memory::Read32(address); |  | ||||||
|                 if (cur_value == value) { |  | ||||||
|                     Memory::Write32(address, (u32)(cur_value + 1)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (cur_value != value) { |  | ||||||
|                 return ERR_INVALID_STATE; |                 return ERR_INVALID_STATE; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -128,18 +120,10 @@ namespace Kernel { | ||||||
|             } else { |             } else { | ||||||
|                 updated_value = value; |                 updated_value = value; | ||||||
|             } |             } | ||||||
|             s32 cur_value; |  | ||||||
|             // Perform an atomic update if equal.
 |  | ||||||
|             { |  | ||||||
|                 std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); |  | ||||||
|                 cur_value = (s32)Memory::Read32(address); |  | ||||||
|                 if (cur_value == value) { |  | ||||||
|                     Memory::Write32(address, (u32)(updated_value)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             // Only continue if equal.
 |             if ((s32)Memory::Read32(address) == value) { | ||||||
|             if (cur_value != value) { |                 Memory::Write32(address, (u32)(updated_value)); | ||||||
|  |             } else { | ||||||
|                 return ERR_INVALID_STATE; |                 return ERR_INVALID_STATE; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -154,17 +138,10 @@ namespace Kernel { | ||||||
|                 return ERR_INVALID_ADDRESS_STATE; |                 return ERR_INVALID_ADDRESS_STATE; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             s32 cur_value; |             s32 cur_value = (s32)Memory::Read32(address); | ||||||
|             // Get value, decrementing if less than
 |             if (cur_value < value) { | ||||||
|             { |                 Memory::Write32(address, (u32)(cur_value - 1)); | ||||||
|                 // Decrement if less than must be an atomic operation.
 |             } else { | ||||||
|                 std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); |  | ||||||
|                 cur_value = (s32)Memory::Read32(address); |  | ||||||
|                 if (cur_value < value) { |  | ||||||
|                     Memory::Write32(address, (u32)(cur_value - 1)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (cur_value >= value) { |  | ||||||
|                 return ERR_INVALID_STATE; |                 return ERR_INVALID_STATE; | ||||||
|             } |             } | ||||||
|             // Short-circuit without rescheduling, if timeout is zero.
 |             // Short-circuit without rescheduling, if timeout is zero.
 | ||||||
|  |  | ||||||
|  | @ -233,7 +233,6 @@ public: | ||||||
| 
 | 
 | ||||||
|     // If waiting for an AddressArbiter, this is the address being waited on.
 |     // If waiting for an AddressArbiter, this is the address being waited on.
 | ||||||
|     VAddr arb_wait_address{0}; |     VAddr arb_wait_address{0}; | ||||||
|     ResultCode arb_wait_result{RESULT_SUCCESS}; ///< Result returned when done waiting on AddressArbiter.
 |  | ||||||
| 
 | 
 | ||||||
|     std::string name; |     std::string name; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Michael Scire
						Michael Scire