xref: /aosp_15_r20/art/tools/signal_dumper/signal_dumper.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2018 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 <dirent.h>
18*795d594fSAndroid Build Coastguard Worker #include <inttypes.h>
19*795d594fSAndroid Build Coastguard Worker #include <poll.h>
20*795d594fSAndroid Build Coastguard Worker #include <sys/prctl.h>
21*795d594fSAndroid Build Coastguard Worker #include <sys/ptrace.h>
22*795d594fSAndroid Build Coastguard Worker #include <sys/types.h>
23*795d594fSAndroid Build Coastguard Worker #include <sys/wait.h>
24*795d594fSAndroid Build Coastguard Worker #include <unistd.h>
25*795d594fSAndroid Build Coastguard Worker 
26*795d594fSAndroid Build Coastguard Worker #include <csignal>
27*795d594fSAndroid Build Coastguard Worker #include <cstdlib>
28*795d594fSAndroid Build Coastguard Worker #include <cstring>
29*795d594fSAndroid Build Coastguard Worker #include <iostream>
30*795d594fSAndroid Build Coastguard Worker #include <thread>
31*795d594fSAndroid Build Coastguard Worker #include <memory>
32*795d594fSAndroid Build Coastguard Worker #include <set>
33*795d594fSAndroid Build Coastguard Worker #include <string>
34*795d594fSAndroid Build Coastguard Worker 
35*795d594fSAndroid Build Coastguard Worker #include <android-base/file.h>
36*795d594fSAndroid Build Coastguard Worker #include <android-base/logging.h>
37*795d594fSAndroid Build Coastguard Worker #include <android-base/macros.h>
38*795d594fSAndroid Build Coastguard Worker #include <android-base/parseint.h>
39*795d594fSAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
40*795d594fSAndroid Build Coastguard Worker #include <android-base/strings.h>
41*795d594fSAndroid Build Coastguard Worker #include <android-base/unique_fd.h>
42*795d594fSAndroid Build Coastguard Worker #include <unwindstack/AndroidUnwinder.h>
43*795d594fSAndroid Build Coastguard Worker 
44*795d594fSAndroid Build Coastguard Worker namespace art {
45*795d594fSAndroid Build Coastguard Worker namespace {
46*795d594fSAndroid Build Coastguard Worker 
47*795d594fSAndroid Build Coastguard Worker using android::base::StringPrintf;
48*795d594fSAndroid Build Coastguard Worker using android::base::unique_fd;
49*795d594fSAndroid Build Coastguard Worker 
50*795d594fSAndroid Build Coastguard Worker constexpr bool kUseAddr2line = true;
51*795d594fSAndroid Build Coastguard Worker 
52*795d594fSAndroid Build Coastguard Worker namespace timeout_signal {
53*795d594fSAndroid Build Coastguard Worker 
54*795d594fSAndroid Build Coastguard Worker class SignalSet {
55*795d594fSAndroid Build Coastguard Worker  public:
SignalSet()56*795d594fSAndroid Build Coastguard Worker   SignalSet() {
57*795d594fSAndroid Build Coastguard Worker     if (sigemptyset(&set_) == -1) {
58*795d594fSAndroid Build Coastguard Worker       PLOG(FATAL) << "sigemptyset failed";
59*795d594fSAndroid Build Coastguard Worker     }
60*795d594fSAndroid Build Coastguard Worker   }
61*795d594fSAndroid Build Coastguard Worker 
Add(int signal)62*795d594fSAndroid Build Coastguard Worker   void Add(int signal) {
63*795d594fSAndroid Build Coastguard Worker     if (sigaddset(&set_, signal) == -1) {
64*795d594fSAndroid Build Coastguard Worker       PLOG(FATAL) << "sigaddset " << signal << " failed";
65*795d594fSAndroid Build Coastguard Worker     }
66*795d594fSAndroid Build Coastguard Worker   }
67*795d594fSAndroid Build Coastguard Worker 
Block()68*795d594fSAndroid Build Coastguard Worker   void Block() {
69*795d594fSAndroid Build Coastguard Worker     if (pthread_sigmask(SIG_BLOCK, &set_, nullptr) != 0) {
70*795d594fSAndroid Build Coastguard Worker       PLOG(FATAL) << "pthread_sigmask failed";
71*795d594fSAndroid Build Coastguard Worker     }
72*795d594fSAndroid Build Coastguard Worker   }
73*795d594fSAndroid Build Coastguard Worker 
Wait()74*795d594fSAndroid Build Coastguard Worker   int Wait() {
75*795d594fSAndroid Build Coastguard Worker     // Sleep in sigwait() until a signal arrives. gdb causes EINTR failures.
76*795d594fSAndroid Build Coastguard Worker     int signal_number;
77*795d594fSAndroid Build Coastguard Worker     int rc = TEMP_FAILURE_RETRY(sigwait(&set_, &signal_number));
78*795d594fSAndroid Build Coastguard Worker     if (rc != 0) {
79*795d594fSAndroid Build Coastguard Worker       PLOG(FATAL) << "sigwait failed";
80*795d594fSAndroid Build Coastguard Worker     }
81*795d594fSAndroid Build Coastguard Worker     return signal_number;
82*795d594fSAndroid Build Coastguard Worker   }
83*795d594fSAndroid Build Coastguard Worker 
84*795d594fSAndroid Build Coastguard Worker  private:
85*795d594fSAndroid Build Coastguard Worker   sigset_t set_;
86*795d594fSAndroid Build Coastguard Worker };
87*795d594fSAndroid Build Coastguard Worker 
88*795d594fSAndroid Build Coastguard Worker }  // namespace timeout_signal
89*795d594fSAndroid Build Coastguard Worker 
90*795d594fSAndroid Build Coastguard Worker namespace addr2line {
91*795d594fSAndroid Build Coastguard Worker 
92*795d594fSAndroid Build Coastguard Worker constexpr const char* kAddr2linePath =
93*795d594fSAndroid Build Coastguard Worker     "/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-addr2line";
94*795d594fSAndroid Build Coastguard Worker 
FindAddr2line()95*795d594fSAndroid Build Coastguard Worker std::unique_ptr<std::string> FindAddr2line() {
96*795d594fSAndroid Build Coastguard Worker   const char* env_value = getenv("ANDROID_BUILD_TOP");
97*795d594fSAndroid Build Coastguard Worker   if (env_value != nullptr) {
98*795d594fSAndroid Build Coastguard Worker     std::string path = std::string(env_value) + kAddr2linePath;
99*795d594fSAndroid Build Coastguard Worker     if (access(path.c_str(), X_OK) == 0) {
100*795d594fSAndroid Build Coastguard Worker       return std::make_unique<std::string>(path);
101*795d594fSAndroid Build Coastguard Worker     }
102*795d594fSAndroid Build Coastguard Worker   }
103*795d594fSAndroid Build Coastguard Worker 
104*795d594fSAndroid Build Coastguard Worker   {
105*795d594fSAndroid Build Coastguard Worker     std::string path = std::string(".") + kAddr2linePath;
106*795d594fSAndroid Build Coastguard Worker     if (access(path.c_str(), X_OK) == 0) {
107*795d594fSAndroid Build Coastguard Worker       return std::make_unique<std::string>(path);
108*795d594fSAndroid Build Coastguard Worker     }
109*795d594fSAndroid Build Coastguard Worker   }
110*795d594fSAndroid Build Coastguard Worker 
111*795d594fSAndroid Build Coastguard Worker   {
112*795d594fSAndroid Build Coastguard Worker     using android::base::Dirname;
113*795d594fSAndroid Build Coastguard Worker 
114*795d594fSAndroid Build Coastguard Worker     std::string exec_dir = android::base::GetExecutableDirectory();
115*795d594fSAndroid Build Coastguard Worker     std::string derived_top = Dirname(Dirname(Dirname(Dirname(exec_dir))));
116*795d594fSAndroid Build Coastguard Worker     std::string path = derived_top + kAddr2linePath;
117*795d594fSAndroid Build Coastguard Worker     if (access(path.c_str(), X_OK) == 0) {
118*795d594fSAndroid Build Coastguard Worker       return std::make_unique<std::string>(path);
119*795d594fSAndroid Build Coastguard Worker     }
120*795d594fSAndroid Build Coastguard Worker   }
121*795d594fSAndroid Build Coastguard Worker 
122*795d594fSAndroid Build Coastguard Worker   constexpr const char* kHostAddr2line = "/usr/bin/addr2line";
123*795d594fSAndroid Build Coastguard Worker   if (access(kHostAddr2line, F_OK) == 0) {
124*795d594fSAndroid Build Coastguard Worker     return std::make_unique<std::string>(kHostAddr2line);
125*795d594fSAndroid Build Coastguard Worker   }
126*795d594fSAndroid Build Coastguard Worker 
127*795d594fSAndroid Build Coastguard Worker   return nullptr;
128*795d594fSAndroid Build Coastguard Worker }
129*795d594fSAndroid Build Coastguard Worker 
130*795d594fSAndroid Build Coastguard Worker // The state of an open pipe to addr2line. In "server" mode, addr2line takes input on stdin
131*795d594fSAndroid Build Coastguard Worker // and prints the result to stdout. This struct keeps the state of the open connection.
132*795d594fSAndroid Build Coastguard Worker struct Addr2linePipe {
Addr2linePipeart::__anon4a4723c20111::addr2line::Addr2linePipe133*795d594fSAndroid Build Coastguard Worker   Addr2linePipe(int in_fd, int out_fd, const std::string& file_name, pid_t pid)
134*795d594fSAndroid Build Coastguard Worker       : in(in_fd), out(out_fd), file(file_name), child_pid(pid), odd(true) {}
135*795d594fSAndroid Build Coastguard Worker 
~Addr2linePipeart::__anon4a4723c20111::addr2line::Addr2linePipe136*795d594fSAndroid Build Coastguard Worker   ~Addr2linePipe() {
137*795d594fSAndroid Build Coastguard Worker     kill(child_pid, SIGKILL);
138*795d594fSAndroid Build Coastguard Worker   }
139*795d594fSAndroid Build Coastguard Worker 
140*795d594fSAndroid Build Coastguard Worker   unique_fd in;      // The file descriptor that is connected to the output of addr2line.
141*795d594fSAndroid Build Coastguard Worker   unique_fd out;     // The file descriptor that is connected to the input of addr2line.
142*795d594fSAndroid Build Coastguard Worker 
143*795d594fSAndroid Build Coastguard Worker   const std::string file;     // The file addr2line is working on, so that we know when to close
144*795d594fSAndroid Build Coastguard Worker                               // and restart.
145*795d594fSAndroid Build Coastguard Worker   const pid_t child_pid;      // The pid of the child, which we should kill when we're done.
146*795d594fSAndroid Build Coastguard Worker   bool odd;                   // Print state for indentation of lines.
147*795d594fSAndroid Build Coastguard Worker };
148*795d594fSAndroid Build Coastguard Worker 
Connect(const std::string & name,const char * args[])149*795d594fSAndroid Build Coastguard Worker std::unique_ptr<Addr2linePipe> Connect(const std::string& name, const char* args[]) {
150*795d594fSAndroid Build Coastguard Worker   int caller_to_addr2line[2];
151*795d594fSAndroid Build Coastguard Worker   int addr2line_to_caller[2];
152*795d594fSAndroid Build Coastguard Worker 
153*795d594fSAndroid Build Coastguard Worker   if (pipe(caller_to_addr2line) == -1) {
154*795d594fSAndroid Build Coastguard Worker     return nullptr;
155*795d594fSAndroid Build Coastguard Worker   }
156*795d594fSAndroid Build Coastguard Worker   if (pipe(addr2line_to_caller) == -1) {
157*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
158*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
159*795d594fSAndroid Build Coastguard Worker     return nullptr;
160*795d594fSAndroid Build Coastguard Worker   }
161*795d594fSAndroid Build Coastguard Worker 
162*795d594fSAndroid Build Coastguard Worker   pid_t pid = fork();
163*795d594fSAndroid Build Coastguard Worker   if (pid == -1) {
164*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
165*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
166*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[0]);
167*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
168*795d594fSAndroid Build Coastguard Worker     return nullptr;
169*795d594fSAndroid Build Coastguard Worker   }
170*795d594fSAndroid Build Coastguard Worker 
171*795d594fSAndroid Build Coastguard Worker   if (pid == 0) {
172*795d594fSAndroid Build Coastguard Worker     dup2(caller_to_addr2line[0], STDIN_FILENO);
173*795d594fSAndroid Build Coastguard Worker     dup2(addr2line_to_caller[1], STDOUT_FILENO);
174*795d594fSAndroid Build Coastguard Worker 
175*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
176*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[1]);
177*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[0]);
178*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
179*795d594fSAndroid Build Coastguard Worker 
180*795d594fSAndroid Build Coastguard Worker     execv(args[0], const_cast<char* const*>(args));
181*795d594fSAndroid Build Coastguard Worker     exit(1);
182*795d594fSAndroid Build Coastguard Worker   } else {
183*795d594fSAndroid Build Coastguard Worker     close(caller_to_addr2line[0]);
184*795d594fSAndroid Build Coastguard Worker     close(addr2line_to_caller[1]);
185*795d594fSAndroid Build Coastguard Worker     return std::make_unique<Addr2linePipe>(addr2line_to_caller[0],
186*795d594fSAndroid Build Coastguard Worker                                            caller_to_addr2line[1],
187*795d594fSAndroid Build Coastguard Worker                                            name,
188*795d594fSAndroid Build Coastguard Worker                                            pid);
189*795d594fSAndroid Build Coastguard Worker   }
190*795d594fSAndroid Build Coastguard Worker }
191*795d594fSAndroid Build Coastguard Worker 
WritePrefix(std::ostream & os,const char * prefix,bool odd)192*795d594fSAndroid Build Coastguard Worker void WritePrefix(std::ostream& os, const char* prefix, bool odd) {
193*795d594fSAndroid Build Coastguard Worker   if (prefix != nullptr) {
194*795d594fSAndroid Build Coastguard Worker     os << prefix;
195*795d594fSAndroid Build Coastguard Worker   }
196*795d594fSAndroid Build Coastguard Worker   os << "  ";
197*795d594fSAndroid Build Coastguard Worker   if (!odd) {
198*795d594fSAndroid Build Coastguard Worker     os << " ";
199*795d594fSAndroid Build Coastguard Worker   }
200*795d594fSAndroid Build Coastguard Worker }
201*795d594fSAndroid Build Coastguard Worker 
Drain(size_t expected,const char * prefix,std::unique_ptr<Addr2linePipe> * pipe,std::ostream & os)202*795d594fSAndroid Build Coastguard Worker void Drain(size_t expected,
203*795d594fSAndroid Build Coastguard Worker            const char* prefix,
204*795d594fSAndroid Build Coastguard Worker            std::unique_ptr<Addr2linePipe>* pipe /* inout */,
205*795d594fSAndroid Build Coastguard Worker            std::ostream& os) {
206*795d594fSAndroid Build Coastguard Worker   DCHECK(pipe != nullptr);
207*795d594fSAndroid Build Coastguard Worker   DCHECK(pipe->get() != nullptr);
208*795d594fSAndroid Build Coastguard Worker   int in = pipe->get()->in.get();
209*795d594fSAndroid Build Coastguard Worker   DCHECK_GE(in, 0);
210*795d594fSAndroid Build Coastguard Worker 
211*795d594fSAndroid Build Coastguard Worker   bool prefix_written = false;
212*795d594fSAndroid Build Coastguard Worker 
213*795d594fSAndroid Build Coastguard Worker   for (;;) {
214*795d594fSAndroid Build Coastguard Worker     constexpr uint32_t kWaitTimeExpectedMilli = 500;
215*795d594fSAndroid Build Coastguard Worker     constexpr uint32_t kWaitTimeUnexpectedMilli = 50;
216*795d594fSAndroid Build Coastguard Worker 
217*795d594fSAndroid Build Coastguard Worker     int timeout = expected > 0 ? kWaitTimeExpectedMilli : kWaitTimeUnexpectedMilli;
218*795d594fSAndroid Build Coastguard Worker     struct pollfd read_fd{in, POLLIN, 0};
219*795d594fSAndroid Build Coastguard Worker     int retval = TEMP_FAILURE_RETRY(poll(&read_fd, 1, timeout));
220*795d594fSAndroid Build Coastguard Worker     if (retval == -1) {
221*795d594fSAndroid Build Coastguard Worker       // An error occurred.
222*795d594fSAndroid Build Coastguard Worker       pipe->reset();
223*795d594fSAndroid Build Coastguard Worker       return;
224*795d594fSAndroid Build Coastguard Worker     }
225*795d594fSAndroid Build Coastguard Worker 
226*795d594fSAndroid Build Coastguard Worker     if (retval == 0) {
227*795d594fSAndroid Build Coastguard Worker       // Timeout.
228*795d594fSAndroid Build Coastguard Worker       return;
229*795d594fSAndroid Build Coastguard Worker     }
230*795d594fSAndroid Build Coastguard Worker 
231*795d594fSAndroid Build Coastguard Worker     if (!(read_fd.revents & POLLIN)) {
232*795d594fSAndroid Build Coastguard Worker       // addr2line call exited.
233*795d594fSAndroid Build Coastguard Worker       pipe->reset();
234*795d594fSAndroid Build Coastguard Worker       return;
235*795d594fSAndroid Build Coastguard Worker     }
236*795d594fSAndroid Build Coastguard Worker 
237*795d594fSAndroid Build Coastguard Worker     constexpr size_t kMaxBuffer = 128;  // Relatively small buffer. Should be OK as we're on an
238*795d594fSAndroid Build Coastguard Worker     // alt stack, but just to be sure...
239*795d594fSAndroid Build Coastguard Worker     char buffer[kMaxBuffer];
240*795d594fSAndroid Build Coastguard Worker     memset(buffer, 0, kMaxBuffer);
241*795d594fSAndroid Build Coastguard Worker     int bytes_read = TEMP_FAILURE_RETRY(read(in, buffer, kMaxBuffer - 1));
242*795d594fSAndroid Build Coastguard Worker     if (bytes_read <= 0) {
243*795d594fSAndroid Build Coastguard Worker       // This should not really happen...
244*795d594fSAndroid Build Coastguard Worker       pipe->reset();
245*795d594fSAndroid Build Coastguard Worker       return;
246*795d594fSAndroid Build Coastguard Worker     }
247*795d594fSAndroid Build Coastguard Worker     buffer[bytes_read] = '\0';
248*795d594fSAndroid Build Coastguard Worker 
249*795d594fSAndroid Build Coastguard Worker     char* tmp = buffer;
250*795d594fSAndroid Build Coastguard Worker     while (*tmp != 0) {
251*795d594fSAndroid Build Coastguard Worker       if (!prefix_written) {
252*795d594fSAndroid Build Coastguard Worker         WritePrefix(os, prefix, (*pipe)->odd);
253*795d594fSAndroid Build Coastguard Worker         prefix_written = true;
254*795d594fSAndroid Build Coastguard Worker       }
255*795d594fSAndroid Build Coastguard Worker       char* new_line = strchr(tmp, '\n');
256*795d594fSAndroid Build Coastguard Worker       if (new_line == nullptr) {
257*795d594fSAndroid Build Coastguard Worker         os << tmp;
258*795d594fSAndroid Build Coastguard Worker 
259*795d594fSAndroid Build Coastguard Worker         break;
260*795d594fSAndroid Build Coastguard Worker       } else {
261*795d594fSAndroid Build Coastguard Worker         os << std::string(tmp, new_line - tmp + 1);
262*795d594fSAndroid Build Coastguard Worker 
263*795d594fSAndroid Build Coastguard Worker         tmp = new_line + 1;
264*795d594fSAndroid Build Coastguard Worker         prefix_written = false;
265*795d594fSAndroid Build Coastguard Worker         (*pipe)->odd = !(*pipe)->odd;
266*795d594fSAndroid Build Coastguard Worker 
267*795d594fSAndroid Build Coastguard Worker         if (expected > 0) {
268*795d594fSAndroid Build Coastguard Worker           expected--;
269*795d594fSAndroid Build Coastguard Worker         }
270*795d594fSAndroid Build Coastguard Worker       }
271*795d594fSAndroid Build Coastguard Worker     }
272*795d594fSAndroid Build Coastguard Worker   }
273*795d594fSAndroid Build Coastguard Worker }
274*795d594fSAndroid Build Coastguard Worker 
Addr2line(const std::string & addr2line,const std::string & map_src,uintptr_t offset,std::ostream & os,const char * prefix,std::unique_ptr<Addr2linePipe> * pipe)275*795d594fSAndroid Build Coastguard Worker void Addr2line(const std::string& addr2line,
276*795d594fSAndroid Build Coastguard Worker                const std::string& map_src,
277*795d594fSAndroid Build Coastguard Worker                uintptr_t offset,
278*795d594fSAndroid Build Coastguard Worker                std::ostream& os,
279*795d594fSAndroid Build Coastguard Worker                const char* prefix,
280*795d594fSAndroid Build Coastguard Worker                std::unique_ptr<Addr2linePipe>* pipe /* inout */) {
281*795d594fSAndroid Build Coastguard Worker   DCHECK(pipe != nullptr);
282*795d594fSAndroid Build Coastguard Worker 
283*795d594fSAndroid Build Coastguard Worker   if (map_src == "[vdso]" || map_src.ends_with(".vdex")) {
284*795d594fSAndroid Build Coastguard Worker     // addr2line will not work on the vdso.
285*795d594fSAndroid Build Coastguard Worker     // vdex files are special frames injected for the interpreter
286*795d594fSAndroid Build Coastguard Worker     // so they don't have any line number information available.
287*795d594fSAndroid Build Coastguard Worker     return;
288*795d594fSAndroid Build Coastguard Worker   }
289*795d594fSAndroid Build Coastguard Worker 
290*795d594fSAndroid Build Coastguard Worker   if (*pipe == nullptr || (*pipe)->file != map_src) {
291*795d594fSAndroid Build Coastguard Worker     if (*pipe != nullptr) {
292*795d594fSAndroid Build Coastguard Worker       Drain(0, prefix, pipe, os);
293*795d594fSAndroid Build Coastguard Worker     }
294*795d594fSAndroid Build Coastguard Worker     pipe->reset();  // Close early.
295*795d594fSAndroid Build Coastguard Worker 
296*795d594fSAndroid Build Coastguard Worker     const char* args[] = {
297*795d594fSAndroid Build Coastguard Worker         addr2line.c_str(),
298*795d594fSAndroid Build Coastguard Worker         "--functions",
299*795d594fSAndroid Build Coastguard Worker         "--inlines",
300*795d594fSAndroid Build Coastguard Worker         "--demangle",
301*795d594fSAndroid Build Coastguard Worker         "-e",
302*795d594fSAndroid Build Coastguard Worker         map_src.c_str(),
303*795d594fSAndroid Build Coastguard Worker         nullptr
304*795d594fSAndroid Build Coastguard Worker     };
305*795d594fSAndroid Build Coastguard Worker     *pipe = Connect(map_src, args);
306*795d594fSAndroid Build Coastguard Worker   }
307*795d594fSAndroid Build Coastguard Worker 
308*795d594fSAndroid Build Coastguard Worker   Addr2linePipe* pipe_ptr = pipe->get();
309*795d594fSAndroid Build Coastguard Worker   if (pipe_ptr == nullptr) {
310*795d594fSAndroid Build Coastguard Worker     // Failed...
311*795d594fSAndroid Build Coastguard Worker     return;
312*795d594fSAndroid Build Coastguard Worker   }
313*795d594fSAndroid Build Coastguard Worker 
314*795d594fSAndroid Build Coastguard Worker   // Send the offset.
315*795d594fSAndroid Build Coastguard Worker   const std::string hex_offset = StringPrintf("%zx\n", offset);
316*795d594fSAndroid Build Coastguard Worker 
317*795d594fSAndroid Build Coastguard Worker   if (!android::base::WriteFully(pipe_ptr->out.get(), hex_offset.data(), hex_offset.length())) {
318*795d594fSAndroid Build Coastguard Worker     // Error. :-(
319*795d594fSAndroid Build Coastguard Worker     pipe->reset();
320*795d594fSAndroid Build Coastguard Worker     return;
321*795d594fSAndroid Build Coastguard Worker   }
322*795d594fSAndroid Build Coastguard Worker 
323*795d594fSAndroid Build Coastguard Worker   // Now drain (expecting two lines).
324*795d594fSAndroid Build Coastguard Worker   Drain(2U, prefix, pipe, os);
325*795d594fSAndroid Build Coastguard Worker }
326*795d594fSAndroid Build Coastguard Worker 
327*795d594fSAndroid Build Coastguard Worker }  // namespace addr2line
328*795d594fSAndroid Build Coastguard Worker 
329*795d594fSAndroid Build Coastguard Worker namespace ptrace {
330*795d594fSAndroid Build Coastguard Worker 
PtraceSiblings(pid_t pid)331*795d594fSAndroid Build Coastguard Worker std::set<pid_t> PtraceSiblings(pid_t pid) {
332*795d594fSAndroid Build Coastguard Worker   std::set<pid_t> ret;
333*795d594fSAndroid Build Coastguard Worker   std::string task_path = android::base::StringPrintf("/proc/%d/task", pid);
334*795d594fSAndroid Build Coastguard Worker 
335*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<DIR, int (*)(DIR*)> d(opendir(task_path.c_str()), closedir);
336*795d594fSAndroid Build Coastguard Worker 
337*795d594fSAndroid Build Coastguard Worker   // Bail early if the task directory cannot be opened.
338*795d594fSAndroid Build Coastguard Worker   if (d == nullptr) {
339*795d594fSAndroid Build Coastguard Worker     PLOG(ERROR) << "Failed to scan task folder";
340*795d594fSAndroid Build Coastguard Worker     return ret;
341*795d594fSAndroid Build Coastguard Worker   }
342*795d594fSAndroid Build Coastguard Worker 
343*795d594fSAndroid Build Coastguard Worker   struct dirent* de;
344*795d594fSAndroid Build Coastguard Worker   while ((de = readdir(d.get())) != nullptr) {
345*795d594fSAndroid Build Coastguard Worker     // Ignore "." and "..".
346*795d594fSAndroid Build Coastguard Worker     if (!strcmp(de->d_name, ".") || !strcmp(de->d_name, "..")) {
347*795d594fSAndroid Build Coastguard Worker       continue;
348*795d594fSAndroid Build Coastguard Worker     }
349*795d594fSAndroid Build Coastguard Worker 
350*795d594fSAndroid Build Coastguard Worker     char* end;
351*795d594fSAndroid Build Coastguard Worker     pid_t tid = strtoul(de->d_name, &end, 10);
352*795d594fSAndroid Build Coastguard Worker     if (*end) {
353*795d594fSAndroid Build Coastguard Worker       continue;
354*795d594fSAndroid Build Coastguard Worker     }
355*795d594fSAndroid Build Coastguard Worker 
356*795d594fSAndroid Build Coastguard Worker     if (tid == pid) {
357*795d594fSAndroid Build Coastguard Worker       continue;
358*795d594fSAndroid Build Coastguard Worker     }
359*795d594fSAndroid Build Coastguard Worker 
360*795d594fSAndroid Build Coastguard Worker     if (::ptrace(PTRACE_ATTACH, tid, 0, 0) != 0) {
361*795d594fSAndroid Build Coastguard Worker       PLOG(ERROR) << "Failed to attach to tid " << tid;
362*795d594fSAndroid Build Coastguard Worker       continue;
363*795d594fSAndroid Build Coastguard Worker     }
364*795d594fSAndroid Build Coastguard Worker 
365*795d594fSAndroid Build Coastguard Worker     ret.insert(tid);
366*795d594fSAndroid Build Coastguard Worker   }
367*795d594fSAndroid Build Coastguard Worker   return ret;
368*795d594fSAndroid Build Coastguard Worker }
369*795d594fSAndroid Build Coastguard Worker 
DumpABI(pid_t forked_pid)370*795d594fSAndroid Build Coastguard Worker void DumpABI(pid_t forked_pid) {
371*795d594fSAndroid Build Coastguard Worker   enum class ABI { kArm, kArm64, kRiscv64, kX86, kX86_64 };
372*795d594fSAndroid Build Coastguard Worker #if defined(__arm__)
373*795d594fSAndroid Build Coastguard Worker   constexpr ABI kDumperABI = ABI::kArm;
374*795d594fSAndroid Build Coastguard Worker #elif defined(__aarch64__)
375*795d594fSAndroid Build Coastguard Worker   constexpr ABI kDumperABI = ABI::kArm64;
376*795d594fSAndroid Build Coastguard Worker #elif defined(__riscv)
377*795d594fSAndroid Build Coastguard Worker   constexpr ABI kDumperABI = ABI::kRiscv64;
378*795d594fSAndroid Build Coastguard Worker #elif defined(__i386__)
379*795d594fSAndroid Build Coastguard Worker   constexpr ABI kDumperABI = ABI::kX86;
380*795d594fSAndroid Build Coastguard Worker #elif defined(__x86_64__)
381*795d594fSAndroid Build Coastguard Worker   constexpr ABI kDumperABI = ABI::kX86_64;
382*795d594fSAndroid Build Coastguard Worker #else
383*795d594fSAndroid Build Coastguard Worker #error Unsupported architecture
384*795d594fSAndroid Build Coastguard Worker #endif
385*795d594fSAndroid Build Coastguard Worker 
386*795d594fSAndroid Build Coastguard Worker   char data[1024];  // Should be more than enough.
387*795d594fSAndroid Build Coastguard Worker   struct iovec io_vec;
388*795d594fSAndroid Build Coastguard Worker   io_vec.iov_base = &data;
389*795d594fSAndroid Build Coastguard Worker   io_vec.iov_len = 1024;
390*795d594fSAndroid Build Coastguard Worker   ABI to_print;
391*795d594fSAndroid Build Coastguard Worker   if (0 != ::ptrace(PTRACE_GETREGSET, forked_pid, /* NT_PRSTATUS */ 1, &io_vec)) {
392*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << "Could not get registers to determine abi.";
393*795d594fSAndroid Build Coastguard Worker     // Use 64-bit as default.
394*795d594fSAndroid Build Coastguard Worker     switch (kDumperABI) {
395*795d594fSAndroid Build Coastguard Worker       case ABI::kArm:
396*795d594fSAndroid Build Coastguard Worker       case ABI::kArm64:
397*795d594fSAndroid Build Coastguard Worker         to_print = ABI::kArm64;
398*795d594fSAndroid Build Coastguard Worker         break;
399*795d594fSAndroid Build Coastguard Worker       case ABI::kRiscv64:
400*795d594fSAndroid Build Coastguard Worker         to_print = ABI::kRiscv64;
401*795d594fSAndroid Build Coastguard Worker         break;
402*795d594fSAndroid Build Coastguard Worker       case ABI::kX86:
403*795d594fSAndroid Build Coastguard Worker       case ABI::kX86_64:
404*795d594fSAndroid Build Coastguard Worker         to_print = ABI::kX86_64;
405*795d594fSAndroid Build Coastguard Worker         break;
406*795d594fSAndroid Build Coastguard Worker     }
407*795d594fSAndroid Build Coastguard Worker   } else {
408*795d594fSAndroid Build Coastguard Worker     // Check the length of the data. Assume that it's the same arch as the tool.
409*795d594fSAndroid Build Coastguard Worker     switch (kDumperABI) {
410*795d594fSAndroid Build Coastguard Worker       case ABI::kArm:
411*795d594fSAndroid Build Coastguard Worker       case ABI::kArm64:
412*795d594fSAndroid Build Coastguard Worker         to_print = io_vec.iov_len == 18 * sizeof(uint32_t) ? ABI::kArm : ABI::kArm64;
413*795d594fSAndroid Build Coastguard Worker         break;
414*795d594fSAndroid Build Coastguard Worker       case ABI::kRiscv64:
415*795d594fSAndroid Build Coastguard Worker         to_print = ABI::kRiscv64;
416*795d594fSAndroid Build Coastguard Worker         break;
417*795d594fSAndroid Build Coastguard Worker       case ABI::kX86:
418*795d594fSAndroid Build Coastguard Worker       case ABI::kX86_64:
419*795d594fSAndroid Build Coastguard Worker         to_print = io_vec.iov_len == 17 * sizeof(uint32_t) ? ABI::kX86 : ABI::kX86_64;
420*795d594fSAndroid Build Coastguard Worker         break;
421*795d594fSAndroid Build Coastguard Worker     }
422*795d594fSAndroid Build Coastguard Worker   }
423*795d594fSAndroid Build Coastguard Worker   std::string abi_str;
424*795d594fSAndroid Build Coastguard Worker   switch (to_print) {
425*795d594fSAndroid Build Coastguard Worker     case ABI::kArm:
426*795d594fSAndroid Build Coastguard Worker       abi_str = "arm";
427*795d594fSAndroid Build Coastguard Worker       break;
428*795d594fSAndroid Build Coastguard Worker     case ABI::kArm64:
429*795d594fSAndroid Build Coastguard Worker       abi_str = "arm64";
430*795d594fSAndroid Build Coastguard Worker       break;
431*795d594fSAndroid Build Coastguard Worker     case ABI::kRiscv64:
432*795d594fSAndroid Build Coastguard Worker       abi_str = "riscv64";
433*795d594fSAndroid Build Coastguard Worker       break;
434*795d594fSAndroid Build Coastguard Worker     case ABI::kX86:
435*795d594fSAndroid Build Coastguard Worker       abi_str = "x86";
436*795d594fSAndroid Build Coastguard Worker       break;
437*795d594fSAndroid Build Coastguard Worker     case ABI::kX86_64:
438*795d594fSAndroid Build Coastguard Worker       abi_str = "x86_64";
439*795d594fSAndroid Build Coastguard Worker       break;
440*795d594fSAndroid Build Coastguard Worker   }
441*795d594fSAndroid Build Coastguard Worker   LOG(ERROR) << "ABI: '" << abi_str << "'" << std::endl;
442*795d594fSAndroid Build Coastguard Worker }
443*795d594fSAndroid Build Coastguard Worker 
444*795d594fSAndroid Build Coastguard Worker }  // namespace ptrace
445*795d594fSAndroid Build Coastguard Worker 
446*795d594fSAndroid Build Coastguard Worker template <typename T>
WaitLoop(uint32_t max_wait_micros,const T & handler)447*795d594fSAndroid Build Coastguard Worker bool WaitLoop(uint32_t max_wait_micros, const T& handler) {
448*795d594fSAndroid Build Coastguard Worker   constexpr uint32_t kWaitMicros = 10;
449*795d594fSAndroid Build Coastguard Worker   const size_t kMaxLoopCount = max_wait_micros / kWaitMicros;
450*795d594fSAndroid Build Coastguard Worker 
451*795d594fSAndroid Build Coastguard Worker   for (size_t loop_count = 1; loop_count <= kMaxLoopCount; ++loop_count) {
452*795d594fSAndroid Build Coastguard Worker     bool ret;
453*795d594fSAndroid Build Coastguard Worker     if (handler(&ret)) {
454*795d594fSAndroid Build Coastguard Worker       return ret;
455*795d594fSAndroid Build Coastguard Worker     }
456*795d594fSAndroid Build Coastguard Worker     usleep(kWaitMicros);
457*795d594fSAndroid Build Coastguard Worker   }
458*795d594fSAndroid Build Coastguard Worker   return false;
459*795d594fSAndroid Build Coastguard Worker }
460*795d594fSAndroid Build Coastguard Worker 
WaitForMainSigStop(const std::atomic<bool> & saw_wif_stopped_for_main)461*795d594fSAndroid Build Coastguard Worker bool WaitForMainSigStop(const std::atomic<bool>& saw_wif_stopped_for_main) {
462*795d594fSAndroid Build Coastguard Worker   auto handler = [&](bool* res) {
463*795d594fSAndroid Build Coastguard Worker     if (saw_wif_stopped_for_main) {
464*795d594fSAndroid Build Coastguard Worker       *res = true;
465*795d594fSAndroid Build Coastguard Worker       return true;
466*795d594fSAndroid Build Coastguard Worker     }
467*795d594fSAndroid Build Coastguard Worker     return false;
468*795d594fSAndroid Build Coastguard Worker   };
469*795d594fSAndroid Build Coastguard Worker   constexpr uint32_t kMaxWaitMicros = 30 * 1000 * 1000;  // 30s wait.
470*795d594fSAndroid Build Coastguard Worker   return WaitLoop(kMaxWaitMicros, handler);
471*795d594fSAndroid Build Coastguard Worker }
472*795d594fSAndroid Build Coastguard Worker 
WaitForSigStopped(pid_t pid,uint32_t max_wait_micros)473*795d594fSAndroid Build Coastguard Worker bool WaitForSigStopped(pid_t pid, uint32_t max_wait_micros) {
474*795d594fSAndroid Build Coastguard Worker   auto handler = [&](bool* res) {
475*795d594fSAndroid Build Coastguard Worker     int status;
476*795d594fSAndroid Build Coastguard Worker     pid_t rc = TEMP_FAILURE_RETRY(waitpid(pid, &status, WNOHANG));
477*795d594fSAndroid Build Coastguard Worker     if (rc == -1) {
478*795d594fSAndroid Build Coastguard Worker       PLOG(ERROR) << "Failed to waitpid for " << pid;
479*795d594fSAndroid Build Coastguard Worker       *res = false;
480*795d594fSAndroid Build Coastguard Worker       return true;
481*795d594fSAndroid Build Coastguard Worker     }
482*795d594fSAndroid Build Coastguard Worker     if (rc == pid) {
483*795d594fSAndroid Build Coastguard Worker       if (!(WIFSTOPPED(status))) {
484*795d594fSAndroid Build Coastguard Worker         LOG(ERROR) << "Did not get expected stopped signal for " << pid;
485*795d594fSAndroid Build Coastguard Worker         *res = false;
486*795d594fSAndroid Build Coastguard Worker       } else {
487*795d594fSAndroid Build Coastguard Worker         *res = true;
488*795d594fSAndroid Build Coastguard Worker       }
489*795d594fSAndroid Build Coastguard Worker       return true;
490*795d594fSAndroid Build Coastguard Worker     }
491*795d594fSAndroid Build Coastguard Worker     return false;
492*795d594fSAndroid Build Coastguard Worker   };
493*795d594fSAndroid Build Coastguard Worker   return WaitLoop(max_wait_micros, handler);
494*795d594fSAndroid Build Coastguard Worker }
495*795d594fSAndroid Build Coastguard Worker 
496*795d594fSAndroid Build Coastguard Worker #ifdef __LP64__
497*795d594fSAndroid Build Coastguard Worker constexpr bool kIs64Bit = true;
498*795d594fSAndroid Build Coastguard Worker #else
499*795d594fSAndroid Build Coastguard Worker constexpr bool kIs64Bit = false;
500*795d594fSAndroid Build Coastguard Worker #endif
501*795d594fSAndroid Build Coastguard Worker 
DumpThread(unwindstack::AndroidRemoteUnwinder & unwinder,pid_t pid,pid_t tid,const std::string * addr2line_path,const char * prefix)502*795d594fSAndroid Build Coastguard Worker void DumpThread(unwindstack::AndroidRemoteUnwinder& unwinder, pid_t pid,
503*795d594fSAndroid Build Coastguard Worker                 pid_t tid,
504*795d594fSAndroid Build Coastguard Worker                 const std::string* addr2line_path,
505*795d594fSAndroid Build Coastguard Worker                 const char* prefix) {
506*795d594fSAndroid Build Coastguard Worker   LOG(ERROR) << std::endl << "=== pid: " << pid << " tid: " << tid << " ===" << std::endl;
507*795d594fSAndroid Build Coastguard Worker 
508*795d594fSAndroid Build Coastguard Worker   constexpr uint32_t kMaxWaitMicros = 1000 * 1000;  // 1s.
509*795d594fSAndroid Build Coastguard Worker   if (pid != tid && !WaitForSigStopped(tid, kMaxWaitMicros)) {
510*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << "Failed to wait for sigstop on " << tid;
511*795d594fSAndroid Build Coastguard Worker   }
512*795d594fSAndroid Build Coastguard Worker 
513*795d594fSAndroid Build Coastguard Worker   unwindstack::AndroidUnwinderData data;
514*795d594fSAndroid Build Coastguard Worker   if (!unwinder.Unwind(tid, data)) {
515*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << prefix << "(Unwind failed for thread " << tid << ": "
516*795d594fSAndroid Build Coastguard Worker                <<  data.GetErrorString() << ")";
517*795d594fSAndroid Build Coastguard Worker     return;
518*795d594fSAndroid Build Coastguard Worker   }
519*795d594fSAndroid Build Coastguard Worker 
520*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<addr2line::Addr2linePipe> addr2line_state;
521*795d594fSAndroid Build Coastguard Worker   data.DemangleFunctionNames();
522*795d594fSAndroid Build Coastguard Worker   for (const unwindstack::FrameData& frame : data.frames) {
523*795d594fSAndroid Build Coastguard Worker     std::ostringstream oss;
524*795d594fSAndroid Build Coastguard Worker     oss << prefix << StringPrintf("#%02zu pc ", frame.num);
525*795d594fSAndroid Build Coastguard Worker     bool try_addr2line = false;
526*795d594fSAndroid Build Coastguard Worker     if (frame.map_info == nullptr) {
527*795d594fSAndroid Build Coastguard Worker       oss << StringPrintf(kIs64Bit ? "%016" PRIx64 "  ???" : "%08" PRIx64 "  ???", frame.pc);
528*795d594fSAndroid Build Coastguard Worker     } else {
529*795d594fSAndroid Build Coastguard Worker       oss << StringPrintf(kIs64Bit ? "%016" PRIx64 "  " : "%08" PRIx64 "  ", frame.rel_pc);
530*795d594fSAndroid Build Coastguard Worker       if (frame.map_info->name().empty()) {
531*795d594fSAndroid Build Coastguard Worker         oss << StringPrintf("<anonymous:%" PRIx64 ">", frame.map_info->start());
532*795d594fSAndroid Build Coastguard Worker       } else {
533*795d594fSAndroid Build Coastguard Worker         oss << frame.map_info->name().c_str();
534*795d594fSAndroid Build Coastguard Worker       }
535*795d594fSAndroid Build Coastguard Worker       if (frame.map_info->offset() != 0) {
536*795d594fSAndroid Build Coastguard Worker         oss << StringPrintf(" (offset %" PRIx64 ")", frame.map_info->offset());
537*795d594fSAndroid Build Coastguard Worker       }
538*795d594fSAndroid Build Coastguard Worker       oss << " (";
539*795d594fSAndroid Build Coastguard Worker       const std::string& function_name = frame.function_name;
540*795d594fSAndroid Build Coastguard Worker       if (!function_name.empty()) {
541*795d594fSAndroid Build Coastguard Worker         oss << function_name;
542*795d594fSAndroid Build Coastguard Worker         if (frame.function_offset != 0) {
543*795d594fSAndroid Build Coastguard Worker           oss << "+" << frame.function_offset;
544*795d594fSAndroid Build Coastguard Worker         }
545*795d594fSAndroid Build Coastguard Worker         // Functions found using the gdb jit interface will be in an empty
546*795d594fSAndroid Build Coastguard Worker         // map that cannot be found using addr2line.
547*795d594fSAndroid Build Coastguard Worker         if (!frame.map_info->name().empty()) {
548*795d594fSAndroid Build Coastguard Worker           try_addr2line = true;
549*795d594fSAndroid Build Coastguard Worker         }
550*795d594fSAndroid Build Coastguard Worker       } else {
551*795d594fSAndroid Build Coastguard Worker         oss << "???";
552*795d594fSAndroid Build Coastguard Worker       }
553*795d594fSAndroid Build Coastguard Worker       oss << ")";
554*795d594fSAndroid Build Coastguard Worker     }
555*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << oss.str() << std::endl;
556*795d594fSAndroid Build Coastguard Worker     if (try_addr2line && addr2line_path != nullptr) {
557*795d594fSAndroid Build Coastguard Worker       addr2line::Addr2line(*addr2line_path,
558*795d594fSAndroid Build Coastguard Worker                            frame.map_info->name(),
559*795d594fSAndroid Build Coastguard Worker                            frame.rel_pc,
560*795d594fSAndroid Build Coastguard Worker                            LOG_STREAM(ERROR),
561*795d594fSAndroid Build Coastguard Worker                            prefix,
562*795d594fSAndroid Build Coastguard Worker                            &addr2line_state);
563*795d594fSAndroid Build Coastguard Worker     }
564*795d594fSAndroid Build Coastguard Worker   }
565*795d594fSAndroid Build Coastguard Worker 
566*795d594fSAndroid Build Coastguard Worker   if (addr2line_state != nullptr) {
567*795d594fSAndroid Build Coastguard Worker     addr2line::Drain(0, prefix, &addr2line_state, LOG_STREAM(ERROR));
568*795d594fSAndroid Build Coastguard Worker   }
569*795d594fSAndroid Build Coastguard Worker }
570*795d594fSAndroid Build Coastguard Worker 
DumpProcess(pid_t forked_pid,const std::atomic<bool> & saw_wif_stopped_for_main)571*795d594fSAndroid Build Coastguard Worker void DumpProcess(pid_t forked_pid, const std::atomic<bool>& saw_wif_stopped_for_main) {
572*795d594fSAndroid Build Coastguard Worker   LOG(ERROR) << "Timeout for process " << forked_pid;
573*795d594fSAndroid Build Coastguard Worker 
574*795d594fSAndroid Build Coastguard Worker   CHECK_EQ(0, ::ptrace(PTRACE_ATTACH, forked_pid, 0, 0));
575*795d594fSAndroid Build Coastguard Worker   std::set<pid_t> tids = ptrace::PtraceSiblings(forked_pid);
576*795d594fSAndroid Build Coastguard Worker   tids.insert(forked_pid);
577*795d594fSAndroid Build Coastguard Worker 
578*795d594fSAndroid Build Coastguard Worker   ptrace::DumpABI(forked_pid);
579*795d594fSAndroid Build Coastguard Worker 
580*795d594fSAndroid Build Coastguard Worker   // Check whether we have and should use addr2line.
581*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<std::string> addr2line_path;
582*795d594fSAndroid Build Coastguard Worker   if (kUseAddr2line) {
583*795d594fSAndroid Build Coastguard Worker     addr2line_path = addr2line::FindAddr2line();
584*795d594fSAndroid Build Coastguard Worker     if (addr2line_path == nullptr) {
585*795d594fSAndroid Build Coastguard Worker       LOG(ERROR) << "Did not find usable addr2line";
586*795d594fSAndroid Build Coastguard Worker     }
587*795d594fSAndroid Build Coastguard Worker   }
588*795d594fSAndroid Build Coastguard Worker 
589*795d594fSAndroid Build Coastguard Worker   if (!WaitForMainSigStop(saw_wif_stopped_for_main)) {
590*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << "Did not receive SIGSTOP for pid " << forked_pid;
591*795d594fSAndroid Build Coastguard Worker   }
592*795d594fSAndroid Build Coastguard Worker 
593*795d594fSAndroid Build Coastguard Worker   unwindstack::AndroidRemoteUnwinder unwinder(forked_pid);
594*795d594fSAndroid Build Coastguard Worker   for (pid_t tid : tids) {
595*795d594fSAndroid Build Coastguard Worker     DumpThread(unwinder, forked_pid, tid, addr2line_path.get(), "  ");
596*795d594fSAndroid Build Coastguard Worker   }
597*795d594fSAndroid Build Coastguard Worker }
598*795d594fSAndroid Build Coastguard Worker 
599*795d594fSAndroid Build Coastguard Worker [[noreturn]]
WaitMainLoop(pid_t forked_pid,std::atomic<bool> * saw_wif_stopped_for_main)600*795d594fSAndroid Build Coastguard Worker void WaitMainLoop(pid_t forked_pid, std::atomic<bool>* saw_wif_stopped_for_main) {
601*795d594fSAndroid Build Coastguard Worker   for (;;) {
602*795d594fSAndroid Build Coastguard Worker     // Consider switching to waitid to not get woken up for WIFSTOPPED.
603*795d594fSAndroid Build Coastguard Worker     int status;
604*795d594fSAndroid Build Coastguard Worker     pid_t res = TEMP_FAILURE_RETRY(waitpid(forked_pid, &status, 0));
605*795d594fSAndroid Build Coastguard Worker     if (res == -1) {
606*795d594fSAndroid Build Coastguard Worker       PLOG(FATAL) << "Failure during waitpid";
607*795d594fSAndroid Build Coastguard Worker       __builtin_unreachable();
608*795d594fSAndroid Build Coastguard Worker     }
609*795d594fSAndroid Build Coastguard Worker 
610*795d594fSAndroid Build Coastguard Worker     if (WIFEXITED(status)) {
611*795d594fSAndroid Build Coastguard Worker       _exit(WEXITSTATUS(status));
612*795d594fSAndroid Build Coastguard Worker       __builtin_unreachable();
613*795d594fSAndroid Build Coastguard Worker     }
614*795d594fSAndroid Build Coastguard Worker     if (WIFSIGNALED(status)) {
615*795d594fSAndroid Build Coastguard Worker       _exit(1);
616*795d594fSAndroid Build Coastguard Worker       __builtin_unreachable();
617*795d594fSAndroid Build Coastguard Worker     }
618*795d594fSAndroid Build Coastguard Worker     if (WIFSTOPPED(status)) {
619*795d594fSAndroid Build Coastguard Worker       *saw_wif_stopped_for_main = true;
620*795d594fSAndroid Build Coastguard Worker       continue;
621*795d594fSAndroid Build Coastguard Worker     }
622*795d594fSAndroid Build Coastguard Worker     if (WIFCONTINUED(status)) {
623*795d594fSAndroid Build Coastguard Worker       continue;
624*795d594fSAndroid Build Coastguard Worker     }
625*795d594fSAndroid Build Coastguard Worker 
626*795d594fSAndroid Build Coastguard Worker     LOG(FATAL) << "Unknown status " << std::hex << status;
627*795d594fSAndroid Build Coastguard Worker   }
628*795d594fSAndroid Build Coastguard Worker }
629*795d594fSAndroid Build Coastguard Worker 
630*795d594fSAndroid Build Coastguard Worker [[noreturn]]
SetupAndWait(pid_t forked_pid,int signal,int timeout_exit_code)631*795d594fSAndroid Build Coastguard Worker void SetupAndWait(pid_t forked_pid, int signal, int timeout_exit_code) {
632*795d594fSAndroid Build Coastguard Worker   timeout_signal::SignalSet signals;
633*795d594fSAndroid Build Coastguard Worker   signals.Add(signal);
634*795d594fSAndroid Build Coastguard Worker   signals.Block();
635*795d594fSAndroid Build Coastguard Worker 
636*795d594fSAndroid Build Coastguard Worker   std::atomic<bool> saw_wif_stopped_for_main(false);
637*795d594fSAndroid Build Coastguard Worker 
638*795d594fSAndroid Build Coastguard Worker   std::thread signal_catcher([&]() {
639*795d594fSAndroid Build Coastguard Worker     signals.Block();
640*795d594fSAndroid Build Coastguard Worker     int sig = signals.Wait();
641*795d594fSAndroid Build Coastguard Worker     CHECK_EQ(sig, signal);
642*795d594fSAndroid Build Coastguard Worker 
643*795d594fSAndroid Build Coastguard Worker     DumpProcess(forked_pid, saw_wif_stopped_for_main);
644*795d594fSAndroid Build Coastguard Worker 
645*795d594fSAndroid Build Coastguard Worker     // Don't clean up. Just kill the child and exit.
646*795d594fSAndroid Build Coastguard Worker     kill(forked_pid, SIGKILL);
647*795d594fSAndroid Build Coastguard Worker     _exit(timeout_exit_code);
648*795d594fSAndroid Build Coastguard Worker   });
649*795d594fSAndroid Build Coastguard Worker 
650*795d594fSAndroid Build Coastguard Worker   WaitMainLoop(forked_pid, &saw_wif_stopped_for_main);
651*795d594fSAndroid Build Coastguard Worker }
652*795d594fSAndroid Build Coastguard Worker 
653*795d594fSAndroid Build Coastguard Worker }  // namespace
654*795d594fSAndroid Build Coastguard Worker }  // namespace art
655*795d594fSAndroid Build Coastguard Worker 
main(int argc,char ** argv)656*795d594fSAndroid Build Coastguard Worker int main([[maybe_unused]] int argc, char** argv) {
657*795d594fSAndroid Build Coastguard Worker   android::base::InitLogging(argv);
658*795d594fSAndroid Build Coastguard Worker 
659*795d594fSAndroid Build Coastguard Worker   int signal = SIGRTMIN + 2;
660*795d594fSAndroid Build Coastguard Worker   int timeout_exit_code = 1;
661*795d594fSAndroid Build Coastguard Worker 
662*795d594fSAndroid Build Coastguard Worker   size_t index = 1u;
663*795d594fSAndroid Build Coastguard Worker   CHECK(argv[index] != nullptr);
664*795d594fSAndroid Build Coastguard Worker 
665*795d594fSAndroid Build Coastguard Worker   bool to_logcat = false;
666*795d594fSAndroid Build Coastguard Worker #ifdef __ANDROID__
667*795d594fSAndroid Build Coastguard Worker   if (strcmp(argv[index], "-l") == 0) {
668*795d594fSAndroid Build Coastguard Worker     index++;
669*795d594fSAndroid Build Coastguard Worker     CHECK(argv[index] != nullptr);
670*795d594fSAndroid Build Coastguard Worker     to_logcat = true;
671*795d594fSAndroid Build Coastguard Worker   }
672*795d594fSAndroid Build Coastguard Worker #endif
673*795d594fSAndroid Build Coastguard Worker   if (!to_logcat) {
674*795d594fSAndroid Build Coastguard Worker     android::base::SetLogger(android::base::StderrLogger);
675*795d594fSAndroid Build Coastguard Worker   }
676*795d594fSAndroid Build Coastguard Worker 
677*795d594fSAndroid Build Coastguard Worker   if (strcmp(argv[index], "-s") == 0) {
678*795d594fSAndroid Build Coastguard Worker     index++;
679*795d594fSAndroid Build Coastguard Worker     CHECK(argv[index] != nullptr);
680*795d594fSAndroid Build Coastguard Worker     uint32_t signal_uint;
681*795d594fSAndroid Build Coastguard Worker     CHECK(android::base::ParseUint(argv[index], &signal_uint)) << "Signal not a number.";
682*795d594fSAndroid Build Coastguard Worker     signal = signal_uint;
683*795d594fSAndroid Build Coastguard Worker     index++;
684*795d594fSAndroid Build Coastguard Worker     CHECK(argv[index] != nullptr);
685*795d594fSAndroid Build Coastguard Worker   }
686*795d594fSAndroid Build Coastguard Worker 
687*795d594fSAndroid Build Coastguard Worker   if (strcmp(argv[index], "-e") == 0) {
688*795d594fSAndroid Build Coastguard Worker     index++;
689*795d594fSAndroid Build Coastguard Worker     CHECK(argv[index] != nullptr);
690*795d594fSAndroid Build Coastguard Worker     uint32_t timeout_exit_code_uint;
691*795d594fSAndroid Build Coastguard Worker     CHECK(android::base::ParseUint(argv[index], &timeout_exit_code_uint))
692*795d594fSAndroid Build Coastguard Worker         << "Exit code not a number.";
693*795d594fSAndroid Build Coastguard Worker     timeout_exit_code = timeout_exit_code_uint;
694*795d594fSAndroid Build Coastguard Worker     index++;
695*795d594fSAndroid Build Coastguard Worker     CHECK(argv[index] != nullptr);
696*795d594fSAndroid Build Coastguard Worker   }
697*795d594fSAndroid Build Coastguard Worker 
698*795d594fSAndroid Build Coastguard Worker   pid_t orig_ppid = getpid();
699*795d594fSAndroid Build Coastguard Worker 
700*795d594fSAndroid Build Coastguard Worker   pid_t pid = fork();
701*795d594fSAndroid Build Coastguard Worker   if (pid == 0) {
702*795d594fSAndroid Build Coastguard Worker     if (prctl(PR_SET_PDEATHSIG, SIGTERM) == -1) {
703*795d594fSAndroid Build Coastguard Worker       _exit(1);
704*795d594fSAndroid Build Coastguard Worker     }
705*795d594fSAndroid Build Coastguard Worker 
706*795d594fSAndroid Build Coastguard Worker     if (getppid() != orig_ppid) {
707*795d594fSAndroid Build Coastguard Worker       _exit(2);
708*795d594fSAndroid Build Coastguard Worker     }
709*795d594fSAndroid Build Coastguard Worker 
710*795d594fSAndroid Build Coastguard Worker     execvp(argv[index], &argv[index]);
711*795d594fSAndroid Build Coastguard Worker 
712*795d594fSAndroid Build Coastguard Worker     _exit(3);
713*795d594fSAndroid Build Coastguard Worker     __builtin_unreachable();
714*795d594fSAndroid Build Coastguard Worker   }
715*795d594fSAndroid Build Coastguard Worker 
716*795d594fSAndroid Build Coastguard Worker   art::SetupAndWait(pid, signal, timeout_exit_code);
717*795d594fSAndroid Build Coastguard Worker   __builtin_unreachable();
718*795d594fSAndroid Build Coastguard Worker }
719