 9f6b24edfd
			
		
	
	
		9f6b24edfd
		
			
		
	
	
	
	
		
			
			* Implement session count decrement when the handle is closed * Remove unused field * Implement SendSyncRequestWithUserBuffer, SendAsyncRequestWithUserBuffer and ReplyAndReceiveWithUserBuffer syscalls * Nits * Fix swapped copy dst/src * Add missing pointer buffer descriptor write on reply * Fix IPC unaligned buffer copy and restoring client attributes on reply * Oops * Fix SetIpcMappingPermission * Fix unaligned copy bugs * Free memory used for temporary IPC buffers
		
			
				
	
	
		
			153 lines
		
	
	
		
			No EOL
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			153 lines
		
	
	
		
			No EOL
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.HLE.HOS.Kernel.Common;
 | |
| using Ryujinx.HLE.HOS.Kernel.Process;
 | |
| using Ryujinx.HLE.HOS.Services;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Kernel.Ipc
 | |
| {
 | |
|     class KClientPort : KSynchronizationObject
 | |
|     {
 | |
|         private int _sessionsCount;
 | |
|         private readonly int _maxSessions;
 | |
| 
 | |
|         private readonly KPort _parent;
 | |
| 
 | |
|         public bool IsLight => _parent.IsLight;
 | |
| 
 | |
|         // TODO: Remove that, we need it for now to allow HLE
 | |
|         // SM implementation to work with the new IPC system.
 | |
|         public IpcService Service { get; set; }
 | |
| 
 | |
|         public KClientPort(KernelContext context, KPort parent, int maxSessions) : base(context)
 | |
|         {
 | |
|             _maxSessions = maxSessions;
 | |
|             _parent      = parent;
 | |
|         }
 | |
| 
 | |
|         public KernelResult Connect(out KClientSession clientSession)
 | |
|         {
 | |
|             clientSession = null;
 | |
| 
 | |
|             KProcess currentProcess = KernelContext.Scheduler.GetCurrentProcess();
 | |
| 
 | |
|             if (currentProcess.ResourceLimit != null &&
 | |
|                !currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
 | |
|             {
 | |
|                 return KernelResult.ResLimitExceeded;
 | |
|             }
 | |
| 
 | |
|             if (!IncrementSessionsCount())
 | |
|             {
 | |
|                 currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
 | |
| 
 | |
|                 return KernelResult.SessionCountExceeded;
 | |
|             }
 | |
| 
 | |
|             KSession session = new KSession(KernelContext, this);
 | |
| 
 | |
|             if (Service != null)
 | |
|             {
 | |
|                 session.ClientSession.Service = Service;
 | |
|             }
 | |
| 
 | |
|             KernelResult result = _parent.EnqueueIncomingSession(session.ServerSession);
 | |
| 
 | |
|             if (result != KernelResult.Success)
 | |
|             {
 | |
|                 session.ClientSession.DecrementReferenceCount();
 | |
|                 session.ServerSession.DecrementReferenceCount();
 | |
| 
 | |
|                 return result;
 | |
|             }
 | |
| 
 | |
|             clientSession = session.ClientSession;
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public KernelResult ConnectLight(out KLightClientSession clientSession)
 | |
|         {
 | |
|             clientSession = null;
 | |
| 
 | |
|             KProcess currentProcess = KernelContext.Scheduler.GetCurrentProcess();
 | |
| 
 | |
|             if (currentProcess.ResourceLimit != null &&
 | |
|                !currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
 | |
|             {
 | |
|                 return KernelResult.ResLimitExceeded;
 | |
|             }
 | |
| 
 | |
|             if (!IncrementSessionsCount())
 | |
|             {
 | |
|                 currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
 | |
| 
 | |
|                 return KernelResult.SessionCountExceeded;
 | |
|             }
 | |
| 
 | |
|             KLightSession session = new KLightSession(KernelContext);
 | |
| 
 | |
|             KernelResult result = _parent.EnqueueIncomingLightSession(session.ServerSession);
 | |
| 
 | |
|             if (result != KernelResult.Success)
 | |
|             {
 | |
|                 session.ClientSession.DecrementReferenceCount();
 | |
|                 session.ServerSession.DecrementReferenceCount();
 | |
| 
 | |
|                 return result;
 | |
|             }
 | |
| 
 | |
|             clientSession = session.ClientSession;
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         private bool IncrementSessionsCount()
 | |
|         {
 | |
|             while (true)
 | |
|             {
 | |
|                 int currentCount = _sessionsCount;
 | |
| 
 | |
|                 if (currentCount < _maxSessions)
 | |
|                 {
 | |
|                     if (Interlocked.CompareExchange(ref _sessionsCount, currentCount + 1, currentCount) == currentCount)
 | |
|                     {
 | |
|                         return true;
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Disconnect()
 | |
|         {
 | |
|             KernelContext.CriticalSection.Enter();
 | |
| 
 | |
|             SignalIfMaximumReached(Interlocked.Decrement(ref _sessionsCount));
 | |
| 
 | |
|             KernelContext.CriticalSection.Leave();
 | |
|         }
 | |
| 
 | |
|         private void SignalIfMaximumReached(int value)
 | |
|         {
 | |
|             if (value == _maxSessions)
 | |
|             {
 | |
|                 Signal();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public new static KernelResult RemoveName(KernelContext context, string name)
 | |
|         {
 | |
|             KAutoObject foundObj = FindNamedObject(context, name);
 | |
| 
 | |
|             if (!(foundObj is KClientPort))
 | |
|             {
 | |
|                 return KernelResult.NotFound;
 | |
|             }
 | |
| 
 | |
|             return KAutoObject.RemoveName(context, name);
 | |
|         }
 | |
|     }
 | |
| } |