* Implemented a very simple wall checker for the hoard allocator to

track down bug #1862.
* It's not complete for now, as it does not support memalign() or
  posix_memalign() yet - handle with care, or feel free to complete :-)


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@24162 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2008-02-28 00:27:30 +00:00
parent d863e59794
commit 79e518749d
2 changed files with 97 additions and 5 deletions

View File

@ -51,7 +51,6 @@
enum { SUPERBLOCK_FULLNESS_GROUP = 9 }; enum { SUPERBLOCK_FULLNESS_GROUP = 9 };
// DO NOT CHANGE THESE. They require running of maketable to replace // DO NOT CHANGE THESE. They require running of maketable to replace
// the values in heap.cpp for the _numBlocks array. // the values in heap.cpp for the _numBlocks array.
@ -63,6 +62,10 @@ enum { SUPERBLOCK_FULLNESS_GROUP = 9 };
#define HEAP_LEAK_CHECK 0 #define HEAP_LEAK_CHECK 0
#define HEAP_CALL_STACK_SIZE 8 #define HEAP_CALL_STACK_SIZE 8
// A simple wall checker
#define HEAP_WALL 1
#define HEAP_WALL_SIZE 64
// CACHE_LINE = The number of bytes in a cache line. // CACHE_LINE = The number of bytes in a cache line.
#if defined(i386) || defined(WIN32) #if defined(i386) || defined(WIN32)

View File

