OllyDBG 入门系列(七)-汇编功能

本文介绍如何使用OllyDBG和LordPE工具修改程序中的字体显示,通过调整CreateWindowExA后的控件字体,实现自定义字体样式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

今天我们的目标程序是MyUninstaller1.34版。这是一个非常小的程序卸载工具,VC6编写,大小只有61K。我拿到的这个是上次闪电狼兄弟给我的,附带在里面的简体中文语言文件是由六芒星制作的。这个程序有个毛病:就是在出的可卸载程序上双击查看属性时,弹出的属性窗口的字体非常难看,应该就是系统字体(SYSTEM_FONT):

http://bbs.pediy.com/upload/2006/4/image/1.gif

我们今天的目标就是利用OllyDBG的汇编功能把上面显示的字体改成我们常见的9号(小五)宋体。首先我们用OllyDBG载入程序,按CTR N组合键查找一下有哪些API函数,只发现一个和设置字体相关的CreateFontIndirectA。现在我们按鼠标右键,选择“在每个参考上设置断点”,关掉名称对话框,F9运行,程序已经运行起来了。我们在程序的表框中随便找一项双击一下,很不幸,那个字体难看的界面又出现了,OllyDBG没有任何动作。可见创建这个窗口的时候根本没调用CreateFontIndirectA,问题现在就变得有点复杂了。先点确定把这个字体难看的对话框关闭,现在我们从另一个方面考虑:既然没有调用设置字体的函数,那我们来看看这个窗口是如何创建的,跟踪窗口创建过程可能会找到一些对我们有用的信息。现在我们再回到我们调试程序的领空,按CTR N看一下,发现CreateWindowExA这个API函数比较可疑。我们在CreateWindowExA函数的每个参考上设上断点,在MyUninstaller的表框中再随便找一项双击一下,被OllyDBG断下:

00408F5E|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/断在这里

上下翻看一下代码:

