blob: 75015d59a4ad3fc9383c8117833e70df391e4093 [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15/*
16 Copyright (c) 2001, 2002, 2003 Eliot Dresselhaus
17
18 Permission is hereby granted, free of charge, to any person obtaining
19 a copy of this software and associated documentation files (the
20 "Software"), to deal in the Software without restriction, including
21 without limitation the rights to use, copy, modify, merge, publish,
22 distribute, sublicense, and/or sell copies of the Software, and to
23 permit persons to whom the Software is furnished to do so, subject to
24 the following conditions:
25
26 The above copyright notice and this permission notice shall be
27 included in all copies or substantial portions of the Software.
28
29 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
30 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
31 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
32 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
33 LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
34 OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
35 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
36*/
37
38#ifndef _included_clib_mem_h
39#define _included_clib_mem_h
40
41#include <stdarg.h>
Damjan Marion01914ce2017-09-14 19:04:50 +020042#include <unistd.h>
43#include <sys/mman.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070044
Dave Barachc3799992016-08-15 11:12:27 -040045#include <vppinfra/clib.h> /* uword, etc */
Damjan Marion01914ce2017-09-14 19:04:50 +020046#include <vppinfra/clib_error.h>
Dave Barach6a5adc32018-07-04 10:56:23 -040047
Ed Warnickecb9cada2015-12-08 15:45:58 -070048#include <vppinfra/os.h>
Dave Barachb7b92992018-10-17 10:38:51 -040049#include <vppinfra/string.h> /* memcpy, clib_memset */
Damjan Marion79934e82022-04-05 12:40:31 +020050#ifdef CLIB_SANITIZE_ADDR
51#include <sanitizer/asan_interface.h>
52#endif
Ed Warnickecb9cada2015-12-08 15:45:58 -070053
Damjan Marionce8debf2016-02-06 19:16:21 +010054#define CLIB_MAX_MHEAPS 256
Damjan Marion6bfd0762020-09-11 22:16:53 +020055#define CLIB_MAX_NUMAS 16
56#define CLIB_MEM_VM_MAP_FAILED ((void *) ~0)
Damjan Marion561ae5d2020-09-24 13:53:46 +020057#define CLIB_MEM_ERROR (-1)
Damjan Marion299571a2022-03-19 00:07:52 +010058#define CLIB_MEM_LOG2_MIN_ALIGN (3)
59#define CLIB_MEM_MIN_ALIGN (1 << CLIB_MEM_LOG2_MIN_ALIGN)
Dave Baracha690fdb2020-01-21 12:34:55 -050060
Damjan Marionb5095042020-09-11 22:13:46 +020061typedef enum
62{
63 CLIB_MEM_PAGE_SZ_UNKNOWN = 0,
64 CLIB_MEM_PAGE_SZ_DEFAULT = 1,
65 CLIB_MEM_PAGE_SZ_DEFAULT_HUGE = 2,
66 CLIB_MEM_PAGE_SZ_4K = 12,
67 CLIB_MEM_PAGE_SZ_16K = 14,
68 CLIB_MEM_PAGE_SZ_64K = 16,
69 CLIB_MEM_PAGE_SZ_1M = 20,
70 CLIB_MEM_PAGE_SZ_2M = 21,
71 CLIB_MEM_PAGE_SZ_16M = 24,
72 CLIB_MEM_PAGE_SZ_32M = 25,
73 CLIB_MEM_PAGE_SZ_512M = 29,
74 CLIB_MEM_PAGE_SZ_1G = 30,
75 CLIB_MEM_PAGE_SZ_16G = 34,
76} clib_mem_page_sz_t;
77
Damjan Marion6bfd0762020-09-11 22:16:53 +020078typedef struct _clib_mem_vm_map_hdr
79{
80 /* base address */
81 uword base_addr;
82
83 /* number of pages */
84 uword num_pages;
85
86 /* page size (log2) */
87 clib_mem_page_sz_t log2_page_sz;
88
Damjan Marion5ef25162020-09-17 13:29:33 +020089 /* file descriptor, -1 if memory is not shared */
90 int fd;
91
Damjan Marion6bfd0762020-09-11 22:16:53 +020092 /* allocation mame */
93#define CLIB_VM_MAP_HDR_NAME_MAX_LEN 64
94 char name[CLIB_VM_MAP_HDR_NAME_MAX_LEN];
95
96 /* linked list */
97 struct _clib_mem_vm_map_hdr *prev, *next;
98} clib_mem_vm_map_hdr_t;
99
Damjan Marion299571a2022-03-19 00:07:52 +0100100#define foreach_clib_mem_heap_flag \
101 _ (0, LOCKED, "locked") \
102 _ (1, UNMAP_ON_DESTROY, "unmap-on-destroy") \
103 _ (2, TRACED, "traced")
Damjan Marionbfa75d62020-10-06 17:46:06 +0200104
105typedef enum
106{
107#define _(i, v, s) CLIB_MEM_HEAP_F_##v = (1 << i),
108 foreach_clib_mem_heap_flag
109#undef _
110} clib_mem_heap_flag_t;
111
112typedef struct
113{
114 /* base address */
115 void *base;
116
117 /* dlmalloc mspace */
118 void *mspace;
119
120 /* heap size */
121 uword size;
122
123 /* page size (log2) */
124 clib_mem_page_sz_t log2_page_sz:8;
125
126 /* flags */
127 clib_mem_heap_flag_t flags:8;
128
129 /* name - _MUST_ be last */
130 char name[0];
131} clib_mem_heap_t;
132
Damjan Marion57d1ec02020-09-16 21:15:44 +0200133typedef struct
134{
Damjan Marionc63e2a42020-09-16 21:36:00 +0200135 /* log2 system page size */
136 clib_mem_page_sz_t log2_page_sz;
137
Damjan Marionff011b22021-09-21 11:38:04 +0200138 /* log2 default hugepage size */
Damjan Marionc63e2a42020-09-16 21:36:00 +0200139 clib_mem_page_sz_t log2_default_hugepage_sz;
140
Damjan Marionff011b22021-09-21 11:38:04 +0200141 /* log2 system default hugepage size */
142 clib_mem_page_sz_t log2_sys_default_hugepage_sz;
143
Damjan Marionc63e2a42020-09-16 21:36:00 +0200144 /* bitmap of available numa nodes */
145 u32 numa_node_bitmap;
146
Damjan Marion57d1ec02020-09-16 21:15:44 +0200147 /* per CPU heaps */
148 void *per_cpu_mheaps[CLIB_MAX_MHEAPS];
149
150 /* per NUMA heaps */
151 void *per_numa_mheaps[CLIB_MAX_NUMAS];
Damjan Marion6bfd0762020-09-11 22:16:53 +0200152
153 /* memory maps */
154 clib_mem_vm_map_hdr_t *first_map, *last_map;
Damjan Marionbdbb0c52020-09-17 10:40:44 +0200155
Damjan Marion70ae0a92020-10-26 10:39:30 +0100156 /* map lock */
157 u8 map_lock;
158
Damjan Marionbdbb0c52020-09-17 10:40:44 +0200159 /* last error */
160 clib_error_t *error;
Damjan Marion57d1ec02020-09-16 21:15:44 +0200161} clib_mem_main_t;
162
163extern clib_mem_main_t clib_mem_main;
164
Dave Baracha690fdb2020-01-21 12:34:55 -0500165/* Unspecified NUMA socket */
166#define VEC_NUMA_UNSPECIFIED (0xFF)
Damjan Marionce8debf2016-02-06 19:16:21 +0100167
Damjan Marion79934e82022-04-05 12:40:31 +0200168static_always_inline void
169clib_mem_poison (const void volatile *p, uword s)
170{
171#ifdef CLIB_SANITIZE_ADDR
172 ASAN_POISON_MEMORY_REGION (p, s);
173#endif
174}
175
176static_always_inline void
177clib_mem_unpoison (const void volatile *p, uword s)
178{
179#ifdef CLIB_SANITIZE_ADDR
180 ASAN_UNPOISON_MEMORY_REGION (p, s);
181#endif
182}
183
Damjan Marionbfa75d62020-10-06 17:46:06 +0200184always_inline clib_mem_heap_t *
Dave Baracha690fdb2020-01-21 12:34:55 -0500185clib_mem_get_per_cpu_heap (void)
186{
187 int cpu = os_get_thread_index ();
Damjan Marion57d1ec02020-09-16 21:15:44 +0200188 return clib_mem_main.per_cpu_mheaps[cpu];
Dave Baracha690fdb2020-01-21 12:34:55 -0500189}
190
191always_inline void *
Damjan Marionbfa75d62020-10-06 17:46:06 +0200192clib_mem_set_per_cpu_heap (void *new_heap)
Dave Baracha690fdb2020-01-21 12:34:55 -0500193{
194 int cpu = os_get_thread_index ();
Damjan Marion57d1ec02020-09-16 21:15:44 +0200195 void *old = clib_mem_main.per_cpu_mheaps[cpu];
196 clib_mem_main.per_cpu_mheaps[cpu] = new_heap;
Dave Baracha690fdb2020-01-21 12:34:55 -0500197 return old;
198}
199
200always_inline void *
201clib_mem_get_per_numa_heap (u32 numa_id)
202{
Damjan Marion57d1ec02020-09-16 21:15:44 +0200203 ASSERT (numa_id < ARRAY_LEN (clib_mem_main.per_numa_mheaps));
204 return clib_mem_main.per_numa_mheaps[numa_id];
Dave Baracha690fdb2020-01-21 12:34:55 -0500205}
206
207always_inline void *
Damjan Marionbfa75d62020-10-06 17:46:06 +0200208clib_mem_set_per_numa_heap (void *new_heap)
Dave Baracha690fdb2020-01-21 12:34:55 -0500209{
210 int numa = os_get_numa_index ();
Damjan Marion57d1ec02020-09-16 21:15:44 +0200211 void *old = clib_mem_main.per_numa_mheaps[numa];
212 clib_mem_main.per_numa_mheaps[numa] = new_heap;
Dave Baracha690fdb2020-01-21 12:34:55 -0500213 return old;
214}
Ed Warnickecb9cada2015-12-08 15:45:58 -0700215
Nathan Skrzypczakd516ca42019-08-01 18:14:06 +0200216always_inline void
217clib_mem_set_thread_index (void)
218{
219 /*
220 * Find an unused slot in the per-cpu-mheaps array,
221 * and grab it for this thread. We need to be able to
222 * push/pop the thread heap without affecting other thread(s).
223 */
224 int i;
225 if (__os_thread_index != 0)
226 return;
Damjan Marion57d1ec02020-09-16 21:15:44 +0200227 for (i = 0; i < ARRAY_LEN (clib_mem_main.per_cpu_mheaps); i++)
228 if (clib_atomic_bool_cmp_and_swap (&clib_mem_main.per_cpu_mheaps[i],
229 0, clib_mem_main.per_cpu_mheaps[0]))
Nathan Skrzypczakd516ca42019-08-01 18:14:06 +0200230 {
231 os_set_thread_index (i);
232 break;
233 }
234 ASSERT (__os_thread_index > 0);
235}
236
Dave Barach241e5222016-10-13 10:53:26 -0400237/* Memory allocator which calls os_out_of_memory() when it fails */
Damjan Marion299571a2022-03-19 00:07:52 +0100238void *clib_mem_alloc (uword size);
239void *clib_mem_alloc_aligned (uword size, uword align);
240void *clib_mem_alloc_or_null (uword size);
241void *clib_mem_alloc_aligned_or_null (uword size, uword align);
242void *clib_mem_realloc (void *p, uword new_size);
243void *clib_mem_realloc_aligned (void *p, uword new_size, uword align);
Damjan Marion24738582022-03-31 15:12:20 +0200244uword clib_mem_is_heap_object (void *p);
245void clib_mem_free (void *p);
246
247void *clib_mem_heap_alloc (void *heap, uword size);
248void *clib_mem_heap_alloc_aligned (void *heap, uword size, uword align);
249void *clib_mem_heap_alloc_or_null (void *heap, uword size);
250void *clib_mem_heap_alloc_aligned_or_null (void *heap, uword size,
251 uword align);
252void *clib_mem_heap_realloc (void *heap, void *p, uword new_size);
253void *clib_mem_heap_realloc_aligned (void *heap, void *p, uword new_size,
254 uword align);
255uword clib_mem_heap_is_heap_object (void *heap, void *p);
256void clib_mem_heap_free (void *heap, void *p);
257
258uword clib_mem_size (void *p);
259void clib_mem_free_s (void *p);
Dave Barach241e5222016-10-13 10:53:26 -0400260
Ed Warnickecb9cada2015-12-08 15:45:58 -0700261/* Memory allocator which panics when it fails.
262 Use macro so that clib_panic macro can expand __FUNCTION__ and __LINE__. */
263#define clib_mem_alloc_aligned_no_fail(size,align) \
264({ \
265 uword _clib_mem_alloc_size = (size); \
266 void * _clib_mem_alloc_p; \
267 _clib_mem_alloc_p = clib_mem_alloc_aligned (_clib_mem_alloc_size, (align)); \
268 if (! _clib_mem_alloc_p) \
269 clib_panic ("failed to allocate %d bytes", _clib_mem_alloc_size); \
270 _clib_mem_alloc_p; \
271})
272
273#define clib_mem_alloc_no_fail(size) clib_mem_alloc_aligned_no_fail(size,1)
274
275/* Alias to stack allocator for naming consistency. */
276#define clib_mem_alloc_stack(bytes) __builtin_alloca(bytes)
277
Damjan Marionbfa75d62020-10-06 17:46:06 +0200278always_inline clib_mem_heap_t *
Dave Barachc3799992016-08-15 11:12:27 -0400279clib_mem_get_heap (void)
280{
281 return clib_mem_get_per_cpu_heap ();
282}
Ed Warnickecb9cada2015-12-08 15:45:58 -0700283
Damjan Marionbfa75d62020-10-06 17:46:06 +0200284always_inline clib_mem_heap_t *
285clib_mem_set_heap (clib_mem_heap_t * heap)
Dave Barachc3799992016-08-15 11:12:27 -0400286{
287 return clib_mem_set_per_cpu_heap (heap);
288}
Ed Warnickecb9cada2015-12-08 15:45:58 -0700289
Damjan Marionbfa75d62020-10-06 17:46:06 +0200290void clib_mem_destroy_heap (clib_mem_heap_t * heap);
291clib_mem_heap_t *clib_mem_create_heap (void *base, uword size, int is_locked,
292 char *fmt, ...);
Damjan Marion4537c302020-09-28 19:03:37 +0200293
Damjan Marionc63e2a42020-09-16 21:36:00 +0200294void clib_mem_main_init ();
Damjan Marionbfa75d62020-10-06 17:46:06 +0200295void *clib_mem_init (void *base, uword size);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200296void *clib_mem_init_with_page_size (uword memory_size,
297 clib_mem_page_sz_t log2_page_sz);
Dave Barach6a5adc32018-07-04 10:56:23 -0400298void *clib_mem_init_thread_safe (void *memory, uword memory_size);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700299
300void clib_mem_exit (void);
301
Ed Warnickecb9cada2015-12-08 15:45:58 -0700302void clib_mem_trace (int enable);
303
Dave Barachd67a4282019-06-15 12:46:13 -0400304int clib_mem_is_traced (void);
305
Dave Barachc3799992016-08-15 11:12:27 -0400306typedef struct
307{
Ed Warnickecb9cada2015-12-08 15:45:58 -0700308 /* Total number of objects allocated. */
309 uword object_count;
310
311 /* Total allocated bytes. Bytes used and free.
312 used + free = total */
313 uword bytes_total, bytes_used, bytes_free;
314
315 /* Number of bytes used by mheap data structure overhead
316 (e.g. free lists, mheap header). */
317 uword bytes_overhead;
318
319 /* Amount of free space returned to operating system. */
320 uword bytes_free_reclaimed;
Dave Barachc3799992016-08-15 11:12:27 -0400321
Ed Warnickecb9cada2015-12-08 15:45:58 -0700322 /* For malloc which puts small objects in sbrk region and
323 large objects in mmap'ed regions. */
324 uword bytes_used_sbrk;
325 uword bytes_used_mmap;
326
327 /* Max. number of bytes in this heap. */
328 uword bytes_max;
329} clib_mem_usage_t;
330
Damjan Marionbfa75d62020-10-06 17:46:06 +0200331void clib_mem_get_heap_usage (clib_mem_heap_t * heap,
332 clib_mem_usage_t * usage);
Damjan Marion4537c302020-09-28 19:03:37 +0200333
Damjan Marionbfa75d62020-10-06 17:46:06 +0200334void *clib_mem_get_heap_base (clib_mem_heap_t * heap);
335uword clib_mem_get_heap_size (clib_mem_heap_t * heap);
336uword clib_mem_get_heap_free_space (clib_mem_heap_t * heap);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700337
Dave Barachc3799992016-08-15 11:12:27 -0400338u8 *format_clib_mem_usage (u8 * s, va_list * args);
Damjan Marion4537c302020-09-28 19:03:37 +0200339u8 *format_clib_mem_heap (u8 * s, va_list * va);
Damjan Marionbfa75d62020-10-06 17:46:06 +0200340u8 *format_clib_mem_page_stats (u8 * s, va_list * va);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700341
Damjan Marion01914ce2017-09-14 19:04:50 +0200342/* Allocate virtual address space. */
343always_inline void *
344clib_mem_vm_alloc (uword size)
345{
346 void *mmap_addr;
347 uword flags = MAP_PRIVATE;
348
349#ifdef MAP_ANONYMOUS
350 flags |= MAP_ANONYMOUS;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700351#endif
352
Damjan Marion01914ce2017-09-14 19:04:50 +0200353 mmap_addr = mmap (0, size, PROT_READ | PROT_WRITE, flags, -1, 0);
354 if (mmap_addr == (void *) -1)
355 mmap_addr = 0;
Benoît Ganne1557d9a2020-02-07 11:58:16 +0100356 else
Damjan Marion79934e82022-04-05 12:40:31 +0200357 clib_mem_unpoison (mmap_addr, size);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700358
Damjan Marion01914ce2017-09-14 19:04:50 +0200359 return mmap_addr;
360}
361
362always_inline void
363clib_mem_vm_free (void *addr, uword size)
364{
365 munmap (addr, size);
366}
367
Damjan Marion6bfd0762020-09-11 22:16:53 +0200368void *clib_mem_vm_map_internal (void *base, clib_mem_page_sz_t log2_page_sz,
369 uword size, int fd, uword offset, char *name);
Damjan Marion01914ce2017-09-14 19:04:50 +0200370
Damjan Marion6bfd0762020-09-11 22:16:53 +0200371void *clib_mem_vm_map (void *start, uword size,
372 clib_mem_page_sz_t log2_page_size, char *fmt, ...);
373void *clib_mem_vm_map_stack (uword size, clib_mem_page_sz_t log2_page_size,
374 char *fmt, ...);
375void *clib_mem_vm_map_shared (void *start, uword size, int fd, uword offset,
376 char *fmt, ...);
377int clib_mem_vm_unmap (void *base);
378clib_mem_vm_map_hdr_t *clib_mem_vm_get_next_map_hdr (clib_mem_vm_map_hdr_t *
379 hdr);
Damjan Marion01914ce2017-09-14 19:04:50 +0200380
Damjan Marion6bfd0762020-09-11 22:16:53 +0200381static_always_inline clib_mem_page_sz_t
382clib_mem_get_log2_page_size (void)
383{
384 return clib_mem_main.log2_page_sz;
385}
386
387static_always_inline uword
388clib_mem_get_page_size (void)
389{
390 return 1ULL << clib_mem_main.log2_page_sz;
391}
392
Damjan Marionff011b22021-09-21 11:38:04 +0200393static_always_inline void
394clib_mem_set_log2_default_hugepage_size (clib_mem_page_sz_t log2_page_sz)
395{
396 clib_mem_main.log2_default_hugepage_sz = log2_page_sz;
397}
398
Damjan Marion6bfd0762020-09-11 22:16:53 +0200399static_always_inline clib_mem_page_sz_t
400clib_mem_get_log2_default_hugepage_size ()
401{
402 return clib_mem_main.log2_default_hugepage_sz;
403}
404
Damjan Marionff011b22021-09-21 11:38:04 +0200405static_always_inline uword
406clib_mem_get_default_hugepage_size (void)
407{
408 return 1ULL << clib_mem_main.log2_default_hugepage_sz;
409}
410
Damjan Marionbdbb0c52020-09-17 10:40:44 +0200411int clib_mem_vm_create_fd (clib_mem_page_sz_t log2_page_size, char *fmt, ...);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200412uword clib_mem_get_fd_page_size (int fd);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200413clib_mem_page_sz_t clib_mem_get_fd_log2_page_size (int fd);
Damjan Marionb5095042020-09-11 22:13:46 +0200414uword clib_mem_vm_reserve (uword start, uword size,
415 clib_mem_page_sz_t log2_page_sz);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200416u64 *clib_mem_vm_get_paddr (void *mem, clib_mem_page_sz_t log2_page_size,
417 int n_pages);
Dave Barach2b793412020-08-28 10:39:00 -0400418void clib_mem_destroy (void);
Damjan Marion561ae5d2020-09-24 13:53:46 +0200419int clib_mem_set_numa_affinity (u8 numa_node, int force);
420int clib_mem_set_default_numa_affinity ();
Damjan Marionb5095042020-09-11 22:13:46 +0200421void clib_mem_vm_randomize_va (uword * requested_va,
422 clib_mem_page_sz_t log2_page_size);
Damjan Marionbfa75d62020-10-06 17:46:06 +0200423void mheap_trace (clib_mem_heap_t * v, int enable);
Dave Barach8fdde3c2019-05-17 10:46:40 -0400424uword clib_mem_trace_enable_disable (uword enable);
425void clib_mem_trace (int enable);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700426
Damjan Marion6bfd0762020-09-11 22:16:53 +0200427always_inline uword
428clib_mem_round_to_page_size (uword size, clib_mem_page_sz_t log2_page_size)
429{
430 ASSERT (log2_page_size != CLIB_MEM_PAGE_SZ_UNKNOWN);
431
432 if (log2_page_size == CLIB_MEM_PAGE_SZ_DEFAULT)
433 log2_page_size = clib_mem_get_log2_page_size ();
434 else if (log2_page_size == CLIB_MEM_PAGE_SZ_DEFAULT_HUGE)
435 log2_page_size = clib_mem_get_log2_default_hugepage_size ();
436
437 return round_pow2 (size, 1ULL << log2_page_size);
438}
439
440typedef struct
441{
Damjan Marionbfa75d62020-10-06 17:46:06 +0200442 clib_mem_page_sz_t log2_page_sz;
443 uword total;
Damjan Marion6bfd0762020-09-11 22:16:53 +0200444 uword mapped;
445 uword not_mapped;
446 uword per_numa[CLIB_MAX_NUMAS];
447 uword unknown;
448} clib_mem_page_stats_t;
449
450void clib_mem_get_page_stats (void *start, clib_mem_page_sz_t log2_page_size,
451 uword n_pages, clib_mem_page_stats_t * stats);
452
453static_always_inline int
454vlib_mem_get_next_numa_node (int numa)
455{
456 clib_mem_main_t *mm = &clib_mem_main;
457 u32 bitmap = mm->numa_node_bitmap;
458
459 if (numa >= 0)
460 bitmap &= ~pow2_mask (numa + 1);
461 if (bitmap == 0)
462 return -1;
463
464 return count_trailing_zeros (bitmap);
465}
466
467static_always_inline clib_mem_page_sz_t
468clib_mem_log2_page_size_validate (clib_mem_page_sz_t log2_page_size)
469{
470 if (log2_page_size == CLIB_MEM_PAGE_SZ_DEFAULT)
471 return clib_mem_get_log2_page_size ();
472 if (log2_page_size == CLIB_MEM_PAGE_SZ_DEFAULT_HUGE)
473 return clib_mem_get_log2_default_hugepage_size ();
474 return log2_page_size;
475}
476
477static_always_inline uword
478clib_mem_page_bytes (clib_mem_page_sz_t log2_page_size)
479{
Dave Barach27c35e32020-10-07 09:37:36 -0400480 return 1ULL << clib_mem_log2_page_size_validate (log2_page_size);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200481}
482
Damjan Marionbdbb0c52020-09-17 10:40:44 +0200483static_always_inline clib_error_t *
484clib_mem_get_last_error (void)
485{
486 return clib_mem_main.error;
487}
488
Damjan Marion0da81682020-12-22 14:58:56 +0100489/* bulk allocator */
490
491typedef void *clib_mem_bulk_handle_t;
492clib_mem_bulk_handle_t clib_mem_bulk_init (u32 elt_sz, u32 align,
493 u32 min_elts_per_chunk);
494void clib_mem_bulk_destroy (clib_mem_bulk_handle_t h);
495void *clib_mem_bulk_alloc (clib_mem_bulk_handle_t h);
496void clib_mem_bulk_free (clib_mem_bulk_handle_t h, void *p);
497u8 *format_clib_mem_bulk (u8 *s, va_list *args);
Damjan Marion6bfd0762020-09-11 22:16:53 +0200498
Dave Barachc3799992016-08-15 11:12:27 -0400499#include <vppinfra/error.h> /* clib_panic */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700500
501#endif /* _included_clib_mem_h */
Dave Barachc3799992016-08-15 11:12:27 -0400502
503/*
504 * fd.io coding-style-patch-verification: ON
505 *
506 * Local Variables:
507 * eval: (c-set-style "gnu")
508 * End:
509 */