From 31153d6374317b4bfaba83fafdcde5a0f3f885de Mon Sep 17 00:00:00 2001 From: Nick Mathewson Date: Tue, 8 Apr 2008 17:29:05 +0000 Subject: r19243@catbus: nickm | 2008-04-08 13:28:59 -0400 Use a freelist to hold a few recent memarea chunks. We do a kazillion memarea allocs and frees; that cant be good for us. svn:r14319 --- src/common/memarea.c | 63 ++++++++++++++++++++++++++++++++++++++++++---------- src/common/memarea.h | 1 + 2 files changed, 52 insertions(+), 12 deletions(-) (limited to 'src/common') diff --git a/src/common/memarea.c b/src/common/memarea.c index 743a451a8b..d4657b7027 100644 --- a/src/common/memarea.c +++ b/src/common/memarea.c @@ -53,6 +53,8 @@ typedef struct memarea_chunk_t { #define CHUNK_HEADER_SIZE STRUCT_OFFSET(memarea_chunk_t, u) +#define CHUNK_SIZE 8192 + /** A memarea_t is an allocation region for a set of small memory requests * that will all be freed at once. */ struct memarea_t { @@ -60,25 +62,49 @@ struct memarea_t { size_t chunk_size; /**chunk_size bytes. */ static memarea_chunk_t * -alloc_chunk(size_t chunk_size) +alloc_chunk(size_t sz) { - memarea_chunk_t *res = tor_malloc_roundup(&chunk_size); - res->next_chunk = NULL; - res->mem_size = chunk_size - CHUNK_HEADER_SIZE; - res->next_mem = res->u.mem; - return res; + (void)sz; /*XXXX021 remove this argument. */ + if (freelist) { + memarea_chunk_t *res = freelist; + freelist = res->next_chunk; + --freelist_len; + return res; + } else { + size_t chunk_size = CHUNK_SIZE; + memarea_chunk_t *res = tor_malloc_roundup(&chunk_size); + res->next_chunk = NULL; + res->mem_size = chunk_size - CHUNK_HEADER_SIZE; + res->next_mem = res->u.mem; + return res; + } } -/** Allocate and return new memarea, with chunks of approximately - * chunk_size bytes. (There is indeed some overhead.) */ +static void +chunk_free(memarea_chunk_t *chunk) +{ + if (freelist_len >= MAX_FREELIST_LEN) { + ++freelist_len; + chunk->next_chunk = freelist; + freelist = chunk; + } else { + tor_free(chunk); + } +} + +/** Allocate and return new memarea. */ memarea_t * -memarea_new(size_t chunk_size) +memarea_new(size_t chunk_size)/*XXXX021 remove this argument.*/ { memarea_t *head = tor_malloc(sizeof(memarea_t)); head->first = alloc_chunk(chunk_size); - head->chunk_size = chunk_size; + (void)chunk_size; return head; } @@ -90,7 +116,7 @@ memarea_drop_all(memarea_t *area) memarea_chunk_t *chunk, *next; for (chunk = area->first; chunk; chunk = next) { next = chunk->next_chunk; - tor_free(chunk); + chunk_free(chunk); } area->first = NULL; /*fail fast on */ tor_free(area); @@ -106,13 +132,26 @@ memarea_clear(memarea_t *area) if (area->first->next_chunk) { for (chunk = area->first->next_chunk; chunk; chunk = next) { next = chunk->next_chunk; - tor_free(chunk); + chunk_free(chunk); } area->first->next_chunk = NULL; } area->first->next_mem = area->first->u.mem; } +/** DOCDOC */ +void +memarea_clear_freelist(void) +{ + memarea_chunk_t *chunk, *next; + freelist_len = 0; + for (chunk = freelist; chunk; chunk = next) { + next = chunk->next_chunk; + tor_free(chunk); + } + freelist = NULL; +} + /** Return true iff p is in a range that has been returned by an * allocation from area. */ int diff --git a/src/common/memarea.h b/src/common/memarea.h index 17ab781a7f..b08a033e4a 100644 --- a/src/common/memarea.h +++ b/src/common/memarea.h @@ -19,6 +19,7 @@ char *memarea_strdup(memarea_t *area, const char *s); char *memarea_strndup(memarea_t *area, const char *s, size_t n); void memarea_get_stats(memarea_t *area, size_t *allocated_out, size_t *used_out); +void memarea_clear_freelist(void); void memarea_assert_ok(memarea_t *area); #endif -- cgit v1.2.3-54-g00ecf