Fix leak_realloc, copy entire allocation.

Bug: 16874447

(cherry picked from commit 5df0839cea)

Change-Id: I9280505c0c1c3b5da24ba590448dcd6e7a230406
This commit is contained in:
Christopher Ferris 2014-08-15 18:42:58 -07:00
parent 5d9e145c62
commit 6d40d34908

View File

@ -385,6 +385,36 @@ extern "C" void* leak_calloc(size_t n_elements, size_t elem_size) {
return ptr; return ptr;
} }
extern "C" size_t leak_malloc_usable_size(const void* mem) {
if (DebugCallsDisabled()) {
return g_malloc_dispatch->malloc_usable_size(mem);
}
if (mem == NULL) {
return 0;
}
// Check the guard to make sure it is valid.
const AllocationEntry* header = const_to_header(mem);
if (header->guard == MEMALIGN_GUARD) {
// If this is a memalign'd pointer, then grab the header from
// entry.
header = const_to_header(header->entry);
} else if (header->guard != GUARD) {
debug_log("WARNING bad header guard: '0x%x'! and invalid entry: %p\n",
header->guard, header->entry);
return 0;
}
size_t ret = g_malloc_dispatch->malloc_usable_size(header);
if (ret != 0) {
// The usable area starts at 'mem' and stops at 'header+ret'.
return reinterpret_cast<uintptr_t>(header) + ret - reinterpret_cast<uintptr_t>(mem);
}
return 0;
}
extern "C" void* leak_realloc(void* oldMem, size_t bytes) { extern "C" void* leak_realloc(void* oldMem, size_t bytes) {
if (DebugCallsDisabled()) { if (DebugCallsDisabled()) {
return g_malloc_dispatch->realloc(oldMem, bytes); return g_malloc_dispatch->realloc(oldMem, bytes);
@ -408,7 +438,7 @@ extern "C" void* leak_realloc(void* oldMem, size_t bytes) {
newMem = leak_malloc(bytes); newMem = leak_malloc(bytes);
if (newMem != NULL) { if (newMem != NULL) {
size_t oldSize = header->entry->size & ~SIZE_FLAG_MASK; size_t oldSize = leak_malloc_usable_size(oldMem);
size_t copySize = (oldSize <= bytes) ? oldSize : bytes; size_t copySize = (oldSize <= bytes) ? oldSize : bytes;
memcpy(newMem, oldMem, copySize); memcpy(newMem, oldMem, copySize);
leak_free(oldMem); leak_free(oldMem);
@ -462,38 +492,6 @@ extern "C" void* leak_memalign(size_t alignment, size_t bytes) {
return base; return base;
} }
extern "C" size_t leak_malloc_usable_size(const void* mem) {
if (DebugCallsDisabled()) {
return g_malloc_dispatch->malloc_usable_size(mem);
}
if (mem != NULL) {
// Check the guard to make sure it is valid.
const AllocationEntry* header = const_to_header((void*)mem);
if (header->guard == MEMALIGN_GUARD) {
// If this is a memalign'd pointer, then grab the header from
// entry.
header = const_to_header(header->entry);
} else if (header->guard != GUARD) {
debug_log("WARNING bad header guard: '0x%x'! and invalid entry: %p\n",
header->guard, header->entry);
return 0;
}
// TODO: Temporary workaround to avoid a crash b/16874447.
return header->entry->size & ~SIZE_FLAG_MASK;
#if 0
size_t ret = g_malloc_dispatch->malloc_usable_size(header);
if (ret != 0) {
// The usable area starts at 'mem' and stops at 'header+ret'.
return reinterpret_cast<uintptr_t>(header) + ret - reinterpret_cast<uintptr_t>(mem);
}
#endif
}
return 0;
}
extern "C" struct mallinfo leak_mallinfo() { extern "C" struct mallinfo leak_mallinfo() {
return g_malloc_dispatch->mallinfo(); return g_malloc_dispatch->mallinfo();
} }