/* TODO: Support malloc() backend for systems without MAP_ANONYMOUS */ #include #include #include #include #include #include #include #include #include "_attrs.h" #include "alloc.h" #include "error.h" #include "macros.h" #define PAD(len, align) (((len) + (align) - 1) & ~((align) - 1)) struct arena_blk { uint8_t *head, *tail, *fngr; struct arena_blk *next; }; 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); void * arena_alloc(allocator_t mem, alloc_mode_t mode, void *ptr, ptrdiff_t oldnmemb, ptrdiff_t newnmemb, ptrdiff_t elemsz, ptrdiff_t align) { switch (mode) { case ALLOC_NEW: return alloc(mem, newnmemb, elemsz, align); case ALLOC_RESIZE: return resize(mem, ptr, oldnmemb, newnmemb, elemsz, align); case ALLOC_FREE: free_(mem, ptr, oldnmemb, elemsz); return nullptr; case ALLOC_FREEALL: freeall(mem); return nullptr; default: unreachable(); } } void * alloc(allocator_t mem, ptrdiff_t nmemb, ptrdiff_t elemsz, ptrdiff_t align) { arena_ctx_t *ctx = mem.ctx; if (ctx->blksz == 0) { long blksz = sysconf(_SC_PAGESIZE); ctx->blksz = blksz == -1 ? 4096 : blksz; } ptrdiff_t bufsz; if (ckd_mul(&bufsz, nmemb, elemsz)) { errno = EOVERFLOW; errjmp(mem.err); } for (arena_blk_t *blk = ctx->_head; blk != nullptr; blk = blk->next) { ptrdiff_t nbufsz, off = PAD((uintptr_t)blk->fngr, align); if (ckd_add(&nbufsz, bufsz, off)) continue; if (blk->tail - blk->fngr >= nbufsz) { void *p = blk->fngr + off; blk->fngr += nbufsz; return p; } } /* No page exists that is large enough for our allocation */ ptrdiff_t padding = PAD(sizeof(arena_blk_t), align); if (ckd_add(&bufsz, bufsz, sizeof(arena_blk_t)) || ckd_add(&bufsz, bufsz, padding)) { errno = EOVERFLOW; errjmp(mem.err); } arena_blk_t *blk = mkblk(MAX(bufsz, ctx->blksz)); if (blk == nullptr) errjmp(mem.err); blk->next = ctx->_head; blk->fngr = blk->head + bufsz; ctx->_head = blk; return blk->head + sizeof(arena_blk_t) + padding; } void * resize(allocator_t mem, void *ptr, ptrdiff_t oldnmemb, ptrdiff_t newnmemb, ptrdiff_t elemsz, ptrdiff_t align) { if (oldnmemb == newnmemb) return ptr; arena_ctx_t *ctx = mem.ctx; arena_blk_t *blk = ctx->_head; /* There are 4 cases we should consider when resizing an allocation: 1. We are shrinking the final allocation 2. We are shrinking any other allocation 3. We are growing the final allocation 4. We are growing any other allocation Shrinking is typically a no-op, however in case 1 we can be a bit smart. If we want to shrink the most recent allocation we can actually shift the finger pointer in the arena block to free up more space in the current block for future allocations, allowing us to potentially cut down on wasteful calls to mmap() and friends in the future. Likewise with growing an allocation, this is typically a very simple alloc+memcpy(), however we can be smart in the case that we are growing the last allocation. If we want to grow the last allocation we can actually first see if there is free-space remaining in the current block and simply move the finger pointer avoiding any unneccesary memory wastage. Not only this but if we don’t have enough space in the current block we can attempt to resize the block using mremap() if we’re on Linux. */ bool resizing_last_alloc = (uint8_t *)ptr + oldnmemb * elemsz == blk->fngr; if (oldnmemb > newnmemb) { if (resizing_last_alloc) blk->fngr = (uint8_t *)ptr + newnmemb * elemsz; return ptr; } ptrdiff_t bufsz; if (ckd_mul(&bufsz, newnmemb, elemsz)) { errno = EOVERFLOW; errjmp(mem.err); } if (resizing_last_alloc) { uint8_t *alloc_tail = (uint8_t *)ptr + bufsz; if (alloc_tail <= blk->tail) { blk->fngr = alloc_tail; return ptr; } #if __linux__ void *p = mremap(blk->head, blk->tail - blk->head, alloc_tail - blk->head, 0); if (p != MAP_FAILED) { blk->fngr = blk->tail = alloc_tail; return ptr; } #endif } void *p = alloc(mem, newnmemb, elemsz, align); return memcpy(p, ptr, oldnmemb * elemsz); } 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; arena_blk_t *blk = ctx->_head; while (blk != nullptr) { arena_blk_t *next = blk->next; (void)munmap(blk, blk->tail - blk->head); blk = next; } ctx->_head = nullptr; } static arena_blk_t * mkblk(ptrdiff_t blksz) { arena_blk_t blk; /* blk.next and blk.fngr get set by the caller */ blk.head = mmap(nullptr, blksz, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANON, -1, 0); if (blk.head == MAP_FAILED) return nullptr; blk.tail = blk.head + blksz; return memcpy(blk.head, &blk, sizeof blk); } void errjmp(jmp_buf *env) { if (env != nullptr) longjmp(*env, 1); err("arena_alloc:"); }