我对 HOOK 派遣例程函数表的体会。。。

关键点:

调试函数得到的信息:

 

kd> u nt!IofCallDriver
nt!IofCallDriver:
804ef120 ff2500d25480    jmp     dword ptr [nt!pIofCallDriver (8054d200)]
804ef126 cc              int     3
804ef127 cc              int     3
804ef128 cc              int     3
804ef129 cc              int     3
804ef12a cc              int     3
804ef12b cc              int     3
nt!IoCheckQuerySetFileInformation:
804ef12c 8bff            mov     edi,edi

 

我们知道,在调用IoCallDriver 的时候,它会先调用 IofCallDriver,所以,我们必须替换 这个 JMP 的地址,也就是

替换成我们的函数地址。。。

 

替换的过程:

 ULONG  uAddress;
 KIRQL  Irql;

 uAddress = (ULONG)IofCallDriver;  // 得到原函数地址

 __asm
 {
  mov eax, uAddress;
  mov esi, [eax+2];                           // 取得 FF 后边的地址
  mov eax, esi;                                  // 保存原来函数的地址(也就是 FF 后边的 4 字节:也就是IoCallDriver 的地址 )
  mov OldIofCallDriver, eax;             //  用我们定义好的指向函数的指针 指向 IoCallDriver 这个函数。。。因为我们处理完后,

                                                         // 可以把控制权还给系统
 }                               

                        

 Irql = KeRaiseIrqlToDpcLevel();       // 提高中断等级,防止被抢占。。。    
 ClearMemoryProtect();                    // 去除内存的写保护,我自己把它们封装了
 __asm
 {
  mov eax, uAddress;                                         // 再次得到函数地址
  mov esi, [eax+2];                                            // 同上
  mov dword ptr [esi], offset MyIofCallDriver;  // 将原来函数的地址覆盖成我们定义的 HOOK 函数,

                                                                        // 然后,当请求到来的时候,程序会首先执行我们,

                                                                        // 然后,在我们的函数中,我们通过前边得到的函数指针,调用原来的函数...
 }
 RecoverMemoryProtect();
 KeLowerIrql(Irql);

卸载呢?

也就是,我们将 jmp 后 的4 个字节 替换成原来的函数地址,一切就是这样,从这里可以看出,指针的威力是很强大的。。。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 C++ 中,虚函数通过虚函数(vtable)来实现。虚函数是一个存储了虚函数指针的数组,每个类的对象都有一个指向其对应虚函数的指针。 要 hook C++ 虚函数并替换其中的虚函数指针,你可以按照以下步骤进行: 1. 获取要 hook 的类的对象指针。 2. 访问对象的虚函数指针。 3. 根据虚函数的索引找到要 hook 的虚函数指针。 4. 替换虚函数指针为你想要调用的函数指针。 以下是一个示例代码,展示了如何 hook C++ 虚函数并替换其中的虚函数指针: ```cpp #include <iostream> // 假设有一个基类 BaseClass class BaseClass { public: virtual void virtualFunction() { std::cout << "BaseClass::virtualFunction()" << std::endl; } }; // 定义一个替代原始虚函数函数 void replacementFunction() { std::cout << "Replacement function called" << std::endl; } int main() { // 创建 BaseClass 对象 BaseClass obj; // 获取对象的虚函数指针 uintptr_t* vtable = *(uintptr_t**)&obj; // 获取要 hook 的虚函数的索引(假设是第一个虚函数) int virtualFunctionIndex = 0; // 替换虚函数指针为 replacementFunction uintptr_t replacementFunctionPtr = (uintptr_t)&replacementFunction; vtable[virtualFunctionIndex] = replacementFunctionPtr; // 调用虚函数,将会调用替代函数 obj.virtualFunction(); return 0; } ``` 请注意,这个方法可能涉及到一些平台相关的细节,因此在不同的编译器和操作系统上可能会有所不同。在实际应用中,确保了解你所使用的编译器和平台的特定要求和限制是非常重要的。此外,修改虚函数可能会破坏程序的稳定性和预期行为,谨慎使用。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值