一、引言

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. 「THUWC 2017」随机二分图

    「THUWC 2017」随机二分图 解题思路 : 首先有一个 \(40pts\) 的做法: 前 \(20pts\) 暴力枚举最终的匹配是怎样的,check一下计算方案数,后 \(20pts\) 令 \ ...

  2. [QSCOJ39]喵哈哈村的代码传说 第五章 找规律

    题目大意: 给你n堆排,两人轮流对其中一堆牌进行以下操作之一: 1.从这堆牌中取出任意数量的牌: 2.将这这堆牌分为任意大小的3堆牌. 不能操作者负. 问先手是否有必胜策略. 思路: 尝试构造sg函数 ...

  3. [转]Android Service完全解析,关于服务你所需知道的一切

      目录(?)[+] Android Service完全解析,关于服务你所需知道的一切(上) 分类: Android疑难解析2013-10-31 08:10 6451人阅读 评论(39) 收藏 举报 ...

  4. ios:设置视图背景图片的方法

    1. 使用一个UIImageView实例做子视图,并且放最后面UIImageView *customBackgournd = [UIImageView alloc] initWithImage:[UI ...

  5. PSCollectionView瀑布流实现

    [-] 一基本原理 二具体实现 相关数据结构 视图更新方式 relayoutViews方法 removeAndAddCellsIfNecessary方法 select方法 重用数据块视图机制 三使用方 ...

  6. Segger Real Time Terminal RTT JLINK 客户端软件 GUI 版本

  7. JavaScript如何获取/计算页面元素的offset?

    问题  通过点击一控件,在控件的下面显示一个浮动层,通常的做法是:获取此控件的offset值,再计算出浮动层的top,left等css属性的值,赋值即可. 那么下面就看一下如何获取控件的offset值 ...

  8. asp动态数组

    本文所说的 ASP 数组是指在 ASP 中以默认语言 VBScript 为语言的数组. 样例: Dim   MyArray()        for   i   =   0   to   10    ...

  9. Android API 中文(14) —— ViewStub

    前言 关键字: android.view.ViewStub,版本为Android 2.2 r1 本章翻译来自唐明 ,这里本博负责整理和发布,欢迎其他译者一起参与Android API 的中文翻译行动, ...

  10. LTE试题

    D 如果出现eNB的告警1018007“小区退服,光口不可用”,不可能是以下哪种原因造成的?(          ) 基带板上Ir接口光模块损坏 基带板上Ir接口光模块被拔出 基带板上Ir接口光模块型 ...