Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

heap.c 4.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. /*-
  2. * Copyright 2016 Vsevolod Stakhov
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #include "config.h"
  17. #include "libutil/heap.h"
  18. struct rspamd_min_heap {
  19. GPtrArray *ar;
  20. };
  21. #define __SWAP(a, b) do { \
  22. __typeof__(a) _a = (a); \
  23. __typeof__(b) _b = (b); \
  24. a = _b; \
  25. b = _a; \
  26. } while (0)
  27. #define heap_swap(h,e1,e2) do { \
  28. __SWAP((h)->ar->pdata[(e1)->idx - 1], (h)->ar->pdata[(e2)->idx - 1]); \
  29. __SWAP((e1)->idx, (e2)->idx); \
  30. } while (0)
  31. #define min_elt(e1, e2) ((e1)->pri <= (e2)->pri ? (e1) : (e2))
  32. /*
  33. * Swims element added (or changed) to preserve heap's invariant
  34. */
  35. static void
  36. rspamd_min_heap_swim (struct rspamd_min_heap *heap,
  37. struct rspamd_min_heap_elt *elt)
  38. {
  39. struct rspamd_min_heap_elt *parent;
  40. while (elt->idx > 1) {
  41. parent = g_ptr_array_index (heap->ar, elt->idx / 2 - 1);
  42. if (parent->pri > elt->pri) {
  43. heap_swap (heap, elt, parent);
  44. }
  45. else {
  46. break;
  47. }
  48. }
  49. }
  50. /*
  51. * Sinks the element popped (or changed) to preserve heap's invariant
  52. */
  53. static void
  54. rspamd_min_heap_sink (struct rspamd_min_heap *heap,
  55. struct rspamd_min_heap_elt *elt)
  56. {
  57. struct rspamd_min_heap_elt *c1, *c2, *m;
  58. while (elt->idx * 2 < heap->ar->len) {
  59. c1 = g_ptr_array_index (heap->ar, elt->idx * 2 - 1);
  60. c2 = g_ptr_array_index (heap->ar, elt->idx * 2);
  61. m = min_elt (c1, c2);
  62. if (elt->pri > m->pri) {
  63. heap_swap (heap, elt, m);
  64. }
  65. else {
  66. break;
  67. }
  68. }
  69. if (elt->idx * 2 - 1 < heap->ar->len) {
  70. m = g_ptr_array_index (heap->ar, elt->idx * 2 - 1);
  71. if (elt->pri > m->pri) {
  72. heap_swap (heap, elt, m);
  73. }
  74. }
  75. }
  76. struct rspamd_min_heap *
  77. rspamd_min_heap_create (gsize reserved_size)
  78. {
  79. struct rspamd_min_heap *heap;
  80. heap = g_malloc (sizeof (*heap));
  81. heap->ar = g_ptr_array_sized_new (reserved_size);
  82. return heap;
  83. }
  84. void
  85. rspamd_min_heap_push (struct rspamd_min_heap *heap,
  86. struct rspamd_min_heap_elt *elt)
  87. {
  88. g_assert (heap != NULL);
  89. g_assert (elt != NULL);
  90. /* Add to the end */
  91. elt->idx = heap->ar->len + 1;
  92. g_ptr_array_add (heap->ar, elt);
  93. /* Now swim it up */
  94. rspamd_min_heap_swim (heap, elt);
  95. }
  96. struct rspamd_min_heap_elt*
  97. rspamd_min_heap_pop (struct rspamd_min_heap *heap)
  98. {
  99. struct rspamd_min_heap_elt *elt, *last;
  100. g_assert (heap != NULL);
  101. if (heap->ar->len == 0) {
  102. return NULL;
  103. }
  104. elt = g_ptr_array_index (heap->ar, 0);
  105. last = g_ptr_array_index (heap->ar, heap->ar->len - 1);
  106. if (elt != last) {
  107. /* Now replace elt with the last element and sink it if needed */
  108. heap_swap (heap, elt, last);
  109. g_ptr_array_remove_index_fast (heap->ar, heap->ar->len - 1);
  110. rspamd_min_heap_sink (heap, last);
  111. }
  112. else {
  113. g_ptr_array_remove_index_fast (heap->ar, heap->ar->len - 1);
  114. }
  115. return elt;
  116. }
  117. void
  118. rspamd_min_heap_update_elt (struct rspamd_min_heap *heap,
  119. struct rspamd_min_heap_elt *elt, guint npri)
  120. {
  121. guint oldpri;
  122. g_assert (heap != NULL);
  123. g_assert (elt->idx > 0 && elt->idx <= heap->ar->len);
  124. oldpri = elt->pri;
  125. elt->pri = npri;
  126. if (npri > oldpri) {
  127. /* We might need to sink */
  128. rspamd_min_heap_sink (heap, elt);
  129. }
  130. else if (npri < oldpri) {
  131. /* We might need to swim */
  132. rspamd_min_heap_swim (heap, elt);
  133. }
  134. }
  135. void
  136. rspamd_min_heap_remove_elt (struct rspamd_min_heap *heap,
  137. struct rspamd_min_heap_elt *elt)
  138. {
  139. struct rspamd_min_heap_elt *first;
  140. g_assert (heap != NULL);
  141. g_assert (elt->idx > 0 && elt->idx <= heap->ar->len);
  142. first = g_ptr_array_index (heap->ar, 0);
  143. if (elt != first) {
  144. elt->pri = first->pri - 1;
  145. rspamd_min_heap_swim (heap, elt);
  146. }
  147. /* Now the desired element is on the top of queue */
  148. (void)rspamd_min_heap_pop (heap);
  149. }
  150. void
  151. rspamd_min_heap_destroy (struct rspamd_min_heap *heap)
  152. {
  153. if (heap) {
  154. g_ptr_array_free (heap->ar, TRUE);
  155. g_free (heap);
  156. }
  157. }
  158. struct rspamd_min_heap_elt*
  159. rspamd_min_heap_index (struct rspamd_min_heap *heap, guint idx)
  160. {
  161. g_assert (heap != NULL);
  162. g_assert (idx < heap->ar->len);
  163. return g_ptr_array_index (heap->ar, idx);
  164. }