00408F3B|.50|PUSHEAX;|hInst
00408F3C|.8B45C0|MOVEAX,DWORDPTRSS:[EBP-40];|
00408F3F|.6A00|PUSH0;|hMenu=NULL
00408F41|.03C6|ADDEAX,ESI;|
00408F43|.FF7508|PUSHDWORDPTRSS:[EBP 8];|hParent
00408F46|.FF75D0|PUSHDWORDPTRSS:[EBP-30];|Height
00408F49|.57|PUSHEDI;|Width
00408F4A|.50|PUSHEAX;|Y
00408F4B|.FF75BC|PUSHDWORDPTRSS:[EBP-44];|X
00408F4E|.FF75EC|PUSHDWORDPTRSS:[EBP-14];|Style
00408F51|.6880DE4000|PUSHmyuninst.0040DE80;|WindowName=""
00408F56|.68DCD94000|PUSHmyuninst.0040D9DC;|Class="STATIC"
00408F5B|.FF75D4|PUSHDWORDPTRSS:[EBP-2C];|ExtStyle
00408F5E|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/断在这里
00408F64|6A00|PUSH0;第一处要修改的地方
00408F66|8945F4|MOVDWORDPTRSS:[EBP-C],EAX
00408F69|.E8A098FFFF|CALL<myuninst.sub_40280E>
00408F6E|.50|PUSHEAX;|hInst
00408F6F|.8B45DC|MOVEAX,DWORDPTRSS:[EBP-24];|
00408F72|.6A00|PUSH0;|hMenu=NULL
00408F74|.03F0|ADDESI,EAX;|
00408F76|.FF7508|PUSHDWORDPTRSS:[EBP 8];|hParent
00408F79|.FF75CC|PUSHDWORDPTRSS:[EBP-34];|Height
00408F7C|.53|PUSHEBX;|Width
00408F7D|.56|PUSHESI;|Y
00408F7E|.FF75D8|PUSHDWORDPTRSS:[EBP-28];|X
00408F81|.FF75E8|PUSHDWORDPTRSS:[EBP-18];|Style
00408F84|.6880DE4000|PUSHmyuninst.0040DE80;|WindowName=""
00408F89|.68D4D94000|PUSHmyuninst.0040D9D4;|Class="EDIT"
00408F8E|.FF75B8|PUSHDWORDPTRSS:[EBP-48];|ExtStyle
00408F91|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F97|8945F0|MOVDWORDPTRSS:[EBP-10],EAX;第二处要修改的地方
00408F9A|8B45F8|MOVEAX,DWORDPTRSS:[EBP-8]
00408F9D|.FF30|PUSHDWORDPTRDS:[EAX];/<%s>
00408F9F|.8D85B0FEFFFF|LEAEAX,DWORDPTRSS:[EBP-150];|
00408FA5|.68D0D94000|PUSHmyuninst.0040D9D0;|format="%s:"
00408FAA|.50|PUSHEAX;|s
00408FAB|.FF1590B14000|CALLDWORDPTRDS:[<&MSVCRT.sprintf>];/sprintf
00408FB1|.8B3584B24000|MOVESI,DWORDPTRDS:[<&USER32.SetWindowTextA>];USER32.SetWindowTextA
00408FB7|.83C40C|ADDESP,0C
00408FBA|.8D85B0FEFFFF|LEAEAX,DWORDPTRSS:[EBP-150]
00408FC0|.50|PUSHEAX;/Text
00408FC1|.FF75F4|PUSHDWORDPTRSS:[EBP-C];|hWnd
00408FC4|.FFD6|CALLESI;/SetWindowTextA
00408FC6|.8D85ACFAFFFF|LEAEAX,DWORDPTRSS:[EBP-554]
00408FCC|.50|PUSHEAX;/Arg3
00408FCD|.FF75FC|PUSHDWORDPTRSS:[EBP-4];|Arg2
00408FD0|.FF3500EF4000|PUSHDWORDPTRDS:[40EF00];|Arg1=00BEADCC
00408FD6|.E81884FFFF|CALL<myuninst.sub_4013F3>;/sub_4013F3
00408FDB|.83C40C|ADDESP,0C
00408FDE|.50|PUSHEAX
00408FDF|.FF75F0|PUSHDWORDPTRSS:[EBP-10]
00408FE2|.FFD6|CALLESI
00408FE4|.FF45FC|INCDWORDPTRSS:[EBP-4]
00408FE7|.8345F814|ADDDWORDPTRSS:[EBP-8],14
00408FEB|.837DFC0F|CMPDWORDPTRSS:[EBP-4],0F
00408FEF|.^0F8C32FFFFFF/JL<myuninst.loc_408F27>
00408FF5|.5FPOPEDI
00408FF6|.5EPOPESI
00408FF7|.5BPOPEBX
00408FF8|.C9LEAVE
00408FF9/.C3RETN
 

我想上面的代码我不需多做解释,OllyDBG自动给出的注释已经够清楚的了。我们双击MyUninstaller表框中的的某项查看属性时,弹出的属性窗口上的STATIC控件和EDIT控件都是由CreateWindowExA函数创建的,然后再调用SetWindowTextA来设置文本,根本没考虑控件上字体显示的问题,所以我们看到的都是系统默认的字体。我们要设置控件上的字体,可以考虑在CreateWindowExA创建完控件后,在使用SetWindowTextA函数设置文本之前调用相关字体创建函数来选择字体,再调用SendMessageA函数发送WM_SETFONT消息来设置控件字体。思路定下来后,我们就开始来实施。首先我们看一下这个程序中的导入函数,CreateFontIndirectA这个字体创建函数已经有了,再看看SendMessageA,呵呵,不错,原程序也有这个函数。这样我们就省事了。有人可能要问,如果原来并没有这两个导入函数,那怎么办呢?其实这也很简单,我们可以直接用LordPE来在程序中添加我们需要的导入函数。我这里用个很小的PE工具zeroadd来示范一下,这个程序里面没有CreateFontIndirectA和SendMessageA函数(这里还有个问题说一下,其实我们编程时调用这两个函数时都是直接写CreateFontIndirect及SendMessage,一般不需指定。但在程序中写补丁代码时我们要指定这是什么类型的函数。这里在函数后面加个“A”表示这是ASCII版本,同样UNICODE版本在后面加个“W”,如SendMessageW。在Win9X下我们一般都用ASCII版本的函数,UNICODE版本的函数很多在Win9X下是不能运行的。而NT系统如WinXP一般都是UNICODE版本的,但如果我们用了ASCII版本的函数,系统会自动转换调用UNICODE版本。这样我们写补丁代码的时候就可以直接指定为ASCII版本的函数,可以兼容各个系统):我们用LordPE的PE编辑器载入zeroadd程序,选择“目录”,再在弹出的目录表对话框中选择输入表后面的那个“...”按钮,会弹出一个对话框:



