heap-inl.h 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. /* Copyright (c) 2013, Ben Noordhuis <info@bnoordhuis.nl>
  2. *
  3. * Permission to use, copy, modify, and/or distribute this software for any
  4. * purpose with or without fee is hereby granted, provided that the above
  5. * copyright notice and this permission notice appear in all copies.
  6. *
  7. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  8. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  9. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  10. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  11. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  12. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  13. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  14. */
  15. #ifndef UV_SRC_HEAP_H_
  16. #define UV_SRC_HEAP_H_
  17. #include <stddef.h> /* NULL */
  18. #if defined(__GNUC__)
  19. # define HEAP_EXPORT(declaration) __attribute__((unused)) static declaration
  20. #else
  21. # define HEAP_EXPORT(declaration) static declaration
  22. #endif
  23. struct heap_node {
  24. struct heap_node* left;
  25. struct heap_node* right;
  26. struct heap_node* parent;
  27. };
  28. /* A binary min heap. The usual properties hold: the root is the lowest
  29. * element in the set, the height of the tree is at most log2(nodes) and
  30. * it's always a complete binary tree.
  31. *
  32. * The heap function try hard to detect corrupted tree nodes at the cost
  33. * of a minor reduction in performance. Compile with -DNDEBUG to disable.
  34. */
  35. struct heap {
  36. struct heap_node* min;
  37. unsigned int nelts;
  38. };
  39. /* Return non-zero if a < b. */
  40. typedef int (*heap_compare_fn)(const struct heap_node* a,
  41. const struct heap_node* b);
  42. /* Public functions. */
  43. HEAP_EXPORT(void heap_init(struct heap* heap));
  44. HEAP_EXPORT(struct heap_node* heap_min(const struct heap* heap));
  45. HEAP_EXPORT(void heap_insert(struct heap* heap,
  46. struct heap_node* newnode,
  47. heap_compare_fn less_than));
  48. HEAP_EXPORT(void heap_remove(struct heap* heap,
  49. struct heap_node* node,
  50. heap_compare_fn less_than));
  51. HEAP_EXPORT(void heap_dequeue(struct heap* heap, heap_compare_fn less_than));
  52. /* Implementation follows. */
  53. HEAP_EXPORT(void heap_init(struct heap* heap)) {
  54. heap->min = NULL;
  55. heap->nelts = 0;
  56. }
  57. HEAP_EXPORT(struct heap_node* heap_min(const struct heap* heap)) {
  58. return heap->min;
  59. }
  60. /* Swap parent with child. Child moves closer to the root, parent moves away. */
  61. static void heap_node_swap(struct heap* heap,
  62. struct heap_node* parent,
  63. struct heap_node* child) {
  64. struct heap_node* sibling;
  65. struct heap_node t;
  66. t = *parent;
  67. *parent = *child;
  68. *child = t;
  69. parent->parent = child;
  70. if (child->left == child) {
  71. child->left = parent;
  72. sibling = child->right;
  73. } else {
  74. child->right = parent;
  75. sibling = child->left;
  76. }
  77. if (sibling != NULL)
  78. sibling->parent = child;
  79. if (parent->left != NULL)
  80. parent->left->parent = parent;
  81. if (parent->right != NULL)
  82. parent->right->parent = parent;
  83. if (child->parent == NULL)
  84. heap->min = child;
  85. else if (child->parent->left == parent)
  86. child->parent->left = child;
  87. else
  88. child->parent->right = child;
  89. }
  90. HEAP_EXPORT(void heap_insert(struct heap* heap,
  91. struct heap_node* newnode,
  92. heap_compare_fn less_than)) {
  93. struct heap_node** parent;
  94. struct heap_node** child;
  95. unsigned int path;
  96. unsigned int n;
  97. unsigned int k;
  98. newnode->left = NULL;
  99. newnode->right = NULL;
  100. newnode->parent = NULL;
  101. /* Calculate the path from the root to the insertion point. This is a min
  102. * heap so we always insert at the left-most free node of the bottom row.
  103. */
  104. path = 0;
  105. for (k = 0, n = 1 + heap->nelts; n >= 2; k += 1, n /= 2)
  106. path = (path << 1) | (n & 1);
  107. /* Now traverse the heap using the path we calculated in the previous step. */
  108. parent = child = &heap->min;
  109. while (k > 0) {
  110. parent = child;
  111. if (path & 1)
  112. child = &(*child)->right;
  113. else
  114. child = &(*child)->left;
  115. path >>= 1;
  116. k -= 1;
  117. }
  118. /* Insert the new node. */
  119. newnode->parent = *parent;
  120. *child = newnode;
  121. heap->nelts += 1;
  122. /* Walk up the tree and check at each node if the heap property holds.
  123. * It's a min heap so parent < child must be true.
  124. */
  125. while (newnode->parent != NULL && less_than(newnode, newnode->parent))
  126. heap_node_swap(heap, newnode->parent, newnode);
  127. }
  128. HEAP_EXPORT(void heap_remove(struct heap* heap,
  129. struct heap_node* node,
  130. heap_compare_fn less_than)) {
  131. struct heap_node* smallest;
  132. struct heap_node** max;
  133. struct heap_node* child;
  134. unsigned int path;
  135. unsigned int k;
  136. unsigned int n;
  137. if (heap->nelts == 0)
  138. return;
  139. /* Calculate the path from the min (the root) to the max, the left-most node
  140. * of the bottom row.
  141. */
  142. path = 0;
  143. for (k = 0, n = heap->nelts; n >= 2; k += 1, n /= 2)
  144. path = (path << 1) | (n & 1);
  145. /* Now traverse the heap using the path we calculated in the previous step. */
  146. max = &heap->min;
  147. while (k > 0) {
  148. if (path & 1)
  149. max = &(*max)->right;
  150. else
  151. max = &(*max)->left;
  152. path >>= 1;
  153. k -= 1;
  154. }
  155. heap->nelts -= 1;
  156. /* Unlink the max node. */
  157. child = *max;
  158. *max = NULL;
  159. if (child == node) {
  160. /* We're removing either the max or the last node in the tree. */
  161. if (child == heap->min) {
  162. heap->min = NULL;
  163. }
  164. return;
  165. }
  166. /* Replace the to be deleted node with the max node. */
  167. child->left = node->left;
  168. child->right = node->right;
  169. child->parent = node->parent;
  170. if (child->left != NULL) {
  171. child->left->parent = child;
  172. }
  173. if (child->right != NULL) {
  174. child->right->parent = child;
  175. }
  176. if (node->parent == NULL) {
  177. heap->min = child;
  178. } else if (node->parent->left == node) {
  179. node->parent->left = child;
  180. } else {
  181. node->parent->right = child;
  182. }
  183. /* Walk down the subtree and check at each node if the heap property holds.
  184. * It's a min heap so parent < child must be true. If the parent is bigger,
  185. * swap it with the smallest child.
  186. */
  187. for (;;) {
  188. smallest = child;
  189. if (child->left != NULL && less_than(child->left, smallest))
  190. smallest = child->left;
  191. if (child->right != NULL && less_than(child->right, smallest))
  192. smallest = child->right;
  193. if (smallest == child)
  194. break;
  195. heap_node_swap(heap, child, smallest);
  196. }
  197. /* Walk up the subtree and check that each parent is less than the node
  198. * this is required, because `max` node is not guaranteed to be the
  199. * actual maximum in tree
  200. */
  201. while (child->parent != NULL && less_than(child, child->parent))
  202. heap_node_swap(heap, child->parent, child);
  203. }
  204. HEAP_EXPORT(void heap_dequeue(struct heap* heap, heap_compare_fn less_than)) {
  205. heap_remove(heap, heap->min, less_than);
  206. }
  207. #undef HEAP_EXPORT
  208. #endif /* UV_SRC_HEAP_H_ */