Coverage Report

Created: 2026-01-25 06:10

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/curl/lib/hash.c
Line
Count
Source
1
/***************************************************************************
2
 *                                  _   _ ____  _
3
 *  Project                     ___| | | |  _ \| |
4
 *                             / __| | | | |_) | |
5
 *                            | (__| |_| |  _ <| |___
6
 *                             \___|\___/|_| \_\_____|
7
 *
8
 * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
9
 *
10
 * This software is licensed as described in the file COPYING, which
11
 * you should have received as part of this distribution. The terms
12
 * are also available at https://curl.se/docs/copyright.html.
13
 *
14
 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15
 * copies of the Software, and permit persons to whom the Software is
16
 * furnished to do so, under the terms of the COPYING file.
17
 *
18
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19
 * KIND, either express or implied.
20
 *
21
 * SPDX-License-Identifier: curl
22
 *
23
 ***************************************************************************/
24
#include "curl_setup.h"
25
26
#include "hash.h"
27
28
/* random patterns for API verification */
29
#ifdef DEBUGBUILD
30
0
#define HASHINIT 0x7017e781
31
0
#define ITERINIT 0x5FEDCBA9
32
#endif
33
34
#if 0 /* useful function for debugging hashes and their contents */
35
void Curl_hash_print(struct Curl_hash *h, void (*func)(void *))
36
{
37
  struct Curl_hash_iterator iter;
38
  struct Curl_hash_element *he;
39
  size_t last_index = UINT_MAX;
40
41
  if(!h)
42
    return;
43
44
  curl_mfprintf(stderr, "=Hash dump=\n");
45
46
  Curl_hash_start_iterate(h, &iter);
47
48
  he = Curl_hash_next_element(&iter);
49
  while(he) {
50
    if(iter.slot_index != last_index) {
51
      curl_mfprintf(stderr, "index %d:", (int)iter.slot_index);
52
      if(last_index != UINT_MAX) {
53
        curl_mfprintf(stderr, "\n");
54
      }
55
      last_index = iter.slot_index;
56
    }
57
58
    if(func)
59
      func(he->ptr);
60
    else
61
      curl_mfprintf(stderr, " [key=%.*s, he=%p, ptr=%p]",
62
                    (int)he->key_len, (char *)he->key,
63
                    (void *)he, (void *)he->ptr);
64
65
    he = Curl_hash_next_element(&iter);
66
  }
67
  curl_mfprintf(stderr, "\n");
68
}
69
#endif
70
71
/* Initializes a hash structure.
72
 * Return 1 on error, 0 is fine.
73
 *
74
 * @unittest: 1602
75
 * @unittest: 1603
76
 */
77
void Curl_hash_init(struct Curl_hash *h,
78
                    size_t slots,
79
                    hash_function hfunc,
80
                    comp_function comparator,
81
                    Curl_hash_dtor dtor)
82
0
{
83
0
  DEBUGASSERT(h);
84
0
  DEBUGASSERT(slots);
85
0
  DEBUGASSERT(hfunc);
86
0
  DEBUGASSERT(comparator);
87
0
  DEBUGASSERT(dtor);
88
89
0
  h->table = NULL;
90
0
  h->hash_func = hfunc;
91
0
  h->comp_func = comparator;
92
0
  h->dtor = dtor;
93
0
  h->size = 0;
94
0
  h->slots = slots;
95
0
#ifdef DEBUGBUILD
96
0
  h->init = HASHINIT;
97
0
#endif
98
0
}
99
100
static struct Curl_hash_element *hash_elem_create(const void *key,
101
                                                  size_t key_len,
102
                                                  const void *p,
103
                                                  Curl_hash_elem_dtor dtor)
104
0
{
105
0
  struct Curl_hash_element *he;
106
107
  /* allocate the struct plus memory after it to store the key */
108
0
  he = curlx_malloc(sizeof(struct Curl_hash_element) + key_len);
109
0
  if(he) {
110
0
    he->next = NULL;
111
    /* copy the key */
112
0
    memcpy(he->key, key, key_len);
113
0
    he->key_len = key_len;
114
0
    he->ptr = CURL_UNCONST(p);
115
0
    he->dtor = dtor;
116
0
  }
117
0
  return he;
118
0
}
119
120
static void hash_elem_clear_ptr(struct Curl_hash *h,
121
                                struct Curl_hash_element *he)
122
0
{
123
0
  DEBUGASSERT(h);
124
0
  DEBUGASSERT(he);
125
0
  if(he->ptr) {
126
0
    if(he->dtor)
127
0
      he->dtor(he->key, he->key_len, he->ptr);
128
0
    else
129
0
      h->dtor(he->ptr);
130
0
    he->ptr = NULL;
131
0
  }
132
0
}
133
134
static void hash_elem_destroy(struct Curl_hash *h,
135
                              struct Curl_hash_element *he)
