1 /* //device/libs/android_runtime/android_ddm_DdmHandleNativeHeap.cpp 2 ** 3 ** Copyright 2006, The Android Open Source Project 4 ** 5 ** Licensed under the Apache License, Version 2.0 (the "License"); 6 ** you may not use this file except in compliance with the License. 7 ** You may obtain a copy of the License at 8 ** 9 ** http://www.apache.org/licenses/LICENSE-2.0 10 ** 11 ** Unless required by applicable law or agreed to in writing, software 12 ** distributed under the License is distributed on an "AS IS" BASIS, 13 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 ** See the License for the specific language governing permissions and 15 ** limitations under the License. 16 */ 17 18 #undef LOG_TAG 19 #define LOG_TAG "DdmHandleNativeHeap" 20 21 #include <nativehelper/JNIHelp.h> 22 #include <jni.h> 23 #include "core_jni_helpers.h" 24 25 #include <android-base/logging.h> 26 #include <bionic/malloc.h> 27 28 #include <utils/Log.h> 29 #include <utils/String8.h> 30 31 #include <fcntl.h> 32 #include <errno.h> 33 #include <sys/types.h> 34 #include <sys/stat.h> 35 36 #define DDMS_HEADER_SIGNATURE 0x812345dd 37 #define DDMS_VERSION 2 38 39 struct Header { 40 #if defined(__LP64__) 41 uint32_t signature; 42 uint16_t version; 43 uint16_t pointerSize; 44 #endif 45 size_t mapSize; 46 size_t allocSize; 47 size_t allocInfoSize; 48 size_t totalMemory; 49 size_t backtraceSize; 50 }; 51 52 namespace android { 53 54 static void ReadFile(const char* path, String8& s) { 55 int fd = open(path, O_RDONLY | O_CLOEXEC); 56 if (fd != -1) { 57 char bytes[1024]; 58 ssize_t byteCount; 59 while ((byteCount = TEMP_FAILURE_RETRY(read(fd, bytes, sizeof(bytes)))) > 0) { 60 s.append(bytes, byteCount); 61 } 62 close(fd); 63 } 64 } 65 66 /* 67 * Retrieve the native heap information and the info from /proc/self/maps, 68 * copy them into a byte[] with a "struct Header" that holds data offsets, 69 * and return the array. 70 */ 71 static jbyteArray DdmHandleNativeHeap_getLeakInfo(JNIEnv* env, jobject) { 72 Header header; 73 memset(&header, 0, sizeof(header)); 74 75 String8 maps; 76 ReadFile("/proc/self/maps", maps); 77 header.mapSize = maps.size(); 78 79 android_mallopt_leak_info_t leak_info; 80 if (!android_mallopt(M_GET_MALLOC_LEAK_INFO, &leak_info, sizeof(leak_info))) { 81 PLOG(ERROR) << "*** Failed to get malloc leak info"; 82 return nullptr; 83 } 84 85 header.allocSize = leak_info.overall_size; 86 header.allocInfoSize = leak_info.info_size; 87 header.totalMemory = leak_info.total_memory; 88 header.backtraceSize = leak_info.backtrace_size; 89 90 ALOGD("*** mapSize: %zu allocSize: %zu allocInfoSize: %zu totalMemory: %zu", 91 header.mapSize, header.allocSize, header.allocInfoSize, header.totalMemory); 92 93 #if defined(__LP64__) 94 header.signature = DDMS_HEADER_SIGNATURE; 95 header.version = DDMS_VERSION; 96 header.pointerSize = sizeof(void*); 97 #endif 98 99 jbyteArray array = env->NewByteArray(sizeof(Header) + header.mapSize + header.allocSize); 100 if (array != NULL) { 101 env->SetByteArrayRegion(array, 0, 102 sizeof(header), reinterpret_cast<jbyte*>(&header)); 103 env->SetByteArrayRegion(array, sizeof(header), 104 maps.size(), reinterpret_cast<const jbyte*>(maps.string())); 105 env->SetByteArrayRegion(array, sizeof(header) + maps.size(), 106 header.allocSize, reinterpret_cast<jbyte*>(leak_info.buffer)); 107 } 108 109 android_mallopt(M_FREE_MALLOC_LEAK_INFO, &leak_info, sizeof(leak_info)); 110 return array; 111 } 112 113 static const JNINativeMethod method_table[] = { 114 { "getLeakInfo", "()[B", (void*) DdmHandleNativeHeap_getLeakInfo }, 115 }; 116 117 int register_android_ddm_DdmHandleNativeHeap(JNIEnv* env) { 118 return RegisterMethodsOrDie(env, "android/ddm/DdmHandleNativeHeap", method_table, 119 NELEM(method_table)); 120 } 121 122 }; 123