1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/crashrpt.cpp 
   3 // Purpose:     code to generate crash dumps (minidumps) 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2003 Vadim Zeitlin <vadim@wxwindows.org> 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  32 #include "wx/msw/debughlp.h" 
  33 #include "wx/msw/crashrpt.h" 
  35 // ---------------------------------------------------------------------------- 
  37 // ---------------------------------------------------------------------------- 
  39 // low level wxBusyCursor replacement: we use Win32 API directly here instead 
  40 // of going through wxWidgets calls as this could be dangerous 
  46         HCURSOR hcursorBusy 
= ::LoadCursor(NULL
, IDC_WAIT
); 
  47         m_hcursorOld 
= ::SetCursor(hcursorBusy
); 
  54             ::SetCursor(m_hcursorOld
); 
  62 // the real crash report generator 
  63 class wxCrashReportImpl
 
  66     wxCrashReportImpl(const wxChar 
*filename
); 
  68     bool Generate(int flags
, EXCEPTION_POINTERS 
*ep
); 
  72         if ( m_hFile 
!= INVALID_HANDLE_VALUE 
) 
  74             ::CloseHandle(m_hFile
); 
  80     // formatted output to m_hFile 
  81     void Output(const wxChar 
*format
, ...); 
  84     void OutputEndl() { Output(_T("\r\n")); } 
  86     // the handle of the report file 
  90 // ---------------------------------------------------------------------------- 
  92 // ---------------------------------------------------------------------------- 
  94 // the file name where the report about exception is written 
  96 // we use fixed buffer to avoid (big) dynamic allocations when the program 
  98 static wxChar gs_reportFilename
[MAX_PATH
]; 
 100 // this is defined in msw/main.cpp 
 101 extern EXCEPTION_POINTERS 
*wxGlobalSEInformation
; 
 103 // ============================================================================ 
 105 // ============================================================================ 
 107 // ---------------------------------------------------------------------------- 
 109 // ---------------------------------------------------------------------------- 
 111 wxCrashReportImpl::wxCrashReportImpl(const wxChar 
*filename
) 
 113     m_hFile 
= ::CreateFile
 
 118                     NULL
,                       // default security 
 120                     FILE_FLAG_WRITE_THROUGH
, 
 121                     NULL                        
// no template file 
 125 void wxCrashReportImpl::Output(const wxChar 
*format
, ...) 
 128     va_start(argptr
, format
); 
 132     wxString s 
= wxString::FormatV(format
, argptr
); 
 134     wxCharBuffer 
buf(s
.mb_str(wxConvUTF8
)); 
 135     ::WriteFile(m_hFile
, buf
.data(), strlen(buf
.data()), &cbWritten
, 0); 
 140 bool wxCrashReportImpl::Generate(int flags
, EXCEPTION_POINTERS 
*ep
) 
 142     if ( m_hFile 
== INVALID_HANDLE_VALUE 
) 
 147         ep 
= wxGlobalSEInformation
; 
 151         Output(_T("Context for crash report generation not available.")); 
 155     // show to the user that we're doing something... 
 156     BusyCursor busyCursor
; 
 158     // user-specified crash report flags override those specified by the 
 161     DWORD dwLen 
= ::GetEnvironmentVariable
 
 163                         _T("WX_CRASH_FLAGS"), 
 169     if ( dwLen 
&& dwLen 
< WXSIZEOF(envFlags
) && 
 170             wxSscanf(envFlags
, _T("%d"), &flagsEnv
) == 1 ) 
 175     if ( wxDbgHelpDLL::Init() ) 
 177         MINIDUMP_EXCEPTION_INFORMATION minidumpExcInfo
; 
 179         minidumpExcInfo
.ThreadId 
= ::GetCurrentThreadId(); 
 180         minidumpExcInfo
.ExceptionPointers 
= ep
; 
 181         minidumpExcInfo
.ClientPointers 
= FALSE
; // in our own address space 
 183         // do generate the dump 
 184         MINIDUMP_TYPE dumpFlags
; 
 185         if ( flags 
& wxCRASH_REPORT_LOCALS 
) 
 187             // the only way to get local variables is to dump the entire 
 188             // process memory space -- but this makes for huge (dozens or 
 189             // even hundreds of Mb) files 
 190             dumpFlags 
= MiniDumpWithFullMemory
; 
 192         else if ( flags 
& wxCRASH_REPORT_GLOBALS 
) 
 194             // MiniDumpWriteDump() has the option for dumping just the data 
 195             // segment which contains all globals -- exactly what we need 
 196             dumpFlags 
= MiniDumpWithDataSegs
; 
 200             // the file size is not much bigger than when using MiniDumpNormal 
 201             // if we use the flags below, but the minidump is much more useful 
 202             // as it contains the values of many (but not all) local variables 
 203             dumpFlags 
