高并发下的hook写法

HOOK原理

在并发的情况下要HOOK函数需要使用CMPXCHG8B 指令,该指令可以原子操作一次性的替换掉8字节的内存。

CMPXCHG8B 指令

该指令判断指定内存中的8字节内容和EDX:EAX (edx高32位,eax,低32位)中的64字节内容是否相同,
如果相同,就把ECX:EBX(ecx高32位,ebx,低32位)中的内容替换到指定内存。
如果不相同,就把指定内存中的值替换到ECX:EBX中。

#include"ntddk.h"
ULONG addr = 0x80546a98;   //这是SwapContext+8的位置
_declspec(naked) void Swap()
{
	__asm{

		push eax
		mov eax,0x8003f048  //记录切换的次数
		inc dword ptr [eax]
		pop  eax
		//下面开始恢复原函数的指令
		or      cl, cl
		mov     byte ptr es : [esi + 0x2d], 0x2
		pushfd
		jmp addr  //在自己的hook函数处理完后,跳转到原函数+8的位置
	}
}

VOID __stdcall UnLoad(PDRIVER_OBJECT pDriver)
{
	DbgPrint("Driver 卸载成功");
}
NTSTATUS DriverEntry(PDRIVER_OBJECT pDriver, PUNICODE_STRING pRegPath)
{
	pDriver->DriverUnload = UnLoad;
	ULONG JmpAddr = (ULONG)Swap - 0x80546a90 - 5;//0x80546a90是SwapContext函数的地址
	ULONG des[2] = {0};
	DbgPrint("JmpAddr=%x\n",JmpAddr);
	//去掉最高的8位  然后+e9
	des[0] = (JmpAddr<<8)+ 0xe9;  //xxxxxx e9    000000xx
	//保留最低的8位  并移到最高位
	des[1] =(JmpAddr>>24);

	DbgPrint("des[0]=%x\n", des[0]);
	DbgPrint("des[1]=%x\n", des[1]);

	//准备好写入的8字节数据
	__asm{
		//把 EDX : EAX设置为目标的值
		mov esi, 0x80546a90
		//eax=0xc626c90a
		mov eax, [esi]
		//edx=0x9c022d46 
		mov edx, [esi+4]
		// 把ECX:EBX  设置为要写入的值
		
		//ebx= xx2d029c
		mov ebx, des[0]
		//ecx=e8xxx
		mov ecx, des[4]
		
		lock CMPXCHG8B [esi] ;
		
	}
	DbgPrint("高并发hook成功");
	return STATUS_SUCCESS;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的 demo,演示了如何使用 Hook 来修改函数的行为,具体来说,这个 demo 实现了一个简单的加法器,但是在加法器计算结束后会输出一段字符串 "Hooked!",以示我们成功地使用了 Hook 修改了函数的行为。 ```c++ #include <iostream> #include <Windows.h> // 原始的加法函数 int add(int a, int b) { return a + b; } // Hook 后的加法函数 int hooked_add(int a, int b) { int result = add(a, b); std::cout << "Hooked!" << std::endl; return result; } int main() { // Hook 前调用原始的加法函数 std::cout << add(1, 2) << std::endl; // Hook DWORD old_protect; VirtualProtect((LPVOID)add, sizeof(add), PAGE_EXECUTE_READWRITE, &old_protect); *(uintptr_t*)add = (uintptr_t)hooked_add; // Hook 后调用加法函数 std::cout << add(1, 2) << std::endl; // 恢复原始函数 *(uintptr_t*)add = (uintptr_t)hooked_add; VirtualProtect((LPVOID)add, sizeof(add), old_protect, &old_protect); // 再次调用原始加法函数 std::cout << add(1, 2) << std::endl; return 0; } ``` 在这个 demo 中,我们首先定义了一个简单的加法函数 `add`。然后我们实现了一个新的函数 `hooked_add`,它会在计算结束后输出一段字符串。 接着,我们使用 `VirtualProtect` 函数将 `add` 函数所在的内存页标记为可写,以便我们修改它。然后,我们将 `add` 函数的地址转换为 `uintptr_t` 类型的指针,并将这个指针的值修改为 `hooked_add` 函数的地址,实现了 Hook。 最后,我们再次将 `add` 函数的地址修改回原始的地址,并将内存页恢复为原始的保护级别。这样,我们就成功地完成了 Hook 的过程,并且在 Hook 后的加法函数计算结束后输出了一段字符串。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值