@ -189,8 +189,50 @@ remove_address(void* address)
hoardUnlock(sUsedLock); hoardUnlock(sUsedLock);
} }
#endif // HEAP_LEAK_CHECK #endif // HEAP_LEAK_CHECK
#if HEAP_WALL
static void*
set_wall(void* addr, size_t size)
{
size_t *start = (size_t*)addr;
start[0] = size;
memset(start + 1, 0x88, HEAP_WALL_SIZE - sizeof(size_t));
memset((uint8*)addr + size - HEAP_WALL_SIZE, 0x66, HEAP_WALL_SIZE);
return (uint8*)addr + HEAP_WALL_SIZE;
}
static void*
check_wall(uint8* buffer)
{
buffer -= HEAP_WALL_SIZE;
size_t size = *(size_t*)buffer;
if (threadHeap::objectSize(buffer) < size)
debugger("invalid size");
for (size_t i = 0; i < HEAP_WALL_SIZE; i++) {
if (i >= sizeof(size_t) && buffer[i] != 0x88) {
debug_printf("allocation %p, size %ld front wall clobbered at byte %ld.\n",
buffer + HEAP_WALL_SIZE, size - 2 * HEAP_WALL_SIZE, i);
debugger("front wall clobbered");
}
if (buffer[i + size - HEAP_WALL_SIZE] != 0x66) {
debug_printf("allocation %p, size %ld back wall clobbered at byte %ld.\n",
buffer + HEAP_WALL_SIZE, size - 2 * HEAP_WALL_SIZE, i);
debugger("back wall clobbered");
}
}
return buffer;
}
#endif // HEAP_WALL
inline static processHeap * inline static processHeap *
getAllocator(void) getAllocator(void)
@ -210,6 +252,10 @@ malloc(size_t size)
{ {
static processHeap *pHeap = getAllocator(); static processHeap *pHeap = getAllocator();
#if HEAP_WALL
size += 2 * HEAP_WALL_SIZE;
#endif
void *addr = pHeap->getHeap(pHeap->getHeapIndex()).malloc(size); void *addr = pHeap->getHeap(pHeap->getHeapIndex()).malloc(size);
if (addr == NULL) { if (addr == NULL) {
errno = B_NO_MEMORY; errno = B_NO_MEMORY;
@ -219,6 +265,9 @@ malloc(size_t size)
#if HEAP_LEAK_CHECK #if HEAP_LEAK_CHECK
add_address(addr, size); add_address(addr, size);
#endif #endif
#if HEAP_WALL
addr = set_wall(addr, size);
#endif
return addr; return addr;
} }
@ -228,18 +277,28 @@ extern "C" void *
calloc(size_t nelem, size_t elsize) calloc(size_t nelem, size_t elsize)
{ {
static processHeap *pHeap = getAllocator(); static processHeap *pHeap = getAllocator();
void *ptr = pHeap->getHeap(pHeap->getHeapIndex()).malloc(nelem * elsize); size_t size = nelem * elsize;
#if HEAP_WALL
size += 2 * HEAP_WALL_SIZE;
#endif
void *ptr = pHeap->getHeap(pHeap->getHeapIndex()).malloc(size);
if (ptr == NULL) { if (ptr == NULL) {
errno = B_NO_MEMORY; errno = B_NO_MEMORY;
return NULL; return NULL;
} }
#if HEAP_LEAK_CHECK #if HEAP_LEAK_CHECK
add_address(ptr, nelem * elsize); add_address(ptr, size);
#endif
#if HEAP_WALL
ptr = set_wall(ptr, size);
size -= 2 * HEAP_WALL_SIZE;
#endif #endif
// Zero out the malloc'd block. // Zero out the malloc'd block.
memset(ptr, 0, nelem * elsize); memset(ptr, 0, size);
return ptr; return ptr;
} }
@ -248,6 +307,11 @@ extern "C" void
free(void *ptr) free(void *ptr)
{ {
static processHeap *pHeap = getAllocator(); static processHeap *pHeap = getAllocator();
#if HEAP_WALL
if (ptr == NULL)
return;
ptr = check_wall((uint8*)ptr);
#endif
#if HEAP_LEAK_CHECK #if HEAP_LEAK_CHECK
if (ptr != NULL) if (ptr != NULL)
remove_address(ptr); remove_address(ptr);
@ -260,6 +324,11 @@ extern "C" void *
memalign(size_t alignment, size_t size) memalign(size_t alignment, size_t size)
{ {
static processHeap *pHeap = getAllocator(); static processHeap *pHeap = getAllocator();
#if HEAP_WALL
debug_printf("memalign() is not yet supported by the wall code.\n");
return NULL;
#endif
void *addr = pHeap->getHeap(pHeap->getHeapIndex()).memalign(alignment, void *addr = pHeap->getHeap(pHeap->getHeapIndex()).memalign(alignment,
size); size);
if (addr == NULL) { if (addr == NULL) {
@ -281,6 +350,10 @@ posix_memalign(void **_pointer, size_t alignment, size_t size)
if ((alignment & (sizeof(void *) - 1)) != 0 || _pointer == NULL) if ((alignment & (sizeof(void *) - 1)) != 0 || _pointer == NULL)
return B_BAD_VALUE; return B_BAD_VALUE;
#if HEAP_WALL
debug_printf("posix_memalign() is not yet supported by the wall code.\n");
return -1;
#endif
static processHeap *pHeap = getAllocator(); static processHeap *pHeap = getAllocator();
void *pointer = pHeap->getHeap(pHeap->getHeapIndex()).memalign(alignment, void *pointer = pHeap->getHeap(pHeap->getHeapIndex()).memalign(alignment,
size); size);
@ -317,9 +390,25 @@ realloc(void *ptr, size_t size)
// If the existing object can hold the new size, // If the existing object can hold the new size,
// just return it. // just return it.
#if HEAP_WALL
size += 2 * HEAP_WALL_SIZE;
ptr = (uint8*)ptr - HEAP_WALL_SIZE;
#endif
size_t objSize = threadHeap::objectSize(ptr); size_t objSize = threadHeap::objectSize(ptr);
if (objSize >= size) if (objSize >= size) {
#if HEAP_WALL
check_wall((uint8*)ptr + HEAP_WALL_SIZE);
ptr = set_wall(ptr, size);
#endif
return ptr; return ptr;
}
#if HEAP_WALL
size -= 2 * HEAP_WALL_SIZE;
objSize -= 2 * HEAP_WALL_SIZE;
ptr = (uint8*)ptr + HEAP_WALL_SIZE;
#endif
// Allocate a new block of size sz. // Allocate a new block of size sz.
void *buffer = malloc(size); void *buffer = malloc(size);