因为SendMessageA在USER32.dll中,我们在右键菜单中点击按钮“添加导入表”,来到下面:

http://bbs.pediy.com/upload/2006/2/image/3.gif

按上面的提示完成后点“确定”,我们回到原先的那个“输入表”对话框:



从上图中我们可以看出多出了一个USER32.dll,这就是我们添加SendMessageA的结果。这也是用工具添加的一个缺点。我们一般希望把添加的函数直接放到已存在的DLL中,而不是多出来一个,这样显得不好看。但用工具就没办法,LordPE默认是建一个1K的新区段来保存添加后的结果,由此出现了上图中的情况。如果你对PE结构比较熟悉的话,也可以直接用16进制编辑工具来添加你需要的函数,这样改出来的东西好看。如果想偷懒,就像我一样用工具吧,呵呵。在上图中我还标出了要注意FirstThunk及那个ThunkRVA的值,并且要把“总是查看FirstThunk”那个选项选上。有人可能不理解其作用,我这里也解释一下:一般讲述PE格式的文章中对FirstThunk的解释是这样的:FirstThunk包含指向一个IMAGE_THUNK_DATA结构数组的RVA偏移地址,当把PE文件装载到内存中时,PE装载器将查找IMAGE_THUNK_DATA和IMAGE_IMPORT_BY_NAME这些结构数组来决定导入函数的地址,随后用导入函数真实地址来替代由FirstThunk指向的IMAGE_THUNK_DATA数组里的元素值。这样说起来还是让人不明白,我举个例子:比如你有个很要好的朋友,他是个大忙人,虽然你知道他的家庭住址,可他很少回家。如果你哪天想找他,直接去他家,很可能吃个闭门羹,找不到他人。怎么办?幸好你有他的手机号码,你就给他拨了一个电话:“小子,你在哪呢?”,他告诉你:“我正在XXX饭店喝酒呢!”这时你怎么办?(当然是杀到他说的那家饭店去蹭饭了!^_^)这里的ThunkRVA就相当于你朋友的手机号码,SendMessageA就相当于你那个朋友。而FirstThunk就是你手机里的号码分组。你把你的多个朋友都放在FirstThunk这样的号码分组里,每个ThunkRVA就是你一个朋友的手机号码。你要找他们,就是通过ThunkRVA这样的手机号码来和他们联系,直接去他家找他你很可能要碰壁。而移动或联通就相当于操作系统,他们负责把你的手机号码和你的朋友对应上。而FirstThunk这样的号码分组还有一个好处就是你可以不记你某个朋友的具体号码,只要记得FirstThunk号码分组的值,你的朋友会按顺序在里面排。比如上图中USER32.dll中的第一个函数是SendMessageA,它的ThunkRVA值就是FirstThunk值。如果还有第二个函数,比如是MessageBoxA,它的值就是FirstThunk值加上4,其余类推。你只要记住各个函数的位置,也可以通过FirstThunk加上位置对应值来找到它。当然这比不上直接看ThunkRVA来得方便。说了上面这些,我们就要考虑怎么在程序中调用了。你可能会说,我在OllyDBG中直接在我们要修改的程序中这样调用:CALLSendMessageA。哦,别这样。这等于我上面说的都是废话,会让我感到伤心的。你这里的CALLSendMessageA就相当于也不跟你朋友打个招呼就直接去他家找他,很有可能你会乘兴而去,败兴而归。别忘了他的手机号码,我们只有通过号码才知道他到底在什么地方。我们应该这样:CALLDWORDPTR[40B01A],这里的40B01A就是上面的SendMessageA在程序载入后的所在的地方,由基址00400000加上ThunkRVA0000B01A得到的。这就是你要找的人所在的地方,不管他跑到哪,你有他的手机号码就能找到他。同样道理,你只要记住了ThunkRVA值,就按这个来调用你需要的函数,在别的Windows系统下也是没有问题的。系统会自动把你要找到函数和ThunkRVA值对应上。而你在OllyDBG中写CALLSendMessageA,可能你在你的系统上成功了,可放到别的系统下就要出错了。为什么?因为你找的人已经不在原来的位置了,他跑到别的地方去了。你还到老地方找他,当然看不见人了。说了这么多废话,也不知大家听明白了没有,别越听越糊涂就行了。总之一句话,别像CALLSendMessageA这样直接调用某个函数,而应该通过ThunkRVA值来调用它。下面我们回到我们要修改的MyUninstaller上来,先用LordPE打开看一下,呵呵,原来CreateFontIndirectA和SendMessageA原程序里面都有了,省了我们不少事情。看一下这两个函数的ThunkRVA值,CreateFontIndirectA在GDI32.dll里面,ThunkRVA值是0000B044,这样我们就知道在程序中调用它的时候就是CALLDWORDPTR[0040B044]。同样,SendMessageA的ThunkRVA值是0000B23C,调用时应该是这样:CALLDWORDPTR[0040B23C]。了解了这些东西我们就来考虑怎么写代码了。首先我们来看一下CreateFontIndirectA和SendMessageA这两个函数的定义:

