]> git.saurik.com Git - wxWidgets.git/blob - src/msw/crashrpt.cpp
ported rest of contrib to bakefile
[wxWidgets.git] / src / msw / crashrpt.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: msw/crashrpt.cpp
3 // Purpose: helpers for structured exception handling (SEH)
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 13.07.03
7 // RCS-ID: $Id$
8 // Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 /*
13 The code in this file is heavily based on Matt Pietrek's column from
14 the March 2002 issue of MSDN Magazine.
15 */
16
17 // ============================================================================
18 // declarations
19 // ============================================================================
20
21 // ----------------------------------------------------------------------------
22 // headers
23 // ----------------------------------------------------------------------------
24
25 // For compilers that support precompilation, includes "wx.h".
26 #include "wx/wxprec.h"
27
28 #ifdef __BORLANDC__
29 #pragma hdrstop
30 #endif
31
32 #if wxUSE_ON_FATAL_EXCEPTION
33
34 #ifndef WX_PRECOMP
35 #endif //WX_PRECOMP
36
37 #include "wx/longlong.h"
38 #include "wx/datetime.h"
39 #include "wx/dynload.h"
40
41 #include "wx/msw/crashrpt.h"
42
43 #include "wx/msw/wrapwin.h"
44 #include <imagehlp.h>
45 #include "wx/msw/private.h"
46
47 // we need to determine whether we have the declarations for the function in
48 // debughlp.dll version 5.81 (at least) and we check for DBHLPAPI to test this
49 //
50 // reasons:
51 // - VC6 version of imagehlp.h doesn't define it
52 // - VC7 one does
53 // - testing for compiler version doesn't work as you can install and use
54 // the new SDK headers with VC6
55 //
56 // in any case, the user may override by defining wxUSE_DBGHELP himself
57 #ifndef wxUSE_DBGHELP
58 #ifdef DBHLPAPI
59 #define wxUSE_DBGHELP 1
60 #else
61 #define wxUSE_DBGHELP 0
62 #endif
63 #endif
64
65 #if wxUSE_DBGHELP
66
67 // ----------------------------------------------------------------------------
68 // types of imagehlp.h functions
69 // ----------------------------------------------------------------------------
70
71 typedef DWORD (WINAPI *SymSetOptions_t)(DWORD);
72 typedef BOOL (WINAPI *SymInitialize_t)(HANDLE, LPSTR, BOOL);
73 typedef BOOL (WINAPI *StackWalk_t)(DWORD, HANDLE, HANDLE, LPSTACKFRAME,
74 LPVOID, PREAD_PROCESS_MEMORY_ROUTINE,
75 PFUNCTION_TABLE_ACCESS_ROUTINE,
76 PGET_MODULE_BASE_ROUTINE,
77 PTRANSLATE_ADDRESS_ROUTINE);
78 typedef BOOL (WINAPI *SymFromAddr_t)(HANDLE, DWORD64, PDWORD64, PSYMBOL_INFO);
79 typedef LPVOID (WINAPI *SymFunctionTableAccess_t)(HANDLE, DWORD);
80 typedef DWORD (WINAPI *SymGetModuleBase_t)(HANDLE, DWORD);
81 typedef BOOL (WINAPI *SymGetLineFromAddr_t)(HANDLE, DWORD,
82 PDWORD, PIMAGEHLP_LINE);
83 typedef BOOL (WINAPI *SymSetContext_t)(HANDLE, PIMAGEHLP_STACK_FRAME,
84 PIMAGEHLP_CONTEXT);
85 typedef BOOL (WINAPI *SymEnumSymbols_t)(HANDLE, ULONG64, PCSTR,
86 PSYM_ENUMERATESYMBOLS_CALLBACK, PVOID);
87 typedef BOOL (WINAPI *SymGetTypeInfo_t)(HANDLE, DWORD64, ULONG,
88 IMAGEHLP_SYMBOL_TYPE_INFO, PVOID);
89
90 // ----------------------------------------------------------------------------
91 // constant
92 // ----------------------------------------------------------------------------
93
94 // Stolen from CVCONST.H in the DIA 2.0 SDK
95 enum BasicType
96 {
97 BASICTYPE_NOTYPE = 0,
98 BASICTYPE_VOID = 1,
99 BASICTYPE_CHAR = 2,
100 BASICTYPE_WCHAR = 3,
101 BASICTYPE_INT = 6,
102 BASICTYPE_UINT = 7,
103 BASICTYPE_FLOAT = 8,
104 BASICTYPE_BCD = 9,
105 BASICTYPE_BOOL = 10,
106 BASICTYPE_LONG = 13,
107 BASICTYPE_ULONG = 14,
108 BASICTYPE_CURRENCY = 25,
109 BASICTYPE_DATE = 26,
110 BASICTYPE_VARIANT = 27,
111 BASICTYPE_COMPLEX = 28,
112 BASICTYPE_BIT = 29,
113 BASICTYPE_BSTR = 30,
114 BASICTYPE_HRESULT = 31
115 };
116
117 // Same as above
118 enum SymbolTag
119 {
120 SYMBOL_TAG_NULL,
121 SYMBOL_TAG_FUNCTION = 5,
122 SYMBOL_TAG_DATA = 7,
123 SYMBOL_TAG_PUBLIC = 10, // appears in .DBGs
124 SYMBOL_TAG_UDT,
125 SYMBOL_TAG_ENUM,
126 SYMBOL_TAG_FUNCTION_TYPE,
127 SYMBOL_TAG_POINTER_TYPE,
128 SYMBOL_TAG_ARRAY_TYPE,
129 SYMBOL_TAG_BASE_TYPE,
130 SYMBOL_TAG_TYPEDEF,
131 SYMBOL_TAG_BASECLASS
132 };
133
134 #endif // wxUSE_DBGHELP
135
136 // ----------------------------------------------------------------------------
137 // classes
138 // ----------------------------------------------------------------------------
139
140 // the real crash report generator
141 class wxCrashReportImpl
142 {
143 public:
144 wxCrashReportImpl(const wxChar *filename);
145
146 bool Generate(int flags);
147
148 ~wxCrashReportImpl()
149 {
150 if ( m_hFile != INVALID_HANDLE_VALUE )
151 {
152 ::CloseHandle(m_hFile);
153 }
154 }
155
156 private:
157 // formatted output to m_hFile
158 void Output(const wxChar *format, ...);
159
160 // output end of line
161 void OutputEndl() { Output(_T("\r\n")); }
162
163 #if wxUSE_DBGHELP
164 // translate exception code to its symbolic name
165 static wxString GetExceptionString(DWORD dwCode);
166
167 // return the type from "type index"
168 static BasicType GetBasicType(DWORD64 modBase, DWORD typeIndex);
169
170 // return the name for the type index
171 static wxString GetSymbolName(DWORD64 modBase, DWORD dwTypeIndex);
172
173 // return the string representation of the variable value
174 static wxString FormatSimpleValue(BasicType bt,
175 DWORD64 length,
176 PVOID pAddress);
177
178 // return string representation of a struct field (which may itself be a
179 // struct, of course)
180 static wxString FormatField(DWORD64 modBase,
181 DWORD dwTypeIndex,
182 void *pVariable,
183 unsigned level);
184
185 // show the name and value of the given symbol
186 static wxString FormatSymbol(PSYMBOL_INFO pSym, STACKFRAME *sf);
187
188 // show value described by SYMBOL_INFO located at pVariable
189 static wxString FormatAnyValue(PSYMBOL_INFO pSym, void *pVariable);
190
191 // show value of possibly complex (user-defined) type
192 static wxString FormatUDT(DWORD64 modBase,
193 DWORD dwTypeIndex,
194 void *pVariable,
195 unsigned level = 0);
196
197 // outputs information about the given symbol
198 void OutputSymbol(PSYMBOL_INFO pSymInfo, STACKFRAME *sf);
199
200 // load all the functions we need from dbghelp.dll, return true if all ok
201 bool ResolveSymFunctions(const wxDynamicLibrary& dllDbgHelp);
202
203 // map address to module (and also section:offset), retunr true if ok
204 static bool GetLogicalAddress(PVOID addr,
205 PTSTR szModule,
206 DWORD len,
207 DWORD& section,
208 DWORD& offset);
209
210 // callback used with SymEnumSymbols() to process all variables
211 static BOOL CALLBACK EnumerateSymbolsCallback(PSYMBOL_INFO pSymInfo,
212 ULONG SymbolSize,
213 PVOID UserContext);
214
215
216 // show the general information about exception which should be always
217 // available
218 //
219 // returns the module of the handle where the crash occured
220 HANDLE OutputBasicContext(EXCEPTION_RECORD *pExceptionRecord, CONTEXT *pCtx);
221
222 // output the call stack and local variables values
223 void OutputStack(const CONTEXT *pCtx, int flags);
224
225 // output the global variables values
226 void OutputGlobals(HANDLE hModuleCrash);
227
228
229 // the current stack frame (may be NULL)
230 STACKFRAME *m_sfCurrent;
231
232
233 // dynamically loaded dbghelp.dll functions
234 #define DECLARE_SYM_FUNCTION(func) static func ## _t func
235
236 DECLARE_SYM_FUNCTION(SymSetOptions);
237 DECLARE_SYM_FUNCTION(SymInitialize);
238 DECLARE_SYM_FUNCTION(StackWalk);
239 DECLARE_SYM_FUNCTION(SymFromAddr);
240 DECLARE_SYM_FUNCTION(SymFunctionTableAccess);
241 DECLARE_SYM_FUNCTION(SymGetModuleBase);
242 DECLARE_SYM_FUNCTION(SymGetLineFromAddr);
243 DECLARE_SYM_FUNCTION(SymSetContext);
244 DECLARE_SYM_FUNCTION(SymEnumSymbols);
245 DECLARE_SYM_FUNCTION(SymGetTypeInfo);
246 #endif // wxUSE_DBGHELP
247
248 // the handle of the report file
249 HANDLE m_hFile;
250 };
251
252 // ----------------------------------------------------------------------------
253 // globals
254 // ----------------------------------------------------------------------------
255
256 // global pointer to exception information, only valid inside OnFatalException
257 extern WXDLLIMPEXP_BASE EXCEPTION_POINTERS *wxGlobalSEInformation = NULL;
258
259
260 // flag telling us whether the application wants to handle exceptions at all
261 static bool gs_handleExceptions = false;
262
263 // the file name where the report about exception is written
264 static wxChar gs_reportFilename[MAX_PATH];
265
266 // ============================================================================
267 // implementation
268 // ============================================================================
269
270 #if wxUSE_DBGHELP
271
272 #define DEFINE_SYM_FUNCTION(func) func ## _t wxCrashReportImpl::func = 0
273
274 DEFINE_SYM_FUNCTION(SymSetOptions);
275 DEFINE_SYM_FUNCTION(SymInitialize);
276 DEFINE_SYM_FUNCTION(StackWalk);
277 DEFINE_SYM_FUNCTION(SymFromAddr);
278 DEFINE_SYM_FUNCTION(SymFunctionTableAccess);
279 DEFINE_SYM_FUNCTION(SymGetModuleBase);
280 DEFINE_SYM_FUNCTION(SymGetLineFromAddr);
281 DEFINE_SYM_FUNCTION(SymSetContext);
282 DEFINE_SYM_FUNCTION(SymEnumSymbols);
283 DEFINE_SYM_FUNCTION(SymGetTypeInfo);
284
285 #undef DEFINE_SYM_FUNCTION
286
287 #endif // wxUSE_DBGHELP
288
289 // ----------------------------------------------------------------------------
290 // wxCrashReportImpl
291 // ----------------------------------------------------------------------------
292
293 wxCrashReportImpl::wxCrashReportImpl(const wxChar *filename)
294 {
295 #if wxUSE_DBGHELP
296 m_sfCurrent = NULL;
297 #endif // wxUSE_DBGHELP
298
299 m_hFile = ::CreateFile
300 (
301 filename,
302 GENERIC_WRITE,
303 0, // no sharing
304 NULL, // default security
305 CREATE_ALWAYS,
306 FILE_FLAG_WRITE_THROUGH,
307 NULL // no template file
308 );
309 }
310
311 void wxCrashReportImpl::Output(const wxChar *format, ...)
312 {
313 va_list argptr;
314 va_start(argptr, format);
315
316 DWORD cbWritten;
317
318 wxString s = wxString::FormatV(format, argptr);
319 ::WriteFile(m_hFile, s, s.length() * sizeof(wxChar), &cbWritten, 0);
320
321 va_end(argptr);
322 }
323
324 #if wxUSE_DBGHELP
325
326 bool
327 wxCrashReportImpl::GetLogicalAddress(PVOID addr,
328 PTSTR szModule,
329 DWORD len,
330 DWORD& section,
331 DWORD& offset)
332 {
333 MEMORY_BASIC_INFORMATION mbi;
334
335 if ( !::VirtualQuery(addr, &mbi, sizeof(mbi)) )
336 return false;
337
338 DWORD hMod = (DWORD)mbi.AllocationBase;
339
340 if ( !::GetModuleFileName((HMODULE)hMod, szModule, len) )
341 return false;
342
343 // Point to the DOS header in memory
344 PIMAGE_DOS_HEADER pDosHdr = (PIMAGE_DOS_HEADER)hMod;
345
346 // From the DOS header, find the NT (PE) header
347 PIMAGE_NT_HEADERS pNtHdr = (PIMAGE_NT_HEADERS)(hMod + pDosHdr->e_lfanew);
348
349 PIMAGE_SECTION_HEADER pSection = IMAGE_FIRST_SECTION( pNtHdr );
350
351 DWORD rva = (DWORD)addr - hMod; // RVA is offset from module load address
352
353 // Iterate through the section table, looking for the one that encompasses
354 // the linear address.
355 const DWORD nSections = pNtHdr->FileHeader.NumberOfSections;
356 for ( DWORD i = 0; i < nSections; i++, pSection++ )
357 {
358 DWORD sectionStart = pSection->VirtualAddress;
359 DWORD sectionEnd = sectionStart
360 + max(pSection->SizeOfRawData, pSection->Misc.VirtualSize);
361
362 // Is the address in this section?
363 if ( (rva >= sectionStart) && (rva <= sectionEnd) )
364 {
365 // Yes, address is in the section. Calculate section and offset,
366 // and store in the "section" & "offset" params, which were
367 // passed by reference.
368 section = i + 1;
369 offset = rva - sectionStart;
370
371 return true;
372 }
373 }
374
375 // failed to map to logical address...
376 return false;
377 }
378
379 /* static */ BasicType
380 wxCrashReportImpl::GetBasicType(DWORD64 modBase, DWORD typeIndex)
381 {
382 const HANDLE hProcess = GetCurrentProcess();
383
384 // try the index we have
385 BasicType bt;
386 if ( SymGetTypeInfo(hProcess, modBase, typeIndex, TI_GET_BASETYPE, &bt) )
387 {
388 return bt;
389 }
390
391 // if failed, try to get the "real" typeid first
392 DWORD typeId;
393 if ( SymGetTypeInfo(hProcess, modBase, typeIndex, TI_GET_TYPEID, &typeId)
394 &&
395 (typeId != typeIndex &&
396 SymGetTypeInfo(hProcess, modBase, typeId, TI_GET_BASETYPE, &bt)) )
397 {
398 return bt;
399 }
400
401 return BASICTYPE_NOTYPE;
402 }
403
404 /* static */ wxString
405 wxCrashReportImpl::FormatSimpleValue(BasicType bt,
406 DWORD64 length,
407 PVOID pAddress)
408 {
409 wxString s;
410 s.reserve(256);
411
412 // Format appropriately (assuming it's a 1, 2, or 4 bytes (!!!)
413 if ( length == 1 )
414 {
415 s.Printf(_T("%#04x"), *(PBYTE)pAddress);
416 }
417 else if ( length == 2 )
418 {
419 s.Printf(_T("%#06x"), *(PWORD)pAddress);
420 }
421 else if ( length == 4 )
422 {
423 bool handled = false;
424
425 if ( bt == BASICTYPE_FLOAT )
426 {
427 s.Printf(_T("%f"), *(PFLOAT)pAddress);
428
429 handled = true;
430 }
431 else if ( bt == BASICTYPE_CHAR )
432 {
433 static const size_t NUM_CHARS = 32;
434
435 const char * const pc = *(PSTR *)pAddress;
436 if ( !::IsBadStringPtrA(pc, NUM_CHARS) )
437 {
438 s << _T('"') << wxString(pc, wxConvLibc, NUM_CHARS) << _T('"');
439
440 handled = true;
441 }
442 }
443
444 if ( !handled )
445 {
446 // treat just as an opaque DWORD
447 s.Printf(_T("%#x"), *(PDWORD)pAddress);
448 }
449 }
450 else if ( length == 8 )
451 {
452 if ( bt == BASICTYPE_FLOAT )
453 {
454 s.Printf(_T("%lf"), *(double *)pAddress);
455 }
456 else // opaque 64 bit value
457 {
458 s.Printf(_T("%#" wxLongLongFmtSpec _T("x")), *(PDWORD *)pAddress);
459 }
460 }
461
462 return s;
463 }
464
465 /* static */
466 wxString wxCrashReportImpl::GetSymbolName(DWORD64 modBase, DWORD dwTypeIndex)
467 {
468 wxString s;
469
470 WCHAR *pwszTypeName;
471 if ( SymGetTypeInfo
472 (
473 GetCurrentProcess(),
474 modBase,
475 dwTypeIndex,
476 TI_GET_SYMNAME,
477 &pwszTypeName
478 ) )
479 {
480 s = wxConvCurrent->cWC2WX(pwszTypeName);
481
482 ::LocalFree(pwszTypeName);
483 }
484
485 return s;
486 }
487
488 // this is called for the struct members/base classes
489 wxString
490 wxCrashReportImpl::FormatField(DWORD64 modBase,
491 DWORD dwTypeIndex,
492 void *pVariable,
493 unsigned level)
494 {
495 wxString s;
496
497 const HANDLE hProcess = GetCurrentProcess();
498
499 DWORD dwTag = 0;
500 SymGetTypeInfo(hProcess, modBase, dwTypeIndex, TI_GET_SYMTAG, &dwTag);
501
502 switch ( dwTag )
503 {
504 case SYMBOL_TAG_UDT:
505 case SYMBOL_TAG_BASECLASS:
506 s = FormatUDT(modBase, dwTypeIndex, pVariable, level);
507 break;
508
509 case SYMBOL_TAG_FUNCTION:
510 // don't show
511 break;
512
513 default:
514 // try to treat all the rest as data even though it's not clear if
515 // it's really a good idea...
516
517 // Get the offset of the child member, relative to its parent
518 DWORD dwMemberOffset = 0;
519 SymGetTypeInfo(hProcess, modBase, dwTypeIndex,
520 TI_GET_OFFSET, &dwMemberOffset);
521
522 // Get the real "TypeId" of the child. We need this for the
523 // SymGetTypeInfo(TI_GET_LENGTH) call below.
524 DWORD typeId;
525 if ( !SymGetTypeInfo(hProcess, modBase, dwTypeIndex,
526 TI_GET_TYPEID, &typeId) )
527 {
528 typeId = dwTypeIndex;
529 }
530
531 // Get the size of the child member
532 ULONG64 size;
533 SymGetTypeInfo(hProcess, modBase, typeId, TI_GET_LENGTH, &size);
534
535 // Calculate the address of the member
536 DWORD_PTR dwFinalOffset = (DWORD_PTR)pVariable + dwMemberOffset;
537
538 BasicType basicType = GetBasicType(modBase, dwTypeIndex);
539
540 s = FormatSimpleValue(basicType, size, (PVOID)dwFinalOffset);
541 break;
542
543 }
544
545 if ( s.empty() )
546 {
547 // don't show if no value -- what for?
548 return s;
549 }
550
551 return wxString(_T('\t'), level + 1) +
552 GetSymbolName(modBase, dwTypeIndex) +
553 _T(" = ") + s + _T("\r\n");
554 }
555
556 // If it's a user defined type (UDT), recurse through its members until we're
557 // at fundamental types.
558 wxString
559 wxCrashReportImpl::FormatUDT(DWORD64 modBase,
560 DWORD dwTypeIndex,
561 void *pVariable,
562 unsigned level)
563 {
564 wxString s;
565 s.reserve(512);
566 s = GetSymbolName(modBase, dwTypeIndex) + _T(" {\r\n");
567
568 const HANDLE hProcess = GetCurrentProcess();
569
570 // Determine how many children this type has.
571 DWORD dwChildrenCount = 0;
572 SymGetTypeInfo(hProcess, modBase, dwTypeIndex, TI_GET_CHILDRENCOUNT,
573 &dwChildrenCount);
574
575 // Prepare to get an array of "TypeIds", representing each of the children.
576 TI_FINDCHILDREN_PARAMS *children = (TI_FINDCHILDREN_PARAMS *)
577 malloc(sizeof(TI_FINDCHILDREN_PARAMS) +
578 (dwChildrenCount - 1)*sizeof(ULONG));
579 if ( !children )
580 return s;
581
582 children->Count = dwChildrenCount;
583 children->Start = 0;
584
585 // Get the array of TypeIds, one for each child type
586 if ( !SymGetTypeInfo(hProcess, modBase, dwTypeIndex, TI_FINDCHILDREN,
587 children) )
588 {
589 return s;
590 }
591
592 // Iterate through all children
593 for ( unsigned i = 0; i < dwChildrenCount; i++ )
594 {
595 s += FormatField(modBase, children->ChildId[i], pVariable, level + 1);
596 }
597
598 free(children);
599
600 s << wxString(_T('\t'), level + 1) << _T('}');
601
602 return s;
603 }
604
605 // return the string containing the symbol of the given symbol
606 /* static */ wxString
607 wxCrashReportImpl::FormatAnyValue(PSYMBOL_INFO pSym, void *pVariable)
608 {
609 DWORD dwTag = 0;
610 SymGetTypeInfo(GetCurrentProcess(), pSym->ModBase, pSym->TypeIndex,
611 TI_GET_SYMTAG, &dwTag);
612
613 wxString s;
614 switch ( dwTag )
615 {
616 case SYMBOL_TAG_FUNCTION:
617 break;
618
619 case SYMBOL_TAG_UDT:
620 case SYMBOL_TAG_BASECLASS:
621 // show UDT recursively
622 s = FormatUDT(pSym->ModBase, pSym->TypeIndex, pVariable);
623 break;
624
625 default:
626 // variable of simple type (but could be array which we don't
627 // handle correctly yet...), format it using its type and size
628 BasicType bt = GetBasicType(pSym->ModBase, pSym->TypeIndex);
629
630 s = FormatSimpleValue(bt, pSym->Size, pVariable);
631 break;
632
633 }
634
635 return s;
636 }
637
638 // display contents and type of the given variable
639 /* static */ wxString
640 wxCrashReportImpl::FormatSymbol(PSYMBOL_INFO pSym, STACKFRAME *sf)
641 {
642 wxString s;
643
644 if ( pSym->Tag == SYMBOL_TAG_FUNCTION )
645 {
646 // If it's a function, don't do anything.
647 return s;
648 }
649
650 if ( pSym->Flags & IMAGEHLP_SYMBOL_INFO_REGISTER )
651 {
652 // Don't try to report register variable
653 return s;
654 }
655
656 s.reserve(512);
657
658 // Indicate if the variable is a local or parameter
659 if ( pSym->Flags & IMAGEHLP_SYMBOL_INFO_PARAMETER )
660 s += _T("\t[param] ");
661 else if ( pSym->Flags & IMAGEHLP_SYMBOL_INFO_LOCAL )
662 s += _T("\t[local] ");
663
664 // Will point to the variable's data in memory
665 DWORD_PTR pVariable = 0;
666
667 if ( (pSym->Flags & IMAGEHLP_SYMBOL_INFO_REGRELATIVE) && sf )
668 {
669 pVariable = sf->AddrFrame.Offset;
670 pVariable += (DWORD_PTR)pSym->Address;
671 }
672 else // It must be a global variable
673 {
674 pVariable = (DWORD_PTR)pSym->Address;
675 }
676
677 s << wxString(pSym->Name, wxConvLibc)
678 << _T(" = ")
679 << FormatAnyValue(pSym, (PVOID)pVariable);
680
681 return s;
682 }
683
684 void
685 wxCrashReportImpl::OutputSymbol(PSYMBOL_INFO pSymInfo, STACKFRAME *sf)
686 {
687 wxString s = FormatSymbol(pSymInfo, sf);
688 if ( !s.empty() )
689 {
690 Output(_T("%s\r\n"), s.c_str());
691 }
692 //else: not an interesting symbol
693 }
694
695 // callback for SymEnumSymbols()
696 /* static */
697 BOOL CALLBACK
698 wxCrashReportImpl::EnumerateSymbolsCallback(PSYMBOL_INFO pSymInfo,
699 ULONG WXUNUSED(SymbolSize),
700 PVOID UserContext)
701 {
702 wxCrashReportImpl *self = (wxCrashReportImpl *)UserContext;
703
704 __try
705 {
706 self->OutputSymbol(pSymInfo, self->m_sfCurrent);
707 }
708 __except ( EXCEPTION_EXECUTE_HANDLER )
709 {
710 self->Output(_T("Can't process symbol %hs\r\n"), pSymInfo->Name);
711 }
712
713 // continue with enumeration
714 return true;
715 }
716
717 HANDLE
718 wxCrashReportImpl::OutputBasicContext(EXCEPTION_RECORD *pExceptionRecord,
719 CONTEXT *pCtx)
720 {
721 // First print information about the type of fault
722 const DWORD dwCode = pExceptionRecord->ExceptionCode;
723 Output(_T("Exception code: %s (%#10x)\r\n"),
724 GetExceptionString(dwCode).c_str(), dwCode);
725
726 // Now print information about where the fault occured
727 TCHAR szFaultingModule[MAX_PATH];
728 DWORD section,
729 offset;
730 void * const pExceptionAddress = pExceptionRecord->ExceptionAddress;
731 if ( !GetLogicalAddress(pExceptionAddress,
732 szFaultingModule,
733 WXSIZEOF(szFaultingModule),
734 section, offset) )
735 {
736 section =
737 offset = 0;
738
739 wxStrcpy(szFaultingModule, _T("<< unknown >>"));
740 }
741
742 Output(_T("Fault address: %08x %02x:%08x %s\r\n"),
743 pExceptionAddress, section, offset, szFaultingModule);
744
745 #ifdef _M_IX86
746 // Show the registers
747 Output( _T("\r\nRegisters:\r\n") );
748
749 Output(_T("EAX: %08x EBX: %08x ECX: %08x EDX: %08x ESI: %08x EDI: %08x\r\n"),
750 pCtx->Eax, pCtx->Ebx, pCtx->Ecx, pCtx->Edx, pCtx->Esi, pCtx->Edi);
751
752 Output(_T("CS:EIP: %04x:%08x SS:ESP: %04x:%08x EBP: %08x\r\n"),
753 pCtx->SegCs, pCtx->Eip, pCtx->SegSs, pCtx->Esp, pCtx->Ebp );
754 Output(_T("DS: %04x ES: %04x FS: %04x GS: %04x\r\n"),
755 pCtx->SegDs, pCtx->SegEs, pCtx->SegFs, pCtx->SegGs);
756 Output(_T("Flags: %08x\r\n"), pCtx->EFlags );
757 #endif // _M_IX86
758
759 return ::GetModuleHandle(szFaultingModule);
760 }
761
762 void wxCrashReportImpl::OutputStack(const CONTEXT *pCtx, int flags)
763 {
764 enum
765 {
766 Output_Stack,
767 Output_Locals,
768 Output_Max
769 #ifndef _M_IX86
770 // can't show locals under other architectures
771 = Output_Locals
772 #endif
773 };
774
775 for ( int step = 0; step < Output_Max; step++ )
776 {
777 // don't do things we're not asked for
778 if ( (step == Output_Stack) && !(flags & wxCRASH_REPORT_STACK) ||
779 (step == Output_Locals) && !(flags & wxCRASH_REPORT_LOCALS) )
780 {
781 continue;
782 }
783
784 // the context is going to be modified below so make a copy
785 CONTEXT ctx = *pCtx;
786
787 Output(_T("\r\n%s\r\n")
788 _T(" # Address Frame Function SourceFile\r\n"),
789 step == Output_Stack ? _T("Call stack") : _T("Local variables"));
790
791 DWORD dwMachineType = 0;
792
793 STACKFRAME sf;
794 wxZeroMemory(sf);
795
796 #ifdef _M_IX86
797 // Initialize the STACKFRAME structure for the first call. This is
798 // only necessary for Intel CPUs, and isn't mentioned in the
799 // documentation.
800 sf.AddrPC.Offset = ctx.Eip;
801 sf.AddrPC.Mode = AddrModeFlat;
802 sf.AddrStack.Offset = ctx.Esp;
803 sf.AddrStack.Mode = AddrModeFlat;
804 sf.AddrFrame.Offset = ctx.Ebp;
805 sf.AddrFrame.Mode = AddrModeFlat;
806
807 dwMachineType = IMAGE_FILE_MACHINE_I386;
808 #endif // _M_IX86
809
810 const HANDLE hProcess = GetCurrentProcess();
811 const HANDLE hThread = GetCurrentThread();
812
813 // first show just the call stack
814 int frame = 0;
815 for ( ;; )
816 {
817 // Get the next stack frame
818 if ( !StackWalk(dwMachineType,
819 hProcess,
820 hThread,
821 &sf,
822 &ctx,
823 0,
824 SymFunctionTableAccess,
825 SymGetModuleBase,
826 0) )
827 {
828 break;
829 }
830
831 // Basic sanity check to make sure the frame is OK.
832 if ( !sf.AddrFrame.Offset )
833 break;
834
835 Output(_T("%2d %08x %08x "),
836 frame++, sf.AddrPC.Offset, sf.AddrFrame.Offset);
837
838 // Get the name of the function for this stack frame entry
839 BYTE symbolBuffer[ sizeof(SYMBOL_INFO) + 1024 ];
840 PSYMBOL_INFO pSymbol = (PSYMBOL_INFO)symbolBuffer;
841 pSymbol->SizeOfStruct = sizeof(symbolBuffer);
842 pSymbol->MaxNameLen = 1024;
843
844 // Displacement of the input address, relative to the start of the
845 // symbol
846 DWORD64 symDisplacement = 0;
847
848 if ( SymFromAddr(hProcess, sf.AddrPC.Offset,
849 &symDisplacement,pSymbol) )
850 {
851 Output(_T("%hs()+%#") wxLongLongFmtSpec _T("x"),
852 pSymbol->Name, symDisplacement);
853 }
854 else // No symbol found. Print out the logical address instead.
855 {
856 TCHAR szModule[MAX_PATH];
857 DWORD section,
858 offset;
859
860 if ( !GetLogicalAddress((PVOID)sf.AddrPC.Offset,
861 szModule, sizeof(szModule),
862 section, offset) )
863 {
864 szModule[0] = _T('\0');
865 section =
866 offset = 0;
867 }
868
869 Output(_T("%04x:%08x %s"), section, offset, szModule);
870 }
871
872 // Get the source line for this stack frame entry
873 IMAGEHLP_LINE lineInfo = { sizeof(IMAGEHLP_LINE) };
874 DWORD dwLineDisplacement;
875 if ( SymGetLineFromAddr(hProcess, sf.AddrPC.Offset,
876 &dwLineDisplacement, &lineInfo ))
877 {
878 Output(_T(" %s line %u"),
879 lineInfo.FileName, lineInfo.LineNumber);
880 }
881
882 OutputEndl();
883
884
885 #ifdef _M_IX86
886 // on the second iteration also show the local variables and
887 // parameters
888 if ( step == Output_Locals )
889 {
890 // Use SymSetContext to get just the locals/params for this
891 // frame
892 IMAGEHLP_STACK_FRAME imagehlpStackFrame;
893 imagehlpStackFrame.InstructionOffset = sf.AddrPC.Offset;
894 SymSetContext(hProcess, &imagehlpStackFrame, 0);
895
896 // Enumerate the locals/parameters
897 m_sfCurrent = &sf;
898 SymEnumSymbols(hProcess, 0, 0, EnumerateSymbolsCallback, this);
899
900 OutputEndl();
901 }
902 #endif // _M_IX86
903 }
904 }
905 }
906
907 void wxCrashReportImpl::OutputGlobals(HANDLE hModule)
908 {
909 #ifdef _M_IX86
910 Output(_T("\r\nGlobal variables:\r\n"));
911
912 m_sfCurrent = NULL;
913 SymEnumSymbols(::GetCurrentProcess(), (DWORD64)hModule, NULL,
914 EnumerateSymbolsCallback, this);
915 #endif // _M_IX86
916 }
917
918 bool wxCrashReportImpl::ResolveSymFunctions(const wxDynamicLibrary& dllDbgHelp)
919 {
920 #define LOAD_SYM_FUNCTION(name) \
921 name = (name ## _t) dllDbgHelp.GetSymbol(_T(#name)); \
922 if ( !name ) \
923 { \
924 Output(_T("\r\nFunction ") _T(#name) \
925 _T("() not found.\r\n")); \
926 return false; \
927 }
928
929 LOAD_SYM_FUNCTION(SymSetOptions);
930 LOAD_SYM_FUNCTION(SymInitialize);
931 LOAD_SYM_FUNCTION(StackWalk);
932 LOAD_SYM_FUNCTION(SymFromAddr);
933 LOAD_SYM_FUNCTION(SymFunctionTableAccess);
934 LOAD_SYM_FUNCTION(SymGetModuleBase);
935 LOAD_SYM_FUNCTION(SymGetLineFromAddr);
936 LOAD_SYM_FUNCTION(SymSetContext);
937 LOAD_SYM_FUNCTION(SymEnumSymbols);
938 LOAD_SYM_FUNCTION(SymGetTypeInfo);
939
940 #undef LOAD_SYM_FUNCTION
941
942 return true;
943 }
944
945 /* static */
946 wxString wxCrashReportImpl::GetExceptionString(DWORD dwCode)
947 {
948 wxString s;
949
950 #define CASE_EXCEPTION( x ) case EXCEPTION_##x: s = _T(#x); break
951
952 switch ( dwCode )
953 {
954 CASE_EXCEPTION(ACCESS_VIOLATION);
955 CASE_EXCEPTION(DATATYPE_MISALIGNMENT);
956 CASE_EXCEPTION(BREAKPOINT);
957 CASE_EXCEPTION(SINGLE_STEP);
958 CASE_EXCEPTION(ARRAY_BOUNDS_EXCEEDED);
959 CASE_EXCEPTION(FLT_DENORMAL_OPERAND);
960 CASE_EXCEPTION(FLT_DIVIDE_BY_ZERO);
961 CASE_EXCEPTION(FLT_INEXACT_RESULT);
962 CASE_EXCEPTION(FLT_INVALID_OPERATION);
963 CASE_EXCEPTION(FLT_OVERFLOW);
964 CASE_EXCEPTION(FLT_STACK_CHECK);
965 CASE_EXCEPTION(FLT_UNDERFLOW);
966 CASE_EXCEPTION(INT_DIVIDE_BY_ZERO);
967 CASE_EXCEPTION(INT_OVERFLOW);
968 CASE_EXCEPTION(PRIV_INSTRUCTION);
969 CASE_EXCEPTION(IN_PAGE_ERROR);
970 CASE_EXCEPTION(ILLEGAL_INSTRUCTION);
971 CASE_EXCEPTION(NONCONTINUABLE_EXCEPTION);
972 CASE_EXCEPTION(STACK_OVERFLOW);
973 CASE_EXCEPTION(INVALID_DISPOSITION);
974 CASE_EXCEPTION(GUARD_PAGE);
975 CASE_EXCEPTION(INVALID_HANDLE);
976
977 default:
978 // unknown exception, ask NTDLL for the name
979 if ( !::FormatMessage
980 (
981 FORMAT_MESSAGE_IGNORE_INSERTS |
982 FORMAT_MESSAGE_FROM_HMODULE,
983 ::GetModuleHandle(_T("NTDLL.DLL")),
984 dwCode,
985 0,
986 wxStringBuffer(s, 1024),
987 1024,
988 0
989 ) )
990 {
991 s = _T("UNKNOWN_EXCEPTION");
992 }
993 }
994
995 #undef CASE_EXCEPTION
996
997 return s;
998 }
999
1000 #endif // wxUSE_DBGHELP
1001
1002 // Remove warning
1003 #if wxUSE_DBGHELP
1004 #define _WXUNUSED(x) x
1005 #else
1006 #define _WXUNUSED WXUNUSED
1007 #endif
1008
1009 bool wxCrashReportImpl::Generate(int _WXUNUSED(flags))
1010 {
1011 if ( m_hFile == INVALID_HANDLE_VALUE )
1012 return false;
1013
1014 #if wxUSE_DBGHELP
1015 if ( !wxGlobalSEInformation )
1016 return false;
1017
1018 PEXCEPTION_RECORD pExceptionRecord = wxGlobalSEInformation->ExceptionRecord;
1019 PCONTEXT pCtx = wxGlobalSEInformation->ContextRecord;
1020
1021 if ( !pExceptionRecord || !pCtx )
1022 return false;
1023
1024 HANDLE hModuleCrash = OutputBasicContext(pExceptionRecord, pCtx);
1025
1026 // for everything else we need dbghelp.dll
1027 wxDynamicLibrary dllDbgHelp(_T("dbghelp.dll"), wxDL_VERBATIM);
1028 if ( dllDbgHelp.IsLoaded() )
1029 {
1030 if ( ResolveSymFunctions(dllDbgHelp) )
1031 {
1032 SymSetOptions(SYMOPT_DEFERRED_LOADS | SYMOPT_UNDNAME);
1033
1034 // Initialize DbgHelp
1035 if ( SymInitialize(GetCurrentProcess(), NULL, TRUE /* invade */) )
1036 {
1037 OutputStack(pCtx, flags);
1038
1039 if ( hModuleCrash && (flags & wxCRASH_REPORT_GLOBALS) )
1040 {
1041 OutputGlobals(hModuleCrash);
1042 }
1043
1044 return true;
1045 }
1046 }
1047 else
1048 {
1049 Output(_T("Please update your dbghelp.dll version, ")
1050 _T("at least version 5.1 is needed!\r\n"));
1051 }
1052 }
1053 else
1054 {
1055 Output(_T("Please install dbghelp.dll available free of charge ")
1056 _T("from Microsoft to get more detailed crash information!"));
1057 }
1058
1059 Output(_T("\r\nLatest dbghelp.dll is available at ")
1060 _T("http://www.microsoft.com/whdc/ddk/debugging/\r\n"));
1061
1062 #else // !wxUSE_DBGHELP
1063 Output(_T("Support for crash report generation was not included ")
1064 _T("in this wxWindows version."));
1065 #endif // wxUSE_DBGHELP/!wxUSE_DBGHELP
1066
1067 return true;
1068 }
1069
1070 // ----------------------------------------------------------------------------
1071 // wxCrashReport
1072 // ----------------------------------------------------------------------------
1073
1074 /* static */
1075 void wxCrashReport::SetFileName(const wxChar *filename)
1076 {
1077 wxStrncpy(gs_reportFilename, filename, WXSIZEOF(gs_reportFilename) - 1);
1078 gs_reportFilename[WXSIZEOF(gs_reportFilename) - 1] = _T('\0');
1079 }
1080
1081 /* static */
1082 const wxChar *wxCrashReport::GetFileName()
1083 {
1084 return gs_reportFilename;
1085 }
1086
1087 /* static */
1088 bool wxCrashReport::Generate(int flags)
1089 {
1090 wxCrashReportImpl impl(gs_reportFilename);
1091
1092 return impl.Generate(flags);
1093 }
1094
1095 // ----------------------------------------------------------------------------
1096 // wxApp::OnFatalException() support
1097 // ----------------------------------------------------------------------------
1098
1099 bool wxHandleFatalExceptions(bool doit)
1100 {
1101 // assume this can only be called from the main thread
1102 gs_handleExceptions = doit;
1103
1104 if ( doit )
1105 {
1106 // try to find a place where we can put out report file later
1107 if ( !::GetTempPath
1108 (
1109 WXSIZEOF(gs_reportFilename),
1110 gs_reportFilename
1111 ) )
1112 {
1113 wxLogLastError(_T("GetTempPath"));
1114
1115 // when all else fails...
1116 wxStrcpy(gs_reportFilename, _T("c:\\"));
1117 }
1118
1119 // use PID and date to make the report file name more unique
1120 wxString fname = wxString::Format
1121 (
1122 _T("%s_%s_%lu.rpt"),
1123 wxTheApp ? wxTheApp->GetAppName().c_str()
1124 : _T("wxwindows"),
1125 wxDateTime::Now().Format(_T("%Y%m%d")).c_str(),
1126 ::GetCurrentProcessId()
1127 );
1128
1129 wxStrncat(gs_reportFilename, fname,
1130 WXSIZEOF(gs_reportFilename) - wxStrlen(gs_reportFilename) - 1);
1131 }
1132
1133 return true;
1134 }
1135
1136 extern unsigned long wxGlobalSEHandler(EXCEPTION_POINTERS *pExcPtrs)
1137 {
1138 if ( gs_handleExceptions && wxTheApp )
1139 {
1140 // store the pointer to exception info
1141 wxGlobalSEInformation = pExcPtrs;
1142
1143 // give the user a chance to do something special about this
1144 __try
1145 {
1146 wxTheApp->OnFatalException();
1147 }
1148 __except ( EXCEPTION_EXECUTE_HANDLER )
1149 {
1150 // nothing to do here, just ignore the exception inside the
1151 // exception handler
1152 ;
1153 }
1154
1155 wxGlobalSEInformation = NULL;
1156
1157 // this will execute our handler and terminate the process
1158 return EXCEPTION_EXECUTE_HANDLER;
1159 }
1160
1161 return EXCEPTION_CONTINUE_SEARCH;
1162 }
1163
1164 #else // !wxUSE_ON_FATAL_EXCEPTION
1165
1166 bool wxHandleFatalExceptions(bool WXUNUSED(doit))
1167 {
1168 wxFAIL_MSG(_T("set wxUSE_ON_FATAL_EXCEPTION to 1 to use this function"));
1169
1170 return false;
1171 }
1172
1173 #endif // wxUSE_ON_FATAL_EXCEPTION/!wxUSE_ON_FATAL_EXCEPTION
1174