Do You Even Bof?
Chall1.png
Surprising that the first challenge in a bof writeup is not even a bof. I’ll keep this one short. Main pointers to remember:
- The program compares some value with a constant 0x539 (1337)
- It uses puts
- Strings in the programs: “Bye”, “Your so leet!”, “%d”
- jne -> means ==
- scanf for input
1 | #include <stdio.h> |
The Process
Since I won’t talk about what commands I executed unless it is important, here’s the commands I use for examining the binaries.
- The classic
file
- Bit less classic, but amazing
rabin2 -z
strings
orrabin2 -zz
if I feel like itrabin2 -s
, mainly because I know there’s gonna be a win function.checksec
to identify the canariescutter
, orradare2
for disassembly. (Mainlycutter
, cause TUI is hard)- ???
cat flag
Jump
Very simple buffer overflow:
- Fill the buffer with padding (64 bytes from the disassembly)
- The program spits out a 32 bit address, and since the return address lies right after the buffer (because there are no parameters), all you have to do is encode the bytes properly.
1 | #!/usr/bin/python3 |
Blind
Same as jump (as the program says), but you don’t know where the win function is.
Now that I have done jump though, this is fairly trivial, as all I have to do is objdump -d blind | grep win
and I would have the address to put it. So same as before, fill the buffer with padding, then properly encode the address, and boom. Bof.
Note: The size of the buffer is very easy to find, but there might be a few cases:
- Look at the offset from
ebp
that is loaded into the buffer, right before thesetbuf
function or the vulnerablegets
- Cutter would list all the variables defined in the vuln function, with offsets: egWhich means you can test 0x44 as the offset to your return address. (The buffer size is still 0x40). With the 32 bit int and the base pointer (presumably) stored on the stack you end up with a padding of 0x48 or 72 bytes.
1
2; var char *s @ ebp-0x44
; var int32_t var_4h @ ebp-0x4
1 | #!/usr/bin/python3 |
Bestsecurity
Ah yes. Fake canaries.
- While I was told that this would be a canary challenge,
checksec
did not say the same thing. Somehow I believechecksec
more. - A simple scan of the disassembly told me that all I had to do was overwrite an integer with the value 1234, which it was comparing with.
- While we did not need to override the return address, if you could, it would be the same as the first one. Except this one would be a ret2libc, as the stack is non-executable.
1 | #!/usr/bin/python3 |
Stack-dump
This one was considerably more difficult, compared to the first 3. It was both the fact that there was an actual stack canary which I had never dealt with before, and because there was so much more stuff to do!
checksec
: Yes there is a canary- After spending more-than-very-little time (a whole day) on figuring out what the binary did, I came up with a game plan. Leak the canary using the fread and then overwrite using the fwrite. I used a combination of cutter, pwndbg and my rapidly depleting sanity to come up with this. Tbf, this is more a test of my patience at this point.
- To leak the canary, you use the “useful stack pointer” which is somehow 1 byte away from the actual useful pointer, to read 22 bytes (0x16) from the place where the canary is located. Which is 0x69 bytes from the pointer provided. The offset can be figured out by a combination of looking at the disassembly and dumping the whole stack in pwndbg. You only need to read 4 bytes, but fread call reads in 22 bytes, so why not.
- By writing the useful pointer to the first 4 bytes of the buffer, and then inspecting the memory at that location, you can get the canary.
- All that remains is figuring out the paddings. You can find the size of the padding in the disassembly, which is 96 bytes (0x60) before the canary (0x30 buffer + 0x30 other variables). The other 8 bytes padding after the canary are a bit hard to find. While there should be a saved version of the stack pointer and the base pointer, why are there 4 null bytes in there?? After a bit of trial and error however, and checking out the disassembly, I realized it was ebp and ebx that was saved. Then I got it to write the correct return address. (which can be found in the disassembly)
- Very important step. Quit the program.
Some amazing pwndbg commands I used
x/w <address>
: Basically allows me to dump the whole stack from any place I wantinfo registers
break
andcontinue
andni
attach
: attaching to pwntools scriptdisassemble
: disassembling on the fly
Interesting points to note/Common pitfalls
- There is a dummy (??) gets call in there. When you put in the amount of characters to write to the buffer, you can give it large values. Perhaps that could be used to overwrite something?
- The overflow is based on the fact that while
fread
andfwrite
do write a specific number of bytesinto the bufferonto the stack, giving control of the number of bytes to write to the user is the same as not asking for them. - I was stuck for a long time because I did not know about the
x/w
command in gdb. Using the disassembly to find where the canary is located and dumping it atleast in gdb is very important. I thought my canary was correct for a while before I figured this out. - The significance of the last step in the process. While the first 5 steps overwrite the buffer safely, only after performing step 6 is a ret instruction executed. So the code flow would not change if that is not done. I was stuck on it for a while because I did not realize this.
1 | #!/usr/bin/python3 |
Conclusion
Bof hard.
Very leet indeed.
The fact that I enjoyed it anyway makes me worried about my future.