首页 > 代码库 > c++ 服务器程序bug dump

c++ 服务器程序bug dump

调试服务器程序时最怕遇到需要运行10天半个月才遇到一次的bug,这种bug很难还原现场,同时还要时刻注意服务器是否挂掉。

本文给出一个解决方法可以极大的提高调试效率。

使用本文方法可以在断言失败时自动dump,可用于还原bug环境进行调试。另外崩溃时也会自动记录crash dump。


断言函数


bool xassert(bool r){

    if(!r)    __asm int 3
   return r;
}

最终异常处理函数,遇到这里的话说明程序只能挂掉了,写crash dump

LONG	WINAPI LastExceptionHandler(PEXCEPTION_POINTERS pEi){
	auto h=CreateFile("crash.dmp",GENERIC_WRITE,0,nullptr,CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL,nullptr);
	if(h!=INVALID_HANDLE_VALUE){
		MINIDUMP_EXCEPTION_INFORMATION mei;
		mei.ClientPointers=false;
		mei.ThreadId=GetCurrentThreadId();
		mei.ExceptionPointers=pEi;
		MiniDumpWriteDump(GetCurrentProcess(),GetCurrentProcessId(),h,MiniDumpWithPrivateReadWriteMemory,&mei,nullptr,nullptr);
		CloseHandle(h);
	}
	return	EXCEPTION_CONTINUE_SEARCH;
}
最先异常处理函数,用于处理xassert里面的int 3,写调试dump

LONG WINAPI FirstExceptionHandler(PEXCEPTION_POINTERS pEi){

	switch(pEi->ExceptionRecord->ExceptionCode){
		//所有的软件断点都是我提交的
		case	EXCEPTION_BREAKPOINT://软件断点int3 0xcc
		{
			static	int	nDump=0;
			static	char	tmp[1024];
			_snprintf_s(tmp,1000,"debug%02d.dmp",++nDump);

			auto h=CreateFile(tmp,GENERIC_WRITE,0,nullptr,CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL,nullptr);
			if(h!=INVALID_HANDLE_VALUE){
				MINIDUMP_EXCEPTION_INFORMATION mei;
				mei.ClientPointers=false;
				mei.ThreadId=GetCurrentThreadId();
				mei.ExceptionPointers=pEi;
				MiniDumpWriteDump(GetCurrentProcess(),GetCurrentProcessId(),h,MiniDumpWithPrivateReadWriteMemory,&mei,nullptr,nullptr);
				CloseHandle(h);
			}
			pEi->ContextRecord->Eip+=1;

			return	EXCEPTION_CONTINUE_EXECUTION;
			
		}break;
		case	EXCEPTION_SINGLE_STEP://debug 硬件断点
		{
		}break;
	}
	return	EXCEPTION_CONTINUE_SEARCH;
}


最后在程序初始化时加上


	AddVectoredExceptionHandler(0,FirstExceptionHandler);
	SetUnhandledExceptionFilter(LastExceptionHandler);


代码原理。

xassert失败时执行一个int3指令,产生调试中断

使用异常处理函数捕获调试中断,用debugAPI写dump后恢复到调试指令下一条继续执行。

最终异常处理函数捕获所有未能处理的异常,写crash dump后程序自动崩溃。


c++ 服务器程序bug dump