Move malloc dispatch table to __libc_globals.

Change-Id: Ic20b980d1e8b6c2d4b773ebe336658fd17c737cb
This commit is contained in:
Josh Gao 2015-10-08 14:49:26 -07:00
parent 25cba7df51
commit 3c8fc2fea9
5 changed files with 153 additions and 80 deletions

View File

@ -50,11 +50,11 @@
#include <elf.h> #include <elf.h>
#include "libc_init_common.h" #include "libc_init_common.h"
#include "private/bionic_globals.h"
#include "private/bionic_tls.h" #include "private/bionic_tls.h"
#include "private/KernelArgumentBlock.h" #include "private/KernelArgumentBlock.h"
extern "C" { extern "C" {
extern void malloc_debug_init(void);
extern void malloc_debug_fini(void); extern void malloc_debug_fini(void);
extern void netdClientInit(void); extern void netdClientInit(void);
extern int __cxa_atexit(void (*)(void *), void *, void *); extern int __cxa_atexit(void (*)(void *), void *, void *);
@ -78,7 +78,7 @@ __attribute__((constructor)) static void __libc_preinit() {
__libc_init_common(*args); __libc_init_common(*args);
// Hooks for various libraries to let them know that we're starting up. // Hooks for various libraries to let them know that we're starting up.
malloc_debug_init(); __libc_globals.mutate(__libc_init_malloc);
netdClientInit(); netdClientInit();
} }

View File

@ -44,6 +44,7 @@
#include <string.h> #include <string.h>
#include <unistd.h> #include <unistd.h>
#include "private/bionic_globals.h"
#include "private/ScopedPthreadMutexLocker.h" #include "private/ScopedPthreadMutexLocker.h"
#if defined(USE_JEMALLOC) #if defined(USE_JEMALLOC)
@ -56,14 +57,8 @@
#error "Either one of USE_DLMALLOC or USE_JEMALLOC must be defined." #error "Either one of USE_DLMALLOC or USE_JEMALLOC must be defined."
#endif #endif
// In a VM process, this is set to 1 after fork()ing out of zygote. static constexpr MallocDebug __libc_malloc_default_dispatch
int gMallocLeakZygoteChild = 0; __attribute__((unused)) = {
static HashTable g_hash_table;
// Support for malloc debugging.
// Table for dispatching malloc calls, initialized with default dispatchers.
static const MallocDebug __libc_malloc_default_dispatch __attribute__((aligned(32))) = {
Malloc(calloc), Malloc(calloc),
Malloc(free), Malloc(free),
Malloc(mallinfo), Malloc(mallinfo),
@ -78,10 +73,12 @@ static const MallocDebug __libc_malloc_default_dispatch __attribute__((aligned(3
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS) #if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
Malloc(valloc), Malloc(valloc),
#endif #endif
}; };
// Selector of dispatch table to use for dispatching malloc calls. // In a VM process, this is set to 1 after fork()ing out of zygote.
static const MallocDebug* __libc_malloc_dispatch = &__libc_malloc_default_dispatch; int gMallocLeakZygoteChild = 0;
static HashTable g_hash_table;
// Handle to shared library where actual memory allocation is implemented. // Handle to shared library where actual memory allocation is implemented.
// This library is loaded and memory allocation calls are redirected there // This library is loaded and memory allocation calls are redirected there
@ -244,46 +241,87 @@ extern "C" void free_malloc_leak_info(uint8_t* info) {
// Allocation functions // Allocation functions
// ============================================================================= // =============================================================================
extern "C" void* calloc(size_t n_elements, size_t elem_size) { extern "C" void* calloc(size_t n_elements, size_t elem_size) {
return __libc_malloc_dispatch->calloc(n_elements, elem_size); auto _calloc = __libc_globals->malloc_dispatch.calloc;
if (__predict_false(_calloc != nullptr)) {
return _calloc(n_elements, elem_size);
}
return Malloc(calloc)(n_elements, elem_size);
} }
extern "C" void free(void* mem) { extern "C" void free(void* mem) {
__libc_malloc_dispatch->free(mem); auto _free = __libc_globals->malloc_dispatch.free;
if (__predict_false(_free != nullptr)) {
_free(mem);
} else {
Malloc(free)(mem);
}
} }
extern "C" struct mallinfo mallinfo() { extern "C" struct mallinfo mallinfo() {
return __libc_malloc_dispatch->mallinfo(); auto _mallinfo = __libc_globals->malloc_dispatch.mallinfo;
if (__predict_false(_mallinfo != nullptr)) {
return _mallinfo();
}
return Malloc(mallinfo)();
} }
extern "C" void* malloc(size_t bytes) { extern "C" void* malloc(size_t bytes) {
return __libc_malloc_dispatch->malloc(bytes); auto _malloc = __libc_globals->malloc_dispatch.malloc;
if (__predict_false(_malloc != nullptr)) {
return _malloc(bytes);
}
return Malloc(malloc)(bytes);
} }
extern "C" size_t malloc_usable_size(const void* mem) { extern "C" size_t malloc_usable_size(const void* mem) {
return __libc_malloc_dispatch->malloc_usable_size(mem); auto _malloc_usable_size = __libc_globals->malloc_dispatch.malloc_usable_size;
if (__predict_false(_malloc_usable_size != nullptr)) {
return _malloc_usable_size(mem);
}
return Malloc(malloc_usable_size)(mem);
} }
extern "C" void* memalign(size_t alignment, size_t bytes) { extern "C" void* memalign(size_t alignment, size_t bytes) {
return __libc_malloc_dispatch->memalign(alignment, bytes); auto _memalign = __libc_globals->malloc_dispatch.memalign;
if (__predict_false(_memalign != nullptr)) {
return _memalign(alignment, bytes);
}
return Malloc(memalign)(alignment, bytes);
} }
extern "C" int posix_memalign(void** memptr, size_t alignment, size_t size) { extern "C" int posix_memalign(void** memptr, size_t alignment, size_t size) {
return __libc_malloc_dispatch->posix_memalign(memptr, alignment, size); auto _posix_memalign = __libc_globals->malloc_dispatch.posix_memalign;
if (__predict_false(_posix_memalign != nullptr)) {
return _posix_memalign(memptr, alignment, size);
}
return Malloc(posix_memalign)(memptr, alignment, size);
} }
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS) #if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
extern "C" void* pvalloc(size_t bytes) { extern "C" void* pvalloc(size_t bytes) {
return __libc_malloc_dispatch->pvalloc(bytes); auto _pvalloc = __libc_globals->malloc_dispatch.pvalloc;
if (__predict_false(_pvalloc != nullptr)) {
return _pvalloc(bytes);
}
return Malloc(pvalloc)(bytes);
} }
#endif #endif
extern "C" void* realloc(void* oldMem, size_t bytes) { extern "C" void* realloc(void* old_mem, size_t bytes) {
return __libc_malloc_dispatch->realloc(oldMem, bytes); auto _realloc = __libc_globals->malloc_dispatch.realloc;
if (__predict_false(_realloc != nullptr)) {
return _realloc(old_mem, bytes);
}
return Malloc(realloc)(old_mem, bytes);
} }
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS) #if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
extern "C" void* valloc(size_t bytes) { extern "C" void* valloc(size_t bytes) {
return __libc_malloc_dispatch->valloc(bytes); auto _valloc = __libc_globals->malloc_dispatch.valloc;
if (__predict_false(_valloc != nullptr)) {
return _valloc(bytes);
}
return Malloc(valloc)(bytes);
} }
#endif #endif
@ -326,7 +364,7 @@ static void InitMalloc(void* malloc_impl_handler, MallocDebug* table, const char
} }
// Initializes memory allocation framework once per process. // Initializes memory allocation framework once per process.
static void malloc_init_impl() { static void malloc_init_impl(libc_globals* globals) {
const char* so_name = NULL; const char* so_name = NULL;
MallocDebugInit malloc_debug_initialize = NULL; MallocDebugInit malloc_debug_initialize = NULL;
unsigned int qemu_running = 0; unsigned int qemu_running = 0;
@ -442,7 +480,7 @@ static void malloc_init_impl() {
// No need to init the dispatch table because we can only get // No need to init the dispatch table because we can only get
// here if debug level is 1, 5, 10, or 20. // here if debug level is 1, 5, 10, or 20.
static MallocDebug malloc_dispatch_table __attribute__((aligned(32))); MallocDebug malloc_dispatch_table;
switch (g_malloc_debug_level) { switch (g_malloc_debug_level) {
case 1: case 1:
InitMalloc(malloc_impl_handle, &malloc_dispatch_table, "leak"); InitMalloc(malloc_impl_handle, &malloc_dispatch_table, "leak");
@ -480,7 +518,7 @@ static void malloc_init_impl() {
getprogname(), g_malloc_debug_level); getprogname(), g_malloc_debug_level);
dlclose(malloc_impl_handle); dlclose(malloc_impl_handle);
} else { } else {
__libc_malloc_dispatch = &malloc_dispatch_table; globals->malloc_dispatch = malloc_dispatch_table;
libc_malloc_impl_handle = malloc_impl_handle; libc_malloc_impl_handle = malloc_impl_handle;
} }
} }
@ -507,15 +545,12 @@ static void malloc_fini_impl() {
#endif // !LIBC_STATIC #endif // !LIBC_STATIC
// Initializes memory allocation framework. // Initializes memory allocation framework.
// This routine is called from __libc_init routines implemented // This routine is called from __libc_init routines in libc_init_dynamic.cpp.
// in libc_init_static.c and libc_init_dynamic.c files. __LIBC_HIDDEN__ void __libc_init_malloc(libc_globals* globals) {
extern "C" __LIBC_HIDDEN__ void malloc_debug_init() { (void)globals;
#if !defined(LIBC_STATIC) #if !defined(LIBC_STATIC)
static pthread_once_t malloc_init_once_ctl = PTHREAD_ONCE_INIT; malloc_init_impl(globals);
if (pthread_once(&malloc_init_once_ctl, malloc_init_impl)) { #endif
error_log("Unable to initialize malloc_debug component.");
}
#endif // !LIBC_STATIC
} }
extern "C" __LIBC_HIDDEN__ void malloc_debug_fini() { extern "C" __LIBC_HIDDEN__ void malloc_debug_fini() {

View File

@ -38,6 +38,7 @@
#include <stdlib.h> #include <stdlib.h>
#include "private/bionic_config.h" #include "private/bionic_config.h"
#include "private/bionic_malloc_dispatch.h"
#include "private/libc_logging.h" #include "private/libc_logging.h"
#define HASHTABLE_SIZE 1543 #define HASHTABLE_SIZE 1543
@ -72,39 +73,6 @@ struct HashTable {
HashEntry* slots[HASHTABLE_SIZE]; HashEntry* slots[HASHTABLE_SIZE];
}; };
/* Entry in malloc dispatch table. */
typedef void* (*MallocDebugCalloc)(size_t, size_t);
typedef void (*MallocDebugFree)(void*);
typedef struct mallinfo (*MallocDebugMallinfo)();
typedef void* (*MallocDebugMalloc)(size_t);
typedef size_t (*MallocDebugMallocUsableSize)(const void*);
typedef void* (*MallocDebugMemalign)(size_t, size_t);
typedef int (*MallocDebugPosixMemalign)(void**, size_t, size_t);
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
typedef void* (*MallocDebugPvalloc)(size_t);
#endif
typedef void* (*MallocDebugRealloc)(void*, size_t);
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
typedef void* (*MallocDebugValloc)(size_t);
#endif
struct MallocDebug {
MallocDebugCalloc calloc;
MallocDebugFree free;
MallocDebugMallinfo mallinfo;
MallocDebugMalloc malloc;
MallocDebugMallocUsableSize malloc_usable_size;
MallocDebugMemalign memalign;
MallocDebugPosixMemalign posix_memalign;
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
MallocDebugPvalloc pvalloc;
#endif
MallocDebugRealloc realloc;
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
MallocDebugValloc valloc;
#endif
};
typedef bool (*MallocDebugInit)(HashTable*, const MallocDebug*); typedef bool (*MallocDebugInit)(HashTable*, const MallocDebug*);
typedef void (*MallocDebugFini)(int); typedef void (*MallocDebugFini)(int);

View File

@ -29,12 +29,14 @@
#define _PRIVATE_BIONIC_GLOBALS_H #define _PRIVATE_BIONIC_GLOBALS_H
#include <sys/cdefs.h> #include <sys/cdefs.h>
#include "private/bionic_malloc_dispatch.h"
#include "private/bionic_vdso.h" #include "private/bionic_vdso.h"
#include "private/WriteProtected.h" #include "private/WriteProtected.h"
struct libc_globals { struct libc_globals {
vdso_entry vdso[VDSO_END]; vdso_entry vdso[VDSO_END];
long setjmp_cookie; long setjmp_cookie;
MallocDebug malloc_dispatch;
}; };
__LIBC_HIDDEN__ extern WriteProtected<libc_globals> __libc_globals; __LIBC_HIDDEN__ extern WriteProtected<libc_globals> __libc_globals;
@ -44,5 +46,5 @@ __LIBC_HIDDEN__ void __libc_init_vdso(libc_globals* globals,
KernelArgumentBlock& args); KernelArgumentBlock& args);
__LIBC_HIDDEN__ void __libc_init_setjmp_cookie(libc_globals* globals, __LIBC_HIDDEN__ void __libc_init_setjmp_cookie(libc_globals* globals,
KernelArgumentBlock& args); KernelArgumentBlock& args);
__LIBC_HIDDEN__ void __libc_init_malloc(libc_globals* globals);
#endif #endif

View File

@ -0,0 +1,68 @@
/*
* Copyright (C) 2015 The Android Open Source Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#ifndef _PRIVATE_BIONIC_MALLOC_DISPATCH_H
#define _PRIVATE_BIONIC_MALLOC_DISPATCH_H
#include <stddef.h>
#include "private/bionic_config.h"
/* Entry in malloc dispatch table. */
typedef void* (*MallocDebugCalloc)(size_t, size_t);
typedef void (*MallocDebugFree)(void*);
typedef struct mallinfo (*MallocDebugMallinfo)();
typedef void* (*MallocDebugMalloc)(size_t);
typedef size_t (*MallocDebugMallocUsableSize)(const void*);
typedef void* (*MallocDebugMemalign)(size_t, size_t);
typedef int (*MallocDebugPosixMemalign)(void**, size_t, size_t);
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
typedef void* (*MallocDebugPvalloc)(size_t);
#endif
typedef void* (*MallocDebugRealloc)(void*, size_t);
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
typedef void* (*MallocDebugValloc)(size_t);
#endif
struct MallocDebug {
MallocDebugCalloc calloc;
MallocDebugFree free;
MallocDebugMallinfo mallinfo;
MallocDebugMalloc malloc;
MallocDebugMallocUsableSize malloc_usable_size;
MallocDebugMemalign memalign;
MallocDebugPosixMemalign posix_memalign;
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
MallocDebugPvalloc pvalloc;
#endif
MallocDebugRealloc realloc;
#if defined(HAVE_DEPRECATED_MALLOC_FUNCS)
MallocDebugValloc valloc;
#endif
} __attribute__((aligned(32)));
#endif