Serpent
Serpent is a modern block cipher published in 1998. It was one of the 5 finalists in the AES contest. It had the least number of negative votes and the second greatest positive vote count after Rijndael.
Construction
Basic facts:
Construction:Key
expansionexpansion:As an AES candidate, Serpent offered 128, 192 and 256-bit keys. By design, it can accept any key between 0 and 256 bits inclusive, any key that has less than the full 256 bits is padded by appending a single $1$ bit and then as many $0$ bits as needed.
The 256-bit key is expanded into 132 32-bit words (collectively called the prekey) using a recurrent expression.
A bitslice-mode S-box (see later) is applied to 4 words of the prekey at a time, producing 33 128-bit subkeys (round keys).
Implementation
Implementation:The algorithm specification describes two possible approaches to implementing Serpent.
Formally, the Serpent round operates on state consisting of 32 4-bit chunks of data (the same S-box is applied 32 times in parallel).
In the so-called "bitslice mode", the state is rearranged into 4 32-bit words, where the first bits of each word correspond to the first 4-bit chunk in the formal description, etc.
The catch is that Serpent treats input data (i.e. plaintexts and ciphertexts) as already converted to the bitslice representation.
This means that when the "formal" implementation is chosen, the input and output bits (as well as individual subkeys) have to be permuted back from the default bitslice mode.
Data
representationrepresentation:In Serpent, for some godforsaken reason, everything is backwards.
- blocks are accepted and presented as 128-bit numbers; the (user) key is interpreted as a sequence of bits. (An incredible formalization choice for a cipher that might commonly be implemented on a system where 128-bit ints aren't a thing.)
The user key is
TODOexpanded
Externally,
256 - blocks are accepted and presented as 128-bit numbers; the (user) key is interpreted as a sequence of bits. (An incredible formalization choice for a cipher that might commonly be implemented on a system where 128-bit ints aren't a thing.)
void serpent_pad_128bit_key(uint8_t key[32])
{
key[16] = 0x01;
memset(&key[16], 16, 0x00);
}
and References:not this:
void bad_pad_128bit_key(uint8_t key[32])
{
key[16] = 0x80;
memset(&key[16], 16, 0x00);
}
References
- https://www.cl.cam.ac.uk/archive/rja14/Papers/serpent.pdf