]>
Commit | Line | Data |
---|---|---|
eaff0f0d | 1 | ///////////////////////////////////////////////////////////////////////////// |
80fdcdb9 | 2 | // Name: src/unix/stackwalk.cpp |
eaff0f0d VZ |
3 | // Purpose: wxStackWalker implementation for Unix/glibc |
4 | // Author: Vadim Zeitlin | |
5 | // Modified by: | |
6 | // Created: 2005-01-18 | |
eaff0f0d VZ |
7 | // Copyright: (c) 2005 Vadim Zeitlin <vadim@wxwindows.org> |
8 | // Licence: wxWindows licence | |
9 | ///////////////////////////////////////////////////////////////////////////// | |
10 | ||
11 | // ============================================================================ | |
12 | // declarations | |
13 | // ============================================================================ | |
14 | ||
15 | // ---------------------------------------------------------------------------- | |
16 | // headers | |
17 | // ---------------------------------------------------------------------------- | |
18 | ||
19 | #include "wx/wxprec.h" | |
20 | ||
21 | #ifdef __BORLANDC__ | |
22 | #pragma hdrstop | |
23 | #endif | |
24 | ||
25 | #if wxUSE_STACKWALKER | |
26 | ||
27 | #ifndef WX_PRECOMP | |
28 | #include "wx/string.h" | |
7a3ba35d KH |
29 | #include "wx/app.h" |
30 | #include "wx/log.h" | |
31 | #include "wx/utils.h" | |
eaff0f0d VZ |
32 | #endif |
33 | ||
34 | #include "wx/stackwalk.h" | |
a82c2299 | 35 | #include "wx/stdpaths.h" |
eaff0f0d VZ |
36 | |
37 | #include <execinfo.h> | |
38 | ||
39 | #ifdef HAVE_CXA_DEMANGLE | |
40 | #include <cxxabi.h> | |
41 | #endif // HAVE_CXA_DEMANGLE | |
42 | ||
ede3a6d6 VZ |
43 | // ---------------------------------------------------------------------------- |
44 | // tiny helper wrapper around popen/pclose() | |
45 | // ---------------------------------------------------------------------------- | |
46 | ||
47 | class wxStdioPipe | |
48 | { | |
49 | public: | |
50 | // ctor parameters are passed to popen() | |
51 | wxStdioPipe(const char *command, const char *type) | |
52 | { | |
53 | m_fp = popen(command, type); | |
54 | } | |
55 | ||
56 | // conversion to stdio FILE | |
57 | operator FILE *() const { return m_fp; } | |
58 | ||
59 | // dtor closes the pipe | |
60 | ~wxStdioPipe() | |
61 | { | |
62 | if ( m_fp ) | |
63 | pclose(m_fp); | |
64 | } | |
65 | ||
66 | private: | |
67 | FILE *m_fp; | |
68 | ||
c0c133e1 | 69 | wxDECLARE_NO_COPY_CLASS(wxStdioPipe); |
ede3a6d6 VZ |
70 | }; |
71 | ||
eaff0f0d VZ |
72 | // ============================================================================ |
73 | // implementation | |
74 | // ============================================================================ | |
75 | ||
eaff0f0d VZ |
76 | // ---------------------------------------------------------------------------- |
77 | // wxStackFrame | |
78 | // ---------------------------------------------------------------------------- | |
79 | ||
80 | void wxStackFrame::OnGetName() | |
81 | { | |
a82c2299 | 82 | if ( !m_name.empty() ) |
eaff0f0d VZ |
83 | return; |
84 | ||
a82c2299 RR |
85 | // we already tried addr2line in wxStackWalker::InitFrames: it always |
86 | // gives us demangled names (even if __cxa_demangle is not available) when | |
87 | // the function is part of the ELF (when it's in a shared object addr2line | |
88 | // will give "??") and because it seems less error-prone. | |
eaff0f0d | 89 | // when it works, backtrace_symbols() sometimes returns incorrect results |
eaff0f0d VZ |
90 | |
91 | // format is: "module(funcname+offset) [address]" but the part in | |
92 | // parentheses can be not present | |
93 | wxString syminfo = wxString::FromAscii(m_syminfo); | |
9a83f860 | 94 | const size_t posOpen = syminfo.find(wxT('(')); |
eaff0f0d VZ |
95 | if ( posOpen != wxString::npos ) |
96 | { | |
9a83f860 | 97 | const size_t posPlus = syminfo.find(wxT('+'), posOpen + 1); |
eaff0f0d VZ |
98 | if ( posPlus != wxString::npos ) |
99 | { | |
9a83f860 | 100 | const size_t posClose = syminfo.find(wxT(')'), posPlus + 1); |
eaff0f0d VZ |
101 | if ( posClose != wxString::npos ) |
102 | { | |
103 | if ( m_name.empty() ) | |
104 | { | |
105 | m_name.assign(syminfo, posOpen + 1, posPlus - posOpen - 1); | |
106 | ||
107 | #ifdef HAVE_CXA_DEMANGLE | |
108 | int rc = -1; | |
109 | char *cppfunc = __cxxabiv1::__cxa_demangle | |
110 | ( | |
111 | m_name.mb_str(), | |
112 | NULL, // output buffer (none, alloc it) | |
113 | NULL, // [out] len of output buffer | |
114 | &rc | |
115 | ); | |
116 | if ( rc == 0 ) | |
117 | m_name = wxString::FromAscii(cppfunc); | |
118 | ||
119 | free(cppfunc); | |
120 | #endif // HAVE_CXA_DEMANGLE | |
121 | } | |
122 | ||
123 | unsigned long ofs; | |
124 | if ( wxString(syminfo, posPlus + 1, posClose - posPlus - 1). | |
125 | ToULong(&ofs, 0) ) | |
126 | m_offset = ofs; | |
127 | } | |
128 | } | |
eaff0f0d | 129 | |
1872e042 VZ |
130 | m_module.assign(syminfo, posOpen); |
131 | } | |
abf99aad | 132 | #ifndef __WXOSX__ |
1872e042 VZ |
133 | else // not in "module(funcname+offset)" format |
134 | { | |
135 | m_module = syminfo; | |
136 | } | |
abf99aad | 137 | #endif // !__WXOSX__ |
eaff0f0d VZ |
138 | } |
139 | ||
a82c2299 RR |
140 | |
141 | // ---------------------------------------------------------------------------- | |
142 | // wxStackWalker | |
143 | // ---------------------------------------------------------------------------- | |
144 | ||
145 | // that many frames should be enough for everyone | |
146 | #define MAX_FRAMES 200 | |
147 | ||
148 | // we need a char buffer big enough to contain a call to addr2line with | |
149 | // up to MAX_FRAMES addresses ! | |
150 | // NB: %p specifier will print the pointer in hexadecimal form | |
151 | // and thus will require 2 chars for each byte + 3 for the | |
152 | // " 0x" prefix | |
153 | #define CHARS_PER_FRAME (sizeof(void*) * 2 + 3) | |
154 | ||
155 | // BUFSIZE will be 2250 for 32 bit machines | |
156 | #define BUFSIZE (50 + MAX_FRAMES*CHARS_PER_FRAME) | |
157 | ||
158 | // static data | |
159 | void *wxStackWalker::ms_addresses[MAX_FRAMES]; | |
160 | char **wxStackWalker::ms_symbols = NULL; | |
161 | int wxStackWalker::m_depth = 0; | |
162 | wxString wxStackWalker::ms_exepath; | |
163 | static char g_buf[BUFSIZE]; | |
164 | ||
165 | ||
166 | void wxStackWalker::SaveStack(size_t maxDepth) | |
167 | { | |
168 | // read all frames required | |
169 | maxDepth = wxMin(WXSIZEOF(ms_addresses)/sizeof(void*), maxDepth); | |
170 | m_depth = backtrace(ms_addresses, maxDepth*sizeof(void*)); | |
171 | if ( !m_depth ) | |
172 | return; | |
173 | ||
174 | ms_symbols = backtrace_symbols(ms_addresses, m_depth); | |
175 | } | |
176 | ||
177 | void wxStackWalker::ProcessFrames(size_t skip) | |
eaff0f0d | 178 | { |
a82c2299 RR |
179 | wxStackFrame frames[MAX_FRAMES]; |
180 | ||
181 | if (!ms_symbols || !m_depth) | |
eaff0f0d VZ |
182 | return; |
183 | ||
1114902a VZ |
184 | // we are another level down from Walk(), so adjust the number of stack |
185 | // frames to skip accordingly | |
d2fd6629 | 186 | skip += 1; |
a82c2299 RR |
187 | |
188 | // call addr2line only once since this call may be very slow | |
189 | // (it has to load in memory the entire EXE of this app which may be quite | |
190 | // big, especially if it contains debug info and is compiled statically!) | |
1114902a VZ |
191 | int numFrames = InitFrames(frames, m_depth - skip, |
192 | &ms_addresses[skip], &ms_symbols[skip]); | |
a82c2299 RR |
193 | |
194 | // now do user-defined operations on each frame | |
1114902a | 195 | for ( int n = 0; n < numFrames; n++ ) |
a82c2299 RR |
196 | OnStackFrame(frames[n]); |
197 | } | |
198 | ||
199 | void wxStackWalker::FreeStack() | |
200 | { | |
201 | // ms_symbols has been allocated by backtrace_symbols() and it's the responsibility | |
202 | // of the caller, i.e. us, to free that pointer | |
203 | if (ms_symbols) | |
204 | free( ms_symbols ); | |
205 | ms_symbols = NULL; | |
206 | m_depth = 0; | |
207 | } | |
eaff0f0d | 208 | |
abf99aad VZ |
209 | namespace |
210 | { | |
211 | ||
212 | // Helper function to read a line from the file and return it without the | |
213 | // trailing newline. Line number is only used for error reporting. | |
214 | bool ReadLine(FILE* fp, unsigned long num, wxString* line) | |
215 | { | |
216 | if ( !fgets(g_buf, WXSIZEOF(g_buf), fp) ) | |
217 | { | |
218 | wxLogDebug(wxS("cannot read address information for stack frame #%lu"), | |
219 | num); | |
220 | return false; | |
221 | } | |
222 | ||
223 | *line = wxString::FromAscii(g_buf); | |
224 | line->RemoveLast(); | |
225 | ||
226 | return true; | |
227 | } | |
228 | ||
229 | } // anonymous namespace | |
230 | ||
a82c2299 RR |
231 | int wxStackWalker::InitFrames(wxStackFrame *arr, size_t n, void **addresses, char **syminfo) |
232 | { | |
eaff0f0d VZ |
233 | // we need to launch addr2line tool to get this information and we need to |
234 | // have the program name for this | |
235 | wxString exepath = wxStackWalker::GetExePath(); | |
236 | if ( exepath.empty() ) | |
237 | { | |
a82c2299 RR |
238 | exepath = wxStandardPaths::Get().GetExecutablePath(); |
239 | if ( exepath.empty() ) | |
240 | { | |
241 | wxLogDebug(wxT("Cannot parse stack frame because the executable ") | |
242 | wxT("path could not be detected")); | |
243 | return 0; | |
244 | } | |
eaff0f0d VZ |
245 | } |
246 | ||
abf99aad VZ |
247 | // build the command line for executing addr2line or atos under OS X using |
248 | // char* directly to avoid the conversions from Unicode | |
249 | #ifdef __WXOSX__ | |
250 | int len = snprintf(g_buf, BUFSIZE, "atos -p %d", (int)getpid()); | |
251 | #else | |
8c8b50a9 | 252 | int len = snprintf(g_buf, BUFSIZE, "addr2line -C -f -e \"%s\"", (const char*) exepath.mb_str()); |
abf99aad | 253 | #endif |
a82c2299 RR |
254 | len = (len <= 0) ? strlen(g_buf) : len; // in case snprintf() is broken |
255 | for (size_t i=0; i<n; i++) | |
eaff0f0d | 256 | { |
a82c2299 RR |
257 | snprintf(&g_buf[len], BUFSIZE - len, " %p", addresses[i]); |
258 | len = strlen(g_buf); | |
ede3a6d6 VZ |
259 | } |
260 | ||
a82c2299 | 261 | //wxLogDebug(wxT("piping the command '%s'"), g_buf); // for debug only |
ede3a6d6 | 262 | |
a82c2299 RR |
263 | wxStdioPipe fp(g_buf, "r"); |
264 | if ( !fp ) | |
265 | return 0; | |
eaff0f0d | 266 | |
abf99aad VZ |
267 | // parse the output reusing the same buffer to avoid any big memory |
268 | // allocations which could fail if our program is in a bad state | |
a82c2299 | 269 | wxString name, filename; |
1d2c115e VZ |
270 | unsigned long line = 0, |
271 | curr = 0; | |
272 | for ( size_t i = 0; i < n; i++ ) | |
ede3a6d6 | 273 | { |
abf99aad VZ |
274 | #ifdef __WXOSX__ |
275 | wxString buffer; | |
276 | if ( !ReadLine(fp, i, &buffer) ) | |
277 | return false; | |
278 | ||
279 | line = 0; | |
280 | filename.clear(); | |
281 | ||
282 | // We can get back either the string in the following format: | |
283 | // | |
284 | // func(args) (in module) (file:line) | |
285 | // | |
286 | // or just the same address back if it couldn't be resolved. | |
287 | const size_t posIn = buffer.find("(in "); | |
288 | if ( posIn != wxString::npos ) | |
a82c2299 | 289 | { |
abf99aad VZ |
290 | name.assign(buffer, 0, posIn); |
291 | ||
292 | size_t posAt = buffer.find(") (", posIn + 3); | |
293 | if ( posAt != wxString::npos ) | |
294 | { | |
295 | posAt += 3; // Skip ") (" | |
296 | ||
297 | // Discard the two last characters which are ")\n" | |
298 | wxString location(buffer, posAt, buffer.length() - posAt - 2); | |
ede3a6d6 | 299 | |
abf99aad VZ |
300 | wxString linenum; |
301 | filename = location.BeforeFirst(':', &linenum); | |
302 | if ( !linenum.empty() ) | |
303 | linenum.ToULong(&line); | |
304 | } | |
a82c2299 | 305 | } |
abf99aad VZ |
306 | #else // !__WXOSX__ |
307 | // 1st line has function name | |
308 | if ( !ReadLine(fp, i, &name) ) | |
a82c2299 | 309 | return false; |
abf99aad VZ |
310 | |
311 | name = wxString::FromAscii(g_buf); | |
312 | name.RemoveLast(); // trailing newline | |
313 | ||
314 | if ( name == wxT("??") ) | |
315 | name.clear(); | |
a82c2299 RR |
316 | |
317 | // 2nd one -- the file/line info | |
abf99aad VZ |
318 | if ( !ReadLine(fp, i, &filename) ) |
319 | return false; | |
a82c2299 | 320 | |
abf99aad VZ |
321 | const size_t posColon = filename.find(wxT(':')); |
322 | if ( posColon != wxString::npos ) | |
323 | { | |
324 | // parse line number (it's ok if it fails, this will just leave | |
325 | // line at its current, invalid, 0 value) | |
326 | wxString(filename, posColon + 1, wxString::npos).ToULong(&line); | |
327 | ||
328 | // remove line number from 'filename' | |
329 | filename.erase(posColon); | |
330 | if ( filename == wxT("??") ) | |
331 | filename.clear(); | |
332 | } | |
333 | else | |
334 | { | |
335 | wxLogDebug(wxT("Unexpected addr2line format: \"%s\" - ") | |
336 | wxT("the semicolon is missing"), | |
337 | filename.c_str()); | |
eaff0f0d | 338 | } |
abf99aad | 339 | #endif // __WXOSX__/!__WXOSX__ |
a82c2299 | 340 | |
dd9b4376 RR |
341 | // now we've got enough info to initialize curr-th stack frame |
342 | // (at worst, only addresses[i] and syminfo[i] have been initialized, | |
343 | // but wxStackFrame::OnGetName may still be able to get function name): | |
344 | arr[curr++].Set(name, filename, syminfo[i], i, line, addresses[i]); | |
eaff0f0d | 345 | } |
eaff0f0d | 346 | |
a82c2299 RR |
347 | return curr; |
348 | } | |
eaff0f0d | 349 | |
a82c2299 | 350 | void wxStackWalker::Walk(size_t skip, size_t maxDepth) |
eaff0f0d | 351 | { |
a82c2299 RR |
352 | // read all frames required |
353 | SaveStack(maxDepth); | |
eaff0f0d | 354 | |
a82c2299 RR |
355 | // process them |
356 | ProcessFrames(skip); | |
eaff0f0d | 357 | |
a82c2299 RR |
358 | // cleanup |
359 | FreeStack(); | |
eaff0f0d VZ |
360 | } |
361 | ||
362 | #endif // wxUSE_STACKWALKER |