ce426f
diff -Nru glibc-2.17-c758a686/malloc/arena.c glibc-2.17-c758a686/malloc/arena.c
ce426f
--- glibc-2.17-c758a686/malloc/arena.c	2012-05-29 16:45:53.000000000 -0600
ce426f
+++ glibc-2.17-c758a686/malloc/arena.c	2012-05-30 00:13:40.683514016 -0600
ce426f
@@ -673,7 +673,7 @@ heap_trim(heap_info *heap, size_t pad)
ce426f
     heap = prev_heap;
ce426f
     if(!prev_inuse(p)) { /* consolidate backward */
ce426f
       p = prev_chunk(p);
ce426f
-      unlink(p, bck, fwd);
ce426f
+      unlink(ar_ptr, p, bck, fwd);
ce426f
     }
ce426f
     assert(((unsigned long)((char*)p + new_size) & (pagesz-1)) == 0);
ce426f
     assert( ((char*)p + new_size) == ((char*)heap + heap->size) );
ce426f
diff -Nru glibc-2.17-c758a686/malloc/hooks.c glibc-2.17-c758a686/malloc/hooks.c
ce426f
--- glibc-2.17-c758a686/malloc/hooks.c	2012-05-29 16:45:53.000000000 -0600
ce426f
+++ glibc-2.17-c758a686/malloc/hooks.c	2012-05-30 00:13:40.684514011 -0600
ce426f
@@ -191,7 +191,9 @@ top_check(void)
ce426f
 	(char*)t + chunksize(t) == mp_.sbrk_base + main_arena.system_mem)))
ce426f
     return 0;
ce426f
 
ce426f
+  mutex_unlock(&main_arena);
ce426f
   malloc_printerr (check_action, "malloc: top chunk is corrupt", t);
ce426f
+  mutex_lock(&main_arena);
ce426f
 
ce426f
   /* Try to set up a new top chunk. */
ce426f
   brk = MORECORE(0);
ce426f
diff -Nru glibc-2.17-c758a686/malloc/malloc.c glibc-2.17-c758a686/malloc/malloc.c
ce426f
--- glibc-2.17-c758a686/malloc/malloc.c	2012-05-29 16:45:53.000000000 -0600
ce426f
+++ glibc-2.17-c758a686/malloc/malloc.c	2012-05-30 00:13:40.686514001 -0600
ce426f
@@ -1424,12 +1424,14 @@ typedef struct malloc_chunk* mbinptr;
ce426f
 #define last(b)      ((b)->bk)
ce426f
 
ce426f
 /* Take a chunk off a bin list */
ce426f
-#define unlink(P, BK, FD) {                                            \
ce426f
+#define unlink(AV, P, BK, FD) {					       \
ce426f
   FD = P->fd;                                                          \
ce426f
   BK = P->bk;                                                          \
