1 | //===-- sanitizer_stacktrace.h ----------------------------------*- C++ -*-===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // This file is shared between AddressSanitizer and ThreadSanitizer |
10 | // run-time libraries. |
11 | //===----------------------------------------------------------------------===// |
12 | #ifndef SANITIZER_STACKTRACE_H |
13 | #define SANITIZER_STACKTRACE_H |
14 | |
15 | #include "sanitizer_common.h" |
16 | #include "sanitizer_internal_defs.h" |
17 | #include "sanitizer_platform.h" |
18 | |
19 | namespace __sanitizer { |
20 | |
21 | struct BufferedStackTrace; |
22 | |
23 | static const u32 kStackTraceMax = 255; |
24 | |
25 | #if SANITIZER_LINUX && defined(__mips__) |
26 | # define SANITIZER_CAN_FAST_UNWIND 0 |
27 | #elif SANITIZER_WINDOWS |
28 | # define SANITIZER_CAN_FAST_UNWIND 0 |
29 | #else |
30 | # define SANITIZER_CAN_FAST_UNWIND 1 |
31 | #endif |
32 | |
33 | // Fast unwind is the only option on Mac for now; we will need to |
34 | // revisit this macro when slow unwind works on Mac, see |
35 | // https://github.com/google/sanitizers/issues/137 |
36 | #if SANITIZER_APPLE |
37 | # define SANITIZER_CAN_SLOW_UNWIND 0 |
38 | #else |
39 | # define SANITIZER_CAN_SLOW_UNWIND 1 |
40 | #endif |
41 | |
42 | struct StackTrace { |
43 | const uptr *trace; |
44 | u32 size; |
45 | u32 tag; |
46 | |
47 | static const int TAG_UNKNOWN = 0; |
48 | static const int TAG_ALLOC = 1; |
49 | static const int TAG_DEALLOC = 2; |
50 | static const int TAG_CUSTOM = 100; // Tool specific tags start here. |
51 | |
52 | StackTrace() : trace(nullptr), size(0), tag(0) {} |
53 | StackTrace(const uptr *trace, u32 size) : trace(trace), size(size), tag(0) {} |
54 | StackTrace(const uptr *trace, u32 size, u32 tag) |
55 | : trace(trace), size(size), tag(tag) {} |
56 | |
57 | // Prints a symbolized stacktrace, followed by an empty line. |
58 | void Print() const; |
59 | |
60 | // Prints a symbolized stacktrace to the output string, followed by an empty |
61 | // line. |
62 | void PrintTo(InternalScopedString *output) const; |
63 | |
64 | // Prints a symbolized stacktrace to the output buffer, followed by an empty |
65 | // line. Returns the number of symbols that should have been written to buffer |
66 | // (not including trailing '\0'). Thus, the string is truncated iff return |
67 | // value is not less than "out_buf_size". |
68 | uptr PrintTo(char *out_buf, uptr out_buf_size) const; |
69 | |
70 | static bool WillUseFastUnwind(bool request_fast_unwind) { |
71 | if (!SANITIZER_CAN_FAST_UNWIND) |
72 | return false; |
73 | if (!SANITIZER_CAN_SLOW_UNWIND) |
74 | return true; |
75 | return request_fast_unwind; |
76 | } |
77 | |
78 | static uptr GetCurrentPc(); |
79 | static inline uptr GetPreviousInstructionPc(uptr pc); |
80 | static uptr GetNextInstructionPc(uptr pc); |
81 | }; |
82 | |
83 | // Performance-critical, must be in the header. |
84 | ALWAYS_INLINE |
85 | uptr StackTrace::GetPreviousInstructionPc(uptr pc) { |
86 | #if defined(__arm__) |
87 | // T32 (Thumb) branch instructions might be 16 or 32 bit long, |
88 | // so we return (pc-2) in that case in order to be safe. |
89 | // For A32 mode we return (pc-4) because all instructions are 32 bit long. |
90 | return (pc - 3) & (~1); |
91 | #elif defined(__sparc__) || defined(__mips__) |
92 | return pc - 8; |
93 | #elif SANITIZER_RISCV64 |
94 | // RV-64 has variable instruction length... |
95 | // C extentions gives us 2-byte instructoins |
96 | // RV-64 has 4-byte instructions |
97 | // + RISC-V architecture allows instructions up to 8 bytes |
98 | // It seems difficult to figure out the exact instruction length - |
99 | // pc - 2 seems like a safe option for the purposes of stack tracing |
100 | return pc - 2; |
101 | #elif SANITIZER_S390 || SANITIZER_I386 || SANITIZER_X32 || SANITIZER_X64 |
102 | return pc - 1; |
103 | #else |
104 | return pc - 4; |
105 | #endif |
106 | } |
107 | |
108 | // StackTrace that owns the buffer used to store the addresses. |
109 | struct BufferedStackTrace : public StackTrace { |
110 | uptr trace_buffer[kStackTraceMax]; |
111 | uptr top_frame_bp; // Optional bp of a top frame. |
112 | |
113 | BufferedStackTrace() : StackTrace(trace_buffer, 0), top_frame_bp(0) {} |
114 | |
115 | void Init(const uptr *pcs, uptr cnt, uptr = 0); |
116 | |
117 | // Get the stack trace with the given pc and bp. |
118 | // The pc will be in the position 0 of the resulting stack trace. |
119 | // The bp may refer to the current frame or to the caller's frame. |
120 | void Unwind(uptr pc, uptr bp, void *context, bool request_fast, |
121 | u32 max_depth = kStackTraceMax) { |
122 | top_frame_bp = (max_depth > 0) ? bp : 0; |
123 | // Small max_depth optimization |
124 | if (max_depth <= 1) { |
125 | if (max_depth == 1) |
126 | trace_buffer[0] = pc; |
127 | size = max_depth; |
128 | return; |
129 | } |
130 | UnwindImpl(pc, bp, context, request_fast, max_depth); |
131 | } |
132 | |
133 | void Unwind(u32 max_depth, uptr pc, uptr bp, void *context, uptr stack_top, |
134 | uptr stack_bottom, bool request_fast_unwind); |
135 | |
136 | void Reset() { |
137 | *static_cast<StackTrace *>(this) = StackTrace(trace_buffer, 0); |
138 | top_frame_bp = 0; |
139 | } |
140 | |
141 | private: |
142 | // Every runtime defines its own implementation of this method |
143 | void UnwindImpl(uptr pc, uptr bp, void *context, bool request_fast, |
144 | u32 max_depth); |
145 | |
146 | // UnwindFast/Slow have platform-specific implementations |
147 | void UnwindFast(uptr pc, uptr bp, uptr stack_top, uptr stack_bottom, |
148 | u32 max_depth); |
149 | void UnwindSlow(uptr pc, u32 max_depth); |
150 | void UnwindSlow(uptr pc, void *context, u32 max_depth); |
151 | |
152 | void PopStackFrames(uptr count); |
153 | uptr LocatePcInTrace(uptr pc); |
154 | |
155 | BufferedStackTrace(const BufferedStackTrace &) = delete; |
156 | void operator=(const BufferedStackTrace &) = delete; |
157 | |
158 | friend class FastUnwindTest; |
159 | }; |
160 | |
161 | #if defined(__s390x__) |
162 | static const uptr kFrameSize = 160; |
163 | #elif defined(__s390__) |
164 | static const uptr kFrameSize = 96; |
165 | #else |
166 | static const uptr kFrameSize = 2 * sizeof(uhwptr); |
167 | #endif |
168 | |
169 | // Check if given pointer points into allocated stack area. |
170 | static inline bool IsValidFrame(uptr frame, uptr stack_top, uptr stack_bottom) { |
171 | return frame > stack_bottom && frame < stack_top - kFrameSize; |
172 | } |
173 | |
174 | } // namespace __sanitizer |
175 | |
176 | // Use this macro if you want to print stack trace with the caller |
177 | // of the current function in the top frame. |
178 | #define GET_CALLER_PC_BP \ |
179 | uptr bp = GET_CURRENT_FRAME(); \ |
180 | uptr pc = GET_CALLER_PC(); |
181 | |
182 | #define GET_CALLER_PC_BP_SP \ |
183 | GET_CALLER_PC_BP; \ |
184 | uptr local_stack; \ |
185 | uptr sp = (uptr)&local_stack |
186 | |
187 | // Use this macro if you want to print stack trace with the current |
188 | // function in the top frame. |
189 | #define GET_CURRENT_PC_BP \ |
190 | uptr bp = GET_CURRENT_FRAME(); \ |
191 | uptr pc = StackTrace::GetCurrentPc() |
192 | |
193 | #define GET_CURRENT_PC_BP_SP \ |
194 | GET_CURRENT_PC_BP; \ |
195 | uptr local_stack; \ |
196 | uptr sp = (uptr)&local_stack |
197 | |
198 | // GET_CURRENT_PC() is equivalent to StackTrace::GetCurrentPc(). |
199 | // Optimized x86 version is faster than GetCurrentPc because |
200 | // it does not involve a function call, instead it reads RIP register. |
201 | // Reads of RIP by an instruction return RIP pointing to the next |
202 | // instruction, which is exactly what we want here, thus 0 offset. |
203 | // It needs to be a macro because otherwise we will get the name |
204 | // of this function on the top of most stacks. Attribute artificial |
205 | // does not do what it claims to do, unfortunatley. And attribute |
206 | // __nodebug__ is clang-only. If we would have an attribute that |
207 | // would remove this function from debug info, we could simply make |
208 | // StackTrace::GetCurrentPc() faster. |
209 | #if defined(__x86_64__) |
210 | # define GET_CURRENT_PC() \ |
211 | (__extension__({ \ |
212 | uptr pc; \ |
213 | asm("lea 0(%%rip), %0" : "=r"(pc)); \ |
214 | pc; \ |
215 | })) |
216 | #else |
217 | # define GET_CURRENT_PC() StackTrace::GetCurrentPc() |
218 | #endif |
219 | |
220 | #endif // SANITIZER_STACKTRACE_H |
221 | |