콘텐츠로 건너뛰기

dungeon-in-1983

Description

해당 문제는 Dreamhack CTF Season 5 Round #8 (🌱Div2) 에 출제된 문제입니다.

Back in 1983, when you only had A and B on your controller…

FYI

Flag Format: DH{...}
Timeout: 5s

Decompiled-src

main

__int64 __fastcall main(__int64 a1, char **a2, char **a3)
{
  int i; // [rsp+14h] [rbp-1BCh]
  char *lineptr; // [rsp+18h] [rbp-1B8h] BYREF
  __int64 ptr; // [rsp+20h] [rbp-1B0h] BYREF
  FILE *stream; // [rsp+28h] [rbp-1A8h]
  __int64 v8[2]; // [rsp+30h] [rbp-1A0h] BYREF
  int v9; // [rsp+40h] [rbp-190h]
  __int64 v10; // [rsp+44h] [rbp-18Ch]
  __int64 v11; // [rsp+4Ch] [rbp-184h]
  int v12; // [rsp+54h] [rbp-17Ch]
  __int64 v13; // [rsp+58h] [rbp-178h]
  __int64 v14; // [rsp+60h] [rbp-170h]
  int v15; // [rsp+68h] [rbp-168h]
  __int64 v16; // [rsp+6Ch] [rbp-164h]
  __int64 v17; // [rsp+74h] [rbp-15Ch]
  int v18; // [rsp+7Ch] [rbp-154h]
  __int64 v19; // [rsp+80h] [rbp-150h]
  __int64 v20; // [rsp+88h] [rbp-148h]
  int v21; // [rsp+90h] [rbp-140h]
  __int64 v22; // [rsp+94h] [rbp-13Ch]
  __int64 v23; // [rsp+9Ch] [rbp-134h]
  int v24; // [rsp+A4h] [rbp-12Ch]
  __int64 v25; // [rsp+A8h] [rbp-128h]
  __int64 v26; // [rsp+B0h] [rbp-120h]
  int v27; // [rsp+B8h] [rbp-118h]
  __int64 v28; // [rsp+BCh] [rbp-114h]
  __int64 v29; // [rsp+C4h] [rbp-10Ch]
  int v30; // [rsp+CCh] [rbp-104h]
  __int64 v31; // [rsp+D0h] [rbp-100h]
  __int64 v32; // [rsp+D8h] [rbp-F8h]
  int v33; // [rsp+E0h] [rbp-F0h]
  __int64 v34; // [rsp+E4h] [rbp-ECh]
  __int64 v35; // [rsp+ECh] [rbp-E4h]
  int v36; // [rsp+F4h] [rbp-DCh]
  char s[200]; // [rsp+100h] [rbp-D0h] BYREF
  unsigned __int64 v38; // [rsp+1C8h] [rbp-8h]

  v38 = __readfsqword(0x28u);
  v8[0] = 'ksilisaB';
  v8[1] = 0LL;
  v9 = 0;
  v10 = 'aremihC';
  v11 = 0LL;
  v12 = 0;
  v13 = 'nekarK';
  v14 = 0LL;
  v15 = 0;
  v16 = 'nogroG';
  v17 = 0LL;
  v18 = 0;
  v19 = 'ogidneW';
  v20 = 0LL;
  v21 = 0;
  v22 = 'ruatoniM';
  v23 = 0LL;
  v24 = 0;
  v25 = 'ahtaiveL';
  v26 = 110LL;
  v27 = 0;
  v28 = 'ardyH';
  v29 = 0LL;
  v30 = 0;
  v31 = 'rocitnaM';
  v32 = 101LL;
  v33 = 0;
  v34 = 'coR';
  v35 = 0LL;
  v36 = 0;
  ((void (__fastcall *)(__int64))sub_130A)(a1);
  stream = fopen("/dev/urandom", "r");
  puts("Welcome to the Dungeon!");
  puts("You have only two buttons: A and B.");
  puts("Each monster requires certain series of key combinations to be defeated, so be careful!");
  for ( i = 0; i <= 9; ++i )
  {
    fread(&ptr, 8uLL, 1uLL, stream);
    printf("[STAGE %2d]: %s\n", (unsigned int)(i + 1), (const char *)v8 + 20 * i);
    sub_138D(ptr);
    printf("Cast your spell!: ");
    fgets(s, 200, stdin);
    if ( s[strlen(s) - 1] == '\n' )
      s[strlen(s) - 1] = 0;
    if ( !sub_1407((__int64)s, ptr) )
    {
      puts("You were defeated. Retreat!");
      exit(-1);
    }
    printf("%s defeated. STAGE %2d cleared!\n", (const char *)v8 + 20 * i, (unsigned int)(i + 1));
  }
  fclose(stream);
  printf("It's dangerous to go alone! Take the flag: ");
  lineptr = 0LL;
  ptr = 0LL;
  stream = fopen("./flag", "r");
  getline(&lineptr, (size_t *)&ptr, stream);
  printf("%s", lineptr);
  free(lineptr);
  fclose(stream);
  return 0LL;
}

