1 // Copyright (c) 2012, Google Inc.
2 // All rights reserved.
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are
8 // * Redistributions of source code must retain the above copyright
9 // notice, this list of conditions and the following disclaimer.
10 // * Redistributions in binary form must reproduce the above
11 // copyright notice, this list of conditions and the following disclaimer
12 // in the documentation and/or other materials provided with the
14 // * Neither the name of Google Inc. nor the names of its
15 // contributors may be used to endorse or promote products derived from
16 // this software without specific prior written permission.
18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 // Author: Sanjay Ghemawat <opensource@google.com>
37 #if defined HAVE_STDINT_H
39 #elif defined HAVE_INTTYPES_H
42 #include <sys/types.h>
45 #include "base/dynamic_annotations.h"
46 #include "base/sysinfo.h" // for FillProcSelfMaps
48 #include "gperftools/heap-checker.h"
50 #include "gperftools/malloc_extension.h"
51 #include "gperftools/malloc_extension_c.h"
52 #include "maybe_threads.h"
55 // Note that malloc_extension can be used without tcmalloc if gperftools'
56 // heap-profiler is enabled without the tcmalloc memory allocator.
57 #include "thread_cache.h"
60 using STL_NAMESPACE::string
;
61 using STL_NAMESPACE::vector
;
63 static void DumpAddressMap(string
* result
) {
64 *result
+= "\nMAPPED_LIBRARIES:\n";
65 // We keep doubling until we get a fit
66 const size_t old_resultlen
= result
->size();
67 for (int amap_size
= 10240; amap_size
< 10000000; amap_size
*= 2) {
68 result
->resize(old_resultlen
+ amap_size
);
69 bool wrote_all
= false;
70 const int bytes_written
=
71 tcmalloc::FillProcSelfMaps(&((*result
)[old_resultlen
]), amap_size
,
73 if (wrote_all
) { // we fit!
74 (*result
)[old_resultlen
+ bytes_written
] = '\0';
75 result
->resize(old_resultlen
+ bytes_written
);
79 result
->reserve(old_resultlen
); // just don't print anything
82 // Note: this routine is meant to be called before threads are spawned.
83 void MallocExtension::Initialize() {
84 static bool initialize_called
= false;
86 if (initialize_called
) return;
87 initialize_called
= true;
90 // GNU libc++ versions 3.3 and 3.4 obey the environment variables
91 // GLIBCPP_FORCE_NEW and GLIBCXX_FORCE_NEW respectively. Setting
92 // one of these variables forces the STL default allocator to call
93 // new() or delete() for each allocation or deletion. Otherwise
94 // the STL allocator tries to avoid the high cost of doing
95 // allocations by pooling memory internally. However, tcmalloc
96 // does allocations really fast, especially for the types of small
97 // items one sees in STL, so it's better off just using us.
98 // TODO: control whether we do this via an environment variable?
99 setenv("GLIBCPP_FORCE_NEW", "1", false /* no overwrite*/);
100 setenv("GLIBCXX_FORCE_NEW", "1", false /* no overwrite*/);
102 // Now we need to make the setenv 'stick', which it may not do since
103 // the env is flakey before main() is called. But luckily stl only
104 // looks at this env var the first time it tries to do an alloc, and
105 // caches what it finds. So we just cause an stl alloc here.
106 string
dummy("I need to be allocated");
107 dummy
+= "!"; // so the definition of dummy isn't optimized out
108 #endif /* __GLIBC__ */
111 // SysAllocator implementation
112 SysAllocator::~SysAllocator() {}
114 // Default implementation -- does nothing
115 MallocExtension::~MallocExtension() { }
116 bool MallocExtension::VerifyAllMemory() { return true; }
117 bool MallocExtension::VerifyNewMemory(const void* p
) { return true; }
118 bool MallocExtension::VerifyArrayNewMemory(const void* p
) { return true; }
119 bool MallocExtension::VerifyMallocMemory(const void* p
) { return true; }
121 bool MallocExtension::GetNumericProperty(const char* property
, size_t* value
) {
125 bool MallocExtension::SetNumericProperty(const char* property
, size_t value
) {
129 void MallocExtension::GetStats(char* buffer
, int length
) {
134 bool MallocExtension::MallocMemoryStats(int* blocks
, size_t* total
,
135 int histogram
[kMallocHistogramSize
]) {
138 memset(histogram
, 0, sizeof(*histogram
) * kMallocHistogramSize
);
142 void** MallocExtension::ReadStackTraces(int* sample_period
) {
146 void** MallocExtension::ReadHeapGrowthStackTraces() {
150 void MallocExtension::MarkThreadIdle() {
151 // Default implementation does nothing
154 void MallocExtension::MarkThreadBusy() {
155 // Default implementation does nothing
158 SysAllocator
* MallocExtension::GetSystemAllocator() {
162 void MallocExtension::SetSystemAllocator(SysAllocator
*a
) {
163 // Default implementation does nothing
166 void MallocExtension::ReleaseToSystem(size_t num_bytes
) {
167 // Default implementation does nothing
170 void MallocExtension::ReleaseFreeMemory() {
171 ReleaseToSystem(static_cast<size_t>(-1)); // SIZE_T_MAX
174 void MallocExtension::SetMemoryReleaseRate(double rate
) {
175 // Default implementation does nothing
178 double MallocExtension::GetMemoryReleaseRate() {
182 size_t MallocExtension::GetEstimatedAllocatedSize(size_t size
) {
186 size_t MallocExtension::GetAllocatedSize(const void* p
) {
187 assert(GetOwnership(p
) != kNotOwned
);
191 MallocExtension::Ownership
MallocExtension::GetOwnership(const void* p
) {
192 return kUnknownOwnership
;
195 void MallocExtension::GetFreeListSizes(
196 vector
<MallocExtension::FreeListInfo
>* v
) {
200 // The current malloc extension object.
202 static pthread_once_t module_init
= PTHREAD_ONCE_INIT
;
203 static MallocExtension
* current_instance
= NULL
;
205 static void InitModule() {
206 current_instance
= new MallocExtension
;
207 #ifndef NO_HEAP_CHECK
208 HeapLeakChecker::IgnoreObject(current_instance
);
212 MallocExtension
* MallocExtension::instance() {
213 perftools_pthread_once(&module_init
, InitModule
);
214 return current_instance
;
217 void MallocExtension::Register(MallocExtension
* implementation
) {
218 perftools_pthread_once(&module_init
, InitModule
);
219 // When running under valgrind, our custom malloc is replaced with
220 // valgrind's one and malloc extensions will not work. (Note:
221 // callers should be responsible for checking that they are the
222 // malloc that is really being run, before calling Register. This
223 // is just here as an extra sanity check.)
224 if (!RunningOnValgrind()) {
225 current_instance
= implementation
;
229 unsigned int MallocExtension::GetBytesAllocatedOnCurrentThread() {
230 // This function is added in Chromium for profiling.
232 // Note that malloc_extension can be used without tcmalloc if gperftools'
233 // heap-profiler is enabled without the tcmalloc memory allocator.
234 return tcmalloc::ThreadCache::GetBytesAllocatedOnCurrentThread();
240 // -----------------------------------------------------------------------
241 // Heap sampling support
242 // -----------------------------------------------------------------------
247 uintptr_t Count(void** entry
) {
248 return reinterpret_cast<uintptr_t>(entry
[0]);
250 uintptr_t Size(void** entry
) {
251 return reinterpret_cast<uintptr_t>(entry
[1]);
253 uintptr_t Depth(void** entry
) {
254 return reinterpret_cast<uintptr_t>(entry
[2]);
256 void* PC(void** entry
, int i
) {
260 void PrintCountAndSize(MallocExtensionWriter
* writer
,
261 uintptr_t count
, uintptr_t size
) {
263 snprintf(buf
, sizeof(buf
),
264 "%6" PRIu64
": %8" PRIu64
" [%6" PRIu64
": %8" PRIu64
"] @",
265 static_cast<uint64
>(count
),
266 static_cast<uint64
>(size
),
267 static_cast<uint64
>(count
),
268 static_cast<uint64
>(size
));
269 writer
->append(buf
, strlen(buf
));
272 void PrintHeader(MallocExtensionWriter
* writer
,
273 const char* label
, void** entries
) {
274 // Compute the total count and total size
275 uintptr_t total_count
= 0;
276 uintptr_t total_size
= 0;
277 for (void** entry
= entries
; Count(entry
) != 0; entry
+= 3 + Depth(entry
)) {
278 total_count
+= Count(entry
);
279 total_size
+= Size(entry
);
282 const char* const kTitle
= "heap profile: ";
283 writer
->append(kTitle
, strlen(kTitle
));
284 PrintCountAndSize(writer
, total_count
, total_size
);
285 writer
->append(" ", 1);
286 writer
->append(label
, strlen(label
));
287 writer
->append("\n", 1);
290 void PrintStackEntry(MallocExtensionWriter
* writer
, void** entry
) {
291 PrintCountAndSize(writer
, Count(entry
), Size(entry
));
293 for (int i
= 0; i
< Depth(entry
); i
++) {
295 snprintf(buf
, sizeof(buf
), " %p", PC(entry
, i
));
296 writer
->append(buf
, strlen(buf
));
298 writer
->append("\n", 1);
303 void MallocExtension::GetHeapSample(MallocExtensionWriter
* writer
) {
304 int sample_period
= 0;
305 void** entries
= ReadStackTraces(&sample_period
);
306 if (entries
== NULL
) {
307 const char* const kErrorMsg
=
308 "This malloc implementation does not support sampling.\n"
309 "As of 2005/01/26, only tcmalloc supports sampling, and\n"
310 "you are probably running a binary that does not use\n"
312 writer
->append(kErrorMsg
, strlen(kErrorMsg
));
317 sprintf(label
, "heap_v2/%d", sample_period
);
318 PrintHeader(writer
, label
, entries
);
319 for (void** entry
= entries
; Count(entry
) != 0; entry
+= 3 + Depth(entry
)) {
320 PrintStackEntry(writer
, entry
);
324 DumpAddressMap(writer
);
327 void MallocExtension::GetHeapGrowthStacks(MallocExtensionWriter
* writer
) {
328 void** entries
= ReadHeapGrowthStackTraces();
329 if (entries
== NULL
) {
330 const char* const kErrorMsg
=
331 "This malloc implementation does not support "
332 "ReadHeapGrowthStackTraces().\n"
333 "As of 2005/09/27, only tcmalloc supports this, and you\n"
334 "are probably running a binary that does not use tcmalloc.\n";
335 writer
->append(kErrorMsg
, strlen(kErrorMsg
));
339 // Do not canonicalize the stack entries, so that we get a
340 // time-ordered list of stack traces, which may be useful if the
341 // client wants to focus on the latest stack traces.
342 PrintHeader(writer
, "growth", entries
);
343 for (void** entry
= entries
; Count(entry
) != 0; entry
+= 3 + Depth(entry
)) {
344 PrintStackEntry(writer
, entry
);
348 DumpAddressMap(writer
);
351 void MallocExtension::Ranges(void* arg
, RangeFunction func
) {
352 // No callbacks by default
355 // These are C shims that work on the current instance.
357 #define C_SHIM(fn, retval, paramlist, arglist) \
358 extern "C" PERFTOOLS_DLL_DECL retval MallocExtension_##fn paramlist { \
359 return MallocExtension::instance()->fn arglist; \
362 C_SHIM(VerifyAllMemory
, int, (void), ());
363 C_SHIM(VerifyNewMemory
, int, (const void* p
), (p
));
364 C_SHIM(VerifyArrayNewMemory
, int, (const void* p
), (p
));
365 C_SHIM(VerifyMallocMemory
, int, (const void* p
), (p
));
366 C_SHIM(MallocMemoryStats
, int,
367 (int* blocks
, size_t* total
, int histogram
[kMallocHistogramSize
]),
368 (blocks
, total
, histogram
));
370 C_SHIM(GetStats
, void,
371 (char* buffer
, int buffer_length
), (buffer
, buffer_length
));
372 C_SHIM(GetNumericProperty
, int,
373 (const char* property
, size_t* value
), (property
, value
));
374 C_SHIM(SetNumericProperty
, int,
375 (const char* property
, size_t value
), (property
, value
));
377 C_SHIM(MarkThreadIdle
, void, (void), ());
378 C_SHIM(MarkThreadBusy
, void, (void), ());
379 C_SHIM(ReleaseFreeMemory
, void, (void), ());
380 C_SHIM(ReleaseToSystem
, void, (size_t num_bytes
), (num_bytes
));
381 C_SHIM(GetEstimatedAllocatedSize
, size_t, (size_t size
), (size
));
382 C_SHIM(GetAllocatedSize
, size_t, (const void* p
), (p
));
384 // Can't use the shim here because of the need to translate the enums.
386 MallocExtension_Ownership
MallocExtension_GetOwnership(const void* p
) {
387 return static_cast<MallocExtension_Ownership
>(
388 MallocExtension::instance()->GetOwnership(p
));