EST! EST!! EST!!!
ESTALLOC is a high-performance memory allocator specifically designed for embedded systems, implementing the Two-Level Segregated Fit (TLSF) algorithm with a First-Fit fallback strategy. It provides deterministic O(1) time complexity for memory allocation and deallocation, making it ideal for real-time applications and resource-constrained environments.
ESTALLOC is derived from the memory allocator originally developed for mrubyc/mrubyc, a lightweight Ruby implementation for embedded systems. The source code was adapted from mruby/c and refactored into a standalone, general-purpose allocator with minimal dependencies and improved portability for broader use beyond the original project.
- Deterministic Performance: O(1) time complexity for malloc and free operations
- Low Memory Fragmentation: Efficient memory management reduces fragmentation
- Configurable: Supports different alignment options (4-byte and 8-byte)
- Embedded-Friendly: Optimized for 16-bit and 32-bit architectures
- Memory Pool Based: Uses pre-allocated memory pools for management
- Debug Support: Optional debugging features for memory leak detection and pool health checks
ESTALLOC provides the following memory management functions:
est_init(void *ptr, unsigned int size)
: Initialize a memory poolest_malloc(ESTALLOC *est, unsigned int size)
: Allocate memoryest_free(ESTALLOC *est, void *ptr)
: Free previously allocated memoryest_realloc(ESTALLOC *est, void *ptr, unsigned int size)
: Resize allocated memoryest_calloc(ESTALLOC *est, unsigned int nmemb, unsigned int size)
: Allocate zero-initialized memoryest_permalloc(ESTALLOC *est, unsigned int size)
: Allocate permanent (non-freeable) memoryest_usable_size(ESTALLOC *est, void *ptr)
: Get usable size of allocated memory block
In any build:
est_take_statistics(ESTALLOC *est)
: Collect memory usage statisticsest_take_statistics(est); est->stat.total; // Total memory est->stat.used; // Used memory est->stat.free; // Free memory est->stat.frag; // Number of fragmentation
When compiled with ESTALLOC_DEBUG
defined:
-
est_start_profiling(ESTALLOC *est)
: Start memory profiling -
est_stop_profiling(ESTALLOC *est)
: Stop memory profilingest_start_profiling(est); { // Do someting. ESTALLOC records the memory usage } est_stop_profiling(est); est->prof.initial; // Initial memory usage at `est_start_profiling()` called est->prof.max; // Maximum memory usage during profiling est->prof.min; // Minimum memory usage during profiling
-
est_sanity_check(ESTALLOC *est)
: Check memory pool integrityint result = est_sanity_check(est); if (result == 0) { // It's sanity } else { // Something went wrong // See comment in estalloc.c for details }
When compiled with ESTALLOC_PRINT_DEBUG
defined:
est_fprint_pool_header(ESTALLOC *est, FILE *fp)
: Print memory pool header informationest_fprint_memory_block(ESTALLOC *est, FILE *fp)
: Print detailed memory block information
#include "estalloc.h"
#include <stdio.h>
#define POOL_SIZE (64 * 1024 - 1) // 64KB memory pool
int
main()
{
static uint8_t memory_pool[POOL_SIZE];
// Initialize memory pool
ESTALLOC *est = est_init(memory_pool, POOL_SIZE);
if (!est) {
printf("Failed to initialize memory pool\n");
return 1;
}
// Allocate memory
int *numbers = (int*)est_malloc(est, 10 * sizeof(int));
if (numbers) {
// Use allocated memory
for (int i = 0; i < 10; i++) {
numbers[i] = i * 10;
}
// Print values
for (int i = 0; i < 10; i++) {
printf("%d ", numbers[i]);
}
printf("\n");
// Free memory
est_free(est, numbers);
}
return 0;
}
ESTALLOC can be configured using the following macros:
ESTALLOC_ALIGNMENT
: Memory alignment (default: N/A. You need to explicitly define4
or8
)ESTALLOC_ADDRESS_16BIT
orESTALLOC_ADDRESS_24BIT
: Addressable memory range bit width (default:ESTALLOC_ADDRESS_24BIT
)
ESTALLOC_ADDRESS_16BIT | ESTALLOC_ADDRESS_24BIT | |
---|---|---|
16-bit Platform | ✅ | ✅ |
32-bit Platform | ✅ | ✅ |
64-bit Platform | ❌ | ✅ |
ESTALLOC_FLI_BIT_WIDTH
: First level index bit width (default:9
)ESTALLOC_SLI_BIT_WIDTH
: Second level index bit width (default:3
)ESTALLOC_IGNORE_LSBS
: Least significant bits to ignore when calculating the first-level index (default:4
or5
)ESTALLOC_MIN_MEMORY_BLOCK_SIZE
: Minimum memory block size (defalut:16
or32
)
(See details in estalloc.c)