一、引言

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. HDU1429 胜利大逃亡 状压bfs

    http://acm.hdu.edu.cn/viewcode.php?rid=22225154 因为总共a-j有10种钥匙,所以可以把有没有钥匙的状态压到一个int数里,然后dfs. 昨天状态特别不好 ...

  2. CodeForces - 1016C Vasya And The Mushrooms

    题面在这里! 好久没有体会这种A题的快感了23333 一开始看错了,以为权值是从1开始的,不过这样不要紧,最后把算的答案减去两行数的和就是正确的答案了. 然后发现位于一个角上的时候,我们其实只有两种选 ...

  3. bzoj 1458: 士兵占领 -- 最大流

    1458: 士兵占领 Time Limit: 10 Sec  Memory Limit: 64 MB Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵 ...

  4. mybatis源码分析(7)-----缓存Cache(一级缓存,二级缓存)

    写在前面  MyBatis 提供查询缓存,用于减轻数据库压力,提高数据库性能. MyBatis缓存分为一级缓存和二级缓存. 通过对于Executor 的设计.也可以发现MyBatis的缓存机制(采用模 ...

  5. web前端笔记整理,从入门到上天,周周更新

    由于大前端知识点太多,所以一一做了分类整理,详情可见本人博客 http://www.cnblogs.com/luxiaoyao/ 一.HTML 1.注释 格式:<!-- 注释内容 --> ...

  6. linux下使用crontab创建定时任务

    在linux中启动crontab服务: /etc/init.d/crond start crontab的命令格式 crontab -l 显示当前的crontab 文件(默认编写的crontab文件会保 ...

  7. Unity3D架构设计NavMesh寻路(未完待续)

    国庆闲来没事把NavMesh巩固一下.以Unity3D引擎为例写一个底层c# NavMesh寻路.由于Unity3D中本身自带的NavMesh寻路不能非常好的融入到游戏项目其中,所以重写一个NavMe ...

  8. 采用FPGA实现音频模数转换器

    http://www.21ic.com/app/eda/200905/42832.htm http://www.eefocus.com/article/09-10/84673s.html 摘 要 简要 ...

  9. UVa-Ecological Premium

    题目地址:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  10. 能使用html/css解决的问题就不要使用JS

    为什么说能使用html/css解决的问题就不要使用JS呢?两个字,因为简单.简单就意味着更快的开发速度,更小的维护成本,同时往往具有更好的体验,下面介绍几个实例. 1. 导航高亮 导航高亮是一种很常见 ...