libcstl
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Modules
heap.h
Go to the documentation of this file.
1/*!
2 * @file
3 */
4
5#ifndef CSTL_HEAP_H
6#define CSTL_HEAP_H
7
8/*!
9 * @defgroup heap Binary heap
10 * @ingroup bintrees
11 * @brief A binary tree organized as a heap
12 *
13 * A heap is a binary tree with the highest valued object (as determined
14 * by the associated comparison function) at the root. Every node in the
15 * tree is less than or equal to its parent. The highest valued object
16 * in the tree can be found in constant time, and adding and removing
17 * objects in the tree can be done in O(log n) where n is the number of
18 * elements in the heap
19 */
20/*!
21 * @addtogroup heap
22 * @{
23 */
24
25#include "cstl/bintree.h"
26
27/*!
28 * @brief Node to anchor an element within a heap
29 *
30 * Users of the heap object declare this object within another
31 * object as follows:
32 * @code{.c}
33 * struct object {
34 * ...
35 * struct cstl_heap_node heap_node;
36 * ...
37 * };
38 * @endcode
39 *
40 * When calling cstl_heap_init(), the caller passes the offset of
41 * @p cstl_heap_node within their object as the @p off parameter of
42 * that function, e.g.
43 * @code{.c}
44 * offsetof(struct object, heap_node)
45 * @endcode
46 */
48{
49 /*! @privatesection */
50 struct cstl_bintree_node bn;
51};
52
53/*!
54 * @brief Heap object
55 *
56 * Callers declare or allocate an object of this type to instantiate
57 * a heap. Users are encouraged to declare (and initialize) this
58 * object with the DECLARE_CSTL_HEAP() macro. Any other declaration or
59 * allocation must be initialized via cstl_heap_init().
60 */
62{
63 /*!
64 * @privatesection
65 *
66 * interestingly the heap code doesn't ever have to
67 * access the heap_node. it goes straight through to
68 * the cstl_bintree_node; therefore, an off member (as found
69 * in the cstl_bintree and rbtree objects isn't necessary)
70 */
71 struct cstl_bintree bt;
72};
73
74/*!
75 * @brief Constant initialization of a heap object
76 *
77 * @param TYPE The type of object that the heap will hold
78 * @param MEMB The name of the @p heap_node member within @p TYPE.
79 * @param CMP A pointer to a function of type @p cstl_compare_func_t that
80 * will be used to compare elements in the heap
81 * @param PRIV A pointer to a private data structure that will be passed
82 * to calls to the @p CMP function
83 *
84 * @see cstl_heap_node for a description of the relationship between
85 * @p TYPE and @p MEMB
86 */
87#define CSTL_HEAP_INITIALIZER(TYPE, MEMB, CMP, PRIV) \
88 { \
89 .bt = CSTL_BINTREE_INITIALIZER(TYPE, MEMB.bn, CMP, PRIV), \
90 }
91/*!
92 * @brief (Statically) declare and initialize a heap
93 *
94 * @param NAME The name of the variable being declared
95 * @param TYPE The type of object that the heap will hold
96 * @param MEMB The name of the @p cstl_heap_node member within @p TYPE.
97 * @param CMP A pointer to a function of type @p cstl_compare_func_t that
98 * will be used to compare elements in the heap
99 * @param PRIV A pointer to a private data structure that will be passed
100 * to calls to the @p CMP function
101 *
102 * @see cstl_heap_node for a description of the relationship between
103 * @p TYPE and @p MEMB
104 */
105#define DECLARE_CSTL_HEAP(NAME, TYPE, MEMB, CMP, PRIV) \
106 struct cstl_heap NAME = \
107 CSTL_HEAP_INITIALIZER(TYPE, MEMB, CMP, PRIV)
108
109/*!
110 * @brief Initialize a heap object
111 *
112 * @param[in,out] h A pointer to the object to be initialized
113 * @param[in] cmp A function that can compare objects in the heap
114 * @param[in] priv A pointer to private data that will be
115 * passed to the @p cmp function
116 * @param[in] off The offset of the @p cstl_heap_node object within the
117 * object(s) that will be stored in the heap
118 */
119static inline void cstl_heap_init(struct cstl_heap * const h,
120 cstl_compare_func_t * const cmp,
121 void * const priv,
122 const size_t off)
123{
125 &h->bt, cmp, priv, off + offsetof(struct cstl_heap_node, bn));
126}
127
128/*!
129 * @brief Get the number of objects in the heap
130 *
131 * @param[in] h A pointer to the heap
132 *
133 * @return The number of objects in the heap
134 */
135static inline size_t cstl_heap_size(const struct cstl_heap * const h)
136{
137 return cstl_bintree_size(&h->bt);
138}
139
140/*!
141 * @brief Insert a new object into the heap
142 *
143 * @param[in] h A pointer to the heap
144 * @param[in] e A pointer to the object to be inserted
145 *
146 * After insertion, the inserted object must not be modified (in such a
147 * way as to affect its comparison with other objects in the heap) as
148 * that modification can cause the assumptions about the ordering of
149 * elements within the heap to become invalid and lead to undefined behavior.
150 */
151void cstl_heap_push(struct cstl_heap * h, void * e);
152
153/*!
154 * @brief Get a pointer to the object at the top of the heap
155 *
156 * @param[in] h A pointer to the heap
157 *
158 * @return A pointer to the object at the top of the heap
159 * @retval NULL The heap is empty
160 */
161const void * cstl_heap_get(const struct cstl_heap * h);
162
163/*!
164 * @brief Remove the highest valued element from the heap
165 *
166 * @param[in] h A pointer to the heap
167 *
168 * @return The highest valued element in the heap
169 * @retval NULL The heap is empty
170 */
171void * cstl_heap_pop(struct cstl_heap * h);
172
173/*!
174 * @brief Remove all elements from the heap
175 *
176 * @param[in] h A pointer to the heap
177 * @param[in] clr A pointer to a function to be called for each
178 * element in the tree
179 *
180 * All elements are removed from the heap and the @p clr function is
181 * called for each element that was in the heap. The order in which
182 * the elements are removed and @p clr is called is not specified, but
183 * the callee may take ownership of an element at the time that @p clr
184 * is called for that element and not before.
185 *
186 * Upon return from this function, the heap contains no elements, and
187 * is as it was immediately after being initialized. No further operations
188 * on the tree are necessary to make it ready to go out of scope or be
189 * destroyed.
190 */
191static inline void cstl_heap_clear(struct cstl_heap * const h,
192 cstl_xtor_func_t * const clr)
193{
194 cstl_bintree_clear(&h->bt, clr, NULL);
195}
196
197/*!
198 * @brief Swap the heap objects at the two given locations
199 *
200 * @param[in,out] a A pointer to a heap
201 * @param[in,out] b A pointer to a(nother) heap
202 *
203 * The heaps at the given locations will be swapped such that upon return,
204 * @p a will contain the heap previously pointed to by @p b and vice versa.
205 */
206static inline void cstl_heap_swap(
207 struct cstl_heap * const a, struct cstl_heap * const b)
208{
209 cstl_bintree_swap(&a->bt, &b->bt);
210}
211
212/*!
213 * @}
214 */
215
216#endif
void cstl_xtor_func_t(void *obj, void *priv)
Type for functions called to construct, clear, or destroy an object.
Definition common.h:97
int cstl_compare_func_t(const void *obj1, const void *obj2, void *priv)
Function type for comparing (in)equality of two objects.
Definition common.h:49
static size_t cstl_bintree_size(const struct cstl_bintree *const bt)
Get the number of objects in the tree.
Definition bintree.h:149
static void cstl_bintree_init(struct cstl_bintree *const bt, cstl_compare_func_t *const cmp, void *const priv, const size_t off)
Initialize a binary tree object.
Definition bintree.h:128
void cstl_bintree_swap(struct cstl_bintree *a, struct cstl_bintree *b)
Swap the tree objects at the two given locations.
Definition bintree.c:506
void cstl_bintree_clear(struct cstl_bintree *bt, cstl_xtor_func_t *clr, void *priv)
Remove all elements from the tree.
Definition bintree.c:552
void * cstl_heap_pop(struct cstl_heap *h)
Remove the highest valued element from the heap.
Definition heap.c:174
static void cstl_heap_init(struct cstl_heap *const h, cstl_compare_func_t *const cmp, void *const priv, const size_t off)
Initialize a heap object.
Definition heap.h:119
const void * cstl_heap_get(const struct cstl_heap *h)
Get a pointer to the object at the top of the heap.
Definition heap.c:165
void cstl_heap_push(struct cstl_heap *h, void *e)
Insert a new object into the heap.
Definition heap.c:122
static size_t cstl_heap_size(const struct cstl_heap *const h)
Get the number of objects in the heap.
Definition heap.h:135
static void cstl_heap_clear(struct cstl_heap *const h, cstl_xtor_func_t *const clr)
Remove all elements from the heap.
Definition heap.h:191
static void cstl_heap_swap(struct cstl_heap *const a, struct cstl_heap *const b)
Swap the heap objects at the two given locations.
Definition heap.h:206
Node to anchor an element within a binary tree.
Definition bintree.h:49
Binary tree object.
Definition bintree.h:63
Node to anchor an element within a heap.
Definition heap.h:48
Heap object.
Definition heap.h:62