KClientPort.cs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. using Ryujinx.HLE.HOS.Kernel.Common;
  2. using Ryujinx.HLE.HOS.Kernel.Process;
  3. using Ryujinx.HLE.HOS.Services;
  4. using System.Threading;
  5. namespace Ryujinx.HLE.HOS.Kernel.Ipc
  6. {
  7. class KClientPort : KSynchronizationObject
  8. {
  9. private int _sessionsCount;
  10. private readonly int _maxSessions;
  11. private readonly KPort _parent;
  12. public bool IsLight => _parent.IsLight;
  13. // TODO: Remove that, we need it for now to allow HLE
  14. // SM implementation to work with the new IPC system.
  15. public IpcService Service { get; set; }
  16. public KClientPort(KernelContext context, KPort parent, int maxSessions) : base(context)
  17. {
  18. _maxSessions = maxSessions;
  19. _parent = parent;
  20. }
  21. public KernelResult Connect(out KClientSession clientSession)
  22. {
  23. clientSession = null;
  24. KProcess currentProcess = KernelContext.Scheduler.GetCurrentProcess();
  25. if (currentProcess.ResourceLimit != null &&
  26. !currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
  27. {
  28. return KernelResult.ResLimitExceeded;
  29. }
  30. if (!IncrementSessionsCount())
  31. {
  32. currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
  33. return KernelResult.SessionCountExceeded;
  34. }
  35. KSession session = new KSession(KernelContext, this);
  36. if (Service != null)
  37. {
  38. session.ClientSession.Service = Service;
  39. }
  40. KernelResult result = _parent.EnqueueIncomingSession(session.ServerSession);
  41. if (result != KernelResult.Success)
  42. {
  43. session.ClientSession.DecrementReferenceCount();
  44. session.ServerSession.DecrementReferenceCount();
  45. return result;
  46. }
  47. clientSession = session.ClientSession;
  48. return result;
  49. }
  50. public KernelResult ConnectLight(out KLightClientSession clientSession)
  51. {
  52. clientSession = null;
  53. KProcess currentProcess = KernelContext.Scheduler.GetCurrentProcess();
  54. if (currentProcess.ResourceLimit != null &&
  55. !currentProcess.ResourceLimit.Reserve(LimitableResource.Session, 1))
  56. {
  57. return KernelResult.ResLimitExceeded;
  58. }
  59. if (!IncrementSessionsCount())
  60. {
  61. currentProcess.ResourceLimit?.Release(LimitableResource.Session, 1);
  62. return KernelResult.SessionCountExceeded;
  63. }
  64. KLightSession session = new KLightSession(KernelContext);
  65. KernelResult result = _parent.EnqueueIncomingLightSession(session.ServerSession);
  66. if (result != KernelResult.Success)
  67. {
  68. session.ClientSession.DecrementReferenceCount();
  69. session.ServerSession.DecrementReferenceCount();
  70. return result;
  71. }
  72. clientSession = session.ClientSession;
  73. return result;
  74. }
  75. private bool IncrementSessionsCount()
  76. {
  77. while (true)
  78. {
  79. int currentCount = _sessionsCount;
  80. if (currentCount < _maxSessions)
  81. {
  82. if (Interlocked.CompareExchange(ref _sessionsCount, currentCount + 1, currentCount) == currentCount)
  83. {
  84. return true;
  85. }
  86. }
  87. else
  88. {
  89. return false;
  90. }
  91. }
  92. }
  93. public void Disconnect()
  94. {
  95. KernelContext.CriticalSection.Enter();
  96. SignalIfMaximumReached(Interlocked.Decrement(ref _sessionsCount));
  97. KernelContext.CriticalSection.Leave();
  98. }
  99. private void SignalIfMaximumReached(int value)
  100. {
  101. if (value == _maxSessions)
  102. {
  103. Signal();
  104. }
  105. }
  106. public new static KernelResult RemoveName(KernelContext context, string name)
  107. {
  108. KAutoObject foundObj = FindNamedObject(context, name);
  109. if (!(foundObj is KClientPort))
  110. {
  111. return KernelResult.NotFound;
  112. }
  113. return KAutoObject.RemoveName(context, name);
  114. }
  115. }
  116. }