Coverage Report

Created: 2026-03-11 07:23

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
1.13M
#define HASHINIT 0x7017e781
31
279k
#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
1.13M
{
83
1.13M
  DEBUGASSERT(h);
84
1.13M
  DEBUGASSERT(slots);
85
1.13M
  DEBUGASSERT(hfunc);
86
1.13M
  DEBUGASSERT(comparator);
87
1.13M
  DEBUGASSERT(dtor);
88
89
1.13M
  h->table = NULL;
90
1.13M
  h->hash_func = hfunc;
91
1.13M
  h->comp_func = comparator;
92
1.13M
  h->dtor = dtor;
93
1.13M
  h->size = 0;
94
1.13M
  h->slots = slots;
95
1.13M
#ifdef DEBUGBUILD
96
1.13M
  h->init = HASHINIT;
97
1.13M
#endif
98
1.13M
}
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
320k
{
105
320k
  struct Curl_hash_element *he;
106
107
  /* allocate the struct plus memory after it to store the key */
108
320k
  he = curlx_malloc(sizeof(struct Curl_hash_element) + key_len);
109
320k
  if(he) {
110
320k
    he->next = NULL;
111
    /* copy the key */
112
320k
    memcpy(he->key, key, key_len);
113
320k
    he->key_len = key_len;
114
320k
    he->ptr = CURL_UNCONST(p);
115
320k
    he->dtor = dtor;
116
320k
  }
117
320k
  return he;
118
320k
}
119
120
static void hash_elem_clear_ptr(struct Curl_hash *h,
121
                                struct Curl_hash_element *he)
122
329k
{
123
329k
  DEBUGASSERT(h);
124
329k
  DEBUGASSERT(he);
125
329k
  if(he->ptr) {
126
329k
    if(he->dtor)
127
120k
      he->dtor(he->key, he->key_len, he->ptr);
128
209k
    else
129
209k
      h->dtor(he->ptr);
130
329k
    he->ptr = NULL;
131
329k
  }
132
329k
}
133
134
static void hash_elem_destroy(struct Curl_hash *h,
135
                              struct Curl_hash_element *he)
