/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "backtrace-map" #include #include #include #include #include #include #include #include #include #include #include "thread_utils.h" using android::base::StringPrintf; std::string backtrace_map_t::Name() const { if (!name.empty()) return name; if (start == 0 && end == 0) return ""; return StringPrintf("", start); } BacktraceMap::BacktraceMap(pid_t pid) : pid_(pid) { if (pid_ < 0) { pid_ = getpid(); } } BacktraceMap::~BacktraceMap() { } void BacktraceMap::FillIn(uint64_t addr, backtrace_map_t* map) { ScopedBacktraceMapIteratorLock lock(this); for (auto it = begin(); it != end(); ++it) { const backtrace_map_t* entry = *it; if (addr >= entry->start && addr < entry->end) { *map = *entry; return; } } *map = {}; } bool BacktraceMap::ParseLine(const char* line, backtrace_map_t* map) { uint64_t start; uint64_t end; char permissions[5]; int name_pos; #if defined(__APPLE__) // Mac OS vmmap(1) output: // __TEXT 0009f000-000a1000 [ 8K 8K] r-x/rwx SM=COW /Volumes/android/dalvik-dev/out/host/darwin-x86/bin/libcorkscrew_test\n // 012345678901234567890123456789012345678901234567890123456789 // 0 1 2 3 4 5 if (sscanf(line, "%*21c %" SCNx64 "-%" SCNx64 " [%*13c] %3c/%*3c SM=%*3c %n", &start, &end, permissions, &name_pos) != 3) { #else // Linux /proc//maps lines: // 6f000000-6f01e000 rwxp 00000000 00:0c 16389419 /system/lib/libcomposer.so\n // 012345678901234567890123456789012345678901234567890123456789 // 0 1 2 3 4 5 if (sscanf(line, "%" SCNx64 "-%" SCNx64 " %4s %*x %*x:%*x %*d %n", &start, &end, permissions, &name_pos) != 3) { #endif return false; } map->start = start; map->end = end; map->flags = PROT_NONE; if (permissions[0] == 'r') { map->flags |= PROT_READ; } if (permissions[1] == 'w') { map->flags |= PROT_WRITE; } if (permissions[2] == 'x') { map->flags |= PROT_EXEC; } map->name = line+name_pos; if (!map->name.empty() && map->name[map->name.length()-1] == '\n') { map->name.erase(map->name.length()-1); } ALOGV("Parsed map: start=%p, end=%p, flags=%x, name=%s", reinterpret_cast(map->start), reinterpret_cast(map->end), map->flags, map->name.c_str()); return true; } bool BacktraceMap::Build() { #if defined(__APPLE__) char cmd[sizeof(pid_t)*3 + sizeof("vmmap -w -resident -submap -allSplitLibs -interleaved ") + 1]; #else char path[sizeof(pid_t)*3 + sizeof("/proc//maps") + 1]; #endif char line[1024]; #if defined(__APPLE__) // cmd is guaranteed to always be big enough to hold this string. snprintf(cmd, sizeof(cmd), "vmmap -w -resident -submap -allSplitLibs -interleaved %d", pid_); FILE* fp = popen(cmd, "r"); #else // path is guaranteed to always be big enough to hold this string. snprintf(path, sizeof(path), "/proc/%d/maps", pid_); FILE* fp = fopen(path, "r"); #endif if (fp == nullptr) { return false; } while(fgets(line, sizeof(line), fp)) { backtrace_map_t map; if (ParseLine(line, &map)) { maps_.push_back(map); } } #if defined(__APPLE__) pclose(fp); #else fclose(fp); #endif return true; } #if defined(__APPLE__) // Corkscrew and libunwind don't compile on the mac, so create a generic // map object. BacktraceMap* BacktraceMap::Create(pid_t pid, bool /*uncached*/) { BacktraceMap* map = new BacktraceMap(pid); if (!map->Build()) { delete map; return nullptr; } return map; } #endif