《CS:APP》二进制炸弹实验(phase_1-3)
《深入理解计算机系统》第三章的bomb lab,拆弹实验:给出一个linux的可执行文件bomb,执行后文件要求分别进行6次输入,每一次输入错误都会导致炸弹爆炸,程序终止。需要通过反汇编来逆向关键代码,得出通关密钥。
相关实验材料可以在CMU的官网下载:http://csapp.cs.cmu.edu/3e/labs.html
目前的水平只能完成到1-3关,详细记录攻关过程。
1.正式开始前的分析
先执行一下bomb,随便输入一个字串123。

首先要做的是在炸弹爆炸前设置断点,看程序都做了什么事情。
输入objdump -t bomb | less查看符号表,内容太多;猜测关键字和bomb有关,所以用正则表达式过滤一下:objdump -t bomb | grep "bomb".

其中explode_bomb应该和炸弹爆炸有关,可以在这里设置断点: b explode_bomb
再反汇编看一下代码:objdump -d bomb > bomb.txt, 搜索“main”,很容易找到主函数入口。

从main函数往下找, 往下有phase_1--phase_6的函数名称,猜测是每个关卡的入口,也在这里设断点:
b phase_1

2.Phase_1攻关
分析完了用GDB打开bomb,执行以下命令,让程序停在phase_1入口:

反汇编查看下代码: disas

从汇编代码来看,执行的过程是比较输入的字符串和预设的字符串,如果相同的话就通关,不相等就爆炸。调用strings_not_equal前,把地址0x402400的内容复制到了寄存器%esi。
X86-64系统CPU使用的16个寄存器中,每个寄存器都有一定的使用规则。比如%rbp, %rbx属于被调用者保存寄存器,被调用的函数必须保证返回调用者时,这两个寄存器的内容不变。所以被调用者一开始就必须先压栈,结束再出栈。
%rdi, %rsi, %rdx, %rcx, %r8, %9, 在执行call函数调用时,分别依次存放第1-6个参数,参数不足6个则后面的不使用;参数大于6个寄存器不够了,则开辟栈存在栈里面。
据此可以猜测,函数string_not_equal有两个形参,分别存在%edi和%esi中,用x/s $rdi 和 x/s $rsi 打印看下,果然一个是输入参数,一个是通关密码。据此第一关通过了。
3、Phase_2攻关
用"b phase_2"命令设置断点,随便输入一个字串后程序在phase_2入口停下,"disas"反汇编查看代码:

可以看到,0x400efc-0x400f02首先把“被调用者保存”寄存器压栈,然后开辟了一个28字节的栈,把栈指针拷贝给了%rsi寄存器,再调用了read_six_numbers函数。反汇编"disas read_six_numbers"看下:

0x40145c-0x40147c都是执行的从栈中拷贝内容到寄存器之类的操作,对于突然出现的地址0x4025c3,打印出来看下,是格式字符串,由此判断第二关要输入的内容是6个int类型的数据。重新运行程序,在第二关时随便输入6个数字:1 2 3 4 5 6,并将断点设置在下图红框的位置,并运行到这里。这里往下都是比较的操作,所以在红框位置之上,相应的数据应该都已经入栈或者放在指定的寄存器了。

阅读这段汇编代码要注意的有寄存器寻址和间接寻址的区别, mov和lea的区别。
比如0x400f17 mov -0x4(%rbx), %eax这一句,第一个操作数属于间接寻址,那么执行的翻译成C语句相当于:%eax = *(%rbx-4)。没错是取地址中的值。0x400f25: add $0x4,%rbx这一行是寄存器寻址,翻译成C语句相当于:%rbx=%rbx-4。
mov和lea的区别是,lea不解引用,所以0x400f30: lea 0x4(%rsp),%rbx相当于%rbx = %rsp+4, x400f35 lea 0x18(%rsp), %rbp相当于:%rbp = %rsp+0x18=%rsp+24.
继续回到汇编代码,输入的数字是int型,在64/32机器上都占用4个字节,所以需要的占用栈24字节的空间。很容易联想到栈中存放着的是我们输入的数字,打印出来证实:


