X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/2bbc1b285523e4ff3d325b0bf36205e5c1aa844a..c263eb03846c1b1439bf67d1d831255024278adf:/src/msw/crashrpt.cpp diff --git a/src/msw/crashrpt.cpp b/src/msw/crashrpt.cpp index d9882ca5db..9ba2dbbb06 100644 --- a/src/msw/crashrpt.cpp +++ b/src/msw/crashrpt.cpp @@ -58,7 +58,7 @@ #include "wx/datetime.h" -#include "wx/dynload.h" +#include "wx/dynlib.h" #include "wx/msw/crashrpt.h" @@ -77,7 +77,7 @@ // // in any case, the user may override by defining wxUSE_DBGHELP himself #ifndef wxUSE_DBGHELP - #ifdef DBHLPAPI + #ifdef DBHLPAPI #define wxUSE_DBGHELP 1 #else #define wxUSE_DBGHELP 0 @@ -172,6 +172,29 @@ enum SymbolTag // classes // ---------------------------------------------------------------------------- +// low level wxBusyCursor replacement: we use Win32 API directly here instead +// of going through wxWidgets calls as this could be dangerous +class BusyCursor +{ +public: + BusyCursor() + { + HCURSOR hcursorBusy = ::LoadCursor(NULL, IDC_WAIT); + m_hcursorOld = ::SetCursor(hcursorBusy); + } + + ~BusyCursor() + { + if ( m_hcursorOld ) + { + ::SetCursor(m_hcursorOld); + } + } + +private: + HCURSOR m_hcursorOld; +}; + // the real crash report generator class wxCrashReportImpl { @@ -1066,14 +1089,13 @@ wxString wxCrashReportImpl::GetExceptionString(DWORD dwCode) #endif // wxUSE_DBGHELP -// Remove warning -#if wxUSE_DBGHELP && !wxUSE_MINIDUMP - #define WXUNUSED_LOCAL(x) x +bool wxCrashReportImpl::Generate( +#if wxUSE_DBGHELP + int flags #else - #define WXUNUSED_LOCAL WXUNUSED + int WXUNUSED(flags) #endif - -bool wxCrashReportImpl::Generate(int WXUNUSED_LOCAL(flags)) +) { if ( m_hFile == INVALID_HANDLE_VALUE ) return false; @@ -1092,6 +1114,25 @@ bool wxCrashReportImpl::Generate(int WXUNUSED_LOCAL(flags)) HANDLE hModuleCrash = OutputBasicContext(pExceptionRecord, pCtx); #endif // !wxUSE_MINIDUMP + // show to the user that we're doing something... + BusyCursor busyCursor; + + // user-specified crash report flags override those specified by the + // programmer + TCHAR envFlags[64]; + DWORD dwLen = ::GetEnvironmentVariable + ( + _T("WX_CRASH_FLAGS"), + envFlags, + WXSIZEOF(envFlags) + ); + + int flagsEnv; + if ( dwLen && dwLen < WXSIZEOF(envFlags) && + wxSscanf(envFlags, _T("%d"), &flagsEnv) == 1 ) + { + flags = flagsEnv; + } // for everything else we need dbghelp.dll wxDynamicLibrary dllDbgHelp(_T("dbghelp.dll"), wxDL_VERBATIM); @@ -1107,12 +1148,31 @@ bool wxCrashReportImpl::Generate(int WXUNUSED_LOCAL(flags)) minidumpExcInfo.ClientPointers = FALSE; // in our own address space // do generate the dump + MINIDUMP_TYPE dumpFlags; + if ( flags & wxCRASH_REPORT_LOCALS ) + { + // the only way to get local variables is to dump the entire + // process memory space -- but this makes for huge (dozens or + // even hundreds of Mb) files + dumpFlags = MiniDumpWithFullMemory; + } + else if ( flags & wxCRASH_REPORT_GLOBALS ) + { + // MiniDumpWriteDump() has the option for dumping just the data + // segment which contains all globals -- exactly what we need + dumpFlags = MiniDumpWithDataSegs; + } + else // minimal dump + { + dumpFlags = MiniDumpNormal; + } + if ( !MiniDumpWriteDump ( ::GetCurrentProcess(), ::GetCurrentProcessId(), m_hFile, // file to write to - MiniDumpNormal, // just the minimum + dumpFlags, // kind of dump to craete &minidumpExcInfo, NULL, // no extra user-defined data NULL // no callbacks @@ -1128,7 +1188,7 @@ bool wxCrashReportImpl::Generate(int WXUNUSED_LOCAL(flags)) SymSetOptions(SYMOPT_DEFERRED_LOADS | SYMOPT_UNDNAME); // Initialize DbgHelp - if ( SymInitialize(GetCurrentProcess(), NULL, TRUE /* invade */) ) + if ( ::SymInitialize(GetCurrentProcess(), NULL, TRUE /* invade */) ) { OutputStack(pCtx, flags); @@ -1160,7 +1220,7 @@ bool wxCrashReportImpl::Generate(int WXUNUSED_LOCAL(flags)) #else // !wxUSE_DBGHELP Output(_T("Support for crash report generation was not included ") - _T("in this wxWindows version.")); + _T("in this wxWidgets version.")); #endif // wxUSE_DBGHELP/!wxUSE_DBGHELP return false;