Api函数钩子

windows 下,api钩子目前比较稳定和健壮的方法。

1、读取pe 文件:ImageDirectoryEntryToData

2、找到函数表中的 函数地址

(PROC *)&pThunk->u1.Function;

3、替换函数地址

WriteProcessMemory(GetCurrentProcess(), ppfn, &pfnNew, sizeof(pfnNew), nullptr)


该方法和cpu无关,不需要汇编知识和其他知识,只需要c++相关知识和pe文件结构。

需要注意的有:动态LoadLibrary之后,也需要再一次挂钩。如果需要在其它进程挂钩api,还需要dll挂载知识。


#include "stdafx.h"
#include <windows.h>
#include <iostream>
#include <ImageHlp.h>
#include <tlhelp32.h>
#include <psapi.h>
#pragma comment(lib, "ImageHlp")


void ReplaceIATEntryInAllMods(PCSTR pszCalleeModName, PROC pfnCurrent, PROC pfnNew);
int WINAPI MyMessageBox(HWND hWnd, LPCWSTR lpText, LPCWSTR lpCaption, UINT uType);
PROC g_pOldMsgBox = nullptr;
int main()
{	
	MessageBox(nullptr, L"修改前", L"", MB_OK);
	g_pOldMsgBox = ::GetProcAddress(GetModuleHandleA("User32.dll"), "MessageBoxW");
	ReplaceIATEntryInAllMods("User32.dll", g_pOldMsgBox, (PROC)MyMessageBox);
	MessageBox(nullptr, L"修改前", L"", MB_OK);
	getchar();
    return 0;
}


int WINAPI MyMessageBox(HWND hWnd, LPCWSTR lpText, LPCWSTR lpCaption, UINT uType)
{
	typedef int(WINAPI *pMsgBoxFunType) (HWND hWnd, LPCWSTR lpText, LPCWSTR lpCaption, UINT uType);
	pMsgBoxFunType pMsgBox = (pMsgBoxFunType)((void *)(g_pOldMsgBox));
	return (*pMsgBox)(nullptr, L"修改后", L"", MB_OK);
}


void ReplaceIATEntryInOneMod(PCSTR pszCalleemodName,PROC pfnCurrent,PROC pfnNew,HMODULE hModCaller)
{
	ULONG ulSize = 0;
	PIMAGE_IMPORT_DESCRIPTOR pImportDesc = nullptr;
	try
	{
		pImportDesc = (PIMAGE_IMPORT_DESCRIPTOR)ImageDirectoryEntryToData(hModCaller,TRUE, IMAGE_DIRECTORY_ENTRY_IMPORT,&ulSize);
	}
	catch (...)
	{


	}
	if (nullptr == pImportDesc)
	{
		return;
	}


	for (; pImportDesc->Name; ++pImportDesc)
	{
		PSTR pszModName = (PSTR)((PBYTE)hModCaller + pImportDesc->Name);
		if (0 != lstrcmpiA(pszModName, pszCalleemodName))
		{
			continue;
		}


		PIMAGE_THUNK_DATA pThunk = (PIMAGE_THUNK_DATA)((PBYTE)hModCaller + pImportDesc->FirstThunk);
		for (;pThunk->u1.Function; ++ pThunk )
		{
			PROC *ppfn = (PROC *)&pThunk->u1.Function;
			bool bFind = (*ppfn == pfnCurrent);
			if (!bFind)
			{
				continue;
			}
			if (!WriteProcessMemory(GetCurrentProcess(), ppfn, &pfnNew, sizeof(pfnNew), nullptr)
				&& (ERROR_NOACCESS == GetLastError()))
			{
				DWORD dwOldProtect = 0;
				if (VirtualProtect(ppfn, sizeof(pfnNew), PAGE_WRITECOPY, &dwOldProtect))
				{
					WriteProcessMemory(GetCurrentProcess(), ppfn, &pfnNew, sizeof(pfnNew), nullptr);
					VirtualProtect(ppfn, sizeof(pfnNew), dwOldProtect, &dwOldProtect);
				}
			}
			return;
		}
	}
}


HMODULE ModuleFromAddress(PVOID pv)
{
	MEMORY_BASIC_INFORMATION mbi = {};
	return ((0 != VirtualQuery(pv, &mbi, sizeof(mbi))) ? (HMODULE)mbi.AllocationBase:nullptr);
}


