⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 firstexc.cpp

📁 微软提供的截取Win32 API函数的开发包和例子detours-src-1.2.rar
💻 CPP
字号:
//////////////////////////////////////////////////////////////////////////////
//
//	Module:		firstexc.lib
//  File:		firstexc.cpp
//	Author:		Galen C. Hunt
//
//	First-Change Exception Handling.  Version 1.2. (Build 35)
//
//	Copyright 1995-1999, Microsoft Corporation
//
//	http://www.research.microsoft.com/sn/detours
//
//	For more information on exception handling, see "A Crash Course on the
//  Depths of Win32 Structured Exception Handling," by Matt Pietrek in the
//  January 1997 issue of Microsoft Systems Journal.
//
#include <stdio.h>
#include <windows.h>
#include "detours.h"

//////////////////////////////////////////////////////////////////////////////
//
static BOOL							s_bExceptionDetourInstalled = FALSE;
static LPTOP_LEVEL_EXCEPTION_FILTER s_pFirstChanceFilter = NULL;

ULONG (NTAPI *s_pNtContinue)(IN PCONTEXT ContextRecord,
							 IN BOOLEAN TestAlerts) = NULL;

VOID (NTAPI *s_pKiUserExceptionDispatcher)(IN PEXCEPTION_RECORD ExceptionRecord,
										   IN PCONTEXT ContextFrame) = NULL;

VOID (NTAPI *s_pTrampolineToKiUserExceptionDispatcher)(IN PEXCEPTION_RECORD pExceptRec,
													   IN CONTEXT *pContext) = NULL;

//////////////////////////////////////////////////////////////////////////////
//
//  This function effectively removes all try..catch frames for the current
//  stack.  It forces all exceptions to be treated as unhandled exceptions.
//
static VOID WINAPI RemoveAllExceptionHandlers(VOID)
{
	// The basic, OS defined exception frame
	struct EXCEPTION_REGISTRATION
	{
		EXCEPTION_REGISTRATION* prev;
		FARPROC handler;
	};	
	
	EXCEPTION_REGISTRATION * pVCExcRec = NULL;
	EXCEPTION_REGISTRATION * pLastGood = NULL;
	
	__asm mov eax, FS:[0];
	__asm mov [pVCExcRec], eax;
	
	for (pLastGood = pVCExcRec; (ULONG)pVCExcRec != ~0ul; ) {
		if ((ULONG)pVCExcRec >= 0x30000000)
			break;
		
		pLastGood = pVCExcRec;
		pVCExcRec = (EXCEPTION_REGISTRATION *)(pVCExcRec->prev);
	}
	
	__asm mov eax, [pLastGood];
	__asm mov FS:[0], eax;
}

//////////////////////////////////////////////////////////////////////////////
// Routine Description:
//
//    This routine is entered on return from kernel mode to dispatch a user
//    mode exception. If a frame based handler handles the exception, then
//    the execution is continued. Else last chance processing is performed.
//
//    NOTE:  This procedure is not called, but rather dispatched to.
//           It depends on there not being a return address on the stack
//           (assumption w.r.t. argument offsets.)
//
// Arguments:
//    ExceptionRecord (esp+0) - Supplies a pointer to an exception record.
//    ContextRecord (esp+4) - Supplies a pointer to a context frame.
//
// Return Value:
//    None.
//
static VOID __declspec(naked) NTAPI
Detour_KiUserExceptionDispatcher(PEXCEPTION_RECORD pExceptRec,
								 CONTEXT *pContext)
{
	__asm {
		xor		eax, eax				; // Create fake return address on stack.
		push 	eax						; // (Generally, we are called by the kernel.)

		push 	ebp						; // Prolog
		mov		ebp, esp				;
		sub		esp, __LOCAL_SIZE		;
	}
	
	LPTOP_LEVEL_EXCEPTION_FILTER pFirstChanceFilter;
	EXCEPTION_POINTERS ep;
	DWORD dwReturn;
	DWORD dwError;
	
	ep.ExceptionRecord = pExceptRec;
	ep.ContextRecord = pContext;
	pFirstChanceFilter = s_pFirstChanceFilter;
	dwReturn = EXCEPTION_CONTINUE_SEARCH;
	dwError = 0;
	
	if (s_pFirstChanceFilter) {
		dwReturn = pFirstChanceFilter(&ep);
	}

	if (dwReturn == EXCEPTION_CONTINUE_EXECUTION) {
		dwError = s_pNtContinue(pContext, 0);
		// This call should *NEVER* return.  If it does, we want to fail to the debugger.
		RemoveAllExceptionHandlers();
	}
	
	if (dwReturn == EXCEPTION_EXECUTE_HANDLER) {		// Special: Call debugger.
		RemoveAllExceptionHandlers();
	}	
	
	__asm {
		mov		ebx, pExceptRec			;
		mov		ecx, pContext			;
		push	ecx						;
		push	ebx						;
		mov		eax, s_pTrampolineToKiUserExceptionDispatcher;
		jmp		eax						;
		;
		; The above code should never return.
		;
		int		3						; // Break!
		;
		mov		esp, ebp				; // Epilog
		pop		ebp						;
		ret								;
	}
}

//////////////////////////////////////////////////////////////////////////////
//
//	Set the first-chance exception filter.
//
//  Returns the pointer to the last first-chance exception filter if there
//  was one.  If this is the first first-chance exception filter, installs
//  the necessary detour and acquires the appropriate function pointers.
//  If the parameter is NULL, first-chance exception filtering is disabled.
//
//  A first-chance exception filter should always return one of three
//  possible codes:
//
//    EXCEPTION_CONTINUE_SEARCH:
//        The exception was not handled by this filter; continue the
//        search for the appropriate exception handler.
//
//    EXCEPTION_CONTINUE_EXECUTION:
//        The exception was handled by this filter; continue execution
//        at the point were the exception was thrown.
//
//    EXCEPTION_EXECUTE_HANDLER:
//        Drastic failure in the exception filter.  Process the
//        exception as if no exception handlers were installed.
//        (i.e. Give the user a chance to invoke the debugger.)
//
LPTOP_LEVEL_EXCEPTION_FILTER WINAPI
DetourFirstChanceExceptionFilter(LPTOP_LEVEL_EXCEPTION_FILTER pNewFirstChanceFilter)
{
	if (!s_bExceptionDetourInstalled) {
		s_bExceptionDetourInstalled = TRUE;

		s_pNtContinue =	(ULONG (NTAPI *)(IN PCONTEXT, IN BOOLEAN))
			DetourFindFunction("ntdll.dll", "NtContinue");
		s_pKiUserExceptionDispatcher =
			(VOID (NTAPI *)(IN PEXCEPTION_RECORD, IN PCONTEXT))
			DetourFindFunction("ntdll.dll", "KiUserExceptionDispatcher");
		s_pTrampolineToKiUserExceptionDispatcher =
			(VOID (NTAPI *)(IN PEXCEPTION_RECORD, IN CONTEXT *))
			DetourFunction((PBYTE)s_pKiUserExceptionDispatcher,
						   (PBYTE)Detour_KiUserExceptionDispatcher);
	}

	LPTOP_LEVEL_EXCEPTION_FILTER pOldFirstChanceFilter = s_pFirstChanceFilter;
	s_pFirstChanceFilter = pNewFirstChanceFilter;
	return pOldFirstChanceFilter;
}
//
///////////////////////////////////////////////////////////////// End of File.

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -