xref: /aosp_15_r20/external/llvm-libc/src/search/hsearch.cpp (revision 71db0c75aadcf003ffe3238005f61d7618a3fead)
1*71db0c75SAndroid Build Coastguard Worker //===-- Implementation of hsearch -------------------------------*- C++ -*-===//
2*71db0c75SAndroid Build Coastguard Worker //
3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information.
5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*71db0c75SAndroid Build Coastguard Worker //
7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
8*71db0c75SAndroid Build Coastguard Worker 
9*71db0c75SAndroid Build Coastguard Worker #include "src/search/hsearch.h"
10*71db0c75SAndroid Build Coastguard Worker #include "src/__support/HashTable/randomness.h"
11*71db0c75SAndroid Build Coastguard Worker #include "src/__support/HashTable/table.h"
12*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/config.h"
13*71db0c75SAndroid Build Coastguard Worker #include "src/errno/libc_errno.h"
14*71db0c75SAndroid Build Coastguard Worker #include "src/search/hsearch/global.h"
15*71db0c75SAndroid Build Coastguard Worker 
16*71db0c75SAndroid Build Coastguard Worker namespace LIBC_NAMESPACE_DECL {
17*71db0c75SAndroid Build Coastguard Worker LLVM_LIBC_FUNCTION(ENTRY *, hsearch, (ENTRY item, ACTION action)) {
18*71db0c75SAndroid Build Coastguard Worker   ENTRY *result = nullptr;
19*71db0c75SAndroid Build Coastguard Worker   if (internal::global_hash_table == nullptr) {
20*71db0c75SAndroid Build Coastguard Worker     // If global_hash_table is null, we create a new hash table with a minimal
21*71db0c75SAndroid Build Coastguard Worker     // capacity. Such hashtable will be expanded as needed.
22*71db0c75SAndroid Build Coastguard Worker     uint64_t randomness = internal::randomness::next_random_seed();
23*71db0c75SAndroid Build Coastguard Worker     internal::global_hash_table = internal::HashTable::allocate(0, randomness);
24*71db0c75SAndroid Build Coastguard Worker   }
25*71db0c75SAndroid Build Coastguard Worker 
26*71db0c75SAndroid Build Coastguard Worker   // In rare cases, the global hashtable may still fail to allocate. We treat it
27*71db0c75SAndroid Build Coastguard Worker   // as ESRCH or ENOMEM depending on the action.
28*71db0c75SAndroid Build Coastguard Worker   switch (action) {
29*71db0c75SAndroid Build Coastguard Worker   case FIND:
30*71db0c75SAndroid Build Coastguard Worker     result = internal::global_hash_table
31*71db0c75SAndroid Build Coastguard Worker                  ? internal::global_hash_table->find(item.key)
32*71db0c75SAndroid Build Coastguard Worker                  : nullptr;
33*71db0c75SAndroid Build Coastguard Worker     if (result == nullptr) {
34*71db0c75SAndroid Build Coastguard Worker       libc_errno = ESRCH;
35*71db0c75SAndroid Build Coastguard Worker     }
36*71db0c75SAndroid Build Coastguard Worker     break;
37*71db0c75SAndroid Build Coastguard Worker   case ENTER:
38*71db0c75SAndroid Build Coastguard Worker     result =
39*71db0c75SAndroid Build Coastguard Worker         internal::global_hash_table
40*71db0c75SAndroid Build Coastguard Worker             ? internal::HashTable::insert(internal::global_hash_table, item)
41*71db0c75SAndroid Build Coastguard Worker             : nullptr;
42*71db0c75SAndroid Build Coastguard Worker     if (result == nullptr) {
43*71db0c75SAndroid Build Coastguard Worker       libc_errno = ENOMEM;
44*71db0c75SAndroid Build Coastguard Worker     }
45*71db0c75SAndroid Build Coastguard Worker     break;
46*71db0c75SAndroid Build Coastguard Worker   }
47*71db0c75SAndroid Build Coastguard Worker   return result;
48*71db0c75SAndroid Build Coastguard Worker }
49*71db0c75SAndroid Build Coastguard Worker 
50*71db0c75SAndroid Build Coastguard Worker } // namespace LIBC_NAMESPACE_DECL
51