Coverage Report

Created: 2026-05-30 06:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/openssl/crypto/hashtable/hashtable.c
Line
Count
Source
1
/*
2
 * Copyright 2024-2026 The OpenSSL Project Authors. All Rights Reserved.
3
 *
4
 * Licensed under the Apache License 2.0 (the "License").  You may not use
5
 * this file except in compliance with the License.  You can obtain a copy
6
 * in the file LICENSE in the source distribution or at
7
 * https://www.openssl.org/source/license.html
8
 *
9
 *
10
 *
11
 * Notes On hash table design and layout
12
 * This hashtable uses a hopscotch algorithm to do indexing.  The data structure
13
 * looks as follows:
14
 *
15
 *   hash          +--------------+
16
 *   value+------->+ HT_VALUE     |
17
 *      +          +--------------+
18
 *  +-------+
19
 *  |       |
20
 *  +---------------------------------------------------------+
21
 *  |       |       |       |       |                         |
22
 *  | entry | entry | entry | entry |                         |
23
 *  |       |       |       |       |                         |
24
 *  +---------------------------------------------------------+
25
 *  |                               |                         |
26
 *  |                               |                         |
27
 *  +---------------------------------------------------------+
28
 *  |              +                             +            +
29
 *  |        neighborhood[0]               neighborhood[1]    |
30
 *  |                                                         |
31
 *  |                                                         |
32
 *  +---------------------------------------------------------+
33
 *                              |
34
 *                              +
35
 *                         neighborhoods
36
 *
37
 * On lookup/insert/delete, the items key is hashed to a 64 bit value
38
 * and the result is masked to provide an index into the neighborhoods
39
 * table.  Once a neighborhood is determined, an in-order search is done
40
 * of the elements in the neighborhood indexes entries for a matching hash
41
 * value, if found, the corresponding HT_VALUE is used for the respective
42
 * operation.  The number of entries in a neighborhood is determined at build
43
 * time based on the cacheline size of the target CPU.  The intent is for a
44
 * neighborhood to have all entries in the neighborhood fit into a single cache
45
 * line to speed up lookups.  If all entries in a neighborhood are in use at the
46
 * time of an insert, the table is expanded and rehashed.
47
 *
48
 * Lockless reads hash table is based on the same design but does not
49
 * allow growing and deletion. Thus subsequent neighborhoods are always
50
 * searched for a match until an empty entry is found.
51
 */
52
53
#include <string.h>
54
#include <internal/rcu.h>
55
#include <internal/hashtable.h>
56
#include <internal/hashfunc.h>
57
#include <openssl/rand.h>
58
59
/*
60
 * gcc defines __SANITIZE_THREAD__
61
 * but clang uses the feature attributes api
62
 * map the latter to the former
63
 */
64
#if defined(__clang__) && defined(__has_feature)
65
#if __has_feature(thread_sanitizer)
66
#define __SANITIZE_THREADS__
67
#endif
68
#endif
69
70
#ifdef __SANITIZE_THREADS__
71
#include <sanitizer/tsan_interface.h>
72
#endif
73
74
/*
75
 * When we do a lookup/insert/delete, there is a high likelihood
76
 * that we will iterate over at least part of the neighborhood list
77
 * As such, because we design a neighborhood entry to fit into a single
78
 * cache line it is advantageous, when supported to fetch the entire
79
 * structure for faster lookups
80
 */
81
#if defined(__GNUC__) || defined(__CLANG__)
82
1.42M
#define PREFETCH_NEIGHBORHOOD(x) __builtin_prefetch(x.entries)
83
2.83M
#define PREFETCH(x) __builtin_prefetch(x)
84
#define ALIGN __attribute__((aligned(8)))
85
#else
86
#define PREFETCH_NEIGHBORHOOD(x)
87
#define PREFETCH(x)
88
#define ALIGN
89
#endif
90
91
/*
92
 * Define our neighborhood list length
93
 * Note: It should always be a power of 2
94
 */
95
408
#define DEFAULT_NEIGH_LEN_LOG 4
96
408
#define DEFAULT_NEIGH_LEN (1 << DEFAULT_NEIGH_LEN_LOG)
97
98
/*
99
 * For now assume cache line size is 64 bytes
100
 */
101
3.21M
#define CACHE_LINE_BYTES 64
102
#define CACHE_LINE_ALIGNMENT CACHE_LINE_BYTES
103
104
3.21M
#define NEIGHBORHOOD_LEN (CACHE_LINE_BYTES / sizeof(struct ht_neighborhood_entry_st))
105
/*
106
 * Defines our chains of values
107
 */
