[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]

[or-cvs] r13428: Add a couple of (currently disabled) strategies for trying t (in tor/trunk: . src/common src/or)



Author: nickm
Date: 2008-02-08 16:13:08 -0500 (Fri, 08 Feb 2008)
New Revision: 13428

Modified:
   tor/trunk/
   tor/trunk/ChangeLog
   tor/trunk/src/common/mempool.c
   tor/trunk/src/common/mempool.h
   tor/trunk/src/or/relay.c
   tor/trunk/src/or/test.c
Log:
 r14061@tombo:  nickm | 2008-02-08 14:30:42 -0500
 Add a couple of (currently disabled) strategies for trying to avoid using too much ram in memory pools: prefer putting new cells in almost-full chunks, and be willing to free the last empty chunk if we have not needed it for a while.  Also add better output to mp_pool_log_status to track how many mallocs a given memory pool strategy is saving us, so we can tune the mempool parameters.



Property changes on: tor/trunk
___________________________________________________________________
 svk:merge ticket from /tor/trunk [r14061] on 49666b30-7950-49c5-bedf-9dc8f3168102

Modified: tor/trunk/ChangeLog
===================================================================
--- tor/trunk/ChangeLog	2008-02-08 21:13:05 UTC (rev 13427)
+++ tor/trunk/ChangeLog	2008-02-08 21:13:08 UTC (rev 13428)
@@ -12,6 +12,8 @@
       bandwidthburst values.
     - Give more descriptive well-formedness errors for out-of-range
       hidden service descriptor/protocol versions.
+    - Make memory debugging output describe more about history of cell
+      allocation.
 
   o Minor features (security):
     - Be slightly more paranoid about overwriting sensitive memory on free,

Modified: tor/trunk/src/common/mempool.c
===================================================================
--- tor/trunk/src/common/mempool.c	2008-02-08 21:13:05 UTC (rev 13427)
+++ tor/trunk/src/common/mempool.c	2008-02-08 21:13:08 UTC (rev 13428)
@@ -8,10 +8,12 @@
 
 #include <stdlib.h>
 #include <string.h>
-
+#include "torint.h"
 #define MEMPOOL_PRIVATE
 #include "mempool.h"
 
+//#define LAZY_CHUNK_SORT
+
 /* OVERVIEW:
  *
  *     This is an implementation of memory pools for Tor cells.  It may be
@@ -59,7 +61,6 @@
  *     if you need doubles.
  *   - Could probably be optimized a bit; the representation contains
  *     a bit more info than it really needs to have.
- *   - probably, chunks should always be a power of 2.
  */
 
 #if 1
@@ -174,6 +175,9 @@
 #else
   mp_chunk_t *chunk = ALLOC(CHUNK_OVERHEAD + sz);
 #endif
+#ifdef MEMPOOL_STATS
+  ++pool->total_chunks_allocated;
+#endif
   CHECK_ALLOC(chunk);
   memset(chunk, 0, sizeof(mp_chunk_t)); /* Doesn't clear the whole thing. */
   chunk->magic = MP_CHUNK_MAGIC;
@@ -189,6 +193,17 @@
   return chunk;
 }
 
+/** DOCDOC */
+static INLINE void
+add_newly_used_chunk_to_used_list(mp_pool_t *pool, mp_chunk_t *chunk)
+{
+  chunk->next = pool->used_chunks;
+  if (chunk->next)
+    chunk->next->prev = chunk;
+  pool->used_chunks = chunk;
+  ASSERT(!chunk->prev);
+}
+
 /** Return an newly allocated item from <b>pool</b>. */
 void *
 mp_pool_get(mp_pool_t *pool)
@@ -214,10 +229,7 @@
       chunk->next->prev = NULL;
 
     /* Put the chunk on the 'used' list*/
-    chunk->next = pool->used_chunks;
-    if (chunk->next)
-      chunk->next->prev = chunk;
-    pool->used_chunks = chunk;
+    add_newly_used_chunk_to_used_list(pool, chunk);
 
     ASSERT(!chunk->prev);
     --pool->n_empty_chunks;
@@ -229,11 +241,7 @@
     CHECK_ALLOC(chunk);
 
     /* Add the new chunk to the used list. */
-    chunk->next = pool->used_chunks;
-    if (chunk->next)
-      chunk->next->prev = chunk;
-    pool->used_chunks = chunk;
-    ASSERT(!chunk->prev);
+    add_newly_used_chunk_to_used_list(pool, chunk);
   }
 
   ASSERT(chunk->n_allocated < chunk->capacity);
@@ -258,6 +266,9 @@
   }
 
   ++chunk->n_allocated;
