学习脱宣宣极速网络电视脱壳去自校验的参考文章
对我这小菜鸟来说 完宣宣困难还是不小的 找来 lhl8730 兄的宣宣极速网络电视脱壳去自校验及算法分析https://www.chinapyg.com/viewthread.php?tid=4460&highlight=%D0%FB%D0%FB%BC%AB%CB%D9%CD%F8%C2%E7%B5%E7%CA%D3
来学习一下 看到几篇参考的帖子 所以一并转过来
对付自校验的杀手锏 -- 偷天换日
: prince
[时间] : 2005.06.30
[声明] : 只做技术交流,不做商业用途,如果你手头宽裕并喜欢这个软件的话请支持正版软件。
: [email protected]
[软件信息]
[软件说明] : 绿鹰PC万能精灵3.92
[保护方式] : 序列号 + 反跟踪 + 自校验
[限制方式] : 功能限制
[外壳保护] : PECompact 2.x -> Jeremy Collake
[编译器/语言]: Microsoft Visual C++ 6.0 / C/C++
[下载地址] : http://www.onlinedown.net/soft/6396.htm
[目的] : 脱壳后自校验的去除
[分析过程] :
现在的软件为了保护自身不被修改,防止脱壳,不少都加上了自身的完整性校验,一旦发现自己的关
键指令被修改或被脱壳了之后,毫无提示地就拒绝运行。这种令人头疼的自校验相信难住了不少跟我一样的菜
鸟!后来再遇到自校验,就只好开2个OD,下断CreateFilaA,然后,一步一步地跟,一句一句地对照,不知道
经过多少步之后,哈,发现了一个跳转不一样,激动地改掉,成功。对于这样直接比较的软件校验还可以用这
种不怕脏不怕累的方法搞定,但是对于那种将自身一个字节一个字节读取出来,然后经过超级变态复杂地算法
运算过后,得到了几个值,偷偷记下来,或作为后面程序运行的关键值,或者分开几个地方偷偷比较,如果不
对,就异常啦!这样的校验,相信光凭耐心和运气是不行的吧?前几天在FLY老大的论坛偶尔看到了fxyang大侠
的文章:浅谈cool edit
pro2.1脱壳后解决自校验(http://fly.ccgforum.info/viewthread.php?tid=1339&fpage=1&highlight=%2Bfxy
ang),就是这样的校验方法。fxyang大侠的解决方法是:跟踪原版程序,由于校验算法复杂,不分析过程,只
看得到几个结果,记下他们的值,然后在脱壳后的文件中强行将原版的值写入,最后问题解决。这种方法虽然
巧妙,但是还是要分析原版计算出的几个值,并且要在脱壳文件中一个一个写入,修改字节过多,容易出错,
毕竟如果有一个地方没有看到就会前功尽弃。有没有更好更简便的方法呢?呵呵,终于到主题了,呼~
静下心来仔细想一下,总结一下程序是如何校验的:首先GetModuleFileName得到自身路径和文件名,
然后CreateFile打开自身,接下来,如果是简单地比较大小,就调用GetFileSize得到程序大小,和原版大小比
较,size大了就OVER,这种比较简单;如果要进行CRC等算法校验,就会CreateFileMapping映射的一块内存中
准备ReadFila读取计算,最后计算出几个值... 呵呵,注意到了吗?无论哪中方法,都要CreateFile打开自身
,才能进行下一步操作,那想到了吗?既然程序要打开文件校验,我们就想办法让它不打开自身而是打开原版
去计算、校验,这样无论它怎么变态,复杂,计算结果都是正确的(当然了,因为原版我们就没动过嘛)!那
可能有朋友说了,怎么做啊?改动复杂不复杂啊?别急,看我举例说明。
PECompact 2.x脱壳很简单(难的我也不会),he eip,F9一步就到OEP,Ollydump脱之,ImportREC修
复,剪切掉一个垃圾指针,FIX,OK,就这么简单。由于有自校验,双击运行毫无反应,OD加载,下断CreateFi
leA,F9来到这里:
----------------------------------------------------------------------------------
0040538D 8B4D 08 mov ecx,dword ptr ss: ; 文件名送ECX
00405390 57 push edi ; 参数入栈
00405391 68 27000008 push 8000027 ; 参数入栈
00405396 6A 03 push 3 ; 参数入栈
00405398 57 push edi ; 参数入栈
00405399 6A 01 push 1 ; 参数入栈
0040539B 68 00000080 push 80000000 ; 参数入栈
004053A0 51 push ecx ; 关键,要打开的文件名入栈
004053A1 FF15 74124600 call dword ptr ds:[<&kernel32.Cr>; KERNEL32.CreateFileA
004053A7 8BF0 mov esi,eax
004053A9 83FE FF cmp esi,-1
004053AC 8975 DC mov dword ptr ss:,esi
004053AF 75 15 jnz short UN2_.004053C6
004053B1 FF15 84124600 call dword ptr ds:[<&kernel32.Ge>; KERNEL32.GetLastError
004053B7 56 push esi
004053B8 8945 EC mov dword ptr ss:,eax
004053BB FF15 78124600 call dword ptr ds:[<&kernel32.Cl>; KERNEL32.CloseHandle
004053C1 E9 13010000 jmp UN2_.004054D9
004053C6 8D55 D4 lea edx,dword ptr ss:
...
文件校验算法没看:
00405465 FF15 6C124600 call dword ptr ds:[<&kernel32.MapV>; KERNEL32.MapViewOfFile
0040546B 8BF8 mov edi,eax
0040546D 8BC6 mov eax,esi
0040546F 8BD7 mov edx,edi
00405471 8BC8 mov ecx,eax
00405473 48 dec eax
00405474 85C9 test ecx,ecx
00405476 8945 08 mov dword ptr ss:,eax
00405479 76 28 jbe short UN2_+.004054A3
0040547B 8B0B mov ecx,dword ptr ds:
0040547D 33DB xor ebx,ebx
0040547F 8A1A mov bl,byte ptr ds: ;PE文件头,表示开始计算了
00405481 8BC1 mov eax,ecx
00405483 25 FF000000 and eax,0FF
00405488 33C3 xor eax,ebx
0040548A 8B5D E0 mov ebx,dword ptr ss:
0040548D C1E9 08 shr ecx,8
00405490 8B5B 04 mov ebx,dword ptr ds:
00405493 8B0483 mov eax,dword ptr ds:
00405496 8B5D 0C mov ebx,dword ptr ss:
00405499 33C1 xor eax,ecx
0040549B 42 inc edx
0040549C 8903 mov dword ptr ds:,eax
0040549E 8B45 08 mov eax,dword ptr ss:
004054A1^ EB CE jmp short UN2_+.00405471
004054A3 57 push edi
004054A4 FF15 70124600 call dword ptr ds:[<&kernel32.Unma>; KERNEL32.UnmapViewOfFile
004054AA 8B55 CC mov edx,dword ptr ss:
004054AD 8B4D D0 mov ecx,dword ptr ss:
004054B0 33C0 xor eax,eax
004054B2 03D6 add edx,esi
004054B4 13C8 adc ecx,eax
004054B6 8955 CC mov dword ptr ss:,edx
004054B9 8B55 D8 mov edx,dword ptr ss:
004054BC 894D D0 mov dword ptr ss:,ecx
004054BF 8B4D D4 mov ecx,dword ptr ss:
004054C2 8B7D D0 mov edi,dword ptr ss:
004054C5 2BCE sub ecx,esi
004054C7 8B75 DC mov esi,dword ptr ss:
004054CA 1BD0 sbb edx,eax
004054CC 894D D4 mov dword ptr ss:,ecx
004054CF 8955 D8 mov dword ptr ss:,edx
004054D2^ E9 5CFFFFFF jmp UN2_+.00405433
004054D7 33FF xor edi,edi
004054D9 8B45 E8 mov eax,dword ptr ss:
004054DC 3BC7 cmp eax,edi
004054DE 74 0C je short UN2_+.004054EC
004054E0 50 push eax
004054E1 FF15 78124600 call dword ptr ds:[<&kernel32.Clos>; KERNEL32.CloseHandle
----------------------------------------------------------------------------------
在0040538D处就是要打开的文件名赋值,我们就是要改这里,让ECX的值为我们指定的原版程序的路径
。首先做好准备工作,比如原版的路径为:D:\Program Files\绿鹰PC万能精灵\adam.exe,我们的目的就是要
将这个字符串赋值给ECX,所以要把这个字符串写到程序里面。由于PE文件Section之间的对齐,文件中通常都
会有一些间隙,用UE打开就是一堆00的地方,我们就用利用这些空地写我们的原版路径字符串。用UE打开脱壳
后的文件,拉动滚动条到文件的结尾,找到空地,将上面的字符串复制拷贝上去,记住我们的字符串的起始地
址,我的是001B1F80,后面用的到,然后保存修改后的文件。回到OD,下断0040538D,重新断下,修改代码为
:
----------------------------------------------------------------------------------
0040538D- E9 1ECC1A00 jmp UN2_+.005B1FB0
00405392 90 nop
00405393 90 nop
00405394 90 nop
00405395 90 nop
00405396 6A 03 push 3
00405398 57 push edi
00405399 6A 01 push 1
0040539B 68 00000080 push 80000000
004053A0 51 push ecx
004053A1 FF15 74124600 call dword ptr ds:[<&kernel32.Cr>; KERNEL32.CreateFileA
----------------------------------------------------------------------------------
因为这里空间不够写入我们的代码,所以要跳到空地写我们要的代码,然后再跳回来继续执行,就好
象什么都没有发生过一样,:) 005B1FB0是我们写好的字符串后面的空地:
----------------------------------------------------------------------------------
005B1FA8 0000 add byte ptr ds:,al ;空地
005B1FAA 0000 add byte ptr ds:,al ;空地
005B1FAC 0000 add byte ptr ds:,al ;空地
005B1FAE 0000 add byte ptr ds:,al ;空地
005B1FB0 B9 801F5B00 mov ecx,UN2_+.005B1F80 ;将写好的原版文件地址赋值给ECX
005B1FB5 57 push edi ;继续压栈参数
005B1FB6 68 27000008 push 8000027 ;继续压栈参数
005B1FBB- E9 D633E5FF jmp UN2_+.00405396 ;跳回去继续执行
005B1FC0 0000 add byte ptr ds:,al
----------------------------------------------------------------------------------
OK,保存修改,完工,运行试试,呵呵,运行了~:)
AntiDebug的去除:下断SetUnhandledExceptFilter,共有两处,NOP掉;程序循环调用CreateToolhelp32Snaps
hot进行父进程检测,将00418807处改为绝对跳转即可。
总结一下:
用这种偷天换日,偷梁换柱,偷龙转凤的手段可以轻易对付利用CreateFilaA进行自校验的程序,代码
修改量小,成功率高,真乃居家旅行必备之良药,^O^ 以后再见到自校验,就又多了一种对付的方法啦~
看来遇到难以解决的问题的时候,不妨转换一下思路,避其锋芒,找到像打太极一样,四量拨千斤的
方法,复杂问题就变得简单了。
呵呵,小弟菜鸟,有不妥的地方请各位大侠见谅、赐教,另外多谢fxyang大侠的文章。
菜鸟写菜文~
补充:以上方法不适用于破解文件的发布,因为要依赖原版的绝对路径,我的目的只是为了让它运行
,从而使我们可以跟踪出它的注册码或者注册算法,毕竟写出注册机才是我们的最终目标。
prince 2005.06.30
任何问题可至:[email protected] 标题: 浅谈cool edit pro V2.1脱壳后解决自校验
发表于: 2005-6-10 09:29
--------------------------------------------------------------------------------
声音编辑软件cool edit Pro2.10脱壳后自效验的解决:
用UPX Shell脱壳后出现错误提示"Internal Error (13)" 然后就死在:
005AA600/$8B4424 04 MOV EAX,DWORD PTR SS:
005AA604|.53 PUSH EBX
005AA605|.55 PUSH EBP
005AA606|.56 PUSH ESI
005AA607|.57 PUSH EDI
005AA608|.6A EB PUSH -15 ; /Index = GWL_USERDATA
005AA60A|.50 PUSH EAX ; |hWnd
005AA60B|.FF15 38166900 CALL DWORD PTR DS:[<&USER32.GetWindowLon>; \GetWindowLongA
005AA611|.8BF8 MOV EDI,EAX
005AA613|.8B5C24 24 MOV EBX,DWORD PTR SS:
005AA617|.8B47 10 MOV EAX,DWORD PTR DS: ;无效的值
DS:=???
EAX=00000000
修改了原程序时间和原程序名,运行正常,说明是对主程序的CRC效验。
但是这个程序的效验是基于消息参数的验证,不容易跟踪到验证的地方:
005F0BFF .FFD6 CALL ESI ; \GetWindowLongA
005F0C01 .8BF8 MOV EDI,EAX
005F0C03 .A1 D04A6E00 MOV EAX,DWORD PTR DS:
005F0C08 .8B9C24 080300>MOV EBX,DWORD PTR SS:
005F0C0F .3BD8 CMP EBX,EAX
看看这里的EBX的值和没有脱壳的EBX值的比较 log EBX:
未脱壳时:
005F0C0FCOND: 00000024
005F0C0FCOND: 00000081
005F0C0FCOND: 00000083
005F0C0FCOND: 00000001
77BA0000模块 C:\WINDOWS\System32\midimap.dll
005F0C0FCOND: 0000000D
005F0C0FCOND: 00000210
005F0C0FCOND: 0000001F
005F0C0FCOND: 0000000A
005F0C0FCOND: 0000001C
005F0C0FCOND: 00002A71 //第10次的值
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
60800000模块 C:\WINDOWS\System32\mslbui.dll
005F0C0FCOND: 00000113
005F0C0FCOND: 0000001C
005F0C0FCOND: 0000000A
005F0C0FCOND: 00000055
005F0C0FCOND: 00000129
005F0C0FCOND: 00000210
77E5D295ID 00000834 的新线程已经创建 //程序运行了
脱壳后:
005F0C0FCOND: 00000024
005F0C0FCOND: 00000081
005F0C0FCOND: 00000083
005F0C0FCOND: 00000001
77BA0000模块 C:\WINDOWS\System32\midimap.dll
005F0C0FCOND: 0000000D
005F0C0FCOND: 00000210
005F0C0FCOND: 0000001F
005F0C0FCOND: 0000000A
005F0C0FCOND: 0000001C
005F0C0FCOND: 00000010 //看看第10 次的值不同了
005F0C0FCOND: 00000210
10000000模块 D:\Program Files\sina\UC\UCIdleHook.dll
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 00000113
005F0C0FCOND: 0000001C
005F0C0FCOND: 00000002
005F0C0FCOND: 00000082
005AA617访问违反: 读取 //程序出错
00400000卸载 C:\Program Files\coolpro2\coolpro20.exe
原想跟踪这个值,看看什么地方验证的,一直找不到,花了不少的时间
后来考虑,既然是CRC验证一般会使用这个CreateFileA函数,试试:
bp CreateFileA几次后来到:
00492570/$55 PUSH EBP
00492571|.8BEC MOV EBP,ESP
00492573|.81EC 24010000 SUB ESP,124
00492579|.53 PUSH EBX
0049257A|.8B1D F8456E00 MOV EBX,DWORD PTR DS:
00492580|.57 PUSH EDI
00492581|.8D85 DCFEFFFF LEA EAX,DWORD PTR SS:
00492587|.68 04010000 PUSH 104 ; /BufSize = 104 (260.)
0049258C|.50 PUSH EAX ; |PathBuffer
0049258D|.6A 00 PUSH 0 ; |hModule = NULL
0049258F|.895D EC MOV DWORD PTR SS:,EBX ; |
00492592|.FF15 C4126900 CALL DWORD PTR DS:[<&KERNEL32.GetModuleF>; \GetModuleFileNameA
00492598|.85C0 TEST EAX,EAX
0049259A|.0F84 74010000 JE coolpro2.00492714
004925A0|.6A 00 PUSH 0 ; /hTemplateFile = NULL
004925A2|.68 80000008 PUSH 8000080 ; |Attributes = NORMAL|SEQUENTIAL_SCAN
004925A7|.6A 03 PUSH 3 ; |Mode = OPEN_EXISTING
004925A9|.6A 00 PUSH 0 ; |pSecurity = NULL
004925AB|.6A 01 PUSH 1 ; |ShareMode = FILE_SHARE_READ
004925AD|.8D8D DCFEFFFF LEA ECX,DWORD PTR SS: ; |
004925B3|.68 00000080 PUSH 80000000 ; |Access = GENERIC_READ
004925B8|.51 PUSH ECX ; |FileName
004925B9|.FF15 10136900 CALL DWORD PTR DS:[<&KERNEL32.CreateFile>; \CreateFileA
004925BF|.8BF8 MOV EDI,EAX
004925C1|.83FF FF CMP EDI,-1
004925C4|.897D E8 MOV DWORD PTR SS:,EDI
004925C7|.0F84 47010000 JE coolpro2.00492714
004925CD|.56 PUSH ESI
004925CE|.6A 00 PUSH 0 ; /pFileSizeHigh = NULL
004925D0|.57 PUSH EDI ; |hFile
004925D1|.FF15 14136900 CALL DWORD PTR DS:[<&KERNEL32.GetFileSiz>; \GetFileSize
004925D7|.8BF0 MOV ESI,EAX
004925D9|.8975 F8 MOV DWORD PTR SS:,ESI
004925DC|.FF15 9C126900 CALL DWORD PTR DS:[<&KERNEL32.GetProcess>; [GetProcessHeap
004925E2|.56 PUSH ESI ; /HeapSize
004925E3|.6A 08 PUSH 8 ; |Flags = HEAP_ZERO_MEMORY
004925E5|.50 PUSH EAX ; |hHeap
004925E6|.8945 E4 MOV DWORD PTR SS:,EAX ; |
004925E9|.FF15 94126900 CALL DWORD PTR DS:[<&KERNEL32.HeapAlloc>>; \HeapAlloc
004925EF|.8BF0 MOV ESI,EAX
004925F1|.85F6 TEST ESI,ESI
004925F3|.8975 E0 MOV DWORD PTR SS:,ESI
004925F6|.0F84 10010000 JE coolpro2.0049270C
004925FC|.8B45 F8 MOV EAX,DWORD PTR SS:
004925FF|.8D55 F4 LEA EDX,DWORD PTR SS:
00492602|.6A 00 PUSH 0 ; /pOverlapped = NULL
00492604|.52 PUSH EDX ; |pBytesRead
00492605|.50 PUSH EAX ; |BytesToRead
00492606|.56 PUSH ESI ; |Buffer
00492607|.57 PUSH EDI ; |hFile
00492608|.FF15 4C136900 CALL DWORD PTR DS:[<&KERNEL32.ReadFile>] ; \ReadFile
0049260E|.85C0 TEST EAX,EAX
00492610|.0F84 E9000000 JE coolpro2.004926FF
00492616|.C745 FC 01000>MOV DWORD PTR SS:,1
0049261D|.BB 03000000 MOV EBX,3
00492622|.C745 F0 00000>MOV DWORD PTR SS:,0
00492629|.C745 F8 06000>MOV DWORD PTR SS:,6
00492630|.B9 02000000 MOV ECX,2
00492635|>66:0FB63E /MOVZX DI,BYTE PTR DS:
00492639|.8B55 F0 |MOV EDX,DWORD PTR SS:
0049263C|.46 |INC ESI
0049263D|.8D0417 |LEA EAX,DWORD PTR DS:
00492640|.33D2 |XOR EDX,EDX
00492642|.8A55 FD |MOV DL,BYTE PTR SS:
00492645|.8D04B8 |LEA EAX,DWORD PTR DS:
00492648|.8B7D F8 |MOV EDI,DWORD PTR SS:
0049264B|.03D0 |ADD EDX,EAX
0049264D|.03D7 |ADD EDX,EDI
0049264F|.03D3 |ADD EDX,EBX
00492651|.33D1 |XOR EDX,ECX
00492653|.8BC2 |MOV EAX,EDX
00492655|.8B55 FC |MOV EDX,DWORD PTR SS:
00492658|.F6C6 80 |TEST DH,80
0049265B|.74 02 |JE SHORT coolpro2.0049265F
0049265D|.0C 01 |OR AL,1
0049265F|>8945 FC |MOV DWORD PTR SS:,EAX
00492662|.8B45 F4 |MOV EAX,DWORD PTR SS:
00492665|.48 |DEC EAX
00492666|.8945 F4 |MOV DWORD PTR SS:,EAX
00492669|.74 70 |JE SHORT coolpro2.004926DB
0049266B|.33C0 |XOR EAX,EAX
0049266D|.66:8B45 FC |MOV AX,WORD PTR SS:
00492671|.BA F1F0F0F0 |MOV EDX,F0F0F0F1
00492676|.F7E2 |MUL EDX
00492678|.C1EA 04 |SHR EDX,4
0049267B|.8955 F0 |MOV DWORD PTR SS:,EDX
0049267E|.8B4D FC |MOV ECX,DWORD PTR SS:
00492681|.66:0FB606 |MOVZX AX,BYTE PTR DS:
00492685|.81E1 FFFF0000 |AND ECX,0FFFF
0049268B|.D1E0 |SHL EAX,1
0049268D|.8D14CD 000000>|LEA EDX,DWORD PTR DS:
00492694|.25 FFFF0000 |AND EAX,0FFFF
00492699|.2BD1 |SUB EDX,ECX
0049269B|.8B4D F0 |MOV ECX,DWORD PTR SS:
0049269E|.03D0 |ADD EDX,EAX
004926A0|.33C0 |XOR EAX,EAX
004926A2|.D1FA |SAR EDX,1
004926A4|.8AC7 |MOV AL,BH
004926A6|.03CA |ADD ECX,EDX
004926A8|.8B55 F8 |MOV EDX,DWORD PTR SS:
004926AB|.03C1 |ADD EAX,ECX
004926AD|.46 |INC ESI
004926AE|.33C2 |XOR EAX,EDX
004926B0|.F6C7 80 |TEST BH,80
004926B3|.74 02 |JE SHORT coolpro2.004926B7
004926B5|.0C 01 |OR AL,1
004926B7|>8BD8 |MOV EBX,EAX
004926B9|.895D F8 |MOV DWORD PTR SS:,EBX
004926BC|.33C0 |XOR EAX,EAX
004926BE|.66:8B45 F8 |MOV AX,WORD PTR SS:
004926C2|.BA 9ED8899D |MOV EDX,9D89D89E
004926C7|.F7E2 |MUL EDX
004926C9|.C1EA 03 |SHR EDX,3
004926CC|.8955 F0 |MOV DWORD PTR SS:,EDX
004926CF|.8B45 FC |MOV EAX,DWORD PTR SS:
004926D2|.8D141B |LEA EDX,DWORD PTR DS:
004926D5|.8955 F8 |MOV DWORD PTR SS:,EDX
004926D8|.8D0C00 |LEA ECX,DWORD PTR DS:
004926DB|>8B45 F4 |MOV EAX,DWORD PTR SS:
004926DE|.48 |DEC EAX
004926DF|.8945 F4 |MOV DWORD PTR SS:,EAX
004926E2|.^ 0F85 4DFFFFFF \JNZ coolpro2.00492635
004926E8|.8B45 EC MOV EAX,DWORD PTR SS:
004926EB|.33C9 XOR ECX,ECX
004926ED|.8B75 E0 MOV ESI,DWORD PTR SS:
004926F0|.8B7D E8 MOV EDI,DWORD PTR SS:
004926F3|.8AEB MOV CH,BL
004926F5|.8A4D FC MOV CL,BYTE PTR SS:
004926F8|.03C1 ADD EAX,ECX
004926FA|.8945 EC MOV DWORD PTR SS:,EAX
004926FD|.8BD8 MOV EBX,EAX
004926FF|>8B55 E4 MOV EDX,DWORD PTR SS:
00492702|.56 PUSH ESI ; /pMemory
00492703|.6A 00 PUSH 0 ; |Flags = 0
00492705|.52 PUSH EDX ; |hHeap
00492706|.FF15 A4126900 CALL DWORD PTR DS:[<&KERNEL32.HeapFree>] ; \HeapFree
0049270C|>57 PUSH EDI ; /hObject
0049270D|.FF15 08126900 CALL DWORD PTR DS:[<&KERNEL32.CloseHandl>; \CloseHandle
00492713|.5E POP ESI
00492714|>A1 F0456E00 MOV EAX,DWORD PTR DS:
00492719|.891D 8C4C6E00 MOV DWORD PTR DS:,EBX
0049271F|.53 PUSH EBX ; /lParam
00492720|.2BD8 SUB EBX,EAX ; |
00492722|.8B45 08 MOV EAX,DWORD PTR SS: ; |
00492725|.83E3 07 AND EBX,7 ; |
00492728|.81C3 592A0000 ADD EBX,2A59 ; |
0049272E|.6A 00 PUSH 0 ; |wParam = 0
00492730|.53 PUSH EBX ; |Message
00492731|.50 PUSH EAX ; |hWnd
00492732|.FF15 EC156900 CALL DWORD PTR DS:[<&USER32.SendMessageA>; \SendMessageA
00492738|.8B0D 24516B00 MOV ECX,DWORD PTR DS:
0049273E|.8B15 60496E00 MOV EDX,DWORD PTR DS:
00492744|.6A 00 PUSH 0 ; /lParam = 0
00492746|.81E1 FFFF0000 AND ECX,0FFFF ; |
0049274C|.6A 00 PUSH 0 ; |wParam = 0
0049274E|.51 PUSH ECX ; |Message
0049274F|.52 PUSH EDX ; |hWnd => 1C02CA
00492750|.FF15 D4156900 CALL DWORD PTR DS:[<&USER32.PostMessageA>; \PostMessageA
00492756|.5F POP EDI
00492757|.B8 01000000 MOV EAX,1
0049275C|.5B POP EBX
0049275D|.8BE5 MOV ESP,EBP
0049275F|.5D POP EBP
00492760\.C2 0400 RETN 4
这是个明显的对程序的二进制验证:
004925B9|.FF15 10136900 CALL DWORD PTR DS:[<&KERNEL32.CreateFile>; \CreateFileA
0012EAB0 004925BF/CALL 到 CreateFileA 来自 coolpro2.004925B9
0012EAB4 0012EAD8|FileName = "C:\Program Files\coolpro2\coolpro20.exe"
0012EAB8 80000000|Access = GENERIC_READ
0012EABC 00000001|ShareMode = FILE_SHARE_READ
0012EAC0 00000000|pSecurity = NULL
0012EAC4 00000003|Mode = OPEN_EXISTING
0012EAC8 08000080|Attributes = NORMAL|SEQUENTIAL_SCAN
0012EACC 00000000\hTemplateFile = NULL
获得Hfile 句柄
004925D1|.FF15 14136900 CALL DWORD PTR DS:[<&KERNEL32.GetFileSiz>; \GetFileSize
0012EAC0 004925D7/CALL 到 GetFileSize 来自 coolpro2.004925D1
0012EAC4 00000138|hFile = 00000138 (window)
0012EAC8 00000000\pFileSizeHigh = NULL
获得文件的size
004925E9|.FF15 94126900 CALL DWORD PTR DS:[<&KERNEL32.HeapAlloc>>; \HeapAlloc
申请一个size大小的临时空间,存放程序数据
00492608|.FF15 4C136900 CALL DWORD PTR DS:[<&KERNEL32.ReadFile>] ; \ReadFile
0012EAB4 0049260E/CALL 到 ReadFile 来自 coolpro2.00492608
0012EAB8 00000138|hFile = 00000138 (window)
0012EABC 01A90020|Buffer = 01A90020
0012EAC0 00903010|BytesToRead = 903010 (9449488.)
0012EAC4 0012EBF0|pBytesRead = 0012EBF0
0012EAC8 00000000\pOverlapped = NULL
把程序数据全部读入Buffer,以便后面验证:
主验证数据获得的代码:
00492635/MOVZX DI,BYTE PTR DS:
DS:=4D ('M')
DI=0138
Jump from 004926E2
//从第一个字节开始
00492639|MOV EDX,DWORD PTR SS:
0049263C|INC ESI
0049263D|LEA EAX,DWORD PTR DS:
00492640|XOR EDX,EDX
00492642|MOV DL,BYTE PTR SS:
00492645|LEA EAX,DWORD PTR DS:
00492648|MOV EDI,DWORD PTR SS:
0049264B|ADD EDX,EAX
0049264D|ADD EDX,EDI
0049264F|ADD EDX,EBX
00492651|XOR EDX,ECX
00492653|MOV EAX,EDX
00492655|MOV EDX,DWORD PTR SS:
00492658|TEST DH,80
0049265B|JE SHORT coolpro2.0049265F
0049265D|OR AL,1
0049265F|MOV DWORD PTR SS:,EAX
00492662|MOV EAX,DWORD PTR SS:
00492665|DEC EAX
00492666|MOV DWORD PTR SS:,EAX
00492669|JE SHORT coolpro2.004926DB
0049266B|XOR EAX,EAX
0049266D|MOV AX,WORD PTR SS:
00492671|MOV EDX,F0F0F0F1
00492676|MUL EDX
00492678|SHR EDX,4
0049267B|MOV DWORD PTR SS:,EDX
0049267E|MOV ECX,DWORD PTR SS:
00492681|MOVZX AX,BYTE PTR DS:
00492685|AND ECX,0FFFF
0049268B|SHL EAX,1
0049268D|LEA EDX,DWORD PTR DS:
00492694|AND EAX,0FFFF
00492699|SUB EDX,ECX
0049269B|MOV ECX,DWORD PTR SS:
0049269E|ADD EDX,EAX
004926A0|XOR EAX,EAX
004926A2|SAR EDX,1
004926A4|MOV AL,BH
004926A6|ADD ECX,EDX
004926A8|MOV EDX,DWORD PTR SS:
004926AB|ADD EAX,ECX
004926AD|INC ESI
004926AE|XOR EAX,EDX
004926B0|TEST BH,80
004926B3|JE SHORT coolpro2.004926B7
004926B5|OR AL,1
004926B7|MOV EBX,EAX
004926B9|MOV DWORD PTR SS:,EBX
004926BC|XOR EAX,EAX
004926BE|MOV AX,WORD PTR SS:
004926C2|MOV EDX,9D89D89E
004926C7|MUL EDX
004926C9|SHR EDX,3
004926CC|MOV DWORD PTR SS:,EDX
004926CF|MOV EAX,DWORD PTR SS:
004926D2|LEA EDX,DWORD PTR DS:
004926D5|MOV DWORD PTR SS:,EDX
004926D8|LEA ECX,DWORD PTR DS:
004926DB|MOV EAX,DWORD PTR SS:
004926DE|DEC EAX
004926DF|MOV DWORD PTR SS:,EAX
004926E2\JNZ coolpro2.00492635
004926E8MOV EAX,DWORD PTR SS:
004926EBXOR ECX,ECX
004926EDMOV ESI,DWORD PTR SS:
004926F0MOV EDI,DWORD PTR SS:
004926F3MOV CH,BL
004926F5MOV CL,BYTE PTR SS:
004926F8ADD EAX,ECX
004926FAMOV DWORD PTR SS:,EAX
004926FDMOV EBX,EAX
不管他的算法 最终得到的值有
和后面:004926F3MOV CH,BL 要用的
EBX
现在跟踪原程序 得到这些值,然后修改这段代码,直接赋值给需要的地址
修改后的代码是:
004926B7|> \33C0 XOR EAX,EAX
004926B9|.B9 CA8FFEB0 MOV ECX,B0FE8FCA
004926BE|.BA CE58437C MOV EDX,7C4358CE
004926C3|.BB 67AC21BE MOV EBX,BE21AC67
004926C8|.BE 30FEF301 MOV ESI,1F3FE30
004926CD|.BF 8C5320BE MOV EDI,BE20538C
004926D2|.C745 F0 430D0>MOV DWORD PTR SS:,0D43
004926D9|.90 NOP
004926DA|.90 NOP
004926DB|>90 NOP
004926DC|.90 NOP
004926DD|.90 NOP
004926DE|.90 NOP
004926DF|.90 NOP
004926E0|.894D FC MOV DWORD PTR SS:,ECX
004926E3|.90 NOP
004926E4|.8955 F8 MOV DWORD PTR SS:,EDX
004926E7|.90 NOP
004926E8|.8B45 EC MOV EAX,DWORD PTR SS:
保存修改的程序,再次运行看看,正常了
by fxyang
2005.6.10 原来7280 和 宣宣 有是同一作者的两个产品 这都什么世道 晕~~ 写得真详细,学习! 精彩.好文章!! 思路很新奇,很实用,感谢~ 很好的方法学习ing 还是坛主牛。支持,致敬
页:
[1]