2013-01-15 16:04:27 +08:00
|
|
|
//===-- sanitizer_allocator_testlib.cc ------------------------------------===//
|
2012-07-19 00:04:55 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Malloc replacement library based on CombinedAllocator.
|
|
|
|
// The primary purpose of this file is an end-to-end integration test
|
|
|
|
// for CombinedAllocator.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2012-12-27 15:37:24 +08:00
|
|
|
/* Usage:
|
|
|
|
clang++ -fno-exceptions -g -fPIC -I. -I../include -Isanitizer \
|
2013-01-15 16:04:27 +08:00
|
|
|
sanitizer_common/tests/sanitizer_allocator_testlib.cc \
|
2013-01-15 16:33:41 +08:00
|
|
|
sanitizer_common/sanitizer_*.cc -shared -lpthread -o testmalloc.so
|
2012-12-27 15:37:24 +08:00
|
|
|
LD_PRELOAD=`pwd`/testmalloc.so /your/app
|
|
|
|
*/
|
2012-12-05 18:09:15 +08:00
|
|
|
#include "sanitizer_common/sanitizer_allocator.h"
|
2012-12-27 15:37:24 +08:00
|
|
|
#include "sanitizer_common/sanitizer_common.h"
|
2012-07-19 00:04:55 +08:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
2013-01-15 16:33:41 +08:00
|
|
|
#include <string.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
|
|
|
|
#ifndef SANITIZER_MALLOC_HOOK
|
|
|
|
# define SANITIZER_MALLOC_HOOK(p, s)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SANITIZER_FREE_HOOK
|
|
|
|
# define SANITIZER_FREE_HOOK(p)
|
|
|
|
#endif
|
2012-07-19 00:04:55 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
static const uptr kAllocatorSpace = 0x600000000000ULL;
|
2013-01-15 16:33:41 +08:00
|
|
|
static const uptr kAllocatorSize = 0x10000000000ULL; // 1T.
|
2012-07-19 00:04:55 +08:00
|
|
|
|
2012-12-27 15:37:24 +08:00
|
|
|
typedef SizeClassAllocator64<kAllocatorSpace, kAllocatorSize, 0,
|
|
|
|
CompactSizeClassMap> PrimaryAllocator;
|
|
|
|
typedef SizeClassAllocatorLocalCache<PrimaryAllocator> AllocatorCache;
|
2012-12-12 22:32:18 +08:00
|
|
|
typedef LargeMmapAllocator<> SecondaryAllocator;
|
2012-07-19 00:04:55 +08:00
|
|
|
typedef CombinedAllocator<PrimaryAllocator, AllocatorCache,
|
|
|
|
SecondaryAllocator> Allocator;
|
|
|
|
|
|
|
|
static Allocator allocator;
|
2013-01-15 20:57:02 +08:00
|
|
|
static bool global_inited;
|
2013-01-15 16:33:41 +08:00
|
|
|
static THREADLOCAL AllocatorCache cache;
|
|
|
|
static THREADLOCAL bool thread_inited;
|
|
|
|
static pthread_key_t pkey;
|
2012-07-19 00:04:55 +08:00
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
static void thread_dtor(void *v) {
|
2013-01-15 16:51:10 +08:00
|
|
|
if ((uptr)v != 3) {
|
|
|
|
pthread_setspecific(pkey, (void*)((uptr)v + 1));
|
2013-01-15 16:33:41 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
allocator.SwallowCache(&cache);
|
2012-07-19 00:04:55 +08:00
|
|
|
}
|
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
static void NOINLINE thread_init() {
|
|
|
|
if (!global_inited) {
|
|
|
|
global_inited = true;
|
|
|
|
allocator.Init();
|
|
|
|
pthread_key_create(&pkey, thread_dtor);
|
|
|
|
}
|
|
|
|
thread_inited = true;
|
|
|
|
pthread_setspecific(pkey, (void*)1);
|
|
|
|
cache.Init();
|
|
|
|
}
|
2012-07-19 00:04:55 +08:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
void *malloc(size_t size) {
|
2013-01-15 16:33:41 +08:00
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
void *p = allocator.Allocate(&cache, size, 8);
|
|
|
|
SANITIZER_MALLOC_HOOK(p, size);
|
|
|
|
return p;
|
2012-07-19 00:04:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void free(void *p) {
|
2013-01-15 16:33:41 +08:00
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
SANITIZER_FREE_HOOK(p);
|
2012-07-19 00:04:55 +08:00
|
|
|
allocator.Deallocate(&cache, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *calloc(size_t nmemb, size_t size) {
|
2013-01-15 16:33:41 +08:00
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
size *= nmemb;
|
|
|
|
void *p = allocator.Allocate(&cache, size, 8, false);
|
|
|
|
memset(p, 0, size);
|
|
|
|
SANITIZER_MALLOC_HOOK(p, size);
|
|
|
|
return p;
|
2012-07-19 00:04:55 +08:00
|
|
|
}
|
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
void *realloc(void *p, size_t size) {
|
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
if (p) {
|
|
|
|
SANITIZER_FREE_HOOK(p);
|
|
|
|
}
|
|
|
|
p = allocator.Reallocate(&cache, p, size, 8);
|
|
|
|
if (p) {
|
|
|
|
SANITIZER_MALLOC_HOOK(p, size);
|
|
|
|
}
|
|
|
|
return p;
|
2012-07-19 00:04:55 +08:00
|
|
|
}
|
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
void *memalign(size_t alignment, size_t size) {
|
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
void *p = allocator.Allocate(&cache, size, alignment);
|
|
|
|
SANITIZER_MALLOC_HOOK(p, size);
|
|
|
|
return p;
|
2012-12-27 15:37:24 +08:00
|
|
|
}
|
2012-07-19 20:15:33 +08:00
|
|
|
|
|
|
|
int posix_memalign(void **memptr, size_t alignment, size_t size) {
|
2013-01-15 16:33:41 +08:00
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
2012-07-19 20:15:33 +08:00
|
|
|
*memptr = allocator.Allocate(&cache, size, alignment);
|
2013-01-15 16:33:41 +08:00
|
|
|
SANITIZER_MALLOC_HOOK(*memptr, size);
|
2012-07-19 20:15:33 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *valloc(size_t size) {
|
2013-01-15 16:33:41 +08:00
|
|
|
if (UNLIKELY(!thread_inited))
|
|
|
|
thread_init();
|
|
|
|
if (size == 0)
|
|
|
|
size = GetPageSizeCached();
|
|
|
|
void *p = allocator.Allocate(&cache, size, GetPageSizeCached());
|
|
|
|
SANITIZER_MALLOC_HOOK(p, size);
|
|
|
|
return p;
|
2012-07-19 20:15:33 +08:00
|
|
|
}
|
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
void cfree(void *p) ALIAS("free");
|
|
|
|
void *pvalloc(size_t size) ALIAS("valloc");
|
|
|
|
void *__libc_memalign(size_t alignment, size_t size) ALIAS("memalign");
|
|
|
|
|
|
|
|
void malloc_usable_size() {
|
2012-07-19 20:15:33 +08:00
|
|
|
}
|
2012-12-27 15:37:24 +08:00
|
|
|
|
2013-01-15 16:33:41 +08:00
|
|
|
void mallinfo() {
|
2012-07-19 00:04:55 +08:00
|
|
|
}
|
2013-01-15 16:33:41 +08:00
|
|
|
|
|
|
|
void mallopt() {
|
|
|
|
}
|
|
|
|
} // extern "C"
|
2013-01-15 20:57:02 +08:00
|
|
|
|
|
|
|
namespace std {
|
|
|
|
struct nothrow_t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *operator new(size_t size) ALIAS("malloc");
|
|
|
|
void *operator new[](size_t size) ALIAS("malloc");
|
|
|
|
void *operator new(size_t size, std::nothrow_t const&) ALIAS("malloc");
|
|
|
|
void *operator new[](size_t size, std::nothrow_t const&) ALIAS("malloc");
|
|
|
|
void operator delete(void *ptr) ALIAS("free");
|
|
|
|
void operator delete[](void *ptr) ALIAS("free");
|
|
|
|
void operator delete(void *ptr, std::nothrow_t const&) ALIAS("free");
|
|
|
|
void operator delete[](void *ptr, std::nothrow_t const&) ALIAS("free");
|