ServiceManager.cs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. using Ryujinx.Horizon.Common;
  2. using Ryujinx.Horizon.Sdk.OsTypes;
  3. using Ryujinx.Horizon.Sdk.Sf;
  4. using Ryujinx.Horizon.Sdk.Sm;
  5. namespace Ryujinx.Horizon.Sm.Impl
  6. {
  7. class ServiceManager
  8. {
  9. private const int MaxServicesCount = 256;
  10. private readonly ServiceInfo[] _services;
  11. public ServiceManager()
  12. {
  13. _services = new ServiceInfo[MaxServicesCount];
  14. }
  15. public Result GetService(out int handle, ulong processId, ServiceName name)
  16. {
  17. handle = 0;
  18. Result result = ValidateServiceName(name);
  19. if (result.IsFailure)
  20. {
  21. return result;
  22. }
  23. // TODO: Validation with GetProcessInfo etc.
  24. int serviceIndex = GetServiceInfo(name);
  25. if (serviceIndex < 0)
  26. {
  27. return SfResult.RequestDeferredByUser;
  28. }
  29. result = GetServiceImpl(out handle, ref _services[serviceIndex]);
  30. if (result == KernelResult.SessionCountExceeded)
  31. {
  32. return SmResult.OutOfSessions;
  33. }
  34. return result;
  35. }
  36. private Result GetServiceImpl(out int handle, ref ServiceInfo serviceInfo)
  37. {
  38. return HorizonStatic.Syscall.ConnectToPort(out handle, serviceInfo.PortHandle);
  39. }
  40. public Result RegisterService(out int handle, ulong processId, ServiceName name, int maxSessions, bool isLight)
  41. {
  42. handle = 0;
  43. Result result = ValidateServiceName(name);
  44. if (result.IsFailure)
  45. {
  46. return result;
  47. }
  48. // TODO: Validation with GetProcessInfo etc.
  49. if (HasServiceInfo(name))
  50. {
  51. return SmResult.AlreadyRegistered;
  52. }
  53. return RegisterServiceImpl(out handle, processId, name, maxSessions, isLight);
  54. }
  55. public Result RegisterServiceForSelf(out int handle, ServiceName name, int maxSessions)
  56. {
  57. return RegisterServiceImpl(out handle, Os.GetCurrentProcessId(), name, maxSessions, false);
  58. }
  59. private Result RegisterServiceImpl(out int handle, ulong processId, ServiceName name, int maxSessions, bool isLight)
  60. {
  61. handle = 0;
  62. Result result = ValidateServiceName(name);
  63. if (!result.IsSuccess)
  64. {
  65. return result;
  66. }
  67. if (HasServiceInfo(name))
  68. {
  69. return SmResult.AlreadyRegistered;
  70. }
  71. int freeServiceIndex = GetFreeService();
  72. if (freeServiceIndex < 0)
  73. {
  74. return SmResult.OutOfServices;
  75. }
  76. ref ServiceInfo freeService = ref _services[freeServiceIndex];
  77. result = HorizonStatic.Syscall.CreatePort(out handle, out int clientPort, maxSessions, isLight, null);
  78. if (!result.IsSuccess)
  79. {
  80. return result;
  81. }
  82. freeService.PortHandle = clientPort;
  83. freeService.Name = name;
  84. freeService.OwnerProcessId = processId;
  85. return Result.Success;
  86. }
  87. public Result UnregisterService(ulong processId, ServiceName name)
  88. {
  89. Result result = ValidateServiceName(name);
  90. if (result.IsFailure)
  91. {
  92. return result;
  93. }
  94. // TODO: Validation with GetProcessInfo etc.
  95. int serviceIndex = GetServiceInfo(name);
  96. if (serviceIndex < 0)
  97. {
  98. return SmResult.NotRegistered;
  99. }
  100. ref var serviceInfo = ref _services[serviceIndex];
  101. if (serviceInfo.OwnerProcessId != processId)
  102. {
  103. return SmResult.NotAllowed;
  104. }
  105. serviceInfo.Free();
  106. return Result.Success;
  107. }
  108. private static Result ValidateServiceName(ServiceName name)
  109. {
  110. if (name[0] == 0)
  111. {
  112. return SmResult.InvalidServiceName;
  113. }
  114. int nameLength = 1;
  115. for (; nameLength < name.Length; nameLength++)
  116. {
  117. if (name[nameLength] == 0)
  118. {
  119. break;
  120. }
  121. }
  122. while (nameLength < name.Length)
  123. {
  124. if (name[nameLength++] != 0)
  125. {
  126. return SmResult.InvalidServiceName;
  127. }
  128. }
  129. return Result.Success;
  130. }
  131. private bool HasServiceInfo(ServiceName name)
  132. {
  133. return GetServiceInfo(name) != -1;
  134. }
  135. private int GetFreeService()
  136. {
  137. return GetServiceInfo(ServiceName.Invalid);
  138. }
  139. private int GetServiceInfo(ServiceName name)
  140. {
  141. for (int index = 0; index < MaxServicesCount; index++)
  142. {
  143. if (_services[index].Name == name)
  144. {
  145. return index;
  146. }
  147. }
  148. return -1;
  149. }
  150. }
  151. }