blob: bda912ecebc989198dc37897c85a0ac102c1264b [file] [log] [blame]
Andreas Gampe5dd44d02016-08-02 17:20:03 -07001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "native_stack_dump.h"
18
Yi Kongc57c6802018-10-29 14:28:56 -070019#include <memory>
Andreas Gampe5dd44d02016-08-02 17:20:03 -070020#include <ostream>
David Srbeckyd4ffc912022-09-26 18:11:38 +010021#include <string_view>
Andreas Gampe5dd44d02016-08-02 17:20:03 -070022
23#include <stdio.h>
24
25#include "art_method.h"
26
27// For DumpNativeStack.
Christopher Ferrisf44070e2022-03-26 09:54:32 -070028#include <unwindstack/AndroidUnwinder.h>
Andreas Gampe5dd44d02016-08-02 17:20:03 -070029
30#if defined(__linux__)
31
Andreas Gampe5dd44d02016-08-02 17:20:03 -070032#include <vector>
33
34#include <linux/unistd.h>
Christopher Ferris453e0e52018-03-06 14:02:55 -080035#include <poll.h>
Andreas Gampe5dd44d02016-08-02 17:20:03 -070036#include <signal.h>
37#include <stdlib.h>
38#include <sys/time.h>
39#include <sys/types.h>
40
Andreas Gampe43e72432019-05-14 16:15:24 -070041#include "android-base/file.h"
Andreas Gampe46ee31b2016-12-14 10:11:49 -080042#include "android-base/stringprintf.h"
Christopher Ferrisb1f23f92018-03-07 14:10:49 -080043#include "android-base/strings.h"
Andreas Gampe46ee31b2016-12-14 10:11:49 -080044
Andreas Gampe5dd44d02016-08-02 17:20:03 -070045#include "arch/instruction_set.h"
Andreas Gampe39b378c2017-12-07 15:44:13 -080046#include "base/aborting.h"
David Srbeckyeea5fd32019-02-13 17:24:17 +000047#include "base/bit_utils.h"
David Sehr891a50e2017-10-27 17:01:07 -070048#include "base/file_utils.h"
Andreas Gampe5dd44d02016-08-02 17:20:03 -070049#include "base/memory_tool.h"
50#include "base/mutex.h"
David Sehrc431b9d2018-03-02 12:01:51 -080051#include "base/os.h"
Andreas Gampefcccbaf2016-08-02 17:20:03 -070052#include "base/unix_file/fd_file.h"
David Sehrc431b9d2018-03-02 12:01:51 -080053#include "base/utils.h"
David Srbeckybb720732019-01-29 16:54:47 +000054#include "class_linker.h"
David Srbeckyeea5fd32019-02-13 17:24:17 +000055#include "entrypoints/runtime_asm_entrypoints.h"
Andreas Gampe5dd44d02016-08-02 17:20:03 -070056#include "oat_quick_method_header.h"
David Srbeckybb720732019-01-29 16:54:47 +000057#include "runtime.h"
Andreas Gampeb486a982017-06-01 13:45:54 -070058#include "thread-current-inl.h"
Andreas Gampe5dd44d02016-08-02 17:20:03 -070059
60#endif
61
62namespace art {
63
64#if defined(__linux__)
65
Andreas Gampe46ee31b2016-12-14 10:11:49 -080066using android::base::StringPrintf;
67
Andreas Gampe5dd44d02016-08-02 17:20:03 -070068static constexpr bool kUseAddr2line = !kIsTargetBuild;
69
David Srbecky87da30e2019-01-30 15:51:23 +000070std::string FindAddr2line() {
David Srbeckya974baa2022-06-08 14:58:39 +010071#if !defined(ART_TARGET) && !defined(ART_CLANG_PATH)
David Srbeckyc4be8812022-05-03 17:25:21 +010072 #error "ART_CLANG_PATH must be defined on host build"
73#endif
74#if defined(ART_CLANG_PATH)
David Srbecky7400a542020-07-09 13:40:57 +010075 const char* env_value = getenv("ANDROID_BUILD_TOP");
David Srbeckyc2042a12023-02-23 17:09:32 +000076 std::string_view top(env_value != nullptr ? env_value : ".");
77 return std::string(top) + "/" + ART_CLANG_PATH + "/bin/llvm-addr2line";
78#else
David Srbeckyc4be8812022-05-03 17:25:21 +010079 return std::string("llvm-addr2line");
David Srbeckyc2042a12023-02-23 17:09:32 +000080#endif
David Srbecky87da30e2019-01-30 15:51:23 +000081}
82
Andreas Gampe5dd44d02016-08-02 17:20:03 -070083ALWAYS_INLINE
Andreas Gampefcccbaf2016-08-02 17:20:03 -070084static inline void WritePrefix(std::ostream& os, const char* prefix, bool odd) {
Andreas Gampe5dd44d02016-08-02 17:20:03 -070085 if (prefix != nullptr) {
Andreas Gampefcccbaf2016-08-02 17:20:03 -070086 os << prefix;
Andreas Gampe5dd44d02016-08-02 17:20:03 -070087 }
Andreas Gampefcccbaf2016-08-02 17:20:03 -070088 os << " ";
Andreas Gampe5dd44d02016-08-02 17:20:03 -070089 if (!odd) {
Andreas Gampefcccbaf2016-08-02 17:20:03 -070090 os << " ";
Andreas Gampe5dd44d02016-08-02 17:20:03 -070091 }
92}
93
Andreas Gampefcccbaf2016-08-02 17:20:03 -070094// The state of an open pipe to addr2line. In "server" mode, addr2line takes input on stdin
95// and prints the result to stdout. This struct keeps the state of the open connection.
96struct Addr2linePipe {
97 Addr2linePipe(int in_fd, int out_fd, const std::string& file_name, pid_t pid)
98 : in(in_fd, false), out(out_fd, false), file(file_name), child_pid(pid), odd(true) {}
99
100 ~Addr2linePipe() {
101 kill(child_pid, SIGKILL);
102 }
103
104 File in; // The file descriptor that is connected to the output of addr2line.
105 File out; // The file descriptor that is connected to the input of addr2line.
106
107 const std::string file; // The file addr2line is working on, so that we know when to close
108 // and restart.
109 const pid_t child_pid; // The pid of the child, which we should kill when we're done.
110 bool odd; // Print state for indentation of lines.
111};
112
113static std::unique_ptr<Addr2linePipe> Connect(const std::string& name, const char* args[]) {
114 int caller_to_addr2line[2];
115 int addr2line_to_caller[2];
116
117 if (pipe(caller_to_addr2line) == -1) {
118 return nullptr;
119 }
120 if (pipe(addr2line_to_caller) == -1) {
121 close(caller_to_addr2line[0]);
122 close(caller_to_addr2line[1]);
123 return nullptr;
124 }
125
126 pid_t pid = fork();
127 if (pid == -1) {
128 close(caller_to_addr2line[0]);
129 close(caller_to_addr2line[1]);
Calin Juravle0ed6c802017-03-27 18:12:05 -0700130 close(addr2line_to_caller[0]);
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700131 close(addr2line_to_caller[1]);
132 return nullptr;
133 }
134
135 if (pid == 0) {
136 dup2(caller_to_addr2line[0], STDIN_FILENO);
137 dup2(addr2line_to_caller[1], STDOUT_FILENO);
138
139 close(caller_to_addr2line[0]);
140 close(caller_to_addr2line[1]);
141 close(addr2line_to_caller[0]);
142 close(addr2line_to_caller[1]);
143
144 execv(args[0], const_cast<char* const*>(args));
145 exit(1);
146 } else {
147 close(caller_to_addr2line[0]);
148 close(addr2line_to_caller[1]);
Yi Kongc57c6802018-10-29 14:28:56 -0700149 return std::make_unique<Addr2linePipe>(addr2line_to_caller[0],
150 caller_to_addr2line[1],
151 name,
152 pid);
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700153 }
154}
155
156static void Drain(size_t expected,
157 const char* prefix,
158 std::unique_ptr<Addr2linePipe>* pipe /* inout */,
159 std::ostream& os) {
160 DCHECK(pipe != nullptr);
161 DCHECK(pipe->get() != nullptr);
162 int in = pipe->get()->in.Fd();
163 DCHECK_GE(in, 0);
164
165 bool prefix_written = false;
166
167 for (;;) {
Christopher Ferris453e0e52018-03-06 14:02:55 -0800168 constexpr uint32_t kWaitTimeExpectedMilli = 500;
169 constexpr uint32_t kWaitTimeUnexpectedMilli = 50;
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700170
Christopher Ferris453e0e52018-03-06 14:02:55 -0800171 int timeout = expected > 0 ? kWaitTimeExpectedMilli : kWaitTimeUnexpectedMilli;
172 struct pollfd read_fd{in, POLLIN, 0};
173 int retval = TEMP_FAILURE_RETRY(poll(&read_fd, 1, timeout));
174 if (retval == -1) {
175 // An error occurred.
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700176 pipe->reset();
177 return;
178 }
179
180 if (retval == 0) {
181 // Timeout.
182 return;
183 }
184
Christopher Ferris453e0e52018-03-06 14:02:55 -0800185 if (!(read_fd.revents & POLLIN)) {
186 // addr2line call exited.
187 pipe->reset();
188 return;
189 }
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700190
191 constexpr size_t kMaxBuffer = 128; // Relatively small buffer. Should be OK as we're on an
192 // alt stack, but just to be sure...
193 char buffer[kMaxBuffer];
194 memset(buffer, 0, kMaxBuffer);
195 int bytes_read = TEMP_FAILURE_RETRY(read(in, buffer, kMaxBuffer - 1));
Christopher Ferris453e0e52018-03-06 14:02:55 -0800196 if (bytes_read <= 0) {
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700197 // This should not really happen...
198 pipe->reset();
199 return;
200 }
Christopher Ferris453e0e52018-03-06 14:02:55 -0800201 buffer[bytes_read] = '\0';
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700202
203 char* tmp = buffer;
204 while (*tmp != 0) {
205 if (!prefix_written) {
206 WritePrefix(os, prefix, (*pipe)->odd);
207 prefix_written = true;
208 }
209 char* new_line = strchr(tmp, '\n');
210 if (new_line == nullptr) {
211 os << tmp;
212
213 break;
214 } else {
215 char saved = *(new_line + 1);
216 *(new_line + 1) = 0;
217 os << tmp;
218 *(new_line + 1) = saved;
219
220 tmp = new_line + 1;
221 prefix_written = false;
222 (*pipe)->odd = !(*pipe)->odd;
223
224 if (expected > 0) {
225 expected--;
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700226 }
227 }
228 }
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700229 }
230}
231
232static void Addr2line(const std::string& map_src,
233 uintptr_t offset,
234 std::ostream& os,
235 const char* prefix,
236 std::unique_ptr<Addr2linePipe>* pipe /* inout */) {
Orion Hodson3e9d7ae2019-11-18 15:21:51 +0000237 std::array<const char*, 3> kIgnoreSuffixes{ ".dex", ".jar", ".vdex" };
238 for (const char* ignore_suffix : kIgnoreSuffixes) {
239 if (android::base::EndsWith(map_src, ignore_suffix)) {
240 // Ignore file names that do not have map information addr2line can consume. e.g. vdex
241 // files are special frames injected for the interpreter so they don't have any line
242 // number information available.
243 return;
244 }
245 }
246 if (map_src == "[vdso]") {
Christopher Ferrisb1f23f92018-03-07 14:10:49 -0800247 // addr2line will not work on the vdso.
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700248 return;
249 }
250
251 if (*pipe == nullptr || (*pipe)->file != map_src) {
252 if (*pipe != nullptr) {
253 Drain(0, prefix, pipe, os);
254 }
255 pipe->reset(); // Close early.
256
David Srbecky87da30e2019-01-30 15:51:23 +0000257 std::string addr2linePath = FindAddr2line();
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700258 const char* args[7] = {
David Srbecky87da30e2019-01-30 15:51:23 +0000259 addr2linePath.c_str(),
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700260 "--functions",
261 "--inlines",
262 "--demangle",
263 "-e",
264 map_src.c_str(),
265 nullptr
266 };
267 *pipe = Connect(map_src, args);
268 }
269
270 Addr2linePipe* pipe_ptr = pipe->get();
271 if (pipe_ptr == nullptr) {
272 // Failed...
273 return;
274 }
275
276 // Send the offset.
David Srbecky7400a542020-07-09 13:40:57 +0100277 const std::string hex_offset = StringPrintf("0x%zx\n", offset);
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700278
279 if (!pipe_ptr->out.WriteFully(hex_offset.data(), hex_offset.length())) {
280 // Error. :-(
281 pipe->reset();
282 return;
283 }
284
285 // Now drain (expecting two lines).
286 Drain(2U, prefix, pipe, os);
287}
288
Andreas Gampeca620d72016-11-08 08:09:33 -0800289static bool RunCommand(const std::string& cmd) {
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700290 FILE* stream = popen(cmd.c_str(), "r");
291 if (stream) {
David Srbeckyf3720542021-11-30 16:40:19 +0000292 // Consume the stdout until we encounter EOF when the tool exits.
293 // Otherwise the tool would complain to stderr when the stream is closed.
294 char buffer[64];
295 while (fread(buffer, 1, sizeof(buffer), stream) == sizeof(buffer)) {}
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700296 pclose(stream);
297 return true;
298 } else {
299 return false;
300 }
301}
302
David Srbecky6940a812022-09-29 14:40:10 +0100303// Remove method parameters by finding matching top-level parenthesis and removing them.
304// Since functions can be defined inside functions, this can remove multiple substrings.
305std::string StripParameters(std::string name) {
306 size_t end = name.size();
David Srbeckyd4ffc912022-09-26 18:11:38 +0100307 int nesting = 0;
308 for (ssize_t i = name.size() - 1; i > 0; i--) {
David Srbecky6940a812022-09-29 14:40:10 +0100309 if (name[i] == ')' && nesting++ == 0) {
310 end = i + 1;
311 }
312 if (name[i] == '(' && --nesting == 0) {
313 name = name.erase(i, end - i);
David Srbeckyd4ffc912022-09-26 18:11:38 +0100314 }
315 }
316 return name;
317}
318
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700319void DumpNativeStack(std::ostream& os,
320 pid_t tid,
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700321 const char* prefix,
322 ArtMethod* current_method,
323 void* ucontext_ptr,
324 bool skip_frames) {
325 unwindstack::AndroidLocalUnwinder unwinder;
326 DumpNativeStack(os, unwinder, tid, prefix, current_method, ucontext_ptr, skip_frames);
327}
328
329void DumpNativeStack(std::ostream& os,
330 unwindstack::AndroidLocalUnwinder& unwinder,
331 pid_t tid,
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700332 const char* prefix,
333 ArtMethod* current_method,
Christopher Ferrisb2749312018-03-23 13:03:45 -0700334 void* ucontext_ptr,
335 bool skip_frames) {
Roland Levillain05e34f42018-05-24 13:19:05 +0000336 // Historical note: This was disabled when running under Valgrind (b/18119146).
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700337
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700338 unwindstack::AndroidUnwinderData data(!skip_frames /*show_all_frames*/);
339 bool unwind_ret;
340 if (ucontext_ptr != nullptr) {
341 unwind_ret = unwinder.Unwind(ucontext_ptr, data);
342 } else {
343 unwind_ret = unwinder.Unwind(tid, data);
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700344 }
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700345 if (!unwind_ret) {
346 os << prefix << "(Unwind failed for thread " << tid << ": "
347 << data.GetErrorString() << ")" << std::endl;
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700348 return;
349 }
350
351 // Check whether we have and should use addr2line.
352 bool use_addr2line;
353 if (kUseAddr2line) {
354 // Try to run it to see whether we have it. Push an argument so that it doesn't assume a.out
355 // and print to stderr.
David Srbecky87da30e2019-01-30 15:51:23 +0000356 use_addr2line = (gAborting > 0) && RunCommand(FindAddr2line() + " -h");
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700357 } else {
358 use_addr2line = false;
359 }
360
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700361 std::unique_ptr<Addr2linePipe> addr2line_state;
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700362 data.DemangleFunctionNames();
Nicolas Geoffray8008d6b2023-01-05 15:03:07 +0000363 bool holds_mutator_lock = Locks::mutator_lock_->IsSharedHeld(Thread::Current());
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700364 for (const unwindstack::FrameData& frame : data.frames) {
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700365 // We produce output like this:
366 // ] #00 pc 000075bb8 /system/lib/libc.so (unwind_backtrace_thread+536)
367 // In order for parsing tools to continue to function, the stack dump
368 // format must at least adhere to this format:
369 // #XX pc <RELATIVE_ADDR> <FULL_PATH_TO_SHARED_LIBRARY> ...
370 // The parsers require a single space before and after pc, and two spaces
371 // after the <RELATIVE_ADDR>. There can be any prefix data before the
372 // #XX. <RELATIVE_ADDR> has to be a hex number but with no 0x prefix.
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700373 os << prefix << StringPrintf("#%02zu pc ", frame.num);
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700374 bool try_addr2line = false;
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700375 if (frame.map_info == nullptr) {
David Srbecky0d847082022-09-26 16:40:56 +0100376 os << StringPrintf("%08" PRIx64 " ???", frame.pc);
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700377 } else {
David Srbecky0d847082022-09-26 16:40:56 +0100378 os << StringPrintf("%08" PRIx64 " ", frame.rel_pc);
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700379 const std::shared_ptr<unwindstack::MapInfo>& map_info = frame.map_info;
380 if (map_info->name().empty()) {
381 os << StringPrintf("<anonymous:%" PRIx64 ">", map_info->start());
Christopher Ferris8bd7d1b2018-01-08 11:12:40 -0800382 } else {
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700383 os << map_info->name().c_str();
Christopher Ferris8bd7d1b2018-01-08 11:12:40 -0800384 }
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700385 if (map_info->elf_start_offset() != 0) {
386 os << StringPrintf(" (offset %" PRIx64 ")", map_info->elf_start_offset());
Christopher Ferris53ef6a62018-02-09 23:13:27 -0800387 }
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700388 os << " (";
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700389 if (!frame.function_name.empty()) {
David Srbeckyd4ffc912022-09-26 18:11:38 +0100390 // Remove parameters from the printed function name to improve signal/noise in the logs.
391 // Also, ANRs are often trimmed, so printing less means we get more useful data out.
392 // We can still symbolize the function based on the PC and build-id (including inlining).
393 os << StripParameters(frame.function_name.c_str());
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700394 if (frame.function_offset != 0) {
395 os << "+" << frame.function_offset;
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700396 }
Christopher Ferris8bd7d1b2018-01-08 11:12:40 -0800397 // Functions found using the gdb jit interface will be in an empty
398 // map that cannot be found using addr2line.
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700399 if (!map_info->name().empty()) {
Christopher Ferris8bd7d1b2018-01-08 11:12:40 -0800400 try_addr2line = true;
401 }
Nicolas Geoffray8008d6b2023-01-05 15:03:07 +0000402 } else if (current_method != nullptr && holds_mutator_lock) {
403 const OatQuickMethodHeader* header = current_method->GetOatQuickMethodHeader(frame.pc);
404 if (header != nullptr) {
405 const void* start_of_code = header->GetCode();
406 os << current_method->JniLongName() << "+"
407 << (frame.pc - reinterpret_cast<uint64_t>(start_of_code));
408 } else {
409 os << "???";
410 }
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700411 } else {
412 os << "???";
413 }
414 os << ")";
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700415 std::string build_id = map_info->GetPrintableBuildID();
David Srbecky7cd509c2021-10-28 16:27:16 +0100416 if (!build_id.empty()) {
417 os << " (BuildId: " << build_id << ")";
418 }
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700419 }
Andreas Gampeef295362016-10-11 20:04:11 -0700420 os << std::endl;
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700421 if (try_addr2line && use_addr2line) {
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700422 // Guaranteed that map_info is not nullptr and name is non-empty.
423 Addr2line(frame.map_info->name(), frame.rel_pc, os, prefix, &addr2line_state);
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700424 }
425 }
Andreas Gampefcccbaf2016-08-02 17:20:03 -0700426
427 if (addr2line_state != nullptr) {
428 Drain(0, prefix, &addr2line_state, os);
429 }
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700430}
431
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700432#elif defined(__APPLE__)
433
Stefano Cianciulli78f3c722023-05-16 10:32:54 +0000434void DumpNativeStack([[maybe_unused]] std::ostream& os,
435 [[maybe_unused]] pid_t tid,
436 [[maybe_unused]] const char* prefix,
437 [[maybe_unused]] ArtMethod* current_method,
438 [[maybe_unused]] void* ucontext_ptr,
439 [[maybe_unused]] bool skip_frames) {}
Christopher Ferrisf44070e2022-03-26 09:54:32 -0700440
Stefano Cianciulli78f3c722023-05-16 10:32:54 +0000441void DumpNativeStack([[maybe_unused]] std::ostream& os,
442 [[maybe_unused]] unwindstack::AndroidLocalUnwinder& existing_map,
443 [[maybe_unused]] pid_t tid,
444 [[maybe_unused]] const char* prefix,
445 [[maybe_unused]] ArtMethod* current_method,
446 [[maybe_unused]] void* ucontext_ptr,
447 [[maybe_unused]] bool skip_frames) {}
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700448
Andreas Gampe5dd44d02016-08-02 17:20:03 -0700449#else
450#error "Unsupported architecture for native stack dumps."
451#endif
452
453} // namespace art