51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

记录:MSVC+Qt生成dump文件

本文只是记录下 MSVC+Qt 生成 dump 的代码。

dump 文件能够保存程序内部的内存、堆栈、句柄、线程等程序运行相关的信息,当程序异常无法在调试环境里解决时,dump 文件是分析问题的重要手段。

相关 win API 文档可以在 MSDN 查看:https://docs.microsoft.com/zh-cn/search/?terms=MiniDumpWriteDump

DUMPTYPE 参数我只使用了 MiniDumpNormal,可以根据文档加上自己需要的。

dump 文件一般配合 pdb 文件使用,通过 pdb 可以找到堆栈地址对应的函数、行号等。

如果使用的 QtCreator,可以打开如下设置:

如果使用 VS ,可以打开如下两个设置:

生成 dump 完整代码:

 
   
 #include "mainwindow.h"
#include "CrashDump.h"
#include <QApplication> 
int main(int argc, char *argv[])
{
    CrashDump::init("Test");
 
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
    return a.exec();
}
 
   
 #pragma once
 
#include <QObject>
#include <QCalendar>
#include <QDateTime>
#include <QFileInfo>
#include <QDir>
#include <Windows.h>
#include <DbgHelp.h>
#pragma comment(lib, "DbgHelp")
#include <iostream>
 
static QString __module_name;
class CrashDump
{
public:
    //初始化注册
    static void init(const QString &module)
    {
        __module_name=module;
        //异常捕获
        ::SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)exceptionFilter);
    }
 
    //获取exe路径
    static QString getAppPath()
    {
        DWORD v;
        QVarLengthArray<char, 2048> buffer;
        size_t size = 0;
        do {
            size += 2048;
            buffer.resize((int)size);
            v = ::GetModuleFileNameA(NULL, buffer.data(), DWORD(buffer.size()));
        } while (v >= size);
 
        return QString::fromLocal8Bit(buffer.data(), v);
    }
 
    //生成dump路径,可以自定义路径以及命名格式
    static QString getDumpPath()
    {
        QFileInfo app_info(getAppPath());
        QString dump_path=QString("%1/dump/%2 %3.dmp")
                .arg(app_info.absolutePath())
                .arg(__module_name)
                .arg(QDateTime::currentDateTime().toString("yyyy_MM_dd hh.mm.ss"));
        QFileInfo dump_info(dump_path);
        if(!dump_info.dir().exists()&&!dump_info.dir().mkpath(dump_info.absolutePath()))
            dump_path=QString("%1.dump").arg(__module_name);
        return dump_path;
    }
 
    //异常处理
    static LONG exceptionFilter(LPEXCEPTION_POINTERS lpExceptionInfo)
    {
        QString dump_path = getDumpPath();
        std::cerr<<"crash dump:"<<dump_path.toStdString()<<std::endl;
 
        //创建文件
        HANDLE file_handle = ::CreateFileA(dump_path.toLocal8Bit().data(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
        if (file_handle != INVALID_HANDLE_VALUE) {
            DWORD dump_type = MiniDumpNormal; //可以加上需要的其他枚举值
            MINIDUMP_EXCEPTION_INFORMATION dump_info;
            dump_info.ClientPointers = TRUE;
            dump_info.ExceptionPointers = lpExceptionInfo;
            dump_info.ThreadId = ::GetCurrentThreadId();
            //写入dump
            ::MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), file_handle, (MINIDUMP_TYPE)dump_type, &dump_info, NULL, NULL);
        }
        ::CloseHandle(file_handle);
        //这里可以把程序拉起来
        return EXCEPTION_CONTINUE_SEARCH; //or EXCEPTION_EXECUTE_HANDLER
    }
};

程序遇到异常生成 dump 文件后,可以使用 VS 打开:

这里符号路径我只设置了 exe 对应 pdb 的路径:

最后点仅限本机调试就可以看到异常现场了:

(2021-3-26)要注意的是,SetUnhandledExceptionFilter 并不能捕获所有的异常,可以搜索其他的相关函数:

 
   
     //terminate()
    set_terminate(terminateHandler);
    //printf(NULL)
    _set_invalid_parameter_handler(invalidParameterHandler);
    //调用纯虚函数
    _set_purecall_handler(purecallHandler);
    //new异常
    _set_new_handler(newHandler);

其他

  1. 国内开源:https://gitee.com/feiyangqingyun

  2. 国际开源:https://github.com/feiyangqingyun

  3. 项目大全:https://qtchina.blog.csdn.net/article/details/97565652


赞(5)
未经允许不得转载:工具盒子 » 记录:MSVC+Qt生成dump文件