xref: /aosp_15_r20/art/runtime/native_stack_dump.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2016 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #include "native_stack_dump.h"
18*795d594fSAndroid Build Coastguard Worker 
19*795d594fSAndroid Build Coastguard Worker #include <memory>
20*795d594fSAndroid Build Coastguard Worker #include <ostream>
21*795d594fSAndroid Build Coastguard Worker #include <string_view>
22*795d594fSAndroid Build Coastguard Worker 
23*795d594fSAndroid Build Coastguard Worker #include <stdio.h>
24*795d594fSAndroid Build Coastguard Worker 
25*795d594fSAndroid Build Coastguard Worker #include "art_method.h"
26*795d594fSAndroid Build Coastguard Worker 
27*795d594fSAndroid Build Coastguard Worker // For DumpNativeStack.
28*795d594fSAndroid Build Coastguard Worker #include <unwindstack/AndroidUnwinder.h>
29*795d594fSAndroid Build Coastguard Worker 
30*795d594fSAndroid Build Coastguard Worker #if defined(__linux__)
31*795d594fSAndroid Build Coastguard Worker 
32*795d594fSAndroid Build Coastguard Worker #include <vector>
33*795d594fSAndroid Build Coastguard Worker 
34*795d594fSAndroid Build Coastguard Worker #include <linux/unistd.h>
35*795d594fSAndroid Build Coastguard Worker #include <poll.h>
36*795d594fSAndroid Build Coastguard Worker #include <signal.h>
37*795d594fSAndroid Build Coastguard Worker #include <stdlib.h>
38*795d594fSAndroid Build Coastguard Worker #include <sys/time.h>
39*795d594fSAndroid Build Coastguard Worker #include <sys/types.h>
40*795d594fSAndroid Build Coastguard Worker 
41*795d594fSAndroid Build Coastguard Worker #include "android-base/file.h"
42*795d594fSAndroid Build Coastguard Worker #include "android-base/stringprintf.h"
43*795d594fSAndroid Build Coastguard Worker #include "android-base/strings.h"
44*795d594fSAndroid Build Coastguard Worker 
45*795d594fSAndroid Build Coastguard Worker #include "arch/instruction_set.h"
46*795d594fSAndroid Build Coastguard Worker #include "base/aborting.h"
47*795d594fSAndroid Build Coastguard Worker #include "base/bit_utils.h"
48*795d594fSAndroid Build Coastguard Worker #include "base/file_utils.h"
49*795d594fSAndroid Build Coastguard Worker #include "base/memory_tool.h"
50*795d594fSAndroid Build Coastguard Worker #include "base/mutex.h"
51*795d594fSAndroid Build Coastguard Worker #include "base/os.h"
52*795d594fSAndroid Build Coastguard Worker #include "base/unix_file/fd_file.h"
53*795d594fSAndroid Build Coastguard Worker #include "base/utils.h"
54*795d594fSAndroid Build Coastguard Worker #include "class_linker.h"
55*795d594fSAndroid Build Coastguard Worker #include "entrypoints/runtime_asm_entrypoints.h"
56*795d594fSAndroid Build Coastguard Worker #include "oat/oat_quick_method_header.h"
57*795d594fSAndroid Build Coastguard Worker #include "runtime.h"
58*795d594fSAndroid Build Coastguard Worker #include "thread-current-inl.h"
59*795d594fSAndroid Build Coastguard Worker 
60*795d594fSAndroid Build Coastguard Worker #endif
61*795d594fSAndroid Build Coastguard Worker 
62*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN {
63*795d594fSAndroid Build Coastguard Worker 
64*795d594fSAndroid Build Coastguard Worker #if defined(__linux__)
65*795d594fSAndroid Build Coastguard Worker 
66*795d594fSAndroid Build Coastguard Worker using android::base::StringPrintf;
67*795d594fSAndroid Build Coastguard Worker 
68*795d594fSAndroid Build Coastguard Worker static constexpr bool kUseAddr2line = !kIsTargetBuild;
69*795d594fSAndroid Build Coastguard Worker 
FindAddr2line()70*795d594fSAndroid Build Coastguard Worker std::string FindAddr2line() {
71*795d594fSAndroid Build Coastguard Worker #if !defined(ART_TARGET) && !defined(ART_CLANG_PATH)
72*795d594fSAndroid Build Coastguard Worker   #error "ART_CLANG_PATH must be defined on host build"
73*795d594fSAndroid Build Coastguard Worker #endif
74*795d594fSAndroid Build Coastguard Worker #if defined(ART_CLANG_PATH)
75*795d594fSAndroid Build Coastguard Worker   const char* env_value = getenv("ANDROID_BUILD_TOP");
76*795d594fSAndroid Build Coastguard Worker   std::string_view top(env_value != nullptr ? env_value : ".");
77*795d594fSAndroid Build Coastguard Worker   return std::string(top) + "/" + ART_CLANG_PATH + "/bin/llvm-addr2line";
78*795d594fSAndroid Build Coastguard Worker #else
79*795d594fSAndroid Build Coastguard Worker   return std::string("llvm-addr2line");
80*795d594fSAndroid Build Coastguard Worker #endif
81*795d594fSAndroid Build Coastguard Worker }
82*795d594fSAndroid Build Coastguard Worker 
83*795d594fSAndroid Build Coastguard Worker ALWAYS_INLINE
WritePrefix(std::ostream & os,const char * prefix,bool odd)84*795d594fSAndroid Build Coastguard Worker static inline void WritePrefix(std::ostream& os, const char* prefix, bool odd) {
85*795d594fSAndroid Build Coastguard Worker   if (prefix != nullptr) {
86*795d594fSAndroid Build Coastguard Worker     os << prefix;
87*795d594fSAndroid Build Coastguard Worker   }
88*795d594fSAndroid Build Coastguard Worker   os << "  ";
89*795d594fSAndroid Build Coastguard Worker   if (!odd) {
90*795d594fSAndroid Build Coastguard Worker     os << " ";
91*795d594fSAndroid Build Coastguard Worker   }
92*795d594fSAndroid Build Coastguard Worker }
93*795d594fSAndroid Build Coastguard Worker 
94*795d594fSAndroid Build Coastguard Worker // The state of an open pipe to addr2line. In "server" mode, addr2line takes input on stdin
95*795d594fSAndroid Build Coastguard Worker // and prints the result to stdout. This struct keeps the state of the open connection.
96*795d594fSAndroid Build Coastguard Worker struct Addr2linePipe {
Addr2linePipeart::Addr2linePipe97*795d594fSAndroid Build Coastguard Worker   Addr2linePipe(int in_fd, int out_fd, const std::string& file_name, pid_t pid)
98*795d594fSAndroid Build Coastguard Worker       : in(in_fd, false), out(out_fd, false), file(file_name), child_pid(pid), odd(true) {}
99*795d594fSAndroid Build Coastguard Worker 
~Addr2linePipeart::Addr2linePipe100*795d594fSAndroid Build Coastguard Worker   ~Addr2linePipe() {
101*795d594fSAndroid Build Coastguard Worker     kill(child_pid, SIGKILL);
102*795d594fSAndroid Build Coastguard Worker   }
103*795d594fSAndroid Build Coastguard Worker 
104*795d594fSAndroid Build Coastguard Worker   File in;      // The file descriptor that is connected to the output of addr2line.
105*795d594fSAndroid Build Coastguard Worker   File out;     // The file descriptor that is connected to the input of addr2line.
106*795d594fSAndroid Build Coastguard Worker 
107*795d594fSAndroid Build Coastguard Worker   const std::string file;     // The file addr2line is working on, so that we know when to close
108*795d594fSAndroid Build Coastguard Worker                               // and restart.
109*795d594fSAndroid Build Coastguard Worker   const pid_t child_pid;      // The pid of the child, which we should kill when we're done.
110*795d594fSAndroid Build Coastguard Worker   bool odd;                   // Print state for indentation of lines.
111*795d594fSAndroid Build Coastguard Worker };
112*795d594fSAndroid Build Coastguard Worker 
Connect(const std::string & name,const char * args[])113*795d594fSAndroid Build Coastguard Worker static std::unique_ptr<Addr2linePipe> Connect(const std::string& name, const char* args[]) {
114*795d594fSAndroid Build Coastguard Worker   int caller_to_addr2line[2];
115*795d594fSAndroid Build Coastguard Worker   int addr2line_to_caller[2];
116*795d594fSAndroid Build Coastguard Worker 
117*795d594fSAndroid Build Coastguard Worker   if (pipe(caller_to_addr2line) == -1) {
118*795d594fSAndroid Build Coastguard Worker     return nullptr;
119*795d594fSAndroid Build Coastguard Worker   }
120*795d594fSAndroid Build Coastguard Worker   if (pipe(addr2line_to_caller) == -1) {
121*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
122*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
123*795d594fSAndroid Build Coastguard Worker     return nullptr;
124*795d594fSAndroid Build Coastguard Worker   }
125*795d594fSAndroid Build Coastguard Worker 
126*795d594fSAndroid Build Coastguard Worker   pid_t pid = fork();
127*795d594fSAndroid Build Coastguard Worker   if (pid == -1) {
128*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
129*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
130*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[0]);
131*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
132*795d594fSAndroid Build Coastguard Worker     return nullptr;
133*795d594fSAndroid Build Coastguard Worker   }
134*795d594fSAndroid Build Coastguard Worker 
135*795d594fSAndroid Build Coastguard Worker   if (pid == 0) {
136*795d594fSAndroid Build Coastguard Worker     dup2(caller_to_addr2line[0], STDIN_FILENO);
137*795d594fSAndroid Build Coastguard Worker     dup2(addr2line_to_caller[1], STDOUT_FILENO);
138*795d594fSAndroid Build Coastguard Worker 
139*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
140*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
141*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[0]);
142*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
143*795d594fSAndroid Build Coastguard Worker 
144*795d594fSAndroid Build Coastguard Worker     execv(args[0], const_cast<char* const*>(args));
145*795d594fSAndroid Build Coastguard Worker     exit(1);
146*795d594fSAndroid Build Coastguard Worker   } else {
147*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
148*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
149*795d594fSAndroid Build Coastguard Worker     return std::make_unique<Addr2linePipe>(addr2line_to_caller[0],
150*795d594fSAndroid Build Coastguard Worker                                            caller_to_addr2line[1],
151*795d594fSAndroid Build Coastguard Worker                                            name,
152*795d594fSAndroid Build Coastguard Worker                                            pid);
153*795d594fSAndroid Build Coastguard Worker   }
154*795d594fSAndroid Build Coastguard Worker }
155*795d594fSAndroid Build Coastguard Worker 
Drain(size_t expected,const char * prefix,std::unique_ptr<Addr2linePipe> * pipe,std::ostream & os)156*795d594fSAndroid Build Coastguard Worker static void Drain(size_t expected,
157*795d594fSAndroid Build Coastguard Worker                   const char* prefix,
158*795d594fSAndroid Build Coastguard Worker                   std::unique_ptr<Addr2linePipe>* pipe /* inout */,
159*795d594fSAndroid Build Coastguard Worker                   std::ostream& os) {
160*795d594fSAndroid Build Coastguard Worker   DCHECK(pipe != nullptr);
161*795d594fSAndroid Build Coastguard Worker   DCHECK(pipe->get() != nullptr);
162*795d594fSAndroid Build Coastguard Worker   int in = pipe->get()->in.Fd();
163*795d594fSAndroid Build Coastguard Worker   DCHECK_GE(in, 0);
164*795d594fSAndroid Build Coastguard Worker 
165*795d594fSAndroid Build Coastguard Worker   bool prefix_written = false;
166*795d594fSAndroid Build Coastguard Worker 
167*795d594fSAndroid Build Coastguard Worker   for (;;) {
168*795d594fSAndroid Build Coastguard Worker     constexpr uint32_t kWaitTimeExpectedMilli = 500;
169*795d594fSAndroid Build Coastguard Worker     constexpr uint32_t kWaitTimeUnexpectedMilli = 50;
170*795d594fSAndroid Build Coastguard Worker 
171*795d594fSAndroid Build Coastguard Worker     int timeout = expected > 0 ? kWaitTimeExpectedMilli : kWaitTimeUnexpectedMilli;
172*795d594fSAndroid Build Coastguard Worker     struct pollfd read_fd{in, POLLIN, 0};
173*795d594fSAndroid Build Coastguard Worker     int retval = TEMP_FAILURE_RETRY(poll(&read_fd, 1, timeout));
174*795d594fSAndroid Build Coastguard Worker     if (retval == -1) {
175*795d594fSAndroid Build Coastguard Worker       // An error occurred.
176*795d594fSAndroid Build Coastguard Worker       pipe->reset();
177*795d594fSAndroid Build Coastguard Worker       return;
178*795d594fSAndroid Build Coastguard Worker     }
179*795d594fSAndroid Build Coastguard Worker 
180*795d594fSAndroid Build Coastguard Worker     if (retval == 0) {
181*795d594fSAndroid Build Coastguard Worker       // Timeout.
182*795d594fSAndroid Build Coastguard Worker       return;
183*795d594fSAndroid Build Coastguard Worker     }
184*795d594fSAndroid Build Coastguard Worker 
185*795d594fSAndroid Build Coastguard Worker     if (!(read_fd.revents & POLLIN)) {
186*795d594fSAndroid Build Coastguard Worker       // addr2line call exited.
187*795d594fSAndroid Build Coastguard Worker       pipe->reset();
188*795d594fSAndroid Build Coastguard Worker       return;
189*795d594fSAndroid Build Coastguard Worker     }
190*795d594fSAndroid Build Coastguard Worker 
191*795d594fSAndroid Build Coastguard Worker     constexpr size_t kMaxBuffer = 128;  // Relatively small buffer. Should be OK as we're on an
192*795d594fSAndroid Build Coastguard Worker     // alt stack, but just to be sure...
193*795d594fSAndroid Build Coastguard Worker     char buffer[kMaxBuffer];
194*795d594fSAndroid Build Coastguard Worker     memset(buffer, 0, kMaxBuffer);
195*795d594fSAndroid Build Coastguard Worker     int bytes_read = TEMP_FAILURE_RETRY(read(in, buffer, kMaxBuffer - 1));
196*795d594fSAndroid Build Coastguard Worker     if (bytes_read <= 0) {
197*795d594fSAndroid Build Coastguard Worker       // This should not really happen...
198*795d594fSAndroid Build Coastguard Worker       pipe->reset();
199*795d594fSAndroid Build Coastguard Worker       return;
200*795d594fSAndroid Build Coastguard Worker     }
201*795d594fSAndroid Build Coastguard Worker     buffer[bytes_read] = '\0';
202*795d594fSAndroid Build Coastguard Worker 
203*795d594fSAndroid Build Coastguard Worker     char* tmp = buffer;
204*795d594fSAndroid Build Coastguard Worker     while (*tmp != 0) {
205*795d594fSAndroid Build Coastguard Worker       if (!prefix_written) {
206*795d594fSAndroid Build Coastguard Worker         WritePrefix(os, prefix, (*pipe)->odd);
207*795d594fSAndroid Build Coastguard Worker         prefix_written = true;
208*795d594fSAndroid Build Coastguard Worker       }
209*795d594fSAndroid Build Coastguard Worker       char* new_line = strchr(tmp, '\n');
210*795d594fSAndroid Build Coastguard Worker       if (new_line == nullptr) {
211*795d594fSAndroid Build Coastguard Worker         os << tmp;
212*795d594fSAndroid Build Coastguard Worker 
213*795d594fSAndroid Build Coastguard Worker         break;
214*795d594fSAndroid Build Coastguard Worker       } else {
215*795d594fSAndroid Build Coastguard Worker         char saved = *(new_line + 1);
216*795d594fSAndroid Build Coastguard Worker         *(new_line + 1) = 0;
217*795d594fSAndroid Build Coastguard Worker         os << tmp;
218*795d594fSAndroid Build Coastguard Worker         *(new_line + 1) = saved;
219*795d594fSAndroid Build Coastguard Worker 
220*795d594fSAndroid Build Coastguard Worker         tmp = new_line + 1;
221*795d594fSAndroid Build Coastguard Worker         prefix_written = false;
222*795d594fSAndroid Build Coastguard Worker         (*pipe)->odd = !(*pipe)->odd;
223*795d594fSAndroid Build Coastguard Worker 
224*795d594fSAndroid Build Coastguard Worker         if (expected > 0) {
225*795d594fSAndroid Build Coastguard Worker           expected--;
226*795d594fSAndroid Build Coastguard Worker         }
227*795d594fSAndroid Build Coastguard Worker       }
228*795d594fSAndroid Build Coastguard Worker     }
229*795d594fSAndroid Build Coastguard Worker   }
230*795d594fSAndroid Build Coastguard Worker }
231*795d594fSAndroid Build Coastguard Worker 
Addr2line(const std::string & map_src,uintptr_t offset,std::ostream & os,const char * prefix,std::unique_ptr<Addr2linePipe> * pipe)232*795d594fSAndroid Build Coastguard Worker static void Addr2line(const std::string& map_src,
233*795d594fSAndroid Build Coastguard Worker                       uintptr_t offset,
234*795d594fSAndroid Build Coastguard Worker                       std::ostream& os,
235*795d594fSAndroid Build Coastguard Worker                       const char* prefix,
236*795d594fSAndroid Build Coastguard Worker                       std::unique_ptr<Addr2linePipe>* pipe /* inout */) {
237*795d594fSAndroid Build Coastguard Worker   std::array<const char*, 3> kIgnoreSuffixes{ ".dex", ".jar", ".vdex" };
238*795d594fSAndroid Build Coastguard Worker   for (const char* ignore_suffix : kIgnoreSuffixes) {
239*795d594fSAndroid Build Coastguard Worker     if (map_src.ends_with(ignore_suffix)) {
240*795d594fSAndroid Build Coastguard Worker       // Ignore file names that do not have map information addr2line can consume. e.g. vdex
241*795d594fSAndroid Build Coastguard Worker       // files are special frames injected for the interpreter so they don't have any line
242*795d594fSAndroid Build Coastguard Worker       // number information available.
243*795d594fSAndroid Build Coastguard Worker       return;
244*795d594fSAndroid Build Coastguard Worker     }
245*795d594fSAndroid Build Coastguard Worker   }
246*795d594fSAndroid Build Coastguard Worker   if (map_src == "[vdso]") {
247*795d594fSAndroid Build Coastguard Worker     // addr2line will not work on the vdso.
248*795d594fSAndroid Build Coastguard Worker     return;
249*795d594fSAndroid Build Coastguard Worker   }
250*795d594fSAndroid Build Coastguard Worker 
251*795d594fSAndroid Build Coastguard Worker   if (*pipe == nullptr || (*pipe)->file != map_src) {
252*795d594fSAndroid Build Coastguard Worker     if (*pipe != nullptr) {
253*795d594fSAndroid Build Coastguard Worker       Drain(0, prefix, pipe, os);
254*795d594fSAndroid Build Coastguard Worker     }
255*795d594fSAndroid Build Coastguard Worker     pipe->reset();  // Close early.
256*795d594fSAndroid Build Coastguard Worker 
257*795d594fSAndroid Build Coastguard Worker     std::string addr2linePath = FindAddr2line();
258*795d594fSAndroid Build Coastguard Worker     const char* args[7] = {
259*795d594fSAndroid Build Coastguard Worker         addr2linePath.c_str(),
260*795d594fSAndroid Build Coastguard Worker         "--functions",
261*795d594fSAndroid Build Coastguard Worker         "--inlines",
262*795d594fSAndroid Build Coastguard Worker         "--demangle",
263*795d594fSAndroid Build Coastguard Worker         "-e",
264*795d594fSAndroid Build Coastguard Worker         map_src.c_str(),
265*795d594fSAndroid Build Coastguard Worker         nullptr
266*795d594fSAndroid Build Coastguard Worker     };
267*795d594fSAndroid Build Coastguard Worker     *pipe = Connect(map_src, args);
268*795d594fSAndroid Build Coastguard Worker   }
269*795d594fSAndroid Build Coastguard Worker 
270*795d594fSAndroid Build Coastguard Worker   Addr2linePipe* pipe_ptr = pipe->get();
271*795d594fSAndroid Build Coastguard Worker   if (pipe_ptr == nullptr) {
272*795d594fSAndroid Build Coastguard Worker     // Failed...
273*795d594fSAndroid Build Coastguard Worker     return;
274*795d594fSAndroid Build Coastguard Worker   }
275*795d594fSAndroid Build Coastguard Worker 
276*795d594fSAndroid Build Coastguard Worker   // Send the offset.
277*795d594fSAndroid Build Coastguard Worker   const std::string hex_offset = StringPrintf("0x%zx\n", offset);
278*795d594fSAndroid Build Coastguard Worker 
279*795d594fSAndroid Build Coastguard Worker   if (!pipe_ptr->out.WriteFully(hex_offset.data(), hex_offset.length())) {
280*795d594fSAndroid Build Coastguard Worker     // Error. :-(
281*795d594fSAndroid Build Coastguard Worker     pipe->reset();
282*795d594fSAndroid Build Coastguard Worker     return;
283*795d594fSAndroid Build Coastguard Worker   }
284*795d594fSAndroid Build Coastguard Worker 
285*795d594fSAndroid Build Coastguard Worker   // Now drain (expecting two lines).
286*795d594fSAndroid Build Coastguard Worker   Drain(2U, prefix, pipe, os);
287*795d594fSAndroid Build Coastguard Worker }
288*795d594fSAndroid Build Coastguard Worker 
RunCommand(const std::string & cmd)289*795d594fSAndroid Build Coastguard Worker static bool RunCommand(const std::string& cmd) {
290*795d594fSAndroid Build Coastguard Worker   FILE* stream = popen(cmd.c_str(), "r");
291*795d594fSAndroid Build Coastguard Worker   if (stream) {
292*795d594fSAndroid Build Coastguard Worker     // Consume the stdout until we encounter EOF when the tool exits.
293*795d594fSAndroid Build Coastguard Worker     // Otherwise the tool would complain to stderr when the stream is closed.
294*795d594fSAndroid Build Coastguard Worker     char buffer[64];
295*795d594fSAndroid Build Coastguard Worker     while (fread(buffer, 1, sizeof(buffer), stream) == sizeof(buffer)) {}
296*795d594fSAndroid Build Coastguard Worker     pclose(stream);
297*795d594fSAndroid Build Coastguard Worker     return true;
298*795d594fSAndroid Build Coastguard Worker   } else {
299*795d594fSAndroid Build Coastguard Worker     return false;
300*795d594fSAndroid Build Coastguard Worker   }
301*795d594fSAndroid Build Coastguard Worker }
302*795d594fSAndroid Build Coastguard Worker 
303*795d594fSAndroid Build Coastguard Worker // Remove method parameters by finding matching top-level parenthesis and removing them.
304*795d594fSAndroid Build Coastguard Worker // Since functions can be defined inside functions, this can remove multiple substrings.
StripParameters(std::string name)305*795d594fSAndroid Build Coastguard Worker std::string StripParameters(std::string name) {
306*795d594fSAndroid Build Coastguard Worker   size_t end = name.size();
307*795d594fSAndroid Build Coastguard Worker   int nesting = 0;
308*795d594fSAndroid Build Coastguard Worker   for (ssize_t i = name.size() - 1; i > 0; i--) {
309*795d594fSAndroid Build Coastguard Worker     if (name[i] == ')' && nesting++ == 0) {
310*795d594fSAndroid Build Coastguard Worker       end = i + 1;
311*795d594fSAndroid Build Coastguard Worker     }
312*795d594fSAndroid Build Coastguard Worker     if (name[i] == '(' && --nesting == 0) {
313*795d594fSAndroid Build Coastguard Worker       name = name.erase(i, end - i);
314*795d594fSAndroid Build Coastguard Worker     }
315*795d594fSAndroid Build Coastguard Worker   }
316*795d594fSAndroid Build Coastguard Worker   return name;
317*795d594fSAndroid Build Coastguard Worker }
318*795d594fSAndroid Build Coastguard Worker 
DumpNativeStack(std::ostream & os,pid_t tid,const char * prefix,ArtMethod * current_method,void * ucontext_ptr,bool skip_frames)319*795d594fSAndroid Build Coastguard Worker void DumpNativeStack(std::ostream& os,
320*795d594fSAndroid Build Coastguard Worker                      pid_t tid,
321*795d594fSAndroid Build Coastguard Worker                      const char* prefix,
322*795d594fSAndroid Build Coastguard Worker                      ArtMethod* current_method,
323*795d594fSAndroid Build Coastguard Worker                      void* ucontext_ptr,
324*795d594fSAndroid Build Coastguard Worker                      bool skip_frames) {
325*795d594fSAndroid Build Coastguard Worker   unwindstack::AndroidLocalUnwinder unwinder;
326*795d594fSAndroid Build Coastguard Worker   DumpNativeStack(os, unwinder, tid, prefix, current_method, ucontext_ptr, skip_frames);
327*795d594fSAndroid Build Coastguard Worker }
328*795d594fSAndroid Build Coastguard Worker 
DumpNativeStack(std::ostream & os,unwindstack::AndroidLocalUnwinder & unwinder,pid_t tid,const char * prefix,ArtMethod * current_method,void * ucontext_ptr,bool skip_frames)329*795d594fSAndroid Build Coastguard Worker void DumpNativeStack(std::ostream& os,
330*795d594fSAndroid Build Coastguard Worker                      unwindstack::AndroidLocalUnwinder& unwinder,
331*795d594fSAndroid Build Coastguard Worker                      pid_t tid,
332*795d594fSAndroid Build Coastguard Worker                      const char* prefix,
333*795d594fSAndroid Build Coastguard Worker                      ArtMethod* current_method,
334*795d594fSAndroid Build Coastguard Worker                      void* ucontext_ptr,
335*795d594fSAndroid Build Coastguard Worker                      bool skip_frames) {
336*795d594fSAndroid Build Coastguard Worker   // Historical note: This was disabled when running under Valgrind (b/18119146).
337*795d594fSAndroid Build Coastguard Worker 
338*795d594fSAndroid Build Coastguard Worker   unwindstack::AndroidUnwinderData data(!skip_frames /*show_all_frames*/);
339*795d594fSAndroid Build Coastguard Worker   bool unwind_ret;
340*795d594fSAndroid Build Coastguard Worker   if (ucontext_ptr != nullptr) {
341*795d594fSAndroid Build Coastguard Worker     unwind_ret = unwinder.Unwind(ucontext_ptr, data);
342*795d594fSAndroid Build Coastguard Worker   } else {
343*795d594fSAndroid Build Coastguard Worker     unwind_ret = unwinder.Unwind(tid, data);
344*795d594fSAndroid Build Coastguard Worker   }
345*795d594fSAndroid Build Coastguard Worker   if (!unwind_ret) {
346*795d594fSAndroid Build Coastguard Worker     os << prefix << "(Unwind failed for thread " << tid << ": "
347*795d594fSAndroid Build Coastguard Worker        <<  data.GetErrorString() << ")" << std::endl;
348*795d594fSAndroid Build Coastguard Worker     return;
349*795d594fSAndroid Build Coastguard Worker   }
350*795d594fSAndroid Build Coastguard Worker 
351*795d594fSAndroid Build Coastguard Worker   // Check whether we have and should use addr2line.
352*795d594fSAndroid Build Coastguard Worker   bool use_addr2line;
353*795d594fSAndroid Build Coastguard Worker   if (kUseAddr2line) {
354*795d594fSAndroid Build Coastguard Worker     // Try to run it to see whether we have it. Push an argument so that it doesn't assume a.out
355*795d594fSAndroid Build Coastguard Worker     // and print to stderr.
356*795d594fSAndroid Build Coastguard Worker     use_addr2line = (gAborting > 0) && RunCommand(FindAddr2line() + " -h");
357*795d594fSAndroid Build Coastguard Worker   } else {
358*795d594fSAndroid Build Coastguard Worker     use_addr2line = false;
359*795d594fSAndroid Build Coastguard Worker   }
360*795d594fSAndroid Build Coastguard Worker 
361*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<Addr2linePipe> addr2line_state;
362*795d594fSAndroid Build Coastguard Worker   data.DemangleFunctionNames();
363*795d594fSAndroid Build Coastguard Worker   bool holds_mutator_lock =  Locks::mutator_lock_->IsSharedHeld(Thread::Current());
364*795d594fSAndroid Build Coastguard Worker   for (const unwindstack::FrameData& frame : data.frames) {
365*795d594fSAndroid Build Coastguard Worker     // We produce output like this:
366*795d594fSAndroid Build Coastguard Worker     // ]    #00 pc 000075bb8  /system/lib/libc.so (unwind_backtrace_thread+536)
367*795d594fSAndroid Build Coastguard Worker     // In order for parsing tools to continue to function, the stack dump
368*795d594fSAndroid Build Coastguard Worker     // format must at least adhere to this format:
369*795d594fSAndroid Build Coastguard Worker     //  #XX pc <RELATIVE_ADDR>  <FULL_PATH_TO_SHARED_LIBRARY> ...
370*795d594fSAndroid Build Coastguard Worker     // The parsers require a single space before and after pc, and two spaces
371*795d594fSAndroid Build Coastguard Worker     // after the <RELATIVE_ADDR>. There can be any prefix data before the
372*795d594fSAndroid Build Coastguard Worker     // #XX. <RELATIVE_ADDR> has to be a hex number but with no 0x prefix.
373*795d594fSAndroid Build Coastguard Worker     os << prefix << StringPrintf("#%02zu pc ", frame.num);
374*795d594fSAndroid Build Coastguard Worker     bool try_addr2line = false;
375*795d594fSAndroid Build Coastguard Worker     if (frame.map_info == nullptr) {
376*795d594fSAndroid Build Coastguard Worker       os << StringPrintf("%08" PRIx64 "  ???", frame.pc);
377*795d594fSAndroid Build Coastguard Worker     } else {
378*795d594fSAndroid Build Coastguard Worker       os << StringPrintf("%08" PRIx64 "  ", frame.rel_pc);
379*795d594fSAndroid Build Coastguard Worker       const std::shared_ptr<unwindstack::MapInfo>& map_info = frame.map_info;
380*795d594fSAndroid Build Coastguard Worker       if (map_info->name().empty()) {
381*795d594fSAndroid Build Coastguard Worker         os << StringPrintf("<anonymous:%" PRIx64 ">", map_info->start());
382*795d594fSAndroid Build Coastguard Worker       } else {
383*795d594fSAndroid Build Coastguard Worker         os << map_info->name().c_str();
384*795d594fSAndroid Build Coastguard Worker       }
385*795d594fSAndroid Build Coastguard Worker       if (map_info->elf_start_offset() != 0) {
386*795d594fSAndroid Build Coastguard Worker         os << StringPrintf(" (offset %" PRIx64 ")", map_info->elf_start_offset());
387*795d594fSAndroid Build Coastguard Worker       }
388*795d594fSAndroid Build Coastguard Worker       os << " (";
389*795d594fSAndroid Build Coastguard Worker       if (!frame.function_name.empty()) {
390*795d594fSAndroid Build Coastguard Worker         // Remove parameters from the printed function name to improve signal/noise in the logs.
391*795d594fSAndroid Build Coastguard Worker         // Also, ANRs are often trimmed, so printing less means we get more useful data out.
392*795d594fSAndroid Build Coastguard Worker         // We can still symbolize the function based on the PC and build-id (including inlining).
393*795d594fSAndroid Build Coastguard Worker         os << StripParameters(frame.function_name.c_str());
394*795d594fSAndroid Build Coastguard Worker         if (frame.function_offset != 0) {
395*795d594fSAndroid Build Coastguard Worker           os << "+" << frame.function_offset;
396*795d594fSAndroid Build Coastguard Worker         }
397*795d594fSAndroid Build Coastguard Worker         // Functions found using the gdb jit interface will be in an empty
398*795d594fSAndroid Build Coastguard Worker         // map that cannot be found using addr2line.
399*795d594fSAndroid Build Coastguard Worker         if (!map_info->name().empty()) {
400*795d594fSAndroid Build Coastguard Worker           try_addr2line = true;
401*795d594fSAndroid Build Coastguard Worker         }
402*795d594fSAndroid Build Coastguard Worker       } else if (current_method != nullptr && holds_mutator_lock) {
403*795d594fSAndroid Build Coastguard Worker         const OatQuickMethodHeader* header = current_method->GetOatQuickMethodHeader(frame.pc);
404*795d594fSAndroid Build Coastguard Worker         if (header != nullptr) {
405*795d594fSAndroid Build Coastguard Worker           const void* start_of_code = header->GetCode();
406*795d594fSAndroid Build Coastguard Worker           os << current_method->JniLongName() << "+"
407*795d594fSAndroid Build Coastguard Worker              << (frame.pc - reinterpret_cast<uint64_t>(start_of_code));
408*795d594fSAndroid Build Coastguard Worker         } else {
409*795d594fSAndroid Build Coastguard Worker           os << "???";
410*795d594fSAndroid Build Coastguard Worker         }
411*795d594fSAndroid Build Coastguard Worker       } else {
412*795d594fSAndroid Build Coastguard Worker         os << "???";
413*795d594fSAndroid Build Coastguard Worker       }
414*795d594fSAndroid Build Coastguard Worker       os << ")";
415*795d594fSAndroid Build Coastguard Worker       std::string build_id = map_info->GetPrintableBuildID();
416*795d594fSAndroid Build Coastguard Worker       if (!build_id.empty()) {
417*795d594fSAndroid Build Coastguard Worker         os << " (BuildId: " << build_id << ")";
418*795d594fSAndroid Build Coastguard Worker       }
419*795d594fSAndroid Build Coastguard Worker     }
420*795d594fSAndroid Build Coastguard Worker     os << std::endl;
421*795d594fSAndroid Build Coastguard Worker     if (try_addr2line && use_addr2line) {
422*795d594fSAndroid Build Coastguard Worker       // Guaranteed that map_info is not nullptr and name is non-empty.
423*795d594fSAndroid Build Coastguard Worker       Addr2line(frame.map_info->name(), frame.rel_pc, os, prefix, &addr2line_state);
424*795d594fSAndroid Build Coastguard Worker     }
425*795d594fSAndroid Build Coastguard Worker   }
426*795d594fSAndroid Build Coastguard Worker 
427*795d594fSAndroid Build Coastguard Worker   if (addr2line_state != nullptr) {
428*795d594fSAndroid Build Coastguard Worker     Drain(0, prefix, &addr2line_state, os);
429*795d594fSAndroid Build Coastguard Worker   }
430*795d594fSAndroid Build Coastguard Worker }
431*795d594fSAndroid Build Coastguard Worker 
432*795d594fSAndroid Build Coastguard Worker #elif defined(__APPLE__)
433*795d594fSAndroid Build Coastguard Worker 
434*795d594fSAndroid Build Coastguard Worker void DumpNativeStack([[maybe_unused]] std::ostream& os,
435*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] pid_t tid,
436*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] const char* prefix,
437*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] ArtMethod* current_method,
438*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] void* ucontext_ptr,
439*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] bool skip_frames) {}
440*795d594fSAndroid Build Coastguard Worker 
441*795d594fSAndroid Build Coastguard Worker void DumpNativeStack([[maybe_unused]] std::ostream& os,
442*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] unwindstack::AndroidLocalUnwinder& existing_map,
443*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] pid_t tid,
444*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] const char* prefix,
445*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] ArtMethod* current_method,
446*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] void* ucontext_ptr,
447*795d594fSAndroid Build Coastguard Worker                      [[maybe_unused]] bool skip_frames) {}
448*795d594fSAndroid Build Coastguard Worker 
449*795d594fSAndroid Build Coastguard Worker #else
450*795d594fSAndroid Build Coastguard Worker #error "Unsupported architecture for native stack dumps."
451*795d594fSAndroid Build Coastguard Worker #endif
452*795d594fSAndroid Build Coastguard Worker 
453*795d594fSAndroid Build Coastguard Worker }  // namespace art
454