136
0
{
137
0
  hash_elem_clear_ptr(h, he);
138
0
  curlx_free(he);
139
0
}
140
141
static void hash_elem_unlink(struct Curl_hash *h,
142
                             struct Curl_hash_element **he_anchor,
143
                             struct Curl_hash_element *he)
144
0
{
145
0
  *he_anchor = he->next;
146
0
  --h->size;
147
0
}
148
149
static void hash_elem_link(struct Curl_hash *h,
150
                           struct Curl_hash_element **he_anchor,
151
                           struct Curl_hash_element *he)
152
0
{
153
0
  he->next = *he_anchor;
154
0
  *he_anchor = he;
155
0
  ++h->size;
156
0
}
157
158
0
#define CURL_HASH_SLOT(x, y, z)      x->table[x->hash_func(y, z, x->slots)]
159
0
#define CURL_HASH_SLOT_ADDR(x, y, z) &CURL_HASH_SLOT(x, y, z)
160
161
void *Curl_hash_add2(struct Curl_hash *h, void *key, size_t key_len, void *p,
162
                     Curl_hash_elem_dtor dtor)
163
0
{
164
0
  struct Curl_hash_element *he, **slot;
165
166
0
  DEBUGASSERT(h);
167
0
  DEBUGASSERT(h->slots);
168
0
  DEBUGASSERT(h->init == HASHINIT);
169
0
  if(!h->table) {
170
0
    h->table = curlx_calloc(h->slots, sizeof(struct Curl_hash_element *));
171
0
    if(!h->table)
172
0
      return NULL; /* OOM */
173
0
  }
174
175
0
  slot = CURL_HASH_SLOT_ADDR(h, key, key_len);
176
0
  for(he = *slot; he; he = he->next) {
177
0
    if(h->comp_func(he->key, he->key_len, key, key_len)) {
178
      /* existing key entry, overwrite by clearing old pointer */
179
0
      hash_elem_clear_ptr(h, he);
180
0
      he->ptr = (void *)p;
181
0
      he->dtor = dtor;
182
0
      return p;
183
0
    }
184
0
  }
185
186
0
  he = hash_elem_create(key, key_len, p, dtor);
187
0
  if(!he)
188
0
    return NULL; /* OOM */
189
190
0
  hash_elem_link(h, slot, he);
191
0
  return p; /* return the new entry */
192
0
}
193
194
/* Insert the data in the hash. If there already was a match in the hash, that
195
 * data is replaced. This function also "lazily" allocates the table if
196
 * needed, as it is not done in the _init function (anymore).
197
 *
198
 * @unittest: 1305
199
 * @unittest: 1602
200
 * @unittest: 1603
201
 */
202
void *Curl_hash_add(struct Curl_hash *h, void *key, size_t key_len, void *p)
203
0
{
204
0
  return Curl_hash_add2(h, key, key_len, p, NULL);
205
0
}
206
207
/* Remove the identified hash entry.
208
 * Returns non-zero on failure.
209
 *
210
 * @unittest: 1603
211
 */
212
int Curl_hash_delete(struct Curl_hash *h, void *key, size_t key_len)
213
0
{
214
0
  DEBUGASSERT(h);
215
0
  DEBUGASSERT(h->slots);
216
0
  DEBUGASSERT(h->init == HASHINIT);
217
0
  if(h->table) {
218
0
    struct Curl_hash_element *he, **he_anchor;
219
220
0
    he_anchor = CURL_HASH_SLOT_ADDR(h, key, key_len);
221
0
    while(*he_anchor) {
222
0
      he = *he_anchor;
223
0
      if(h->comp_func(he->key, he->key_len, key, key_len)) {
224
0
        hash_elem_unlink(h, he_anchor, he);
225
0
        hash_elem_destroy(h, he);
226
0
        return 0;
227
0
      }
228
0
      he_anchor = &he->next;
229
0
    }
230
0
  }
231
0
  return 1;
232
0
}
233
234
/* Retrieves a hash element.
235
 *
236
 * @unittest: 1603
237
 */
238
void *Curl_hash_pick(struct Curl_hash *h, void *key, size_t key_len)
239
0
{
240
0
  DEBUGASSERT(h);
241
0
  DEBUGASSERT(h->init == HASHINIT);
242
0
  if(h->table) {
243
0
    struct Curl_hash_element *he;
244
0
    DEBUGASSERT(h->slots);
245
0
    he = CURL_HASH_SLOT(h, key, key_len);
246
0
    while(he) {
247
0
      if(h->comp_func(he->key, he->key_len, key, key_len)) {
248
0
        return he->ptr;
249
0
      }
250
0
      he = he->next;
251
0
    }
252
0
  }
253
0
  return NULL;
254
0
}
255
256
/* Destroys all the entries in the given hash and resets its attributes,
257
 * prepping the given hash for [static|dynamic] deallocation.
258
 *
259
 * @unittest: 1305
260
 * @unittest: 1602
261
 * @unittest: 1603
262
 */