108
struct ht_internal_value_st {
109
    HT_VALUE value;
110
    HT *ht;
111
};
112
113
struct ht_neighborhood_entry_st {
114
    uint64_t hash;
115
    struct ht_internal_value_st *value;
116
} ALIGN;
117
118
struct ht_neighborhood_st {
119
    struct ht_neighborhood_entry_st entries[NEIGHBORHOOD_LEN];
120
};
121
122
/*
123
 * Updates to data in this struct
124
 * require an rcu sync after modification
125
 * prior to free
126
 */
127
struct ht_mutable_data_st {
128
    struct ht_neighborhood_st *neighborhoods;
129
    void *neighborhood_ptr_to_free;
130
    uint64_t neighborhood_mask;
131
};
132
133
/*
134
 * Private data may be updated on the write
135
 * side only, and so do not require rcu sync
136
 */
137
struct ht_write_private_data_st {
138
    size_t neighborhood_len;
139
    size_t value_count;
140
    int need_sync;
141
};
142
143
struct ht_internal_st {
144
    HT_CONFIG config;
145
    CRYPTO_RCU_LOCK *lock;
146
    CRYPTO_RWLOCK *atomic_lock;
147
    struct ht_mutable_data_st *md;
148
    struct ht_write_private_data_st wpd;
149
};
150
151
static void free_value(struct ht_internal_value_st *v);
152
153
static struct ht_neighborhood_st *alloc_new_neighborhood_list(size_t len,
154
    void **freeptr)
155
239
{
156
239
    struct ht_neighborhood_st *ret;
157
158
239
#if !defined(OPENSSL_SMALL_FOOTPRINT)
159
239
    ret = OPENSSL_aligned_alloc_array(len, sizeof(struct ht_neighborhood_st),
160
239
        CACHE_LINE_BYTES, freeptr);
161
162
    /* fall back to regular malloc */
163
239
    if (ret == NULL)
164
0
#endif
165
0
    {
166
0
        ret = *freeptr = OPENSSL_malloc_array(len, sizeof(struct ht_neighborhood_st));
167
0
        if (ret == NULL)
168
0
            return NULL;
169
0
    }
170
239
    memset(ret, 0, sizeof(struct ht_neighborhood_st) * len);
171
239
    return ret;
172
239
}
173
174
static void internal_free_nop(HT_VALUE *v)
175
0
{
176
0
    return;
177
0
}
178
179
static uint64_t internal_ht_hash_fn(HT_KEY *key)
180
1.42M
{
181
1.42M
    return ossl_fnv1a_hash(key->keybuf, key->keysize);
182
1.42M
}
183
184
HT *ossl_ht_new(const HT_CONFIG *conf)
185
102
{
186
102
    HT *new = OPENSSL_zalloc(sizeof(*new));
187
188
102
    if (new == NULL)
189
0
        return NULL;
190
191
102
    if (conf->lockless_reads && conf->no_rcu)
192
0
        goto err;
193
194
102
    if (!conf->no_rcu) {
195
6
        new->atomic_lock = CRYPTO_THREAD_lock_new();
196
6
        if (new->atomic_lock == NULL)
197
0
            goto err;
198
6
    }
199
102
    memcpy(&new->config, conf, sizeof(*conf));
200
201
102
    if (new->config.init_neighborhoods != 0) {
202
102
        new->wpd.neighborhood_len = new->config.init_neighborhoods;
203
        /* round up to the next power of 2 */
204
102
        new->wpd.neighborhood_len--;
205
102
        new->wpd.neighborhood_len |= new->wpd.neighborhood_len >> 1;
206
102
        new->wpd.neighborhood_len |= new->wpd.neighborhood_len >> 2;
207
102
        new->wpd.neighborhood_len |= new->wpd.neighborhood_len >> 4;
208
102
        new->wpd.neighborhood_len |= new->wpd.neighborhood_len >> 8;
209
102
        new->wpd.neighborhood_len |= new->wpd.neighborhood_len >> 16;
210
102
        new->wpd.neighborhood_len++;
211
102
    } else {
212
0
        new->wpd.neighborhood_len = DEFAULT_NEIGH_LEN;
213
0
    }
214
215
102
    if (new->config.ht_free_fn == NULL)
216
6
        new->config.ht_free_fn = internal_free_nop;
217
218
102
    new->md = OPENSSL_zalloc(sizeof(*new->md));
219
102
    if (new->md == NULL)
220
0
        goto err;
221
222
102
    new->md->neighborhoods = alloc_new_neighborhood_list(new->wpd.neighborhood_len,
223
102
        &new->md->neighborhood_ptr_to_free);
224
102
    if (new->md->neighborhoods == NULL)
225
0
        goto err;
226
102
    new->md->neighborhood_mask = new->wpd.neighborhood_len - 1;
227
228
102
    if (!conf->no_rcu) {
229
6
        new->lock = ossl_rcu_lock_new(1, conf->ctx);
230
6
        if (new->lock == NULL)
231
0
            goto err;
232
6
    }
233
102
    if (new->config.ht_hash_fn == NULL)
234
102
        new->config.ht_hash_fn = internal_ht_hash_fn;
235
236
102
    return new;
237
238
0
err:
239
0
    if (!conf->no_rcu) {
240
0
        CRYPTO_THREAD_lock_free(new->atomic_lock);
241
0
        ossl_rcu_lock_free(new->lock);
242
0
    }
243
0
    if (new->md != NULL)
244
0
        OPENSSL_free(new->md->neighborhood_ptr_to_free);
245
0
    OPENSSL_free(new->md);
246
0
    OPENSSL_free(new);
247
0
    return NULL;
248
102
}
249
250
int ossl_ht_read_lock(HT *htable)
251
0
{
252
0
    if (htable->config.no_rcu)
253
0
        return 1;
254
255
0
    return ossl_rcu_read_lock(htable->lock);
256
0
}
257
258
void ossl_ht_read_unlock(HT *htable)
259
0
{
260
0
    if (htable->config.no_rcu)
261
0
        return;
262
263
0
    ossl_rcu_read_unlock(htable->lock);
264
0
}
265
266
void ossl_ht_write_lock(HT *htable)
267
0
{
268
0
    if (htable->config.no_rcu)
269
0
        return;
270
271
0
    ossl_rcu_write_lock(htable->lock);
272
0
    htable->wpd.need_sync = 0;
273
0
}
274
275
void ossl_ht_write_unlock(HT *htable)
276
0
{
277
0
    int need_sync = htable->wpd.need_sync;
278
279
0
    if (htable->config.no_rcu)
280
0
        return;
281
282
0
    htable->wpd.need_sync = 0;
283
0
    ossl_rcu_write_unlock(htable->lock);
284
0
    if (need_sync)
285
0
        ossl_synchronize_rcu(htable->lock);
286
0
}
287
288
static void free_oldmd(void *arg)
289
136
{
290
136
    struct ht_mutable_data_st *oldmd = arg;
291
136
    size_t i, j;
292
136
    size_t neighborhood_len = (size_t)oldmd->neighborhood_mask + 1;
293
136
    struct ht_internal_value_st *v;
294
295
1.35k
    for (i = 0; i < neighborhood_len; i++) {
296
1.21k
        PREFETCH_NEIGHBORHOOD(oldmd->neighborhoods[i + 1]);
297
6.08k
        for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
298
4.86k
            if (oldmd->neighborhoods[i].entries[j].value != NULL) {
299
0
                v = oldmd->neighborhoods[i].entries[j].value;
300
0
                v->ht->config.ht_free_fn((HT_VALUE *)v);
301
0
                free_value(v);
302
0
            }
303
4.86k
        }
304
1.21k
    }
