首页
社区
课程
招聘
[原创]《游戏修改器DIY》
发表于: 2005-2-6 22:43 17366

[原创]《游戏修改器DIY》

2005-2-6 22:43
17366

《游戏修改器DIY》

《潜艇大战》v1.0 修改器及源码下载(内含游戏)              

   近日在整理电脑光盘时,在其中发现了一款名叫《潜艇大战》的小游戏,做得挺有趣的,可怎么也打不通关……郁闷中!于是找来《金山游侠》开始作弊,嘿嘿……结果嘛――当然是爆机喽!

   每次玩都用《游侠》修改岂不是很不爽?!那就让我们一起来做个游戏修改器吧!

   我们的目标是――
   《潜艇大战》(版本:1.0)
   所需工具――
   《金山游侠III》(玩家们都应该知道吧?版本:III)、
   SOFTICE(一款强大的程序调试工具,版本:4.0.5 Build334+IceDump)、
   Microsoft Visual C++ (编程利器!版本:6.0+SP5)

   预备知识――
   游戏修改器的原理很简单,就是修改内存中数据。那么就要用到下面这个API函数:WriteProcessMemory,其函数原型为:
BOOL WriteProcessMemory(

HANDLE hProcess, // handle to process whose memory is written to
LPVOID lpBaseAddress, // address to start writing to
LPVOID lpBuffer, // pointer to buffer to write data to
DWORD nSize, // number of bytes to write
LPDWORD lpNumberOfBytesWritten // actual number of bytes written
);
有了这个API函数我们就可以修改内存中的数据了,让我们一步一步来,把这个游戏做成一个不死版的游戏――

   第一部分:获得游戏中关键的数据――这是制作游戏修改器的前提。

   首先运行《潜艇大战》,在游戏开始后你只有5次机会,《金山游侠》的使用方法我就不在赘述了,很简单,相信众玩家都会使用。通过一系列的操作并测试后,发现用于存放生命的内存地址在:“0x007B0BE8”。OK!让我们调出SOFTICE,输入下面的命令(其作用是:在对内存“0x007B0BE8”处有写操作时中断程序):
   bpm 007B0BE8 w
回车返回游戏后故意使潜艇中弹使生命数减1,这时SOFTICE会自动跳出并停留在:
   0167:00405686 3BCD CMP ECX,EBP(图一)

       

往上看,大家看到了什么?对,是
   0167:0040567D 4A DEC EDX
而EDX存放的就是潜艇中弹之前的生命数,DEC是什么我就不用解释了吧(减一)?到了这里想必大家也就弄清楚了制作不死版游戏的关键,那就是把“0167:0040567D”处的代码改成什么都不做――
   0167:0040567D 90 NOP
也就是――即使潜艇中弹了也不会减少生命数!

   第二部分:用VC制作游戏修改器。

   我们要制作的这个游戏修改器很简单,不需要MFC,SDK即可完全搞定!

   启动VC,依次――[File]-[New]-[Projects]-[Win32 Application]-[输入-Project name]后[OK],选择[An empty project]-[OK]。然后[File]-[New]-[Files]-[选择-C++Source File],起名后[OK]。输入以下代码――

//////////////////////////////////////////////////////////////////////////////////////////

/*
--《潜艇大战》v1.0 修改器--
作者:赵春生
制作时间:17:18 04-1-15
主页:HTTP://TIMW.YEAH.NET
*/

#include<windows.h>

int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)

{

//OBJIP
unsigned int OBJ_ip=0x0040567D;//修改潜艇生命数代码的偏移地址
unsigned int OBJ_patch[1]={0x89CA8B90};//其中的90则是NOP的十六进制

HWND hwndOBJ=FindWindow(NULL,"潜艇大战");//获得游戏的窗口句柄
DWORD PID;
HANDLE hProcess;

if (hwndOBJ!=0)
{
SetForegroundWindow(hwndOBJ);//找到句柄后将游戏设置为当前窗口
GetWindowThreadProcessId(hwndOBJ,&PID);//得到游戏的进程ID
hProcess=OpenProcess(PROCESS_ALL_ACCESS,false,PID);//打开游戏进程
WriteProcessMemory(hProcess,(void *)OBJ_ip,OBJ_patch,1,0);//写入修改后的代码

CloseHandle(hwndOBJ);
CloseHandle(hProcess);
}
else
MessageBox(NULL,"游戏还没有启动吧?","错误!",NULL);

return 0;
}

