由于Debug和Release模式下,编译器的行为不同,导致有些bug在Debug模式下并不能检查出来,而Release模式下又无法单步调试。因此在Release模式下记录程序崩溃的信息,并定位代码所在的行号是非常有必要的。

该过程分为3步:

  1. 在当前程序的Release版本中生成pdb调试信息文件
  2. 将程序崩溃时的执行信息保存为dump文件;
  3. 指定pdbexe文件路径,使用WinDbg打开dump文件定位崩溃时的代码位置。

测试的环境:Qt Creator 4.11.2 + Qt 5.12.8 + MSVC2017

1. Qt Creator在Release下生成 .pdb 调试文件

项目采用qmake编译,在.pro文件中加入如下语句:

QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUG

或者在项目 —> 构建设置 —> qmake详情 中,选择Release构建配置,将Generate separate debug info勾选。

重新构建当前项目,会在release文件夹下生成.pdb文件。

2. 将程序崩溃时的执行信息保存为dump文件

这部分网上文章较多,参考QT 异常崩溃处理,在windows系统下的主要思路是:

  1. 使用SetUnhandledExceptionFilter注册程序的异常捕获回调函数;
  2. 将捕获的异常通过MiniDumpWriteDump函数保存。

参考代码(crash.dmp文件将会保存在程序运行根目录下):

在Pro文件里加入LIBS += -lDbgHelp

#include <Windows.h>
#include <DbgHelp.h>
#include <QApplication>

// 保存程序异常崩溃的信息
LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
{
    //创建 Dump 文件
    HANDLE hDumpFile = CreateFile(L"crash.dmp", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
    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);
    }

    //弹出一个错误对话框
    QMessageBox msgBox;
    msgBox.setText("application crash!");
    msgBox.exec();

    return EXCEPTION_EXECUTE_HANDLER;
}


int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    //注册异常捕获函数
    SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);

    // set printf and fprintf out immediately
    setvbuf(stdout, NULL, _IONBF, 0);
    setvbuf(stderr, NULL, _IONBF, 0);

    OctNisus w;
    w.show();
    return a.exec();
}

非 windows 操作系统,可以采用Google breakpad实现,参考:跨平台的Qt程序崩溃生成Dump文件Breakpad

3. 使用 WinDbg 分析dump文件

  1. 打开WinDbg(x64)(windbg.exe);
  2. File —> Symbol File Path,设置.pdb文件所在的文件夹(注意是文件夹,而不是文件);
  3. File —> Source File Path,设置release版本的.exe执行文件所在的文件夹
  4. File —> Open Crash Dump,打开crash.dmp文件。

注意:

每次发布程序的时候,记得保存对应版本的.pdb文件,当客户使用软件出现崩溃时,回传dump文件,即可定位问题的原因,不用拼命的复现场景。

分析过程中可能会提示:缺少ntdll.pdb缺少kernelbase.pdb等等,不用理会,这些是为了定位系统内部方法的。我们只需要知道错误在自己程序的什么函数里面发生。

如果是在同一台电脑上,而且release包、pdb文件、应用程序都是同一个版本时,可以直接定位到具体的代码行号。此时用visual studio打开crash.dmp文件,可以直接定位到代码文件中。