+#ifdef MEMPOOL_STATS
+  ++pool->total_items_allocated;
+#endif
 
   if (PREDICT_UNLIKELY(chunk->n_allocated == chunk->capacity)) {
     /* This chunk just became full. */
@@ -275,7 +286,6 @@
       chunk->next->prev = chunk;
     pool->full_chunks = chunk;
   }
-
   /* And return the memory portion of the mp_allocated_t. */
   return A2M(allocated);
 }
@@ -393,43 +403,111 @@
   return pool;
 }
 
+#ifdef LAZY_CHUNK_SORT
+/** DOCDOC */
+static int
+mp_pool_sort_used_chunks_helper(const void *_a, const void *_b)
+{
+  mp_chunk_t *a = *(mp_chunk_t**)_a;
+  mp_chunk_t *b = *(mp_chunk_t**)_b;
+  return b->n_allocated - a->n_allocated;
+}
+
+/** DOCDOC */
+static void
+mp_pool_sort_used_chunks(mp_pool_t *pool)
+{
+  int i, n=0, inverted=0;
+  mp_chunk_t **chunks, *chunk;
+  for (chunk = pool->used_chunks; chunk; chunk = chunk->next) {
+    ++n;
+    if (chunk->next && chunk->next->n_allocated > chunk->n_allocated)
+      ++inverted;
+  }
+  if (!inverted)
+    return;
+  ASSERT(n);
+  //printf("Sort %d/%d\n",inverted,n);
+  chunks = ALLOC(sizeof(mp_chunk_t *)*n);
+#ifdef ALLOC_CAN_RETURN_NULL
+  if (PREDICT_UNLIKELY(!chunks)) return;
+#endif
+  for (i=0,chunk = pool->used_chunks; chunk; chunk = chunk->next)
+    chunks[i++] = chunk;
+  qsort(chunks, n, sizeof(mp_chunk_t *), mp_pool_sort_used_chunks_helper);
+  pool->used_chunks = chunks[0];
+  chunks[0]->prev = NULL;
+  for (i=1;i<n;++i) {
+    chunks[i-1]->next = chunks[i];
+    chunks[i]->prev = chunks[i-1];
+  }
+  chunks[n-1]->next = NULL;
+  FREE(chunks);
+#if 0
+  inverted = 0;
+  for (chunk = pool->used_chunks; chunk; chunk = chunk->next) {
+    if (chunk->next) {
+      ASSERT(chunk->next->n_allocated <= chunk->n_allocated);
+    }
+  }
+#endif
+  mp_pool_assert_ok(pool);
+}
+#endif
+
 /** If there are more than <b>n</b> empty chunks in <b>pool</b>, free the
  * excess ones that have been empty for the longest.  (If <b>n</b> is less
  * than zero, free only empty chunks that were not used since the last
- * call to mp_pool_clean(), leaving only -<b>n</b>.) */
+ * call to mp_pool_clean(), leaving only -<b>n</b>.)
+ * DOCDOC Keep_recently_used, n_to_keep
+ * XXXX020 maybe dump negative n_to_keep behavior, if k_r_u turns out to be
+ *   smarter.
+ **/
 void