sub_138D

int __fastcall sub_138D(__int64 a1)
{
  return printf(
           "[INFO] HP: %5hu, STR: %5hhu, AGI: %5hhu, VIT: %5hhu, INT: %5hhu, END: %5hhu, DEX: %5hhu\n",
           HIWORD(a1),
           (unsigned __int8)a1,
           BYTE1(a1),
           BYTE2(a1),
           BYTE3(a1),
           BYTE4(a1),
           BYTE5(a1));
}

Solution

from pwn import *
#context.log_level = 'debug'
context(arch='amd64',os='linux')
warnings.filterwarnings('ignore')

p = remote('host3.dreamhack.games', 14036)
#p = process('./prob')

for i in range(10):
    p.recvuntil("[INFO]")
    info = p.recvline()
    _hp = info.split(b":")[1].split(b",")[0].strip()
    _str = info.split(b":")[2].split(b",")[0].strip()
    _agi = info.split(b":")[3].split(b",")[0].strip()
    _vit = info.split(b":")[4].split(b",")[0].strip()
    _int = info.split(b":")[5].split(b",")[0].strip()
    _end = info.split(b":")[6].split(b",")[0].strip()
    _dex = info.split(b":")[7].split(b"\n")[0].strip()

    _hp =  "0x{:02x}".format(int(_hp))
    _str = "0x{:02x}".format(int(_str))
    _agi = "0x{:02x}".format(int(_agi))
    _vit = "0x{:02x}".format(int(_vit))
    _int = "0x{:02x}".format(int(_int))
    _end = "0x{:02x}".format(int(_end))
    _dex = "0x{:02x}".format(int(_dex))

    #print(f"_hp: {_hp}, _str: {_str}, _agi: {_agi}, _vit: {_vit}, _int: {_int}, _end: {_end}, _dex: {_dex}")

    random_val = ''
    random_val += _hp
    random_val += _dex 
    random_val += _end
    random_val += _int
    random_val += _vit
    random_val += _agi 
    random_val += _str
    random_val = random_val.replace("0x", "")
    random_val = int(random_val, 16)

    print(f"random_val: {hex(random_val)}")

    val = random_val
    _pattern = ''

    while True:
        if val == 1:
            _pattern = "A" + _pattern
            break
        if val % 2 == 1:
            val = val - 1
            _pattern = "A" + _pattern
            continue
        val = val // 2
        _pattern = "B" + _pattern

    #print(_pattern)
    p.sendlineafter("Cast your spell!: ", _pattern)

p.interactive()

Result

ubuntu@wh1te4ever-main:~/Desktop/dreamhack-CTF/dungeon-in-1983$ python3 solve.py
[+] Opening connection to host3.dreamhack.games on port 14036: Done
random_val: 0xa4061985c50cfe3e
random_val: 0x1ba3a5802a7ed17
random_val: 0xf2e1d9d858b41b16
random_val: 0xab093dd1fe9ac28b
random_val: 0x140d91e3641adec
random_val: 0xead8e07deee1753a
random_val: 0x1e15ead778b4c55d
random_val: 0x42615069ca925013
random_val: 0xfa93152a956ff2a4
random_val: 0xaa212fbb6a04e4fa
[*] Switching to interactive mode
Roc defeated. STAGE 10 cleared!
It's dangerous to go alone! Take the flag: DH{ddfd394275e80d059f04cee16667ad40e81820a0b105e2f80f0d17544a274a62}
[*] Got EOF while reading in interactive
$ 
[*] Interrupted
[*] Closed connection to host3.dreamhack.games port 14036
태그:

답글 남기기