1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker * Copyright 2011, The Android Open Source Project
3*ec779b8eSAndroid Build Coastguard Worker *
4*ec779b8eSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*ec779b8eSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*ec779b8eSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*ec779b8eSAndroid Build Coastguard Worker *
8*ec779b8eSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*ec779b8eSAndroid Build Coastguard Worker *
10*ec779b8eSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*ec779b8eSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*ec779b8eSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ec779b8eSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*ec779b8eSAndroid Build Coastguard Worker * limitations under the License.
15*ec779b8eSAndroid Build Coastguard Worker */
16*ec779b8eSAndroid Build Coastguard Worker
17*ec779b8eSAndroid Build Coastguard Worker
18*ec779b8eSAndroid Build Coastguard Worker //#define LOG_NDEBUG 0
19*ec779b8eSAndroid Build Coastguard Worker #define LOG_TAG "MemoryLeackTrackUtil"
20*ec779b8eSAndroid Build Coastguard Worker #include <utils/Log.h>
21*ec779b8eSAndroid Build Coastguard Worker
22*ec779b8eSAndroid Build Coastguard Worker #include <mediautils/MemoryLeakTrackUtil.h>
23*ec779b8eSAndroid Build Coastguard Worker #include <sstream>
24*ec779b8eSAndroid Build Coastguard Worker
25*ec779b8eSAndroid Build Coastguard Worker #include <bionic/malloc.h>
26*ec779b8eSAndroid Build Coastguard Worker
27*ec779b8eSAndroid Build Coastguard Worker /*
28*ec779b8eSAndroid Build Coastguard Worker * The code here originally resided in MediaPlayerService.cpp
29*ec779b8eSAndroid Build Coastguard Worker */
30*ec779b8eSAndroid Build Coastguard Worker
31*ec779b8eSAndroid Build Coastguard Worker // Figure out the abi based on defined macros.
32*ec779b8eSAndroid Build Coastguard Worker #if defined(__arm__)
33*ec779b8eSAndroid Build Coastguard Worker #define ABI_STRING "arm"
34*ec779b8eSAndroid Build Coastguard Worker #elif defined(__aarch64__)
35*ec779b8eSAndroid Build Coastguard Worker #define ABI_STRING "arm64"
36*ec779b8eSAndroid Build Coastguard Worker #elif defined(__riscv)
37*ec779b8eSAndroid Build Coastguard Worker #define ABI_STRING "riscv64"
38*ec779b8eSAndroid Build Coastguard Worker #elif defined(__i386__)
39*ec779b8eSAndroid Build Coastguard Worker #define ABI_STRING "x86"
40*ec779b8eSAndroid Build Coastguard Worker #elif defined(__x86_64__)
41*ec779b8eSAndroid Build Coastguard Worker #define ABI_STRING "x86_64"
42*ec779b8eSAndroid Build Coastguard Worker #else
43*ec779b8eSAndroid Build Coastguard Worker #error "Unsupported ABI"
44*ec779b8eSAndroid Build Coastguard Worker #endif
45*ec779b8eSAndroid Build Coastguard Worker
46*ec779b8eSAndroid Build Coastguard Worker extern std::string backtrace_string(const uintptr_t* frames, size_t frame_count);
47*ec779b8eSAndroid Build Coastguard Worker
48*ec779b8eSAndroid Build Coastguard Worker namespace android {
49*ec779b8eSAndroid Build Coastguard Worker
dumpMemoryAddresses(size_t limit)50*ec779b8eSAndroid Build Coastguard Worker std::string dumpMemoryAddresses(size_t limit)
51*ec779b8eSAndroid Build Coastguard Worker {
52*ec779b8eSAndroid Build Coastguard Worker android_mallopt_leak_info_t leak_info;
53*ec779b8eSAndroid Build Coastguard Worker if (!android_mallopt(M_GET_MALLOC_LEAK_INFO, &leak_info, sizeof(leak_info))) {
54*ec779b8eSAndroid Build Coastguard Worker return "";
55*ec779b8eSAndroid Build Coastguard Worker }
56*ec779b8eSAndroid Build Coastguard Worker
57*ec779b8eSAndroid Build Coastguard Worker size_t count;
58*ec779b8eSAndroid Build Coastguard Worker if (leak_info.buffer == nullptr || leak_info.overall_size == 0 || leak_info.info_size == 0
59*ec779b8eSAndroid Build Coastguard Worker || (count = leak_info.overall_size / leak_info.info_size) == 0) {
60*ec779b8eSAndroid Build Coastguard Worker ALOGD("no malloc info, libc.debug.malloc.program property should be set");
61*ec779b8eSAndroid Build Coastguard Worker return "";
62*ec779b8eSAndroid Build Coastguard Worker }
63*ec779b8eSAndroid Build Coastguard Worker
64*ec779b8eSAndroid Build Coastguard Worker std::ostringstream oss;
65*ec779b8eSAndroid Build Coastguard Worker oss << leak_info.total_memory << " bytes in " << count << " allocations\n";
66*ec779b8eSAndroid Build Coastguard Worker oss << " ABI: '" ABI_STRING "'" << "\n\n";
67*ec779b8eSAndroid Build Coastguard Worker if (count > limit) count = limit;
68*ec779b8eSAndroid Build Coastguard Worker
69*ec779b8eSAndroid Build Coastguard Worker // The memory is sorted based on total size which is useful for finding
70*ec779b8eSAndroid Build Coastguard Worker // worst memory offenders. For diffs, sometimes it is preferable to sort
71*ec779b8eSAndroid Build Coastguard Worker // based on the backtrace.
72*ec779b8eSAndroid Build Coastguard Worker for (size_t i = 0; i < count; i++) {
73*ec779b8eSAndroid Build Coastguard Worker struct AllocEntry {
74*ec779b8eSAndroid Build Coastguard Worker size_t size; // bit 31 is set if this is zygote allocated memory
75*ec779b8eSAndroid Build Coastguard Worker size_t allocations;
76*ec779b8eSAndroid Build Coastguard Worker uintptr_t backtrace[];
77*ec779b8eSAndroid Build Coastguard Worker };
78*ec779b8eSAndroid Build Coastguard Worker
79*ec779b8eSAndroid Build Coastguard Worker const AllocEntry * const e = (AllocEntry *)(leak_info.buffer + i * leak_info.info_size);
80*ec779b8eSAndroid Build Coastguard Worker
81*ec779b8eSAndroid Build Coastguard Worker oss << (e->size * e->allocations)
82*ec779b8eSAndroid Build Coastguard Worker << " bytes ( " << e->size << " bytes * " << e->allocations << " allocations )\n";
83*ec779b8eSAndroid Build Coastguard Worker oss << backtrace_string(e->backtrace, leak_info.backtrace_size) << "\n";
84*ec779b8eSAndroid Build Coastguard Worker }
85*ec779b8eSAndroid Build Coastguard Worker oss << "\n";
86*ec779b8eSAndroid Build Coastguard Worker android_mallopt(M_FREE_MALLOC_LEAK_INFO, &leak_info, sizeof(leak_info));
87*ec779b8eSAndroid Build Coastguard Worker return oss.str();
88*ec779b8eSAndroid Build Coastguard Worker }
89*ec779b8eSAndroid Build Coastguard Worker
90*ec779b8eSAndroid Build Coastguard Worker } // namespace android
91