* 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:
parent
d863e59794
commit
79e518749d
@ -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)
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user