From b2b15d53798213327796e15341dd77875726c09e Mon Sep 17 00:00:00 2001 From: Brooks Davis Date: Fri, 28 Jun 2024 15:55:42 +0100 Subject: [PATCH] mrs_realloc: avoid copies when space is available When the passed pointer is valid and already has enough space, return it unless it's at least twice the requested size. Some software uses realloc to extend allocations by small, fixed amounts. In a naive implementation of realloc that blindly performs a new allocation, copies the data, and then frees the old allocation, this is O(n^2). For small values of `n` this isn't too noticable, but asymptotically it's quite bad and revocation makes it much worse. With current tuning, we eventually reach a point where approximately every 4th allocation triggers a revocation pass. We can make this somewhat less visible (or fix it entirely for snmalloc with it's power-of-two sized buckets) by omitting unnecessary allocations and copies. Bias towards improving linear increase performance over space saving and only reallocate to shrink the allocation if we'd save at least 50% space. If we chose a smaller value then we end up doing full reallocs constantly with an allocator like snmalloc. --- lib/libc/stdlib/malloc/mrs/mrs.c | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/lib/libc/stdlib/malloc/mrs/mrs.c b/lib/libc/stdlib/malloc/mrs/mrs.c index f54807c54fbf..e2c11765526f 100644 --- a/lib/libc/stdlib/malloc/mrs/mrs.c +++ b/lib/libc/stdlib/malloc/mrs/mrs.c @@ -1515,6 +1515,19 @@ mrs_realloc(void *ptr, size_t size) mrs_debug_printf("mrs_realloc: called ptr %p ptr size %zu new size %zu\n", ptr, old_size, size); + /* + * Just return the pointer if our desired size fits. + * + * Only try to reclaim space by copying if we'd recover at least + * half of the allocated storage. In any other case we can't + * tell the difference between shrinking and linear growth into + * a very large over-allocation (e.g., growing into snmalloc's + * power-of-two buckets by 1K). + */ + if (ptr != NULL && cheri_gettag(ptr) && cheri_getoffset(ptr) == 0 && + size <= old_size && old_size - size > (old_size >> 1)) + return (ptr); + void *new_alloc = mrs_malloc(size); /*