aboutsummaryrefslogtreecommitdiff
path: root/lib/alloc
diff options
context:
space:
mode:
Diffstat (limited to 'lib/alloc')
-rw-r--r--lib/alloc/arena_alloc.c13
1 files changed, 12 insertions, 1 deletions
diff --git a/lib/alloc/arena_alloc.c b/lib/alloc/arena_alloc.c
index 16fd758..c25c132 100644
--- a/lib/alloc/arena_alloc.c
+++ b/lib/alloc/arena_alloc.c
@@ -26,6 +26,8 @@ static void *alloc(allocator_t mem, ptrdiff_t nmemb, ptrdiff_t elemsz,
ptrdiff_t align);
static void *resize(allocator_t mem, void *ptr, ptrdiff_t oldnmemb,
ptrdiff_t newnmemb, ptrdiff_t elemsz, ptrdiff_t align);
+static void free_(allocator_t mem, void *ptr, ptrdiff_t oldnmemb,
+ ptrdiff_t elemsz);
static void freeall(allocator_t mem);
static arena_blk_t *mkblk(ptrdiff_t blksz);
[[noreturn]] static void errjmp(jmp_buf *env);
@@ -40,7 +42,7 @@ arena_alloc(allocator_t mem, alloc_mode_t mode, void *ptr, ptrdiff_t oldnmemb,
case ALLOC_RESIZE:
return resize(mem, ptr, oldnmemb, newnmemb, elemsz, align);
case ALLOC_FREE:
- /* TODO: Allow freeing the very last allocation */
+ free_(mem, ptr, oldnmemb, elemsz);
return nullptr;
case ALLOC_FREEALL:
freeall(mem);
@@ -163,6 +165,15 @@ resize(allocator_t mem, void *ptr, ptrdiff_t oldnmemb, ptrdiff_t newnmemb,
}
void
+free_(allocator_t mem, void *ptr, ptrdiff_t oldnmemb, ptrdiff_t elemsz)
+{
+ arena_ctx_t *ctx = mem.ctx;
+ arena_blk_t *blk = ctx->_head;
+ if ((uint8_t *)ptr + oldnmemb * elemsz == blk->fngr)
+ blk->fngr = ptr;
+}
+
+void
freeall(allocator_t mem)
{
arena_ctx_t *ctx = mem.ctx;