zoukankan      html  css  js  c++  java
  • 逆向工程试验报告

    逆向工程这门课也已经上完了,作为考查课需要交一份实验报告,这里我学习(抄袭)了我是用户大佬的破解实战-第一战,由于大佬水平高深,步骤间跳跃很大,我这里把大佬的意思加上自己的理解写成实验报告,仅供小白参考。

    出处:https://www.52pojie.cn/thread-197281-1-1.html

    注:中间有几处坑,表示自己掉进去的地方,如果想一次性实现,可以看完再练习。

    开始吧!

    超级U盘锁的逆向分析 

    超级U盘锁是一款功能强大的锁定系统U盘使用软件,限制他人使用U盘或其它移动存储设备从你的电脑复制或盗取重要资料。另外新版增加锁定光驱和锁定网络功能,防止更多方式盗走你的数据。软件提供密码设置,可以自由设置解除和锁定限制的功能。锁定U盘后,插上任意U盘或移动存储设备到计算机是没有任何反应的,需要解锁后才能正常使用U盘设备。

    它全面支持Win 2000/XP/2003/Win7/Win8 32/64位。但其试用版不能添加密码,需要购买了才能设置密码。

    超级U盘锁下载地址:http://www.onlinedown.net/soft/52481.htm

    2.2 逆向过程及分析

    2.2.1安装逆向相关工具

    下载并安装逆向相关工具,本人使用的工具如下:

    VMware虚拟机,WIN7操作系统,PEiD查壳工具以及OD。

    2.2.2 脱壳

    首先将超级U盘锁软件下载安装到虚拟机中,并使用PEiD查看该软件,如图2-1所示,显示该程序加的是ASpack压缩加密壳。

     

     

    图2-1 查壳

    然后使用OD手动脱壳,我采用的是单步跟踪法,即向上的跳转不能让他实现,向下的跳转让他实现。

    把超级U盘锁(Ulock)软件载入OD,会出现如图2-2所示的入口点警告,这就是因为软件加壳,因而无法直接找到程序入口点。

    这里不管他,我们点击确认,然后进行单步跟踪法,按F8或者菜单栏的单步步过。期间在汇编的一个CALL命令后步过,软件就直接运行了。这里我们称为软件跑飞(图2-3)了,即没有找到OEP就直接运行了软件。

     

    图2-3 软件跑飞

    所以我们需要重新再次跑程序,不能步过(F8)这个函数,而应该步入F7这个函数继续去找OEP。

    在遇到如图2-4示的需要往回跳转时候,按照单步跟踪法,我们点击下一行即005C205F 然后按F4(运行到选定位置)执行到这一行。

     

    图2-4 单步跟踪法

    按照这种方法,我们很快就找到了有RETN的地方,这里也就会进入OEP了,如图2-5边寄存器也显示模块入口点了

     

    图2-5 程序OEP进入点

    我们继续跟踪步入,会进入图2-6所示的地方,这里没有任何的反汇编,只需要右键分析即可显示OEP了(图2-7)。

     

    图2-6 未分析的OEP

     

    图2-7分析后的OEP

    这时,我们就可以使用插件中的OD脱壳模块来脱壳了(图2-8)。

     

    图2-8 OD自带脱壳来脱壳

    我们随机保存为ssozh.exe文件,然后打开发现无法运行该程序(图2-9),百度发现这是因为ASLR基地址随机化的原因,这样脱壳dump插件获取的地址不对,而ASLR是在WIN7/8或以上系统才有,因而,我改换成了XP的系统重新脱壳。

     

    图 2-9 程序错误

     

    图 2-10 编程语言E language

    2.2.3去自校验

    二次脱壳我采用了ESP定律法进行手动脱壳,这里不再赘述,很成功,发现是用E language写的(图2-10)。

    打开脱壳后的软件(名字叫123),这时候直接运行OD是看不出来程序有自校验了。因为这个程序不能同时开两个,在电脑去壳期间已经运行着一个程序,而去壳后用OD再次载入,他并没有再次打开该程序而是OD自己结束了,没有运行起来。这时候会把之前运行的误认为是当前OD调试的。因而无法出现如下的自校验窗口,你需要先把软件强制关闭,再次打开。

    你会发现这个软件很难关闭,因此需要使用上面推荐的PChunter强制关闭,再次打开OD调试就会发现程序有自效验(图2-11)。

     

    图 2-11 自校验

     

    OD打开123.exe(代码2-1)

    00401000 >/$  E8 06000000   call 123.0040100B

    00401005  |.  50 push eax                                   ; ExitCode = 0x0

    00401006  .  E8 BB010000 call <jmp.&kernel32.ExitProcess> ; ExitProcess

    0040100B   $  55           push ebp

    0040100C   .  8BEC         mov ebp,esp

    0040100E   .  81C4 F0FEFFFF add esp,-0x110

    00401014   .  E9 83000000   jmp 123.0040109C

    00401019   .  6B 72 6E 6C 6>ascii "krnln.fnr",0

    00401023   .  20 20 20 20 2>ascii "         ",0

    0040102D   .  47 65 74 4E 6>ascii "GetNewSock",0

    00401038   .  20 20 20 20 2>ascii "                "

    00401048   .  20 20 20 20 2>ascii "         ",0

    00401052   .  20 20 20 20 0>ascii "    ",0

    00401057   .  c8 b1c9 d9    enter 0xc9b1,0xd9

    0040105B   .  c8 edbc fe    enter 0xbced,0xfe

    0040105F   .  BA CBD0C4D6   mov edx,0xD6C4D0CB

    00401064   .  A7            cmps dword ptr ds:[esi],dword ptr es:[ed>

    00401065   .  B3 D6         mov bl,0xD6

    00401067   .  BF E22CC7EB   mov edi,0xEBC72CE2

    0040106C   .  D6            salc

    0040106D   .  D8D0          fcom st

    代码2-1

    命令行对window下的MessageBoxA 下断点,即 BP MessageBoxA。然后进行堆栈回溯(图2-12)。图中右边是堆栈区,显示了图2-11中“程序已经被破坏”的位置,因为操作系统是小端模式,为了寻找代码段所在的位置,应该向之前的高地址寻找到第一次出现“程序已经被破坏”字符前面对应的代码段堆栈地址(代码2)找到堆栈地址0012FBD0进行反汇编窗口跟随(按回车)。图2-13所示即反汇编窗口跟随的用户代码领空,上一句CALL调用函数应该就是自校验,我们需要跳过这个自校验,即继续向上找找到可以跳过的JMP之类的语句。最后在005AECD9地址把反汇编语句jle 123.005AED5B改成JMP 123.005AED5B即可。

     

    图 2-12 下断,堆栈回溯

     

    0012FBD0 005AED3E  返回到 123.005AED3E 来自 123.005B2D86;第一句出现堆栈回溯的用户领空

    0012FBD4 00000003

    0012FBD8 00166868  ASCII "程序已经被破坏!"

    0012FBDC 00000000

    0012FBEC 80000301

    0012FBF0 00403452  123.00403452

    0012FBF4 00000000

    0012FC00 0017A9FC

    0012FC04 100E7A1C  krnln.100E7A1C

    0012FC08 100E5948  krnln.100E5948

    0012FC0C 0012FC50

    0012FC10 001C5400 ASCII "((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((("

    0012FC14   00166868  ASCII "程序已经被破坏!";第一句关键字符串

    0012FC18   00464100  ASCII "B3CCD0F2D2D1BEADB1BBC6C6BBB521"

    代码2-2

     

     

    图2-14 反汇编窗口跟随

           双击右下角图标,,弹出程序被损坏(图2-15)

     

    图2-15程序被破坏

     

    BP ExitProcess,堆栈回溯,定位程序退出为005B0B27处,向上找出关键跳转为005B0A76 改为JMP。完成去自校验(如图2-16)

     

    图2-16

    2.2.4 爆破

    如图2-16试用版不能保存密码,点击密码修改按钮,弹出提示,试用版不能保存密码。为了保存多个情况下的软件去自校验后的名字为ssozh.exe。

    前面我们查壳的时候已经知道这款软件是用易语言写的,我们采用二进制字符串查找法,查找易语言的按钮事件(特征码 )FF55FC,然后找到关键代码即可实现注册。

    具体过程如下:

           第一步,直接在反汇编窗口的00401000处搜索特征码FF55FC,发现找不到,说明这个软件不是静态编译,而是动态编译的,因此我们需要到易语言下的运行库(krnln.fnr,注一般来说运行库(动态库)是dll格式的,易语言的运行库后缀不太一样)里面查找特征码。

           第二步,因此我们先在OD中运行程序,打开模块窗口(Alt+E)或者点击上面的e按钮,找到krnln.fnr并回车,在CPU中查看代码(如图2-17)。并再次在CPU反汇编窗口查找二进制字符串FF55FC,找到相关代码段(代码2-3),第一行二进制代码就是FF55FC,在这里设置断点。

     

    图2-17 在模块中找到易语言的运行库

    10028B46  |> FF55 FC call [local.1] ;  krnln.100E5948

    10028B49  |.  5F pop edi ;  user32.77D191BE

    10028B4A  |.  5E pop esi ;  user32.77D191BE

    10028B4B  |.  895D F4 mov [local.3],ebx

    10028B4E  |.  8945 F8 mov [local.2],eax

    10028B51  |.  8B55 E8 mov edx,[local.6]

    10028B54  |.  C782 0C020000>mov dword ptr ds:[edx+0x20C],0x0

    10028B5E  |.  837D F4 00 cmp [local.3],0x0

    代码2-3

    第三步,点击设置密码按钮,此时程序会运行到断点处停止,这里F7步入CALL,即找到关键代码处(代码2-4),代码2-4中005AE463地址显然这两行是一个if-else语句,在动态调试中表明会直接跳转到代码2-4最后一行005AE4AA,显然是无法设置密码,因此我们可以猜测,比较一定是关于是否注册的比较,而紧接着if成立就是已经注册的程序,不成立则是试用版。

    如果想要实现设置密码,则可以直接对005AE463填充成nop。

    005AE453    55              push ebp

    005AE454    8BEC           mov ebp,esp

    005AE456    81EC 08000000   sub esp,0x8

    005AE45C    833D F40CB600 0>cmp dword ptr ds:[0xB60CF4],0x0

    005AE463    0F84 41000000   je ssozh.005AE4AA

    005AE469    68 02000080     push 0x80000002

    005AE46E    6A 00           push 0x0

    005AE470    68 01000000     push 0x1

    005AE475    68 01000100     push 0x10001

    005AE47A    68 00000106     push 0x6010000

    005AE47F    68 01000152     push 0x52010001

    005AE484    68 01000100     push 0x10001

    005AE489    68 CB000106     push 0x60100CB

    005AE48E    68 CC000152     push 0x520100CC

    005AE493    68 03000000     push 0x3

    005AE498    BB 20030000     mov ebx,0x320

    005AE49D    E8 E4480000     call ssozh.005B2D86     ; 弹出密码窗口

    005AE4A2    83C4 28         add esp,0x28

    005AE4A5    E9 90000000     jmp ssozh.005AE53A

    005AE4AA    68 02000080     push 0x80000002

    代码2-4

    重新运行程序,定位[0xB60674],下断硬件断点,重启程序,因为一开始内存中没有0xB60674这个地址使用,因此在多次执行到代码段005xxxxxx的时候找到这段代码在代码2-5的第三句。显然这段代码表示未注册的意思,我们向上找到跳转到这里的语句(第一句),我们把第一句nop掉,或把005AD381 代码处的0x0改为0x1,即可表示已经注册。

    005AD31D   /0F84 5E000000   je ssozh1.005AD381

    005AD323    C705 F40CB600 0>mov dword ptr ds:[0xB60CF4],0x1

    //关于注册的相关语句

    005AD381    C705 F40CB600 0>mov dword ptr ds:[0xB60CF4],0x0

    第三章 实验总结

    3.1 实验结果总结

    成功爆破后可以使用密码了。

     

    3-1 设置密码

    3.2 实验心得

    写实验报告,自己实践是提升自我解决问题能力以及快速掌握一些知识点的良好途径,作为一名跨专业的学生,在本课程中学会了很多,感谢何老师的引领入门,感谢同学们的帮助,感谢网络上的各位网友的指导。

  • 相关阅读:
    浅析JTable与TableModel、TableCellRenderer、TableCellEditor接口——使用JComboBox显示单元格的值
    设计模式学习笔记-观察者模式
    swing布局管理器简介
    java调用博思得打印机的心得
    Eclipse 创建web项目后没有 Java EE 5 Library,没有web开发相关基础java包,myeclipse中有。
    三层架构与MVC的区别
    浅析MVC模式与三层架构的区别
    用正则表达式判断字符串是否是数字
    gdb用法
    解决linux的-bash: ./xx: Permission denied
  • 原文地址:https://www.cnblogs.com/SsoZhNO-1/p/9983769.html
Copyright © 2011-2022 走看看