본문 바로가기
CTF

[CTF] ASIS CTF 2021 writeup

by snwo 2021. 11. 29.

beanstalk

먼저, license key 를 입력받고, 그 key 를 기반으로 어떤 값들을 생성하고, 그 값들중 어느 값만 추출해서 license key 를 비교한다.

맞으면, FLAG 를 입력받고, 어떤값들을 이용해 암호화한 뒤, 비교한다.

first step

license_to_key()

license 키 형식이 BEAN-XXXX-XXXXXXXXXX-XXXXXX 인지 검사하고

std::hex 를 이용해 하위 바이트부터 first_enc 에 1바이트씩 채운다.

ex : 1234-1234567890-123456 → 0x56341290785634123412 (메모리에는 0x12부터 차례대로 저장)

second step

Beanstalk::Beanstalk()

void __fastcall Beanstalk::Beanstalk(struct_this *this, const unsigned __int8 *license_key) {   _QWORD *v2; // rbx   __int64 v3; // rax   int i; // [rsp+18h] [rbp-18h]   int j; // [rsp+1Ch] [rbp-14h]    this->bufs = operator new[](0x50uLL);   for ( i = 0; i <= 9; ++i )   {     v2 = &this->bufs[i];     *v2 = operator new[](0x100uLL);     for ( j = 0; j <= 255; ++j )     {       *(j + this->bufs[i]) = *(t() + (j ^ license_key[i]));     }     this->license[i] = *(this->bufs[i] + 0x77LL);   }

this→bufs 에 배열포인터를 할당한 뒤,

this→bufs[i] 에 0x100 크기의 배열을 할당한다.

this→bufs[i][j] 에는 t()[j^license_Key[i]] 값들을 차례대로 채운다.

t() 함수는 k() 함수와 다르게 인자를 받지않고, opcode 의 주소를 반환해서

j^license_key[i] 를 인덱스로해서 값을 가져온다.

다행히도, t() 함수에서 리턴하는 주소에 있는 opcode 들은 값이 중복되지 않아서 역연산이 가능하다.

이렇게 생성된 bufs 배열 중, 각 인덱스에 0x77 에 있는 값을 license 로 가져온다.

이 부분을 다시보자. Beanstalk 으로 license 값을 이용해 bufs 배열의 값들을 채워놓은 뒤,

각 인덱스의 0x77 에 있는 것을 this→license 배열에 넣고,

k() 함수로 opcode 10바이트의 위치를 리턴해 this→license 와 비교한다.

int *k(int x);

k(0) ~ k(1) 값과 생성된 license 값과 비교하는데,

이 함수가 좀 변태같다.

내부에 있는 sub_2605 를 호출하면, ret address 가 mov edi,0A5A~~ 부분을 가리키고 있는데,

그거를 rax 에 pop 한 뒤,

인자 x 를 0으로 주면, mov edi,0A5A~~ 부분을,

인자 x 를 1으로 주면, mov edi,0A5A~~ + 0xa 부분을 리턴한다.

그렇다는 것은, mov edi,0A5A~~ 부터의 opcode 들을 차례대로 가져와서 비교한다는 것 이다.

list(ida_bytes.get_bytes(here(),0xA))

위 명령어로 저 위치에 있는 opcode 를 가져올 수 있다.

license=[191, 11, 15, 162, 165, 148, 14, 215, 203, 133]

license[i] = t()[0x77^original_license[i]]

복호화 → t().index(license[i])^0x77 = original_license[i]

#BEAN-2A21-B91DC84834-24E676 for i in range(len(key)):         original_license.append(t.index(license[i])^0x77)

third step

std::__cxx11::basic_string<char,std::char_traits<char>,std::allocator<char>>::basic_string(MyKey, v7, v9);       std::operator<<<std::char_traits<char>>(&std::cout, "Flag: ");       std::operator>><char>(&std::cin, MyKey);       if ( std::__cxx11::basic_string<char,std::char_traits<char>,std::allocator<char>>::size(MyKey) <= 0x2F )       {         std::__cxx11::basic_string<char,std::char_traits<char>,std::allocator<char>>::size(MyKey);         if ( std::__cxx11::basic_string<char,std::char_traits<char>,std::allocator<char>>::size(MyKey) )         {           for ( i = 0; i <= 7; ++i )             flags[i] = *std::__cxx11::basic_string<char,std::char_traits<char>,std::allocator<char>>::at(MyKey, i);           Beanstalk::encrypt(this, flags, v19);           f();                         // flags[i] 에 MyKey[i+j] 차례대로 대입 (8바이트)                         // f()[i+j] 와 flags[i] 비교 (8바이트)                         // 틀리면 중간에 종료하고, 다 맞으면 j 에 8을 더하고,                          // for문 위에 if문으로 점프          v12 = std::operator<<<std::char_traits<char>>(&std::cout, "[+] Correct!");         std::ostream::operator<<(v12, &std::endl<char,std::char_traits<char>>);         v5 = 0;       }

알맞은 license key 를 입력하면 이 부분으로 온다.

여기서 디컴파일이 이상하게돼서 처음에는 플래그 8자리밖에 못 얻었는데,

이중 for문이였다.

f() 함수도 마찬가지로 함수 내의 opcode 주소를 리턴하는데 따로 설명은 안하겠습니다.

fourth step

unsigned __int8 *__fastcall Beanstalk::encrypt(struct_this *this, const unsigned __int8 *a2, unsigned __int8 *a3) {   unsigned __int8 *result; // rax   int i; // [rsp+28h] [rbp-8h]   int j; // [rsp+2Ch] [rbp-4h]    for ( i = 0; i <= 3; ++i )     *&this->flags[2 * i] = (a2[2 * i] << 8) | a2[2 * i + 1];// 6 7 4 5 2 3 0 1   srand((*(this->bufs[8] + 0xALL) << 16) | (*(this->bufs[5] + 14LL) << 8) | *(this->bufs[1] + 14LL) | (*(this->bufs[9] + 31LL) << 24));   Beanstalk::g0(this, 3);   Beanstalk::z(this, 2, 3);   Beanstalk::g1(this, 2);   Beanstalk::z(this, 1, 2);  -------------------------------------------------    Beanstalk::z(this, 3, 2);   Beanstalk::g4(this, 2);   Beanstalk::z(this, 2, 1);   Beanstalk::g0(this, 1);   Beanstalk::z(this, 1, 0);   Beanstalk::g1(this, 0);   for ( j = 0; j <= 3; ++j )   {     a3[2 * j] = HIBYTE(*&this->flags[2 * j]);     result = &a3[2 * j + 1];     *result = *&this->flags[2 * j];   }   return result; }

처음에 딱 보고 멘붕이왔는데, 그냥 비트연산을 조금 많이하는 것 뿐이였다.

this→flags 배열에 내가 입력한 flag 를 2바이트씩 넣는데, 이때 순서를 바꿔서 넣는다.

ex : 0x3412 → 0x1234

그리고, srand 함수를 호출하는데, bufs 배열에 있는 값을 이용한다.

이미 알맞은 license key 를 구했으니, seed 값을 알 수 있다.

sr=(bufs[8][0xa]<<16)|(bufs[5][14] <<8) |(bufs[1][14]) |(bufs[9][31] << 24)

그 후, g0~g4, z 함수를 차례대로 호출하고, 다시 순서를 바꿔준다.

z(struct_this *this, int a1, int a2)

this→flags[2*n1] ^= this→flags[2*a2] ^ rand()

rand() 함수 리턴값을 구할 수 있으니, 그대로 다시 실행시키면 된다.

복호화 → this->flags[2*n1] ^= this->flags[2*a2] ^ rand()

근데, flag 를 뒤에서부터 복호화해야되기 때문에,

rand 값을 z 함수가 호출된 수 만큼 리스트에 넣어서, 뒤에서부터 rand 값을 사용해야한다.

(g 시리즈 함수에서는 rand 함수 사용안됨)

def z(n1,n2):         global flags         global c         global rands         flags[n1]^=flags[n2]^rands[c]         c-=1

이런식으로 구현했다.

g 시리즈 함수

g0 → g(this,n,0,1,2,3)

g1 → g(this,n,4,5,6,7)

g2 → g(this,n,8,9,0,1)

g3 → g(this,n,2,3,4,5)

g4 → g(this,n,6,7,8,9)

다 똑같이 g 함수를 호출하는데, 나머지 4개 인자가 다르다.

g(n,l0,l1,l2,l3)

struct_this *__fastcall Beanstalk::g(struct_this *this, int n1, int l0, int l1, int l2, int l3) {                                                this->flags[2 * n1] ^= *(this->bufs[l3] + this->flags[2 * n1]) << 8;   this->flags[2 * n1] ^= *(this->bufs[l2] + (this->flags[2 * n1] >> 8));   this->flags[2 * n1] ^= *(this->bufs[l1] + this->flags[2 * n1]) << 8;   this->flags[2 * n1] ^= *(this->bufs[l0] + (this->flags[2 * n1] >> 8)); }

언뜻 보면 복호화되지 않을 것 같았다.

하지만 2바이트씩 word 로 참조한다는 사실을 알면 복호화 하기 쉽다.

마지막 구문부터 보자.

this->flags[2 * n1] ^= this->bufs[l0][this->flags[2 * n1] >> 8];

flags[2*n1] 의 상위바이트를 인덱스로 해서 flags[2*n1] 와 xor 한다.

이 때, 상위바이트는 변형되지 않으므로, xor 하는 값을 알 수 있다.

그래서 그냥 한번 더 실행하면 복호화된다. ^^

그래서, encrypt 함수에서 호출하는 g시리즈 함수, z 함수에 대해 복호화함수를 만든 뒤,

뒤에서부터 호출하면 플래그가 복호화된다.

뭔가 걸리는게 있다면 rand 값인데,

encrypt 함수 내부에 srand 가 있기 때문에 8바이트씩 암호화 할 때 마다

srand 로 초기화해주기 때문에, c 변수를 31로 초기화 하면서 사용하면 된다.

full code

#!/usr/bin/python
from ctypes import *
rands=[]
r=CDLL("/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.28")
#g0 0 1 2 3
#g1 4 5 6 7
#g1 8 9 0 1
#g3 2 3 4 5
#g4 6 7 8 9
def g0(n1):
        global flags
        flags[n1]^=bufs[0][flags[n1]>>8]
        flags[n1]^=bufs[1][flags[n1]&0xff]<<8
        flags[n1]^=bufs[2][flags[n1]>>8]
        flags[n1]^=bufs[3][flags[n1]&0xff]<<8
def g1(n1):
        global flags
        flags[n1]^=bufs[4][flags[n1]>>8]
        flags[n1]^=bufs[5][flags[n1]&0xff]<<8
        flags[n1]^=bufs[6][flags[n1]>>8]
        flags[n1]^=bufs[7][flags[n1]&0xff]<<8
def g2(n1):
        global flags
        flags[n1]^=bufs[8][flags[n1]>>8]
        flags[n1]^=bufs[9][flags[n1]&0xff]<<8
        flags[n1]^=bufs[0][flags[n1]>>8]
        flags[n1]^=bufs[1][flags[n1]&0xff]<<8
def g3(n1):
        global flags
        flags[n1]^=bufs[2][flags[n1]>>8]
        flags[n1]^=bufs[3][flags[n1]&0xff]<<8
        flags[n1]^=bufs[4][flags[n1]>>8]
        flags[n1]^=bufs[5][flags[n1]&0xff]<<8
def g4(n1):
        global flags
        flags[n1]^=bufs[6][flags[n1]>>8]
        flags[n1]^=bufs[7][flags[n1]&0xff]<<8
        flags[n1]^=bufs[8][flags[n1]>>8]
        flags[n1]^=bufs[9][flags[n1]&0xff]<<8

def z(n1,n2):
        global flags
        global c
        global rands
        flags[n1]^=flags[n2]^rands[c]
        c-=1
t=[163, 215, 9, 131, 248, 72, 246, 244, 179, 33, 21, 120, 153, 177, 175, 249, 231, 45, 77, 138, 206, 76, 202, 46, 82, 149, 217, 30, 78, 56, 68, 40, 10, 223, 2, 160, 23, 241, 96, 104, 18, 183, 122, 195, 233, 250, 61, 83, 150, 132, 107, 186, 242, 99, 154, 25, 124, 174, 229, 245, 247, 22, 106, 162, 57, 182, 123, 15, 193, 147, 129, 27, 238, 180, 26, 234, 208, 145, 47, 184, 85, 185, 218, 133, 63, 65, 191, 224, 90, 88, 128, 95, 102, 11, 216, 144, 53, 213, 192, 167, 51, 6, 101, 105, 69, 0, 148, 86, 109, 152, 155, 118, 151, 252, 178, 194, 176, 254, 219, 32, 225, 235, 214, 228, 221, 71, 74, 29, 66, 237, 158, 110, 73, 60, 205, 67, 39, 210, 7, 212, 222, 199, 103, 24, 137, 203, 48, 31, 141, 198, 143, 170, 200, 116, 220, 201, 93, 92, 49, 164, 112, 136, 97, 44, 159, 13, 43, 135, 80, 130, 84, 100, 38, 125, 3, 64, 52, 75, 28, 115, 209, 196, 253, 59, 204, 251, 127, 171, 230, 62, 91, 165, 173, 4, 35, 156, 20, 81, 34, 240, 41, 121, 113, 126, 255, 140, 14, 226, 12, 239, 188, 114, 117, 111, 55, 161, 236, 211, 142, 98, 139, 134, 16, 232, 8, 119, 17, 190, 146, 79, 36, 197, 50, 54, 157, 207, 243, 166, 187, 172, 94, 108, 169, 19, 87, 37, 181, 227, 189, 168, 58, 1, 5, 89, 42,70]
key=[191, 11, 15, 162, 165, 148, 14, 215, 203, 133]
lic=[]
bufs=[]

#BEAN-2A21-B91DC84834-24E676
for i in range(len(key)):
        lic.append(t.index(key[i])^0x77)

for i in range(len(lic)):
        temp=[]
        for j in range(0x100):
                temp.append(t[lic[i]^j])
        bufs.append(temp)

sr=(bufs[8][0xa]<<16)|(bufs[5][14] <<8) |(bufs[1][14]) |(bufs[9][31] << 24)
r.srand(sr)
c=0
for i in range(32):
        rands.append(r.rand()&0xffff)
        c+=1
c-=1

flags=[70, 157, 250, 50, 81, 226, 101, 244]
flags+=[128, 198, 190, 179, 198, 110, 126, 60]
flags+=[101, 193, 53, 224, 17, 25, 13, 134]
flags+=[46, 147, 254, 234, 214, 103, 215, 177]
flags+=[205, 236, 82, 228, 83, 62, 59, 225]
flags+=[10, 253, 80, 126, 180, 248, 208, 67]
flags+=[70, 157, 250, 50, 81, 226, 101, 244]

def dec():
        count=0
        for i in range(0,8,2):
                flags[count]=(flags[i+1]<<8) | flags[i]
                count+=1
        for i in range(4):
                flags[i]=(flags[i]>>8)|((flags[i]&0xff)<<8)
        g1(0);
        z(1, 0);
        g0(1);
        z(2, 1);
        g4(2);
        z(3, 2);
        g3(3);
        z(0, 3);
        g2(0);
        z(1, 0);
        g1(1);
        z(2, 1);
        g0(2);
        z(3, 2);
        g4(3);
        z(0, 3);
        z(3, 0);
        g3(0);
        z(0, 1);
        g2(1);
        z(1, 2);
        g1(2);
        z(2, 3);
        g0(3);
        z(3, 0);
        g4(0);
        z(0, 1);
        g3(1);
        z(1, 2);
        g2(2);
        z(2, 3);
        g1(3);
        g0(0);
        z(1, 0);
        g4(1);
        z(2, 1);
        g3(2);
        z(3, 2);
        g2(3);
        z(0, 3);
        g1(0);
        z(1, 0);
        g0(1);
        z(2, 1);
        g4(2);
        z(3, 2);
        g3(3);
        z(0, 3);
        z(3, 0);
        g2(0);
        z(0, 1);
        g1(1);
        z(1, 2);
        g0(2);
        z(2, 3);
        g4(3);
        z(3, 0);
        g3(0);
        z(0, 1);
        g2(1);
        z(1, 2);
        g1(2);
        z(2, 3);
        g0(3);
        for i in range(4):
                flags[i]=(flags[i]>>8)|((flags[i]&0xff)<<8)
for k in range(6):
        c=31
        dec()
        for i in range(4):
                print(chr(flags[i]&0xff)+chr(flags[i]>>8),end='')
        flags=flags[8:]
print("")

Uploaded by Notion2Tistory v1.1.0