PriorityQueue.cs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  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.Collections.Generic;
  35. using System.Diagnostics;
  36. namespace UnityEngine.Experimental.Rendering.Universal
  37. {
  38. namespace LibTessDotNet
  39. {
  40. internal class PriorityQueue<TValue> where TValue : class
  41. {
  42. private PriorityHeap<TValue>.LessOrEqual _leq;
  43. private PriorityHeap<TValue> _heap;
  44. private TValue[] _keys;
  45. private int[] _order;
  46. private int _size, _max;
  47. private bool _initialized;
  48. public bool Empty { get { return _size == 0 && _heap.Empty; } }
  49. public PriorityQueue(int initialSize, PriorityHeap<TValue>.LessOrEqual leq)
  50. {
  51. _leq = leq;
  52. _heap = new PriorityHeap<TValue>(initialSize, leq);
  53. _keys = new TValue[initialSize];
  54. _size = 0;
  55. _max = initialSize;
  56. _initialized = false;
  57. }
  58. class StackItem
  59. {
  60. internal int p, r;
  61. };
  62. static void Swap(ref int a, ref int b)
  63. {
  64. int tmp = a;
  65. a = b;
  66. b = tmp;
  67. }
  68. public void Init()
  69. {
  70. var stack = new Stack<StackItem>();
  71. int p, r, i, j, piv;
  72. uint seed = 2016473283;
  73. p = 0;
  74. r = _size - 1;
  75. _order = new int[_size + 1];
  76. for (piv = 0, i = p; i <= r; ++piv, ++i)
  77. {
  78. _order[i] = piv;
  79. }
  80. stack.Push(new StackItem { p = p, r = r });
  81. while (stack.Count > 0)
  82. {
  83. var top = stack.Pop();
  84. p = top.p;
  85. r = top.r;
  86. while (r > p + 10)
  87. {
  88. seed = seed * 1539415821 + 1;
  89. i = p + (int)(seed % (r - p + 1));
  90. piv = _order[i];
  91. _order[i] = _order[p];
  92. _order[p] = piv;
  93. i = p - 1;
  94. j = r + 1;
  95. do {
  96. do { ++i; } while (!_leq(_keys[_order[i]], _keys[piv]));
  97. do { --j; } while (!_leq(_keys[piv], _keys[_order[j]]));
  98. Swap(ref _order[i], ref _order[j]);
  99. } while (i < j);
  100. Swap(ref _order[i], ref _order[j]);
  101. if (i - p < r - j)
  102. {
  103. stack.Push(new StackItem { p = j + 1, r = r });
  104. r = i - 1;
  105. }
  106. else
  107. {
  108. stack.Push(new StackItem { p = p, r = i - 1 });
  109. p = j + 1;
  110. }
  111. }
  112. for (i = p + 1; i <= r; ++i)
  113. {
  114. piv = _order[i];
  115. for (j = i; j > p && !_leq(_keys[piv], _keys[_order[j - 1]]); --j)
  116. {
  117. _order[j] = _order[j - 1];
  118. }
  119. _order[j] = piv;
  120. }
  121. }
  122. #if DEBUG
  123. p = 0;
  124. r = _size - 1;
  125. for (i = p; i < r; ++i)
  126. {
  127. Debug.Assert(_leq(_keys[_order[i + 1]], _keys[_order[i]]), "Wrong sort");
  128. }
  129. #endif
  130. _max = _size;
  131. _initialized = true;
  132. _heap.Init();
  133. }
  134. public PQHandle Insert(TValue value)
  135. {
  136. if (_initialized)
  137. {
  138. return _heap.Insert(value);
  139. }
  140. int curr = _size;
  141. if (++_size >= _max)
  142. {
  143. _max <<= 1;
  144. Array.Resize(ref _keys, _max);
  145. }
  146. _keys[curr] = value;
  147. return new PQHandle { _handle = -(curr + 1) };
  148. }
  149. public TValue ExtractMin()
  150. {
  151. Debug.Assert(_initialized);
  152. if (_size == 0)
  153. {
  154. return _heap.ExtractMin();
  155. }
  156. TValue sortMin = _keys[_order[_size - 1]];
  157. if (!_heap.Empty)
  158. {
  159. TValue heapMin = _heap.Minimum();
  160. if (_leq(heapMin, sortMin))
  161. return _heap.ExtractMin();
  162. }
  163. do {
  164. --_size;
  165. } while (_size > 0 && _keys[_order[_size - 1]] == null);
  166. return sortMin;
  167. }
  168. public TValue Minimum()
  169. {
  170. Debug.Assert(_initialized);
  171. if (_size == 0)
  172. {
  173. return _heap.Minimum();
  174. }
  175. TValue sortMin = _keys[_order[_size - 1]];
  176. if (!_heap.Empty)
  177. {
  178. TValue heapMin = _heap.Minimum();
  179. if (_leq(heapMin, sortMin))
  180. return heapMin;
  181. }
  182. return sortMin;
  183. }
  184. public void Remove(PQHandle handle)
  185. {
  186. Debug.Assert(_initialized);
  187. int curr = handle._handle;
  188. if (curr >= 0)
  189. {
  190. _heap.Remove(handle);
  191. return;
  192. }
  193. curr = -(curr + 1);
  194. Debug.Assert(curr < _max && _keys[curr] != null);
  195. _keys[curr] = null;
  196. while (_size > 0 && _keys[_order[_size - 1]] == null)
  197. {
  198. --_size;
  199. }
  200. }
  201. }
  202. }
  203. }