xref: /aosp_15_r20/frameworks/av/media/utils/MemoryLeakTrackUtil.cpp (revision ec779b8e0859a360c3d303172224686826e6e0e1)
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