Extended functionality. More...
Macros | |
#define | MI_SMALL_SIZE_MAX |
Maximum size allowed for small allocations in mi_malloc_small and mi_zalloc_small (usually 128*sizeof(void*) (= 1KB on 64-bit systems)) | |
Typedefs | |
typedef void | mi_deferred_free_fun(bool force, unsigned long long heartbeat, void *arg) |
Type of deferred free functions. | |
typedef void | mi_output_fun(const char *msg, void *arg) |
Type of output functions. | |
typedef void | mi_error_fun(int err, void *arg) |
Type of error callback functions. | |
typedef int | mi_arena_id_t |
Mimalloc uses large (virtual) memory areas, called "arena"s, from the OS to manage its memory. | |
typedef void * | mi_subproc_id_t |
A process can associate threads with sub-processes. | |
Functions | |
void * | mi_malloc_small (size_t size) |
Allocate a small object. | |
void * | mi_zalloc_small (size_t size) |
Allocate a zero initialized small object. | |
size_t | mi_usable_size (void *p) |
Return the available bytes in a memory block. | |
size_t | mi_good_size (size_t size) |
Return the used allocation size. | |
void | mi_collect (bool force) |
Eagerly free memory. | |
void | mi_stats_print (void *out) |
Deprecated. | |
void | mi_stats_print_out (mi_output_fun *out, void *arg) |
Print the main statistics. | |
void | mi_stats_reset (void) |
Reset statistics. | |
void | mi_stats_merge (void) |
Merge thread local statistics with the main statistics and reset. | |
void | mi_thread_init (void) |
Initialize mimalloc on a thread. | |
void | mi_thread_done (void) |
Uninitialize mimalloc on a thread. | |
void | mi_thread_stats_print_out (mi_output_fun *out, void *arg) |
Print out heap statistics for this thread. | |
void | mi_register_deferred_free (mi_deferred_free_fun *deferred_free, void *arg) |
Register a deferred free function. | |
void | mi_register_output (mi_output_fun *out, void *arg) |
Register an output function. | |
void | mi_register_error (mi_error_fun *errfun, void *arg) |
Register an error callback function. | |
bool | mi_is_in_heap_region (const void *p) |
Is a pointer part of our heap? | |
int | mi_reserve_os_memory (size_t size, bool commit, bool allow_large) |
Reserve OS memory for use by mimalloc. | |
bool | mi_manage_os_memory (void *start, size_t size, bool is_committed, bool is_large, bool is_zero, int numa_node) |
Manage a particular memory area for use by mimalloc. | |
int | mi_reserve_huge_os_pages_interleave (size_t pages, size_t numa_nodes, size_t timeout_msecs) |
Reserve pages of huge OS pages (1GiB) evenly divided over numa_nodes nodes, but stops after at most timeout_msecs seconds. | |
int | mi_reserve_huge_os_pages_at (size_t pages, int numa_node, size_t timeout_msecs) |
Reserve pages of huge OS pages (1GiB) at a specific numa_node, but stops after at most timeout_msecs seconds. | |
bool | mi_is_redirected () |
Is the C runtime malloc API redirected? | |
void | mi_process_info (size_t *elapsed_msecs, size_t *user_msecs, size_t *system_msecs, size_t *current_rss, size_t *peak_rss, size_t *current_commit, size_t *peak_commit, size_t *page_faults) |
Return process information (time and memory usage). | |
void | mi_debug_show_arenas (bool show_inuse, bool show_abandoned, bool show_purge) |
Show all current arena's. | |
void * | mi_arena_area (mi_arena_id_t arena_id, size_t *size) |
Return the size of an arena. | |
int | mi_reserve_huge_os_pages_at_ex (size_t pages, int numa_node, size_t timeout_msecs, bool exclusive, mi_arena_id_t *arena_id) |
Reserve huge OS pages (1GiB) into a single arena. | |
int | mi_reserve_os_memory_ex (size_t size, bool commit, bool allow_large, bool exclusive, mi_arena_id_t *arena_id) |
Reserve OS memory to be managed in an arena. | |
bool | mi_manage_os_memory_ex (void *start, size_t size, bool is_committed, bool is_large, bool is_zero, int numa_node, bool exclusive, mi_arena_id_t *arena_id) |
Manage externally allocated memory as a mimalloc arena. | |
mi_heap_t * | mi_heap_new_in_arena (mi_arena_id_t arena_id) |
Create a new heap that only allocates in the specified arena. | |
mi_heap_t * | mi_heap_new_ex (int heap_tag, bool allow_destroy, mi_arena_id_t arena_id) |
Create a new heap. | |
mi_subproc_id_t | mi_subproc_main (void) |
Get the main sub-process identifier. | |
mi_subproc_id_t | mi_subproc_new (void) |
Create a fresh sub-process (with no associated threads yet). | |
void | mi_subproc_delete (mi_subproc_id_t subproc) |
Delete a previously created sub-process. | |
void | mi_subproc_add_current_thread (mi_subproc_id_t subproc) |
Add the current thread to the given sub-process. | |
Extended functionality.
#define MI_SMALL_SIZE_MAX |
Maximum size allowed for small allocations in mi_malloc_small and mi_zalloc_small (usually 128*sizeof(void*)
(= 1KB on 64-bit systems))
typedef int mi_arena_id_t |
Mimalloc uses large (virtual) memory areas, called "arena"s, from the OS to manage its memory.
Each arena has an associated identifier.
typedef void mi_deferred_free_fun(bool force, unsigned long long heartbeat, void *arg) |
Type of deferred free functions.
force | If true all outstanding items should be freed. |
heartbeat | A monotonically increasing count. |
arg | Argument that was passed at registration to hold extra state. |
typedef void mi_error_fun(int err, void *arg) |
Type of error callback functions.
err | Error code (see mi_register_error() for a complete list). |
arg | Argument that was passed at registration to hold extra state. |
typedef void mi_output_fun(const char *msg, void *arg) |
Type of output functions.
msg | Message to output. |
arg | Argument that was passed at registration to hold extra state. |
typedef void* mi_subproc_id_t |
A process can associate threads with sub-processes.
A sub-process will not reclaim memory from (abandoned heaps/threads) other subprocesses.
void * mi_arena_area | ( | mi_arena_id_t | arena_id, |
size_t * | size ) |
Return the size of an arena.
arena_id | The arena identifier. |
size | Returned size in bytes of the (virtual) arena area. |
void mi_collect | ( | bool | force | ) |
Eagerly free memory.
force | If true, aggressively return memory to the OS (can be expensive!) |
Regular code should not have to call this function. It can be beneficial in very narrow circumstances; in particular, when a long running thread allocates a lot of blocks that are freed by other threads it may improve resource usage by calling this every once in a while.
void mi_debug_show_arenas | ( | bool | show_inuse, |
bool | show_abandoned, | ||
bool | show_purge ) |
Show all current arena's.
show_inuse | Show the arena blocks that are in use. |
show_abandoned | Show the abandoned arena blocks. |
show_purge | Show arena blocks scheduled for purging. |
size_t mi_good_size | ( | size_t | size | ) |
Return the used allocation size.
size | The minimal required size in bytes. |
n
that will be allocated, where n >= size
.Generally, mi_usable_size(mi_malloc(size)) == mi_good_size(size)
. This can be used to reduce internal wasted space when allocating buffers for example.
mi_heap_t * mi_heap_new_ex | ( | int | heap_tag, |
bool | allow_destroy, | ||
mi_arena_id_t | arena_id ) |
Create a new heap.
heap_tag | The heap tag associated with this heap; heaps only reclaim memory between heaps with the same tag. |
allow_destroy | Is mi_heap_destroy allowed? Not allowing this allows the heap to reclaim memory from terminated threads. |
arena_id | If not 0, the heap will only allocate from the specified arena. |
NULL
on failure.The arena_id can be used by runtimes to allocate only in a specified pre-reserved arena. This is used for example for a compressed pointer heap in Koka. The heap_tag enables heaps to keep objects of a certain type isolated to heaps with that tag. This is used for example in the CPython integration.
mi_heap_t * mi_heap_new_in_arena | ( | mi_arena_id_t | arena_id | ) |
Create a new heap that only allocates in the specified arena.
arena_id | The arena identifier. |
NULL
. bool mi_is_in_heap_region | ( | const void * | p | ) |
Is a pointer part of our heap?
p | The pointer to check. |
bool mi_is_redirected | ( | ) |
Is the C runtime malloc API redirected?
Currently only used on Windows.
void * mi_malloc_small | ( | size_t | size | ) |
Allocate a small object.
size | The size in bytes, can be at most MI_SMALL_SIZE_MAX. |
bool mi_manage_os_memory | ( | void * | start, |
size_t | size, | ||
bool | is_committed, | ||
bool | is_large, | ||
bool | is_zero, | ||
int | numa_node ) |
Manage a particular memory area for use by mimalloc.
This is just like mi_reserve_os_memory
except that the area should already be allocated in some manner and available for use my mimalloc.
start | Start of the memory area |
size | The size of the memory area. |
is_committed | Is the area already committed? |
is_large | Does it consist of large OS pages? Set this to true as well for memory that should not be decommitted or protected (like rdma etc.) |
is_zero | Does the area consists of zero's? |
numa_node | Possible associated numa node or -1 . |
bool mi_manage_os_memory_ex | ( | void * | start, |
size_t | size, | ||
bool | is_committed, | ||
bool | is_large, | ||
bool | is_zero, | ||
int | numa_node, | ||
bool | exclusive, | ||
mi_arena_id_t * | arena_id ) |
Manage externally allocated memory as a mimalloc arena.
This memory will not be freed by mimalloc.
start | Start address of the area. |
size | Size in bytes of the area. |
is_committed | Is the memory already committed? |
is_large | Does it consist of (pinned) large OS pages? |
is_zero | Is the memory zero-initialized? |
numa_node | Associated NUMA node, or -1 to have no NUMA preference. |
exclusive | Is the arena exclusive (where only heaps associated with the arena can allocate in it) |
arena_id | The new arena identifier. |
true
if successful. void mi_process_info | ( | size_t * | elapsed_msecs, |
size_t * | user_msecs, | ||
size_t * | system_msecs, | ||
size_t * | current_rss, | ||
size_t * | peak_rss, | ||
size_t * | current_commit, | ||
size_t * | peak_commit, | ||
size_t * | page_faults ) |
Return process information (time and memory usage).
elapsed_msecs | Optional. Elapsed wall-clock time of the process in milli-seconds. |
user_msecs | Optional. User time in milli-seconds (as the sum over all threads). |
system_msecs | Optional. System time in milli-seconds. |
current_rss | Optional. Current working set size (touched pages). |
peak_rss | Optional. Peak working set size (touched pages). |
current_commit | Optional. Current committed memory (backed by the page file). |
peak_commit | Optional. Peak committed memory (backed by the page file). |
page_faults | Optional. Count of hard page faults. |
The current_rss is precise on Windows and MacOSX; other systems estimate this using current_commit. The commit is precise on Windows but estimated on other systems as the amount of read/write accessible memory reserved by mimalloc.
void mi_register_deferred_free | ( | mi_deferred_free_fun * | deferred_free, |
void * | arg ) |
Register a deferred free function.
deferred_free | Address of a deferred free-ing function or NULL to unregister. |
arg | Argument that will be passed on to the deferred free function. |
Some runtime systems use deferred free-ing, for example when using reference counting to limit the worst case free time. Such systems can register (re-entrant) deferred free function to free more memory on demand. When the force parameter is true all possible memory should be freed. The per-thread heartbeat parameter is monotonically increasing and guaranteed to be deterministic if the program allocates deterministically. The deferred_free function is guaranteed to be called deterministically after some number of allocations (regardless of freeing or available free memory). At most one deferred_free function can be active.
void mi_register_error | ( | mi_error_fun * | errfun, |
void * | arg ) |
Register an error callback function.
errfun | The error function that is called on an error (use NULL for default) |
arg | Extra argument that will be passed on to the error function. |
The errfun function is called on an error in mimalloc after emitting an error message (through the output function). It as always legal to just return from the errfun function in which case allocation functions generally return NULL or ignore the condition. The default function only calls abort() when compiled in secure mode with an EFAULT error. The possible error codes are:
void mi_register_output | ( | mi_output_fun * | out, |
void * | arg ) |
Register an output function.
out | The output function, use NULL to output to stderr. |
arg | Argument that will be passed on to the output function. |
The out
function is called to output any information from mimalloc, like verbose or warning messages.
int mi_reserve_huge_os_pages_at | ( | size_t | pages, |
int | numa_node, | ||
size_t | timeout_msecs ) |
Reserve pages of huge OS pages (1GiB) at a specific numa_node, but stops after at most timeout_msecs
seconds.
pages | The number of 1GiB pages to reserve. |
numa_node | The NUMA node where the memory is reserved (start at 0). Use -1 for no affinity. |
timeout_msecs | Maximum number of milli-seconds to try reserving, or 0 for no timeout. |
The reserved memory is used by mimalloc to satisfy allocations. May quit before timeout_msecs are expired if it estimates it will take more than 1.5 times timeout_msecs. The time limit is needed because on some operating systems it can take a long time to reserve contiguous memory if the physical memory is fragmented.
int mi_reserve_huge_os_pages_at_ex | ( | size_t | pages, |
int | numa_node, | ||
size_t | timeout_msecs, | ||
bool | exclusive, | ||
mi_arena_id_t * | arena_id ) |
Reserve huge OS pages (1GiB) into a single arena.
pages | Number of 1GiB pages to reserve. |
numa_node | The associated NUMA node, or -1 for no NUMA preference. |
timeout_msecs | Max amount of milli-seconds this operation is allowed to take. (0 is infinite) |
exclusive | If exclusive, only a heap associated with this arena can allocate in it. |
arena_id | The arena identifier. |
int mi_reserve_huge_os_pages_interleave | ( | size_t | pages, |
size_t | numa_nodes, | ||
size_t | timeout_msecs ) |
Reserve pages of huge OS pages (1GiB) evenly divided over numa_nodes nodes, but stops after at most timeout_msecs
seconds.
pages | The number of 1GiB pages to reserve. |
numa_nodes | The number of nodes do evenly divide the pages over, or 0 for using the actual number of NUMA nodes. |
timeout_msecs | Maximum number of milli-seconds to try reserving, or 0 for no timeout. |
The reserved memory is used by mimalloc to satisfy allocations. May quit before timeout_msecs are expired if it estimates it will take more than 1.5 times timeout_msecs. The time limit is needed because on some operating systems it can take a long time to reserve contiguous memory if the physical memory is fragmented.
int mi_reserve_os_memory | ( | size_t | size, |
bool | commit, | ||
bool | allow_large ) |
Reserve OS memory for use by mimalloc.
Reserved areas are used before allocating from the OS again. By reserving a large area upfront, allocation can be more efficient, and can be better managed on systems without mmap
/VirtualAlloc
(like WASM for example).
size | The size to reserve. |
commit | Commit the memory upfront. |
allow_large | Allow large OS pages (2MiB) to be used? |
ENOMEM
). int mi_reserve_os_memory_ex | ( | size_t | size, |
bool | commit, | ||
bool | allow_large, | ||
bool | exclusive, | ||
mi_arena_id_t * | arena_id ) |
Reserve OS memory to be managed in an arena.
size | Size the reserve. |
commit | Should the memory be initially committed? |
allow_large | Allow the use of large OS pages? |
exclusive | Is the returned arena exclusive? |
arena_id | The new arena identifier. |
void mi_stats_merge | ( | void | ) |
Merge thread local statistics with the main statistics and reset.
void mi_stats_print | ( | void * | out | ) |
Deprecated.
out | Ignored, outputs to the registered output function or stderr by default. |
Most detailed when using a debug build.
void mi_stats_print_out | ( | mi_output_fun * | out, |
void * | arg ) |
Print the main statistics.
out | An output function or NULL for the default. |
arg | Optional argument passed to out (if not NULL) |
Most detailed when using a debug build.
void mi_stats_reset | ( | void | ) |
Reset statistics.
void mi_subproc_add_current_thread | ( | mi_subproc_id_t | subproc | ) |
Add the current thread to the given sub-process.
This should be called right after a thread is created (and no allocation has taken place yet)
void mi_subproc_delete | ( | mi_subproc_id_t | subproc | ) |
Delete a previously created sub-process.
subproc | The sub-process identifier. Only delete sub-processes if all associated threads have terminated. |
mi_subproc_id_t mi_subproc_main | ( | void | ) |
Get the main sub-process identifier.
mi_subproc_id_t mi_subproc_new | ( | void | ) |
Create a fresh sub-process (with no associated threads yet).
void mi_thread_done | ( | void | ) |
Uninitialize mimalloc on a thread.
Should not be used as on most systems (pthreads, windows) this is done automatically. Ensures that any memory that is not freed yet (but will be freed by other threads in the future) is properly handled.
void mi_thread_init | ( | void | ) |
Initialize mimalloc on a thread.
Should not be used as on most systems (pthreads, windows) this is done automatically.
void mi_thread_stats_print_out | ( | mi_output_fun * | out, |
void * | arg ) |
Print out heap statistics for this thread.
out | An output function or NULL for the default. |
arg | Optional argument passed to out (if not NULL) |
Most detailed when using a debug build.
size_t mi_usable_size | ( | void * | p | ) |
Return the available bytes in a memory block.
p | Pointer to previously allocated memory (or NULL) |
The returned size can be used to call mi_expand successfully. The returned size is always at least equal to the allocated size of p.
void * mi_zalloc_small | ( | size_t | size | ) |
Allocate a zero initialized small object.
size | The size in bytes, can be at most MI_SMALL_SIZE_MAX. |