]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/basic/hashmap.h
pkgconfig: define variables relative to ${prefix}/${rootprefix}/${sysconfdir}
[thirdparty/systemd.git] / src / basic / hashmap.h
1 /* SPDX-License-Identifier: LGPL-2.1+ */
2 #pragma once
3
4 #include <limits.h>
5 #include <stdbool.h>
6 #include <stddef.h>
7
8 #include "hash-funcs.h"
9 #include "macro.h"
10 #include "util.h"
11
12 /*
13 * A hash table implementation. As a minor optimization a NULL hashmap object
14 * will be treated as empty hashmap for all read operations. That way it is not
15 * necessary to instantiate an object for each Hashmap use.
16 *
17 * If ENABLE_DEBUG_HASHMAP is defined (by configuring with --enable-debug=hashmap),
18 * the implementation will:
19 * - store extra data for debugging and statistics (see tools/gdb-sd_dump_hashmaps.py)
20 * - perform extra checks for invalid use of iterators
21 */
22
23 #define HASH_KEY_SIZE 16
24
25 /* The base type for all hashmap and set types. Many functions in the
26 * implementation take (HashmapBase*) parameters and are run-time polymorphic,
27 * though the API is not meant to be polymorphic (do not call functions
28 * internal_*() directly). */
29 typedef struct HashmapBase HashmapBase;
30
31 /* Specific hashmap/set types */
32 typedef struct Hashmap Hashmap; /* Maps keys to values */
33 typedef struct OrderedHashmap OrderedHashmap; /* Like Hashmap, but also remembers entry insertion order */
34 typedef struct Set Set; /* Stores just keys */
35
36 typedef struct IteratedCache IteratedCache; /* Caches the iterated order of one of the above */
37
38 /* Ideally the Iterator would be an opaque struct, but it is instantiated
39 * by hashmap users, so the definition has to be here. Do not use its fields
40 * directly. */
41 typedef struct {
42 unsigned idx; /* index of an entry to be iterated next */
43 const void *next_key; /* expected value of that entry's key pointer */
44 #if ENABLE_DEBUG_HASHMAP
45 unsigned put_count; /* hashmap's put_count recorded at start of iteration */
46 unsigned rem_count; /* hashmap's rem_count in previous iteration */
47 unsigned prev_idx; /* idx in previous iteration */
48 #endif
49 } Iterator;
50
51 #define _IDX_ITERATOR_FIRST (UINT_MAX - 1)
52 #define ITERATOR_FIRST ((Iterator) { .idx = _IDX_ITERATOR_FIRST, .next_key = NULL })
53
54 /* Macros for type checking */
55 #define PTR_COMPATIBLE_WITH_HASHMAP_BASE(h) \
56 (__builtin_types_compatible_p(typeof(h), HashmapBase*) || \
57 __builtin_types_compatible_p(typeof(h), Hashmap*) || \
58 __builtin_types_compatible_p(typeof(h), OrderedHashmap*) || \
59 __builtin_types_compatible_p(typeof(h), Set*))
60
61 #define PTR_COMPATIBLE_WITH_PLAIN_HASHMAP(h) \
62 (__builtin_types_compatible_p(typeof(h), Hashmap*) || \
63 __builtin_types_compatible_p(typeof(h), OrderedHashmap*)) \
64
65 #define HASHMAP_BASE(h) \
66 __builtin_choose_expr(PTR_COMPATIBLE_WITH_HASHMAP_BASE(h), \
67 (HashmapBase*)(h), \
68 (void)0)
69
70 #define PLAIN_HASHMAP(h) \
71 __builtin_choose_expr(PTR_COMPATIBLE_WITH_PLAIN_HASHMAP(h), \
72 (Hashmap*)(h), \
73 (void)0)
74
75 #if ENABLE_DEBUG_HASHMAP
76 # define HASHMAP_DEBUG_PARAMS , const char *func, const char *file, int line
77 # define HASHMAP_DEBUG_SRC_ARGS , __func__, __FILE__, __LINE__
78 # define HASHMAP_DEBUG_PASS_ARGS , func, file, line
79 #else
80 # define HASHMAP_DEBUG_PARAMS
81 # define HASHMAP_DEBUG_SRC_ARGS
82 # define HASHMAP_DEBUG_PASS_ARGS
83 #endif
84
85 Hashmap *internal_hashmap_new(const struct hash_ops *hash_ops HASHMAP_DEBUG_PARAMS);
86 OrderedHashmap *internal_ordered_hashmap_new(const struct hash_ops *hash_ops HASHMAP_DEBUG_PARAMS);
87 #define hashmap_new(ops) internal_hashmap_new(ops HASHMAP_DEBUG_SRC_ARGS)
88 #define ordered_hashmap_new(ops) internal_ordered_hashmap_new(ops HASHMAP_DEBUG_SRC_ARGS)
89
90 HashmapBase *internal_hashmap_free(HashmapBase *h);
91 static inline Hashmap *hashmap_free(Hashmap *h) {
92 return (void*)internal_hashmap_free(HASHMAP_BASE(h));
93 }
94 static inline OrderedHashmap *ordered_hashmap_free(OrderedHashmap *h) {
95 return (void*)internal_hashmap_free(HASHMAP_BASE(h));
96 }
97
98 HashmapBase *internal_hashmap_free_free(HashmapBase *h);
99 static inline Hashmap *hashmap_free_free(Hashmap *h) {
100 return (void*)internal_hashmap_free_free(HASHMAP_BASE(h));
101 }
102 static inline OrderedHashmap *ordered_hashmap_free_free(OrderedHashmap *h) {
103 return (void*)internal_hashmap_free_free(HASHMAP_BASE(h));
104 }
105
106 Hashmap *hashmap_free_free_free(Hashmap *h);
107 static inline OrderedHashmap *ordered_hashmap_free_free_free(OrderedHashmap *h) {
108 return (void*)hashmap_free_free_free(PLAIN_HASHMAP(h));
109 }
110
111 IteratedCache *iterated_cache_free(IteratedCache *cache);
112 int iterated_cache_get(IteratedCache *cache, const void ***res_keys, const void ***res_values, unsigned *res_n_entries);
113
114 HashmapBase *internal_hashmap_copy(HashmapBase *h);
115 static inline Hashmap *hashmap_copy(Hashmap *h) {
116 return (Hashmap*) internal_hashmap_copy(HASHMAP_BASE(h));
117 }
118 static inline OrderedHashmap *ordered_hashmap_copy(OrderedHashmap *h) {
119 return (OrderedHashmap*) internal_hashmap_copy(HASHMAP_BASE(h));
120 }
121
122 int internal_hashmap_ensure_allocated(Hashmap **h, const struct hash_ops *hash_ops HASHMAP_DEBUG_PARAMS);
123 int internal_ordered_hashmap_ensure_allocated(OrderedHashmap **h, const struct hash_ops *hash_ops HASHMAP_DEBUG_PARAMS);
124 #define hashmap_ensure_allocated(h, ops) internal_hashmap_ensure_allocated(h, ops HASHMAP_DEBUG_SRC_ARGS)
125 #define ordered_hashmap_ensure_allocated(h, ops) internal_ordered_hashmap_ensure_allocated(h, ops HASHMAP_DEBUG_SRC_ARGS)
126
127 IteratedCache *internal_hashmap_iterated_cache_new(HashmapBase *h);
128 static inline IteratedCache *hashmap_iterated_cache_new(Hashmap *h) {
129 return (IteratedCache*) internal_hashmap_iterated_cache_new(HASHMAP_BASE(h));
130 }
131 static inline IteratedCache *ordered_hashmap_iterated_cache_new(OrderedHashmap *h) {
132 return (IteratedCache*) internal_hashmap_iterated_cache_new(HASHMAP_BASE(h));
133 }
134
135 int hashmap_put(Hashmap *h, const void *key, void *value);
136 static inline int ordered_hashmap_put(OrderedHashmap *h, const void *key, void *value) {
137 return hashmap_put(PLAIN_HASHMAP(h), key, value);
138 }
139
140 int hashmap_update(Hashmap *h, const void *key, void *value);
141 static inline int ordered_hashmap_update(OrderedHashmap *h, const void *key, void *value) {
142 return hashmap_update(PLAIN_HASHMAP(h), key, value);
143 }
144
145 int hashmap_replace(Hashmap *h, const void *key, void *value);
146 static inline int ordered_hashmap_replace(OrderedHashmap *h, const void *key, void *value) {
147 return hashmap_replace(PLAIN_HASHMAP(h), key, value);
148 }
149
150 void *internal_hashmap_get(HashmapBase *h, const void *key);
151 static inline void *hashmap_get(Hashmap *h, const void *key) {
152 return internal_hashmap_get(HASHMAP_BASE(h), key);
153 }
154 static inline void *ordered_hashmap_get(OrderedHashmap *h, const void *key) {
155 return internal_hashmap_get(HASHMAP_BASE(h), key);
156 }
157
158 void *hashmap_get2(Hashmap *h, const void *key, void **rkey);
159 static inline void *ordered_hashmap_get2(OrderedHashmap *h, const void *key, void **rkey) {
160 return hashmap_get2(PLAIN_HASHMAP(h), key, rkey);
161 }
162
163 bool internal_hashmap_contains(HashmapBase *h, const void *key);
164 static inline bool hashmap_contains(Hashmap *h, const void *key) {
165 return internal_hashmap_contains(HASHMAP_BASE(h), key);
166 }
167 static inline bool ordered_hashmap_contains(OrderedHashmap *h, const void *key) {
168 return internal_hashmap_contains(HASHMAP_BASE(h), key);
169 }
170
171 void *internal_hashmap_remove(HashmapBase *h, const void *key);
172 static inline void *hashmap_remove(Hashmap *h, const void *key) {
173 return internal_hashmap_remove(HASHMAP_BASE(h), key);
174 }
175 static inline void *ordered_hashmap_remove(OrderedHashmap *h, const void *key) {
176 return internal_hashmap_remove(HASHMAP_BASE(h), key);
177 }
178
179 void *hashmap_remove2(Hashmap *h, const void *key, void **rkey);
180 static inline void *ordered_hashmap_remove2(OrderedHashmap *h, const void *key, void **rkey) {
181 return hashmap_remove2(PLAIN_HASHMAP(h), key, rkey);
182 }
183
184 void *hashmap_remove_value(Hashmap *h, const void *key, void *value);
185 static inline void *ordered_hashmap_remove_value(OrderedHashmap *h, const void *key, void *value) {
186 return hashmap_remove_value(PLAIN_HASHMAP(h), key, value);
187 }
188
189 int hashmap_remove_and_put(Hashmap *h, const void *old_key, const void *new_key, void *value);
190 static inline int ordered_hashmap_remove_and_put(OrderedHashmap *h, const void *old_key, const void *new_key, void *value) {
191 return hashmap_remove_and_put(PLAIN_HASHMAP(h), old_key, new_key, value);
192 }
193
194 int hashmap_remove_and_replace(Hashmap *h, const void *old_key, const void *new_key, void *value);
195 static inline int ordered_hashmap_remove_and_replace(OrderedHashmap *h, const void *old_key, const void *new_key, void *value) {
196 return hashmap_remove_and_replace(PLAIN_HASHMAP(h), old_key, new_key, value);
197 }
198
199 /* Since merging data from a OrderedHashmap into a Hashmap or vice-versa
200 * should just work, allow this by having looser type-checking here. */
201 int internal_hashmap_merge(Hashmap *h, Hashmap *other);
202 #define hashmap_merge(h, other) internal_hashmap_merge(PLAIN_HASHMAP(h), PLAIN_HASHMAP(other))
203 #define ordered_hashmap_merge(h, other) hashmap_merge(h, other)
204
205 int internal_hashmap_reserve(HashmapBase *h, unsigned entries_add);
206 static inline int hashmap_reserve(Hashmap *h, unsigned entries_add) {
207 return internal_hashmap_reserve(HASHMAP_BASE(h), entries_add);
208 }
209 static inline int ordered_hashmap_reserve(OrderedHashmap *h, unsigned entries_add) {
210 return internal_hashmap_reserve(HASHMAP_BASE(h), entries_add);
211 }
212
213 int internal_hashmap_move(HashmapBase *h, HashmapBase *other);
214 /* Unlike hashmap_merge, hashmap_move does not allow mixing the types. */
215 static inline int hashmap_move(Hashmap *h, Hashmap *other) {
216 return internal_hashmap_move(HASHMAP_BASE(h), HASHMAP_BASE(other));
217 }
218 static inline int ordered_hashmap_move(OrderedHashmap *h, OrderedHashmap *other) {
219 return internal_hashmap_move(HASHMAP_BASE(h), HASHMAP_BASE(other));
220 }
221
222 int internal_hashmap_move_one(HashmapBase *h, HashmapBase *other, const void *key);
223 static inline int hashmap_move_one(Hashmap *h, Hashmap *other, const void *key) {
224 return internal_hashmap_move_one(HASHMAP_BASE(h), HASHMAP_BASE(other), key);
225 }
226 static inline int ordered_hashmap_move_one(OrderedHashmap *h, OrderedHashmap *other, const void *key) {
227 return internal_hashmap_move_one(HASHMAP_BASE(h), HASHMAP_BASE(other), key);
228 }
229
230 unsigned internal_hashmap_size(HashmapBase *h) _pure_;
231 static inline unsigned hashmap_size(Hashmap *h) {
232 return internal_hashmap_size(HASHMAP_BASE(h));
233 }
234 static inline unsigned ordered_hashmap_size(OrderedHashmap *h) {
235 return internal_hashmap_size(HASHMAP_BASE(h));
236 }
237
238 static inline bool hashmap_isempty(Hashmap *h) {
239 return hashmap_size(h) == 0;
240 }
241 static inline bool ordered_hashmap_isempty(OrderedHashmap *h) {
242 return ordered_hashmap_size(h) == 0;
243 }
244
245 unsigned internal_hashmap_buckets(HashmapBase *h) _pure_;
246 static inline unsigned hashmap_buckets(Hashmap *h) {
247 return internal_hashmap_buckets(HASHMAP_BASE(h));
248 }
249 static inline unsigned ordered_hashmap_buckets(OrderedHashmap *h) {
250 return internal_hashmap_buckets(HASHMAP_BASE(h));
251 }
252
253 bool internal_hashmap_iterate(HashmapBase *h, Iterator *i, void **value, const void **key);
254 static inline bool hashmap_iterate(Hashmap *h, Iterator *i, void **value, const void **key) {
255 return internal_hashmap_iterate(HASHMAP_BASE(h), i, value, key);
256 }
257 static inline bool ordered_hashmap_iterate(OrderedHashmap *h, Iterator *i, void **value, const void **key) {
258 return internal_hashmap_iterate(HASHMAP_BASE(h), i, value, key);
259 }
260
261 void internal_hashmap_clear(HashmapBase *h);
262 static inline void hashmap_clear(Hashmap *h) {
263 internal_hashmap_clear(HASHMAP_BASE(h));
264 }
265 static inline void ordered_hashmap_clear(OrderedHashmap *h) {
266 internal_hashmap_clear(HASHMAP_BASE(h));
267 }
268
269 void internal_hashmap_clear_free(HashmapBase *h);
270 static inline void hashmap_clear_free(Hashmap *h) {
271 internal_hashmap_clear_free(HASHMAP_BASE(h));
272 }
273 static inline void ordered_hashmap_clear_free(OrderedHashmap *h) {
274 internal_hashmap_clear_free(HASHMAP_BASE(h));
275 }
276
277 void hashmap_clear_free_free(Hashmap *h);
278 static inline void ordered_hashmap_clear_free_free(OrderedHashmap *h) {
279 hashmap_clear_free_free(PLAIN_HASHMAP(h));
280 }
281
282 /*
283 * Note about all *_first*() functions
284 *
285 * For plain Hashmaps and Sets the order of entries is undefined.
286 * The functions find whatever entry is first in the implementation
287 * internal order.
288 *
289 * Only for OrderedHashmaps the order is well defined and finding
290 * the first entry is O(1).
291 */
292
293 void *internal_hashmap_first_key_and_value(HashmapBase *h, bool remove, void **ret_key);
294 static inline void *hashmap_steal_first_key_and_value(Hashmap *h, void **ret) {
295 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), true, ret);
296 }
297 static inline void *ordered_hashmap_steal_first_key_and_value(OrderedHashmap *h, void **ret) {
298 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), true, ret);
299 }
300 static inline void *hashmap_first_key_and_value(Hashmap *h, void **ret) {
301 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), false, ret);
302 }
303 static inline void *ordered_hashmap_first_key_and_value(OrderedHashmap *h, void **ret) {
304 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), false, ret);
305 }
306
307
308 static inline void *hashmap_steal_first(Hashmap *h) {
309 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), true, NULL);
310 }
311 static inline void *ordered_hashmap_steal_first(OrderedHashmap *h) {
312 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), true, NULL);
313 }
314 static inline void *hashmap_first(Hashmap *h) {
315 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), false, NULL);
316 }
317 static inline void *ordered_hashmap_first(OrderedHashmap *h) {
318 return internal_hashmap_first_key_and_value(HASHMAP_BASE(h), false, NULL);
319 }
320
321 static inline void *internal_hashmap_first_key(HashmapBase *h, bool remove) {
322 void *key = NULL;
323
324 (void) internal_hashmap_first_key_and_value(HASHMAP_BASE(h), remove, &key);
325 return key;
326 }
327 static inline void *hashmap_steal_first_key(Hashmap *h) {
328 return internal_hashmap_first_key(HASHMAP_BASE(h), true);
329 }
330 static inline void *ordered_hashmap_steal_first_key(OrderedHashmap *h) {
331 return internal_hashmap_first_key(HASHMAP_BASE(h), true);
332 }
333 static inline void *hashmap_first_key(Hashmap *h) {
334 return internal_hashmap_first_key(HASHMAP_BASE(h), false);
335 }
336 static inline void *ordered_hashmap_first_key(OrderedHashmap *h) {
337 return internal_hashmap_first_key(HASHMAP_BASE(h), false);
338 }
339
340 #define hashmap_clear_with_destructor(_s, _f) \
341 ({ \
342 void *_item; \
343 while ((_item = hashmap_steal_first(_s))) \
344 _f(_item); \
345 })
346 #define hashmap_free_with_destructor(_s, _f) \
347 ({ \
348 hashmap_clear_with_destructor(_s, _f); \
349 hashmap_free(_s); \
350 })
351 #define ordered_hashmap_clear_with_destructor(_s, _f) \
352 ({ \
353 void *_item; \
354 while ((_item = ordered_hashmap_steal_first(_s))) \
355 _f(_item); \
356 })
357 #define ordered_hashmap_free_with_destructor(_s, _f) \
358 ({ \
359 ordered_hashmap_clear_with_destructor(_s, _f); \
360 ordered_hashmap_free(_s); \
361 })
362
363 /* no hashmap_next */
364 void *ordered_hashmap_next(OrderedHashmap *h, const void *key);
365
366 char **internal_hashmap_get_strv(HashmapBase *h);
367 static inline char **hashmap_get_strv(Hashmap *h) {
368 return internal_hashmap_get_strv(HASHMAP_BASE(h));
369 }
370 static inline char **ordered_hashmap_get_strv(OrderedHashmap *h) {
371 return internal_hashmap_get_strv(HASHMAP_BASE(h));
372 }
373
374 /*
375 * Hashmaps are iterated in unpredictable order.
376 * OrderedHashmaps are an exception to this. They are iterated in the order
377 * the entries were inserted.
378 * It is safe to remove the current entry.
379 */
380 #define HASHMAP_FOREACH(e, h, i) \
381 for ((i) = ITERATOR_FIRST; hashmap_iterate((h), &(i), (void**)&(e), NULL); )
382
383 #define ORDERED_HASHMAP_FOREACH(e, h, i) \
384 for ((i) = ITERATOR_FIRST; ordered_hashmap_iterate((h), &(i), (void**)&(e), NULL); )
385
386 #define HASHMAP_FOREACH_KEY(e, k, h, i) \
387 for ((i) = ITERATOR_FIRST; hashmap_iterate((h), &(i), (void**)&(e), (const void**) &(k)); )
388
389 #define ORDERED_HASHMAP_FOREACH_KEY(e, k, h, i) \
390 for ((i) = ITERATOR_FIRST; ordered_hashmap_iterate((h), &(i), (void**)&(e), (const void**) &(k)); )
391
392 DEFINE_TRIVIAL_CLEANUP_FUNC(Hashmap*, hashmap_free);
393 DEFINE_TRIVIAL_CLEANUP_FUNC(Hashmap*, hashmap_free_free);
394 DEFINE_TRIVIAL_CLEANUP_FUNC(Hashmap*, hashmap_free_free_free);
395 DEFINE_TRIVIAL_CLEANUP_FUNC(OrderedHashmap*, ordered_hashmap_free);
396 DEFINE_TRIVIAL_CLEANUP_FUNC(OrderedHashmap*, ordered_hashmap_free_free);
397 DEFINE_TRIVIAL_CLEANUP_FUNC(OrderedHashmap*, ordered_hashmap_free_free_free);
398
399 #define _cleanup_hashmap_free_ _cleanup_(hashmap_freep)
400 #define _cleanup_hashmap_free_free_ _cleanup_(hashmap_free_freep)
401 #define _cleanup_hashmap_free_free_free_ _cleanup_(hashmap_free_free_freep)
402 #define _cleanup_ordered_hashmap_free_ _cleanup_(ordered_hashmap_freep)
403 #define _cleanup_ordered_hashmap_free_free_ _cleanup_(ordered_hashmap_free_freep)
404 #define _cleanup_ordered_hashmap_free_free_free_ _cleanup_(ordered_hashmap_free_free_freep)
405
406 DEFINE_TRIVIAL_CLEANUP_FUNC(IteratedCache*, iterated_cache_free);
407
408 #define _cleanup_iterated_cache_free_ _cleanup_(iterated_cache_freep)