Not really my forte, but here we go, I can only get better.
If we disassemble, the solution is pretty clear.
gets()
is used to take in input, then the contents of another local variable are compared to 0xdeadbeef
. Basic buffer overflow then overwrite a local variable:
X-MAS{ah_yes__i_d0_rememb3r_you}
Once we visit the URL, we are shown some code:
Clearly this is some type of Type Juggling exploit, but I'm not that familiar with it except for 0e
md5 hashes and stuff. However, there are some restrictions here:
There can be no e
character in either parameter
The two parameters must be the same length
They can't strictly equal each other (!==
) but they must loosely equal each other (==
)
PHP comparision is a known piece of junk, so we can find some weaknesses using PayloadsAllTheThings.
Once set of possible parameters is 01
and 1
, as they are both two characters long and - according to PHP's loose comparison - equal each other (thanks to nrabulinski for this solution after the CTF). It appears that objetcs are automatically converted to numbers for loose comparisions, as loose only compares values while strict also compares types. Therefore the example above would both equal 1
under loose comparison.
Another, more interesting set is 200
and 2E3
(thanks to 03sunf). Note that 2E3
is an exponential, equivalent to 2 * 10^2
. Once both are converted to integers, they pass the check.
We receive a file called chall
. NX is disabled, which is helpful. We inject shellcode, use a jmp rsp
gadget and execute our own shellcode.
main()
is a fairly simple binary:
The buffer is 48
bytes long. After the buffer there is 16-bit integer check
, which acts as a canary. Then there are 8 bytes for the stored RBP. The total input it 71
, meaning after the stored RBP we have 13 bytes of overflow, including the RIP. No ROP is possible.
Note that the value -6913
is actually 0xe4ff
.
This was rather misleading as they gave you the LIBC.
Firstly:
Now we need some shellcode. pwntools' shellcraft.sh()
is 2
bytes too long, so we'll have to make it manually.
The general payload is as follows:
/bin/sh\x00
so we have it in a known location (relative to RSP)
Shellcode
Padding
0xe4ff
to overwrite the pseudo-canary
Padding
jmp rsp
Now we need to decide what shellcode we want to run. Well, since RSP points at the stack, we know that it will always be a static offset off our buffer. If we calculate it, we can just do
And execute the other half of our code! And at this point RSP will be exactly 8
bytes off /bin/sh\x00
, so we can use it to populate RDI as well!
X-MAS{sant4_w1ll_f0rg1ve_y0u_th1s_y3ar}
I didn't manage to solve a huge number of these - I was quite busy, plus I suck - but I'll dump some writeups here for those I caught up on later.