blob: dc1665b49f5f3f7034040ded2525357827178811 [file] [log] [blame]
Austin Schuh906616c2019-01-21 20:25:11 -08001// Copyright (c) 2005 - 2007, Google Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8// * Redistributions of source code must retain the above copyright
9// notice, this list of conditions and the following disclaimer.
10// * Redistributions in binary form must reproduce the above
11// copyright notice, this list of conditions and the following disclaimer
12// in the documentation and/or other materials provided with the
13// distribution.
14// * Neither the name of Google Inc. nor the names of its
15// contributors may be used to endorse or promote products derived from
16// this software without specific prior written permission.
17//
18// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29//
30// Author: Arun Sharma
31//
32// Produce stack trace using libgcc
33
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070034#include <cstdlib> // for NULL
Austin Schuh906616c2019-01-21 20:25:11 -080035#include <unwind.h> // ABI defined unwinder
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070036
Austin Schuh906616c2019-01-21 20:25:11 -080037#include "stacktrace.h"
38
39_START_GOOGLE_NAMESPACE_
40
41typedef struct {
42 void **result;
43 int max_depth;
44 int skip_count;
45 int count;
46} trace_arg_t;
47
48
49// Workaround for the malloc() in _Unwind_Backtrace() issue.
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070050static _Unwind_Reason_Code nop_backtrace(struct _Unwind_Context */*uc*/, void */*opq*/) {
Austin Schuh906616c2019-01-21 20:25:11 -080051 return _URC_NO_REASON;
52}
53
54
55// This code is not considered ready to run until
56// static initializers run so that we are guaranteed
57// that any malloc-related initialization is done.
58static bool ready_to_run = false;
59class StackTraceInit {
60 public:
61 StackTraceInit() {
62 // Extra call to force initialization
63 _Unwind_Backtrace(nop_backtrace, NULL);
64 ready_to_run = true;
65 }
66};
67
68static StackTraceInit module_initializer; // Force initialization
69
70static _Unwind_Reason_Code GetOneFrame(struct _Unwind_Context *uc, void *opq) {
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070071 trace_arg_t *targ = static_cast<trace_arg_t *>(opq);
Austin Schuh906616c2019-01-21 20:25:11 -080072
73 if (targ->skip_count > 0) {
74 targ->skip_count--;
75 } else {
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070076 targ->result[targ->count++] = reinterpret_cast<void *>(_Unwind_GetIP(uc));
Austin Schuh906616c2019-01-21 20:25:11 -080077 }
78
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070079 if (targ->count == targ->max_depth) {
Austin Schuh906616c2019-01-21 20:25:11 -080080 return _URC_END_OF_STACK;
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070081 }
Austin Schuh906616c2019-01-21 20:25:11 -080082
83 return _URC_NO_REASON;
84}
85
86// If you change this function, also change GetStackFrames below.
87int GetStackTrace(void** result, int max_depth, int skip_count) {
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070088 if (!ready_to_run) {
Austin Schuh906616c2019-01-21 20:25:11 -080089 return 0;
James Kuszmaulba0ac1a2022-08-12 16:29:30 -070090 }
Austin Schuh906616c2019-01-21 20:25:11 -080091
92 trace_arg_t targ;
93
94 skip_count += 1; // Do not include the "GetStackTrace" frame
95
96 targ.result = result;
97 targ.max_depth = max_depth;
98 targ.skip_count = skip_count;
99 targ.count = 0;
100
101 _Unwind_Backtrace(GetOneFrame, &targ);
102
103 return targ.count;
104}
105
106_END_GOOGLE_NAMESPACE_