CreateFontIndirectA:

HFONTCreateFontIndirect(
CONSTLOGFONT*lplf//pointertologicalfontstructure
);
CreateFontIndirect的返回值就是字体的句柄。

对于这个函数我们需要的参数就是给它一个LOGFONT的字体结构指针,我们只要在要修改程序的空白处建一个标准的9号(小五)宋体的LOGFONT字体结构,再把指针给CreateFontIndirectA就可以了。

SendMessageA:

LRESULTSendMessage(
HWNDhWnd,//handleofdestinationwindow
UINTMsg,//messagetosend
WPARAMwParam,//firstmessageparameter
LPARAMlParam//secondmessageparameter
);
上面的第一个参数是窗口句柄,我们知道CreateWindowExA返回的就是窗口句柄,我们可以直接拿来用。第二个消息参数我们这里是设置字体,选WM_SETFONT,这个值是30H。第三个参数是字体句柄,可以由上面的CreateFontIndirectA获得。第四个参数我们不需要,留空。现在我们准备开始写代码,首先我们要在程序中建一个标准9号宋体的LOGFONT,以便于我们调用。对于LOGFONT,我们再来看一下定义:

typedefstructtagLOGFONT{//lf
LONGlfHeight;
LONGlfWidth;
LONGlfEscapement;
LONGlfOrientation;
LONGlfWeight;
BYTElfItalic;
BYTElfUnderline;
BYTElfStrikeOut;
BYTElfCharSet;
BYTElfOutPrecision;
BYTElfClipPrecision;
BYTElfQuality;
BYTElfPitchAndFamily;
TCHARlfFaceName[LF_FACESIZE];
}LOGFONT;