305
306
136
    OPENSSL_free(oldmd->neighborhood_ptr_to_free);
307
136
    OPENSSL_free(oldmd);
308
136
}
309
310
static int ossl_ht_flush_internal(HT *h)
311
136
{
312
136
    struct ht_mutable_data_st *newmd = NULL;
313
136
    struct ht_mutable_data_st *oldmd = NULL;
314
136
    CRYPTO_RCU_CB_ITEM *cbi = NULL;
315
316
136
    newmd = OPENSSL_zalloc(sizeof(*newmd));
317
136
    if (newmd == NULL)
318
0
        return 0;
319
320
136
    newmd->neighborhoods = alloc_new_neighborhood_list(DEFAULT_NEIGH_LEN,
321
136
        &newmd->neighborhood_ptr_to_free);
322
136
    if (newmd->neighborhoods == NULL) {
323
0
        OPENSSL_free(newmd);
324
0
        return 0;
325
0
    }
326
327
136
    newmd->neighborhood_mask = DEFAULT_NEIGH_LEN - 1;
328
329
136
    if (!h->config.no_rcu) {
330
0
        cbi = ossl_rcu_cb_item_new();
331
0
        if (cbi == NULL) {
332
0
            OPENSSL_free(newmd->neighborhood_ptr_to_free);
333
0
            OPENSSL_free(newmd);
334
0
            return 0;
335
0
        }
336
337
        /* Swap the old and new mutable data sets */
338
0
        oldmd = ossl_rcu_deref(&h->md);
339
0
        ossl_rcu_assign_ptr(&h->md, &newmd);
340
136
    } else {
341
136
        oldmd = h->md;
342
136
        h->md = newmd;
343
136
    }
344
345
    /* Set the number of entries to 0 */
346
136
    h->wpd.value_count = 0;
347
136
    h->wpd.neighborhood_len = DEFAULT_NEIGH_LEN;
348
349
136
    if (!h->config.no_rcu) {
350
0
        ossl_rcu_call(h->lock, cbi, free_oldmd, oldmd);
351
0
        h->wpd.need_sync = 1;
352
136
    } else {
353
136
        free_oldmd(oldmd);
354
136
    }
355
356
136
    return 1;
357
136
}
358
359
int ossl_ht_flush(HT *h)
360
136
{
361
136
    return ossl_ht_flush_internal(h);
362
136
}
363
364
void ossl_ht_free(HT *h)
365
0
{
366
0
    int flush_ok;
367
368
0
    if (h == NULL)
369
0
        return;
370
371
0
    if (h->config.no_rcu) {
372
0
        free_oldmd(h->md);
373
0
    } else {
374
0
        ossl_ht_write_lock(h);
375
0
        flush_ok = ossl_ht_flush_internal(h);
376
0
        ossl_ht_write_unlock(h);
377
        /* Freeing the lock does a final sync for us */
378
0
        CRYPTO_THREAD_lock_free(h->atomic_lock);
379
0
        ossl_rcu_lock_free(h->lock);
380
0
        if (flush_ok) {
381
0
            OPENSSL_free(h->md->neighborhood_ptr_to_free);
382
0
            OPENSSL_free(h->md);
383
0
        } else {
384
0
            free_oldmd(h->md);
385
0
        }
386
0
    }
387
0
    OPENSSL_free(h);
388
0
}
389
390
size_t ossl_ht_count(HT *h)
391
43
{
392
43
    size_t count;
393
394
43
    count = h->wpd.value_count;
395
43
    return count;
396
43
}
397
398
void ossl_ht_foreach_until(HT *h, int (*cb)(HT_VALUE *obj, void *arg),
399
    void *arg)
