ctf-hash.c revision 1.1.1.3 1 /* Interface to hashtable implementations.
2 Copyright (C) 2006-2024 Free Software Foundation, Inc.
3
4 This file is part of libctf.
5
6 libctf is free software; you can redistribute it and/or modify it under
7 the terms of the GNU General Public License as published by the Free
8 Software Foundation; either version 3, or (at your option) any later
9 version.
10
11 This program is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
14 See the GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with this program; see the file COPYING. If not see
18 <http://www.gnu.org/licenses/>. */
19
20 #include <ctf-impl.h>
21 #include <string.h>
22 #include "libiberty.h"
23 #include "hashtab.h"
24
25 /* We have two hashtable implementations:
26
27 - ctf_dynhash_* is an interface to a dynamically-expanding hash with
28 unknown size that should support addition of large numbers of items,
29 and removal as well, and is used only at type-insertion time and during
30 linking. It can be constructed with an expected initial number of
31 elements, but need not be.
32
33 - ctf_dynset_* is an interface to a dynamically-expanding hash that contains
34 only keys: no values.
35
36 These can be implemented by the same underlying hashmap if you wish. */
37
38 /* The helem is used for general key/value mappings in the ctf_dynhash: the
39 owner may not have space allocated for it, and will be garbage (not
40 NULL!) in that case. */
41
42 typedef struct ctf_helem
43 {
44 void *key; /* Either a pointer, or a coerced ctf_id_t. */
45 void *value; /* The value (possibly a coerced int). */
46 ctf_dynhash_t *owner; /* The hash that owns us. */
47 } ctf_helem_t;
48
49 /* Equally, the key_free and value_free may not exist. */
50
51 struct ctf_dynhash
52 {
53 struct htab *htab;
54 ctf_hash_free_fun key_free;
55 ctf_hash_free_fun value_free;
56 };
57
58 /* Hash and eq functions for the dynhash and hash. */
59
60 unsigned int
61 ctf_hash_integer (const void *ptr)
62 {
63 ctf_helem_t *hep = (ctf_helem_t *) ptr;
64
65 return htab_hash_pointer (hep->key);
66 }
67
68 int
69 ctf_hash_eq_integer (const void *a, const void *b)
70 {
71 ctf_helem_t *hep_a = (ctf_helem_t *) a;
72 ctf_helem_t *hep_b = (ctf_helem_t *) b;
73
74 return htab_eq_pointer (hep_a->key, hep_b->key);
75 }
76
77 unsigned int
78 ctf_hash_string (const void *ptr)
79 {
80 ctf_helem_t *hep = (ctf_helem_t *) ptr;
81
82 return htab_hash_string (hep->key);
83 }
84
85 int
86 ctf_hash_eq_string (const void *a, const void *b)
87 {
88 ctf_helem_t *hep_a = (ctf_helem_t *) a;
89 ctf_helem_t *hep_b = (ctf_helem_t *) b;
90
91 return !strcmp((const char *) hep_a->key, (const char *) hep_b->key);
92 }
93
94 /* Hash a type_key. */
95 unsigned int
96 ctf_hash_type_key (const void *ptr)
97 {
98 ctf_helem_t *hep = (ctf_helem_t *) ptr;
99 ctf_link_type_key_t *k = (ctf_link_type_key_t *) hep->key;
100
101 return htab_hash_pointer (k->cltk_fp) + 59
102 * htab_hash_pointer ((void *) (uintptr_t) k->cltk_idx);
103 }
104
105 int
106 ctf_hash_eq_type_key (const void *a, const void *b)
107 {
108 ctf_helem_t *hep_a = (ctf_helem_t *) a;
109 ctf_helem_t *hep_b = (ctf_helem_t *) b;
110 ctf_link_type_key_t *key_a = (ctf_link_type_key_t *) hep_a->key;
111 ctf_link_type_key_t *key_b = (ctf_link_type_key_t *) hep_b->key;
112
113 return (key_a->cltk_fp == key_b->cltk_fp)
114 && (key_a->cltk_idx == key_b->cltk_idx);
115 }
116
117 /* Hash a type_id_key. */
118 unsigned int
119 ctf_hash_type_id_key (const void *ptr)
120 {
121 ctf_helem_t *hep = (ctf_helem_t *) ptr;
122 ctf_type_id_key_t *k = (ctf_type_id_key_t *) hep->key;
123
124 return htab_hash_pointer ((void *) (uintptr_t) k->ctii_input_num)
125 + 59 * htab_hash_pointer ((void *) (uintptr_t) k->ctii_type);
126 }
127
128 int
129 ctf_hash_eq_type_id_key (const void *a, const void *b)
130 {
131 ctf_helem_t *hep_a = (ctf_helem_t *) a;
132 ctf_helem_t *hep_b = (ctf_helem_t *) b;
133 ctf_type_id_key_t *key_a = (ctf_type_id_key_t *) hep_a->key;
134 ctf_type_id_key_t *key_b = (ctf_type_id_key_t *) hep_b->key;
135
136 return (key_a->ctii_input_num == key_b->ctii_input_num)
137 && (key_a->ctii_type == key_b->ctii_type);
138 }
139
140 /* The dynhash, used for hashes whose size is not known at creation time. */
141
142 /* Free a single ctf_helem with arbitrary key/value functions. */
143
144 static void
145 ctf_dynhash_item_free (void *item)
146 {
147 ctf_helem_t *helem = item;
148
149 if (helem->owner->key_free && helem->key)
150 helem->owner->key_free (helem->key);
151 if (helem->owner->value_free && helem->value)
152 helem->owner->value_free (helem->value);
153 free (helem);
154 }
155
156 ctf_dynhash_t *
157 ctf_dynhash_create_sized (unsigned long nelems, ctf_hash_fun hash_fun,
158 ctf_hash_eq_fun eq_fun, ctf_hash_free_fun key_free,
159 ctf_hash_free_fun value_free)
160 {
161 ctf_dynhash_t *dynhash;
162 htab_del del = ctf_dynhash_item_free;
163
164 if (key_free || value_free)
165 dynhash = malloc (sizeof (ctf_dynhash_t));
166 else
167 dynhash = malloc (offsetof (ctf_dynhash_t, key_free));
168 if (!dynhash)
169 return NULL;
170
171 if (key_free == NULL && value_free == NULL)
172 del = free;
173
174 if ((dynhash->htab = htab_create_alloc (nelems, (htab_hash) hash_fun, eq_fun,
175 del, xcalloc, free)) == NULL)
176 {
177 free (dynhash);
178 return NULL;
179 }
180
181 if (key_free || value_free)
182 {
183 dynhash->key_free = key_free;
184 dynhash->value_free = value_free;
185 }
186
187 return dynhash;
188 }
189
190 ctf_dynhash_t *
191 ctf_dynhash_create (ctf_hash_fun hash_fun, ctf_hash_eq_fun eq_fun,
192 ctf_hash_free_fun key_free, ctf_hash_free_fun value_free)
193 {
194 /* 7 is arbitrary and not benchmarked yet. */
195
196 return ctf_dynhash_create_sized (7, hash_fun, eq_fun, key_free, value_free);
197 }
198
199 static ctf_helem_t **
200 ctf_hashtab_lookup (struct htab *htab, const void *key, enum insert_option insert)
201 {
202 ctf_helem_t tmp = { .key = (void *) key };
203 return (ctf_helem_t **) htab_find_slot (htab, &tmp, insert);
204 }
205
206 static ctf_helem_t *
207 ctf_hashtab_insert (struct htab *htab, void *key, void *value,
208 ctf_hash_free_fun key_free,
209 ctf_hash_free_fun value_free)
210 {
211 ctf_helem_t **slot;
212
213 slot = ctf_hashtab_lookup (htab, key, INSERT);
214
215 if (!slot)
216 {
217 errno = ENOMEM;
218 return NULL;
219 }
220
221 if (!*slot)
222 {
223 /* Only spend space on the owner if we're going to use it: if there is a
224 key or value freeing function. */
225 if (key_free || value_free)
226 *slot = malloc (sizeof (ctf_helem_t));
227 else
228 *slot = malloc (offsetof (ctf_helem_t, owner));
229 if (!*slot)
230 return NULL;
231 (*slot)->key = key;
232 }
233 else
234 {
235 if (key_free)
236 key_free (key);
237 if (value_free)
238 value_free ((*slot)->value);
239 }
240 (*slot)->value = value;
241 return *slot;
242 }
243
244 int
245 ctf_dynhash_insert (ctf_dynhash_t *hp, void *key, void *value)
246 {
247 ctf_helem_t *slot;
248 ctf_hash_free_fun key_free = NULL, value_free = NULL;
249
250 if (hp->htab->del_f == ctf_dynhash_item_free)
251 {
252 key_free = hp->key_free;
253 value_free = hp->value_free;
254 }
255 slot = ctf_hashtab_insert (hp->htab, key, value,
256 key_free, value_free);
257
258 if (!slot)
259 return errno;
260
261 /* Keep track of the owner, so that the del function can get at the key_free
262 and value_free functions. Only do this if one of those functions is set:
263 if not, the owner is not even present in the helem. */
264
265 if (key_free || value_free)
266 slot->owner = hp;
267
268 return 0;
269 }
270
271 void
272 ctf_dynhash_remove (ctf_dynhash_t *hp, const void *key)
273 {
274 ctf_helem_t hep = { (void *) key, NULL, NULL };
275 htab_remove_elt (hp->htab, &hep);
276 }
277
278 void
279 ctf_dynhash_empty (ctf_dynhash_t *hp)
280 {
281 htab_empty (hp->htab);
282 }
283
284 size_t
285 ctf_dynhash_elements (ctf_dynhash_t *hp)
286 {
287 return htab_elements (hp->htab);
288 }
289
290 void *
291 ctf_dynhash_lookup (ctf_dynhash_t *hp, const void *key)
292 {
293 ctf_helem_t **slot;
294
295 slot = ctf_hashtab_lookup (hp->htab, key, NO_INSERT);
296
297 if (slot)
298 return (*slot)->value;
299
300 return NULL;
301 }
302
303 /* TRUE/FALSE return. */
304 int
305 ctf_dynhash_lookup_kv (ctf_dynhash_t *hp, const void *key,
306 const void **orig_key, void **value)
307 {
308 ctf_helem_t **slot;
309
310 slot = ctf_hashtab_lookup (hp->htab, key, NO_INSERT);
311
312 if (slot)
313 {
314 if (orig_key)
315 *orig_key = (*slot)->key;
316 if (value)
317 *value = (*slot)->value;
318 return 1;
319 }
320 return 0;
321 }
322
323 typedef struct ctf_traverse_cb_arg
324 {
325 ctf_hash_iter_f fun;
326 void *arg;
327 } ctf_traverse_cb_arg_t;
328
329 static int
330 ctf_hashtab_traverse (void **slot, void *arg_)
331 {
332 ctf_helem_t *helem = *((ctf_helem_t **) slot);
333 ctf_traverse_cb_arg_t *arg = (ctf_traverse_cb_arg_t *) arg_;
334
335 arg->fun (helem->key, helem->value, arg->arg);
336 return 1;
337 }
338
339 void
340 ctf_dynhash_iter (ctf_dynhash_t *hp, ctf_hash_iter_f fun, void *arg_)
341 {
342 ctf_traverse_cb_arg_t arg = { fun, arg_ };
343 htab_traverse (hp->htab, ctf_hashtab_traverse, &arg);
344 }
345
346 typedef struct ctf_traverse_find_cb_arg
347 {
348 ctf_hash_iter_find_f fun;
349 void *arg;
350 void *found_key;
351 } ctf_traverse_find_cb_arg_t;
352
353 static int
354 ctf_hashtab_traverse_find (void **slot, void *arg_)
355 {
356 ctf_helem_t *helem = *((ctf_helem_t **) slot);
357 ctf_traverse_find_cb_arg_t *arg = (ctf_traverse_find_cb_arg_t *) arg_;
358
359 if (arg->fun (helem->key, helem->value, arg->arg))
360 {
361 arg->found_key = helem->key;
362 return 0;
363 }
364 return 1;
365 }
366
367 void *
368 ctf_dynhash_iter_find (ctf_dynhash_t *hp, ctf_hash_iter_find_f fun, void *arg_)
369 {
370 ctf_traverse_find_cb_arg_t arg = { fun, arg_, NULL };
371 htab_traverse (hp->htab, ctf_hashtab_traverse_find, &arg);
372 return arg.found_key;
373 }
374
375 typedef struct ctf_traverse_remove_cb_arg
376 {
377 struct htab *htab;
378 ctf_hash_iter_remove_f fun;
379 void *arg;
380 } ctf_traverse_remove_cb_arg_t;
381
382 static int
383 ctf_hashtab_traverse_remove (void **slot, void *arg_)
384 {
385 ctf_helem_t *helem = *((ctf_helem_t **) slot);
386 ctf_traverse_remove_cb_arg_t *arg = (ctf_traverse_remove_cb_arg_t *) arg_;
387
388 if (arg->fun (helem->key, helem->value, arg->arg))
389 htab_clear_slot (arg->htab, slot);
390 return 1;
391 }
392
393 void
394 ctf_dynhash_iter_remove (ctf_dynhash_t *hp, ctf_hash_iter_remove_f fun,
395 void *arg_)
396 {
397 ctf_traverse_remove_cb_arg_t arg = { hp->htab, fun, arg_ };
398 htab_traverse (hp->htab, ctf_hashtab_traverse_remove, &arg);
399 }
400
401 /* Traverse a dynhash in arbitrary order, in _next iterator form.
402
403 Mutating the dynhash while iterating is not supported (just as it isn't for
404 htab_traverse).
405
406 Note: unusually, this returns zero on success and a *positive* value on
407 error, because it does not take an fp, taking an error pointer would be
408 incredibly clunky, and nearly all error-handling ends up stuffing the result
409 of this into some sort of errno or ctf_errno, which is invariably
410 positive. So doing this simplifies essentially all callers. */
411 int
412 ctf_dynhash_next (ctf_dynhash_t *h, ctf_next_t **it, void **key, void **value)
413 {
414 ctf_next_t *i = *it;
415 ctf_helem_t *slot;
416
417 if (!i)
418 {
419 size_t size = htab_size (h->htab);
420
421 /* If the table has too many entries to fit in an ssize_t, just give up.
422 This might be spurious, but if any type-related hashtable has ever been
423 nearly as large as that then something very odd is going on. */
424 if (((ssize_t) size) < 0)
425 return EDOM;
426
427 if ((i = ctf_next_create ()) == NULL)
428 return ENOMEM;
429
430 i->u.ctn_hash_slot = h->htab->entries;
431 i->cu.ctn_h = h;
432 i->ctn_n = 0;
433 i->ctn_size = (ssize_t) size;
434 i->ctn_iter_fun = (void (*) (void)) ctf_dynhash_next;
435 *it = i;
436 }
437
438 if ((void (*) (void)) ctf_dynhash_next != i->ctn_iter_fun)
439 return ECTF_NEXT_WRONGFUN;
440
441 if (h != i->cu.ctn_h)
442 return ECTF_NEXT_WRONGFP;
443
444 if ((ssize_t) i->ctn_n == i->ctn_size)
445 goto hash_end;
446
447 while ((ssize_t) i->ctn_n < i->ctn_size
448 && (*i->u.ctn_hash_slot == HTAB_EMPTY_ENTRY
449 || *i->u.ctn_hash_slot == HTAB_DELETED_ENTRY))
450 {
451 i->u.ctn_hash_slot++;
452 i->ctn_n++;
453 }
454
455 if ((ssize_t) i->ctn_n == i->ctn_size)
456 goto hash_end;
457
458 slot = *i->u.ctn_hash_slot;
459
460 if (key)
461 *key = slot->key;
462 if (value)
463 *value = slot->value;
464
465 i->u.ctn_hash_slot++;
466 i->ctn_n++;
467
468 return 0;
469
470 hash_end:
471 ctf_next_destroy (i);
472 *it = NULL;
473 return ECTF_NEXT_END;
474 }
475
476 int
477 ctf_dynhash_sort_by_name (const ctf_next_hkv_t *one, const ctf_next_hkv_t *two,
478 void *unused _libctf_unused_)
479 {
480 return strcmp ((char *) one->hkv_key, (char *) two->hkv_key);
481 }
482
483 /* Traverse a sorted dynhash, in _next iterator form.
484
485 See ctf_dynhash_next for notes on error returns, etc.
486
487 Sort keys before iterating over them using the SORT_FUN and SORT_ARG.
488
489 If SORT_FUN is null, thunks to ctf_dynhash_next. */
490 int
491 ctf_dynhash_next_sorted (ctf_dynhash_t *h, ctf_next_t **it, void **key,
492 void **value, ctf_hash_sort_f sort_fun, void *sort_arg)
493 {
494 ctf_next_t *i = *it;
495
496 if (sort_fun == NULL)
497 return ctf_dynhash_next (h, it, key, value);
498
499 if (!i)
500 {
501 size_t els = ctf_dynhash_elements (h);
502 ctf_next_t *accum_i = NULL;
503 void *key, *value;
504 int err;
505 ctf_next_hkv_t *walk;
506
507 if (((ssize_t) els) < 0)
508 return EDOM;
509
510 if ((i = ctf_next_create ()) == NULL)
511 return ENOMEM;
512
513 if ((i->u.ctn_sorted_hkv = calloc (els, sizeof (ctf_next_hkv_t))) == NULL)
514 {
515 ctf_next_destroy (i);
516 return ENOMEM;
517 }
518 walk = i->u.ctn_sorted_hkv;
519
520 i->cu.ctn_h = h;
521
522 while ((err = ctf_dynhash_next (h, &accum_i, &key, &value)) == 0)
523 {
524 walk->hkv_key = key;
525 walk->hkv_value = value;
526 walk++;
527 }
528 if (err != ECTF_NEXT_END)
529 {
530 ctf_next_destroy (i);
531 return err;
532 }
533
534 if (sort_fun)
535 ctf_qsort_r (i->u.ctn_sorted_hkv, els, sizeof (ctf_next_hkv_t),
536 (int (*) (const void *, const void *, void *)) sort_fun,
537 sort_arg);
538 i->ctn_n = 0;
539 i->ctn_size = (ssize_t) els;
540 i->ctn_iter_fun = (void (*) (void)) ctf_dynhash_next_sorted;
541 *it = i;
542 }
543
544 if ((void (*) (void)) ctf_dynhash_next_sorted != i->ctn_iter_fun)
545 return ECTF_NEXT_WRONGFUN;
546
547 if (h != i->cu.ctn_h)
548 return ECTF_NEXT_WRONGFP;
549
550 if ((ssize_t) i->ctn_n == i->ctn_size)
551 {
552 ctf_next_destroy (i);
553 *it = NULL;
554 return ECTF_NEXT_END;
555 }
556
557 if (key)
558 *key = i->u.ctn_sorted_hkv[i->ctn_n].hkv_key;
559 if (value)
560 *value = i->u.ctn_sorted_hkv[i->ctn_n].hkv_value;
561 i->ctn_n++;
562 return 0;
563 }
564
565 void
566 ctf_dynhash_destroy (ctf_dynhash_t *hp)
567 {
568 if (hp != NULL)
569 htab_delete (hp->htab);
570 free (hp);
571 }
572
573 /* The dynset, used for sets of keys with no value. The implementation of this
574 can be much simpler, because without a value the slot can simply be the
575 stored key, which means we don't need to store the freeing functions and the
576 dynset itself is just a htab. */
577
578 ctf_dynset_t *
579 ctf_dynset_create (htab_hash hash_fun, htab_eq eq_fun,
580 ctf_hash_free_fun key_free)
581 {
582 /* 7 is arbitrary and untested for now. */
583 return (ctf_dynset_t *) htab_create_alloc (7, (htab_hash) hash_fun, eq_fun,
584 key_free, xcalloc, free);
585 }
586
587 /* The dynset has one complexity: the underlying implementation reserves two
588 values for internal hash table implementation details (empty versus deleted
589 entries). These values are otherwise very useful for pointers cast to ints,
590 so transform the ctf_dynset_inserted value to allow for it. (This
591 introduces an ambiguity in that one can no longer store these two values in
592 the dynset, but if we pick high enough values this is very unlikely to be a
593 problem.)
594
595 We leak this implementation detail to the freeing functions on the grounds
596 that any use of these functions is overwhelmingly likely to be in sets using
597 real pointers, which will be unaffected. */
598
599 #define DYNSET_EMPTY_ENTRY_REPLACEMENT ((void *) (uintptr_t) -64)
600 #define DYNSET_DELETED_ENTRY_REPLACEMENT ((void *) (uintptr_t) -63)
601
602 static void *
603 key_to_internal (const void *key)
604 {
605 if (key == HTAB_EMPTY_ENTRY)
606 return DYNSET_EMPTY_ENTRY_REPLACEMENT;
607 else if (key == HTAB_DELETED_ENTRY)
608 return DYNSET_DELETED_ENTRY_REPLACEMENT;
609
610 return (void *) key;
611 }
612
613 static void *
614 internal_to_key (const void *internal)
615 {
616 if (internal == DYNSET_EMPTY_ENTRY_REPLACEMENT)
617 return HTAB_EMPTY_ENTRY;
618 else if (internal == DYNSET_DELETED_ENTRY_REPLACEMENT)
619 return HTAB_DELETED_ENTRY;
620 return (void *) internal;
621 }
622
623 int
624 ctf_dynset_insert (ctf_dynset_t *hp, void *key)
625 {
626 struct htab *htab = (struct htab *) hp;
627 void **slot;
628
629 slot = htab_find_slot (htab, key, INSERT);
630
631 if (!slot)
632 {
633 errno = ENOMEM;
634 return -errno;
635 }
636
637 if (*slot)
638 {
639 if (htab->del_f)
640 (*htab->del_f) (*slot);
641 }
642
643 *slot = key_to_internal (key);
644
645 return 0;
646 }
647
648 void
649 ctf_dynset_remove (ctf_dynset_t *hp, const void *key)
650 {
651 htab_remove_elt ((struct htab *) hp, key_to_internal (key));
652 }
653
654 void
655 ctf_dynset_destroy (ctf_dynset_t *hp)
656 {
657 if (hp != NULL)
658 htab_delete ((struct htab *) hp);
659 }
660
661 void *
662 ctf_dynset_lookup (ctf_dynset_t *hp, const void *key)
663 {
664 void **slot = htab_find_slot ((struct htab *) hp,
665 key_to_internal (key), NO_INSERT);
666
667 if (slot)
668 return internal_to_key (*slot);
669 return NULL;
670 }
671
672 /* TRUE/FALSE return. */
673 int
674 ctf_dynset_exists (ctf_dynset_t *hp, const void *key, const void **orig_key)
675 {
676 void **slot = htab_find_slot ((struct htab *) hp,
677 key_to_internal (key), NO_INSERT);
678
679 if (orig_key && slot)
680 *orig_key = internal_to_key (*slot);
681 return (slot != NULL);
682 }
683
684 /* Look up a completely random value from the set, if any exist.
685 Keys with value zero cannot be distinguished from a nonexistent key. */
686 void *
687 ctf_dynset_lookup_any (ctf_dynset_t *hp)
688 {
689 struct htab *htab = (struct htab *) hp;
690 void **slot = htab->entries;
691 void **limit = slot + htab_size (htab);
692
693 while (slot < limit
694 && (*slot == HTAB_EMPTY_ENTRY || *slot == HTAB_DELETED_ENTRY))
695 slot++;
696
697 if (slot < limit)
698 return internal_to_key (*slot);
699 return NULL;
700 }
701
702 /* Traverse a dynset in arbitrary order, in _next iterator form.
703
704 Otherwise, just like ctf_dynhash_next. */
705 int
706 ctf_dynset_next (ctf_dynset_t *hp, ctf_next_t **it, void **key)
707 {
708 struct htab *htab = (struct htab *) hp;
709 ctf_next_t *i = *it;
710 void *slot;
711
712 if (!i)
713 {
714 size_t size = htab_size (htab);
715
716 /* If the table has too many entries to fit in an ssize_t, just give up.
717 This might be spurious, but if any type-related hashtable has ever been
718 nearly as large as that then somthing very odd is going on. */
719
720 if (((ssize_t) size) < 0)
721 return EDOM;
722
723 if ((i = ctf_next_create ()) == NULL)
724 return ENOMEM;
725
726 i->u.ctn_hash_slot = htab->entries;
727 i->cu.ctn_s = hp;
728 i->ctn_n = 0;
729 i->ctn_size = (ssize_t) size;
730 i->ctn_iter_fun = (void (*) (void)) ctf_dynset_next;
731 *it = i;
732 }
733
734 if ((void (*) (void)) ctf_dynset_next != i->ctn_iter_fun)
735 return ECTF_NEXT_WRONGFUN;
736
737 if (hp != i->cu.ctn_s)
738 return ECTF_NEXT_WRONGFP;
739
740 if ((ssize_t) i->ctn_n == i->ctn_size)
741 goto set_end;
742
743 while ((ssize_t) i->ctn_n < i->ctn_size
744 && (*i->u.ctn_hash_slot == HTAB_EMPTY_ENTRY
745 || *i->u.ctn_hash_slot == HTAB_DELETED_ENTRY))
746 {
747 i->u.ctn_hash_slot++;
748 i->ctn_n++;
749 }
750
751 if ((ssize_t) i->ctn_n == i->ctn_size)
752 goto set_end;
753
754 slot = *i->u.ctn_hash_slot;
755
756 if (key)
757 *key = internal_to_key (slot);
758
759 i->u.ctn_hash_slot++;
760 i->ctn_n++;
761
762 return 0;
763
764 set_end:
765 ctf_next_destroy (i);
766 *it = NULL;
767 return ECTF_NEXT_END;
768 }
769
770 /* Helper functions for insertion/removal of types. */
771
772 int
773 ctf_dynhash_insert_type (ctf_dict_t *fp, ctf_dynhash_t *hp, uint32_t type,
774 uint32_t name)
775 {
776 const char *str;
777 int err;
778
779 if (type == 0)
780 return EINVAL;
781
782 if ((str = ctf_strptr_validate (fp, name)) == NULL)
783 return ctf_errno (fp);
784
785 if (str[0] == '\0')
786 return 0; /* Just ignore empty strings on behalf of caller. */
787
788 if ((err = ctf_dynhash_insert (hp, (char *) str,
789 (void *) (ptrdiff_t) type)) == 0)
790 return 0;
791
792 return err;
793 }
794
795 ctf_id_t
796 ctf_dynhash_lookup_type (ctf_dynhash_t *hp, const char *key)
797 {
798 void *value;
799
800 if (ctf_dynhash_lookup_kv (hp, key, NULL, &value))
801 return (ctf_id_t) (uintptr_t) value;
802
803 return 0;
804 }
805