这样我们的标准9号宋体的LOGFONT值应该是32字节,16进制就像这样:F4FFFFFF000000000000000000000000900100000000008600000000CBCECCE5。现在在程序中找个空地。我们用PEiD来帮助我们寻找,用PEiD打开程序,点EP段后面的那个>号,随便选择一个区段右击,选“搜索全0处”(原版好像是cave什么的):

http://bbs.pediy.com/upload/2006/2/image/5.gif

我们看到PEiD把搜索到的空间都给我们出来了:

http://bbs.pediy.com/upload/2006/2/image/6.gif

现在我们用WinHEX打开我们要修改的程序,转到偏移9815处,从9815处选择32字节(16进制是0X20)的一个选块,把光标定位到9815处,右键选择菜单剪贴板数据->写入(从当前位置覆写),随后的格式选择ASCIIHex,把我们LOGFONT的16进制值

F4FFFFFF000000000000000000000000900100000000008600000000CBCECCE5

写入保存。现在我们用OllyDBG载入已添加了LOGFONT数据的程序,先转到VA40A415处(从上图中看到的)往下看一下:

http://bbs.pediy.com/upload/2006/2/image/7.gif

因为SendMessageA还要用到一个窗口句柄,我们可以通过前面的CreateWindowExA来获得。现在我们就把前一张图中的.rdata区段中的地址0040C56E作为我们保存窗口句柄HWND值的临时空间。一切就绪,开始写代码。先回顾一下我们最先说的那两个要修改的地方:

第一个要改的地方:

00408F5E|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F646A00PUSH0;修改前
00408F668945F4MOVDWORDPTRSS:[EBP-C],EAX
00408F69|.E8A098FFFF|CALL<myuninst.sub_40280E>

修改后:

00408F5E|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F64E9D5140000JMPmyuninst.0040A43E;跳转到我们的补丁代码处
00408F69|.E8A098FFFF|CALL<myuninst.sub_40280E>

第二个要改的地方:

00408F91|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F978945F0MOVDWORDPTRSS:[EBP-10],EAX;改这里
00408F9A8B45F8MOVEAX,DWORDPTRSS:[EBP-8]
00408F9D|.FF30|PUSHDWORDPTRDS:[EAX];/<%s>
00408F9F|.8D85B0FEFFFF|LEAEAX,DWORDPTRSS:[EBP-150];|
00408FA5|.68D0D94000|PUSHmyuninst.0040D9D0;|format="%s:"
00408FAA|.50|PUSHEAX;|s
00408FAB|.FF1590B14000|CALLDWORDPTRDS:[<&MSVCRT.sprintf>];/sprintf
00408FB1|.8B3584B24000|MOVESI,DWORDPTRDS:[<&USER32.SetWindowTextA>];USER32.SetWindowTextA

修改后:

00408F91|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F97E9D4140000JMPmyuninst.0040A470;跳到我们的第二部分补丁代码处
00408F9C90NOP
00408F9D|.FF30|PUSHDWORDPTRDS:[EAX];/<%s>
00408F9F|.8D85B0FEFFFF|LEAEAX,DWORDPTRSS:[EBP-150];|
00408FA5|.68D0D94000|PUSHmyuninst.0040D9D0;|format="%s:"
00408FAA|.50|PUSHEAX;|s
00408FAB|.FF1590B14000|CALLDWORDPTRDS:[<&MSVCRT.sprintf>];/sprintf
00408FB1|.8B3584B24000|MOVESI,DWORDPTRDS:[<&USER32.SetWindowTextA>];USER32.SetWindowTextA

这两个地方的修改都是把原代码改成跳转,跳到我们的补丁代码那继续执行。在修改之前先把原代码复制下来,以便恢复。我们在OllyDBG中按CTR G组合键,来到0040A43E地址处,开始输补丁代码:



同样,我们也在0040A470地址处输入我们另一部分的补丁代码。两部分的补丁代码分别如下:

补丁代码1:

