ServiceManager.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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. return result == KernelResult.SessionCountExceeded ? SmResult.OutOfSessions : result;
  31. }
  32. private Result GetServiceImpl(out int handle, ref ServiceInfo serviceInfo)
  33. {
  34. return HorizonStatic.Syscall.ConnectToPort(out handle, serviceInfo.PortHandle);
  35. }
  36. public Result RegisterService(out int handle, ulong processId, ServiceName name, int maxSessions, bool isLight)
  37. {
  38. handle = 0;
  39. Result result = ValidateServiceName(name);
  40. if (result.IsFailure)
  41. {
  42. return result;
  43. }
  44. // TODO: Validation with GetProcessInfo etc.
  45. return HasServiceInfo(name) ? SmResult.AlreadyRegistered : RegisterServiceImpl(out handle, processId, name, maxSessions, isLight);
  46. }
  47. public Result RegisterServiceForSelf(out int handle, ServiceName name, int maxSessions)
  48. {
  49. return RegisterServiceImpl(out handle, Os.GetCurrentProcessId(), name, maxSessions, false);
  50. }
  51. private Result RegisterServiceImpl(out int handle, ulong processId, ServiceName name, int maxSessions, bool isLight)
  52. {
  53. handle = 0;
  54. Result result = ValidateServiceName(name);
  55. if (!result.IsSuccess)
  56. {
  57. return result;
  58. }
  59. if (HasServiceInfo(name))
  60. {
  61. return SmResult.AlreadyRegistered;
  62. }
  63. int freeServiceIndex = GetFreeService();
  64. if (freeServiceIndex < 0)
  65. {
  66. return SmResult.OutOfServices;
  67. }
  68. ref ServiceInfo freeService = ref _services[freeServiceIndex];
  69. result = HorizonStatic.Syscall.CreatePort(out handle, out int clientPort, maxSessions, isLight, null);
  70. if (!result.IsSuccess)
  71. {
  72. return result;
  73. }
  74. freeService.PortHandle = clientPort;
  75. freeService.Name = name;
  76. freeService.OwnerProcessId = processId;
  77. return Result.Success;
  78. }
  79. public Result UnregisterService(ulong processId, ServiceName name)
  80. {
  81. Result result = ValidateServiceName(name);
  82. if (result.IsFailure)
  83. {
  84. return result;
  85. }
  86. // TODO: Validation with GetProcessInfo etc.
  87. int serviceIndex = GetServiceInfo(name);
  88. if (serviceIndex < 0)
  89. {
  90. return SmResult.NotRegistered;
  91. }
  92. ref var serviceInfo = ref _services[serviceIndex];
  93. if (serviceInfo.OwnerProcessId != processId)
  94. {
  95. return SmResult.NotAllowed;
  96. }
  97. serviceInfo.Free();
  98. return Result.Success;
  99. }
  100. private static Result ValidateServiceName(ServiceName name)
  101. {
  102. if (name[0] == 0)
  103. {
  104. return SmResult.InvalidServiceName;
  105. }
  106. int nameLength = 1;
  107. for (; nameLength < name.Length; nameLength++)
  108. {
  109. if (name[nameLength] == 0)
  110. {
  111. break;
  112. }
  113. }
  114. while (nameLength < name.Length)
  115. {
  116. if (name[nameLength++] != 0)
  117. {
  118. return SmResult.InvalidServiceName;
  119. }
  120. }
  121. return Result.Success;
  122. }
  123. private bool HasServiceInfo(ServiceName name)
  124. {
  125. return GetServiceInfo(name) != -1;
  126. }
  127. private int GetFreeService()
  128. {
  129. return GetServiceInfo(ServiceName.Invalid);
  130. }
  131. private int GetServiceInfo(ServiceName name)
  132. {
  133. for (int index = 0; index < MaxServicesCount; index++)
  134. {
  135. if (_services[index].Name == name)
  136. {
  137. return index;
  138. }
  139. }
  140. return -1;
  141. }
  142. }
  143. }