ServerDomainManager.cs 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. using Ryujinx.Horizon.Common;
  2. using System;
  3. using System.Collections.Generic;
  4. namespace Ryujinx.Horizon.Sdk.Sf.Cmif
  5. {
  6. class ServerDomainManager
  7. {
  8. private class EntryManager
  9. {
  10. public class Entry
  11. {
  12. public int Id { get; }
  13. public Domain Owner { get; set; }
  14. public ServiceObjectHolder Obj { get; set; }
  15. public LinkedListNode<Entry> Node { get; set; }
  16. public Entry(int id)
  17. {
  18. Id = id;
  19. }
  20. }
  21. private readonly LinkedList<Entry> _freeList;
  22. private readonly Entry[] _entries;
  23. public EntryManager(int count)
  24. {
  25. _freeList = new LinkedList<Entry>();
  26. _entries = new Entry[count];
  27. for (int i = 0; i < count; i++)
  28. {
  29. _freeList.AddLast(_entries[i] = new Entry(i + 1));
  30. }
  31. }
  32. public Entry AllocateEntry()
  33. {
  34. lock (_freeList)
  35. {
  36. if (_freeList.Count == 0)
  37. {
  38. return null;
  39. }
  40. var entry = _freeList.First.Value;
  41. _freeList.RemoveFirst();
  42. return entry;
  43. }
  44. }
  45. public void FreeEntry(Entry entry)
  46. {
  47. lock (_freeList)
  48. {
  49. DebugUtil.Assert(entry.Owner == null);
  50. DebugUtil.Assert(entry.Obj == null);
  51. _freeList.AddFirst(entry);
  52. }
  53. }
  54. public Entry GetEntry(int id)
  55. {
  56. if (id == 0)
  57. {
  58. return null;
  59. }
  60. int index = id - 1;
  61. if ((uint)index >= (uint)_entries.Length)
  62. {
  63. return null;
  64. }
  65. return _entries[index];
  66. }
  67. }
  68. private class Domain : DomainServiceObject, IDisposable
  69. {
  70. private readonly ServerDomainManager _manager;
  71. private readonly LinkedList<EntryManager.Entry> _entries;
  72. public Domain(ServerDomainManager manager)
  73. {
  74. _manager = manager;
  75. _entries = new LinkedList<EntryManager.Entry>();
  76. }
  77. public override ServiceObjectHolder GetObject(int id)
  78. {
  79. var entry = _manager._entryManager.GetEntry(id);
  80. if (entry == null)
  81. {
  82. return null;
  83. }
  84. lock (_manager._entryOwnerLock)
  85. {
  86. if (entry.Owner != this)
  87. {
  88. return null;
  89. }
  90. }
  91. return entry.Obj.Clone();
  92. }
  93. public override ServerDomainBase GetServerDomain()
  94. {
  95. return this;
  96. }
  97. public override void RegisterObject(int id, ServiceObjectHolder obj)
  98. {
  99. var entry = _manager._entryManager.GetEntry(id);
  100. DebugUtil.Assert(entry != null);
  101. lock (_manager._entryOwnerLock)
  102. {
  103. DebugUtil.Assert(entry.Owner == null);
  104. entry.Owner = this;
  105. entry.Node = _entries.AddLast(entry);
  106. }
  107. entry.Obj = obj;
  108. }
  109. public override Result ReserveIds(Span<int> outIds)
  110. {
  111. for (int i = 0; i < outIds.Length; i++)
  112. {
  113. var entry = _manager._entryManager.AllocateEntry();
  114. if (entry == null)
  115. {
  116. return SfResult.OutOfDomainEntries;
  117. }
  118. DebugUtil.Assert(entry.Owner == null);
  119. outIds[i] = entry.Id;
  120. }
  121. return Result.Success;
  122. }
  123. public override ServiceObjectHolder UnregisterObject(int id)
  124. {
  125. var entry = _manager._entryManager.GetEntry(id);
  126. if (entry == null)
  127. {
  128. return null;
  129. }
  130. ServiceObjectHolder obj;
  131. lock (_manager._entryOwnerLock)
  132. {
  133. if (entry.Owner != this)
  134. {
  135. return null;
  136. }
  137. entry.Owner = null;
  138. obj = entry.Obj;
  139. entry.Obj = null;
  140. _entries.Remove(entry.Node);
  141. entry.Node = null;
  142. }
  143. _manager._entryManager.FreeEntry(entry);
  144. return obj;
  145. }
  146. public override void UnreserveIds(ReadOnlySpan<int> ids)
  147. {
  148. for (int i = 0; i < ids.Length; i++)
  149. {
  150. var entry = _manager._entryManager.GetEntry(ids[i]);
  151. DebugUtil.Assert(entry != null);
  152. DebugUtil.Assert(entry.Owner == null);
  153. _manager._entryManager.FreeEntry(entry);
  154. }
  155. }
  156. public void Dispose()
  157. {
  158. foreach (var entry in _entries)
  159. {
  160. if (entry.Obj.ServiceObject is IDisposable disposableObj)
  161. {
  162. disposableObj.Dispose();
  163. }
  164. }
  165. _manager.FreeDomain(this);
  166. }
  167. }
  168. private readonly EntryManager _entryManager;
  169. private readonly object _entryOwnerLock;
  170. private readonly HashSet<Domain> _domains;
  171. private int _maxDomains;
  172. public ServerDomainManager(int entryCount, int maxDomains)
  173. {
  174. _entryManager = new EntryManager(entryCount);
  175. _entryOwnerLock = new object();
  176. _domains = new HashSet<Domain>();
  177. _maxDomains = maxDomains;
  178. }
  179. public DomainServiceObject AllocateDomainServiceObject()
  180. {
  181. lock (_domains)
  182. {
  183. if (_domains.Count == _maxDomains)
  184. {
  185. return null;
  186. }
  187. var domain = new Domain(this);
  188. _domains.Add(domain);
  189. return domain;
  190. }
  191. }
  192. public static void DestroyDomainServiceObject(DomainServiceObject obj)
  193. {
  194. ((Domain)obj).Dispose();
  195. }
  196. private void FreeDomain(Domain domain)
  197. {
  198. lock (_domains)
  199. {
  200. _domains.Remove(domain);
  201. }
  202. }
  203. }
  204. }