2022-06-04 17:04:51 +00:00
|
|
|
#pragma once
|
|
|
|
|
2024-02-20 21:41:37 -05:00
|
|
|
/**
|
|
|
|
** \defgroup mem Memory management
|
|
|
|
** tf_malloc() and friends use malloc() behind the scenes but optionally
|
|
|
|
** track memory per system (OpenSSL, sqlite, libuv, ...) and store callstacks
|
|
|
|
** to help debug leaks.
|
|
|
|
** @{
|
|
|
|
*/
|
|
|
|
|
2023-02-18 21:00:39 +00:00
|
|
|
#include <stdbool.h>
|
2022-06-04 17:04:51 +00:00
|
|
|
#include <stddef.h>
|
2023-02-18 21:00:39 +00:00
|
|
|
#include <stdint.h>
|
2022-06-04 17:04:51 +00:00
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/** JS malloc functions. */
|
2022-06-20 17:57:07 +00:00
|
|
|
typedef struct JSMallocFunctions JSMallocFunctions;
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Do early setup for memory tracking.
|
|
|
|
** @param tracking Whether tracking will be enabled, which adds a time and
|
|
|
|
** memory cost of storing stack traces for every allocation.
|
|
|
|
*/
|
2023-02-18 21:00:39 +00:00
|
|
|
void tf_mem_startup(bool tracking);
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Clean up the memory system.
|
|
|
|
*/
|
2023-02-18 21:00:39 +00:00
|
|
|
void tf_mem_shutdown();
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Register a custom allocator with libuv.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
void tf_mem_replace_uv_allocator();
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Get the number of bytes currently allocated by libuv.
|
|
|
|
** @return The allocated size in bytes.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
size_t tf_mem_get_uv_malloc_size();
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Register a custom allocator with OpenSSL.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
void tf_mem_replace_tls_allocator();
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Get the number of bytes currently allocated by OpenSSL.
|
|
|
|
** @return The allocated size in bytes.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
size_t tf_mem_get_tls_malloc_size();
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Register a custom allocator with SQLite.
|
|
|
|
*/
|
2023-02-18 19:14:06 +00:00
|
|
|
void tf_mem_replace_sqlite_allocator();
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Get the number of bytes currently allocated by SQLite.
|
|
|
|
** @return The allocated size in bytes.
|
|
|
|
*/
|
2023-02-18 19:14:06 +00:00
|
|
|
size_t tf_mem_get_sqlite_malloc_size();
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Get the number of bytes currently allocated by tf_malloc().
|
|
|
|
** @return The allocated size in bytes.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
size_t tf_mem_get_tf_malloc_size();
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Allocate memory. Like malloc() but with more tracking.
|
|
|
|
** @param size The number of bytes to allocate.
|
|
|
|
** @return The allocated memory.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
void* tf_malloc(size_t size);
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Reallocate memory. Like realloc() but with more tracking.
|
|
|
|
** @param ptr The previously allocated memory or NULL.
|
|
|
|
** @param size The new desired size.
|
|
|
|
** @return The new allocation.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
void* tf_realloc(void* ptr, size_t size);
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Free memory allocated by tf_malloc() or tf_realloc().
|
|
|
|
** @param ptr The allocation.
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
void tf_free(void* ptr);
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Duplicate a string.
|
|
|
|
** @param string The string to copy.
|
|
|
|
** @return The newly allocated string. Free with tf_free().
|
|
|
|
*/
|
2022-06-04 17:04:51 +00:00
|
|
|
char* tf_strdup(const char* string);
|
2022-06-17 21:18:10 +00:00
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Resize a vector. Like tf_realloc() but overallocatess and prefers not to
|
|
|
|
** shrink in order to speed up repeated growth.
|
|
|
|
** @param ptr The allocation to resize.
|
|
|
|
** @param size The desired new size.
|
|
|
|
** @return The new allocation.
|
|
|
|
*/
|
2022-06-17 21:18:10 +00:00
|
|
|
void* tf_resize_vec(void* ptr, size_t size);
|
2022-06-20 17:57:07 +00:00
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Populate a struct with custom JS allocation functions.
|
|
|
|
** @param[out] out The struct to receive the functions.
|
|
|
|
*/
|
2022-06-20 17:57:07 +00:00
|
|
|
void tf_get_js_malloc_functions(JSMallocFunctions* out);
|
2024-03-05 12:47:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
** Get the number of bytes currently allocated by JS allocators.
|
|
|
|
** @return The allocated size in bytes.
|
|
|
|
*/
|
2022-06-20 17:57:07 +00:00
|
|
|
size_t tf_mem_get_js_malloc_size();
|
2023-02-18 21:00:39 +00:00
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Call a function for every live memory allocation.
|
|
|
|
** @param callback The callback to call.
|
|
|
|
** @param user_data User data to pass to the callback.
|
|
|
|
*/
|
2023-02-18 21:00:39 +00:00
|
|
|
void tf_mem_walk_allocations(void (*callback)(void* ptr, size_t size, int frames_count, void* const* frames, void* user_data), void* user_data);
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Information about a memory allocation.
|
|
|
|
*/
|
2023-02-18 21:00:39 +00:00
|
|
|
typedef struct _tf_mem_allocation_t
|
|
|
|
{
|
2024-03-05 12:47:58 -05:00
|
|
|
/** A hash of the callstack used for determining uniqueness. */
|
2023-02-18 21:00:39 +00:00
|
|
|
uint32_t stack_hash;
|
2024-03-05 12:47:58 -05:00
|
|
|
/** The number of instances of this allocation. */
|
2023-02-18 21:00:39 +00:00
|
|
|
int count;
|
2024-03-05 12:47:58 -05:00
|
|
|
/** The size of this allocation. */
|
2023-02-18 21:00:39 +00:00
|
|
|
size_t size;
|
2024-03-05 12:47:58 -05:00
|
|
|
/** The callstack from which this allocation was made. */
|
2023-02-18 21:00:39 +00:00
|
|
|
void* frames[32];
|
2024-03-05 12:47:58 -05:00
|
|
|
/** The number of frames in the callstack. */
|
2023-02-18 21:00:39 +00:00
|
|
|
int frames_count;
|
|
|
|
} tf_mem_allocation_t;
|
|
|
|
|
2024-03-05 12:47:58 -05:00
|
|
|
/**
|
|
|
|
** Generate a list of live allocations.
|
|
|
|
** @param[out] out_count The number of allocations returned.
|
|
|
|
** @return An array of allocation information. Free with tf_free().
|
|
|
|
*/
|
2023-02-18 21:00:39 +00:00
|
|
|
tf_mem_allocation_t* tf_mem_summarize_allocations(int* out_count);
|
2024-02-20 21:41:37 -05:00
|
|
|
|
|
|
|
/** @} */
|