项目需要监测在Windows平台的崩溃情况,折腾了两天终于弄好了,记录一下。
1.捕获崩溃信息
接到这个需求,心想应用崩溃系统会收到信号,应该有相应的函数可以监听,上网搜索,果不其然函数
SetUnhandledExceptionFilter
便有此功能,用法也不复杂,在项目里实现之后发现监测不到【Debug/Release】~~~我勒个擦
搜索 SetUnhandledExceptionFilter无效 貌似蛮多人遇到这个问题的,参考几个博客的内容
尝试了几次依然还是捕获失败,遂放弃了此种方式。
搜索有没有相关的SDK可以使用,还好找到了CrashRpt库,接口简洁接入方便,官方提供的示例
CrashRpt: An Example of Using CrashRpt API
参照示例接入,发现确实可以捕获到崩溃(*^▽^*),而且捕获的信息参数里有崩溃的必要信息
但是来了,但是造了几个崩溃测试了几把,这几个参数一直为空~~我去~~
没办法只能上传崩溃堆栈了,幸好参数里有崩溃信息文件的地址
读取该文件,哎呀,崩溃时改文件还未写入,尼玛~~~看来要自己写崩溃信息文件了
参考两种Dump(崩溃日志)文件生成的方法及比较博文成功将崩溃信息写入文件,
2.信息上报给服务器
信息上报需要注意,可能由于还未上传程序便结束了,因此主线程最好短暂休眠,给上报线程预留时间
3.分析崩溃信息,定位崩溃原因
使用WinDbg工具【需下载Windows SDK】
打开WinDbg,依次进行下面操作:
File -> Symbol File Path -> 选择pdb文件存放路径。
File -> Image File Path -> 选择exe文件存放路径。
File -> Open Crash Dump -> 选择DMP文件存放路径。
最后会弹出WinDbg对崩溃文件的初步分析的结果,在下面的输入框中输入“!analyze -v”,意思是软件进行对崩溃文件进行分析,并显示出来,然后根据分析结果定位崩溃原因。
Debug->Stop Debugging停止当前分析
至此,Windows平台捕获C++崩溃,上报并分析定位崩溃原因的功能便完成了。
实现代码如下:
#include <iostream>
#include <stdio.h>
#include <tchar.h>
#include <string>
#include <stdlib.h>
#include <windows.h>#include <strsafe.h>
#pragma comment(lib,"DbgHelp.lib")#include "include/CrashRpt.h" void log(const std::string& str)
{std::cout << str.c_str() << std::endl;
}std::string StringWideCharToUtf8(const std::wstring& strWideChar)
{std::string ret;if (!strWideChar.empty()){int nNum = WideCharToMultiByte(CP_UTF8, 0, strWideChar.c_str(), -1, nullptr, 0, nullptr, FALSE);if (nNum){char* utf8String = new char[nNum + 1];utf8String[0] = 0;nNum = WideCharToMultiByte(CP_UTF8, 0, strWideChar.c_str(), -1, utf8String, nNum + 1, nullptr, FALSE);ret = utf8String;delete[] utf8String;}}return ret;
}//将崩溃信息写入文件
typedef BOOL(WINAPI* TKPGetModuleHandleEx)(DWORD dwFlags, LPCTSTR lpModuleName, HMODULE* phModule);
VOID createCrashDump(struct _EXCEPTION_POINTERS* pExceptionPointers, std::string& path)
{//收集信息HMODULE hModule;WCHAR szModuleName[MAX_PATH] = { 0 };TKPGetModuleHandleEx pFun = (TKPGetModuleHandleEx)GetProcAddress(GetModuleHandle(L"kernel32.dll"), "GetModuleHandleExW");if (!pFun) {return;}pFun(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, (LPCWSTR)pExceptionPointers->ExceptionRecord->ExceptionAddress, &hModule);GetModuleFileName(hModule, szModuleName, ARRAYSIZE(szModuleName));//生成 mini crash dumpBOOL bMiniDumpSuccessful;WCHAR szPath[MAX_PATH];WCHAR szFileName[MAX_PATH];WCHAR* szAppName = L"DumpFile";WCHAR* szVersion = L"v1.0";DWORD dwBufferSize = MAX_PATH;HANDLE hDumpFile;SYSTEMTIME stLocalTime;MINIDUMP_EXCEPTION_INFORMATION ExpParam;GetLocalTime(&stLocalTime);GetTempPath(dwBufferSize, szPath);StringCchPrintf(szFileName, MAX_PATH, L"%s%s", szPath, szAppName);CreateDirectory(szFileName, NULL);StringCchPrintf(szFileName, MAX_PATH, L"%s%s//%s-%04d%02d%02d-%02d%02d%02d-%ld-%ld.dmp",szPath, szAppName, szVersion,stLocalTime.wYear, stLocalTime.wMonth, stLocalTime.wDay,stLocalTime.wHour, stLocalTime.wMinute, stLocalTime.wSecond,GetCurrentProcessId(), GetCurrentThreadId());hDumpFile = CreateFile(szFileName, GENERIC_READ | GENERIC_WRITE,FILE_SHARE_WRITE | FILE_SHARE_READ, 0, CREATE_ALWAYS, 0, 0);ExpParam.ThreadId = GetCurrentThreadId();ExpParam.ExceptionPointers = pExceptionPointers;ExpParam.ClientPointers = TRUE;MINIDUMP_TYPE MiniDumpWithDataSegs = (MINIDUMP_TYPE)(MiniDumpNormal| MiniDumpWithHandleData| MiniDumpWithUnloadedModules| MiniDumpWithIndirectlyReferencedMemory| MiniDumpScanMemory| MiniDumpWithProcessThreadData| MiniDumpWithThreadInfo);bMiniDumpSuccessful = MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(),hDumpFile, MiniDumpWithDataSegs, &ExpParam, NULL, NULL);path = StringWideCharToUtf8(szFileName);return;
}//程序崩溃时CrashRpt库抛出的监听事件
int CALLBACK crashCallback(CR_CRASH_CALLBACK_INFO* pInfo)
{// The application has crashed!// Close the log file here// to ensure CrashRpt is able to include it into error reportstd::string filePath;createCrashDump(pInfo->pExceptionInfo->pexcptrs, filePath);unsigned len = 0;FILE* fp = fopen(filePath.c_str(), "rb");if (fp){fseek(fp, 0, SEEK_END);len = ftell(fp);fclose(fp);}//此处将数据上报服务器【子线程上报】//主线程暂时休眠1秒Sleep(1000);log("crash------");return CR_CB_CANCEL;
}int installCrashRpt()
{CR_INSTALL_INFO info;memset(&info, 0, sizeof(CR_INSTALL_INFO));info.cb = sizeof(CR_INSTALL_INFO);info.pszAppName = _T("Test");info.pszAppVersion = _T("1.0.0");info.pszEmailSubject = _T("Test Error Report");//info.uPriorities[CR_HTTP] = 3; // 首先使用HTTP的方式发送错误报告info.uPriorities[CR_SMTP] = 2; // 然后使用SMTP的方式发送错误报告 info.uPriorities[CR_SMAPI] = 1; //最后尝试使用SMAPI的方式发送错误报告 // 捕获所有能够捕获的异常, 使用HTTP二进制编码的方式传输info.dwFlags |= CR_INST_ALL_POSSIBLE_HANDLERS;info.dwFlags |= CR_INST_HTTP_BINARY_ENCODING;info.dwFlags |= CR_INST_APP_RESTART;info.dwFlags |= CR_INST_SEND_QUEUED_REPORTS;info.pszRestartCmdLine = _T("/restart");// 隐私策略URLinfo.pszPrivacyPolicyURL = _T("http://myapp.com/privacypolicy.html");int nResult = crInstall(&info);if (nResult != 0){TCHAR szErrorMsg[512] = _T("");crGetLastErrorMsg(szErrorMsg, 512);_tprintf_s(_T("%s\n"), szErrorMsg);log("crash rpt set falied");return 1;}log("crash rpt set success");crSetCrashCallback(crashCallback, NULL);
}int main(void)
{installCrashRpt();int* p=new int;delete p;*p = 5;system("pause");return 0;
}
附:接入CrashRpt除了头文件需要的文件列表
捕获崩溃需Release模式
参考资料:
应用程序异常处理与崩溃收集
SetUnhandledExceptionFilter拦不住的崩溃
分析两种Dump(崩溃日志)文件生成的方法及比较
vs2010 利用DMP文件、pdb文件查找release下的异常行号的方法