-mp_pool_clean(mp_pool_t *pool, int n)
+mp_pool_clean(mp_pool_t *pool, int n_to_keep, int keep_recently_used)
 {
-  /* XXXX020 this is stupid.  We shouldn't care about empty chunks if there
-   * is lots of space in used chunks. */
+  mp_chunk_t *chunk, **first_to_free;
 
-  mp_chunk_t *chunk, **first_to_free;
-  if (n < 0) {
+#ifdef LAZY_CHUNK_SORT
+  mp_pool_sort_used_chunks(pool);
+#endif
+
+  if (n_to_keep < 0) {
     /* As said in the documentation, "negative n" means "leave an additional
      * -n chunks". So replace n with a positive number. */
-    n = pool->min_empty_chunks + (-n);
-    if (n < pool->n_empty_chunks)
-      pool->min_empty_chunks = n;
+    n_to_keep = pool->min_empty_chunks + (-n_to_keep);
   }
-  ASSERT(n>=0);
+  if (keep_recently_used) {
+    int n_recently_used = pool->n_empty_chunks - pool->min_empty_chunks;
+    if (n_to_keep < n_recently_used)
+      n_to_keep = n_recently_used;
+  }
 
+  ASSERT(n_to_keep >= 0);
+
   first_to_free = &pool->empty_chunks;
-  while (*first_to_free && n > 0) {
+  while (*first_to_free && n_to_keep > 0) {
     first_to_free = &(*first_to_free)->next;
-    --n;
+    --n_to_keep;
   }
-  if (!*first_to_free)
+  if (!*first_to_free) {
+    pool->min_empty_chunks = pool->n_empty_chunks;
     return;
+  }
 
   chunk = *first_to_free;
   while (chunk) {
     mp_chunk_t *next = chunk->next;
     chunk->magic = 0xdeadbeef;
     FREE(chunk);
+#ifdef MEMPOOL_STATS
+    ++pool->total_chunks_freed;
+#endif
     --pool->n_empty_chunks;
     chunk = next;
   }
 
+  pool->min_empty_chunks = pool->n_empty_chunks;
   *first_to_free = NULL;
 }
 
@@ -535,6 +613,8 @@
     ++n_used;
     bu += chunk->n_allocated * pool->item_alloc_size;
     ba += chunk->mem_size;
+    log_fn(severity, LD_MM, "   used chunk: %d items allocated",
+           chunk->n_allocated);
   }
   log_fn(severity, LD_MM, U64_FORMAT"/"U64_FORMAT
          " bytes in %d partially full chunks",
@@ -556,6 +636,15 @@
   log_fn(severity, LD_MM, "Total: "U64_FORMAT"/"U64_FORMAT" bytes allocated "
          "for cell pools are full.",
          U64_PRINTF_ARG(bytes_used), U64_PRINTF_ARG(bytes_allocated));
+
+#ifdef MEMPOOL_STATS
+  log_fn(severity, LD_MM, U64_FORMAT" cell allocations ever; "
+         U64_FORMAT" chunk allocations ever; "
+         U64_FORMAT" chunk frees ever.",
+         U64_PRINTF_ARG(pool->total_items_allocated),
+         U64_PRINTF_ARG(pool->total_chunks_allocated),
+         U64_PRINTF_ARG(pool->total_chunks_freed));
+#endif
 }
 #endif
 

Modified: tor/trunk/src/common/mempool.h
===================================================================
--- tor/trunk/src/common/mempool.h	2008-02-08 21:13:05 UTC (rev 13427)
+++ tor/trunk/src/common/mempool.h	2008-02-08 21:13:08 UTC (rev 13428)
@@ -18,11 +18,13 @@
 void *mp_pool_get(mp_pool_t *pool);
 void mp_pool_release(void *item);
 mp_pool_t *mp_pool_new(size_t item_size, size_t chunk_capacity);
-void mp_pool_clean(mp_pool_t *pool, int n);
+void mp_pool_clean(mp_pool_t *pool, int n_to_keep, int keep_recently_used);
 void mp_pool_destroy(mp_pool_t *pool);
 void mp_pool_assert_ok(mp_pool_t *pool);
 void mp_pool_log_status(mp_pool_t *pool, int severity);
 
+#define MEMPOOL_STATS
+
 #ifdef MEMPOOL_PRIVATE
 /* These declarations are only used by mempool.c and test.c */
 
@@ -47,6 +49,14 @@
   /** Size to allocate for each item, including overhead and alignment
    * padding. */
   size_t item_alloc_size;
+#ifdef MEMPOOL_STATS
+  /** Total number of items allocated ever */
+  uint64_t total_items_allocated;
+  /** Total number of chunks allocated ever */
+  uint64_t total_chunks_allocated;
+  /** Total number of chunks freed ever */
+  uint64_t total_chunks_freed;
+#endif
 };
 #endif
 

Modified: tor/trunk/src/or/relay.c
===================================================================
--- tor/trunk/src/or/relay.c	2008-02-08 21:13:05 UTC (rev 13427)
+++ tor/trunk/src/or/relay.c	2008-02-08 21:13:08 UTC (rev 13428)
@@ -1538,7 +1538,7 @@
 clean_cell_pool(void)
 {
   tor_assert(cell_pool);
-  mp_pool_clean(cell_pool, -1);
+  mp_pool_clean(cell_pool, -1, 0);
 }
 
 /** Release storage held by <b>cell</b>. */

Modified: tor/trunk/src/or/test.c
===================================================================
--- tor/trunk/src/or/test.c	2008-02-08 21:13:05 UTC (rev 13427)
+++ tor/trunk/src/or/test.c	2008-02-08 21:13:08 UTC (rev 13428)
@@ -3120,14 +3120,14 @@
       //mp_pool_assert_ok(pool);
     }
     if (crypto_rand_int(777)==0)
-      mp_pool_clean(pool, -1);
+      mp_pool_clean(pool, -1, 0);
 
     if (i % 777)
       mp_pool_assert_ok(pool);
   }
   SMARTLIST_FOREACH(allocated, void *, m, mp_pool_release(m));
   mp_pool_assert_ok(pool);
-  mp_pool_clean(pool, 0);
+  mp_pool_clean(pool, 0, 0);
   mp_pool_assert_ok(pool);
   mp_pool_destroy(pool);
   smartlist_free(allocated);