summaryrefslogtreecommitdiff
path: root/src/or
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2007-05-09 21:43:41 +0000
committerNick Mathewson <nickm@torproject.org>2007-05-09 21:43:41 +0000
commitb248ed620fb9ab4eb2534fd493d83e26e0cdfbcc (patch)
tree709dd6271befa2e03f72dc70b8d1bf48ab91b0fc /src/or
parenta7696a936d75cfe86438d9bb71498c4dff044bf0 (diff)
downloadtor-b248ed620fb9ab4eb2534fd493d83e26e0cdfbcc.tar.gz
tor-b248ed620fb9ab4eb2534fd493d83e26e0cdfbcc.zip
r12704@catbus: nickm | 2007-05-09 17:43:34 -0400
Periodically clean the freelist of buffer memory chunks. svn:r10149
Diffstat (limited to 'src/or')
-rw-r--r--src/or/buffers.c29
-rw-r--r--src/or/main.c1
-rw-r--r--src/or/or.h1
3 files changed, 29 insertions, 2 deletions
diff --git a/src/or/buffers.c b/src/or/buffers.c
index 6273c6699b..6296ffa7c7 100644
--- a/src/or/buffers.c
+++ b/src/or/buffers.c
@@ -161,7 +161,7 @@ _split_range(buf_t *buf, char *at, size_t *len,
/** DOCDOC */
static char *free_mem_list = NULL;
static int free_mem_list_len = 0;
-/*XXXX020 Actually remove stuff from freelist when it gets too big */
+static int free_mem_list_lowwater = 0;
/** DOCDOC */
static void
@@ -194,7 +194,8 @@ buf_get_initial_mem(buf_t *buf)
if (free_mem_list) {
mem = free_mem_list;
free_mem_list = *(char**)mem;
- --free_mem_list_len;
+ if (--free_mem_list_len < free_mem_list_lowwater)
+ free_mem_list_lowwater = free_mem_list_len;
log_info(LD_GENERAL, "Got buf mem from freelist. Freelist has %d entries.",
free_mem_list_len);
} else {
@@ -209,6 +210,30 @@ buf_get_initial_mem(buf_t *buf)
buf->cur = buf->mem;
}
+/** DOCDOC 64k of 4k buffers. */
+#define BUF_FREELIST_SLACK 16
+
+/** DOCDOC */
+void
+buf_shrink_freelist(void)
+{
+ if (free_mem_list_lowwater > BUF_FREELIST_SLACK) {
+ int i;
+ log_info(LD_GENERAL, "We haven't used %d/%d allocated buffer memory "
+ "chunks since the last call(); freeing all but %d of them",
+ free_mem_list_lowwater, free_mem_list_len,
+ BUF_FREELIST_SLACK);
+ for (i = BUF_FREELIST_SLACK; i < free_mem_list_lowwater; ++i) {
+ char *mem = free_mem_list;
+ tor_assert(mem);
+ free_mem_list = *(char**)mem;
+ tor_free(mem);
+ --free_mem_list_len;
+ }
+ }
+ free_mem_list_lowwater = free_mem_list_len;
+}
+
/** Change a buffer's capacity. <b>new_capacity</b> must be \>=
* buf->datalen. */
static void
diff --git a/src/or/main.c b/src/or/main.c
index d87dd7fecf..2c2387255a 100644
--- a/src/or/main.c
+++ b/src/or/main.c
@@ -1021,6 +1021,7 @@ run_scheduled_events(time_t now)
buf_shrink(conn->inbuf);
}
clean_cell_pool();
+ buf_shrink_freelist();
time_to_shrink_memory = now + MEM_SHRINK_INTERVAL;
}
diff --git a/src/or/or.h b/src/or/or.h
index f07865d82d..4d3f193320 100644
--- a/src/or/or.h
+++ b/src/or/or.h
@@ -2020,6 +2020,7 @@ buf_t *buf_new_with_capacity(size_t size);
void buf_free(buf_t *buf);
void buf_clear(buf_t *buf);
void buf_shrink(buf_t *buf);
+void buf_shrink_freelist(void);
size_t buf_datalen(const buf_t *buf);
size_t buf_capacity(const buf_t *buf);