Alexey Samsonov | e5f5895 | 2012-06-04 13:50:10 +0000 | [diff] [blame] | 1 | //===-- asan_malloc_linux.cc ----------------------------------------------===// |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
| 10 | // This file is a part of AddressSanitizer, an address sanity checker. |
| 11 | // |
| 12 | // Linux-specific malloc interception. |
| 13 | // We simply define functions like malloc, free, realloc, etc. |
| 14 | // They will replace the corresponding libc functions automagically. |
| 15 | //===----------------------------------------------------------------------===// |
Evgeniy Stepanov | 24e1372 | 2013-03-19 14:33:38 +0000 | [diff] [blame] | 16 | |
| 17 | #include "sanitizer_common/sanitizer_platform.h" |
Stephen Hines | 2d1fdb2 | 2014-05-28 23:58:16 -0700 | [diff] [blame] | 18 | #if SANITIZER_FREEBSD || SANITIZER_LINUX |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 19 | |
Stephen Hines | 2d1fdb2 | 2014-05-28 23:58:16 -0700 | [diff] [blame] | 20 | #include "sanitizer_common/sanitizer_tls_get_addr.h" |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 21 | #include "asan_allocator.h" |
| 22 | #include "asan_interceptors.h" |
| 23 | #include "asan_internal.h" |
| 24 | #include "asan_stack.h" |
| 25 | |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 26 | // ---------------------- Replacement functions ---------------- {{{1 |
| 27 | using namespace __asan; // NOLINT |
| 28 | |
Alexey Samsonov | f2598fc | 2012-02-02 10:39:40 +0000 | [diff] [blame] | 29 | INTERCEPTOR(void, free, void *ptr) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 30 | GET_STACK_TRACE_FREE; |
Kostya Serebryany | fe6d916 | 2012-12-21 08:53:59 +0000 | [diff] [blame] | 31 | asan_free(ptr, &stack, FROM_MALLOC); |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 32 | } |
| 33 | |
Alexey Samsonov | f2598fc | 2012-02-02 10:39:40 +0000 | [diff] [blame] | 34 | INTERCEPTOR(void, cfree, void *ptr) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 35 | GET_STACK_TRACE_FREE; |
Kostya Serebryany | fe6d916 | 2012-12-21 08:53:59 +0000 | [diff] [blame] | 36 | asan_free(ptr, &stack, FROM_MALLOC); |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 37 | } |
| 38 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 39 | INTERCEPTOR(void*, malloc, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 40 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 41 | return asan_malloc(size, &stack); |
| 42 | } |
| 43 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 44 | INTERCEPTOR(void*, calloc, uptr nmemb, uptr size) { |
Stephen Hines | 2d1fdb2 | 2014-05-28 23:58:16 -0700 | [diff] [blame] | 45 | if (UNLIKELY(!asan_inited)) { |
Alexey Samsonov | 09672ca | 2012-02-08 13:45:31 +0000 | [diff] [blame] | 46 | // Hack: dlsym calls calloc before REAL(calloc) is retrieved from dlsym. |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 47 | const uptr kCallocPoolSize = 1024; |
Kostya Serebryany | 3f4c387 | 2012-05-31 14:35:53 +0000 | [diff] [blame] | 48 | static uptr calloc_memory_for_dlsym[kCallocPoolSize]; |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 49 | static uptr allocated; |
| 50 | uptr size_in_words = ((nmemb * size) + kWordSize - 1) / kWordSize; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 51 | void *mem = (void*)&calloc_memory_for_dlsym[allocated]; |
| 52 | allocated += size_in_words; |
| 53 | CHECK(allocated < kCallocPoolSize); |
| 54 | return mem; |
| 55 | } |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 56 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 57 | return asan_calloc(nmemb, size, &stack); |
| 58 | } |
| 59 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 60 | INTERCEPTOR(void*, realloc, void *ptr, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 61 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 62 | return asan_realloc(ptr, size, &stack); |
| 63 | } |
| 64 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 65 | INTERCEPTOR(void*, memalign, uptr boundary, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 66 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | fe6d916 | 2012-12-21 08:53:59 +0000 | [diff] [blame] | 67 | return asan_memalign(boundary, size, &stack, FROM_MALLOC); |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 68 | } |
| 69 | |
Stephen Hines | 6a211c5 | 2014-07-21 00:49:56 -0700 | [diff] [blame] | 70 | INTERCEPTOR(void*, aligned_alloc, uptr boundary, uptr size) { |
| 71 | GET_STACK_TRACE_MALLOC; |
| 72 | return asan_memalign(boundary, size, &stack, FROM_MALLOC); |
| 73 | } |
| 74 | |
Stephen Hines | 2d1fdb2 | 2014-05-28 23:58:16 -0700 | [diff] [blame] | 75 | INTERCEPTOR(void*, __libc_memalign, uptr boundary, uptr size) { |
| 76 | GET_STACK_TRACE_MALLOC; |
| 77 | void *res = asan_memalign(boundary, size, &stack, FROM_MALLOC); |
| 78 | DTLS_on_libc_memalign(res, size * boundary); |
| 79 | return res; |
| 80 | } |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 81 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 82 | INTERCEPTOR(uptr, malloc_usable_size, void *ptr) { |
Alexey Samsonov | 1b17f5b | 2013-11-13 14:46:58 +0000 | [diff] [blame] | 83 | GET_CURRENT_PC_BP_SP; |
| 84 | (void)sp; |
| 85 | return asan_malloc_usable_size(ptr, pc, bp); |
Alexey Samsonov | 4fd95f1 | 2012-01-17 06:39:10 +0000 | [diff] [blame] | 86 | } |
| 87 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 88 | // We avoid including malloc.h for portability reasons. |
| 89 | // man mallinfo says the fields are "long", but the implementation uses int. |
| 90 | // It doesn't matter much -- we just need to make sure that the libc's mallinfo |
| 91 | // is not called. |
| 92 | struct fake_mallinfo { |
| 93 | int x[10]; |
| 94 | }; |
| 95 | |
| 96 | INTERCEPTOR(struct fake_mallinfo, mallinfo, void) { |
| 97 | struct fake_mallinfo res; |
Alexey Samsonov | 09672ca | 2012-02-08 13:45:31 +0000 | [diff] [blame] | 98 | REAL(memset)(&res, 0, sizeof(res)); |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 99 | return res; |
| 100 | } |
| 101 | |
Alexey Samsonov | f2598fc | 2012-02-02 10:39:40 +0000 | [diff] [blame] | 102 | INTERCEPTOR(int, mallopt, int cmd, int value) { |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 103 | return -1; |
| 104 | } |
| 105 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 106 | INTERCEPTOR(int, posix_memalign, void **memptr, uptr alignment, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 107 | GET_STACK_TRACE_MALLOC; |
Evgeniy Stepanov | 739eb79 | 2012-03-21 11:32:46 +0000 | [diff] [blame] | 108 | // Printf("posix_memalign: %zx %zu\n", alignment, size); |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 109 | return asan_posix_memalign(memptr, alignment, size, &stack); |
| 110 | } |
| 111 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 112 | INTERCEPTOR(void*, valloc, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 113 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 114 | return asan_valloc(size, &stack); |
| 115 | } |
| 116 | |
Kostya Serebryany | e7539bf | 2012-06-25 09:58:29 +0000 | [diff] [blame] | 117 | INTERCEPTOR(void*, pvalloc, uptr size) { |
Kostya Serebryany | a30c8f9 | 2012-12-13 09:34:23 +0000 | [diff] [blame] | 118 | GET_STACK_TRACE_MALLOC; |
Kostya Serebryany | 1e172b4 | 2011-11-30 01:07:02 +0000 | [diff] [blame] | 119 | return asan_pvalloc(size, &stack); |
| 120 | } |
Kostya Serebryany | d6567c5 | 2011-12-01 21:40:52 +0000 | [diff] [blame] | 121 | |
Kostya Serebryany | 73bad81 | 2012-12-20 11:54:21 +0000 | [diff] [blame] | 122 | INTERCEPTOR(void, malloc_stats, void) { |
Kostya Serebryany | 709a33e | 2012-12-26 12:20:35 +0000 | [diff] [blame] | 123 | __asan_print_accumulated_stats(); |
Kostya Serebryany | 73bad81 | 2012-12-20 11:54:21 +0000 | [diff] [blame] | 124 | } |
| 125 | |
Stephen Hines | 6d18623 | 2014-11-26 17:56:19 -0800 | [diff] [blame] | 126 | #if SANITIZER_ANDROID |
| 127 | // Format of __libc_malloc_dispatch has changed in Android L. |
| 128 | // While we are moving towards a solution that does not depend on bionic |
| 129 | // internals, here is something to support both K* and L releases. |
| 130 | struct MallocDebugK { |
| 131 | void *(*malloc)(uptr bytes); |
| 132 | void (*free)(void *mem); |
| 133 | void *(*calloc)(uptr n_elements, uptr elem_size); |
| 134 | void *(*realloc)(void *oldMem, uptr bytes); |
| 135 | void *(*memalign)(uptr alignment, uptr bytes); |
| 136 | uptr (*malloc_usable_size)(void *mem); |
| 137 | }; |
| 138 | |
| 139 | struct MallocDebugL { |
| 140 | void *(*calloc)(uptr n_elements, uptr elem_size); |
| 141 | void (*free)(void *mem); |
| 142 | fake_mallinfo (*mallinfo)(void); |
| 143 | void *(*malloc)(uptr bytes); |
| 144 | uptr (*malloc_usable_size)(void *mem); |
| 145 | void *(*memalign)(uptr alignment, uptr bytes); |
| 146 | int (*posix_memalign)(void **memptr, uptr alignment, uptr size); |
| 147 | void* (*pvalloc)(uptr size); |
| 148 | void *(*realloc)(void *oldMem, uptr bytes); |
| 149 | void* (*valloc)(uptr size); |
| 150 | }; |
| 151 | |
| 152 | ALIGNED(32) const MallocDebugK asan_malloc_dispatch_k = { |
| 153 | WRAP(malloc), WRAP(free), WRAP(calloc), |
| 154 | WRAP(realloc), WRAP(memalign), WRAP(malloc_usable_size)}; |
| 155 | |
| 156 | ALIGNED(32) const MallocDebugL asan_malloc_dispatch_l = { |
| 157 | WRAP(calloc), WRAP(free), WRAP(mallinfo), |
| 158 | WRAP(malloc), WRAP(malloc_usable_size), WRAP(memalign), |
| 159 | WRAP(posix_memalign), WRAP(pvalloc), WRAP(realloc), |
| 160 | WRAP(valloc)}; |
| 161 | |
| 162 | namespace __asan { |
| 163 | void ReplaceSystemMalloc() { |
| 164 | void **__libc_malloc_dispatch_p = |
| 165 | (void **)AsanDlSymNext("__libc_malloc_dispatch"); |
| 166 | if (__libc_malloc_dispatch_p) { |
| 167 | // Decide on K vs L dispatch format by the presence of |
| 168 | // __libc_malloc_default_dispatch export in libc. |
| 169 | void *default_dispatch_p = AsanDlSymNext("__libc_malloc_default_dispatch"); |
| 170 | if (default_dispatch_p) |
| 171 | *__libc_malloc_dispatch_p = (void *)&asan_malloc_dispatch_k; |
| 172 | else |
| 173 | *__libc_malloc_dispatch_p = (void *)&asan_malloc_dispatch_l; |
| 174 | } |
| 175 | } |
| 176 | } // namespace __asan |
| 177 | |
| 178 | #else // SANITIZER_ANDROID |
| 179 | |
| 180 | namespace __asan { |
| 181 | void ReplaceSystemMalloc() { |
| 182 | } |
| 183 | } // namespace __asan |
| 184 | #endif // SANITIZER_ANDROID |
| 185 | |
Stephen Hines | 2d1fdb2 | 2014-05-28 23:58:16 -0700 | [diff] [blame] | 186 | #endif // SANITIZER_FREEBSD || SANITIZER_LINUX |