~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/tools/perf/util/hashmap.c

Version: ~ [ linux-6.11.5 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.58 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.114 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.169 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.228 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.284 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.322 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  1 // SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
  2 
  3 /*
  4  * Generic non-thread safe hash map implementation.
  5  *
  6  * Copyright (c) 2019 Facebook
  7  */
  8 #include <stdint.h>
  9 #include <stdlib.h>
 10 #include <stdio.h>
 11 #include <errno.h>
 12 #include <linux/err.h>
 13 #include "hashmap.h"
 14 
 15 /* make sure libbpf doesn't use kernel-only integer typedefs */
 16 #pragma GCC poison u8 u16 u32 u64 s8 s16 s32 s64
 17 
 18 /* prevent accidental re-addition of reallocarray() */
 19 #pragma GCC poison reallocarray
 20 
 21 /* start with 4 buckets */
 22 #define HASHMAP_MIN_CAP_BITS 2
 23 
 24 static void hashmap_add_entry(struct hashmap_entry **pprev,
 25                               struct hashmap_entry *entry)
 26 {
 27         entry->next = *pprev;
 28         *pprev = entry;
 29 }
 30 
 31 static void hashmap_del_entry(struct hashmap_entry **pprev,
 32                               struct hashmap_entry *entry)
 33 {
 34         *pprev = entry->next;
 35         entry->next = NULL;
 36 }
 37 
 38 void hashmap__init(struct hashmap *map, hashmap_hash_fn hash_fn,
 39                    hashmap_equal_fn equal_fn, void *ctx)
 40 {
 41         map->hash_fn = hash_fn;
 42         map->equal_fn = equal_fn;
 43         map->ctx = ctx;
 44 
 45         map->buckets = NULL;
 46         map->cap = 0;
 47         map->cap_bits = 0;
 48         map->sz = 0;
 49 }
 50 
 51 struct hashmap *hashmap__new(hashmap_hash_fn hash_fn,
 52                              hashmap_equal_fn equal_fn,
 53                              void *ctx)
 54 {
 55         struct hashmap *map = malloc(sizeof(struct hashmap));
 56 
 57         if (!map)
 58                 return ERR_PTR(-ENOMEM);
 59         hashmap__init(map, hash_fn, equal_fn, ctx);
 60         return map;
 61 }
 62 
 63 void hashmap__clear(struct hashmap *map)
 64 {
 65         struct hashmap_entry *cur, *tmp;
 66         size_t bkt;
 67 
 68         hashmap__for_each_entry_safe(map, cur, tmp, bkt) {
 69                 free(cur);
 70         }
 71         free(map->buckets);
 72         map->buckets = NULL;
 73         map->cap = map->cap_bits = map->sz = 0;
 74 }
 75 
 76 void hashmap__free(struct hashmap *map)
 77 {
 78         if (IS_ERR_OR_NULL(map))
 79                 return;
 80 
 81         hashmap__clear(map);
 82         free(map);
 83 }
 84 
 85 size_t hashmap__size(const struct hashmap *map)
 86 {
 87         return map->sz;
 88 }
 89 
 90 size_t hashmap__capacity(const struct hashmap *map)
 91 {
 92         return map->cap;
 93 }
 94 
 95 static bool hashmap_needs_to_grow(struct hashmap *map)
 96 {
 97         /* grow if empty or more than 75% filled */
 98         return (map->cap == 0) || ((map->sz + 1) * 4 / 3 > map->cap);
 99 }
100 
101 static int hashmap_grow(struct hashmap *map)
102 {
103         struct hashmap_entry **new_buckets;
104         struct hashmap_entry *cur, *tmp;
105         size_t new_cap_bits, new_cap;
106         size_t h, bkt;
107 
108         new_cap_bits = map->cap_bits + 1;
109         if (new_cap_bits < HASHMAP_MIN_CAP_BITS)
110                 new_cap_bits = HASHMAP_MIN_CAP_BITS;
111 
112         new_cap = 1UL << new_cap_bits;
113         new_buckets = calloc(new_cap, sizeof(new_buckets[0]));
114         if (!new_buckets)
115                 return -ENOMEM;
116 
117         hashmap__for_each_entry_safe(map, cur, tmp, bkt) {
118                 h = hash_bits(map->hash_fn(cur->key, map->ctx), new_cap_bits);
119                 hashmap_add_entry(&new_buckets[h], cur);
120         }
121 
122         map->cap = new_cap;
123         map->cap_bits = new_cap_bits;
124         free(map->buckets);
125         map->buckets = new_buckets;
126 
127         return 0;
128 }
129 
130 static bool hashmap_find_entry(const struct hashmap *map,
131                                const long key, size_t hash,
132                                struct hashmap_entry ***pprev,
133                                struct hashmap_entry **entry)
134 {
135         struct hashmap_entry *cur, **prev_ptr;
136 
137         if (!map->buckets)
138                 return false;
139 
140         for (prev_ptr = &map->buckets[hash], cur = *prev_ptr;
141              cur;
142              prev_ptr = &cur->next, cur = cur->next) {
143                 if (map->equal_fn(cur->key, key, map->ctx)) {
144                         if (pprev)
145                                 *pprev = prev_ptr;
146                         *entry = cur;
147                         return true;
148                 }
149         }
150 
151         return false;
152 }
153 
154 int hashmap_insert(struct hashmap *map, long key, long value,
155                    enum hashmap_insert_strategy strategy,
156                    long *old_key, long *old_value)
157 {
158         struct hashmap_entry *entry;
159         size_t h;
160         int err;
161 
162         if (old_key)
163                 *old_key = 0;
164         if (old_value)
165                 *old_value = 0;
166 
167         h = hash_bits(map->hash_fn(key, map->ctx), map->cap_bits);
168         if (strategy != HASHMAP_APPEND &&
169             hashmap_find_entry(map, key, h, NULL, &entry)) {
170                 if (old_key)
171                         *old_key = entry->key;
172                 if (old_value)
173                         *old_value = entry->value;
174 
175                 if (strategy == HASHMAP_SET || strategy == HASHMAP_UPDATE) {
176                         entry->key = key;
177                         entry->value = value;
178                         return 0;
179                 } else if (strategy == HASHMAP_ADD) {
180                         return -EEXIST;
181                 }
182         }
183 
184         if (strategy == HASHMAP_UPDATE)
185                 return -ENOENT;
186 
187         if (hashmap_needs_to_grow(map)) {
188                 err = hashmap_grow(map);
189                 if (err)
190                         return err;
191                 h = hash_bits(map->hash_fn(key, map->ctx), map->cap_bits);
192         }
193 
194         entry = malloc(sizeof(struct hashmap_entry));
195         if (!entry)
196                 return -ENOMEM;
197 
198         entry->key = key;
199         entry->value = value;
200         hashmap_add_entry(&map->buckets[h], entry);
201         map->sz++;
202 
203         return 0;
204 }
205 
206 bool hashmap_find(const struct hashmap *map, long key, long *value)
207 {
208         struct hashmap_entry *entry;
209         size_t h;
210 
211         h = hash_bits(map->hash_fn(key, map->ctx), map->cap_bits);
212         if (!hashmap_find_entry(map, key, h, NULL, &entry))
213                 return false;
214 
215         if (value)
216                 *value = entry->value;
217         return true;
218 }
219 
220 bool hashmap_delete(struct hashmap *map, long key,
221                     long *old_key, long *old_value)
222 {
223         struct hashmap_entry **pprev, *entry;
224         size_t h;
225 
226         h = hash_bits(map->hash_fn(key, map->ctx), map->cap_bits);
227         if (!hashmap_find_entry(map, key, h, &pprev, &entry))
228                 return false;
229 
230         if (old_key)
231                 *old_key = entry->key;
232         if (old_value)
233                 *old_value = entry->value;
234 
235         hashmap_del_entry(pprev, entry);
236         free(entry);
237         map->sz--;
238 
239         return true;
240 }
241 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php