首先进行栈顶元素和1的比较,不相等则调用explode_bomb,相等则进行跳转1(红笔标注的数字)【所以输入的第一个数字应该是1】,1箭头末尾的两个lea操作,第一个是让%rbx=%rsp+4,即指向第二个我们输入的数字,第二个是让%rbp=%rsp+24,即指向栈的第六个元素末尾,用于判断循环结束的。
接着进行跳转2,%eax = *(%rbx-4),即%eax=1;接着%eax翻倍等于2,判断%eax是否等于*(%rbx),即*(%rsp+4),不等就调用explode_bomb【所以输入的第二个数字是2】。
接着进行跳转3,%rbx=%rbx+4,注意%rbx目前存放的是指针不是数据,所以%rbx又往栈底移动了一个4个字节,现在指向第3个我们输入的数字了.cmp %rbp, %rbx这是比较%rbx有没有到第六个元素末尾,如果到达,循环就结束了。
接着进行跳转6,%eax = *(%rbx-4),即%eax=2;接着%eax翻倍等于4,判断%eax是否等于*(%rbx),即*(%rsp+8),不等就调用explode_bomb【所以输入的第三个数字是4】。
翻译成C语言类似
int val = 1;
int * end = rsp+6;
do
{
if(*rsp==val)
{
val = *rsp * 2;
rsp++;
}
else{
explode_bomb();
}
}while(rsp!=end);
依次类推,需要我们输入的数字分别是:1 2 4 8 16 32
4.Phase_3攻关


Phase_3和Phase_2类似,也是在一开始开辟了18个字节的栈,分别令%rcx, %rdx指向栈的两个位置,这里并没有从栈顶开始使用。
出现0x4025cf比较突兀,打印出来看是格式化字符串“%d %d”,推断这次要输入的是两个int类型数据。调用sscanf回来后比较返回值是否大于1,大于1则跳到<Phase_3+39>。可以输入两个数字,运行到这里后,打印%rsp+8和%rsp+0xc来确认,这两个位置存放的就是我们输入的第一个和第二个数字。
0x400f6a判断第一个参数是否小于8,如果大于或等于则跳到<phase_3+106>处的explode_bomb处。负数的补码相当于正无穷,也会跳到explode_bomb,由此判断第一个参数参数是[0,7]之间的数。看起来有点像一个switch结构了。
接下来出现的无条件调整语句,jmpq *0x402470(,%rax,8), 也是一个间接跳转,跳转地址是*(0x402470+%rax*8)。