0040A43E60PUSHAD;保护现场
0040A43FA36EC54000MOVDWORDPTRDS:[40C56E],EAX;保存窗口句柄
0040A4446815A44000PUSHmyuninst.0040A415;传递字体句柄LOGFONT
0040A449FF1544B04000CALLDWORDPTRDS:[<&GDI32.CreateFontIndirectA>];GDI32.CreateFontIndirectA
0040A44F6A00PUSH0;lParam参数留空
0040A45150PUSHEAX;字体句柄LOGFONT
0040A4526A30PUSH30;WM_SETFONT
0040A4548B0D6EC54000MOVECX,DWORDPTRDS:[40C56E];窗口句柄送ECX
0040A45A51PUSHECX;压入窗口句柄参数
0040A45BFF153CB24000CALLDWORDPTRDS:[<&USER32.SendMessageA>];USER32.SendMessageA
0040A46161POPAD;恢复现场
0040A4626A00PUSH0;恢复原代码
0040A4648945F4MOVDWORDPTRSS:[EBP-C],EAX
0040A467^E9FDEAFFFFJMPmyuninst.00408F69;返回

补丁代码2:

0040A470>/60PUSHAD
0040A471.A36EC54000MOVDWORDPTRDS:[40C56E],EAX
0040A476.6815A44000PUSHmyuninst.0040A415;/pLogfont=myuninst.0040A415
0040A47B.FF1544B04000CALLDWORDPTRDS:[<&GDI32.CreateFontIndirectA>];/CreateFontIndirectA
0040A481.6A00PUSH0;/lParam=0
0040A483.50PUSHEAX;|wParam
0040A484.6A30PUSH30;|Message=WM_SETFONT
0040A486.8B0D6EC54000MOVECX,DWORDPTRDS:[40C56E];|
0040A48C.51PUSHECX;|hWnd=>NULL
0040A48D.FF153CB24000CALLDWORDPTRDS:[<&USER32.SendMessageA>];/SendMessageA
0040A493.61POPAD
0040A494.8945F0MOVDWORDPTRSS:[EBP-10],EAX
0040A497.8B45F8MOVEAX,DWORDPTRSS:[EBP-8]
0040A49A.^E9FEEAFFFFJMPmyuninst.00408F9D

补丁代码2因为与补丁代码1类似,我就不做详细解释了。现在我们的代码都写完了,现在我们开始保存我们的工作,选中我们修改的代码,点击鼠标右键,会出来一个菜单:



我们左键选所有修改(当然选它了,要不然只会保存我们选定的这一部分。关于这个地方还要说一下,有的时候我们修改完程序选“复制到可执行文件”时只有“选择”菜单,没有“所有修改”菜单项。按OllyDBG帮助里关于备份功能的说法,好像是受内存块限制的,补丁功能也同样是这样。对于备份及补丁功能我用的比较少,并不是很了解,这方面的内容还是大家自己去研究吧,有什么好的心得也希望能共享一下。我遇到不能保存所有修改的情况就是先把补丁代码全部复制下来,同时利用二进制功能复制代码,先选一段补丁代码保存为文件,再用OllyDBG打开保存后的文件,转到相应位置分别把我们复制下来的补丁二进制代码粘贴上去后保存。纯属笨办法,当然你也可以用HexView这样的工具来修改代码),随后会出来一个“把选中的内容复制到可执行文件”的对话框,我们选“全部复制”,又出来一个对话框,我们在上面点右键,在弹出的菜单上选“保存文件”:

http://bbs.pediy.com/upload/2006/2/image/10.gif

这时会出来一个另存文件的对话框,我们另选一个名字如myuninst1.exe来保存,不要直接覆盖原文件myuninst.exe,以便于出错后好修改。现在关闭OllyDBG,先不要急着运行刚刚修改过的文件,因为我们还有个地方要改一下。大家还记得我们在.rdata中用了个地方作为我们保存临时变量的地方吧?原先的.rdata段属性设置是不可写的,现在我们写入了数据,运行时是会出错的。现在我们要修改一下.rdata段的属性。用LordPE的PE编辑器打开我们修改后的程序,点“区段”按钮,在弹出的对话框中点击.rdata段,右键选择弹出菜单中的“编辑区段”:

