Windows生成dump文件
// dumpcreate.h 文件
// dumpcreate.h
#if defined (_WIN32) || defined (WIN32)
#include <Windows.h>
long __stdcall exceptionHander(_EXCEPTION_POINTERS* excp); // 全局异常处理捕获
#endif
// dumpcreate.cpp 文件
// dumpcreate.cpp
#if defined (_WIN32) || defined (WIN32)
#include <time.h>
#include <Windows.h>
#include <dbghelp.h>
#include <tchar.h>
#pragma comment(lib, "dbghelp.lib")
#endif
#include "dumpcreate.h"
#define USER_DATA_BUFFER_SIZE (4096)
/*
*ExceptionCode 最可能遇到的几种类型
*C0000005h读写内存冲突
*C0000094h非法除0
*C00000FDh堆栈溢出或者说越界
*80000001h由Virtual Alloc建立起来的属性页冲突
*C0000025h不可持续异常,程序无法恢复执行,异常处理例程不应处理这个异常
*C0000026h在异常处理过程中系统使用的代码,如果系统从某个例程莫名奇妙的返回,则出现此代码,例如调用RtlUnwind时没有ExceptionRecord参数时产生的异常填入的就是这个代码
*80000003h调试时因代码中int3中断
*80000004h处于被单步调试状态
*
*返回值:
*
*EXCEPTION_EXECUTE_HANDLER equ 1 表示我已经处理了异常,可以结束了
*EXCEPTION_CONTINUE_SEARCH equ 0 表示我不处理,其他人来吧,于是windows调用默认的处理程序显示一个错误框,并结束
*EXCEPTION_CONTINUE_EXECUTION equ -1 表示错误已经被修复,请从异常发生处继续执行
*/
long /*__stdcall*/ exceptionHander(_EXCEPTION_POINTERS* excp)
{
TCHAR m_szDumpPath[_MAX_PATH];
TCHAR m_szAppPath[_MAX_PATH];
TCHAR m_szAppBaseName[_MAX_PATH];
time_t m_lTime;
time(&m_lTime);
LONG retval = EXCEPTION_CONTINUE_SEARCH;
HMODULE hDll = NULL;
//TCHAR szDbgHelpPath[_MAX_PATH];
if (GetModuleFileName(NULL, m_szAppPath, _MAX_PATH))
{
TCHAR *pSlash = _tcsrchr(m_szAppPath, '\\');
if (pSlash)
{
_tcsncpy_s(m_szAppBaseName, _MAX_PATH, pSlash + 1, _MAX_PATH);
*(pSlash + 1) = 0;
// SetDumpFileName
wsprintf(m_szDumpPath, _T("%s%s.%ld.dmp"), m_szAppPath, m_szAppBaseName, m_lTime);
}
}
TCHAR szScratch[USER_DATA_BUFFER_SIZE];
// ask the user if they want to save a dump file
wsprintf(szScratch, _T("There was an unexpected error:\n\n%s\nWould you like to save a diagnostic file?\n\nFilename: %s"), "", m_szDumpPath);
if (::MessageBox(NULL, szScratch, NULL, MB_YESNO) != IDYES)
return EXCEPTION_EXECUTE_HANDLER;
//创建转储文件
HANDLE hFile = CreateFile(m_szDumpPath, GENERIC_WRITE, FILE_SHARE_WRITE, NULL,
CREATE_NEW, FILE_ATTRIBUTE_NORMAL, NULL);
LPCTSTR szResult = NULL;
if (hFile != INVALID_HANDLE_VALUE)
{
MINIDUMP_EXCEPTION_INFORMATION exptInfo;
exptInfo.ThreadId = ::GetCurrentThreadId();
exptInfo.ExceptionPointers = excp;
exptInfo.ClientPointers = NULL;
//将dump信息写入到文件中
BOOL bOK = MiniDumpWriteDump(GetCurrentProcess(),
GetCurrentProcessId(),
hFile, MiniDumpNormal,
&exptInfo, NULL, NULL);
if (bOK)
{
szResult = NULL;
retval = EXCEPTION_EXECUTE_HANDLER;
}
else
{
wsprintf(szScratch, _T("Failed to save dump file to '%s' (error %d)"), m_szDumpPath, GetLastError());
szResult = szScratch;
}
::CloseHandle(hFile);
}
else
{
wsprintf(szScratch, _T("Failed to create dump file '%s' (error %d)"), m_szDumpPath, GetLastError());
szResult = szScratch;
}
if (szResult)
::MessageBox(NULL, szResult, NULL, MB_OK);
TerminateProcess(GetCurrentProcess(), 0);
return EXCEPTION_CONTINUE_SEARCH;
}
在main.cpp中设置异常句柄,如下:
// main.cpp 文件
// main.cpp 文件
#include "dumpcreate.h"
int main(int argc, char *argv[])
{
#if defined (_WIN32) || defined (WIN32)
SetUnhandledExceptionFilter(exceptionHander);
#endif
// 测试生成 dump 文件
char *str = NULL;
strcpy(str, "test");
return 0;
}
程序运行结果:
在发生运行错误时,会弹出是否要生成dump文件的对话框,选择是(Y)
时,即开始生成dump文件,选择否(N)
即不生成。
若不想有确认生成的弹窗,屏蔽dumpcreate.cpp
中的两处MessageBox(...)
方法调用即可
生成的文件默认在工作目录下,如下图。