= (MINIDUMP_TYPE
)(MiniDumpScanMemory
 
 205                                         |MiniDumpWithIndirectlyReferencedMemory
 
 210         if ( !wxDbgHelpDLL::MiniDumpWriteDump
 
 212                 ::GetCurrentProcess(), 
 213                 ::GetCurrentProcessId(), 
 214                 m_hFile
,                    // file to write to 
 215                 dumpFlags
,                  // kind of dump to craete 
 217                 NULL
,                       // no extra user-defined data 
 221             Output(_T("MiniDumpWriteDump() failed.")); 
 228     else // dbghelp.dll couldn't be loaded 
 230         Output(_T("%s"), wxDbgHelpDLL::GetErrorMessage().c_str()); 
 232 #else // !wxUSE_DBGHELP 
 236     Output(_T("Support for crash report generation was not included ") 
 237            _T("in this wxWidgets version.")); 
 238 #endif // wxUSE_DBGHELP/!wxUSE_DBGHELP 
 243 // ---------------------------------------------------------------------------- 
 245 // ---------------------------------------------------------------------------- 
 248 void wxCrashReport::SetFileName(const wxString
& filename
) 
 250     wxStrncpy(gs_reportFilename
, filename
.c_str(), WXSIZEOF(gs_reportFilename
) - 1); 
 251     gs_reportFilename
[WXSIZEOF(gs_reportFilename
) - 1] = _T('\0'); 
 255 wxString 
wxCrashReport::GetFileName() 
 257     return gs_reportFilename
; 
 261 bool wxCrashReport::Generate(int flags
, EXCEPTION_POINTERS 
*ep
) 
 263     wxCrashReportImpl 
impl(gs_reportFilename
); 
 265     return impl
.Generate(flags
, ep
); 
 269 bool wxCrashReport::GenerateNow(int flags
) 
 275         RaiseException(0x1976, 0, 0, NULL
); 
 277     __except( rc 
= Generate(flags
, (EXCEPTION_POINTERS 
*)GetExceptionInformation()), 
 278               EXCEPTION_CONTINUE_EXECUTION 
) 
 280         // never executed because of EXCEPTION_CONTINUE_EXECUTION above 
 286 // ---------------------------------------------------------------------------- 
 288 // ---------------------------------------------------------------------------- 
 290 wxCrashContext::wxCrashContext(_EXCEPTION_POINTERS 
*ep
) 
 296         wxCHECK_RET( wxGlobalSEInformation
, _T("no exception info available") ); 
 297         ep 
= wxGlobalSEInformation
; 
 300     // TODO: we could also get the operation (read/write) and address for which 
 301     //       it failed for EXCEPTION_ACCESS_VIOLATION code 
 302     const EXCEPTION_RECORD
& rec 
= *ep
->ExceptionRecord
; 
 303     code 
= rec
.ExceptionCode
; 
 304     addr 
= rec
.ExceptionAddress
; 
 307     const CONTEXT
& ctx 
= *ep
->ContextRecord
; 
 326     regs
.flags 
= ctx
.EFlags
; 
 330 wxString 
wxCrashContext::GetExceptionString() const 
 334     #define CASE_EXCEPTION( x ) case EXCEPTION_##x: s = _T(#x); break 
 338         CASE_EXCEPTION(ACCESS_VIOLATION
); 
 339         CASE_EXCEPTION(DATATYPE_MISALIGNMENT
); 
 340         CASE_EXCEPTION(BREAKPOINT
); 
 341         CASE_EXCEPTION(SINGLE_STEP
); 
 342         CASE_EXCEPTION(ARRAY_BOUNDS_EXCEEDED
); 
 343         CASE_EXCEPTION(FLT_DENORMAL_OPERAND
); 
 344         CASE_EXCEPTION(FLT_DIVIDE_BY_ZERO
); 
 345         CASE_EXCEPTION(FLT_INEXACT_RESULT
); 
 346         CASE_EXCEPTION(FLT_INVALID_OPERATION
); 
 347         CASE_EXCEPTION(FLT_OVERFLOW
); 
 348         CASE_EXCEPTION(FLT_STACK_CHECK
); 
 349         CASE_EXCEPTION(FLT_UNDERFLOW
); 
 350         CASE_EXCEPTION(INT_DIVIDE_BY_ZERO
); 
 351         CASE_EXCEPTION(INT_OVERFLOW
); 
 352         CASE_EXCEPTION(PRIV_INSTRUCTION
); 
 353         CASE_EXCEPTION(IN_PAGE_ERROR
); 
 354         CASE_EXCEPTION(ILLEGAL_INSTRUCTION
); 
 355         CASE_EXCEPTION(NONCONTINUABLE_EXCEPTION
); 
 356         CASE_EXCEPTION(STACK_OVERFLOW
); 
 357         CASE_EXCEPTION(INVALID_DISPOSITION
); 
 358         CASE_EXCEPTION(GUARD_PAGE
); 
 359         CASE_EXCEPTION(INVALID_HANDLE
); 
 362             // unknown exception, ask NTDLL for the name 
 363             if ( !::FormatMessage
 
 365                      FORMAT_MESSAGE_IGNORE_INSERTS 
| 
 366                      FORMAT_MESSAGE_FROM_HMODULE
, 
 367                      ::GetModuleHandle(_T("NTDLL.DLL")), 
 370                      wxStringBuffer(s
, 1024), 
 375                 s
.Printf(_T("UNKNOWN_EXCEPTION(%d)"), code
); 
 379     #undef CASE_EXCEPTION 
 384 #endif // wxUSE_CRASHREPORT/!wxUSE_CRASHREPORT