ce426f
-  if (__builtin_expect (FD->bk != P || BK->fd != P, 0))                \
ce426f
+  if (__builtin_expect (FD->bk != P || BK->fd != P, 0)) {	       \
ce426f
+    mutex_unlock(&(AV)->mutex);					       \
ce426f
     malloc_printerr (check_action, "corrupted double-linked list", P); \
ce426f
-  else {                                                               \
ce426f
+    mutex_lock(&(AV)->mutex);					       \
ce426f
+  } else {							       \
ce426f
     FD->bk = BK;                                                       \
ce426f
     BK->fd = FD;                                                       \
ce426f
     if (!in_smallbin_range (P->size)				       \
ce426f
@@ -2511,7 +2513,9 @@ static void* sysmalloc(INTERNAL_SIZE_T n
ce426f
 
ce426f
     else if (contiguous(av) && old_size && brk < old_end) {
ce426f
       /* Oops!  Someone else killed our space..  Can't touch anything.  */
ce426f
+      mutex_unlock(&av->mutex);
ce426f
       malloc_printerr (3, "break adjusted to free malloc space", brk);
ce426f
+      mutex_lock(&av->mutex);
ce426f
     }
ce426f
 
ce426f
     /*
ce426f
@@ -3345,7 +3349,9 @@ _int_malloc(mstate av, size_t bytes)
ce426f
 	{
ce426f
 	  errstr = "malloc(): memory corruption (fast)";
ce426f
 	errout:
ce426f
+	  mutex_unlock(&av->mutex);
ce426f
 	  malloc_printerr (check_action, errstr, chunk2mem (victim));
ce426f
+	  mutex_lock(&av->mutex);
ce426f
 	  return NULL;
ce426f
 	}
ce426f
       check_remalloced_chunk(av, victim, nb);
ce426f
@@ -3430,8 +3436,12 @@ _int_malloc(mstate av, size_t bytes)
ce426f
       bck = victim->bk;
ce426f
       if (__builtin_expect (victim->size <= 2 * SIZE_SZ, 0)
ce426f
 	  || __builtin_expect (victim->size > av->system_mem, 0))
ce426f
-	malloc_printerr (check_action, "malloc(): memory corruption",
ce426f
-			 chunk2mem (victim));
ce426f
+	{
ce426f
+	  void *p = chunk2mem(victim);
ce426f
+	  mutex_unlock(&av->mutex);
ce426f
+	  malloc_printerr (check_action, "malloc(): memory corruption", p);
ce426f
+	  mutex_lock(&av->mutex);
ce426f
+	}
ce426f
       size = chunksize(victim);
ce426f
 
ce426f
       /*
ce426f
@@ -3572,7 +3582,7 @@ _int_malloc(mstate av, size_t bytes)
ce426f
 	  victim = victim->fd;
ce426f
 
ce426f
 	remainder_size = size - nb;
ce426f
-	unlink(victim, bck, fwd);
ce426f
+	unlink(av, victim, bck, fwd);
ce426f
 
ce426f
 	/* Exhaust */
ce426f
 	if (remainder_size < MINSIZE)  {
ce426f
@@ -3670,7 +3680,7 @@ _int_malloc(mstate av, size_t bytes)
ce426f
 	remainder_size = size - nb;
ce426f
 
ce426f
 	/* unlink */
ce426f
-	unlink(victim, bck, fwd);
ce426f
+	unlink(av, victim, bck, fwd);
ce426f
 
ce426f
 	/* Exhaust */
ce426f
 	if (remainder_size < MINSIZE) {
ce426f
@@ -3805,9 +3815,11 @@ _int_free(mstate av, mchunkptr p, int ha
ce426f
     {
ce426f
       errstr = "free(): invalid pointer";
ce426f
     errout:
ce426f
-      if (! have_lock && locked)
ce426f
+      if (have_lock || locked)
ce426f
 	(void)mutex_unlock(&av->mutex);
ce426f
       malloc_printerr (check_action, errstr, chunk2mem(p));
ce426f
+      if (have_lock)
ce426f
+	mutex_lock(&av->mutex);
ce426f
       return;
ce426f
     }
ce426f
   /* We know that each chunk is at least MINSIZE bytes in size or a
ce426f
@@ -3952,7 +3964,7 @@ _int_free(mstate av, mchunkptr p, int ha
ce426f
       prevsize = p->prev_size;
ce426f
       size += prevsize;
ce426f
       p = chunk_at_offset(p, -((long) prevsize));
ce426f
-      unlink(p, bck, fwd);
ce426f
+      unlink(av, p, bck, fwd);
ce426f
     }
ce426f
 
ce426f
     if (nextchunk != av->top) {
ce426f
@@ -3961,7 +3973,7 @@ _int_free(mstate av, mchunkptr p, int ha
ce426f
 
ce426f
       /* consolidate forward */
ce426f
       if (!nextinuse) {
ce426f
-	unlink(nextchunk, bck, fwd);
ce426f
+	unlink(av, nextchunk, bck, fwd);
ce426f
 	size += nextsize;
ce426f
       } else
ce426f
 	clear_inuse_bit_at_offset(nextchunk, 0);
ce426f
@@ -4122,7 +4134,7 @@ static void malloc_consolidate(mstate av
ce426f
 	    prevsize = p->prev_size;
ce426f
 	    size += prevsize;
ce426f
 	    p = chunk_at_offset(p, -((long) prevsize));
ce426f
-	    unlink(p, bck, fwd);
ce426f
+	    unlink(av, p, bck, fwd);
ce426f
 	  }
ce426f
 
ce426f
 	  if (nextchunk != av->top) {
ce426f
@@ -4130,7 +4142,7 @@ static void malloc_consolidate(mstate av
ce426f
 
ce426f
 	    if (!nextinuse) {
ce426f
 	      size += nextsize;
ce426f
-	      unlink(nextchunk, bck, fwd);
ce426f
+	      unlink(av, nextchunk, bck, fwd);
ce426f
 	    } else
ce426f
 	      clear_inuse_bit_at_offset(nextchunk, 0);
ce426f
 
ce426f
@@ -4199,7 +4211,9 @@ _int_realloc(mstate av, mchunkptr oldp,
ce426f
     {
ce426f
       errstr = "realloc(): invalid old size";
ce426f
     errout:
ce426f
+      mutex_unlock(&av->mutex);
ce426f
       malloc_printerr (check_action, errstr, chunk2mem(oldp));
ce426f
+      mutex_lock(&av->mutex);
ce426f
       return NULL;
ce426f
     }
ce426f
 
ce426f
@@ -4241,7 +4255,7 @@ _int_realloc(mstate av, mchunkptr oldp,
ce426f
 	     (unsigned long)(newsize = oldsize + nextsize) >=
ce426f
 	     (unsigned long)(nb)) {
ce426f
       newp = oldp;
ce426f
-      unlink(next, bck, fwd);
ce426f
+      unlink(av, next, bck, fwd);
ce426f
     }
ce426f
 
ce426f
     /* allocate, copy, free */