Nisy 发表于 2006-7-28 23:07:41

学习脱宣宣极速网络电视脱壳去自校验的参考文章

对我这小菜鸟来说 完宣宣困难还是不小的 找来 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]

Nisy 发表于 2006-7-28 23:09:05

标题: 浅谈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

Nisy 发表于 2006-7-28 23:46:30

原来7280 和 宣宣 有是同一作者的两个产品 这都什么世道 晕~~

wzwgp 发表于 2006-7-29 22:49:22

写得真详细,学习!

bfqyygy 发表于 2006-7-30 00:13:41

精彩.好文章!!

ssxx164 发表于 2006-7-30 19:11:34

思路很新奇,很实用,感谢~

madshime 发表于 2007-1-30 04:31:38

很好的方法学习ing

honghe 发表于 2008-1-22 08:30:33

还是坛主牛。支持,致敬
页: [1]
查看完整版本: 学习脱宣宣极速网络电视脱壳去自校验的参考文章