Table of Content
Intro
Arab Security Conference was held this year from 5th till 7th of September and as always the last day was for the cyber wargames competition where ten qualified teams compete with each other and solve sophisticated challenges in 5 different categories in information security. In this write-up, we’re going to discuss and solve the 4 crypto challenges I created in the ASC wargames finals this year 2021 with my bro X-Vector. The challenges was a bit heavy and full of tricks for those who aren’t up-todate with the latest crypto systems and techniques. However, I’ll show you my thought process when I face a tough challenge and how to divide it into sub-challenges in order to find a correct solution.
:DISCLAIMER:
You’re not gonna find here all the tips and tricks in crypto since I wanted to leave some space for you to think and ask. So, feel free to think more about the challenge before jumping into the solution.
Black Box
Description:
A plane was attacked by a highly skilled hacker and in order to control it, he has to break the black box encryption. Can you help him to decrypt the commands and control the plane?
The challenge’s name and describtion seems to have a good hint
about what type of attack we’re gonna do on this server. When
connecting to the
ip/port
it gives us the following
Analysis
The server uses AES CBC and of course from its name, we’ll assume it’s a padding oracle attack on AES. In order to test our assumtion, we should modify the last byte of ciphertext and pass it to the oracle and it gives us Padding Error! also after modifying the first byte and pass it to the oracle it given us MAC Error!, now we’re 100% sure it’s a padding oracle attack. The CBC uses PKCS7 scheme in order to pad the messages before the encryption. The given cipherhex contains 80 bytes which means we have to deal with 5 blocks of ciphertext 16 bytes each.
Solution
For those who unaware of this type of attack, I found this great article explaining the attack in detail.
The decryption process goes as following:
\[P_k = D(k,C_k) \oplus C_{k-1} \\ P_{k-1} = D(k,C_{k-1}) \oplus C_{k-2} \\ ... \\ P_1 = D(k,C_1) \oplus iv\]
Briefly, if we have just one block beside the
iv
, we
have to start from the previous block (or
iv
) in
our case we don’t know the
iv
, so
we shall start attacking each block based on the previous block.
To do so, we’ll start by the last or 5th block to
determine the padding size and the following steps will be
repeated for each block:
- Pick a random byte \(B\) from \([0,255]\) and replace it with the last byte of \(C_k\).
- If the decryptoin of \(C'_{k-1} \cdot C_k\) succeeds e.g.
- Rearraging the above equation, we shall get the plaintext byte
where \(i = 1,..,15\).
This attack is pretty straightforward, but the implementation may be a little confusing due to multiple runs for trial and error.
And we’ll get the flag:
ASCWG{0r4cl3_0r4cl3_0N_A3S_L3f7_R19H7_U9_D0wN!!}
Time To Guess
This challenge has no describtion and has 2 pitfalls. The first one is the intended solution and the other one for those who focus on details. It also gives us the source code and a server to connect. When connecting to the server it gives the following.
Let’s look at the source code:
Analysis
After analyzing the code, the server wants from us to sumbit the
next 1000 bits after it prints
N
and
v
based on some sort of function called
BBS
and the next 128 bits are the secret key for the AES encryption
of the flag. Looking at
BBS
,
it simply do the following:
- Generate 512 bit random integer as a seed and check it has no divisors with \(N\).
- Calculate \(X = seed^2 \pmod N\).
- Loop forever and for each iteration return \(X^2 \pmod N\) and its least significant bit (e.g. right-most bit).
Rearranging the equations gives:
\[X_{i+1} = X_i^{2^i} \pmod N : \ X_0 = seed\]
Surprisingly, this algorithm is called
Blum Blum Shub, a pseudorandom number generator based on hardness of
factorization of the modulus
N
and
our goal is to find the
seed
in order to crack the
BBS
.
1st Solution
Connecting to the server multiple times gives us the same
N
but
different
v
, so
let’s try to factor
N
since it’s only 271 bits.
Factordb couldn’t make it,
let’s go to
Alpertron. At
first, there was no hope, but after awhile it seems to be able
to factor it in almost 20 min.
N = 46404135538347313105324845256452679880827 ×
62935919582590588934355071817888577583483
Now, we know that \(v = seed ^ {2^2} \pmod N\), so we want to find the 4th root module \(N\) which leads to a quadratic residue problem and hopefully it’s easy to compute after we found the primes of the modulus \(N\).
So, the whole attack goes as following:
- Find \(x, y\) such that \(v = x ^ 4 \pmod p\) and \(v = y ^ 4 \pmod q\) where \(p, q\) are the only prime factors of \(N\).
- Use The Chinees Remainder Theorem to get a unique solution \(seed \pmod {N=pq}\) for the following equations:
2nd Solution (Unintended)
If you noriced that
v
is
the first value that
BBS
generates which means \(v = X^2 \pmod N\), however if you square
it again you’ll get the second value of the function and of
course its right-most bit and so on.
Therefore, the next random bits can be obtained by squaring
v
1000
times each of which contains the bit you wanted in order to send
it to the server.
And it gives us the flag:
ASCWG{7h3_P0wer_0f_F@c7oR1z4t10n_T0_Br3aK_R4nd0mNe$$!}
Winter On The Street
Description:
Winter was walking around the street and he has found a new way of a public-key cryptosystem to encrypt messages in a quantum channel. He thinks that his idea is fascinating and no one can break it. Could you prove him wrong?
The challenge comes with a python script
challenge.py
and
output.txt
files. The name looks unfamiliar, but the description sounds a
bit tricky since it mentioned a public-key system with a quantum
sense.
The text file:
The script contains:
Analysis
Looking at the code, it has a function
H
that
hashing a message
n
times with
sha512
. It also generates a private key based on a secret and hash it
65536
times, and a public key based on the same secret, but hashing it
with 16 bit random number from the private key after converting
it into binary. Then, it encrypts the flag with the middle 16
bytes of the private key.
After searching about such known quantum cryptosystems, we found Winternitz One Time Signature. It’s a quantum resistant digital signature hash based scheme that uses relatively small key and signature sizes, but the challenge looks a modified version of it.
The scheme works as follow:
If you want to sign a message between \(0\) and \(w-1\), where \(w\) is some parameter of the scheme.
- The private key \(K\) to sign a message \(M\), where \(0 < M < w\), compute \(H^M (K)\).
- The public key is \(H^w(K)\).
- WOTS signature \(S\) verified by checking that \(H^{w-M}(S)\) is equal to the public key, where \(S\) is the private key after \(M\) hashing.
From the above, we shall get \(M + (w - M) = w\) times which is the public key.
Solution
To get all the pieces together, we don’t know what
K = secret_key
and \(M\), but we know only \(w = 2^{16}\). Notice that, \(M\)
in the code generated by 16 bit integer chosen randomly from the
private key and also we know that \(M < w\). Thus, our goal
is to brute force over \(M\) in the range \([0, 2^{16}]\) and
hash the public key \(M\) times in order to get the flag.
The flag:
Learn With Encryption
Description:
Y4mm1 has created a super-secure communication on a quantum computer so that he can easily send any messages to Vector, but somehow he messed up with his secret so that no one can retrieve it. Could you help Vector to read the message?
This was the toughest one among all the previous challenges. The
description gives us a big hit along with the name that this one
deals with a quantum system as well. However, this time it has 5
file
cipher.txt
,
error.txt
,
U.txt
,
Vector.pub
, and
Y4mm1.pub
and a Python script
challenge.py
. Looking at the text file, all of them contains a lot of lines
contains nonsense long integers.
Let’s look at the script:
Analysis
The script has 2 main functions
encrypt
and
random_sampling
. Then it does the following:
- Get the flag from the secret file, then converts it into binary.
- Generate a known random 1024 bit prime \(q\).
- Generate 50 random integers 4096 bit each as a Vector’s public key.
- Generate a 2048 bit random integer \(s\) along with 50 random integers 256 bit each.
-
Generate Y4mm1’s public key based on Vector’s key and the
Error
with this equation \(Y = Vs + Error\).
All the previous are reduced module \(q\), then it encrypts the bits of the flag with the Vector and Y4mm1 public keys.
The encryption function get random samples from both public keys, iterate over the bits of the flag and in each iteration does the folowing:
-
Sum the first sample that belongs to Vector’s public and
append it into a list
u
. -
Sum the second sample that belongs to Y4mm1’s public and add
to it the current flag bit multiplied by \(\frac{q}{2}\), then
append it into a list
v
.
Solution
After some digging about quantum cryptography, this technique is pretty known in the robust quantum world of crypto systems called Learning With Errors. It’s a lattice based cryptography algorithm built on top of SVP & CVP hard problems. This challenge is a simplfied version of the LWE system.
The secret parameters should be \(s\) and \(Error\), but we know only the \(Error\) vector and from this equation \(Y = Vs + Error \pmod q\) the only unknown here is the secret integer \(s\), where \(Y\) and \(V\) are the public keys. Rearranging the equation, we shall get \(s = V^{-1}(Y - Error) \pmod q\) where \(xV = 1 \pmod q\) the equation to get the modular inverse of \(V \pmod q\).
Note that, the public keys are lists and the secret is a single integer, so for any values of the public keys, we should get the same secret value \(s\) from the previous equation.
Then, we can apply the decryption process \(dec = c - su \pmod
q\) where \(c\) is the encrypted bit integer from
cipher.txt
and \(u\) is the value from
U.txt
.
In order to get each bit of the flag, we should check \(dec <
\frac{q}{2}\) for 0 and 1 otherwise.
It doesn’t matter \(dec < \frac{q}{2^i}\), for any arbitrary \(i>0\). The \(dec\) should be close to 0, you could notice that from the encryption function.
Finally, we’ll get the flag:
Conclusion
This write-up discussed and analyzed not only the crypto challenges in the competition but also how to deal with difficult problems and how to cut them into pieces in order to be able to find a solution. In addition, it covered new topics in cryptography and post-quantum crypto in a simplified way to get into the concept and understand how it works.
The topics included:
- Padding Oracle Attack on AES-CBC mode and implementation steps.
- Attacking Blum Blum Shub pseudo random number generator algorithm.
- Post Quantum Hash-Based Cryptography and Winternitz One Time Signature.
- Post Quantum Lattice-Based Cryptography and Learning With Errors.
All post-quantum schemes are considered NP-Hard which means its security remains hard to quantify, but when security looks too good to be true, it often is. Because of the lack of understanding of the constructions of such a scheme, however you get the first step in the future of cryptography.
Finally, there isn’t one way of thinking about a challenge to find a solution. Although, you may take hours, days, weeks, or even months in order to understand how such a cryptosystem works and find flaws to break it.