Line data Source code
1 : /* Generate prototypes, inlines and/or implementations for ultra high
2 : performance dynamic key-val maps of gigantic size. A giant map can
3 : be persisting beyond the lifetime of creating process, used
4 : concurrently, used IPC, relocated in memory, naively
5 : serialized/deserialized and/or moved between hosts. Memory
6 : efficiency and flexible footprint are prioritized. Elements that are
7 : recently used can be optionally moved to the front of the chains to
8 : adaptively optimize the maps for recent queries. Typical usage:
9 :
10 : struct mymap {
11 : ulong key; // Technically "MAP_KEY_T MAP_KEY;" (default is ulong key), should not be touched by user
12 : ulong next; // Technically "ulong MAP_NEXT;", should not be touched by user
13 : ... key and next can be located arbitrarily in struct
14 : ... the mapping of a key to an array index is arbitrary but is
15 : ... fixed over the lifetime of the key in the map
16 : };
17 :
18 : typedef struct mymap mymap_t;
19 :
20 : #define MAP_NAME mymap
21 : #define MAP_T mymap_t
22 : #include "tmpl/fd_map_giant.c"
23 :
24 : will declare the following APIs as a header only style library in the
25 : compilation unit:
26 :
27 : // mymap_{align,footprint} returns alignment and footprint needed
28 : // for a memory region to be used as a mymap. align will be an
29 : // integer power-of-two and footprint will be a multiple of align.
30 : // footprint will returns 0 if key_max requires a footprint that
31 : // would overflow 64-bit. key_max is the maximum number of keys
32 : // (elements) the map can hold.
33 : //
34 : // mymap_new formats a memory region with the appropriate alignment
35 : // and footprint whose first byte in the caller's address space is
36 : // pointed by shmem. Returns shmem on success and NULL on failure
37 : // (logs details). Caller is not joined on return. The map will be
38 : // empty.
39 : //
40 : // mymap_join joins a mymap. Assumes shmap points at a memory
41 : // region formatted as a mymap in the caller's address space.
42 : // Returns a handle to the caller's local join on success and NULL
43 : // on failure (logs details). THIS IS NOT A SIMPLE CAST OF SHMAP!
44 : // The join can be indexed as a flat array with key_max elements in
45 : // the caller's address space.
46 : //
47 : // mymap_leave leaves a mymap. Assumes join points to a current
48 : // local join. Returns shmap used on join. THIS IS NOT A SIMPLE
49 : // CASE OF JOIN!
50 : //
51 : // mymap_delete unformats a memory region used as a mymap. Assumes
52 : // shmap points to a memory region in the caller's local address
53 : // space formatted as a mymap, that there are no joins to the mymap
54 : // and that any application cleanup of the entries has already been
55 : // done. Returns shmap on success and NULL on failure.
56 :
57 : ulong mymap_align ( void );
58 : ulong mymap_footprint( ulong key_max );
59 : void * mymap_new ( void * shmem, ulong key_max, ulong seed );
60 : mymap_t * mymap_join ( void * shmap ); // Indexed [0,key_max)
61 : void * mymap_leave ( mymap_t * join );
62 : void * mymap_delete ( void * shmap );
63 :
64 : // mymap_key_max and mymap_seed return the values used to construct
65 : // the map. They assume join is a current local join. The values
66 : // will be constant for the map lifetime.
67 :
68 : ulong mymap_key_max( mymap_t const * join );
69 : ulong mymap_seed ( mymap_t const * join );
70 :
71 : // mymap_key_cnt returns the current number of keys in the map.
72 : // Will be in [0,key_max]. mymap_is_full returns 1 if
73 : // key_cnt==key_max. Assumes join is a current local join. The
74 : // values will be constant until the next map insert / remove.
75 :
76 : ulong mymap_key_cnt( mymap_t const * join );
77 : int mymap_is_full( mymap_t const * join );
78 :
79 : // mymap_key_{eq,hash,copy} expose the provided
80 : // MAP_KEY_{EQ,HASH,COPY} macros as inline functions with strict
81 : // semantics. They assume that the provided pointers are in the
82 : // caller's address space to keys that will not be changed
83 : // concurrently. They retains no interest in key on return.
84 : //
85 : // mymap_key_eq returns 1 if the *k0 and *k1 are equal and 0
86 : // otherwise.
87 : //
88 : // mymap_key_hash returns the hash *key using the hash function
89 : // seed. Should ideally be a random mapping from MAP_KEY_T -> ulong
90 : // but this depends on what the user actually passed in for
91 : // MAP_HASH. The seed used by a particular instance of a giant map
92 : // can be obtained above.
93 : //
94 : // mymap_key_copy deep copies *ks into *kd and returns kd.
95 :
96 : int mymap_key_eq ( ulong * k0, ulong * k1 );
97 : ulong mymap_key_hash( ulong * key, ulong seed );
98 : ulong * mymap_key_copy( ulong * kd, ulong const * ks );
99 :
100 : // mymap_insert inserts the key pointed to by key into the map.
101 : // Returns the location in the caller's address space of the element
102 : // for key in the map or NULL if there was no space in the map.
103 : //
104 : // Assumes map is a current local join, key points to a valid key in
105 : // the caller's address space, there are no concurrent operations on
106 : // the map or key. The map retains no interest in key. The
107 : // returned element will be valid for the lesser of the lifetime of
108 : // the join and until key is removed from the map.
109 : //
110 : // Critically, as this is used in high performance contexts where
111 : // the application already knows this, THE CALLER PROMISES THE KEY
112 : // IS NOT IN THE MAP AND THAT THE MAP HAS SPACE FOR KEY.
113 : //
114 : // This always succeeds (with the above requirements) and returns
115 : // non NULL.
116 :
117 : mymap_t * mymap_insert( mymap_t * join, ulong const * key );
118 :
119 : // mymap_remove removes the key pointed to by key from the map. A
120 : // pointer in the caller's address space to the former element is
121 : // returned to allow additional cleanup of the application fields.
122 : // Returns NULL if key is not in the map.
123 : //
124 : // Assumes map is a current local join, key points to a valid key in
125 : // the caller's address space and there are no concurrent operations
126 : // on the map or key. The map retains no interest in key. Any
127 : // returned pointer will be valid for the lesser of the lifetime of
128 : // the join and until the next insert.
129 :
130 : mymap_t * mymap_remove( mymap_t * join, ulong const * key );
131 :
132 : // mymap_query finds the element for the key pointed to by key in
133 : // the map and returns a pointer to it in the caller's address
134 : // space. If key is not found, returns sentinel (usually pass
135 : // NULL).
136 : //
137 : // Assumes map is a current local join, key points to a valid key in
138 : // the caller's address space and there are no concurrent operations
139 : // on the map or key. The map retains no interest in key. On
140 : // success, the returned pointer will be valid for the lesser of the
141 : // lifetime of join or until key is removed. On failure, the
142 : // returned pointer lifetime will be that of the sentinel.
143 :
144 : mymap_t * mymap_query( mymap_t * join, ulong const * key, mymap_t * sentinel );
145 :
146 : // mymap_query_const is the same as mymap_query but supports
147 : // concurrent queries so long as there no concurrently running
148 : // insert/remove/query operations. The value fields of the mymap_t
149 : // returned by this function can be changed by the application (but
150 : // it is up to the application to manage concurrency between
151 : // different users modifying the same key).
152 :
153 : mymap_t const * mymap_query_const( mymap_t const * join, ulong const * key, mymap_t const * sentinel );
154 :
155 : // mymap_query_safe is the same as mymap_query_const but with
156 : // additional safety checks. If a concurrent write is occuring,
157 : // this API will still return a resonable map entry without
158 : // crashing, even if it is wrong.
159 :
160 : mymap_t const * mymap_query_safe( mymap_t const * join, ulong const * key, mymap_t const * sentinel );
161 :
162 : // mymap_iter_* allow for iteration over all the keys inserted into
163 : // a mymap. The iteration will be in a random order but the order
164 : // will be identical if repeated with no insert/remove/query
165 : // operations done in between. Assumes no concurrent
166 : // insert/remove/query operations. Example usage:
167 : //
168 : // for( mymap_t iter = mymap_iter_init( join ); !mymap_iter_done( join, iter ); iter = mymap_iter_next( join, iter ) ) {
169 : // mymap_t * ele = mymap_iter_ele( join, iter );
170 : //
171 : // ... process ele here
172 : //
173 : // ... IMPORTANT! It is _okay_ to insert, remove, query or
174 : // ... query_const here. In particular, if an element is
175 : // ... inserted, it may or may not be covered by this iteration.
176 : // ... If an element is removed that has not already been
177 : // ... iterated over, it will not be iterated over in this
178 : // ... iteration. It is fine to remove the current item being
179 : // ... iterated over.
180 : //
181 : // ... WARNING! While this looks like an O(key_cnt) operation,
182 : // ... technically, this is an O(key_max) operation under the
183 : // ... hood. As such, use outside critical paths (e.g.
184 : // ... checkpointing), on dense maps (e.g. key_cnt/key_max ~
185 : // ... O(1)) and/or on maps where key_max is small enough not to
186 : // ... matter practically.
187 : // }
188 :
189 : struct mymap_iter_private { ... internal use only ... };
190 : typedef struct mymap_iter_private mymap_iter_t;
191 :
192 : mymap_iter_t mymap_iter_init ( mymap_t const * join, mymap_iter_t iter ); // returns iter, NULL join fine
193 : int mymap_iter_done ( mymap_t const * join, mymap_iter_t iter ); // returns 1 if no more iterations, 0 o.w.
194 : mymap_iter_t mymap_iter_next ( mymap_t const * join, mymap_iter_t iter ); // returns next iter value iter
195 : mymap_t * mymap_iter_ele ( mymap_t * join, mymap_iter_t iter ); // assumes not done, return non-NULL ele
196 : mymap_t const * mymap_iter_ele_const( mymap_t const * join, mymap_iter_t iter ); // assumes not done, return non-NULL ele
197 :
198 : // mymap_verify returns 0 if the mymap is not obviously corrupt or a
199 : // -1 (i.e. ERR_INVAL) if it is obviously corrupt (logs details).
200 : // join is the handle of a current local join to mymap.
201 :
202 : int mymap_verify( mymap_t const * join );
203 :
204 : You can do this as often as you like in a compilation unit to get
205 : different types of gigantic maps. Variants exist for making header
206 : protoypes only and/or implementations if doing a library with multiple
207 : compilation units. Further, options exist to use different hashing
208 : functions, comparison functions, etc as detailed below. */
209 :
210 : /* MAP_NAME gives the API prefix to use for map */
211 :
212 : #ifndef MAP_NAME
213 : #error "Define MAP_NAME"
214 : #endif
215 :
216 : /* MAP_T is the map element type. */
217 :
218 : #ifndef MAP_T
219 : #error "Define MAP_T"
220 : #endif
221 :
222 : /* MAP_KEY_T is the map key type */
223 :
224 : #ifndef MAP_KEY_T
225 : #define MAP_KEY_T ulong
226 : #endif
227 :
228 : /* MAP_KEY is the MAP_T key field */
229 :
230 : #ifndef MAP_KEY
231 0 : #define MAP_KEY key
232 : #endif
233 :
234 : /* MAP_NEXT is the MAP_T next field */
235 :
236 : #ifndef MAP_NEXT
237 4078590000 : #define MAP_NEXT next
238 : #endif
239 :
240 : /* MAP_KEY_EQ returns 0/1 if *k0 is the same/different as *k1 */
241 :
242 : #ifndef MAP_KEY_EQ
243 4148822646 : #define MAP_KEY_EQ(k0,k1) ((*(k0))==(*(k1)))
244 : #endif
245 :
246 : /* MAP_KEY_HASH maps *key into ulong uniform pseudo randomly. */
247 :
248 : #ifndef MAP_KEY_HASH
249 5003218794 : #define MAP_KEY_HASH(key,seed) fd_ulong_hash( (*(key)) ^ (seed) )
250 : #endif
251 :
252 : /* MAP_KEY_COPY copys the contents from *ks to *kd. Non-POD key types
253 : might need to customize this accordingly. Defaults to the copy
254 : operator. */
255 :
256 : #ifndef MAP_KEY_COPY
257 9072000 : #define MAP_KEY_COPY(kd,ks) (*(kd))=(*(ks))
258 : #endif
259 :
260 : /* MAP_MAGIC is the magic number to use for the structure to aid in
261 : persistent and or IPC usage. */
262 :
263 : #ifndef MAP_MAGIC
264 407865 : #define MAP_MAGIC (0xf17eda2c3763a900UL) /* firedancer gmap version 0 */
265 : #endif
266 :
267 : /* 0 - local use only
268 : 1 - library header declaration
269 : 2 - library implementation */
270 :
271 : #ifndef MAP_IMPL_STYLE
272 : #define MAP_IMPL_STYLE 0
273 : #endif
274 :
275 : /* If MAP_MEMOIZE is defined to non-zero, the MAP_T requires a
276 : "hash" field that will hold the value of the MAP_KEY_HASH of the
277 : MAP_T's MAP_KEY when the map slot is not empty (undefined otherwise).
278 : This is useful for accelerating user operations that might need a
279 : hash of the key and for accelerating remove operations. It is also
280 : potentially useful as a way to accelerate slow key comparison
281 : operations (see MAP_KEY_EQUAL_IS_SLOW). */
282 :
283 : #ifndef MAP_MEMOIZE
284 : #define MAP_MEMOIZE 0
285 : #endif
286 :
287 : /* If MAP_MEMOIZE is non-zero, MAP_HASH is the element member to store the hash in. */
288 :
289 : #ifndef MAP_HASH
290 : #define MAP_HASH map_hash
291 : #endif
292 :
293 : /* Implementation *****************************************************/
294 :
295 : /* Constructors and verification logs detail on failure (rest only needs
296 : fd_bits.h, consider making logging a compile time option). */
297 :
298 : #include "../log/fd_log.h"
299 :
300 96962132552 : #define MAP_IDX_NULL (~(1UL<<63)) /* 2^63-1 */
301 >18818*10^7 : #define MAP_(n) FD_EXPAND_THEN_CONCAT3(MAP_NAME,_,n)
302 :
303 : #if MAP_IMPL_STYLE==0 || MAP_IMPL_STYLE==1 /* need structures and inlines */
304 :
305 : struct MAP_(private) {
306 :
307 : /* This point is max(128,alignof(MAP_T)) aligned */
308 :
309 : /* 2 ulong here, index 0 is MAP_MAGIC, index 1 is offset from shmem
310 : region to key_max map array below. */
311 :
312 : /* alignment padding here */
313 :
314 : /* list_cnt ulong here, indexed [0,list_cnt)
315 : list[ list_idx ], idx is in [0,key_max) or MAP_IDX_NULL, tag is used */
316 :
317 : ulong key_max; /* yields non-zero footprint, <2^63 */
318 : ulong seed; /* hash seed, arbitrary */
319 : ulong list_cnt; /* == MAP_(private_list_cnt)( key_max ) */
320 : ulong key_cnt; /* in [0,key_max] */
321 : ulong free_stack; /* idx is in [0,key_max) or MAP_IDX_NULL, tag is free */
322 :
323 : /* Elements on the free_stack will have bit 63 of their next field set
324 : to facilitate iteration and validation. Elements in lists will
325 : have their bit 63 of their next field clear. key_max<2^63 ensures
326 : no confusion possible with MAP_IDX_NULL though this is more
327 : theoretical as ~2^63 keys is impractical physically. */
328 :
329 : /* This point is max(128,alignof(MAP_T)) aligned */
330 :
331 : /* key_max MAP_T here, hdr[1] above points here */
332 :
333 : };
334 :
335 : typedef struct MAP_(private) MAP_(private_t);
336 :
337 : typedef ulong MAP_(iter_t);
338 :
339 : FD_PROTOTYPES_BEGIN
340 :
341 : /* map_private_list_cnt returns the number of lists a map with the given
342 : key_max should use. */
343 :
344 : FD_FN_CONST static inline ulong
345 103839357 : MAP_(private_list_cnt)( ulong key_max ) {
346 : /* Compute the number of lists as the power of 2 that makes the
347 : average chain length between ~1 and ~2 */
348 103839357 : ulong list_min = (key_max>>1) + (key_max&1UL); /* list_min = ceil(key_max/2), at most 2^63, computed without overflow */
349 103839357 : ulong list_cnt = fd_ulong_pow2_up( list_min ); /* at most 2^63 */
350 103839357 : return list_cnt;
351 103839357 : }
352 :
353 : FD_FN_CONST static inline ulong
354 55593534 : MAP_(private_meta_footprint)( ulong list_cnt ) {
355 55593534 : return fd_ulong_align_up( (2UL + list_cnt)*sizeof(ulong) + sizeof(MAP_(private_t)), fd_ulong_max( 128UL, alignof(MAP_T) ) );
356 55593534 : }
357 :
358 : /* map_private returns the location of the map private for a current
359 : local join. Assumes join is a current local join. map_private_const
360 : is a const correct version. */
361 :
362 : FD_FN_CONST static inline MAP_(private_t) *
363 2592585651 : MAP_(private)( MAP_T * join ) {
364 2592585651 : return (MAP_(private_t) *)(((ulong)join) - sizeof(MAP_(private_t)));
365 2592585651 : }
366 :
367 : FD_FN_CONST static inline MAP_(private_t) const *
368 3604356723 : MAP_(private_const)( MAP_T const * join ) {
369 3604356723 : return (MAP_(private_t) const *)(((ulong)join) - sizeof(MAP_(private_t)));
370 3604356723 : }
371 :
372 : /* map_private_{list,ele}[_const] returns the location in the caller's
373 : address space of the map lists and elements. The _const variants are
374 : for const correctness. Assumes map is valid. */
375 :
376 : FD_FN_PURE static inline ulong *
377 2589917679 : MAP_(private_list)( MAP_(private_t) * map ) {
378 2589917679 : return ((ulong *)map) - map->list_cnt;
379 2589917679 : }
380 :
381 : FD_FN_PURE static inline ulong const *
382 3751304844 : MAP_(private_list_const)( MAP_(private_t) const * map ) {
383 3751304844 : return ((ulong const *)map) - map->list_cnt;
384 3751304844 : }
385 :
386 : /* map_private_list_idx returns the index of the list (in [0,list_cnt)
387 : that will contain the element corresponding to key (if present at
388 : all) for a map with list_cnt elements and seed. Assumes list_cnt is
389 : an integer power-of 2. Assumes key points to a key is in the
390 : caller's address space that will not be changed concurrently.
391 : Retains no interest in key on return. */
392 :
393 : FD_FN_PURE static inline ulong
394 : MAP_(private_list_idx)( MAP_KEY_T const * key,
395 : ulong seed,
396 2550721767 : ulong list_cnt ) {
397 2550721767 : return (MAP_KEY_HASH( (key), (seed) )) & (list_cnt-1UL);
398 2550721767 : }
399 :
400 : /* map_private_box_next boxes idx and tag into a map next field value.
401 : Assumes idx is in [0,2^63-1==MAP_IDX_NULL] and tag is in [0,1].
402 :
403 : map_private_unbox_idx unboxes the idx from a map next field value.
404 : Return will be in [0,2^63-1==MAP_IDX_NULL].
405 :
406 : map_private_unbox_tag unboxes the tag from a map next field value.
407 : Return will be in [0,1] */
408 :
409 8720972376 : FD_FN_CONST static inline ulong MAP_(private_box_next) ( ulong idx, int tag ) { return idx | (((ulong)tag)<<63); }
410 46688435206 : FD_FN_CONST static inline ulong MAP_(private_unbox_idx) ( ulong next ) { return next & MAP_IDX_NULL; }
411 >11528*10^7 : FD_FN_CONST static inline int MAP_(private_unbox_tag) ( ulong next ) { return (int)(next>>63); }
412 :
413 : /* map_private_is_null returns 1 if idx is the NULL map idx value
414 : and 0 otherwise. */
415 :
416 46665319774 : FD_FN_CONST static inline int MAP_(private_is_null)( ulong idx ) { return idx==MAP_IDX_NULL; }
417 :
418 : FD_FN_PURE static inline ulong
419 44550222 : MAP_(key_max)( MAP_T const * join ) {
420 44550222 : if( FD_UNLIKELY( !join ) ) return 0UL;
421 44550222 : return MAP_(private_const)( join )->key_max;
422 44550222 : }
423 :
424 : FD_FN_PURE static inline ulong
425 44550222 : MAP_(seed)( MAP_T const * join ) {
426 44550222 : if( FD_UNLIKELY( !join ) ) return 0UL;
427 44550222 : return MAP_(private_const)( join )->seed;
428 44550222 : }
429 :
430 : FD_FN_PURE static inline ulong
431 79187064 : MAP_(key_cnt)( MAP_T const * join ) {
432 79187064 : if( FD_UNLIKELY( !join ) ) return 0UL;
433 79187064 : return MAP_(private_const)( join )->key_cnt;
434 79187064 : }
435 :
436 : FD_FN_PURE static inline int
437 98549985 : MAP_(is_full)( MAP_T const * join ) {
438 98549985 : if( FD_UNLIKELY( !join ) ) return 0UL;
439 98549985 : MAP_(private_t) const * map = MAP_(private_const)( join );
440 98549985 : return MAP_(private_is_null)( MAP_(private_unbox_idx)( map->free_stack ) );
441 98549985 : }
442 :
443 : FD_FN_PURE static inline int
444 : MAP_(key_eq)( MAP_KEY_T const * k0,
445 7045342497 : MAP_KEY_T const * k1 ) {
446 7045342497 : return !!(MAP_KEY_EQ( (k0), (k1) ));
447 7045342497 : }
448 :
449 : FD_FN_PURE static inline ulong
450 : MAP_(key_hash)( MAP_KEY_T const * key,
451 6000000 : ulong seed ) {
452 6000000 : return (MAP_KEY_HASH( (key), (seed) ));
453 6000000 : }
454 :
455 : static inline MAP_KEY_T *
456 : MAP_(key_copy)( MAP_KEY_T * kd,
457 20686716 : MAP_KEY_T const * ks ) {
458 20686716 : (MAP_KEY_COPY( (kd), (ks) ));
459 20686716 : return kd;
460 20686716 : }
461 :
462 : FD_FN_PURE static inline MAP_(iter_t)
463 118101798 : MAP_(iter_init)( MAP_T const * join ) {
464 118101798 : if( FD_UNLIKELY( !join ) ) return 0UL; /* Debatable */
465 118101798 : MAP_(private_t) const * map = MAP_(private_const)( join );
466 118101798 : ulong ele_rem = map->key_max;
467 70479385104 : for( ; ele_rem; ele_rem-- ) if( !MAP_(private_unbox_tag)( join[ ele_rem-1UL ].MAP_NEXT ) ) break;
468 118101798 : return ele_rem;
469 118101798 : }
470 :
471 : FD_FN_CONST static inline int
472 : MAP_(iter_done)( MAP_T const * join,
473 3851819049 : MAP_(iter_t) ele_rem ) {
474 3851819049 : (void)join;
475 3851819049 : return !ele_rem;
476 3851819049 : }
477 :
478 : FD_FN_PURE static inline MAP_(iter_t)
479 : MAP_(iter_next)( MAP_T const * join,
480 3733717251 : MAP_(iter_t) ele_rem ) {
481 7527887382 : for( ele_rem--; ele_rem; ele_rem-- ) if( !MAP_(private_unbox_tag)( join[ ele_rem-1UL ].MAP_NEXT ) ) break;
482 3733717251 : return ele_rem;
483 3733717251 : }
484 :
485 : FD_FN_CONST static inline MAP_T *
486 : MAP_(iter_ele)( MAP_T * join,
487 3733717251 : MAP_(iter_t) ele_rem ) {
488 3733717251 : return join + ele_rem - 1UL;
489 3733717251 : }
490 :
491 : FD_FN_CONST static inline MAP_T const *
492 : MAP_(iter_ele_const)( MAP_T const * join,
493 787968000 : MAP_(iter_t) ele_rem ) {
494 787968000 : return join + ele_rem - 1UL;
495 787968000 : }
496 :
497 : FD_PROTOTYPES_END
498 :
499 : #endif
500 :
501 : #if MAP_IMPL_STYLE==1 /* need prototypes */
502 :
503 : FD_PROTOTYPES_BEGIN
504 :
505 : FD_FN_CONST ulong MAP_(align) ( void );
506 : FD_FN_CONST ulong MAP_(footprint)( ulong key_max );
507 : void * MAP_(new) ( void * shmem, ulong key_max, ulong seed );
508 : MAP_T * MAP_(join) ( void * shmap );
509 : void * MAP_(leave) ( MAP_T * join );
510 : void * MAP_(delete) ( void * shmap );
511 :
512 : MAP_T *
513 : MAP_(insert)( MAP_T * join,
514 : MAP_KEY_T const * key );
515 :
516 : MAP_T *
517 : MAP_(remove)( MAP_T * join,
518 : MAP_KEY_T const * key );
519 :
520 : FD_FN_PURE MAP_T *
521 : MAP_(query)( MAP_T * join,
522 : MAP_KEY_T const * key,
523 : MAP_T * null );
524 :
525 : FD_FN_PURE MAP_T *
526 : MAP_(query2)( MAP_T * join,
527 : MAP_KEY_T const * key,
528 : MAP_T * null );
529 :
530 : FD_FN_PURE MAP_T const *
531 : MAP_(query_const)( MAP_T const * join,
532 : MAP_KEY_T const * key,
533 : MAP_T const * null );
534 :
535 : FD_FN_PURE MAP_T const *
536 : MAP_(query_safe)( MAP_T const * join,
537 : MAP_KEY_T const * key,
538 : MAP_T const * null );
539 :
540 : FD_FN_PURE int
541 : MAP_(verify)( MAP_T const * join );
542 :
543 : MAP_T *
544 : MAP_(pop_free_ele)( MAP_T * join );
545 :
546 : MAP_T *
547 : MAP_(push_free_ele)( MAP_T * join,
548 : MAP_T * ele );
549 :
550 : FD_PROTOTYPES_END
551 :
552 : #else /* need implementations */
553 :
554 : #if MAP_IMPL_STYLE==0 /* local only */
555 : #define MAP_IMPL_STATIC FD_FN_UNUSED static
556 : #else
557 : #define MAP_IMPL_STATIC
558 : #endif
559 :
560 : FD_FN_CONST MAP_IMPL_STATIC ulong
561 25134477 : MAP_(align)( void ) {
562 25134477 : return fd_ulong_max( 128UL, alignof(MAP_T) );
563 25134477 : }
564 :
565 : FD_FN_CONST MAP_IMPL_STATIC ulong
566 53143896 : MAP_(footprint)( ulong key_max ) {
567 53143896 : ulong align = MAP_(align)();
568 :
569 : /* memory layout is:
570 :
571 : 2 ulong | pad | list_cnt ulong | map_private_t | key_max map_t | pad
572 : <------ meta_footprint, align multiple ------>
573 : <------------------ footprint, align multiple -------------------->
574 :
575 : Noting that list_cnt is in [key_max/2,key_max], footprint is
576 : (conservatively) at most:
577 :
578 : 2*sizeof(ulong) + align-1 + key_max*sizeof(ulong) + sizeof(map_private_t) + key_max*sizeof(map_t) + align-1
579 :
580 : Requiring this to be at most ULONG_MAX (such that no calculations
581 : will overflow) yields the below. We also must have
582 : key_max<=2^63-1==MAP_IDX_NULL given how elements are marked. */
583 :
584 53143896 : ulong key_thresh = fd_ulong_min( (ULONG_MAX - 2UL*(align-1UL) - 2UL*sizeof(ulong) - sizeof(MAP_(private_t))) /
585 53143896 : (sizeof(MAP_T) + sizeof(ulong)), MAP_IDX_NULL );
586 53143896 : if( FD_UNLIKELY( key_max>key_thresh ) ) return 0UL;
587 53143884 : ulong list_cnt = MAP_(private_list_cnt) ( key_max );
588 53143884 : ulong meta_footprint = MAP_(private_meta_footprint)( list_cnt );
589 53143884 : return meta_footprint + fd_ulong_align_up( key_max*sizeof(MAP_T), align );
590 53143896 : }
591 :
592 : MAP_IMPL_STATIC void *
593 : MAP_(new)( void * shmem,
594 : ulong key_max,
595 1224846 : ulong seed ) {
596 1224846 : ulong * hdr = (ulong *)shmem;
597 :
598 1224846 : if( FD_UNLIKELY( !hdr ) ) {
599 6 : FD_LOG_WARNING(( "NULL shmem" ));
600 6 : return NULL;
601 6 : }
602 :
603 1224840 : if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)hdr, MAP_(align)() ) ) ) {
604 6 : FD_LOG_WARNING(( "misaligned shmem" ));
605 6 : return NULL;
606 6 : }
607 :
608 1224834 : ulong footprint = MAP_(footprint)( key_max );
609 1224834 : if( FD_UNLIKELY( !footprint ) ) {
610 6 : FD_LOG_WARNING(( "bad key_max" ));
611 6 : return NULL;
612 6 : }
613 :
614 : /* Init the metadata */
615 :
616 1224828 : ulong list_cnt = MAP_(private_list_cnt)( key_max );
617 1224828 : ulong meta_footprint = MAP_(private_meta_footprint)( list_cnt );
618 :
619 1224828 : MAP_T * join = (MAP_T *)( ((ulong)shmem) + meta_footprint );
620 1224828 : MAP_(private_t) * map = MAP_(private)( join );
621 :
622 1224828 : map->key_max = key_max;
623 1224828 : map->seed = seed;
624 1224828 : map->list_cnt = list_cnt;
625 1224828 : map->key_cnt = 0UL;
626 :
627 : /* Init the free stack */
628 :
629 1224828 : if( FD_UNLIKELY( !key_max ) ) map->free_stack = MAP_(private_box_next)( MAP_IDX_NULL, 1 );
630 1224816 : else {
631 1224816 : map->free_stack = MAP_(private_box_next)( 0UL, 1 );
632 4503020490 : for( ulong ele_idx=1UL; ele_idx<key_max; ele_idx++ ) join[ ele_idx-1UL ].MAP_NEXT = MAP_(private_box_next)( ele_idx, 1 );
633 1224816 : join[ key_max-1UL ].MAP_NEXT = MAP_(private_box_next)( MAP_IDX_NULL, 1 );
634 1224816 : }
635 :
636 : /* Set all the lists to null */
637 :
638 1224828 : ulong * list = MAP_(private_list)( map );
639 3555233676 : for( ulong list_idx=0UL; list_idx<list_cnt; list_idx++ ) list[ list_idx ] = MAP_(private_box_next)( MAP_IDX_NULL, 0 );
640 :
641 1224828 : hdr[1] = meta_footprint;
642 :
643 1224828 : FD_COMPILER_MFENCE();
644 1224828 : FD_VOLATILE( hdr[0] ) = MAP_MAGIC;
645 1224828 : FD_COMPILER_MFENCE();
646 :
647 1224828 : return hdr;
648 1224834 : }
649 :
650 : MAP_IMPL_STATIC MAP_T *
651 1224846 : MAP_(join)( void * shmap ) {
652 1224846 : ulong * hdr = (ulong *)shmap;
653 :
654 1224846 : if( FD_UNLIKELY( !hdr ) ) {
655 6 : FD_LOG_WARNING(( "NULL shmap" ));
656 6 : return NULL;
657 6 : }
658 :
659 1224840 : if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)hdr, MAP_(align)() ) ) ) {
660 6 : FD_LOG_WARNING(( "misaligned shmap" ));
661 6 : return NULL;
662 6 : }
663 :
664 1224834 : if( FD_UNLIKELY( hdr[0]!=MAP_MAGIC ) ) {
665 6 : FD_LOG_WARNING(( "bad magic" ));
666 6 : return NULL;
667 6 : }
668 :
669 1224828 : return (MAP_T *)(((ulong)shmap) + hdr[1]);
670 1224834 : }
671 :
672 : MAP_IMPL_STATIC void *
673 1224828 : MAP_(leave)( MAP_T * join ) {
674 :
675 1224828 : if( FD_UNLIKELY( !join ) ) {
676 6 : FD_LOG_WARNING(( "NULL join" ));
677 6 : return NULL;
678 6 : }
679 :
680 1224822 : return (void *)(((ulong)join) - MAP_(private_meta_footprint)( MAP_(private)( join )->list_cnt ));
681 1224828 : }
682 :
683 : MAP_IMPL_STATIC void *
684 1224840 : MAP_(delete)( void * shmap ) {
685 1224840 : ulong * hdr = (ulong *)shmap;
686 :
687 1224840 : if( FD_UNLIKELY( !hdr ) ) {
688 6 : FD_LOG_WARNING(( "NULL shmap" ));
689 6 : return NULL;
690 6 : }
691 :
692 1224834 : if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)hdr, MAP_(align)() ) ) ) {
693 6 : FD_LOG_WARNING(( "misaligned shmap" ));
694 6 : return NULL;
695 6 : }
696 :
697 1224828 : if( FD_UNLIKELY( hdr[0]!=MAP_MAGIC ) ) {
698 6 : FD_LOG_WARNING(( "bad magic" ));
699 6 : return NULL;
700 6 : }
701 :
702 1224822 : FD_COMPILER_MFENCE();
703 1224822 : FD_VOLATILE( hdr[0] ) = 0UL;
704 1224822 : FD_COMPILER_MFENCE();
705 :
706 1224822 : return shmap;
707 1224828 : }
708 :
709 : static inline long
710 : MAP_(verify_key)( MAP_T * join,
711 : MAP_KEY_T const * key,
712 0 : ulong cnt ) {
713 0 : # define MAP_TEST(c) do { \
714 0 : if( FD_UNLIKELY( !(c) ) ) { FD_LOG_WARNING(( "FAIL: %s", #c )); return -1; } \
715 0 : } while(0)
716 0 :
717 0 : MAP_(private_t) * map = MAP_(private)( join );
718 0 : ulong list_idx = MAP_(private_list_idx)( key, map->seed, map->list_cnt );
719 0 : ulong key_max = map->key_max;
720 0 : ulong key_cnt = map->key_cnt;
721 0 : ulong ele_idx = MAP_(private_list)( map )[ MAP_(private_list_idx)( key, map->seed, map->list_cnt ) ];
722 0 : while( !MAP_(private_is_null)( ele_idx ) ) {
723 0 : MAP_TEST( cnt<key_cnt );
724 0 : MAP_TEST( ele_idx<key_max );
725 0 : MAP_T const * ele = join + ele_idx;
726 0 : MAP_TEST( MAP_(private_list_idx)( &ele->MAP_KEY, map->seed, map->list_cnt )==list_idx );
727 0 : cnt++;
728 0 : ele_idx = MAP_(private_unbox_idx)( ele->MAP_NEXT );
729 0 : MAP_TEST( !MAP_(private_unbox_tag)( ele->MAP_NEXT ) ); /* Element marked as used */
730 0 : }
731 0 :
732 0 : # undef MAP_TEST
733 0 : return (long)cnt;;
734 0 : }
735 :
736 : MAP_IMPL_STATIC MAP_T *
737 : MAP_(insert)( MAP_T * join,
738 14686716 : MAP_KEY_T const * key ) {
739 14686716 : MAP_(private_t) * map = MAP_(private)( join );
740 :
741 : /* Pop the free stack to allocate an element (this is guaranteed to
742 : succeed as per contract) */
743 :
744 14686716 : ulong ele_idx = MAP_(private_unbox_idx)( map->free_stack );
745 14686716 : MAP_T * ele = join + ele_idx;
746 14686716 : map->free_stack = ele->MAP_NEXT; /* already tagged free */
747 14686716 : map->key_cnt++; /* Consider eliminating this to help make completely concurrent lockfree? */
748 :
749 : /* ... and map the newly allocated element to key (this is also
750 : guaranteed to not have collisions as per contract). Note that
751 : elements appear in the chain in order of newest to oldest. This
752 : property is NECESSARY for an important optimization in
753 : fd_funk_rec_query_global. */
754 :
755 14686716 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
756 14686716 : ulong * head = MAP_(private_list)( map ) + ( hash & (map->list_cnt-1UL) );
757 14686716 : MAP_(key_copy)( &ele->MAP_KEY, key );
758 14686716 : ele->MAP_NEXT = MAP_(private_box_next)( MAP_(private_unbox_idx)( *head ), 0 );
759 : #if MAP_MEMOIZE
760 13150716 : ele->MAP_HASH = hash;
761 : #endif
762 14686716 : *head = MAP_(private_box_next)( ele_idx, 0 );
763 :
764 14686716 : return ele;
765 14686716 : }
766 :
767 : MAP_IMPL_STATIC MAP_T *
768 0 : MAP_(pop_free_ele)( MAP_T * join ) {
769 0 : MAP_(private_t) * map = MAP_(private)( join );
770 :
771 : /* Pop the free stack to allocate an element (this is guaranteed to
772 : succeed as per contract) */
773 :
774 0 : ulong ele_idx = MAP_(private_unbox_idx)( map->free_stack );
775 0 : MAP_T * ele = join + ele_idx;
776 0 : map->free_stack = ele->MAP_NEXT; /* already tagged free */
777 :
778 0 : return ele;
779 0 : }
780 :
781 : MAP_IMPL_STATIC MAP_T *
782 : MAP_(push_free_ele)( MAP_T * join,
783 0 : MAP_T * ele ) {
784 0 : MAP_(private_t) * map = MAP_(private)( join );
785 :
786 0 : ulong ele_idx = (ulong)(ele - join);
787 :
788 0 : ele->MAP_NEXT = map->free_stack; /* already tagged free */
789 0 : map->free_stack = MAP_(private_box_next)( ele_idx, 1 );
790 :
791 0 : return ele;
792 0 : }
793 :
794 : MAP_IMPL_STATIC MAP_T *
795 : MAP_(insert_free_ele)( MAP_T * join,
796 : MAP_T * ele,
797 0 : MAP_KEY_T const * key ) {
798 0 : MAP_(private_t) * map = MAP_(private)( join );
799 : /* Map the allocated element to key (this is also
800 : guaranteed to not have collisions as per contract). */
801 :
802 0 : ulong ele_idx = (ulong)(ele - join);
803 :
804 0 : ulong * head = MAP_(private_list)( map ) + MAP_(private_list_idx)( key, map->seed, map->list_cnt );
805 0 : MAP_(key_copy)( &ele->MAP_KEY, key );
806 0 : ele->MAP_NEXT = MAP_(private_box_next)( MAP_(private_unbox_idx)( *head ), 0 );
807 0 : *head = MAP_(private_box_next)( ele_idx, 0 );
808 :
809 0 : return ele;
810 0 : }
811 :
812 : MAP_IMPL_STATIC MAP_T *
813 : MAP_(remove)( MAP_T * join,
814 16314603 : MAP_KEY_T const * key ) {
815 16314603 : MAP_(private_t) * map = MAP_(private)( join );
816 :
817 :
818 : /* Find the key */
819 :
820 16314603 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
821 16314603 : ulong * head = MAP_(private_list)( map ) + ( hash & (map->list_cnt-1UL) );
822 16314603 : ulong * cur = head;
823 20967942 : for(;;) {
824 20967942 : ulong ele_idx = MAP_(private_unbox_idx)( *cur );
825 20967942 : if( FD_UNLIKELY( MAP_(private_is_null)( ele_idx ) ) ) break;
826 17895942 : MAP_T * ele = join + ele_idx;
827 17895942 : if(
828 : #if MAP_MEMOIZE
829 14766942 : hash == ele->MAP_HASH &&
830 : #endif
831 15936723 : FD_LIKELY( MAP_(key_eq)( key, &ele->MAP_KEY ) ) ) { /* Optimize for found (it is remove after all) */
832 : /* Found, remove the mapping and push it to free stack. */
833 13242603 : *cur = ele->MAP_NEXT; /* already tagged empty */
834 13242603 : ele->MAP_NEXT = map->free_stack; /* already tagged free */
835 13242603 : map->free_stack = MAP_(private_box_next)( ele_idx, 1 );
836 13242603 : map->key_cnt--;
837 13242603 : return ele;
838 13242603 : }
839 4653339 : cur = &ele->MAP_NEXT; /* Retain the pointer to next so we can rewrite it later. */
840 4653339 : }
841 :
842 : /* Not found */
843 :
844 3072000 : return NULL;
845 16314603 : }
846 :
847 : FD_FN_PURE MAP_IMPL_STATIC MAP_T *
848 : MAP_(query)( MAP_T * join,
849 : MAP_KEY_T const * key,
850 2107871076 : MAP_T * sentinel ) {
851 2107871076 : MAP_(private_t) * map = MAP_(private)( join );
852 :
853 :
854 : /* Find the key */
855 :
856 2107871076 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
857 2107871076 : ulong * head = MAP_(private_list)( map ) + ( hash & (map->list_cnt-1UL) );
858 2107871076 : ulong * cur = head;
859 3866930360 : for(;;) {
860 3866930360 : ulong ele_idx = MAP_(private_unbox_idx)( *cur );
861 3866930360 : if( FD_UNLIKELY( MAP_(private_is_null)( ele_idx ) ) ) break; /* optimize for found */
862 2846494787 : MAP_T * ele = join + ele_idx;
863 2846494787 : if(
864 : #if MAP_MEMOIZE
865 1635354647 : hash == ele->MAP_HASH &&
866 : #endif
867 1914769911 : FD_LIKELY( MAP_(key_eq)( key, &ele->MAP_KEY ) ) ) { /* Optimize for found (it is remove after all) */
868 : /* Found, move it to the front of the chain. (FIXME: BRANCH PROB? DO BRANCHLESS?) */
869 1087435503 : if( FD_UNLIKELY( cur!=head ) ) { /* Assume already at head from previous query */
870 620102175 : *cur = ele->MAP_NEXT; /* Already tagged free */
871 620102175 : ele->MAP_NEXT = *head; /* Already tagged free */
872 620102175 : *head = MAP_(private_box_next)( ele_idx, 0 );
873 620102175 : }
874 1087435503 : return ele;
875 1087435503 : }
876 1759059284 : cur = &ele->MAP_NEXT; /* Retain the pointer to next so we can rewrite it later. */
877 1759059284 : }
878 :
879 : /* Not found */
880 :
881 1020435573 : return sentinel;
882 2107871076 : }
883 :
884 : FD_FN_PURE MAP_IMPL_STATIC MAP_T *
885 : MAP_(query2)( MAP_T * join,
886 : MAP_KEY_T const * key,
887 0 : MAP_T * sentinel ) {
888 0 : MAP_(private_t) * map = MAP_(private)( join );
889 :
890 :
891 : /* Find the key */
892 :
893 0 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
894 0 : ulong * head = MAP_(private_list)( map ) + ( hash & (map->list_cnt-1UL) );
895 0 : ulong * cur = head;
896 0 : for(;;) {
897 0 : ulong ele_idx = MAP_(private_unbox_idx)( *cur );
898 0 : if( FD_UNLIKELY( MAP_(private_is_null)( ele_idx ) ) ) break; /* optimize for found */
899 0 : MAP_T * ele = join + ele_idx;
900 0 : if(
901 0 : #if MAP_MEMOIZE
902 0 : hash == ele->MAP_HASH &&
903 0 : #endif
904 0 : FD_LIKELY( MAP_(key_eq)( key, &ele->MAP_KEY ) ) ) { /* Optimize for found (it is remove after all) */
905 0 : return ele; /* optimize for found */
906 0 : }
907 0 : cur = &ele->MAP_NEXT;
908 0 : }
909 :
910 : /* Not found */
911 :
912 0 : return sentinel;
913 0 : }
914 :
915 : FD_FN_PURE MAP_IMPL_STATIC MAP_T const *
916 : MAP_(query_const)( MAP_T const * join,
917 : MAP_KEY_T const * key,
918 3700610622 : MAP_T const * sentinel ) {
919 3700610622 : MAP_(private_t) const * map = MAP_(private_const)( join );
920 :
921 : /* Find the key */
922 :
923 3700610622 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
924 3700610622 : ulong const * head = MAP_(private_list_const)( map ) + ( hash & (map->list_cnt-1UL) );
925 3700610622 : ulong const * cur = head;
926 7013583095 : for(;;) {
927 7013583095 : ulong ele_idx = MAP_(private_unbox_idx)( *cur );
928 7013583095 : if( FD_UNLIKELY( MAP_(private_is_null)( ele_idx ) ) ) break; /* optimize for found */
929 6999146042 : MAP_T const * ele = join + ele_idx;
930 6999146042 : if(
931 : #if MAP_MEMOIZE
932 5402177933 : hash == ele->MAP_HASH &&
933 : #endif
934 5402177933 : FD_LIKELY( MAP_(key_eq)( key, &ele->MAP_KEY ) ) ) { /* Optimize for found (it is remove after all) */
935 3686173569 : return ele; /* optimize for found */
936 3686173569 : }
937 3312972473 : cur = &ele->MAP_NEXT;
938 3312972473 : }
939 :
940 : /* Not found */
941 :
942 14437053 : return sentinel;
943 3700610622 : }
944 :
945 : FD_FN_PURE MAP_IMPL_STATIC MAP_T const *
946 : MAP_(query_safe)( MAP_T const * join,
947 : MAP_KEY_T const * key,
948 0 : MAP_T const * sentinel ) {
949 0 : MAP_(private_t) const * map = MAP_(private_const)( join );
950 :
951 0 : ulong key_max = map->key_max;
952 :
953 : /* Find the key */
954 :
955 0 : ulong hash = MAP_KEY_HASH( (key), (map->seed) );
956 0 : ulong const * head = MAP_(private_list_const)( map ) + ( hash & (map->list_cnt-1UL) );
957 0 : ulong const * cur = head;
958 0 : for( ulong i = 0; i < key_max; ++i ) {
959 0 : ulong ele_idx = MAP_(private_unbox_idx)( *cur );
960 0 : if( FD_UNLIKELY( MAP_(private_is_null)( ele_idx ) || ele_idx >= key_max ) ) break; /* optimize for found */
961 0 : MAP_T const * ele = join + ele_idx;
962 0 : if(
963 : #if MAP_MEMOIZE
964 0 : hash == ele->MAP_HASH &&
965 : #endif
966 0 : FD_LIKELY( MAP_(key_eq)( key, &ele->MAP_KEY ) ) ) { /* Optimize for found (it is remove after all) */
967 0 : return ele; /* optimize for found */
968 0 : }
969 0 : cur = &ele->MAP_NEXT;
970 0 : }
971 :
972 : /* Not found */
973 :
974 0 : return sentinel;
975 0 : }
976 :
977 : FD_FN_PURE MAP_IMPL_STATIC int
978 50694222 : MAP_(verify)( MAP_T const * join ) {
979 :
980 87339367401 : # define MAP_TEST(c) do { \
981 87339367401 : if( FD_UNLIKELY( !(c) ) ) { FD_LOG_WARNING(( "FAIL: %s", #c )); return -1; } \
982 87339367401 : } while(0)
983 :
984 50694222 : MAP_TEST( join );
985 :
986 50694222 : MAP_(private_t) const * map = MAP_(private_const)( join ); MAP_TEST( map!=NULL );
987 :
988 50694222 : MAP_TEST( MAP_(footprint)( map->key_max ) );
989 :
990 : /* seed can be anything as far as map is concerned */
991 :
992 50694222 : MAP_TEST( map->list_cnt==MAP_(private_list_cnt)( map->key_max ) );
993 50694222 : MAP_TEST( map->key_cnt <=map->key_max );
994 :
995 50694222 : ulong key_max = map->key_max;
996 50694222 : ulong key_cnt = map->key_cnt;
997 50694222 : ulong seed = map->seed;
998 50694222 : ulong list_cnt = map->list_cnt;
999 50694222 : ulong const * list = MAP_(private_list_const)( map ); MAP_TEST( list!=NULL );
1000 :
1001 50694222 : ulong free_cnt = key_max - key_cnt;
1002 :
1003 50694222 : ulong ele_idx;
1004 50694222 : ulong cnt;
1005 :
1006 50694222 : cnt = 0UL;
1007 11607905358 : for( ulong list_idx=0UL; list_idx<list_cnt; list_idx++ ) {
1008 11557211136 : ele_idx = MAP_(private_unbox_idx)( list[ list_idx ] );
1009 11557211136 : MAP_TEST( !MAP_(private_unbox_tag)( list[ list_idx ] ) ); /* Head marked as used */
1010 14107932903 : while( !MAP_(private_is_null)( ele_idx ) ) {
1011 2550721767 : MAP_TEST( cnt<key_cnt );
1012 2550721767 : MAP_TEST( ele_idx<key_max );
1013 2550721767 : MAP_T const * ele = join + ele_idx;
1014 : #if MAP_MEMOIZE
1015 1764289767 : MAP_TEST( ele->MAP_HASH == MAP_KEY_HASH( (&ele->MAP_KEY), (map->seed) ) );
1016 1764289767 : #endif
1017 2550721767 : MAP_TEST( MAP_(private_list_idx)( &ele->MAP_KEY, seed, list_cnt )==list_idx );
1018 2550721767 : cnt++;
1019 2550721767 : ele_idx = MAP_(private_unbox_idx)( ele->MAP_NEXT );
1020 2550721767 : MAP_TEST( !MAP_(private_unbox_tag)( ele->MAP_NEXT ) ); /* Element marked as used */
1021 2550721767 : }
1022 11557211136 : }
1023 :
1024 50694222 : MAP_TEST( cnt==key_cnt );
1025 :
1026 50694222 : cnt = 0UL;
1027 :
1028 50694222 : ele_idx = MAP_(private_unbox_idx)( map->free_stack );
1029 50694222 : MAP_TEST( MAP_(private_unbox_tag)( map->free_stack ) ); /* Head marked as free */
1030 20614394727 : while( !MAP_(private_is_null)( ele_idx ) ) {
1031 20563700505 : MAP_TEST( cnt<free_cnt );
1032 20563700505 : MAP_TEST( ele_idx<key_max );
1033 20563700505 : MAP_T const * ele = join + ele_idx;
1034 20563700505 : cnt++;
1035 20563700505 : ele_idx = MAP_(private_unbox_idx)( ele->MAP_NEXT );
1036 20563700505 : MAP_TEST( MAP_(private_unbox_tag)( ele->MAP_NEXT ) ); /* Element marked as free */
1037 20563700505 : }
1038 :
1039 50694222 : MAP_TEST( cnt==free_cnt );
1040 :
1041 2601415989 : for( ulong ele_idx=0UL; ele_idx<key_cnt; ele_idx++ ) {
1042 2550721767 : if( MAP_(private_unbox_tag)( join[ele_idx].MAP_NEXT ) ) continue;
1043 1667629917 : MAP_TEST( MAP_(query_const)( join, &join[ele_idx].MAP_KEY, NULL )==&join[ele_idx] );
1044 1667629917 : }
1045 :
1046 50694222 : # undef MAP_TEST
1047 :
1048 50694222 : return 0;
1049 50694222 : }
1050 :
1051 : #undef MAP_IMPL_STATIC
1052 :
1053 : #endif
1054 :
1055 : #undef MAP_
1056 : #undef MAP_IDX_NULL
1057 :
1058 : #undef MAP_IMPL_STYLE
1059 : #undef MAP_MAGIC
1060 : #undef MAP_KEY_COPY
1061 : #undef MAP_KEY_HASH
1062 : #undef MAP_KEY_EQ
1063 : #undef MAP_NEXT
1064 : #undef MAP_KEY
1065 : #undef MAP_KEY_T
1066 : #undef MAP_T
1067 : #undef MAP_NAME
1068 : #undef MAP_MEMOIZE
1069 : #undef MAP_HASH
|