400
0
{
401
0
    size_t i, j;
402
0
    struct ht_mutable_data_st *md;
403
404
0
    md = ossl_rcu_deref(&h->md);
405
0
    for (i = 0; i < md->neighborhood_mask + 1; i++) {
406
0
        PREFETCH_NEIGHBORHOOD(md->neighborhoods[i + 1]);
407
0
        for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
408
0
            if (md->neighborhoods[i].entries[j].value != NULL) {
409
0
                if (!cb((HT_VALUE *)md->neighborhoods[i].entries[j].value, arg))
410
0
                    goto out;
411
0
            }
412
0
        }
413
0
    }
414
0
out:
415
0
    return;
416
0
}
417
418
HT_VALUE_LIST *ossl_ht_filter(HT *h, size_t max_len,
419
    int (*filter)(HT_VALUE *obj, void *arg),
420
    void *arg)
421
0
{
422
0
    struct ht_mutable_data_st *md;
423
0
    HT_VALUE_LIST *list = OPENSSL_zalloc(sizeof(HT_VALUE_LIST)
424
0
        + (sizeof(HT_VALUE *) * max_len));
425
0
    size_t i, j;
426
0
    struct ht_internal_value_st *v;
427
428
0
    if (list == NULL)
429
0
        return NULL;
430
431
    /*
432
     * The list array lives just beyond the end of
433
     * the struct
434
     */
435
0
    list->list = (HT_VALUE **)(list + 1);
436
437
0
    md = ossl_rcu_deref(&h->md);
438
0
    for (i = 0; i < md->neighborhood_mask + 1; i++) {
439
0
        PREFETCH_NEIGHBORHOOD(md->neighborhoods[i + 1]);
440
0
        for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
441
0
            v = md->neighborhoods[i].entries[j].value;
442
0
            if (v != NULL && filter((HT_VALUE *)v, arg)) {
443
0
                list->list[list->list_len++] = (HT_VALUE *)v;
444
0
                if (list->list_len == max_len)
445
0
                    goto out;
446
0
            }
447
0
        }
448
0
    }
449
0
out:
450
0
    return list;
451
0
}
452
453
void ossl_ht_value_list_free(HT_VALUE_LIST *list)
454
0
{
455
0
    OPENSSL_free(list);
456
0
}
457
458
static int compare_hash(uint64_t hash1, uint64_t hash2)
459
3.14M
{
460
3.14M
    return (hash1 == hash2);
461
3.14M
}
462
463
static void free_old_neigh_table(void *arg)
464
1
{
465
1
    struct ht_mutable_data_st *oldmd = arg;
466
467
1
    OPENSSL_free(oldmd->neighborhood_ptr_to_free);
468
1
    OPENSSL_free(oldmd);
469
1
}
470
471
/*
472
 * Increase hash table bucket list
473
 * must be called with write_lock held
474
 */
