123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286 |
- /*
- * Copyright (C) 2015 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.
- */
- #include "OfflineUnwinder.h"
- #include <sys/mman.h>
- #include <android-base/logging.h>
- #include <unwindstack/MachineArm.h>
- #include <unwindstack/MachineArm64.h>
- #include <unwindstack/MachineX86.h>
- #include <unwindstack/MachineX86_64.h>
- #include <unwindstack/Maps.h>
- #include <unwindstack/Regs.h>
- #include <unwindstack/RegsArm.h>
- #include <unwindstack/RegsArm64.h>
- #include <unwindstack/RegsX86.h>
- #include <unwindstack/RegsX86_64.h>
- #include <unwindstack/Unwinder.h>
- #include <unwindstack/UserArm.h>
- #include <unwindstack/UserArm64.h>
- #include <unwindstack/UserX86.h>
- #include <unwindstack/UserX86_64.h>
- #include "environment.h"
- #include "perf_regs.h"
- #include "read_apk.h"
- #include "thread_tree.h"
- static_assert(simpleperf::map_flags::PROT_JIT_SYMFILE_MAP ==
- unwindstack::MAPS_FLAGS_JIT_SYMFILE_MAP, "");
- namespace simpleperf {
- // Max frames seen so far is 463, in http://b/110923759.
- static constexpr size_t MAX_UNWINDING_FRAMES = 512;
- static unwindstack::Regs* GetBacktraceRegs(const RegSet& regs) {
- switch (regs.arch) {
- case ARCH_ARM: {
- unwindstack::arm_user_regs arm_user_regs;
- memset(&arm_user_regs, 0, sizeof(arm_user_regs));
- static_assert(
- static_cast<int>(unwindstack::ARM_REG_R0) == static_cast<int>(PERF_REG_ARM_R0), "");
- static_assert(
- static_cast<int>(unwindstack::ARM_REG_LAST) == static_cast<int>(PERF_REG_ARM_MAX), "");
- for (size_t i = unwindstack::ARM_REG_R0; i < unwindstack::ARM_REG_LAST; ++i) {
- arm_user_regs.regs[i] = static_cast<uint32_t>(regs.data[i]);
- }
- return unwindstack::RegsArm::Read(&arm_user_regs);
- }
- case ARCH_ARM64: {
- unwindstack::arm64_user_regs arm64_user_regs;
- memset(&arm64_user_regs, 0, sizeof(arm64_user_regs));
- static_assert(
- static_cast<int>(unwindstack::ARM64_REG_R0) == static_cast<int>(PERF_REG_ARM64_X0), "");
- static_assert(
- static_cast<int>(unwindstack::ARM64_REG_R30) == static_cast<int>(PERF_REG_ARM64_LR), "");
- memcpy(&arm64_user_regs.regs[unwindstack::ARM64_REG_R0], ®s.data[PERF_REG_ARM64_X0],
- sizeof(uint64_t) * (PERF_REG_ARM64_LR - PERF_REG_ARM64_X0 + 1));
- arm64_user_regs.sp = regs.data[PERF_REG_ARM64_SP];
- arm64_user_regs.pc = regs.data[PERF_REG_ARM64_PC];
- return unwindstack::RegsArm64::Read(&arm64_user_regs);
- }
- case ARCH_X86_32: {
- unwindstack::x86_user_regs x86_user_regs;
- memset(&x86_user_regs, 0, sizeof(x86_user_regs));
- x86_user_regs.eax = static_cast<uint32_t>(regs.data[PERF_REG_X86_AX]);
- x86_user_regs.ebx = static_cast<uint32_t>(regs.data[PERF_REG_X86_BX]);
- x86_user_regs.ecx = static_cast<uint32_t>(regs.data[PERF_REG_X86_CX]);
- x86_user_regs.edx = static_cast<uint32_t>(regs.data[PERF_REG_X86_DX]);
- x86_user_regs.ebp = static_cast<uint32_t>(regs.data[PERF_REG_X86_BP]);
- x86_user_regs.edi = static_cast<uint32_t>(regs.data[PERF_REG_X86_DI]);
- x86_user_regs.esi = static_cast<uint32_t>(regs.data[PERF_REG_X86_SI]);
- x86_user_regs.esp = static_cast<uint32_t>(regs.data[PERF_REG_X86_SP]);
- x86_user_regs.eip = static_cast<uint32_t>(regs.data[PERF_REG_X86_IP]);
- return unwindstack::RegsX86::Read(&x86_user_regs);
- }
- case ARCH_X86_64: {
- unwindstack::x86_64_user_regs x86_64_user_regs;
- memset(&x86_64_user_regs, 0, sizeof(x86_64_user_regs));
- x86_64_user_regs.rax = regs.data[PERF_REG_X86_AX];
- x86_64_user_regs.rbx = regs.data[PERF_REG_X86_BX];
- x86_64_user_regs.rcx = regs.data[PERF_REG_X86_CX];
- x86_64_user_regs.rdx = regs.data[PERF_REG_X86_DX];
- x86_64_user_regs.r8 = regs.data[PERF_REG_X86_R8];
- x86_64_user_regs.r9 = regs.data[PERF_REG_X86_R9];
- x86_64_user_regs.r10 = regs.data[PERF_REG_X86_R10];
- x86_64_user_regs.r11 = regs.data[PERF_REG_X86_R11];
- x86_64_user_regs.r12 = regs.data[PERF_REG_X86_R12];
- x86_64_user_regs.r13 = regs.data[PERF_REG_X86_R13];
- x86_64_user_regs.r14 = regs.data[PERF_REG_X86_R14];
- x86_64_user_regs.r15 = regs.data[PERF_REG_X86_R15];
- x86_64_user_regs.rdi = regs.data[PERF_REG_X86_DI];
- x86_64_user_regs.rsi = regs.data[PERF_REG_X86_SI];
- x86_64_user_regs.rbp = regs.data[PERF_REG_X86_BP];
- x86_64_user_regs.rsp = regs.data[PERF_REG_X86_SP];
- x86_64_user_regs.rip = regs.data[PERF_REG_X86_IP];
- return unwindstack::RegsX86_64::Read(&x86_64_user_regs);
- }
- default:
- return nullptr;
- }
- }
- static unwindstack::MapInfo* CreateMapInfo(const MapEntry* entry) {
- const char* name = entry->dso->GetDebugFilePath().c_str();
- uint64_t pgoff = entry->pgoff;
- if (entry->pgoff == 0) {
- auto tuple = SplitUrlInApk(entry->dso->GetDebugFilePath());
- if (std::get<0>(tuple)) {
- // The unwinder does not understand the ! format, so change back to
- // the previous format (apk, offset).
- EmbeddedElf* elf = ApkInspector::FindElfInApkByName(std::get<1>(tuple), std::get<2>(tuple));
- if (elf != nullptr) {
- name = elf->filepath().c_str();
- pgoff = elf->entry_offset();
- }
- }
- }
- return new unwindstack::MapInfo(nullptr, entry->start_addr, entry->get_end_addr(), pgoff,
- PROT_READ | PROT_EXEC | entry->flags, name);
- }
- void UnwindMaps::UpdateMaps(const MapSet& map_set) {
- if (version_ == map_set.version) {
- return;
- }
- version_ = map_set.version;
- size_t i = 0;
- size_t old_size = entries_.size();
- for (auto it = map_set.maps.begin(); it != map_set.maps.end();) {
- const MapEntry* entry = it->second;
- if (i < old_size && entry == entries_[i]) {
- i++;
- ++it;
- } else if (i == old_size || entry->start_addr <= entries_[i]->start_addr) {
- // Add an entry.
- entries_.push_back(entry);
- maps_.emplace_back(CreateMapInfo(entry));
- ++it;
- } else {
- // Remove an entry.
- entries_[i] = nullptr;
- maps_[i++] = nullptr;
- }
- }
- while (i < old_size) {
- entries_[i] = nullptr;
- maps_[i++] = nullptr;
- }
- std::sort(entries_.begin(), entries_.end(), [](const auto& e1, const auto& e2) {
- if (e1 == nullptr || e2 == nullptr) {
- return e1 != nullptr;
- }
- return e1->start_addr < e2->start_addr;
- });
- std::sort(maps_.begin(), maps_.end(),
- [](const auto& m1, const auto& m2) {
- if (m1 == nullptr || m2 == nullptr) {
- return m1 != nullptr;
- }
- return m1->start < m2->start;
- });
- entries_.resize(map_set.maps.size());
- maps_.resize(map_set.maps.size());
- }
- OfflineUnwinder::OfflineUnwinder(bool collect_stat) : collect_stat_(collect_stat) {
- unwindstack::Elf::SetCachingEnabled(true);
- }
- bool OfflineUnwinder::UnwindCallChain(const ThreadEntry& thread, const RegSet& regs,
- const char* stack, size_t stack_size,
- std::vector<uint64_t>* ips, std::vector<uint64_t>* sps) {
- uint64_t start_time;
- if (collect_stat_) {
- start_time = GetSystemClock();
- }
- is_callchain_broken_for_incomplete_jit_debug_info_ = false;
- ips->clear();
- sps->clear();
- std::vector<uint64_t> result;
- uint64_t sp_reg_value;
- if (!regs.GetSpRegValue(&sp_reg_value)) {
- LOG(ERROR) << "can't get sp reg value";
- return false;
- }
- uint64_t stack_addr = sp_reg_value;
- UnwindMaps& cached_map = cached_maps_[thread.pid];
- cached_map.UpdateMaps(*thread.maps);
- std::shared_ptr<unwindstack::MemoryOfflineBuffer> stack_memory(
- new unwindstack::MemoryOfflineBuffer(reinterpret_cast<const uint8_t*>(stack),
- stack_addr, stack_addr + stack_size));
- std::unique_ptr<unwindstack::Regs> unwind_regs(GetBacktraceRegs(regs));
- if (!unwind_regs) {
- return false;
- }
- unwindstack::Unwinder unwinder(MAX_UNWINDING_FRAMES, &cached_map, unwind_regs.get(),
- stack_memory);
- unwinder.SetResolveNames(false);
- unwinder.Unwind();
- size_t last_jit_method_frame = UINT_MAX;
- for (auto& frame : unwinder.frames()) {
- // Unwinding in arm architecture can return 0 pc address.
- // If frame.map.start == 0, this frame doesn't hit any map, it could be:
- // 1. In an executable map not backed by a file. Note that RecordCommand::ShouldOmitRecord()
- // may omit maps only exist memory.
- // 2. An incorrectly unwound frame. Like caused by invalid stack data, as in
- // SampleRecord::GetValidStackSize(). Or caused by incomplete JIT debug info.
- // We want to remove this frame and callchains following it in either case.
- if (frame.pc == 0 || frame.map_start == 0) {
- is_callchain_broken_for_incomplete_jit_debug_info_ = true;
- break;
- }
- if (frame.map_flags & unwindstack::MAPS_FLAGS_JIT_SYMFILE_MAP) {
- last_jit_method_frame = ips->size();
- }
- ips->push_back(frame.pc);
- sps->push_back(frame.sp);
- }
- // If the unwound frames stop near to a JITed method, it may be caused by incomplete JIT debug
- // info.
- if (last_jit_method_frame != UINT_MAX && last_jit_method_frame + 3 > ips->size()) {
- is_callchain_broken_for_incomplete_jit_debug_info_ = true;
- }
- uint64_t ip_reg_value;
- if (!regs.GetIpRegValue(&ip_reg_value)) {
- LOG(ERROR) << "can't get ip reg value";
- return false;
- }
- if (ips->empty()) {
- ips->push_back(ip_reg_value);
- sps->push_back(sp_reg_value);
- } else {
- // Check if the unwinder returns ip reg value as the first ip address in callstack.
- CHECK_EQ((*ips)[0], ip_reg_value);
- }
- if (collect_stat_) {
- unwinding_result_.used_time = GetSystemClock() - start_time;
- switch (unwinder.LastErrorCode()) {
- case unwindstack::ERROR_MAX_FRAMES_EXCEEDED:
- unwinding_result_.stop_reason = UnwindingResult::EXCEED_MAX_FRAMES_LIMIT;
- break;
- case unwindstack::ERROR_MEMORY_INVALID: {
- uint64_t addr = unwinder.LastErrorAddress();
- // Because we don't have precise stack range here, just guess an addr is in stack
- // if sp - 128K <= addr <= sp.
- if (addr <= stack_addr && addr >= stack_addr - 128 * 1024) {
- unwinding_result_.stop_reason = UnwindingResult::ACCESS_STACK_FAILED;
- } else {
- unwinding_result_.stop_reason = UnwindingResult::ACCESS_MEM_FAILED;
- }
- unwinding_result_.stop_info.addr = addr;
- break;
- }
- case unwindstack::ERROR_INVALID_MAP:
- unwinding_result_.stop_reason = UnwindingResult::MAP_MISSING;
- break;
- default:
- unwinding_result_.stop_reason = UnwindingResult::UNKNOWN_REASON;
- break;
- }
- unwinding_result_.stack_start = stack_addr;
- unwinding_result_.stack_end = stack_addr + stack_size;
- }
- return true;
- }
- } // namespace simpleperf
|