一、引言

dump文件是C++程序发生异常时,保存当时程序运行状态的文件,是调试异常程序重要的方法,所以程序崩溃时,除了日志文件,dump文件便成了我们查找错误的最后一根救命的稻草。windows程序产生dump文件和linux程序产生dump文件的方式不一样,linux默认是不让产生core dump文件,只要在用户自己的~/.bash_profile文件中增加

ulimit -S -c unlimited > /dev/null 2>&1

这样程序崩溃就可以产生可调试的core dump文件了。但是windows环境就得写代码才能实现了。

二、原理

windows程序当遇到异常,没有try-catch或者try-catch也无法捕获到的异常时,程序就会自动退出,如果这时候没有dump文件的话,我们是没有得到任何程序退出的信息。在windows程序异常退出之前,会预先调用一个在程序中注册的异常处理回调函数(默认是没有设置),只要我们在这个回调函数中调用MiniDumpWriteDump函数就可以产生我们想要的dump文件。

三、实现

1.调用SetUnhandledExceptionFilter注册一个自定义的异常处理回调函数

SetUnhandledExceptionFilter(MyUnhandledExceptionFilter);

异常处理回调函数的原型

LONG __stdcall MyUnhandledExceptionFilter(PEXCEPTION_POINTERS pExceptionInfo);

