1*288bf522SAndroid Build Coastguard Worker /*
2*288bf522SAndroid Build Coastguard Worker * Copyright (C) 2015 The Android Open Source Project
3*288bf522SAndroid Build Coastguard Worker *
4*288bf522SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*288bf522SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*288bf522SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*288bf522SAndroid Build Coastguard Worker *
8*288bf522SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*288bf522SAndroid Build Coastguard Worker *
10*288bf522SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*288bf522SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*288bf522SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*288bf522SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*288bf522SAndroid Build Coastguard Worker * limitations under the License.
15*288bf522SAndroid Build Coastguard Worker */
16*288bf522SAndroid Build Coastguard Worker
17*288bf522SAndroid Build Coastguard Worker #include "OfflineUnwinder.h"
18*288bf522SAndroid Build Coastguard Worker
19*288bf522SAndroid Build Coastguard Worker #include <inttypes.h>
20*288bf522SAndroid Build Coastguard Worker #include <sys/mman.h>
21*288bf522SAndroid Build Coastguard Worker
22*288bf522SAndroid Build Coastguard Worker #include <unordered_map>
23*288bf522SAndroid Build Coastguard Worker
24*288bf522SAndroid Build Coastguard Worker #include <android-base/logging.h>
25*288bf522SAndroid Build Coastguard Worker #include <android-base/parseint.h>
26*288bf522SAndroid Build Coastguard Worker #include <unwindstack/MachineArm.h>
27*288bf522SAndroid Build Coastguard Worker #include <unwindstack/MachineArm64.h>
28*288bf522SAndroid Build Coastguard Worker #include <unwindstack/MachineX86.h>
29*288bf522SAndroid Build Coastguard Worker #include <unwindstack/MachineX86_64.h>
30*288bf522SAndroid Build Coastguard Worker #include <unwindstack/MachineRiscv64.h>
31*288bf522SAndroid Build Coastguard Worker #include <unwindstack/Maps.h>
32*288bf522SAndroid Build Coastguard Worker #include <unwindstack/RegsArm.h>
33*288bf522SAndroid Build Coastguard Worker #include <unwindstack/RegsArm64.h>
34*288bf522SAndroid Build Coastguard Worker #include <unwindstack/RegsX86.h>
35*288bf522SAndroid Build Coastguard Worker #include <unwindstack/RegsX86_64.h>
36*288bf522SAndroid Build Coastguard Worker #include <unwindstack/RegsRiscv64.h>
37*288bf522SAndroid Build Coastguard Worker #include <unwindstack/Unwinder.h>
38*288bf522SAndroid Build Coastguard Worker #include <unwindstack/UserArm.h>
39*288bf522SAndroid Build Coastguard Worker #include <unwindstack/UserArm64.h>
40*288bf522SAndroid Build Coastguard Worker #include <unwindstack/UserX86.h>
41*288bf522SAndroid Build Coastguard Worker #include <unwindstack/UserX86_64.h>
42*288bf522SAndroid Build Coastguard Worker #include <unwindstack/UserRiscv64.h>
43*288bf522SAndroid Build Coastguard Worker
44*288bf522SAndroid Build Coastguard Worker #include "JITDebugReader.h"
45*288bf522SAndroid Build Coastguard Worker #include "OfflineUnwinder_impl.h"
46*288bf522SAndroid Build Coastguard Worker #include "environment.h"
47*288bf522SAndroid Build Coastguard Worker #include "perf_regs.h"
48*288bf522SAndroid Build Coastguard Worker #include "read_apk.h"
49*288bf522SAndroid Build Coastguard Worker #include "thread_tree.h"
50*288bf522SAndroid Build Coastguard Worker
51*288bf522SAndroid Build Coastguard Worker namespace simpleperf {
52*288bf522SAndroid Build Coastguard Worker
53*288bf522SAndroid Build Coastguard Worker // unwindstack only builds on linux. So simpleperf redefines flags in unwindstack, to use them on
54*288bf522SAndroid Build Coastguard Worker // darwin/windows. Use static_assert to make sure they are on the same page.
55*288bf522SAndroid Build Coastguard Worker static_assert(map_flags::PROT_JIT_SYMFILE_MAP == unwindstack::MAPS_FLAGS_JIT_SYMFILE_MAP);
56*288bf522SAndroid Build Coastguard Worker
57*288bf522SAndroid Build Coastguard Worker #define CHECK_ERROR_CODE(error_code_name) \
58*288bf522SAndroid Build Coastguard Worker static_assert(UnwindStackErrorCode::error_code_name == \
59*288bf522SAndroid Build Coastguard Worker (UnwindStackErrorCode)unwindstack::ErrorCode::error_code_name)
60*288bf522SAndroid Build Coastguard Worker
61*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_NONE);
62*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_MEMORY_INVALID);
63*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_UNWIND_INFO);
64*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_UNSUPPORTED);
65*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_INVALID_MAP);
66*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_MAX_FRAMES_EXCEEDED);
67*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_REPEATED_FRAME);
68*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_INVALID_ELF);
69*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_THREAD_DOES_NOT_EXIST);
70*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_THREAD_TIMEOUT);
71*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_SYSTEM_CALL);
72*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_BAD_ARCH);
73*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_MAPS_PARSE);
74*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_INVALID_PARAMETER);
75*288bf522SAndroid Build Coastguard Worker CHECK_ERROR_CODE(ERROR_MAX);
76*288bf522SAndroid Build Coastguard Worker
77*288bf522SAndroid Build Coastguard Worker // Max frames seen so far is 463, in http://b/110923759.
78*288bf522SAndroid Build Coastguard Worker static constexpr size_t MAX_UNWINDING_FRAMES = 512;
79*288bf522SAndroid Build Coastguard Worker
GetBacktraceRegs(const RegSet & regs)80*288bf522SAndroid Build Coastguard Worker unwindstack::Regs* OfflineUnwinderImpl::GetBacktraceRegs(const RegSet& regs) {
81*288bf522SAndroid Build Coastguard Worker switch (regs.arch) {
82*288bf522SAndroid Build Coastguard Worker case ARCH_ARM: {
83*288bf522SAndroid Build Coastguard Worker unwindstack::arm_user_regs arm_user_regs;
84*288bf522SAndroid Build Coastguard Worker memset(&arm_user_regs, 0, sizeof(arm_user_regs));
85*288bf522SAndroid Build Coastguard Worker static_assert(static_cast<int>(unwindstack::ARM_REG_R0) == static_cast<int>(PERF_REG_ARM_R0),
86*288bf522SAndroid Build Coastguard Worker "");
87*288bf522SAndroid Build Coastguard Worker static_assert(
88*288bf522SAndroid Build Coastguard Worker static_cast<int>(unwindstack::ARM_REG_LAST) == static_cast<int>(PERF_REG_ARM_MAX), "");
89*288bf522SAndroid Build Coastguard Worker for (size_t i = unwindstack::ARM_REG_R0; i < unwindstack::ARM_REG_LAST; ++i) {
90*288bf522SAndroid Build Coastguard Worker arm_user_regs.regs[i] = static_cast<uint32_t>(regs.data[i]);
91*288bf522SAndroid Build Coastguard Worker }
92*288bf522SAndroid Build Coastguard Worker return unwindstack::RegsArm::Read(&arm_user_regs);
93*288bf522SAndroid Build Coastguard Worker }
94*288bf522SAndroid Build Coastguard Worker case ARCH_ARM64: {
95*288bf522SAndroid Build Coastguard Worker unwindstack::arm64_user_regs arm64_user_regs;
96*288bf522SAndroid Build Coastguard Worker memset(&arm64_user_regs, 0, sizeof(arm64_user_regs));
97*288bf522SAndroid Build Coastguard Worker static_assert(
98*288bf522SAndroid Build Coastguard Worker static_cast<int>(unwindstack::ARM64_REG_R0) == static_cast<int>(PERF_REG_ARM64_X0), "");
99*288bf522SAndroid Build Coastguard Worker static_assert(
100*288bf522SAndroid Build Coastguard Worker static_cast<int>(unwindstack::ARM64_REG_R30) == static_cast<int>(PERF_REG_ARM64_LR), "");
101*288bf522SAndroid Build Coastguard Worker memcpy(&arm64_user_regs.regs[unwindstack::ARM64_REG_R0], ®s.data[PERF_REG_ARM64_X0],
102*288bf522SAndroid Build Coastguard Worker sizeof(uint64_t) * (PERF_REG_ARM64_LR - PERF_REG_ARM64_X0 + 1));
103*288bf522SAndroid Build Coastguard Worker arm64_user_regs.sp = regs.data[PERF_REG_ARM64_SP];
104*288bf522SAndroid Build Coastguard Worker arm64_user_regs.pc = regs.data[PERF_REG_ARM64_PC];
105*288bf522SAndroid Build Coastguard Worker auto regs =
106*288bf522SAndroid Build Coastguard Worker static_cast<unwindstack::RegsArm64*>(unwindstack::RegsArm64::Read(&arm64_user_regs));
107*288bf522SAndroid Build Coastguard Worker regs->SetPACMask(arm64_pac_mask_);
108*288bf522SAndroid Build Coastguard Worker return regs;
109*288bf522SAndroid Build Coastguard Worker }
110*288bf522SAndroid Build Coastguard Worker case ARCH_X86_32: {
111*288bf522SAndroid Build Coastguard Worker unwindstack::x86_user_regs x86_user_regs;
112*288bf522SAndroid Build Coastguard Worker memset(&x86_user_regs, 0, sizeof(x86_user_regs));
113*288bf522SAndroid Build Coastguard Worker x86_user_regs.eax = static_cast<uint32_t>(regs.data[PERF_REG_X86_AX]);
114*288bf522SAndroid Build Coastguard Worker x86_user_regs.ebx = static_cast<uint32_t>(regs.data[PERF_REG_X86_BX]);
115*288bf522SAndroid Build Coastguard Worker x86_user_regs.ecx = static_cast<uint32_t>(regs.data[PERF_REG_X86_CX]);
116*288bf522SAndroid Build Coastguard Worker x86_user_regs.edx = static_cast<uint32_t>(regs.data[PERF_REG_X86_DX]);
117*288bf522SAndroid Build Coastguard Worker x86_user_regs.ebp = static_cast<uint32_t>(regs.data[PERF_REG_X86_BP]);
118*288bf522SAndroid Build Coastguard Worker x86_user_regs.edi = static_cast<uint32_t>(regs.data[PERF_REG_X86_DI]);
119*288bf522SAndroid Build Coastguard Worker x86_user_regs.esi = static_cast<uint32_t>(regs.data[PERF_REG_X86_SI]);
120*288bf522SAndroid Build Coastguard Worker x86_user_regs.esp = static_cast<uint32_t>(regs.data[PERF_REG_X86_SP]);
121*288bf522SAndroid Build Coastguard Worker x86_user_regs.eip = static_cast<uint32_t>(regs.data[PERF_REG_X86_IP]);
122*288bf522SAndroid Build Coastguard Worker return unwindstack::RegsX86::Read(&x86_user_regs);
123*288bf522SAndroid Build Coastguard Worker }
124*288bf522SAndroid Build Coastguard Worker case ARCH_X86_64: {
125*288bf522SAndroid Build Coastguard Worker unwindstack::x86_64_user_regs x86_64_user_regs;
126*288bf522SAndroid Build Coastguard Worker memset(&x86_64_user_regs, 0, sizeof(x86_64_user_regs));
127*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rax = regs.data[PERF_REG_X86_AX];
128*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rbx = regs.data[PERF_REG_X86_BX];
129*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rcx = regs.data[PERF_REG_X86_CX];
130*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rdx = regs.data[PERF_REG_X86_DX];
131*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r8 = regs.data[PERF_REG_X86_R8];
132*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r9 = regs.data[PERF_REG_X86_R9];
133*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r10 = regs.data[PERF_REG_X86_R10];
134*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r11 = regs.data[PERF_REG_X86_R11];
135*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r12 = regs.data[PERF_REG_X86_R12];
136*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r13 = regs.data[PERF_REG_X86_R13];
137*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r14 = regs.data[PERF_REG_X86_R14];
138*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.r15 = regs.data[PERF_REG_X86_R15];
139*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rdi = regs.data[PERF_REG_X86_DI];
140*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rsi = regs.data[PERF_REG_X86_SI];
141*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rbp = regs.data[PERF_REG_X86_BP];
142*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rsp = regs.data[PERF_REG_X86_SP];
143*288bf522SAndroid Build Coastguard Worker x86_64_user_regs.rip = regs.data[PERF_REG_X86_IP];
144*288bf522SAndroid Build Coastguard Worker return unwindstack::RegsX86_64::Read(&x86_64_user_regs);
145*288bf522SAndroid Build Coastguard Worker }
146*288bf522SAndroid Build Coastguard Worker case ARCH_RISCV64: {
147*288bf522SAndroid Build Coastguard Worker unwindstack::riscv64_user_regs riscv64_user_regs;
148*288bf522SAndroid Build Coastguard Worker memset(&riscv64_user_regs, 0, sizeof(riscv64_user_regs));
149*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_PC] = regs.data[PERF_REG_RISCV_PC];
150*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_RA] = regs.data[PERF_REG_RISCV_RA];
151*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_SP] = regs.data[PERF_REG_RISCV_SP];
152*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_GP] = regs.data[PERF_REG_RISCV_GP];
153*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_TP] = regs.data[PERF_REG_RISCV_TP];
154*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T0] = regs.data[PERF_REG_RISCV_T0];
155*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T1] = regs.data[PERF_REG_RISCV_T1];
156*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T2] = regs.data[PERF_REG_RISCV_T2];
157*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S0] = regs.data[PERF_REG_RISCV_S0];
158*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S1] = regs.data[PERF_REG_RISCV_S1];
159*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A0] = regs.data[PERF_REG_RISCV_A0];
160*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A1] = regs.data[PERF_REG_RISCV_A1];
161*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A2] = regs.data[PERF_REG_RISCV_A2];
162*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A3] = regs.data[PERF_REG_RISCV_A3];
163*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A4] = regs.data[PERF_REG_RISCV_A4];
164*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A5] = regs.data[PERF_REG_RISCV_A5];
165*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A6] = regs.data[PERF_REG_RISCV_A6];
166*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_A7] = regs.data[PERF_REG_RISCV_A7];
167*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S2] = regs.data[PERF_REG_RISCV_S2];
168*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S3] = regs.data[PERF_REG_RISCV_S3];
169*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S4] = regs.data[PERF_REG_RISCV_S4];
170*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S5] = regs.data[PERF_REG_RISCV_S5];
171*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S6] = regs.data[PERF_REG_RISCV_S6];
172*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S7] = regs.data[PERF_REG_RISCV_S7];
173*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S8] = regs.data[PERF_REG_RISCV_S8];
174*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S9] = regs.data[PERF_REG_RISCV_S9];
175*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S10] = regs.data[PERF_REG_RISCV_S10];
176*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_S11] = regs.data[PERF_REG_RISCV_S11];
177*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T3] = regs.data[PERF_REG_RISCV_T3];
178*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T4] = regs.data[PERF_REG_RISCV_T4];
179*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T5] = regs.data[PERF_REG_RISCV_T5];
180*288bf522SAndroid Build Coastguard Worker riscv64_user_regs.regs[PERF_REG_RISCV_T6] = regs.data[PERF_REG_RISCV_T6];
181*288bf522SAndroid Build Coastguard Worker return unwindstack::RegsRiscv64::Read(&riscv64_user_regs);
182*288bf522SAndroid Build Coastguard Worker }
183*288bf522SAndroid Build Coastguard Worker default:
184*288bf522SAndroid Build Coastguard Worker return nullptr;
185*288bf522SAndroid Build Coastguard Worker }
186*288bf522SAndroid Build Coastguard Worker }
187*288bf522SAndroid Build Coastguard Worker
CreateMapInfo(const MapEntry * entry)188*288bf522SAndroid Build Coastguard Worker static std::shared_ptr<unwindstack::MapInfo> CreateMapInfo(const MapEntry* entry) {
189*288bf522SAndroid Build Coastguard Worker std::string name_holder;
190*288bf522SAndroid Build Coastguard Worker const char* name = entry->dso->GetDebugFilePath().data();
191*288bf522SAndroid Build Coastguard Worker uint64_t pgoff = entry->pgoff;
192*288bf522SAndroid Build Coastguard Worker auto tuple = SplitUrlInApk(entry->dso->GetDebugFilePath());
193*288bf522SAndroid Build Coastguard Worker if (std::get<0>(tuple)) {
194*288bf522SAndroid Build Coastguard Worker // The unwinder does not understand the ! format, so change back to
195*288bf522SAndroid Build Coastguard Worker // the previous format (apk, offset).
196*288bf522SAndroid Build Coastguard Worker EmbeddedElf* elf = ApkInspector::FindElfInApkByName(std::get<1>(tuple), std::get<2>(tuple));
197*288bf522SAndroid Build Coastguard Worker if (elf != nullptr) {
198*288bf522SAndroid Build Coastguard Worker name = elf->filepath().data();
199*288bf522SAndroid Build Coastguard Worker pgoff += elf->entry_offset();
200*288bf522SAndroid Build Coastguard Worker }
201*288bf522SAndroid Build Coastguard Worker } else if (entry->flags & map_flags::PROT_JIT_SYMFILE_MAP) {
202*288bf522SAndroid Build Coastguard Worker // Remove location_in_file suffix, which isn't recognized by libunwindstack.
203*288bf522SAndroid Build Coastguard Worker const std::string& path = entry->dso->GetDebugFilePath();
204*288bf522SAndroid Build Coastguard Worker if (JITDebugReader::IsPathInJITSymFile(path)) {
205*288bf522SAndroid Build Coastguard Worker size_t colon_pos = path.rfind(':');
206*288bf522SAndroid Build Coastguard Worker CHECK_NE(colon_pos, std::string::npos);
207*288bf522SAndroid Build Coastguard Worker name_holder = path.substr(0, colon_pos);
208*288bf522SAndroid Build Coastguard Worker name = name_holder.data();
209*288bf522SAndroid Build Coastguard Worker }
210*288bf522SAndroid Build Coastguard Worker }
211*288bf522SAndroid Build Coastguard Worker return unwindstack::MapInfo::Create(entry->start_addr, entry->get_end_addr(), pgoff,
212*288bf522SAndroid Build Coastguard Worker PROT_READ | entry->flags, name);
213*288bf522SAndroid Build Coastguard Worker }
214*288bf522SAndroid Build Coastguard Worker
UpdateMaps(const MapSet & map_set)215*288bf522SAndroid Build Coastguard Worker void UnwindMaps::UpdateMaps(const MapSet& map_set) {
216*288bf522SAndroid Build Coastguard Worker if (version_ == map_set.version) {
217*288bf522SAndroid Build Coastguard Worker return;
218*288bf522SAndroid Build Coastguard Worker }
219*288bf522SAndroid Build Coastguard Worker version_ = map_set.version;
220*288bf522SAndroid Build Coastguard Worker size_t i = 0;
221*288bf522SAndroid Build Coastguard Worker size_t old_size = entries_.size();
222*288bf522SAndroid Build Coastguard Worker bool has_removed_entry = false;
223*288bf522SAndroid Build Coastguard Worker for (auto it = map_set.maps.begin(); it != map_set.maps.end();) {
224*288bf522SAndroid Build Coastguard Worker const MapEntry* entry = it->second;
225*288bf522SAndroid Build Coastguard Worker if (i < old_size && entry == entries_[i]) {
226*288bf522SAndroid Build Coastguard Worker i++;
227*288bf522SAndroid Build Coastguard Worker ++it;
228*288bf522SAndroid Build Coastguard Worker } else if (i == old_size || entry->start_addr <= entries_[i]->start_addr) {
229*288bf522SAndroid Build Coastguard Worker // Add an entry.
230*288bf522SAndroid Build Coastguard Worker entries_.push_back(entry);
231*288bf522SAndroid Build Coastguard Worker maps_.emplace_back(CreateMapInfo(entry));
232*288bf522SAndroid Build Coastguard Worker ++it;
233*288bf522SAndroid Build Coastguard Worker } else {
234*288bf522SAndroid Build Coastguard Worker // Remove an entry.
235*288bf522SAndroid Build Coastguard Worker has_removed_entry = true;
236*288bf522SAndroid Build Coastguard Worker entries_[i] = nullptr;
237*288bf522SAndroid Build Coastguard Worker maps_[i++] = nullptr;
238*288bf522SAndroid Build Coastguard Worker }
239*288bf522SAndroid Build Coastguard Worker }
240*288bf522SAndroid Build Coastguard Worker while (i < old_size) {
241*288bf522SAndroid Build Coastguard Worker has_removed_entry = true;
242*288bf522SAndroid Build Coastguard Worker entries_[i] = nullptr;
243*288bf522SAndroid Build Coastguard Worker maps_[i++] = nullptr;
244*288bf522SAndroid Build Coastguard Worker }
245*288bf522SAndroid Build Coastguard Worker
246*288bf522SAndroid Build Coastguard Worker if (has_removed_entry) {
247*288bf522SAndroid Build Coastguard Worker entries_.resize(std::remove(entries_.begin(), entries_.end(), nullptr) - entries_.begin());
248*288bf522SAndroid Build Coastguard Worker maps_.resize(std::remove(maps_.begin(), maps_.end(), std::shared_ptr<unwindstack::MapInfo>()) -
249*288bf522SAndroid Build Coastguard Worker maps_.begin());
250*288bf522SAndroid Build Coastguard Worker }
251*288bf522SAndroid Build Coastguard Worker
252*288bf522SAndroid Build Coastguard Worker std::sort(entries_.begin(), entries_.end(),
253*288bf522SAndroid Build Coastguard Worker [](const auto& e1, const auto& e2) { return e1->start_addr < e2->start_addr; });
254*288bf522SAndroid Build Coastguard Worker // Use Sort() to sort maps_ and create prev_real_map links.
255*288bf522SAndroid Build Coastguard Worker // prev_real_map is needed by libunwindstack to find the start of an embedded lib in an apk.
256*288bf522SAndroid Build Coastguard Worker // See http://b/120981155.
257*288bf522SAndroid Build Coastguard Worker Sort();
258*288bf522SAndroid Build Coastguard Worker }
259*288bf522SAndroid Build Coastguard Worker
CollectMetaInfo(std::unordered_map<std::string,std::string> * info_map)260*288bf522SAndroid Build Coastguard Worker void OfflineUnwinder::CollectMetaInfo(std::unordered_map<std::string, std::string>* info_map
261*288bf522SAndroid Build Coastguard Worker __attribute__((unused))) {
262*288bf522SAndroid Build Coastguard Worker #if defined(__aarch64__)
263*288bf522SAndroid Build Coastguard Worker // Find pac_mask for ARMv8.3-A Pointer Authentication by below steps:
264*288bf522SAndroid Build Coastguard Worker // 1. Create a 64 bit value with every bit set, but clear bit 55. Because linux user space uses
265*288bf522SAndroid Build Coastguard Worker // TTBR0.
266*288bf522SAndroid Build Coastguard Worker // 2. Use XPACLRI to clear auth code bits.
267*288bf522SAndroid Build Coastguard Worker // 3. Flip every bit to get pac_mask, excluding bit 55.
268*288bf522SAndroid Build Coastguard Worker // We can also use ptrace(PTRACE_GETREGSET, pid, NT_ARM_PAC_MASK). But it needs a tracee.
269*288bf522SAndroid Build Coastguard Worker register uint64_t x30 __asm("x30") = ~(1ULL << 55);
270*288bf522SAndroid Build Coastguard Worker // This is XPACLRI on ARMv8.3-A, and nop on prev ARMv8.3-A.
271*288bf522SAndroid Build Coastguard Worker asm("hint 0x7" : "+r"(x30));
272*288bf522SAndroid Build Coastguard Worker uint64_t pac_mask = ~x30 & ~(1ULL << 55);
273*288bf522SAndroid Build Coastguard Worker if (pac_mask != 0) {
274*288bf522SAndroid Build Coastguard Worker (*info_map)[META_KEY_ARM64_PAC_MASK] = android::base::StringPrintf("0x%" PRIx64, pac_mask);
275*288bf522SAndroid Build Coastguard Worker }
276*288bf522SAndroid Build Coastguard Worker #endif
277*288bf522SAndroid Build Coastguard Worker }
278*288bf522SAndroid Build Coastguard Worker
LoadMetaInfo(const std::unordered_map<std::string,std::string> & info_map)279*288bf522SAndroid Build Coastguard Worker void OfflineUnwinderImpl::LoadMetaInfo(
280*288bf522SAndroid Build Coastguard Worker const std::unordered_map<std::string, std::string>& info_map) {
281*288bf522SAndroid Build Coastguard Worker if (auto it = info_map.find(META_KEY_ARM64_PAC_MASK); it != info_map.end()) {
282*288bf522SAndroid Build Coastguard Worker CHECK(android::base::ParseUint(it->second, &arm64_pac_mask_));
283*288bf522SAndroid Build Coastguard Worker }
284*288bf522SAndroid Build Coastguard Worker }
285*288bf522SAndroid Build Coastguard Worker
UnwindCallChain(const ThreadEntry & thread,const RegSet & regs,const char * stack,size_t stack_size,std::vector<uint64_t> * ips,std::vector<uint64_t> * sps)286*288bf522SAndroid Build Coastguard Worker bool OfflineUnwinderImpl::UnwindCallChain(const ThreadEntry& thread, const RegSet& regs,
287*288bf522SAndroid Build Coastguard Worker const char* stack, size_t stack_size,
288*288bf522SAndroid Build Coastguard Worker std::vector<uint64_t>* ips, std::vector<uint64_t>* sps) {
289*288bf522SAndroid Build Coastguard Worker uint64_t start_time;
290*288bf522SAndroid Build Coastguard Worker if (collect_stat_) {
291*288bf522SAndroid Build Coastguard Worker start_time = GetSystemClock();
292*288bf522SAndroid Build Coastguard Worker }
293*288bf522SAndroid Build Coastguard Worker is_callchain_broken_for_incomplete_jit_debug_info_ = false;
294*288bf522SAndroid Build Coastguard Worker ips->clear();
295*288bf522SAndroid Build Coastguard Worker sps->clear();
296*288bf522SAndroid Build Coastguard Worker std::vector<uint64_t> result;
297*288bf522SAndroid Build Coastguard Worker uint64_t sp_reg_value;
298*288bf522SAndroid Build Coastguard Worker if (!regs.GetSpRegValue(&sp_reg_value)) {
299*288bf522SAndroid Build Coastguard Worker LOG(ERROR) << "can't get sp reg value";
300*288bf522SAndroid Build Coastguard Worker return false;
301*288bf522SAndroid Build Coastguard Worker }
302*288bf522SAndroid Build Coastguard Worker uint64_t stack_addr = sp_reg_value;
303*288bf522SAndroid Build Coastguard Worker
304*288bf522SAndroid Build Coastguard Worker UnwindMaps& cached_map = cached_maps_[thread.pid];
305*288bf522SAndroid Build Coastguard Worker cached_map.UpdateMaps(*thread.maps);
306*288bf522SAndroid Build Coastguard Worker std::unique_ptr<unwindstack::Regs> unwind_regs(GetBacktraceRegs(regs));
307*288bf522SAndroid Build Coastguard Worker if (!unwind_regs) {
308*288bf522SAndroid Build Coastguard Worker return false;
309*288bf522SAndroid Build Coastguard Worker }
310*288bf522SAndroid Build Coastguard Worker unwindstack::Unwinder unwinder(
311*288bf522SAndroid Build Coastguard Worker MAX_UNWINDING_FRAMES, &cached_map, unwind_regs.get(),
312*288bf522SAndroid Build Coastguard Worker unwindstack::Memory::CreateOfflineMemory(reinterpret_cast<const uint8_t*>(stack), stack_addr,
313*288bf522SAndroid Build Coastguard Worker stack_addr + stack_size));
314*288bf522SAndroid Build Coastguard Worker unwinder.SetResolveNames(false);
315*288bf522SAndroid Build Coastguard Worker unwinder.Unwind();
316*288bf522SAndroid Build Coastguard Worker size_t last_jit_method_frame = UINT_MAX;
317*288bf522SAndroid Build Coastguard Worker for (auto& frame : unwinder.frames()) {
318*288bf522SAndroid Build Coastguard Worker // Unwinding in arm architecture can return 0 pc address.
319*288bf522SAndroid Build Coastguard Worker
320*288bf522SAndroid Build Coastguard Worker // If frame.map_info == nullptr, this frame doesn't hit any map, it could be:
321*288bf522SAndroid Build Coastguard Worker // 1. In an executable map not backed by a file. Note that RecordCommand::ShouldOmitRecord()
322*288bf522SAndroid Build Coastguard Worker // may omit maps only exist memory.
323*288bf522SAndroid Build Coastguard Worker // 2. An incorrectly unwound frame. Likely caused by invalid stack data, as in
324*288bf522SAndroid Build Coastguard Worker // SampleRecord::GetValidStackSize(). Or caused by incomplete JIT debug info.
325*288bf522SAndroid Build Coastguard Worker // We want to remove this frame and callchains following it in either case.
326*288bf522SAndroid Build Coastguard Worker if (frame.map_info == nullptr) {
327*288bf522SAndroid Build Coastguard Worker is_callchain_broken_for_incomplete_jit_debug_info_ = true;
328*288bf522SAndroid Build Coastguard Worker break;
329*288bf522SAndroid Build Coastguard Worker }
330*288bf522SAndroid Build Coastguard Worker if (frame.map_info->flags() & unwindstack::MAPS_FLAGS_JIT_SYMFILE_MAP) {
331*288bf522SAndroid Build Coastguard Worker last_jit_method_frame = ips->size();
332*288bf522SAndroid Build Coastguard Worker }
333*288bf522SAndroid Build Coastguard Worker ips->push_back(frame.pc);
334*288bf522SAndroid Build Coastguard Worker sps->push_back(frame.sp);
335*288bf522SAndroid Build Coastguard Worker }
336*288bf522SAndroid Build Coastguard Worker // If the unwound frames stop near to a JITed method, it may be caused by incomplete JIT debug
337*288bf522SAndroid Build Coastguard Worker // info.
338*288bf522SAndroid Build Coastguard Worker if (last_jit_method_frame != UINT_MAX && last_jit_method_frame + 3 > ips->size()) {
339*288bf522SAndroid Build Coastguard Worker is_callchain_broken_for_incomplete_jit_debug_info_ = true;
340*288bf522SAndroid Build Coastguard Worker }
341*288bf522SAndroid Build Coastguard Worker
342*288bf522SAndroid Build Coastguard Worker uint64_t ip_reg_value;
343*288bf522SAndroid Build Coastguard Worker if (!regs.GetIpRegValue(&ip_reg_value)) {
344*288bf522SAndroid Build Coastguard Worker LOG(ERROR) << "can't get ip reg value";
345*288bf522SAndroid Build Coastguard Worker return false;
346*288bf522SAndroid Build Coastguard Worker }
347*288bf522SAndroid Build Coastguard Worker if (ips->empty()) {
348*288bf522SAndroid Build Coastguard Worker ips->push_back(ip_reg_value);
349*288bf522SAndroid Build Coastguard Worker sps->push_back(sp_reg_value);
350*288bf522SAndroid Build Coastguard Worker } else {
351*288bf522SAndroid Build Coastguard Worker // Check if the unwinder returns ip reg value as the first ip address in callstack.
352*288bf522SAndroid Build Coastguard Worker CHECK_EQ((*ips)[0], ip_reg_value);
353*288bf522SAndroid Build Coastguard Worker }
354*288bf522SAndroid Build Coastguard Worker if (collect_stat_) {
355*288bf522SAndroid Build Coastguard Worker unwinding_result_.used_time = GetSystemClock() - start_time;
356*288bf522SAndroid Build Coastguard Worker unwinding_result_.error_code = unwinder.LastErrorCode();
357*288bf522SAndroid Build Coastguard Worker unwinding_result_.error_addr = unwinder.LastErrorAddress();
358*288bf522SAndroid Build Coastguard Worker unwinding_result_.stack_start = stack_addr;
359*288bf522SAndroid Build Coastguard Worker unwinding_result_.stack_end = stack_addr + stack_size;
360*288bf522SAndroid Build Coastguard Worker }
361*288bf522SAndroid Build Coastguard Worker return true;
362*288bf522SAndroid Build Coastguard Worker }
363*288bf522SAndroid Build Coastguard Worker
Create(bool collect_stat)364*288bf522SAndroid Build Coastguard Worker std::unique_ptr<OfflineUnwinder> OfflineUnwinder::Create(bool collect_stat) {
365*288bf522SAndroid Build Coastguard Worker return std::unique_ptr<OfflineUnwinder>(new OfflineUnwinderImpl(collect_stat));
366*288bf522SAndroid Build Coastguard Worker }
367*288bf522SAndroid Build Coastguard Worker
368*288bf522SAndroid Build Coastguard Worker } // namespace simpleperf
369