475
static int grow_hashtable(HT *h, size_t oldsize)
476
1
{
477
1
    struct ht_mutable_data_st *newmd;
478
1
    struct ht_mutable_data_st *oldmd = ossl_rcu_deref(&h->md);
479
1
    CRYPTO_RCU_CB_ITEM *cbi = NULL;
480
1
    int rc = 0;
481
1
    uint64_t oldi, oldj, newi, newj;
482
1
    uint64_t oldhash;
483
1
    struct ht_internal_value_st *oldv;
484
1
    int rehashed;
485
1
    size_t newsize = oldsize * 2;
486
487
1
    if (h->config.lockless_reads)
488
0
        goto out;
489
490
1
    if ((newmd = OPENSSL_zalloc(sizeof(*newmd))) == NULL)
491
0
        goto out;
492
493
    /* bucket list is always a power of 2 */
494
1
    newmd->neighborhoods = alloc_new_neighborhood_list(oldsize * 2,
495
1
        &newmd->neighborhood_ptr_to_free);
496
1
    if (newmd->neighborhoods == NULL)
497
0
        goto out_free;
498
499
    /* being a power of 2 makes for easy mask computation */
500
1
    newmd->neighborhood_mask = (newsize - 1);
501
502
    /*
503
     * Now we need to start rehashing entries
504
     * Note we don't need to use atomics here as the new
505
     * mutable data hasn't been published
506
     */
507
17
    for (oldi = 0; oldi < h->wpd.neighborhood_len; oldi++) {
508
16
        PREFETCH_NEIGHBORHOOD(oldmd->neighborhoods[oldi + 1]);
509
80
        for (oldj = 0; oldj < NEIGHBORHOOD_LEN; oldj++) {
510
64
            oldv = oldmd->neighborhoods[oldi].entries[oldj].value;
511
64
            if (oldv == NULL)
512
17
                continue;
513
47
            oldhash = oldmd->neighborhoods[oldi].entries[oldj].hash;
514
47
            newi = oldhash & newmd->neighborhood_mask;
515
47
            rehashed = 0;
516
72
            for (newj = 0; newj < NEIGHBORHOOD_LEN; newj++) {
517
72
                if (newmd->neighborhoods[newi].entries[newj].value == NULL) {
518
47
                    newmd->neighborhoods[newi].entries[newj].value = oldv;
519
47
                    newmd->neighborhoods[newi].entries[newj].hash = oldhash;
520
47
                    rehashed = 1;
521
47
                    break;
522
47
                }
523
72
            }
524
47
            if (rehashed == 0) {
525
                /* we ran out of space in a neighborhood, grow again */
526
0
                OPENSSL_free(newmd->neighborhood_ptr_to_free);
527
0
                OPENSSL_free(newmd);
528
0
                return grow_hashtable(h, newsize);
529
0
            }
530
47
        }
531
16
    }
532
533
    /*
534
     * Pre allocate the rcu callback item before assigning the newmd.
535
     */
536
1
    if (!h->config.no_rcu) {
537
0
        cbi = ossl_rcu_cb_item_new();
538
0
        if (cbi == NULL)
539
0
            goto out_free;
540
0
    }
541
542
    /*
543
     * Now that our entries are all hashed into the new bucket list
544
     * update our bucket_len and target_max_load
545
     */
546
1
    h->wpd.neighborhood_len = newsize;
547
548
    /*
549
     * Now we replace the old mutable data with the new
550
     */
551
1
    if (!h->config.no_rcu) {
552
0
        ossl_rcu_assign_ptr(&h->md, &newmd);
553
0
        ossl_rcu_call(h->lock, cbi, free_old_neigh_table, oldmd);
554
0
        h->wpd.need_sync = 1;
555
1
    } else {
556
1
        h->md = newmd;
557
1
        free_old_neigh_table(oldmd);
558
1
    }
559
    /*
560
     * And we're done
561
     */
562
1
    rc = 1;
563
564
1
out:
565
1
    return rc;
566
0
out_free:
567
0
    OPENSSL_free(newmd->neighborhood_ptr_to_free);
568
0
    OPENSSL_free(newmd);
569
0
    goto out;
570
1
}
571
572
static void free_old_ht_value(void *arg)
573
0
{
574
0
    HT_VALUE *h = (HT_VALUE *)arg;
575
576
    /*
577
     * Note, this is only called on replacement,
578
     * the caller is responsible for freeing the
579
     * held data, we just need to free the wrapping
580
     * struct here
581
     */
582
0
    OPENSSL_free(h);
583
0
}
584
585
static ossl_inline int match_key(HT_KEY *a, HT_KEY *b)
586
1.41M
{
587
    /*
588
     * keys match if they are both present, the same size
589
     * and compare equal in memory
590
     */
591
1.41M
    PREFETCH(a->keybuf);
592
1.41M
    PREFETCH(b->keybuf);
593
1.41M
    if (a->keybuf != NULL && b->keybuf != NULL && a->keysize == b->keysize)
594
1.41M
        return !memcmp(a->keybuf, b->keybuf, a->keysize);
595
596
0
    return 1;
597
1.41M
}
598
599
static int ossl_ht_insert_locked(HT *h, uint64_t hash,
600
    struct ht_internal_value_st *newval,
601
    HT_VALUE **olddata)
