PriorityHeap.cs 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. /*
  2. ** SGI FREE SOFTWARE LICENSE B (Version 2.0, Sept. 18, 2008)
  3. ** Copyright (C) 2011 Silicon Graphics, Inc.
  4. ** All Rights Reserved.
  5. **
  6. ** Permission is hereby granted, free of charge, to any person obtaining a copy
  7. ** of this software and associated documentation files (the "Software"), to deal
  8. ** in the Software without restriction, including without limitation the rights
  9. ** to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
  10. ** of the Software, and to permit persons to whom the Software is furnished to do so,
  11. ** subject to the following conditions:
  12. **
  13. ** The above copyright notice including the dates of first publication and either this
  14. ** permission notice or a reference to http://oss.sgi.com/projects/FreeB/ shall be
  15. ** included in all copies or substantial portions of the Software.
  16. **
  17. ** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
  18. ** INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
  19. ** PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL SILICON GRAPHICS, INC.
  20. ** BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
  21. ** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
  22. ** OR OTHER DEALINGS IN THE SOFTWARE.
  23. **
  24. ** Except as contained in this notice, the name of Silicon Graphics, Inc. shall not
  25. ** be used in advertising or otherwise to promote the sale, use or other dealings in
  26. ** this Software without prior written authorization from Silicon Graphics, Inc.
  27. */
  28. /*
  29. ** Original Author: Eric Veach, July 1994.
  30. ** libtess2: Mikko Mononen, http://code.google.com/p/libtess2/.
  31. ** LibTessDotNet: Remi Gillig, https://github.com/speps/LibTessDotNet
  32. */
  33. using System;
  34. using System.Diagnostics;
  35. namespace UnityEngine.Experimental.Rendering.Universal
  36. {
  37. namespace LibTessDotNet
  38. {
  39. internal struct PQHandle
  40. {
  41. public static readonly int Invalid = 0x0fffffff;
  42. internal int _handle;
  43. }
  44. internal class PriorityHeap<TValue> where TValue : class
  45. {
  46. public delegate bool LessOrEqual(TValue lhs, TValue rhs);
  47. protected class HandleElem
  48. {
  49. internal TValue _key;
  50. internal int _node;
  51. }
  52. private LessOrEqual _leq;
  53. private int[] _nodes;
  54. private HandleElem[] _handles;
  55. private int _size, _max;
  56. private int _freeList;
  57. private bool _initialized;
  58. public bool Empty { get { return _size == 0; } }
  59. public PriorityHeap(int initialSize, LessOrEqual leq)
  60. {
  61. _leq = leq;
  62. _nodes = new int[initialSize + 1];
  63. _handles = new HandleElem[initialSize + 1];
  64. _size = 0;
  65. _max = initialSize;
  66. _freeList = 0;
  67. _initialized = false;
  68. _nodes[1] = 1;
  69. _handles[1] = new HandleElem { _key = null };
  70. }
  71. private void FloatDown(int curr)
  72. {
  73. int child;
  74. int hCurr, hChild;
  75. hCurr = _nodes[curr];
  76. while (true)
  77. {
  78. child = curr << 1;
  79. if (child < _size && _leq(_handles[_nodes[child + 1]]._key, _handles[_nodes[child]]._key))
  80. {
  81. ++child;
  82. }
  83. Debug.Assert(child <= _max);
  84. hChild = _nodes[child];
  85. if (child > _size || _leq(_handles[hCurr]._key, _handles[hChild]._key))
  86. {
  87. _nodes[curr] = hCurr;
  88. _handles[hCurr]._node = curr;
  89. break;
  90. }
  91. _nodes[curr] = hChild;
  92. _handles[hChild]._node = curr;
  93. curr = child;
  94. }
  95. }
  96. private void FloatUp(int curr)
  97. {
  98. int parent;
  99. int hCurr, hParent;
  100. hCurr = _nodes[curr];
  101. while (true)
  102. {
  103. parent = curr >> 1;
  104. hParent = _nodes[parent];
  105. if (parent == 0 || _leq(_handles[hParent]._key, _handles[hCurr]._key))
  106. {
  107. _nodes[curr] = hCurr;
  108. _handles[hCurr]._node = curr;
  109. break;
  110. }
  111. _nodes[curr] = hParent;
  112. _handles[hParent]._node = curr;
  113. curr = parent;
  114. }
  115. }
  116. public void Init()
  117. {
  118. for (int i = _size; i >= 1; --i)
  119. {
  120. FloatDown(i);
  121. }
  122. _initialized = true;
  123. }
  124. public PQHandle Insert(TValue value)
  125. {
  126. int curr = ++_size;
  127. if ((curr * 2) > _max)
  128. {
  129. _max <<= 1;
  130. Array.Resize(ref _nodes, _max + 1);
  131. Array.Resize(ref _handles, _max + 1);
  132. }
  133. int free;
  134. if (_freeList == 0)
  135. {
  136. free = curr;
  137. }
  138. else
  139. {
  140. free = _freeList;
  141. _freeList = _handles[free]._node;
  142. }
  143. _nodes[curr] = free;
  144. if (_handles[free] == null)
  145. {
  146. _handles[free] = new HandleElem { _key = value, _node = curr };
  147. }
  148. else
  149. {
  150. _handles[free]._node = curr;
  151. _handles[free]._key = value;
  152. }
  153. if (_initialized)
  154. {
  155. FloatUp(curr);
  156. }
  157. Debug.Assert(free != PQHandle.Invalid);
  158. return new PQHandle { _handle = free };
  159. }
  160. public TValue ExtractMin()
  161. {
  162. Debug.Assert(_initialized);
  163. int hMin = _nodes[1];
  164. TValue min = _handles[hMin]._key;
  165. if (_size > 0)
  166. {
  167. _nodes[1] = _nodes[_size];
  168. _handles[_nodes[1]]._node = 1;
  169. _handles[hMin]._key = null;
  170. _handles[hMin]._node = _freeList;
  171. _freeList = hMin;
  172. if (--_size > 0)
  173. {
  174. FloatDown(1);
  175. }
  176. }
  177. return min;
  178. }
  179. public TValue Minimum()
  180. {
  181. Debug.Assert(_initialized);
  182. return _handles[_nodes[1]]._key;
  183. }
  184. public void Remove(PQHandle handle)
  185. {
  186. Debug.Assert(_initialized);
  187. int hCurr = handle._handle;
  188. Debug.Assert(hCurr >= 1 && hCurr <= _max && _handles[hCurr]._key != null);
  189. int curr = _handles[hCurr]._node;
  190. _nodes[curr] = _nodes[_size];
  191. _handles[_nodes[curr]]._node = curr;
  192. if (curr <= --_size)
  193. {
  194. if (curr <= 1 || _leq(_handles[_nodes[curr >> 1]]._key, _handles[_nodes[curr]]._key))
  195. {
  196. FloatDown(curr);
  197. }
  198. else
  199. {
  200. FloatUp(curr);
  201. }
  202. }
  203. _handles[hCurr]._key = null;
  204. _handles[hCurr]._node = _freeList;
  205. _freeList = hCurr;
  206. }
  207. }
  208. }
  209. }