 4314a8f3e5
			
		
	
	
		4314a8f3e5
		
			
		
	
	
	
	
		
			
			* Add support for events, move concept of domains to IpcService * Support waiting for KThread, remove some test code, other tweaks * Use move handle on NIFM since I can't test that now, it's better to leave it how it was
		
			
				
	
	
		
			122 lines
		
	
	
		
			No EOL
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			No EOL
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Core.OsHle.Handles;
 | |
| using System.Collections.Concurrent;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace Ryujinx.Core.OsHle
 | |
| {
 | |
|     class Mutex
 | |
|     {
 | |
|         private const int MutexHasListenersMask = 0x40000000;
 | |
| 
 | |
|         private Process Process;
 | |
| 
 | |
|         private long MutexAddress;
 | |
| 
 | |
|         private bool OwnsMutexValue;
 | |
| 
 | |
|         private object EnterWaitLock;
 | |
| 
 | |
|         private ConcurrentQueue<KThread> WaitingThreads;
 | |
| 
 | |
|         public Mutex(Process Process, long MutexAddress, int OwnerThreadHandle)
 | |
|         {
 | |
|             this.Process      = Process;
 | |
|             this.MutexAddress = MutexAddress;
 | |
| 
 | |
|             //Process.Memory.WriteInt32(MutexAddress, OwnerThreadHandle);
 | |
| 
 | |
|             EnterWaitLock = new object();
 | |
| 
 | |
|             WaitingThreads = new ConcurrentQueue<KThread>();
 | |
|         }
 | |
| 
 | |
|         public void WaitForLock(KThread RequestingThread, int RequestingThreadHandle)
 | |
|         {
 | |
|             AcquireMutexValue();
 | |
| 
 | |
|             lock (EnterWaitLock)
 | |
|             {
 | |
|                 int CurrentThreadHandle = Process.Memory.ReadInt32(MutexAddress) & ~MutexHasListenersMask;
 | |
| 
 | |
|                 if (CurrentThreadHandle == RequestingThreadHandle ||
 | |
|                     CurrentThreadHandle == 0)
 | |
|                 {
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 Process.Memory.WriteInt32(MutexAddress, CurrentThreadHandle | MutexHasListenersMask);
 | |
| 
 | |
|                 ReleaseMutexValue();
 | |
| 
 | |
|                 WaitingThreads.Enqueue(RequestingThread);
 | |
|             }
 | |
| 
 | |
|             Process.Scheduler.WaitForSignal(RequestingThread);
 | |
|         }
 | |
| 
 | |
|         public void GiveUpLock(int ThreadHandle)
 | |
|         {
 | |
|             AcquireMutexValue();
 | |
| 
 | |
|             lock (EnterWaitLock)
 | |
|             {
 | |
|                 int CurrentThread = Process.Memory.ReadInt32(MutexAddress) & ~MutexHasListenersMask;
 | |
| 
 | |
|                 if (CurrentThread == ThreadHandle)
 | |
|                 {
 | |
|                     Unlock();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             ReleaseMutexValue();
 | |
|         }
 | |
| 
 | |
|         public void Unlock()
 | |
|         {
 | |
|             AcquireMutexValue();
 | |
| 
 | |
|             lock (EnterWaitLock)
 | |
|             {
 | |
|                 int HasListeners = WaitingThreads.Count > 1 ? MutexHasListenersMask : 0;
 | |
| 
 | |
|                 Process.Memory.WriteInt32(MutexAddress, HasListeners);
 | |
| 
 | |
|                 ReleaseMutexValue();
 | |
| 
 | |
|                 KThread[] UnlockedThreads = new KThread[WaitingThreads.Count];
 | |
| 
 | |
|                 int Index = 0;
 | |
| 
 | |
|                 while (WaitingThreads.TryDequeue(out KThread Thread))
 | |
|                 {
 | |
|                     UnlockedThreads[Index++] = Thread;
 | |
|                 }
 | |
| 
 | |
|                 Process.Scheduler.Signal(UnlockedThreads);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void AcquireMutexValue()
 | |
|         {
 | |
|             if (!OwnsMutexValue)
 | |
|             {
 | |
|                 while (!Process.Memory.AcquireAddress(MutexAddress))
 | |
|                 {
 | |
|                     Thread.Yield();
 | |
|                 }
 | |
| 
 | |
|                 OwnsMutexValue = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void ReleaseMutexValue()
 | |
|         {
 | |
|             if (OwnsMutexValue)
 | |
|             {
 | |
|                 OwnsMutexValue = false;
 | |
| 
 | |
|                 Process.Memory.ReleaseAddress(MutexAddress);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| } |