602
811
{
603
811
    struct ht_mutable_data_st *md = h->md;
604
811
    uint64_t neigh_idx_start = hash & md->neighborhood_mask;
605
811
    uint64_t neigh_idx = neigh_idx_start;
606
811
    size_t j;
607
811
    uint64_t ihash;
608
811
    HT_VALUE *ival;
609
811
    size_t empty_idx = SIZE_MAX;
610
811
    int lockless_reads = h->config.lockless_reads;
611
612
812
    do {
613
812
        PREFETCH_NEIGHBORHOOD(md->neighborhoods[neigh_idx]);
614
615
1.40k
        for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
616
1.31k
            if (!h->config.no_rcu)
617
970
                ival = ossl_rcu_deref(&md->neighborhoods[neigh_idx].entries[j].value);
618
348
            else
619
348
                ival = (HT_VALUE *)md->neighborhoods[neigh_idx].entries[j].value;
620
1.31k
            if (ival == NULL) {
621
984
                empty_idx = j;
622
                /* lockless_reads implies no deletion, we can break out */
623
984
                if (lockless_reads)
624
724
                    goto not_found;
625
260
                continue;
626
984
            }
627
334
            if (!h->config.no_rcu) {
628
246
                if (!CRYPTO_atomic_load(&md->neighborhoods[neigh_idx].entries[j].hash,
629
246
                        &ihash, h->atomic_lock))
630
0
                    return 0;
631
246
            } else {
632
88
                ihash = md->neighborhoods[neigh_idx].entries[j].hash;
633
88
            }
634
334
            if (compare_hash(hash, ihash) && match_key(&newval->value.key, &ival->key)) {
635
0
                if (olddata == NULL) {
636
                    /* This would insert a duplicate -> fail */
637
0
                    return 0;
638
0
                }
639
                /* Do a replacement */
640
0
                if (!h->config.no_rcu) {
641
0
                    CRYPTO_RCU_CB_ITEM *cbi = ossl_rcu_cb_item_new();
642
0
                    if (cbi == NULL)
643
0
                        return 0;
644
0
                    if (!CRYPTO_atomic_store(&md->neighborhoods[neigh_idx].entries[j].hash,
645
0
                            hash, h->atomic_lock)) {
646
0
                        ossl_rcu_cb_item_free(cbi);
647
0
                        return 0;
648
0
                    }
649
0
                    *olddata = (HT_VALUE *)md->neighborhoods[neigh_idx].entries[j].value;
650
0
                    ossl_rcu_assign_ptr(&md->neighborhoods[neigh_idx].entries[j].value,
651
0
                        &newval);
652
0
                    ossl_rcu_call(h->lock, cbi, free_old_ht_value, *olddata);
653
0
                } else {
654
0
                    md->neighborhoods[neigh_idx].entries[j].hash = hash;
655
0
                    *olddata = (HT_VALUE *)md->neighborhoods[neigh_idx].entries[j].value;
656
0
                    md->neighborhoods[neigh_idx].entries[j].value = newval;
657
0
                }
658
0
                h->wpd.need_sync = 1;
659
0
                return 1;
660
0
            }
661
334
        }
662
88
        if (!lockless_reads)
663
87
            break;
664
        /* Continue search in subsequent neighborhoods */
665
1
        neigh_idx = (neigh_idx + 1) & md->neighborhood_mask;
666
1
    } while (neigh_idx != neigh_idx_start);
667
668
811
not_found:
669
    /* If we get to here, its just an insert */
670
811
    if (empty_idx == SIZE_MAX)
671
1
        return -1; /* out of space */