http://bbs.pediy.com/upload/2006/2/image/11.gif

在弹出的对话框中选标志后面那个“...”按钮:

http://bbs.pediy.com/upload/2006/2/image/12.gif

现在我们把区段标志添加一个可写入的属性:

http://bbs.pediy.com/upload/2006/2/image/13.gif

完成后按确定保存我们所做的工作,运行一下修改后的程序,呵呵,终于把字体改过来了:

http://bbs.pediy.com/upload/2006/2/image/14.gif

如果你运行出错也没关系,用OllyDBG调试一下你修改后的程序,看看错在什么地方。这一般都是输入补丁代码时造成的,你只要看一下你补丁代码运行的情况就可以了。到这里我们的任务似乎也完成了,但细心的朋友可能会发现补丁代码1和补丁代码2前面的代码基本上是相同的。一个两个这样的补丁还好,如果要是多的话,这样重复就要浪费不少空间了,况且工作量也相应加大了。既然前面有很多代码都是重复的,为什么我们不把这些重复的代码做成一个子程序呢?这样调用起来要方便的多。下面我们把前面的补丁代码修改一下,我们先把补丁代码1的代码改成这样:

0040A43E60PUSHAD;保护现场
0040A43FA36EC54000MOVDWORDPTRDS:[40C56E],EAX;保存窗口句柄
0040A4446815A44000PUSHmyuninst.0040A415;我们建的LOGFONT对应指针
0040A449FF1544B04000CALLDWORDPTRDS:[<&GDI32.CreateFontIndirectA>];GDI32.CreateFontIndirectA
0040A44F6A00PUSH0;lParam参数留空
0040A45150PUSHEAX;字体句柄
0040A4526A30PUSH30;WM_SETFONT
0040A4548B0D6EC54000MOVECX,DWORDPTRDS:[40C56E];窗口句柄
0040A45A51PUSHECX;窗口句柄压栈
0040A45BFF153CB24000CALLDWORDPTRDS:[<&USER32.SendMessageA>];USER32.SendMessageA
0040A46161POPAD;恢复现场
0040A462C3RETN;返回

这样我们的子程序代码就写好了。现在我们再在子程序代码后面写上两个补丁代码,当然不要忘了改前面原程序中的跳转:

修改后的补丁代码1:

0040A467E8D2FFFFFFCALLmyuninst.0040A43E;调用子程序
0040A46C6A00PUSH0;恢复前面修改过的代码
0040A46E8945F4MOVDWORDPTRSS:[EBP-C],EAX
0040A471^E9F3EAFFFFJMPmyuninst.00408F69;返回继续执行

修改后的补丁代码2:

0040A47AE8BFFFFFFFCALLmyuninst.0040A43E
0040A47F8945F0MOVDWORDPTRSS:[EBP-10],EAX
0040A4828B45F8MOVEAX,DWORDPTRSS:[EBP-8]
0040A485^E913EBFFFFJMPmyuninst.00408F9D

我在每个补丁代码片断间留了4个字节来分隔。同样,我们还要修改一下我们前面的跳转:

第一个要修改跳转的地方:

00408F5E|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/断在这里
00408F64E9FE140000JMPmyuninst.0040A467;跳到我们的第一部分补丁代码处
00408F69|.E8A098FFFF|CALL<myuninst.sub_40280E>

第二个要修改跳转的地方:

00408F91|.FF1598B24000|CALLDWORDPTRDS:[<&USER32.CreateWindowExA>];/CreateWindowExA
00408F97E9DE140000JMPmyuninst.0040A47A;跳到我们的第二部分补丁代码处
00408F9C90NOP
00408F9D|.FF30|PUSHDWORDPTRDS:[EAX];/<%s>

修改好后保存,同样不要忘了再修改一下.rdata区段的属性。运行一下,一切OK!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值