2.CreateFile创建dump文件,调用MiniDumpWriteDump函数往dump文件写异常信息

  1. inline void CreateMiniDump(PEXCEPTION_POINTERS pep, LPCTSTR strFileName)
  2. {
  3. HANDLE hFile = CreateFile(strFileName, GENERIC_READ | GENERIC_WRITE,
  4. FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  5. if((hFile != NULL) && (hFile != INVALID_HANDLE_VALUE))
  6. {
  7. MINIDUMP_EXCEPTION_INFORMATION mdei;
  8. mdei.ThreadId           = GetCurrentThreadId();
  9. mdei.ExceptionPointers  = pep;
  10. mdei.ClientPointers     = NULL;
  11. MINIDUMP_CALLBACK_INFORMATION mci;
  12. mci.CallbackRoutine     = (MINIDUMP_CALLBACK_ROUTINE)MiniDumpCallback;
  13. mci.CallbackParam       = 0;
  14. ::MiniDumpWriteDump(::GetCurrentProcess(), ::GetCurrentProcessId(), hFile, MiniDumpNormal, (pep != 0) ? &mdei : 0, NULL, &mci);
  15. CloseHandle(hFile);
  16. }
  17. }

CreateMiniDump函数是在异常处理回调函数MyUnhandledExceptionFilter中调用的

  1. LONG __stdcall MyUnhandledExceptionFilter(PEXCEPTION_POINTERS pExceptionInfo)
  2. {
  3. CreateMiniDump(pExceptionInfo, "core.dmp");
  4. return EXCEPTION_EXECUTE_HANDLER;
  5. }

3.将SetUnhandledExceptionFilter失效

vs2005中,编译的过程中,编译器会自动给你的程序加上一句SetUnhandledExceptionFilter(NULL),这就会导致你之前自定义的

SetUnhandledExceptionFilter(MyUnhandledExceptionFilter);

无效,就有可能不会产生dump文件,因此我们必须在自定义的SetUnhandledExceptionFilter之后,让之后调用的SetUnhandledExceptionFilter无效。增加以下代码:

  1. // 此函数一旦成功调用,之后对 SetUnhandledExceptionFilter 的调用将无效
  2. void DisableSetUnhandledExceptionFilter()
  3. {
  4. void* addr = (void*)GetProcAddress(LoadLibrary("kernel32.dll"),
  5. "SetUnhandledExceptionFilter");
  6. if (addr)
  7. {
  8. unsigned char code[16];
  9. int size = 0;
  10. code[size++] = 0x33;
  11. code[size++] = 0xC0;
  12. code[size++] = 0xC2;
  13. code[size++] = 0x04;
  14. code[size++] = 0x00;
  15. DWORD dwOldFlag, dwTempFlag;
  16. VirtualProtect(addr, size, PAGE_READWRITE, &dwOldFlag);
  17. WriteProcessMemory(GetCurrentProcess(), addr, code, size, NULL);
  18. VirtualProtect(addr, size, dwOldFlag, &dwTempFlag);
  19. }
  20. }

最终代码整理:

//minidump.h

  1. #pragma once
  2. #include <windows.h>
  3. #include <DbgHelp.h>
  4. #include <stdlib.h>
  5. #pragma comment(lib, "dbghelp.lib")
  6. #ifndef _M_IX86
  7. #error "The following code only works for x86!"
  8. #endif
  9. inline BOOL IsDataSectionNeeded(const WCHAR* pModuleName)
  10. {
  11. if(pModuleName == 0)
  12. {
  13. return FALSE;
  14. }
  15. WCHAR szFileName[_MAX_FNAME] = L"";
  16. _wsplitpath(pModuleName, NULL, NULL, szFileName, NULL);
  17. if(wcsicmp(szFileName, L"ntdll") == 0)
  18. return TRUE;
  19. return FALSE;
  20. }
  21. inline BOOL CALLBACK MiniDumpCallback(PVOID                            pParam,
  22. const PMINIDUMP_CALLBACK_INPUT   pInput,
  23. PMINIDUMP_CALLBACK_OUTPUT        pOutput)
  24. {
  25. if(pInput == 0 || pOutput == 0)
  26. return FALSE;
  27. switch(pInput->CallbackType)
  28. {
  29. case ModuleCallback:
  30. if(pOutput->ModuleWriteFlags & ModuleWriteDataSeg)
  31. if(!IsDataSectionNeeded(pInput->Module.FullPath))
  32. pOutput->ModuleWriteFlags &= (~ModuleWriteDataSeg);
  33. case IncludeModuleCallback:
  34. case IncludeThreadCallback:
  35. case ThreadCallback:
  36. case ThreadExCallback:
  37. return TRUE;
  38. default:;
  39. }
  40. return FALSE;
  41. }
  42. inline void CreateMiniDump(PEXCEPTION_POINTERS pep, LPCTSTR strFileName)
  43. {
  44. HANDLE hFile = CreateFile(strFileName, GENERIC_READ | GENERIC_WRITE,
  45. FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  46. if((hFile != NULL) && (hFile != INVALID_HANDLE_VALUE))
  47. {
  48. MINIDUMP_EXCEPTION_INFORMATION mdei;
  49. mdei.ThreadId           = GetCurrentThreadId();
  50. mdei.ExceptionPointers  = pep;
  51. mdei.ClientPointers     = NULL;
  52. MINIDUMP_CALLBACK_INFORMATION mci;
  53. mci.CallbackRoutine     = (MINIDUMP_CALLBACK_ROUTINE)MiniDumpCallback;
  54. mci.CallbackParam       = 0;
  55. ::MiniDumpWriteDump(::GetCurrentProcess(), ::GetCurrentProcessId(), hFile, MiniDumpNormal, (pep != 0) ? &mdei : 0, NULL, &mci);
  56. CloseHandle(hFile);
  57. }
  58. }
  59. LONG __stdcall MyUnhandledExceptionFilter(PEXCEPTION_POINTERS pExceptionInfo)
  60. {
  61. CreateMiniDump(pExceptionInfo, "core.dmp");
  62. return EXCEPTION_EXECUTE_HANDLER;
  63. }
  64. // 此函数一旦成功调用,之后对 SetUnhandledExceptionFilter 的调用将无效
  65. void DisableSetUnhandledExceptionFilter()
  66. {
  67. void* addr = (void*)GetProcAddress(LoadLibrary("kernel32.dll"),
  68. "SetUnhandledExceptionFilter");
  69. if (addr)
  70. {
  71. unsigned char code[16];
  72. int size = 0;
  73. code[size++] = 0x33;
  74. code[size++] = 0xC0;
  75. code[size++] = 0xC2;
  76. code[size++] = 0x04;
  77. code[size++] = 0x00;
  78. DWORD dwOldFlag, dwTempFlag;
  79. VirtualProtect(addr, size, PAGE_READWRITE, &dwOldFlag);
  80. WriteProcessMemory(GetCurrentProcess(), addr, code, size, NULL);
  81. VirtualProtect(addr, size, dwOldFlag, &dwTempFlag);
  82. }
  83. }
  84. void InitMinDump()
  85. {
  86. //注册异常处理函数
  87. SetUnhandledExceptionFilter(MyUnhandledExceptionFilter);
  88. //使SetUnhandledExceptionFilter
  89. DisableSetUnhandledExceptionFilter();
  90. }

4.测试代码

//test.cpp

    1. #include <iostream>
    2. #include "minidump.h"
    3. void test()
    4. {
    5. std::string s = "abcd";
    6. try{
    7. s[100] = 'b';
    8. }
    9. catch(std::exception& e)
    10. {
    11. std::cout << "with exception:[" << e.what() << "]" << std::endl;
    12. }
    13. catch(...)
    14. {
    15. std::cout << "with unknown exception" << std::endl;
    16. }
    17. }
    18. void main()
    19. {
    20. InitMinDump();
    21. test();
    22. system("pause");
    23. }

编写的windows程序,崩溃时产生crash dump文件的办法的更多相关文章

  1. 如何在.NET程序崩溃时自动创建Dump?

    今天在浏览张队转载文章的留言时,遇到一个读者问了这样的问题,如下图所示: 首先能明确的一点是"程序崩溃退出了是不能用常规的方式dump的",因为整个进程树都已经退出.现场已经无法使 ...

  2. 如何定位Release 版本中程序崩溃的位置 ---利用map文件 拦截windows崩溃函数

    1       案例描述 作为Windows程序员,平时最担心见到的事情可能就是程序发生了崩溃(异常),这时Windows会提示该程序执行了非法操作,即将关闭.请与您的供应商联系.呵呵,这句微软的“名 ...

  3. Ubuntu16.04下写的Qt程序,调试时没问题,运行时偶现崩溃 (需要在运行时生成core dump文件,QMAKE_CC += -g)

    记录一下 Ubuntu16.04下写的Qt程序,调试时没问题,运行时偶现崩溃 需要在运行时生成core dump文件 首先在pro结尾里加入 QMAKE_CC += -g QMAKE_CXX += - ...

  4. 使用SetUnhandledExceptionFilter转储程序崩溃时内存DMP .

    关于程序崩溃时转储内存DMP,可以设置注册表,使程序崩溃时自动转储内存DMP,见程序崩溃时利用注册表自动转储内存DMP.本文要介绍的是使用SetUnhandledExceptionFilter函数在程 ...

  5. 让linux中的程序崩溃时生成core文件

    当我们的linux程序崩溃的时候,常常会有这样的提示:    Segmentation fault (core dumped)    段错误 (核心已转储)    提示说生成了core文件,但是此功能 ...

  6. android在程序崩溃时Catch异常并处理

    Android系统的"程序异常退出",给应用的用户体验造成不良影响.为了捕获应用运行时异常并给出友好提示,便可继承UncaughtExceptionHandler类来处理.通过Th ...

  7. C++ 程序崩溃时生成Dump文件

    #include <DbgHelp.h> //生产DUMP文件 int GenerateMiniDump(HANDLE hFile, PEXCEPTION_POINTERS pExcept ...

  8. .NET程序崩溃了怎么抓 Dump ? 我总结了三种方案

    一:背景 1. 讲故事 最近几天接到了几个crash的求助,可能这几个朋友没玩过怎么去生成dump,只能手把手教,感觉也不是一个办法,所以有必要总结一下,后续再有朋友咨询的话,我就可以把这篇文章丢过去 ...

  9. 记录linux 生成crash dump文件步骤

    执行文件编译时加入-g 命令 例如 g++ -g test.cpp 查看当前系统限制情况 ulimit -a 设置crash dump 文件大小 ulimit -c unlimited unlimit ...

随机推荐

  1. [ZROI 9.15模拟赛] Tutorial

    Link: 传送门 可能要补一补之前的题了 题目名字天(Sky)的(De)炭(C)好评啊…… A: 从买/卖物品的配对来考虑: 可以发现如果当前物品为卖,肯定从之前选最小的(无论其为买/卖),因为贡献 ...

  2. SpringMVC集成Swagger插件以及Swagger注解的简单使用

    一.简介 Swagger 是一个规范和完整的框架,用于生成.描述.调用和可视化 RESTful 风格的 Web 服务.总体目标是使客户端和文件系统作为服务器以同样的速度来更新 .接口的方法,参数和模型 ...

  3. hdu 3507 斜率优化

    我的第一道斜率优化. 就这道题而言,写出原始的方程: dp[i] = min{ dp[j] + (sum[i]-sum[j])2  + M | j in [0,i) } O(n^2)的复杂度肯定超时, ...

  4. [转]MinGW编译wxWidgets问题

    其实利用MinGW编译wxWidgets是一件挺简单的事情,但是最近在编译的时候遇到一些问题,并找到解决方案,因此在此分享一下个人经验.编译出错的原因是由于安装了MSYS的缘故.   错误信息类似于如 ...

  5. 如何解决…has been modified since the precompiled header… was built的问题

    如何解决…has been modified since the precompiled header… was built 的问题 xcode5.1在程序中报错: File '/Applicatio ...

  6. FreeRTOS Customisation -- FreeRTOSConfig.h

    http://www.freertos.org/a00110.html FreeRTOS is customised using a configuration file called FreeRTO ...

  7. SQL Server 的事务和锁 图解

    http://www.cnblogs.com/lxconan/archive/2011/10/20/2218396.html

  8. Linux X86-64 进程内存空间布局

    http://blog.csdn.net/woshinia/article/details/41722085 http://www.lenky.info/archives/2012/04/1424 h ...

  9. python笔记23-unittest单元测试之mock

    什么是mock unittest.mock是一个用于在Python中进行单元测试的库,Mock翻译过来就是模拟的意思,顾名思义这个库的主要功能是模拟一些东西. 它的主要功能是使用mock对象替代掉指定 ...

  10. Xcode的插件的路径

    /Users/dllo/Library/Application\ Support/Developer