- 关于Dump的文件介绍在这里,今天要介绍的是如何在Qt工程生成Dump文件。
- 首先我们在Qt的pro文件(工程配置文件)要取消优化,并加入调试信息,如下:
#加入调试信息
QMAKE_CFLAGS_RELEASE += -g
QMAKE_CXXFLAGS_RELEASE += -g
#禁止优化
QMAKE_CFLAGS_RELEASE -= -O2
QMAKE_CXXFLAGS_RELEASE -= -O2
#release在最后link时默认有"-s”参数,表示"Omit all symbol information from the output file",因此要去掉该参
win32:!msvc{
QMAKE_LFLAGS_RELEASE = -mthreads
}else{
#message("msvc version")
#QMAKE_CFLAGS_RELEASE = -O2 -MD -Zi
#QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUG
#QMAKE_LFLAGS_RELEASE += /MAP
#QMAKE_CFLAGS_RELEASE += /Zi
#QMAKE_LFLAGS_RELEASE += /debug /opt:ref
QMAKE_LFLAGS_RELEASE += /MAP /DEBUG /opt:ref /INCREMENTAL:NO
}
- 同样,在pro文件中加入生成Dump文件的依赖库——DBghelp.lib,User32.lib
LIBS += \
-lUser32 \
-lDbghelp
- 接着我们要定义生成dump文件的函数入口,然后把入口添加在main函数的入口处,一般是在main函数的第一行。如下
#include <Windows.h>
#include <DbgHelp.h>
#include <winnt.h>
#include <time.h>
#include <locale>
#include <codecvt>
LONG __stdcall ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
{
//程式异常捕获
//创建 Dump 文件
char str_time[100];
memset(str_time, 0, 100);
struct tm *local_time = NULL;
time_t utc_time;
utc_time = time(NULL);
local_time = localtime(&utc_time);
strftime(str_time, 100, "%Y%m%d%H%M%S", local_time);
std::string timeStr(str_time);
std::string dumpFilePath = timeStr + ".dmp";
std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> converter;
std::wstring wide = converter.from_bytes(dumpFilePath);
HANDLE hDumpFile = CreateFile(wide.c_str(),GENERIC_WRITE, 0, nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, nullptr);
if (hDumpFile != INVALID_HANDLE_VALUE) {
//Dump信息
MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
dumpInfo.ExceptionPointers = pException;
dumpInfo.ThreadId = GetCurrentThreadId();
dumpInfo.ClientPointers = TRUE;
//写入Dump文件内容
MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
}
//这里弹出一个错误对话框并退出程序
EXCEPTION_RECORD* record = pException->ExceptionRecord;
fprintf(stderr, "record->ExceptionCode:%lu\n", record->ExceptionCode);
return EXCEPTION_EXECUTE_HANDLER;
}
#include<tchar.h>
void DisableSetUnhandledExceptionFilter()
{
HMODULE h_hand = LoadLibrary(_T("kernel32.dll"));
void *addr = (void*)GetProcAddress(h_hand,"SetUnhandledExceptionFilter");
if (addr)
{
unsigned char code[16];
int size = 0;
code[size++] = 0x33;
code[size++] = 0xC0;
code[size++] = 0xC2;
code[size++] = 0x04;
code[size++] = 0x00;
DWORD dwOldFlag, dwTempFlag;
VirtualProtect(addr, size, PAGE_READWRITE, &dwOldFlag);
WriteProcessMemory(GetCurrentProcess(), addr, code, size, NULL);
VirtualProtect(addr, size, dwOldFlag, &dwTempFlag);
}
}
void RunCrashHandlerLocal()
{
SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);//注册异常捕获函数
}
int main()
{
RunCrashHandlerLocal();
int a = 9;
int b = 0;
int c = 0;
c = a / b;//产生程序异常并崩溃
return 0;
}
以上就完成一个最简版本的
注:
1). 生成dump必须得是msvc的编译器才行,使用mingw编译器生成不了dump文件
2). pro文件适用于qt工程,以上代码同样试用于vs工程,只需要在vs工程中取消优化,并将相应的库——DBghelp.lib,User32.lib添加到工程即可