Upstream version 5.34.104.0
[platform/framework/web/crosswalk.git] / src / third_party / tcmalloc / chromium / src / malloc_extension.cc
1 // Copyright (c) 2012, Google Inc.
2 // All rights reserved.
3 // 
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are
6 // met:
7 // 
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
13 // distribution.
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.
17 // 
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.
29
30 // ---
31 // Author: Sanjay Ghemawat <opensource@google.com>
32
33 #include <config.h>
34 #include <assert.h>
35 #include <string.h>
36 #include <stdio.h>
37 #if defined HAVE_STDINT_H
38 #include <stdint.h>
39 #elif defined HAVE_INTTYPES_H
40 #include <inttypes.h>
41 #else
42 #include <sys/types.h>
43 #endif
44 #include <string>
45 #include "base/dynamic_annotations.h"
46 #include "base/sysinfo.h"    // for FillProcSelfMaps
47 #ifndef NO_HEAP_CHECK
48 #include "gperftools/heap-checker.h"
49 #endif
50 #include "gperftools/malloc_extension.h"
51 #include "gperftools/malloc_extension_c.h"
52 #include "maybe_threads.h"
53
54 #ifdef USE_TCMALLOC
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"
58 #endif
59
60 using STL_NAMESPACE::string;
61 using STL_NAMESPACE::vector;
62
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,
72                                    &wrote_all);
73     if (wrote_all) {   // we fit!
74       (*result)[old_resultlen + bytes_written] = '\0';
75       result->resize(old_resultlen + bytes_written);
76       return;
77     }
78   }
79   result->reserve(old_resultlen);   // just don't print anything
80 }
81
82 // Note: this routine is meant to be called before threads are spawned.
83 void MallocExtension::Initialize() {
84   static bool initialize_called = false;
85
86   if (initialize_called) return;
87   initialize_called = true;
88
89 #ifdef __GLIBC__
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*/);
101
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__ */
109 }
110
111 // SysAllocator implementation
112 SysAllocator::~SysAllocator() {}
113
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; }
120
121 bool MallocExtension::GetNumericProperty(const char* property, size_t* value) {
122   return false;
123 }
124
125 bool MallocExtension::SetNumericProperty(const char* property, size_t value) {
126   return false;
127 }
128
129 void MallocExtension::GetStats(char* buffer, int length) {
130   assert(length > 0);
131   buffer[0] = '\0';
132 }
133
134 bool MallocExtension::MallocMemoryStats(int* blocks, size_t* total,
135                                        int histogram[kMallocHistogramSize]) {
136   *blocks = 0;
137   *total = 0;
138   memset(histogram, 0, sizeof(*histogram) * kMallocHistogramSize);
139   return true;
140 }
141
142 void** MallocExtension::ReadStackTraces(int* sample_period) {
143   return NULL;
144 }
145
146 void** MallocExtension::ReadHeapGrowthStackTraces() {
147   return NULL;
148 }
149
150 void MallocExtension::MarkThreadIdle() {
151   // Default implementation does nothing
152 }
153
154 void MallocExtension::MarkThreadBusy() {
155   // Default implementation does nothing
156 }
157
158 SysAllocator* MallocExtension::GetSystemAllocator() {
159   return NULL;
160 }
161
162 void MallocExtension::SetSystemAllocator(SysAllocator *a) {
163   // Default implementation does nothing
164 }
165
166 void MallocExtension::ReleaseToSystem(size_t num_bytes) {
167   // Default implementation does nothing
168 }
169
170 void MallocExtension::ReleaseFreeMemory() {
171   ReleaseToSystem(static_cast<size_t>(-1));   // SIZE_T_MAX
172 }
173
174 void MallocExtension::SetMemoryReleaseRate(double rate) {
175   // Default implementation does nothing
176 }
177
178 double MallocExtension::GetMemoryReleaseRate() {
179   return -1.0;
180 }
181
182 size_t MallocExtension::GetEstimatedAllocatedSize(size_t size) {
183   return size;
184 }
185
186 size_t MallocExtension::GetAllocatedSize(const void* p) {
187   assert(GetOwnership(p) != kNotOwned);
188   return 0;
189 }
190
191 MallocExtension::Ownership MallocExtension::GetOwnership(const void* p) {
192   return kUnknownOwnership;
193 }
194
195 void MallocExtension::GetFreeListSizes(
196     vector<MallocExtension::FreeListInfo>* v) {
197   v->clear();
198 }
199
200 // The current malloc extension object.
201
202 static pthread_once_t module_init = PTHREAD_ONCE_INIT;
203 static MallocExtension* current_instance = NULL;
204
205 static void InitModule() {
206   current_instance = new MallocExtension;
207 #ifndef NO_HEAP_CHECK
208   HeapLeakChecker::IgnoreObject(current_instance);
209 #endif
210 }
211
212 MallocExtension* MallocExtension::instance() {
213   perftools_pthread_once(&module_init, InitModule);
214   return current_instance;
215 }
216
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;
226   }
227 }
228
229 unsigned int MallocExtension::GetBytesAllocatedOnCurrentThread() {
230   // This function is added in Chromium for profiling.
231 #ifdef USE_TCMALLOC
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();
235 #else
236   return 0;
237 #endif
238 }
239
240 // -----------------------------------------------------------------------
241 // Heap sampling support
242 // -----------------------------------------------------------------------
243
244 namespace {
245
246 // Accessors
247 uintptr_t Count(void** entry) {
248   return reinterpret_cast<uintptr_t>(entry[0]);
249 }
250 uintptr_t Size(void** entry) {
251   return reinterpret_cast<uintptr_t>(entry[1]);
252 }
253 uintptr_t Depth(void** entry) {
254   return reinterpret_cast<uintptr_t>(entry[2]);
255 }
256 void* PC(void** entry, int i) {
257   return entry[3+i];
258 }
259
260 void PrintCountAndSize(MallocExtensionWriter* writer,
261                        uintptr_t count, uintptr_t size) {
262   char buf[100];
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));
270 }
271
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);
280   }
281
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);
288 }
289
290 void PrintStackEntry(MallocExtensionWriter* writer, void** entry) {
291   PrintCountAndSize(writer, Count(entry), Size(entry));
292
293   for (int i = 0; i < Depth(entry); i++) {
294     char buf[32];
295     snprintf(buf, sizeof(buf), " %p", PC(entry, i));
296     writer->append(buf, strlen(buf));
297   }
298   writer->append("\n", 1);
299 }
300
301 }
302
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"
311         "tcmalloc.\n";
312     writer->append(kErrorMsg, strlen(kErrorMsg));
313     return;
314   }
315
316   char label[32];
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);
321   }
322   delete[] entries;
323
324   DumpAddressMap(writer);
325 }
326
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));
336     return;
337   }
338
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);
345   }
346   delete[] entries;
347
348   DumpAddressMap(writer);
349 }
350
351 void MallocExtension::Ranges(void* arg, RangeFunction func) {
352   // No callbacks by default
353 }
354
355 // These are C shims that work on the current instance.
356
357 #define C_SHIM(fn, retval, paramlist, arglist)          \
358   extern "C" PERFTOOLS_DLL_DECL retval MallocExtension_##fn paramlist {    \
359     return MallocExtension::instance()->fn arglist;     \
360   }
361
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));
369
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));
376
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));
383
384 // Can't use the shim here because of the need to translate the enums.
385 extern "C"
386 MallocExtension_Ownership MallocExtension_GetOwnership(const void* p) {
387   return static_cast<MallocExtension_Ownership>(
388       MallocExtension::instance()->GetOwnership(p));
389 }