//////////////////////////////////////////////////////////////////////////////////////////

   编译后运行程序,看看效果吧?!怎么样?游戏成为不死版了吧?

   在这里我只是给大家介绍了制作游戏修改器的一种思路,本程序还可以做得更加完美,例如:直接使用游戏修改器来加载游戏进行修改,而不是像现在这样:要先运行游戏后才能使用游戏修改器;再有,游戏还有一处修改潜艇生命数的地方,这处在:“0167:00404BEE”,大家有兴趣的话可以自己尝试……欢迎大家和我一起进行交流!本游戏的修改器及其源码可从我的主页下载。

17:18 04-1-15


[招生]科锐逆向工程师培训(2024年11月15日实地,远程教学同时开班, 第51期)

收藏
免费 7
支持
分享
最新回复 (14)
雪    币: 431
活跃值: (442)
能力值: ( LV12,RANK:530 )
在线值:
发帖
回帖
粉丝
2
《游戏修改器DIY之二》的前一篇。写得太烂,开始没敢放出来,为了完整,还是斗胆……
2005-2-6 22:51
0
雪    币: 200
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
3
8错8错^_^
2005-2-6 23:23
0
雪    币: 202
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
4
支持一吓楼主!~`
2005-2-6 23:27
0
雪    币: 214
活跃值: (100)
能力值: ( LV4,RANK:50 )
在线值:
发帖
回帖
粉丝
5
支持,正想学学游戏修改
2005-2-7 08:42
0
雪    币: 12342
活跃值: (4055)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
6
收下学习
谢谢!
2005-2-7 09:13
0
雪    币: 202
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
7
unsigned int OBJ_patch[1]={0x89CA8B90};//其中的90则是NOP的十六进制

这句老是想不通``为什么不写:
unsigned int OBJ_patch[1]={0x90};呢?指教``!
2005-2-7 09:20
0
雪    币: 431
活跃值: (442)
能力值: ( LV12,RANK:530 )
在线值:
发帖
回帖
粉丝
8
我是为了让大家弄清指令代码的赋值顺序。
2005-2-7 16:23
0
雪    币: 202
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
9
如此啊``:)
2005-2-7 19:25
0
雪    币: 202
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
10
嘿``还有一个地方不明白``
这句:
long OBJ_patch[1] = {0x7D5};
flag = WriteProcessMemory(HProcess,(LPVOID)OBJ_ip,OBJ_patch,1,0);
像上面那样写就行``但换一吓`
long OBJ_patch1 = 0x7D5;  //这里
flag = WriteProcessMemory(HProcess,(LPVOID)OBJ_ip,(LPVOID)OBJ_patch,1,0);
像上面那样写就不行```不明白为什么写成数组就不用把缓冲区转类型?
2005-2-9 10:36
0
雪    币: 431
活跃值: (442)
能力值: ( LV12,RANK:530 )
在线值:
发帖
回帖
粉丝
11
BOOL WriteProcessMemory(

    HANDLE hProcess,        // handle to process whose memory is written to  
    LPVOID lpBaseAddress,        // address to start writing to
    LPVOID lpBuffer,        // pointer to buffer to write data to
    DWORD nSize,        // number of bytes to write
    LPDWORD lpNumberOfBytesWritten         // actual number of bytes written
   );

lpBuffer要的是地址存放(0x7D5)的地址。

long OBJ_patch[1] = {0x7D5};
flag = WriteProcessMemory(HProcess,(LPVOID)OBJ_ip,OBJ_patch,1,0);
OBJ_patch就是0x7D5(OBJ_patch[0])的地址。

long OBJ_patch1 = 0x7D5;  //这里
flag = WriteProcessMemory(HProcess,(LPVOID)OBJ_ip,(LPVOID)OBJ_patch1,1,0);
OBJ_patch1是0x7D5,这样写编译应该无错,但程序就错了,她写的内容不是D5,而是内存0x7D5位置上的一个字节。
2005-2-9 13:55
0
雪    币: 205
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
12
C++如何搜锁一串机械码呢?thanks.
2005-2-9 15:03
0
雪    币: 431
活跃值: (442)
能力值: ( LV12,RANK:530 )
在线值:
发帖
回帖
粉丝
13
如何编写游戏修改器 第二版(SE)昆明 李维

前言
       本文乃作者多年研究游戏修改器经验的结晶,通过浅显易懂的语言,并晓之以原理,示之以代码,将为您揭开游戏修改器的技术奥秘。本文包含了当今Win9x/Me/2000/xp平台下游戏修改器的大部分流行甚至可以说“先进”的技术,相信您通过本文的指引,定能写出一款属于自己的游戏修改器。 (SE: 本文所述的技术适用于 Windows 95/97/98/Me/2000/Xp)

       在正式开始前,我将假定您已经具备了如下的阅读条件:

1.        具备在32位Windows 下使用 C/C++ 编程的基本知识(SE:最好是VC),大致了解 Windows的结构及基本原理。

2.        可供查询的Windows编程资料,如MSDN和著名的《Windows 核心编程》等。

3.        使用过游戏修改器,开个玩笑:)

目录
如何编写游戏修改器... 1

前言... 1

目录... 1

游戏修改器的基本工作原理... 1

如何访问游戏程序的内存... 2

如何实现热键... 3

如何实现暂停游戏... 3

如何在游戏中弹出自己的界面... 3

如何调整游戏速度... 4

抓图功能... 4

  

游戏修改器的基本工作原理
所谓游戏修改器,主要是通过修改游戏程序的内存数据或存盘文件来修改游戏中的相关数据,使之达到“无敌”等效果。

游戏修改器主要分为两类:单一游戏的修改器和通用游戏修改器。顾名思义,前者只能修改特定的游戏,此类修改器也叫“无敌引导程序”或“游戏作弊器”;而后者则能够以“不变应万变”,可以修改大多数的游戏。本文主要讨论后者,相比之下,前者是后者只留下修改功能的“简化版”。

总的来说,游戏修改器主要的功能便是反复搜索并筛选某一特定内存地址并将其按照固定的周期修改为特定的值(所谓的“锁定”),当然,将“内存”换为“文件”便可以以同样的方式搜索文件了。

搜索和筛选是如何实现的呢?假如我们已经可以访问游戏的内存,我们将游戏的内存分块读入一个缓冲区,然后借由下面的函数搜索:

  

//在一个内存块中搜索内容

//成功返回位置

//失败返回-1

//(SE:现在我发现这个顺序查找函数完全可以用 STL 的 std::find() 函数取代,原因是写文章的时候我还不太了解 STL:)

int MemFind(int iStartPosition, LPBYTE pDestBuffer, int iDestBufferLength, LPBYTE pPatternBuffer, int iPatternBufferLength)

{

   

    signed int iFoundPosition, i;

   

    iFoundPosition = -1;

    if(iStartPosition > iDestBufferLength)return -1;

   

    for(i = iStartPosition; i < (iDestBufferLength + 1); i++)

    {

        if(memcmp(&pDestBuffer[i], pPatternBuffer, iPatternBufferLength) == 0)

        {

            iFoundPosition = i;

            break;

        }

    }

    return iFoundPosition;

}

  

由于游戏在内存或文件中保存的数据是二进制格式,因此,当我们搜索一个为123的整数时应用如下的格式:

int nPattern = 123;

dwOffset = MemFind(nStartPosition, pDestBuffer, nDestBufferLength, (LPBYTE)(&nPattern), sizeof(nPattern))

同样,可以这样搜索浮点类型:

float rPattern = 123;

dwOffset = MemFind(nStartPosition, pDestBuffer, nDestBufferLength, (LPBYTE)(&rPattern), sizeof(rPattern))

  

由于内存中肯定会存在许多相同的数据,所以第一次我们肯定会搜索到许多地址,而真正我们要找的地址一定包含在其中。所以,我们建立一个临时文件将这些地址保存起来,并设置“有效”标志,如下代码所示:

struct CTempItem

{

    DWORD dwAddress;

    BOOL bEnable;

};

CTempItem item;

item.dwAddress = ?????????;

item.bEnable = TRUE;

fwrite(&item, sizeof(item), 1, fp);

接下来,当用户进行第二次搜索时,将这些保存在临时文件中的数据取出来,先看item.bEnable若等于FALSE则跳过,否则读取item.dwAddress所指示的游戏内存,并于用户第二次输入的数值相比较,若发现相同,则设置item.bEnable = TRUE;若不同设置item.bEnable=FALSE,表示废弃。完成之后,再次把item写回文件,当所有item分析完之后,我们就完成了第二次搜索。再接下来,bEnable=TRUE的地址还有很多,则仍然用第二次搜索的方式反复搜索,直到剩下1-2个地址为止。 (SE:应用读写缓冲区,即成批地读写,可以大大加快速度)

由以上介绍可以看出,游戏修改器的搜索分为2个阶段:第一次搜索和第2、3、4……次搜索,游戏修改器在第一次搜索出的很多地址中分析出与用户输入的数据始终相同的地址。当我们有了目标地址,就可以按照用户的意愿定时或手动方式写入用户指定的数据,这便是游戏修改器的基本原理。

当然,这只是基本原理,当具体编写修改器将遇到许多具体的技术困难,以下章节将为您一一解答。

如何访问游戏程序的内存
当我们的修改器运行于Windows时,首先遇到的问题便是如何访问游戏的内存。

首先,在访问游戏的内存前我们还必须获得游戏进程的句柄,这可以通过ToolHelp函数获取系统中当前运行的所有进程的列表和各进程的ID,经由用户选择之后通过OpenProcess函数来获取。若您的修改器运行于后台,而前台是游戏的话,可以在用户按下“弹出”热键时使用GetForegroundWindow函数获取游戏窗口的HWND,再使用GetWindowThreadProcessId转换成游戏进程的ID,再使用OpenProcess函数获取游戏进程的句柄。

有了游戏进程的句柄之后,便可以使用Windows提供的ReadProcessMemory和WriteProcessMemory这两个API来读写游戏的内存了。但是,在Windows9x中每个进程均拥有各自独立的1GB虚拟地址空间,而在Win2000/XP下更是达到了2GB。显然,搜索这样大的地址空间是不现实的,而且游戏也仅仅用了其中的几十到几百MB。所以,我们需要使用VirtualQueryEx这个函数来查询哪些是已经分配的地址,哪些是未用的地址。以下查询与搜索相结合的示范代码:

  

DWORD dwBaseAddress;

SYSTEM_INFO si;

GetSystemInfo(si);

  

dwBaseAddress = si.lpMinimumApplicationAddress;

while(dwBaseAddress < si.lpMaximumApplicationAddress)

{

    mbi.BaseAddress = (LPVOID)dwBaseAddress;

ProcessMem.Query((PVOID)dwBaseAddress, &mbi);

VirtualQueryEx(hProcess, (LPVOID)dwAddress), mbi, sizeof(mbi)

    dwBaseAddress = (DWORD)mbi.BaseAddress + mbi.RegionSize;

    if(mbi.State != MEM_COMMIT || mbi.AllocationProtect != PAGE_READWRITE); //跳过未分配或不可读写的区域

    {

       continue;

    }

      

    //搜索这块内存区域

}

资源:请到 http://alphasun.betajin.com下载我写的一个很“简陋”的游戏修改器――GameProbe的源程序。

如何实现热键
       热键的原理很简单,使用全局键盘HOOK就可以了,鉴于这方面的资料较多,具体的Hook使用方法请参阅MSDN或相关资料。

如何实现暂停游戏
       这是游戏修改器必须具备的一项功能了,若不暂停游戏,搜索之前数值很可能会改变,从而造成找不到数据的现象。暂停游戏的办法有很多种,主要有:

1.        Debug法,这种方法利用DebugActiveProcess这个Windows API将游戏修改器作为游戏的调试程序,游戏修改器可通过Windows提供的调试事件(DebugEvents)来获取游戏的各个线程的句柄。但此法有一个缺点就是不能在关闭游戏前关闭修改器,否则Windows将会自动终止游戏的运行,运用此方法的典型是FPE2000。

2.        通过使用ToolHelp系列函数获取游戏进程的所有线程,并使用公开的OpenThread这个API来获得各线程的句柄并使用SuspendThread 和 ResumeThread来暂停或恢复游戏的运行,目前大部分程序运用此法。(BTW:著名的 GameMaster使用的是CreateProcess并通过Suspend游戏主线程的方法暂停游戏的,很显然,若游戏采用了多线程,此方法是欠妥的。)

但恼人的是Win9x下并没有OpenThread这个API,不过我们可以通过一个未公开的使用 TCB 的方法在Win9x下替代OpenThread而获取线程的句柄,从而达到了暂停游戏运行的目的。您可以到 http://www.windrun.com/ 下载Ligtest前辈编写的PauseProcess程序的源代码来学习,或者下载我的 TestPopup。

如何在游戏中弹出自己的界面
       这个问题可以和热键问题一并解决:众所周知,Windows是一个消息驱动的32位操作系统。在Windows中,所有正在运的进程都有一个独立的2GB的虚拟地址空间,进程之间相互不可见。Windows的绝大多数API与消息是不能跨越进程的。

“Hook”在Windows中主要是用来截取消息的,形象说,就是用来“钩” 消息的。Hook实际上是一个处理消息的程序段,每当特定的消息发出,在没有到达目的窗口前,Hook函数就先捕获该消息,即Hook函数先得到处理消息的控制权。而且如果你把Hook实现在DLL文件中,那么Hook函数将会自动被系统映射到会处理那个特定消息的窗口所在的进程虚拟地址空间中。例如,你可以用Hook来捕获系统中所有的键盘输入消息(WM_KEYDOWN)来实现对电脑使用者的输入进行记录(关于Windows进程管理与Hook 的详细用法,请参阅MSDN与相关资料)。

微软的 DirectX 为Windows下的游戏带来了华丽的声光效果。但是由于DirectX采用直接访问硬件的方法提高多媒体与游戏程序的速度,因此导致了人们误以为在DirectX(确切地说是DirectDraw)下不能显示普通的Windows对话框。

       幸运的是,DirectX 是支持GDI的,也就是说游戏程序可以用常规的方法在DirectX下显示对话框(在微软 DirectX 8 SDK 中有名为“FullScreenDialog”的例子)。所以现在我们的问题变为:如上所述,如何让我们的程序进入游戏的程序的内部并显示对话框。

       这似乎是一个很棘手的问题。但是,有了上面所讲的Hook情况就大为不同了。我们知道既然Hook可以映射到别的进程内部,那么只要将显示对话框的函数以及对话框资源包括在Hook DLL 中不就可以调用DialogBox() 了吗?完全正确!我们用 SetWindowsHookEx() 为系统设置一个键盘消息Hook,系统会自动将这个DLL映射到游戏的进程中。每当有键盘消息,我们只要判断是不是我们所设定的热键,如果是就调用 DialogBox() 显示对话框即可。 (SE: 还有一种方法可以把 DLL 插入别的进程,那就是利用 RemoteThread  远线程,具体程序请参阅上文提到的 PauseProcess 程序。)

       您可以参阅我的TestPopup程序,就演示了在DirectX下弹出界面、暂停游戏、热键等功能。

如何调整游戏速度
       游戏通常是通过timeGetTime、GetTickCount等几个与时间相关的API来控制速度的。因此,我们只要抢在游戏调用这些API之前调用它们,并修改返回值,便可以调整游戏的速度了。具体的程序可以参阅我的 SpeedMan,您可以免费下载它的源程序。SpeedMan 使用了《Windows核心编程》所提供的 CApiHook 类来拦截 timeGetTime 等 Windows API。

抓图功能
       抓图功能很好实现。具体就是先暂停游戏,再通过使用GetForegroundWindow函数获取游戏窗口的HWND,最后DC就可以获得Bitmap格式的屏幕图像了,至于如何保存成文件,或转换为Jpeg(可以使用IJLib)等,则不在本文讨论范围里了。

结束语
       相信您通过阅读本文,已经大致了解了游戏修改器的关键技术了吧?希望本文能助您写出更新、更好的游戏修改器。

       有任何问题欢迎您来信讨论:malloc@km169.net

       若要转载(包括传统媒体),必须提前通知作者,征得作者同意。 此文仅供研究和学习,作者不对文章造成的后果负责。

  

版权所有? 李维 (CSDN 帐号: xenix)

保留所有权力
2005-2-9 16:59
0
雪    币: 202
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
14
谢谢````
2005-2-9 19:18
0
雪    币: 205
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
15
偶个人认为VB(包括.NET)最难做的是搜索部份,速度或长度都不如DELPHI或VC,ASM.
2005-2-9 23:32
0
游客
登录 | 注册 方可回帖
返回
//