void ReplaceIATEntryInAllMods(PCSTR pszCalleeModName, PROC pfnCurrent, PROC pfnNew)
{
	HMODULE hmodThisMod = ModuleFromAddress(ReplaceIATEntryInAllMods);
	hmodThisMod = GetModuleHandle(NULL);
	hmodThisMod = nullptr;


	auto hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPMODULE, GetCurrentProcessId());
	if (INVALID_HANDLE_VALUE == hSnap)
	{
		return;
	}


	MODULEENTRY32 me = { sizeof(me) };
	if (!Module32First(hSnap, &me))
	{
		CloseHandle(hSnap);
		return;
	}


	do
	{
		if (me.hModule != hmodThisMod)
		{
			ReplaceIATEntryInOneMod(pszCalleeModName, pfnCurrent, pfnNew, me.hModule);
		}
	} while (Module32Next(hSnap, &me));
	CloseHandle(hSnap);
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
目前最好的EasyHook的完整Demo程序,包括了Hook.dll动态库和Inject.exe注入程序。 Hook.dll动态库封装了一套稳定的下钩子的机制,以后对函数钩子,只需要填下数组表格就能实现了,极大的方便了今后的使用。 Inject.exe部分是用MFC写的界面程序,只需要在界面上输入进程ID就能正确的HOOK上相应的进程,操作起来非常的简便。 这个Demo的代码风格也非常的好,用VS2010成功稳定编译通过,非常值得下载使用。 部分代码片段摘录如下: //【Inject.exe注入程序的代码片段】 void CInjectHelperDlg::OnBnClickedButtonInjectDllProcessId() { ////////////////////////////////////////////////////////////////////////// //【得到进程ID值】 UINT nProcessID = 0; if (!GetProcessID(nProcessID)) { TRACE(_T("%s GetProcessID 失败"), __FUNCTION__); return; } ////////////////////////////////////////////////////////////////////////// //【得到DLL完整路径】 CString strPathDLL; if (!GetDllFilePath(strPathDLL)) { TRACE(_T("%s GetDllFilePath 失败"), __FUNCTION__); return; } ////////////////////////////////////////////////////////////////////////// //【注入DLL】 NTSTATUS ntStatus = RhInjectLibrary(nProcessID, 0, EASYHOOK_INJECT_DEFAULT, strPathDLL.GetBuffer(0), NULL, NULL, 0); if (!ShowStatusInfo(ntStatus)) { TRACE(_T("%s ShowStatusInfo 失败"), __FUNCTION__); return; } } //【Hook.dll动态库的代码片段】 extern "C" __declspec(dllexport) void __stdcall NativeInjectionEntryPoint(REMOTE_ENTRY_INFO* InRemoteInfo) { if (!DylibMain()) { TRACE(_T("%s DylibMain 失败"), __FUNCTION__); return; } } FUNCTIONOLDNEW_FRMOSYMBOL array_stFUNCTIONOLDNEW_FRMOSYMBOL[]= { {_T("kernel32"), "CreateFileW", (void*)CreateFileW_new}, {_T("kernel32"), "CreateFileA", (void*)CreateFileA_new}, {_T("kernel32"), "ReadFile", (void*)ReadFile_new} }; BOOL HookFunctionArrayBySymbol() { /////////////////////////////////////////////////////////////// int nPos = 0; do { /////////////////////////////// FUNCTIONOLDNEW_FRMOSYMBOL* stFunctionOldNew = &g_stFUNCTIONOLDNEW_FRMOSYMBOL[nPos]; if (NULL == stFunctionOldNew->strModulePath) { break; } /////////////////////////////// if (!HookFunctionBySymbol(stFunctionOldNew->strModulePath, stFunctionOldNew->strNameFunction, stFunctionOldNew->pFunction_New)) { TRACE(_T("%s HookFunctionBySymbol 失败"), __FUNCTION__); return FALSE; } } while(++nPos); /////////////////////////////////////////////////////////////// return TRUE; } HANDLE WINAPI CreateFileW_new( PWCHAR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode, LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes, HANDLE hTemplateFile ) { TRACE(_T("CreateFileW_new. lpFileName = %s"), lpFileName); return CreateFileW( lpFileName, dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile); }

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值