Cryptanalysis is the study of cryptographic algorithms and methods. The basic goal is to break cryptographic algorithms in one way or another. For example, the attacker may have acquired knowledge of some cryptotexts and the corresponding plaintexts. Then he tries to use this information for finding the used key. In another setting, we assume that the attacker has only some cryptotexts based on which the key should be found. It is always assumed that the attacker knows all details of the encryption and decryption processes, only the secret key is not known. If this cannot be assumed then the encryption method is considered to be very weak.
Let us take a look at the substitution cipher. Now every occurrence of the letter ’e’ in the plaintext is encrypted as the same letter in the cryptotext, lets say ’Å’. Because ’e’ is the most common letter in the English language, ’Å’ should be one of the most common letters of the cryptotext. So we may assume that the common letters in the cryptotext correspond to common letters in the language of the plaintext and use this information for making educated guesses about how each letter is encrypted. This cryptanalytic approach is called frequency analysis.
The same approach cannot be used against OTP. If the key is chosen randomly, both 0 and 1 appear as often in the cryptotext, in average. This happens regardless of how common 1 is in the plaintext. In fact, any cryptotext could result from any plaintext with a suitable key. Assume the known cryptotext bit is C. Now the corresponding plaintext bit could be either C (which happens if the key bit is zero) or 1-C (which happens if the key bit is one). Therefore, knowing the cryptotext does not provide any new information about the plaintext to the attacker. This means the OTP is unconditionally secure.
For the OTP, we applied the setting where the attacker knows only the cryptotext. If the attacker knows both the cryptotext and the corresponding plaintext, then he can easily recover the used key. However, breaking OTP in this setting is not relevant because the recovered key is not used to encrypt anything else than the plaintext that the attacker already knows.
Modes of operation
A block cipher, like AES, is used to encrypt blocks of a certain size. What should be done if the message is longer than that size?
The simplest way to encrypt a long message is to take the first block, encrypt it using the key to produce the first cryptotext block, then take the second block, encrypt it using the same key to produce the second cryptotext block etc. This approach is one of the block cipher modes of operation, called Electronic Codebook (ECB). It is the simplest way but often could be broken by frequency analysis. The attacker notices if you encrypt the same plaintext block twice because the two cryptotexts are also the same. This happens, for example, if plaintext contains some commonly used short pattern, like 'OK'.
Other modes of operation avoid this problem by using extra input in addition to plaintext and key. For instance, previously computed cryptotext blocks or counters could be used for this purpose. Our HTTPS example (
TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384_256 bit keys,TLS 1.2) makes use of AES algorithm in Galois/Counter Mode. The key length is 256 bits.
The next 3 exercises should be done in order.
Padding oracle attack
Padding oracle attack shows that a tiny amount of additional information can be enough to break the cipher.
Earlier versions of CBC decipher implementations would return an error message to the sender if the padding of the sent message was correct. This information, assuming that we have access to submit our own messages to the decipher, is enough to break CBC encryption! Furthermore, the breakage doesn't depend on the underlying block cipher.
Assume that we have an oracle that given a ciphertext will tell whether the decrypted message has a valid padding. Note that we don't get to see the decrypted message, we only observe whether the padding is valid.
Assume that we have two blocks of ciphertext and , each of length 8. Write . We know from previous exercise that . If we can solve , then we can solve .
Let us focus on solving the last byte of . There are 256 possible values for . Consider 256 different ciphertexts of form , where is an array of zeros, except for the last entry where . Let be the 2nd block of the decrypted message .
Note that is different for every ; there is an index for which . If we know , then we can deduce since .
We don't know but we know that, since , has a valid padding! So to find we can test every with an oracle to see which ciphertexts produce a valid padding.
There is a small complication as the oracle can find multiple messages with a valid padding. For example, if or , then has a valid padding. Note that does not change as we vary . This means that the oracle can find at most 2 messages with a valid padding (can you see why?).
This gives us the following approach. We test 256 ciphertexts to see which ones have valid padding. If there is only one, then we have found and we can solve . If there are two, say and , we need to figure out which one is which. This can be done by changing and to a different value (for example by xoring with 1). This will change the 7th byte of the decrypted message. Let and be the modified ciphers. Then will still produce a valid padding while no longer has a valid padding since the last and the second last byte in the decrypted message no longer match. In summary, we can find and solve .
Once we have solved we can move to solving . This is done by setting and . Using the oracle we can find the index for which , resulting in a valid padding. Note that, unlike in the case for the last byte, the oracle will find only one index (can you see why?). Finding allows us to solve . We can now move to to by setting , , and . We continue until we have solved , which will give us .
If the cipher has more than two blocks, say, we can decrypt each individual block, say , by running the previous procedure for each pair .
Remember to check your points from the ball on the bottom-right corner of the material!