分别打印出%rax的值的不同情况:
%rax--------*(0x402470+%rax*8)----------%跳转地址----------%跳转后的操作
0 0x402470 0x00400f7c %eax=0xcf (d: 207)
1 0x402470+0x8 0x00400fb9 %eax=0x137 (d:311)
2 0x402470+0x10 0x00400f83 %eax=0x2c3 (d:707)
3 0x402470+0x18 0x00400f8a %eax=0x100 (d:256)
4 0x402470+0x20 0x00400f91 %eax=0x185 (d:389)
5 0x402470+0x28 0x00400f98 %eax=0xce (d:206)
6 0x402470+0x30 0x00400f9f %eax=0x2aa (d:682)
7 0x402470+0x38 0x00400fa6 %eax=0x147 (d:327)
所以Phase_3其实等价于一个switch结构:
int args1, args2,val;
scanf("%d %d", &args1, &args2);
switch(args1)
{
case 0: val = 207;break;
case 1: val = 311;break;
case 2: val = 707;break;
case 3: val = 256;break;
case 4: val = 389;break;
case 5: val = 206;break;
case 6: val = 682;break;
case 7: val = 327;break;
default:
explode_bomb();
}
if(args2!=val)
explode_bomb();
所以正确答案有7组:(0,207), (1,311),(2,707),(3,256),(4,389),(5,206),(6,682),(7,327).
至此,Phase_3攻关完毕。后面的关卡未来水平提高了再战。
《CS:APP》二进制炸弹实验(phase_1-3)的更多相关文章
- BinaryBombs(二进制炸弹实验)
实验介绍 使用所学知识拆除Binary Bombs来增强对程序的机器级表示.汇编语言.调试器和逆向工程等理解. Binary Bombs(二进制炸弹)是一个可执行程序,是C语言编译链接成的,包含pha ...
- 深入理解计算机系统 (CS:APP) 缓冲区漏洞实验 – Buffer Lab 解析
原文地址:https://billc.io/2019/05/csapp-cachelab/ 写在前面 这是 CSAPP 官网上的第 4 个实验 buflab,也是学校要求的第三个实验.这个实验比上一个 ...
- 《CSAPP》实验二:二进制炸弹
二进制炸弹是第三章<程序的机器级表示>的配套实验,这章主要介绍了x64汇编,包括:操作数的表示方式,数据传送指令,算术和逻辑指令,控制流跳转指令,过程(procedure)的实现与运行时栈 ...
- CS:APP配套实验 Data Lab
刚刚完成注册博客,想写一篇随笔,方便以后自己回顾.如果恰好也能帮助到你,是我的荣幸. 这次随笔是记载我的计算机系统(CS:APP,Computer Systems:A Programer's Pers ...
- 深入理解计算机系统 (CS:APP) Lab2 - Bomb Lab 解析
原文地址:https://billc.io/2019/04/csapp-bomblab/ 写在前面 CS:APP是这学期的一门硬核课程,应该是目前接触到最底层的课程了.学校的教学也是尝试着尽量和CMU ...
- csapp lab2 bomb 二进制炸弹《深入理解计算机系统》
bomb炸弹实验 首先对bomb这个文件进行反汇编,得到一个1000+的汇编程序,看的头大. phase_1: 0000000000400ef0 <phase_1>: 400ef0: 48 ...
- CSAPP:逆向工程【二进制炸弹】
转载请注明出处:https://www.cnblogs.com/ustca/p/11694127.html 二进制炸弹任务描述 拓展:缓冲区溢出攻击 "二进制炸弹包含若干个阶段,每个阶段需要 ...
- 图文并茂-超详解 CS:APP: Lab3-Attack(附带栈帧分析)
CS:APP:Lab3-ATTACK 0. 环境要求 关于环境已经在lab1里配置过了.lab1的连接如下 实验的下载地址如下 说明文档如下 http://csapp.cs.cmu.edu/3e/at ...
- 深入理解计算机系统 (CS:APP) - 高速缓存实验 Cache Lab 解析
原文地址:https://billc.io/2019/05/csapp-cachelab/ 这个实验是这学期的第四个实验.作为缓存这一章的配套实验,设计得非常精妙.难度上来讲,相比之前的修改现成文件, ...
随机推荐
- JDBC相关知识
一.连接数据库 1. 步骤 //1.创建一个Driver实现类的对象 Driver driver = new com.mysql.jdbc.Driver();//注意抛异常 //2.准备 url 和 ...
- POJ 3041 Asteroids / UESTC 253 Asteroids(二分图最大匹配,最小点匹配)
POJ 3041 Asteroids / UESTC 253 Asteroids(二分图最大匹配,最小点匹配) Description Bessie wants to navigate her spa ...
- Android事件分发理解
Android事件分发机制是个难点和重点,结合下各家,写点自己的理解.. 首先抛出一个小问题,写一个button的点击事件 button.setOnClickListener(new OnClickL ...
- 高效测试用例组织算法pairwise之Python实现
------------------------------------------本文专为<光荣之路培训 >原创,如有转载请注明出处--------------------------- ...
- Es6 新增函数
====函数的扩展 -----ES6 之前,不能直接为函数的参数指定默认值,只能采用变通的方法. function log(x, y) { y = y || 'World'; console.log( ...
- rsync定时同步配置
附上脚本 三大配置文件请看rsync安装与配置 #!/bin/sh #linuxsir.org home backup #/usr/bin/rsync -avzP --password-file=/e ...
- 2017年1月1日 App Store中的所有应用都必须启用 App Transport Security安全功能
2017年1月1日 App Store中的所有应用都必须启用 App Transport Security安全功能,否则极有可能被拒! 在WWDC 2016开发者大会上,苹果宣布了一个最后期限:到20 ...
- linux下 git 安装
1.使用yum安装 yum -y install git yum remove git 2.源代码安装 a.下载git源码 网址为 https://github.com/git/git/releas ...
- (转)java中对集合对象list的几种循环访问总结
Java集合的Stack.Queue.Map的遍历 在集合操作中,常常离不开对集合的遍历,对集合遍历一般来说一个foreach就搞定了,但是,对于Stack.Queue.Map类型的遍历,还是有一 ...
- (转)ZXing生成二维码和带logo的二维码,模仿微信生成二维码效果
场景:移动支付需要对二维码的生成与部署有所了解,掌握目前主流的二维码生成技术. 1 ZXing 生成二维码 首先说下,QRCode是日本人开发的,ZXing是google开发,barcode4j也是老 ...