136
320k
{
137
320k
  hash_elem_clear_ptr(h, he);
138
320k
  curlx_free(he);
139
320k
}
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
320k
{
145
320k
  *he_anchor = he->next;
146
320k
  --h->size;
147
320k
}
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
320k
{
153
320k
  he->next = *he_anchor;
154
320k
  *he_anchor = he;
155
320k
  ++h->size;
156
320k
}
157
158
66.8M
#define CURL_HASH_SLOT(x, y, z)      x->table[(x)->hash_func(y, z, (x)->slots)]
159
546k
#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
329k
{
164
329k
  struct Curl_hash_element *he, **slot;
165
166
329k
  DEBUGASSERT(h);
167
329k
  DEBUGASSERT(h->slots);
168
329k
  DEBUGASSERT(h->init == HASHINIT);
169
329k
  if(!h->table) {
170
308k
    h->table = curlx_calloc(h->slots, sizeof(struct Curl_hash_element *));
171
308k
    if(!h->table)
172
0
      return NULL; /* OOM */
173
308k
  }
174
175
329k
  slot = CURL_HASH_SLOT_ADDR(h, key, key_len);
176
329k
  for(he = *slot; he; he = he->next) {
177
9.48k
    if(h->comp_func(he->key, he->key_len, key, key_len)) {
178
      /* existing key entry, overwrite by clearing old pointer */
179
9.07k
      hash_elem_clear_ptr(h, he);
180
9.07k
      he->ptr = p;
181
9.07k
      he->dtor = dtor;
182
9.07k
      return p;
183
9.07k
    }
184
9.48k
  }
185
186
320k
  he = hash_elem_create(key, key_len, p, dtor);
187
320k
  if(!he)
188
0
    return NULL; /* OOM */
189
190
320k
  hash_elem_link(h, slot, he);
191
320k
  return p; /* return the new entry */
192
320k
}
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
209k
{
204
209k
  return Curl_hash_add2(h, key, key_len, p, NULL);
205
209k
}
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
391k
{
214
391k
  DEBUGASSERT(h);
215
391k
  DEBUGASSERT(h->slots);
216
391k
  DEBUGASSERT(h->init == HASHINIT);
217
391k
  if(h->table) {
218
216k
    struct Curl_hash_element *he, **he_anchor;
219
220
216k
    he_anchor = CURL_HASH_SLOT_ADDR(h, key, key_len);
221
216k
    while(*he_anchor) {
222
115k
      he = *he_anchor;
223
115k
      if(h->comp_func(he->key, he->key_len, key, key_len)) {
224
115k
        hash_elem_unlink(h, he_anchor, he);
225
115k
        hash_elem_destroy(h, he);
226
115k
        return 0;
227
115k
      }
228
0
      he_anchor = &he->next;
229
0
    }
230
216k
  }
231
276k
  return 1;
232
391k
}
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
66.7M
{
240
66.7M
  DEBUGASSERT(h);
241
66.7M
  DEBUGASSERT(h->init == HASHINIT);
242
66.7M
  if(h->table) {
243
66.3M
    struct Curl_hash_element *he;
244
66.3M
    DEBUGASSERT(h->slots);
245
66.3M
    he = CURL_HASH_SLOT(h, key, key_len);
246
67.0M
    while(he) {
247
64.6M
      if(h->comp_func(he->key, he->key_len, key, key_len)) {
248
63.9M
        return he->ptr;
249
63.9M
      }
250
707k
      he = he->next;
251
707k
    }
252
66.3M
  }
253
2.82M
  return NULL;
254
66.7M
}
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
1.13M
{
265
1.13M
  DEBUGASSERT(h->init == HASHINIT);
266
1.13M
  if(h->table) {
267
308k
    Curl_hash_clean(h);
268
308k
    Curl_safefree(h->table);
269
308k
  }
270
1.13M
  DEBUGASSERT(h->size == 0);
271
1.13M
  h->slots = 0;
272
1.13M
}
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
308k
{
280
308k
  if(h && h->table) {
281
308k
    struct Curl_hash_element *he, **he_anchor;
282
308k
    size_t i;
283
308k
    DEBUGASSERT(h->init == HASHINIT);
284
19.5M
    for(i = 0; i < h->slots; ++i) {
285
19.2M
      he_anchor = &h->table[i];
286
19.4M
      while(*he_anchor) {
287
204k
        he = *he_anchor;
288
204k
        hash_elem_unlink(h, he_anchor, he);
289
204k
        hash_elem_destroy(h, he);
290
204k
      }
291
19.2M
    }
292
308k
  }
293
308k
}
294
295
size_t Curl_hash_count(struct Curl_hash *h)
296
117k
{
297
117k
  DEBUGASSERT(h->init == HASHINIT);
298
117k
  return h->size;
299
117k
}
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
117k
{
305
117k
  size_t i;
306
307
117k
  if(!h || !h->table)
308
8.90k
    return;
309
310
108k
  DEBUGASSERT(h->init == HASHINIT);
311
7.83M
  for(i = 0; i < h->slots; ++i) {
312
7.72M
    struct Curl_hash_element *he, **he_anchor = &h->table[i];
313
7.83M
    while(*he_anchor) {
314
      /* ask the callback function if we shall remove this entry or not */
315
110k
      if(!comp || comp(user, (*he_anchor)->ptr)) {
316
133
        he = *he_anchor;
317
133
        hash_elem_unlink(h, he_anchor, he);
318
133
        hash_elem_destroy(h, he);
319
133
      }
320
110k
      else
321
110k
        he_anchor = &(*he_anchor)->next;
322
110k
    }
323
7.72M
  }
324
108k
}
325
326
size_t Curl_hash_str(void *key, size_t key_length, size_t slots_num)
327
66.8M
{
328
66.8M
  const char *key_str = (const char *)key;
329
66.8M
  const char *end = key_str + key_length;
330
66.8M
  size_t h = 5381;
331
332
1.39G
  while(key_str < end) {
333
1.32G
    size_t j = (size_t)*key_str++;
334
1.32G
    h += h << 5;
335
1.32G
    h ^= j;
336
1.32G
  }
337
338
66.8M
  return (h % slots_num);
339
66.8M
}
340
341
size_t curlx_str_key_compare(void *k1, size_t key1_len,
342
                             void *k2, size_t key2_len)
343
64.7M
{
344
64.7M
  if((key1_len == key2_len) && !memcmp(k1, k2, key1_len))
345
64.0M
    return 1;
346
347
707k
  return 0;
348
64.7M
}
349
350
void Curl_hash_start_iterate(struct Curl_hash *hash,
351
                             struct Curl_hash_iterator *iter)
352
279k
{
353
279k
  DEBUGASSERT(hash->init == HASHINIT);
354
279k
  iter->hash = hash;
355
279k
  iter->slot_index = 0;
356
279k
  iter->current = NULL;
357
279k
#ifdef DEBUGBUILD
358
279k
  iter->init = ITERINIT;
359
279k
#endif
360
279k
}
361
362
struct Curl_hash_element *
363
Curl_hash_next_element(struct Curl_hash_iterator *iter)
364
288k
{
365
288k
  struct Curl_hash *h;
366
288k
  DEBUGASSERT(iter->init == ITERINIT);
367
288k
  h = iter->hash;
368
288k
  if(!h->table)
369
158k
    return NULL; /* empty hash, nothing to return */
370
371
  /* Get the next element in the current list, if any */
372
129k
  if(iter->current)
373
8.91k
    iter->current = iter->current->next;
374
375
  /* If we have reached the end of the list, find the next one */
376
129k
  if(!iter->current) {
377
129k
    size_t i;
378
11.6M
    for(i = iter->slot_index; i < h->slots; i++) {
379
11.5M
      if(h->table[i]) {
380
14.7k
        iter->current = h->table[i];
381
14.7k
        iter->slot_index = i + 1;
382
14.7k
        break;
383
14.7k
      }
384
11.5M
    }
385
129k
  }
386
387
129k
  return iter->current;
388
288k
}