天马阁

 找回密码
 立即注册
                                        →→→→→→→→→→→→ 1点击查看所有VIP教程目录长列表(总教程数269个) 2办理VIP详情进入 ←←←←←←←←←←←←
1 x64CE与x64dbg入门基础教程 7课 已完结 2 x64汇编语言基础教程 16课 已完结 3 x64辅助入门基础教程 9课 已完结 4 C++x64内存辅助实战技术教程 149课 已完结
5 C++x64内存检测与过检测技术教程 10课 已完结 6 C+x64二叉树分析遍历与LUA自动登陆教程 19课已完结 7 C++BT功能原理与x64实战教程 29课 已完结 8 C+FPS框透视与自瞄x64实现原理及防护思路 30课完结
64驱?封? 9 64反驱? 10 64位V? 11 绝? 12 ???课?
13 64透 ? 14 64U ? 15 64Q ? 16 64功 ?
17 64U ? 18 64模 ? 19 64多 ? 20 64网 ?
21 64注 ? 22 64火 ? 23 64棋 ? 24 64自二链L?
25 64破 ? VIP会员办理QQ: 89986068   
【请先加好友,然后到好友列表双击联系客服办理,不然可能无法接受到信息。】
27 加入2000人交流群637034024 3 28 免责声明?
查看: 1328|回复: 0

根据特征码搜索基址,VC++ 研究了一天,初步写成。测试...

[复制链接]

13

主题

0

回帖

16

积分

编程入门

Rank: 1

天马币
26
发表于 2024-3-5 09:33:53 | 显示全部楼层 |阅读模式

研究了一天,初步写成。测试了几个没发现bug,如果有问题可以跟贴共同讨论

代码如下:

需要引入的头文件:

#include <stdio.h>
#include <stdlib.h>
#include <windows.h>

union Base   
{
    DWORD   address;
    BYTE    data[4];
};




/************************************************************************/
/* 函数说明:根据特征码扫描基址
/* 参数一:process 要查找的进程
/* 参数二:markCode 特征码字符串,不能有空格
/* 参数三:特征码离基址的距离,默认距离:1
/* 参数四:findMode 扫描方式,找到特征码后,默认为:1
/*                  0:往上找基址(特征码在基址下面)
/*                  1:往下找基址(特征码在基址上面)
/* 参数五:offset 保存基址距离进程的偏移,默认为:不保存
/************************************************************************/
DWORD ScanAddress(HANDLE process, char *markCode,
                  DWORD distinct = 1, DWORD findMode = 1,
                  LPDWORD offset = NULL)
{
    //起始地址
    const DWORD beginAddr = 0x00400000;
    //结束地址
    const DWORD endAddr = 0x7FFFFFFF;
    //每次读取游戏内存数目的大小
    const DWORD pageSize = 4096;

    ////////////////////////处理特征码/////////////////////
    //特征码长度不能为单数
    if (strlen(markCode) % 2 != 0) return 0;
    //特征码长度
    int len = strlen(markCode) / 2;
    //将特征码转换成byte型
    BYTE *m_code = new BYTE[len];
    for (int i = 0; i < len; i++){
        char c[] = {markCode[i*2], markCode[i*2+1], '\0'};
        m_code = (BYTE)::strtol(c, NULL, 16);
    }

    /////////////////////////查找特征码/////////////////////
    BOOL _break = FALSE;
    //用来保存在第几页中的第几个找到的特征码
    int curPage = 0;
    int curIndex = 0;
    Base base;
    //每页读取4096个字节
    BYTE page[pageSize];
    DWORD tmpAddr = beginAddr;
    while (tmpAddr <= endAddr - len){
        ::ReadProcessMemory(process, (LPCVOID)tmpAddr, &page, pageSize, 0);
        //在该页中查找特征码
        for (int i = 0; i < pageSize; i++){
            for (int j = 0; j < len; j++){
                //只要有一个与特征码对应不上则退出循环
                if (m_code[j] != page[i + j])break;
                //找到退出所有循环
                if (j == len - 1){
                    _break = TRUE;
                    if (!findMode){
                        curIndex = i;
                        base.data[0] = page[curIndex-distinct-4];
                        base.data[1] = page[curIndex-distinct-3];
                        base.data[2] = page[curIndex-distinct-2];
                        base.data[3] = page[curIndex-distinct-1];
                    }else{
                        curIndex = i + j;
                        base.data[0] = page[curIndex+distinct+1];
                        base.data[1] = page[curIndex+distinct+2];
                        base.data[2] = page[curIndex+distinct+3];
                        base.data[3] = page[curIndex+distinct+4];
                    }
                    break;
                }
            }
            if (_break) break;
        }
        if (_break) break;
        curPage++;
        tmpAddr += pageSize;
    }
    if(offset != NULL){
        *offset = curPage * pageSize + curIndex + beginAddr;
    }
    return base.address;
}


/************************************************************************/
/* 函数说明:根据特征码扫描call地址
/* 参数一:process 要查找的进程
/* 参数二:markCode 特征码字符串,不能有空格
/* 参数三:特征码离基址的距离,默认距离:1
/* 参数四:findMode 扫描方式,找到特征码后,默认为:1
/*                  0:往上找基址
/*                  1:往下找基址
/************************************************************************/
DWORD ScanCall(HANDLE process, char *markCode,
               DWORD distinct = 1, DWORD findMode = 1)
{
    DWORD offset;
    DWORD call = ScanAddress(process, markCode, distinct, findMode, &offset);
    call += offset;
    if(findMode) call = call + 5 + distinct;
    else call = call - distinct;
    return call;
}


测试代码如下:

int main(int argc, char* argv[])
{
    //查找游戏窗口
    HWND hGame = ::FindWindow("DxFirst", NULL);
    if(hGame == NULL) return FALSE;
   
    DWORD processId;
    HANDLE process;
    ::GetWindowThreadProcessId(hGame, &processId);
    process = ::OpenProcess(PROCESS_ALL_ACCESS, false, processId);
    //83C404C3CCCCA1              1           人物基址往下搜索
    //C3CCCCCCCCCCCCCCCCCCCC8B442404A3ECA72001      0       人物基址往上搜索
    //5557535152C6400801E8            1        打怪call
     
     //基址在特征码下面
     DWORD addr = ScanAddress(process, "83C404C3CCCCA1");
     printf("人物基址:%X\n",addr);
     
     //基址在特征码上面
     DWORD addr = ScanAddress(process, "C3CCCCCCCCCCCCCCCCCCCC8B442404A3ECA72001", 3, 0);
     printf("人物基址:%X\n",addr);

    DWORD call = ScanCall(process, "5557535152C6400801E8");
    printf("call基址:%X\n",call);
    ::CloseHandle(process);
    return 0;
}

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

天马阁|C/C++辅助教程|安卓逆向安全| 论坛导航|免责申明|Archiver||网站地图
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表天马阁立场!
任何人不得以任何方式翻录、盗版或出售本站视频,一经发现我们将追究其相关责任!
我们一直在努力成为最好的编程论坛!
Copyright© 2010-2021 All Right Reserved.
快速回复 返回顶部 返回列表