672
810
    if (!h->config.no_rcu) {
673
724
        if (!CRYPTO_atomic_store(&md->neighborhoods[neigh_idx].entries[empty_idx].hash,
674
724
                hash, h->atomic_lock))
675
0
            return 0;
676
724
        ossl_rcu_assign_ptr(&md->neighborhoods[neigh_idx].entries[empty_idx].value,
677
724
            &newval);
678
724
    } else {
679
86
        md->neighborhoods[neigh_idx].entries[empty_idx].hash = hash;
680
86
        md->neighborhoods[neigh_idx].entries[empty_idx].value = newval;
681
86
    }
682
810
    h->wpd.value_count++;
683
810
    return 1;
684
810
}
685
686
static struct ht_internal_value_st *alloc_new_value(HT *h, HT_KEY *key,
687
    void *data,
688
    uintptr_t *type)
689
810
{
690
810
    struct ht_internal_value_st *tmp;
691
810
    size_t nvsize = sizeof(*tmp);
692
693
810
    if (h->config.collision_check == 1)
694
810
        nvsize += key->keysize;
695
696
810
    tmp = OPENSSL_malloc(nvsize);
697
698
810
    if (tmp == NULL)
699
0
        return NULL;
700
701
810
    tmp->ht = h;
702
810
    tmp->value.value = data;
703
810
    tmp->value.type_id = type;
704
810
    tmp->value.key.keybuf = NULL;
705
810
    if (h->config.collision_check) {
706
810
        tmp->value.key.keybuf = (uint8_t *)(tmp + 1);
707
810
        tmp->value.key.keysize = key->keysize;
708
810
        memcpy(tmp->value.key.keybuf, key->keybuf, key->keysize);
709
810
    }
710
711
810
    return tmp;
712
810
}
713
714
static void free_value(struct ht_internal_value_st *v)
715
0
{
716
0
    OPENSSL_free(v);
717
0
}
718
719
int ossl_ht_insert(HT *h, HT_KEY *key, HT_VALUE *data, HT_VALUE **olddata)
720
810
{
721
810
    struct ht_internal_value_st *newval = NULL;
722
810
    uint64_t hash;
723
810
    int rc = 0;
724
810
    int i;
725
726
810
    if (data->value == NULL)
727
0
        goto out;
728
729
810
    rc = -1;
730
810
    newval = alloc_new_value(h, key, data->value, data->type_id);
731
810
    if (newval == NULL)
732
0
        goto out;
733
734
810
    if (key->cached_hash)
735
0
        hash = key->cached_hash;
736
810
    else
737
810
        hash = key->cached_hash = newval->value.key.cached_hash = h->config.ht_hash_fn(key);
738
739
    /*
740
     * we have to take our lock here to prevent other changes
741
     * to the bucket list
742
     */
743
810
    for (i = 0;
744
811
        (rc = ossl_ht_insert_locked(h, hash, newval, olddata)) == -1
745
1
        && i <= (int)NEIGHBORHOOD_LEN;
746
810
        ++i)
747
1
        if (!grow_hashtable(h, h->wpd.neighborhood_len)) {
748
0
            rc = -1;
749
0
            break;
750
0
        }
751
752
810
    if (rc <= 0)
753
0
        free_value(newval);
754
755
810
out:
756
810
    return rc;
757
810
}
758
759
HT_VALUE *ossl_ht_get(HT *h, HT_KEY *key)
760
1.42M
{
761
1.42M
    struct ht_mutable_data_st *md;
762
1.42M
    uint64_t hash;
763
1.42M
    uint64_t neigh_idx_start;
764
1.42M
    uint64_t neigh_idx;
765
1.42M
    struct ht_internal_value_st *ival = NULL;
766
1.42M
    size_t j;
767
1.42M
    uint64_t ehash;
768
1.42M
    int lockless_reads = h->config.lockless_reads;
769
770
1.42M
    if (key->cached_hash)
771
0
        hash = key->cached_hash;
772
1.42M
    else
773
1.42M
        hash = key->cached_hash = h->config.ht_hash_fn(key);
774
775
1.42M
    if (!h->config.no_rcu)
776
713k
        md = ossl_rcu_deref(&h->md);
777
708k
    else
778
708k
        md = h->md;
779
1.42M
    neigh_idx = neigh_idx_start = hash & md->neighborhood_mask;
780
1.42M
    do {
781
1.42M
        PREFETCH_NEIGHBORHOOD(md->neighborhoods[neigh_idx]);
782
3.20M
        for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
783
3.20M
            if (!h->config.no_rcu)
784
1.67M
                ival = ossl_rcu_deref(&md->neighborhoods[neigh_idx].entries[j].value);
785
1.52M
            else
786
1.52M
                ival = md->neighborhoods[neigh_idx].entries[j].value;
787
3.20M
            if (ival == NULL) {
788
54.1k
                if (lockless_reads)
789
                    /* lockless_reads implies no deletion, we can break out */
790
2.73k
                    return NULL;
791
51.4k
                continue;
792
54.1k
            }
793
3.14M
            if (!h->config.no_rcu) {
794
1.67M
                if (!CRYPTO_atomic_load(&md->neighborhoods[neigh_idx].entries[j].hash,
795
1.67M
                        &ehash, h->atomic_lock))
796
0
                    return NULL;
797
1.67M
            } else {
798
1.47M
                ehash = md->neighborhoods[neigh_idx].entries[j].hash;
799
1.47M
            }
800
3.14M
            if (compare_hash(hash, ehash) && match_key(&ival->value.key, key))
801
1.41M
                return (HT_VALUE *)ival;
802
3.14M
        }
803
44
        if (!lockless_reads)
804
41
            break;
805
        /* Continue search in subsequent neighborhoods */
806
3
        neigh_idx = (neigh_idx + 1) & md->neighborhood_mask;
807
3
    } while (neigh_idx != neigh_idx_start);
