写一个Windows上的守护进程(7)捕获异常并生成dump
写一个Windows上的守护进程(7)捕获异常并生成dump
谁都不能保证自己的代码不出bug。一旦出了bug,最好是崩溃掉,这样很快就能被发现,若是不崩溃,只是业务处理错了,就麻烦了,可能很长时间之后才能被发现。
那么如果崩溃掉,怎么查错呢?
写过Windows驱动的同学应该知道,一旦崩溃,系统会生成dump文件,然后就可以根据dump文件、pdb文件、源码用windbg分析了。应用层的程序同样可以在崩溃的时候生成dump文件,只是没人帮你完成这个步骤,得自己动手。
1. API
这里涉及到的主要API就一个:MiniDumpWriteDump——生成dump文件,我说再多也没有官方权威,所以要看详情,还是请移步MSDN。其它辅助API就是设置异常处理器的函数,此处不介绍。
2. 异常类型
读过《Windows核心编程》的同学应该知道SEH——Windows结构化异常处理,当时我读了之后,就以为所有的异常都是在SEH体系内的,还接着“以为”了很多年。后来发现并不是:因为我写的捕获异常并生成dump文件的库并没有发挥应有的作用,该崩溃的地方还是崩溃了并且没有生成dump文件!
google后发现了这么一篇文章:http://www.codeproject.com/Articles/207464/Exception-Handling-in-Visual-Cplusplus。
我总结一下:Windows下并非所有的异常都走SEH体系,还有CRT错误处理体系;也并不是只有异常才会导致程序退出,接收到Windows支持的几个信号也会导致程序退出,比如SIGABRT等等;有的“异常”处理器不是针对进程的,而只对当前线程有效,必须每个线程都设置处理器。
所以,要想捕获所有的“异常”,就得:
l 调用SetUnhandledExceptionFilter设置SEH异常处理函数
l 调用_set_purecall_handler设置处理纯虚函数调用导致的错误的处理函数
l 调用_set_new_handler设置new失败导致的错误的处理函数
l 调用_set_invalid_parameter_handler设置无效参数导致的错误的处理函数
l 调用signal设置信号SIGABRT、SIGINT、SIGTERM的处理函数
l 调用set_terminate设置terminate处理函数。这个只对当前线程有效
l 调用set_unexpected设置unexpected处理函数。这个也是只对当前线程有效
l 调用signal设置信号SIGFPE、SIGILL、SIGSEGV的处理函数。这个也是对当前线程有效
3. 异常信息获取
MiniDumpWriteDump 的ExceptionParam需要填充上异常信息,那么如何填充呢?
上面那篇文章已经指出了方法:用vc源码文件invarg.c中函数_invoke_watson的代码。
他代码中用的是vc8.0的源码,我是用vc2008编译的,所以就取的vc2008的源码,但是好像不用在意取哪个vc版本的源码。
4. 调试dump文件
我把生成的dump文件放在了应用程序所在目录,这样一眼就可以看出程序有没有崩溃过。一旦生成了dump文件,我们只要把二进制、pdb文件、dump文件放在一起,然后用vs打开dump文件,直接运行就可以调试了。(所以我们提交二进制的时候要把pdb也一同提交。)
我修改了下上面文章中的源码,封装成了两个函数,把我用它的测试代码测试的结果写在了代码中。
我的使用:main函数起来后,设置所有处理函数;在每个任务线程设置线程相关的处理函数。
源码:https://git.oschina.net/mkdym/DaemonSvc.git (主)&& https://github.com/mkdym/DaemonSvc.git (提升逼格用的)。
2015年11月10日星期二
写一个Windows上的守护进程(7)捕获异常并生成dump的更多相关文章
- 写一个Windows上的守护进程(8)获取进程路径
写一个Windows上的守护进程(8)获取进程路径 要想守护某个进程,就先得知道这个进程在不在.我们假设要守护的进程只会存在一个实例(这也是绝大部分情形). 我是遍历系统上的所有进程,然后判断他们的路 ...
- 写一个Windows上的守护进程(6)Windows服务
写一个Windows上的守护进程(6)Windows服务 守护进程因为要开机启动,还要高权限,所以我就把它做成Windows服务了. 关于Windows服务的官方文档,大家可以看https://msd ...
- 写一个Windows上的守护进程(5)文件系统重定向
写一个Windows上的守护进程(5)文件系统重定向 在Windows上经常操作文件或注册表的同学可能知道,有"文件系统/注册表重定向"这么一回事.大致来说就是32位程序在64位的 ...
- 写一个Windows上的守护进程(4)日志其余
写一个Windows上的守护进程(4)日志其余 这次把和日志相关的其他东西一并说了. 一.vaformat C++日志接口通常有两种形式:流输入形式,printf形式. 我采用printf形式,因为流 ...
- 写一个Windows上的守护进程(3)句柄的管理
写一个Windows上的守护进程(3)句柄的管理 在Windows中编程,跟HANDLE打交道是家常便饭.为了防止忘记CloseHandle,我都是使用do-while-false手法: void f ...
- 写一个Windows上的守护进程(2)单例
写一个Windows上的守护进程(2)单例 上一篇的日志类的实现里有个这: class Singleton<CLoggerImpl> 看名字便知其意--单例.这是一个单例模板类. 一个进程 ...
- 写一个Windows上的守护进程(1)开篇
写一个Windows上的守护进程(1)开篇 最近由于工作需要,要写一个守护进程,主要就是要在被守护进程挂了的时候再把它启起来.说起来这个功能是比较简单的,但是我前一阵子写了好多现在回头看起来比较糟糕的 ...
- 写一个Windows服务
做了两个和Windows服务有关的项目了,最开始的时候没做过,不懂,现在明白了许多.需要注意的是,如果不想登录什么的,最后在添加安装程序的那里选择那个字长的右键属性,把启动方式改为local syst ...
- 写一个限制上传文件大小和格式的jQuery插件
在客户端上传文件,通常需要限制文件的尺寸和格式,最常用的做法是使用某款插件,一些成熟的插件的确界面好看,且功能强大,但美中不足的是:有时候会碰到浏览器兼容问题.本篇就来写一个"原生态&quo ...
随机推荐
- C/C++中的常成员函数
代码: #include <iostream> using namespace std; class A{ public: void func1(){ cout<<" ...
- mysql update语句,修改字段,,或者是批量修改字段
更新一个字段,在它的后面加一个字符串,不查询数据库得到这个字段值 怎么添加?? 例如: 我的test表,有个user字段,我现在想在它后面加了另一个用户的名字 我在mysql数据库这样写 UPDATE ...
- 使用canvas来实时播放RTSP视频
HTML5的标签可以用使用下来面的方式来播放静态视频 <video width="320" height="240" controls="con ...
- JavaScript语法支持严格模式:"use strict"
如果给JavaScript代码标志为“严格模式”,则其中运行的所有代码都必然是严格模式下的.其一:如果在语法检测时发现语法问题,则整个代码块失效,并导致一个语法异常.其二:如果在运行期出现了违反严格模 ...
- mac删除顽固图标
cd /Users/shelley/Library/Application\ Support/Dock cp 10CCA448-0975-41DE-B47A-8E89FD634227.db 10 ...
- javascript第二课练习
<script type="text/javascript"> window.onload=function() //网页全部加载完后执行 { va ...
- JDK Linux环境配置
① $sudo vi /etc/profile ② 在末尾行添加 #set java environment JAVA_HOME=/usr/local/jdk1.7.0 CLASSPATH=.:$JA ...
- php 和 apache的关系
例如在客户端游览器输入他也回把这个地址传送到192.168.1.100里的apache里的,apache一看你传过来的是Php文件,如果在服务器没装php的情况下,他也会把这个文件打开,把里面的代码全 ...
- Idea使用记录--每次修改JS文件都需要重启Idea才能生效解决方法
最近开始使用Idea,有些地方的确比eclipse方便.但是我发现工程每次修改JS或者是JSP页面后,并没有生效,每次修改都需要重启一次Tomcat这样的确不方便.我想Idea肯定有设置的方法,不可能 ...
- [TYVJ] P1023 奶牛的锻炼
奶牛的锻炼 背景 Background USACO 描述 Description 奶牛Bessie有N分钟时间跑步,每分钟她可以跑步或者休息.若她在第i分钟跑步,可以跑出D_i米,同时疲倦程度增加 ...