C++异常捕获生成minidump windows

#if defined (_WIN32) || defined (WIN32)
#include <Windows.h>
#include <dbghelp.h>
#pragma comment(lib, "dbghelp.lib")
long __stdcall exceptionHander(_EXCEPTION_POINTERS*   excp);			//全局异常处理捕获
#endif

int main(int argc, char *argv[])
{	
#if defined (_WIN32) || defined (WIN32)
	SetUnhandledExceptionFilter(exceptionHander);
#endif
	QApplication a(argc, argv);
	MainWindow w;
	w.show();
	QTextCodec::setCodecForLocale(QTextCodec::codecForName("utf8"));
	QTextCodec::setCodecForTr(QTextCodec::codecForName("utf8"));
	QTextCodec::setCodecForCStrings(QTextCodec::codecForName("utf8"));
	return a.exec();
}

/*
 *ExceptionCode 最可能遇到的几种类型
 *C0000005h读写内存冲突
 *C0000094h非法除0
 *C00000FDh堆栈溢出或者说越界
 *80000001h由Virtual Alloc建立起来的属性页冲突
 *C0000025h不可持续异常,程序无法恢复执行,异常处理例程不应处理这个异常
 *C0000026h在异常处理过程中系统使用的代码,如果系统从某个例程莫名奇妙的返回,则出现此代码,例如调用RtlUnwind时没有ExceptionRecord参数时产生的异常填入的就是这个代码
 *80000003h调试时因代码中int3中断
 *80000004h处于被单步调试状态
 *
 *返回值:
 *
 *EXCEPTION_EXECUTE_HANDLER equ 1 表示我已经处理了异常,可以结束了
 *EXCEPTION_CONTINUE_SEARCH equ 0 表示我不处理,其他人来吧,于是windows调用默认的处理程序显示一个错误框,并结束
 *EXCEPTION_CONTINUE_EXECUTION equ -1 表示错误已经被修复,请从异常发生处继续执行
*/

long __stdcall exceptionHander(_EXCEPTION_POINTERS*   excp)
{
	QDir dumpDir(QApplication::applicationDirPath() + "/dump/");
	if(!dumpDir.exists())
	{
		dumpDir.mkdir(QApplication::applicationDirPath() + "/dump/");
	}
	QString strPath = QApplication::applicationDirPath() + "/dump/"
			+  QDateTime::currentDateTime().toString("yyyy-MM-dd-hh-mm-ss")
			+ ".dmp";
	qDebug()<<strPath;
	//创建转储文件
	HANDLE hFile = CreateFile(strPath.toStdWString().c_str(), GENERIC_WRITE,
							  FILE_SHARE_WRITE, NULL, CREATE_NEW,
							  FILE_ATTRIBUTE_NORMAL, NULL );
	if(hFile != INVALID_HANDLE_VALUE)
	{
		MINIDUMP_EXCEPTION_INFORMATION exptInfo;
		exptInfo.ThreadId = ::GetCurrentThreadId();
		exptInfo.ExceptionPointers = excp;
		//将dump信息写入到文件中
		MiniDumpWriteDump(GetCurrentProcess(),
						  GetCurrentProcessId(),
						  hFile, MiniDumpNormal,
						  &exptInfo, NULL, NULL);
	}
	
	QMessageBox::critical(NULL,TR("程序异常"),TR("程序产生异常,已生成dump转储文件"));
	return EXCEPTION_EXECUTE_HANDLER;
}

程序是采用Qt写的,当程序运行过程中出现了crash,就会自动生成dump文件到和exe同级的dump文件夹下

dump文件可以直接拖到VS里面进行分析

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要获取MINIDUMP_SYSTEM_INFO,您可以使用Windows API函数MiniDumpWriteDumpMiniDumpReadDumpStream。以下是一个示例代码片段,演示如何使用这些函数来获取MINIDUMP_SYSTEM_INFO: ```c++ // 定义MINIDUMP_SYSTEM_INFO结构体 typedef struct _MINIDUMP_SYSTEM_INFO { USHORT ProcessorArchitecture; USHORT ProcessorLevel; USHORT ProcessorRevision; USHORT Reserved0; ULONG64 CpuMaxMhz; ULONG64 CpuCurMhz; ULONG64 CpuLimitMhz; ULONG64 CpuAvgMhz; ULONG64 MemorySize; ULONG64 BootTime; ULONG64 SystemTime; ULONG64 TimeZoneBias; ULONG TimeZoneId; ULONG Reserved1[8]; } MINIDUMP_SYSTEM_INFO, *PMINIDUMP_SYSTEM_INFO; // 获取MINIDUMP_SYSTEM_INFO BOOL GetMiniDumpSystemInfo(HANDLE hProcess, LPVOID lpMiniDumpBuffer, DWORD dwBufferSize, PMINIDUMP_SYSTEM_INFO pSystemInfo) { DWORD dwStreamSize = 0; PVOID pStream = MiniDumpReadDumpStream(lpMiniDumpBuffer, SystemInfoStream, NULL, &dwStreamSize); if (pStream == NULL) { return FALSE; } if (dwStreamSize != sizeof(MINIDUMP_SYSTEM_INFO)) { return FALSE; } CopyMemory(pSystemInfo, pStream, dwStreamSize); return TRUE; } // 使用MiniDumpWriteDump生成MiniDump文件 BOOL GenerateMiniDump(HANDLE hProcess, DWORD dwProcessId, LPCWSTR lpFilePath) { HANDLE hDumpFile = CreateFileW(lpFilePath, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if (hDumpFile == INVALID_HANDLE_VALUE) { return FALSE; } MINIDUMP_EXCEPTION_INFORMATION exceptionInfo = { 0 }; exceptionInfo.ThreadId = GetCurrentThreadId(); exceptionInfo.ExceptionPointers = NULL; exceptionInfo.ClientPointers = FALSE; BOOL bResult = MiniDumpWriteDump(hProcess, dwProcessId, hDumpFile, MiniDumpNormal, &exceptionInfo, NULL, NULL); CloseHandle(hDumpFile); return bResult; } // 使用示例 HANDLE hProcess = GetCurrentProcess(); DWORD dwProcessId = GetCurrentProcessId(); WCHAR szFilePath[MAX_PATH] = L"C:\\crash.dmp"; if (GenerateMiniDump(hProcess, dwProcessId, szFilePath)) { HANDLE hFile = CreateFileW(szFilePath, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hFile != INVALID_HANDLE_VALUE) { DWORD dwFileSize = GetFileSize(hFile, NULL); LPVOID lpMiniDumpBuffer = VirtualAlloc(NULL, dwFileSize, MEM_COMMIT, PAGE_READWRITE); if (lpMiniDumpBuffer != NULL) { DWORD dwBytesRead = 0; if (ReadFile(hFile, lpMiniDumpBuffer, dwFileSize, &dwBytesRead, NULL)) { MINIDUMP_SYSTEM_INFO systemInfo = { 0 }; if (GetMiniDumpSystemInfo(hProcess, lpMiniDumpBuffer, dwFileSize, &systemInfo)) { // 在这里使用MINIDUMP_SYSTEM_INFO } } VirtualFree(lpMiniDumpBuffer, 0, MEM_RELEASE); } CloseHandle(hFile); } } ``` 请注意,上述示例代码仅用于说明如何获取MINIDUMP_SYSTEM_INFO,您需要根据您的具体情况进行修改和适应。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值