808
809
41
    return NULL;
810
1.42M
}
811
812
static void free_old_entry(void *arg)
813
0
{
814
0
    struct ht_internal_value_st *v = arg;
815
816
0
    v->ht->config.ht_free_fn((HT_VALUE *)v);
817
0
    free_value(v);
818
0
}
819
820
int ossl_ht_delete(HT *h, HT_KEY *key)
821
0
{
822
0
    uint64_t hash;
823
0
    uint64_t neigh_idx;
824
0
    size_t j;
825
0
    struct ht_internal_value_st *v = NULL;
826
0
    HT_VALUE *nv = NULL;
827
0
    int rc = 0;
828
829
0
    if (h->config.lockless_reads)
830
0
        return 0;
831
832
0
    if (key->cached_hash)
833
0
        hash = key->cached_hash;
834
0
    else
835
0
        hash = key->cached_hash = h->config.ht_hash_fn(key);
836
837
0
    neigh_idx = hash & h->md->neighborhood_mask;
838
0
    PREFETCH_NEIGHBORHOOD(h->md->neighborhoods[neigh_idx]);
839
0
    for (j = 0; j < NEIGHBORHOOD_LEN; j++) {
840
0
        v = (struct ht_internal_value_st *)h->md->neighborhoods[neigh_idx].entries[j].value;
841
0
        if (v == NULL)
842
0
            continue;
843
0
        if (compare_hash(hash, h->md->neighborhoods[neigh_idx].entries[j].hash)
844
0
            && match_key(key, &v->value.key)) {
845
0
            if (!h->config.no_rcu) {
846
0
                CRYPTO_RCU_CB_ITEM *cbi = ossl_rcu_cb_item_new();
847
0
                if (cbi == NULL)
848
0
                    break;
849
0
                if (!CRYPTO_atomic_store(&h->md->neighborhoods[neigh_idx].entries[j].hash,
850
0
                        0, h->atomic_lock)) {
851
0
                    ossl_rcu_cb_item_free(cbi);
852
0
                    break;
853
0
                }
854
0
                ossl_rcu_assign_ptr(&h->md->neighborhoods[neigh_idx].entries[j].value, &nv);
855
0
                ossl_rcu_call(h->lock, cbi, free_old_entry, v);
856
0
            } else {
857
0
                h->md->neighborhoods[neigh_idx].entries[j].hash = 0;
858
0
                h->md->neighborhoods[neigh_idx].entries[j].value = NULL;
859
0
                free_old_entry(v);
860
0
            }
861
0
            h->wpd.value_count--;
862
0
            h->wpd.need_sync = 1;
863
0
            rc = 1;
864
0
            break;
865
0
        }
866
0
    }
867
868
0
    return rc;
869
0
}
870
871
HT_VALUE *ossl_ht_deref_value(HT *h, HT_VALUE **val)
872
708k
{
873
708k
    HT_VALUE *v;
874
875
708k
    if (!h->config.no_rcu)
876
0
        v = ossl_rcu_deref(val);
877
708k
    else
878
708k
        v = *val;
879
880
708k
    return v;
881
708k
}
882
883
void *ossl_ht_inner_value(HT *h, HT_VALUE *v)
884
0
{
885
0
    void *inner;
886
887
0
    if (!h->config.no_rcu) {
888
0
        inner = v->value;
889
0
    } else {
890
0
        inner = v->value;
891
0
        OPENSSL_free(v);
892
0
    }
893
894
0
    return inner;
895
0
}