forked from procxx/cpp-stacktrace-proposal
-
Notifications
You must be signed in to change notification settings - Fork 1
/
stacktrace.html
270 lines (217 loc) · 15.3 KB
/
stacktrace.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html><head>
<title>A Proposal to add stack trace library</title>
<meta content="http://schemas.microsoft.com/intellisense/ie5" name="vs_targetSchema">
<meta http-equiv="Content-Language" content="en-us">
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<style type="text/css">
.addition { color: green; }
.right { float:right; }
.changed-deleted { background-color: #CFF0FC ; text-decoration: line-through; display: none; }
.addition.changed-deleted { color: green; background-color: #CFF0FC ; text-decoration: line-through; text-decoration: black double line-through; display: none; }
.changed-added { background-color: #CFF0FC ;}
.notes { background-color: #80D080 ;}
pre { line-height: 1.2; font-size: 10pt; margin-top: 25px; }
.desc { margin-left: 35px; margin-top: 10px; padding:0; white-space: normal; }
body {max-width: 1024px; margin-left: 25px;}
.cppkeyword { color: blue; }
.cppcomment { color: green; }
.cppcomment > .cppkeyword{ color: green; }
.cpptext { color: #2E8B57; }
</style>
</head>
<body bgcolor="#ffffff">
<address>Document number: D0881R1</address>
<address>Project: Programming Language C++</address>
<address>Audience: Library Evolution</address>
<address> </address>
<address>Alexey Gorgurov <<a href="mailto:[email protected]">[email protected]</a>>, <<a href="mailto:[email protected]">[email protected]</a>></address>
<address>Antony Polukhin <<a href="mailto:[email protected]">[email protected]</a>>, <<a href="mailto:[email protected]">[email protected]</a>></address>
<address> </address>
<address>Date: 2018-04-13</address>
<h1>A Proposal to add stack trace library</h1>
<p class='changed-added'>Significant changes to <a href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2018/p0881r0.html">P0881R0</a> are marked with blue.<p>
<p><input type="checkbox" id="show_deletions" onchange="show_hide_deleted()"> Show deleted lines from P0881R0.</p>
<h2>I. Motivation</h2>
<p>In the current working draft [<a href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2017/n4713.pdf">N4713</a>] there is no way to get,store and decode the current call sequence.
Such call sequences are useful for debugging and post mortem debugging. They are popular in other programming languages (like Java, C#, Python).</p>
<p>Pretty often assertions can't describe the whole picture of a bug and do not provide enough information to locate the problem.
For example, you can see the following message on out-of-range access:</p>
<pre>
boost/array.hpp:123: T& boost::array<T, N>::operator[](boost::array<T, N>::size_type): Assertion '(i < N)&&("out of range")' failed.
Aborted (core dumped)</pre>
<p>That's not enough information in the assert message to locate the problem without debugger.</p>
<p>This paper proposes classes that could simplify debugging and may change the assertion meassage into the following:</p>
<pre>
Expression 'i < N' is false in function 'T& boost::array<T, N>::operator[](boost::array<T, N>::size_type) [with T = int; long unsigned int N = 5ul; boost::array<T, N>::reference = int&; boost::array<T, N>::size_type = long unsigned int]': out of range.
Backtrace:
0# boost::assertion_failed_msg(char const*, char const*, char const*, char const*, long) at ../example/assert_handler.cpp:39
1# boost::array<int, 5ul>::operator[](unsigned long) at ../../../boost/array.hpp:124
2# bar(int) at ../example/assert_handler.cpp:17
3# foo(int) at ../example/assert_handler.cpp:25
4# bar(int) at ../example/assert_handler.cpp:17
5# foo(int) at ../example/assert_handler.cpp:25
6# main at ../example/assert_handler.cpp:54
7# 0x00007F991FD69F45 in /lib/x86_64-linux-gnu/libc.so.6
8# 0x0000000000401139
</pre>
<h2>II. Impact on the Standard</h2>
<p>This proposal is a pure library extension and it does not break the existing code and does not degrade performance.
It does not require any changes in the core language and could be implemented in the standard C++.</p>
<h2>III. Design Decisions</h2>
<p>The design is based on the Boost.Stacktrace library, a popular library that does not depend on any non-standard library components.</p>
<p><b>Note about signal safety:</b> this proposal does not attempt to provide a signal-safe solution for capturing and decoding stacktraces.
Such functionality currently is not implementable on some of the popular platforms. However, the paper attempts to provide extensible solution, that may be made signal safe some day
<span class="changed-added">by providing a signal safe allocator and changing the <code>stacktrace</code> implementation details</span>.</p>
<p><b>Note on performance:</b> during Boost.Stacktrace development phase many users requested a fast way to store stack trace, without decoding the function names. This functionality is preserved in the paper.
<span class="changed-added">All the <code>stack_frame</code> functions and constructors are lazy and won't decode the pointer information if there was no explicit request from class user.</span></p>
<p class="changed-added"><b>Note on allocations:</b> initial implementations of Boost.Stacktrace were not using allocator and all the frames were placed inside a fixed size internal storage.
That was a mistake! Sometimes the most important information is located at the bottom of the stack. For example if you run Boost.Test, then the test name will be located low on the stack.
With a fixed size storage the bottom of the stack could be lost along with the information.</p>
<p class="changed-added">Current design assumes that by default users wish to see the whole stack and OK with dynamic allocations, because do not construct <code>stacktrace</code>
in performance critical places.
For those users, who wish to use <code>stacktrace</code> on a hot path or in embedded environments <code>basic_stacktrace</code> allows to provide a custom allocator that allocates
on the stack or in some other place, where users thinks it is appropriate.</p>
<p class="changed-added"><b>Note on returning <code>std::string</code> and not having <code>noexcept</code> on <code>stack_frame::source_line()</code></b>:
Unfortunately this is a necessarity on some platforms, where getting source line requires allocating or where source
file name <a href="https://github.com/boostorg/stacktrace/blob/a0f948e9f505cb53baf582fccbcb3024fd255ee1/include/boost/stacktrace/detail/frame_msvc.ipp#L213-L219">retuned into</a> a storage
<a href="http://www.qnx.com/developers/docs/6.5.0/index.jsp?topic=%2Fcom.qnx.doc.neutrino_lib_ref%2Fb%2Fbt_get_backtrace.html"> provided by user</a>.</p>
<h2>IV. Proposed Interface</h2>
<h3>Header <stacktrace> synopsis</h3>
<pre>
namespace std {
class stack_frame;
template<typename Allocator>
class basic_stacktrace;
// free functions
// This is the alias to use unless you'd like to provide a specific allocator to basic_stacktrace.
using stacktrace = basic_stacktrace<allocator<stack_frame>>;
// Outputs stacktrace in a human readable format to output stream.
template<typename CharT, typename TraitsT, typename Allocator>
basic_ostream< CharT, TraitsT > & operator<<(basic_ostream<CharT, TraitsT>& os, const basic_stacktrace<Allocator>& bt);
// Outputs stacktrace in a human readable format to string.
template<typename Allocator>
string to_string(const basic_stacktrace<Allocator>& f);
// Outputs frame in a human readable format to string.
string to_string(const stack_frame& f);
// Outputs frame in a human readable format to output stream.
template<typename CharT, typename TraitsT>
basic_ostream< CharT, TraitsT >& operator<<(basic_ostream<CharT, TraitsT>& os, const stack_frame& f);
}
</pre>
<h3>Class stack_frame</h3>
<p>The stack_frame class stores a pointer to function and allows querying information about that function.</p>
<pre>
namespace std {
class stack_frame {
public:
using native_frame_ptr_t = <i>unspecified</i>;
// construct/copy/destruct
constexpr stack_frame() noexcept;
constexpr stack_frame(const stack_frame&) noexcept = default;
constexpr stack_frame& operator=(const stack_frame&)<span class="changed-added"> noexcept</span> = default;
constexpr explicit stack_frame(native_frame_ptr_t f) noexcept;
template<typename T> explicit stack_frame(T* address) noexcept;
constexpr native_frame_ptr_t address() const noexcept;
constexpr explicit operator bool() const noexcept;
constexpr strong_ordering operator <=>(const stack_frame& rhs) = default;
// functions that query<span class="changed-deleted">ing</span> information about stored address
string name() const;
string source_file() const;
size_t source_line() const;
private:
native_frame_ptr_t data; // exposition only
};
}
</pre>
<h3><code>stack_frame</code> constructors</h3>
<pre>stack_frame() noexcept;</pre>
<div class="desc">Constructs stack_frame that references nullptr address. Calls to <code>source_file()</code> and <code>source_line()</code> will return empty string. Calls to <code>source_line()</code> will return 0.</div>
<pre>explicit stack_frame(native_frame_ptr_t addr) noexcept;
template<typename T> explicit stack_frame(T * addr) noexcept;</pre>
<div class="desc">Constructs stack_frame that references addr and could later generate information about that address using platform specific features.</div>
<h3><code>stack_frame</code> member functions</h3>
<pre>std::string name() const;</pre>
<div class="desc">Returns empty string or platform specific name of the function that the stored address is pointing to. Throws std::bad_alloc if not enough memory to construct resulting string.</div>
<pre>constexpr native_frame_ptr_t address() const noexcept;</pre>
<div class="desc">Returns address stored by this.</div>
<pre>std::string source_file() const;</pre>
<div class="desc">Returns empty string or path to the source file, where the function of the frame is defined. Returns empty string if this->source_line() == 0. Throws std::bad_alloc if not enough memory to construct resulting string.</div>
<pre>std::string source_line() const;</pre>
<div class="desc">Returns 0 or code line in the source file, where the function of the stored address is defined. Throws std::bad_alloc if not enough memory to construct resulting string.</div>
<pre>explicit operator bool() const noexcept;</pre>
<div class="desc">Returns true if this stores and address other than NULL.</div>
<h3>Class template <basic_stacktrace></h3>
<p>The basic_stacktrace template class stores current call sequence on construction and provides functions for querying information about the call sequence.</p>
<pre>
namespace std {
template<typename Allocator>
class basic_stacktrace {
public:
using value_type = stack_frame;
using const_reference = const value_type &;
using size_type = <i>implementation-defined</i>;
using const_iterator = <i>implementation-defined</i>;
<span class="changed-added">using allocator_type = Allocator;</span>
<span class="changed-deleted">using allocaotr_type = Allocator;</span>
// functions that capture current call sequence
basic_stacktrace() noexcept;
explicit basic_stacktrace(const allocator_type& a) noexcept;
basic_stacktrace(size_type skip, size_type max_depth, const allocator_type& a = allocator_type()) noexcept;
// construct/copy/destruct
basic_stacktrace(const basic_stacktrace &);
basic_stacktrace(basic_stacktrace &&) noexcept;
basic_stacktrace & operator=(const basic_stacktrace &);
basic_stacktrace & operator=(basic_stacktrace &&)<span class="changed-added"> noexcept</span>;
~basic_stacktrace();
// public member functions
size_type size() const noexcept;
const_reference operator[](size_type ) const noexcept;
const_iterator begin() const noexcept;
const_iterator end() const noexcept;
explicit operator bool() const noexcept;
template <typename Allocator2>
strong_ordering operator <=>(const basic_stacktrace< Allocator2 >& rhs) <span class="changed-added"> noexcept</span> = default;
private:
vector<value_type> stack_frames; // exposition only
};
}
</pre>
<h3><code>basic_stacktrace</code> constructors</h3>
<pre>basic_stacktrace() noexcept;
explicit basic_stacktrace(const allocator_type & a) noexcept;</pre>
<div class="desc">Stores the current call sequence inside *this without querying information about each call.</div>
<div class="desc">Any exception raised during this operation is
silently ignored. In case of exception <code>(bool)*this</code> is <code>false</code></div>
<pre>basic_stacktrace(size_type skip, size_type max_depth, const allocator_type& a = allocator_type()) noexcept;</pre>
<div class="desc">Stores [skip; skip + max_depth) of the current function call sequence inside *this without querying information about each call.</div>
<div class="desc">Any exception raised during this operation is
silently ignored. In case of exception <code>(bool)*this</code> is <code>false</code></div>
<h3><code>basic_stacktrace</code> member functions</h3>
<pre>const_reference operator[](size_type frame_no) const noexcept;</pre>
<div class="desc">Returns frame that references the actual frame info, stored inside *this.</div>
<div class="desc">Parameters: <code>frame_no</code> - zero-based index of frame to return. 0 is the function index where stacktrace was constructed and index close to this->size() contains function main().</div>
<pre>explicit operator bool() const noexcept;</pre>
<div class="desc">Returns true if call sequence was successfully stored.</div>
<h2>V. Feature-testing macro</h2>
<p>For the purposes of SG10 we recommend the feature-testing macro name <code>__cpp_lib_stacktrace</code>.</p>
<script type="text/javascript">
function colorize_texts(texts) {
for (var i = 0; i < texts.length; ++i) {
var text = texts[i].innerHTML;
text = text.replace(/namespace|enum|void|constexpr|extern|noexcept|bool|template|class |struct|auto|const |typename|explicit|public|private|#include|inline|typedef|static_assert|static_cast|static/g,"<span class='cppkeyword'>$&<\/span>");
text = text.replace(/\/\/[\s\S]+?\n/g,"<span class='cppcomment'>$&<\/span>");
texts[i].innerHTML = text;
}
}
colorize_texts(document.getElementsByTagName("pre"));
colorize_texts(document.getElementsByTagName("code"));
function show_hide_deleted() {
var to_change = document.getElementsByClassName('changed-deleted');
for (var i = 0; i < to_change.length; ++i) {
to_change[i].style.display = (document.getElementById("show_deletions").checked ? 'block' : 'none');
}
}
show_hide_deleted()
</script>
</body></html>