263
void Curl_hash_destroy(struct Curl_hash *h)
264
0
{
265
0
  DEBUGASSERT(h->init == HASHINIT);
266
0
  if(h->table) {
267
0
    Curl_hash_clean(h);
268
0
    Curl_safefree(h->table);
269
0
  }
270
0
  DEBUGASSERT(h->size == 0);
271
0
  h->slots = 0;
272
0
}
273
274
/* Removes all the entries in the given hash.
275
 *
276
 * @unittest: 1602
277
 */
278
void Curl_hash_clean(struct Curl_hash *h)
279
0
{
280
0
  if(h && h->table) {
281
0
    struct Curl_hash_element *he, **he_anchor;
282
0
    size_t i;
283
0
    DEBUGASSERT(h->init == HASHINIT);
284
0
    for(i = 0; i < h->slots; ++i) {
285
0
      he_anchor = &h->table[i];
286
0
      while(*he_anchor) {
287
0
        he = *he_anchor;
288
0
        hash_elem_unlink(h, he_anchor, he);
289
0
        hash_elem_destroy(h, he);
290
0
      }
291
0
    }
292
0
  }
293
0
}
294
295
size_t Curl_hash_count(struct Curl_hash *h)
296
0
{
297
0
  DEBUGASSERT(h->init == HASHINIT);
298
0
  return h->size;
299
0
}
300
301
/* Cleans all entries that pass the comp function criteria. */
302
void Curl_hash_clean_with_criterium(struct Curl_hash *h, void *user,
303
                                    int (*comp)(void *, void *))
304
0
{
305
0
  size_t i;
306
307
0
  if(!h || !h->table)
308
0
    return;
309
310
0
  DEBUGASSERT(h->init == HASHINIT);
311
0
  for(i = 0; i < h->slots; ++i) {
312
0
    struct Curl_hash_element *he, **he_anchor = &h->table[i];
313
0
    while(*he_anchor) {
314
      /* ask the callback function if we shall remove this entry or not */
315
0
      if(!comp || comp(user, (*he_anchor)->ptr)) {
316
0
        he = *he_anchor;
317
0
        hash_elem_unlink(h, he_anchor, he);
318
0
        hash_elem_destroy(h, he);
319
0
      }
320
0
      else
321
0
        he_anchor = &(*he_anchor)->next;
322
0
    }
323
0
  }
324
0
}
325
326
size_t Curl_hash_str(void *key, size_t key_length, size_t slots_num)
327
0
{
328
0
  const char *key_str = (const char *)key;
329
0
  const char *end = key_str + key_length;
330
0
  size_t h = 5381;
331
332
0
  while(key_str < end) {
333
0
    size_t j = (size_t)*key_str++;
334
0
    h += h << 5;
335
0
    h ^= j;
336
0
  }
337
338
0
  return (h % slots_num);
339
0
}
340
341
size_t curlx_str_key_compare(void *k1, size_t key1_len,
342
                             void *k2, size_t key2_len)
343
0
{
344
0
  if((key1_len == key2_len) && !memcmp(k1, k2, key1_len))
345
0
    return 1;
346
347
0
  return 0;
348
0
}
349
350
void Curl_hash_start_iterate(struct Curl_hash *hash,
351
                             struct Curl_hash_iterator *iter)
352
0
{
353
0
  DEBUGASSERT(hash->init == HASHINIT);
354
0
  iter->hash = hash;
355
0
  iter->slot_index = 0;
356
0
  iter->current = NULL;
357
0
#ifdef DEBUGBUILD
358
0
  iter->init = ITERINIT;
359
0
#endif
360
0
}
361
362
struct Curl_hash_element *
363
Curl_hash_next_element(struct Curl_hash_iterator *iter)
364
0
{
365
0
  struct Curl_hash *h;
366
0
  DEBUGASSERT(iter->init == ITERINIT);
367
0
  h = iter->hash;
368
0
  if(!h->table)
369
0
    return NULL; /* empty hash, nothing to return */
370
371
  /* Get the next element in the current list, if any */
372
0
  if(iter->current)
373
0
    iter->current = iter->current->next;
374
375
  /* If we have reached the end of the list, find the next one */
376
0
  if(!iter->current) {
377
0
    size_t i;
378
0
    for(i = iter->slot_index; i < h->slots; i++) {
379
0
      if(h->table[i]) {
380
0
        iter->current = h->table[i];
381
0
        iter->slot_index = i + 1;
382
0
        break;
383
0
      }
384
0
    }
385
0
  }
386
387
0
  return iter->current;
388
0
}