hook虚表,快速获取怪物动向,boss技能

有同学问我什么是虚表hook,是不是很难? 其实相较于其他hook,虚表hook是最简单的.

我们先来复习下什么是虚函数.

虚函数是指父类中被声明, 子类中被重新定义的成员函数.

当类有虚函数的时候,会自动产生虚表.

比如这个人物对象下, 前8字节就是虚表地址(32位的话是4字节续表地址), 后面就是对象的属性.

图片

游戏中的类继承关系极为复杂,重写的函数也特别多,所以基本上都是有虚表的.

虚函数被调用的时候是这样的流程

图片

首先rcx是对象

通过mov rax,[rcx] 就得到了虚表地址

虚表地址+偏移里的值 可以得到其中某个虚函数的地址 这里是[rax+C08],对应的是走路,如下图,7FF8CD2AC46D就是走路函数地址

然后直接  call [rax+C08] 即可

图片

如果把这个地址修改了他是不是就可以执行其他的函数了,甚至是我们自己写的函数.

下载xdbg等工具可以公众号 任鸟飞逆向.

那么了解了这个调用过程,和修改的基本思路,我们就可以对他进行hook了.

原理非常简单:

虚函数地址是通过虚表中获得的,

那么我们把虚表中的虚函数地址修改成我们自己的函数地址,

这样虚函数调用的时候就会调用到我们的函数了.

同时,为了不影响他原来的功能,

执行完我们的代码,我们再调用他原来的真正的虚函数即可.

那么可以上代码了:

void Ctextdialog::OnBnClickedButton14()
{
    
    hookVirtualTable(*(uintptr_t*)Base_人物, 0xC08);

}


void Ctextdialog::OnBnClickedButton15()
{
    uhookVirtualTable(*(uintptr_t*)Base_人物, 0xC08);

}
uintptr_t oldVirtualTable = 0;
void hookVirtualTFunc(uintptr_t RCX,uintptr_t RDX,uintptr_t R8,uintptr_t R9,uintptr_t arg5,uintptr_t arg6)
{
    if (R8 == 1)
        printf_rnf("人物跑: %.0f,%.0f,%.0f", *(FLOAT*)RDX, *(FLOAT*)(RDX+4), *(FLOAT*)(RDX+8));
    else
        printf_rnf("人物走: %.0f,%.0f,%.0f", *(FLOAT*)RDX, *(FLOAT*)(RDX + 4), *(FLOAT*)(RDX + 8));

    Call(oldVirtualTable, RCX, RDX, R8, R9, arg5, arg6);//调用原来的虚函数  不影响他原来的代码    
}

void hookVirtualTable(uintptr_t obj, uintptr_t offset)
{
    uintptr_t virtualTable = *(uintptr_t*)obj;// 第一步先获得虚表
    oldVirtualTable = *(uintptr_t*)(virtualTable + offset);//第二步保存原来的虚函数地址
    DWORD old = 0;
    VirtualProtect(PVOID(virtualTable + offset), 0x100, PAGE_EXECUTE_READWRITE, &old);
    *(uintptr_t*)(virtualTable + offset) = (uintptr_t)hookVirtualTFunc;//第三步把虚函数地址写成我们自己的函数地址
    VirtualProtect(PVOID(virtualTable + offset),0x100, old,&old);
}

void uhookVirtualTable(uintptr_t obj, uintptr_t offset)
{
    uintptr_t virtualTable = *(uintptr_t*)obj;//第一步获得虚表
    DWORD old = 0;
    VirtualProtect(PVOID(virtualTable + offset), 0x100, PAGE_EXECUTE_READWRITE, &old);
    *(uintptr_t*)(virtualTable + offset) = oldVirtualTable;//第二步 把原来虚函数地址 写回去
    VirtualProtect(PVOID(virtualTable + offset), 0x100, old, &old);
}

这样就对走路功能进行了时时hook

拓展题目:

我们怎么快速获得 每个怪物的下一步动向?比如BOSS技能?

可以给老师留言你的答案.

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值