mirror of
https://github.com/GaloisInc/cryptol.git
synced 2025-01-07 08:19:12 +03:00
1524 lines
60 KiB
TeX
1524 lines
60 KiB
TeX
\commentout{
|
|
\begin{code}
|
|
module AES where
|
|
|
|
import Enigma
|
|
// import Classic
|
|
\end{code}
|
|
}
|
|
|
|
%#####################################################################
|
|
\chapter{AES: The Advanced Encryption Standard}
|
|
\label{chapter:aes}
|
|
|
|
AES\indAES is a symmetric key encryption algorithm (a symmetric
|
|
cipher, per the discussion in~\autoref{chapter:classic}), based on the
|
|
Rijndael algorithm designed by Joan Daemen and Vincent
|
|
Rijmen~\cite{DaemenR02}\glosAES. (The term {\em symmetric key} means
|
|
that the algorithm uses the same key for encryption and
|
|
decryption.\indSymKey) AES was adopted in 2001 by the US government,
|
|
deemed suitable for protecting classified information up to {\em
|
|
secret} level for the key size 128, and up to the {\em top-secret}
|
|
level for key sizes 192 and 256.
|
|
|
|
In this chapter, we will program AES in Cryptol. Our emphasis will be
|
|
on clarity, as opposed to efficiency, and we shall follow the NIST
|
|
standard description of AES\indAES fairly closely~\cite{aes}\glosNIST.
|
|
Referring to the standard as you work your way through this chapter is
|
|
recommended.
|
|
|
|
Some surprises may be at hand for the reader who has never deeply
|
|
examined a modern cryptography algorithm before.
|
|
|
|
First, algorithms like AES are typically composed of many smaller
|
|
units of varying kinds. Consequently, the entire algorithm is
|
|
constructed bottom-up by specifying and verifying each of its
|
|
component pieces. It is wise to handle smaller and simpler components
|
|
first. It is also a good practice, though hard to accomplish the
|
|
first one or two times you write such a specification, to write
|
|
specifications with an eye toward reuse in multiple instantiations of
|
|
the same algorithm (e.g., different key sizes or configurations). The
|
|
choice between encoding configurations at the type level or the value
|
|
level is aesthetic and practical: some verification is only possible
|
|
when one encodes information at the type level.
|
|
|
|
Second, algorithms frequently depend upon interesting data structures
|
|
and mathematical constructs, the latter of which can be though of as
|
|
data structures in a pure mathematics sense. The definition, shape,
|
|
form, and subtleties of these data structures are critical to the
|
|
\emph{correct definition} of the crypto algorithm \emph{as well as its
|
|
security properties}. Implementing an algorithm using an
|
|
alternative datatype construction that you believe has the same
|
|
properties as that which is stipulated in a standard is nearly always
|
|
the wrong thing to do. Also, the subtleties of these constructions
|
|
usually boils down to what an engineer might think of as ``magic
|
|
numbers''---strange initial values or specific polynomials that appear
|
|
out of thin air. Just remind yourself that the discovery and analysis
|
|
of those magic values was, in general, the joint hard work of a
|
|
community of cryptographers.
|
|
|
|
%=====================================================================
|
|
\section{Parameters}
|
|
\label{sec:aesparams}
|
|
|
|
The AES algorithm always takes 128-bits of input, and always produces
|
|
128-bits of output, regardless of the key size. The key-size can be
|
|
one of 128 (AES128), 192 (AES192), or 256 (AES256). Following the
|
|
standard, we define the following three parameters~\cite[Section
|
|
2.2]{aes}:
|
|
\begin{itemize}
|
|
\item {\tt Nb}: Number of columns, always set to 4 by the standard.
|
|
\item {\tt Nk}: Number of key-blocks, which is the number of 32-bit
|
|
words in the key: 4 for AES128, 6 for AES192, and 8 for AES256;
|
|
\item {\tt Nr}: Number of rounds, which {\tt Nr} is always $6 +
|
|
\mbox{\tt Nk}$, according to the standard. Thus, 10 for AES128, 12
|
|
for AES192, and 14 for AES256.
|
|
\end{itemize}
|
|
The Cryptol definitions follow the above descriptions verbatim:
|
|
\begin{code}
|
|
type AES128 = 4
|
|
type AES192 = 6
|
|
type AES256 = 8
|
|
|
|
type Nk = AES128
|
|
type Nb = 4
|
|
type Nr = 6 + Nk
|
|
\end{code}
|
|
The following derived type is helpful in signatures:
|
|
\begin{code}
|
|
type AESKeySize = (Nk*32)
|
|
\end{code}
|
|
|
|
%=====================================================================
|
|
% \section{Polynomials in \texorpdfstring{GF($2^8$)}{GF(2,8)}}
|
|
% \label{sec:polynomials}
|
|
\sectionWithAnswers{Polynomials in \texorpdfstring{GF($2^8$)}{GF(2,8)}}{sec:polynomials}
|
|
|
|
AES\indAES works on a two-dimensional representation of the input
|
|
arranged into bytes, called the {\em state}.\indAESState For an
|
|
128-bit input, we have precisely 4 rows, each containing {\tt Nb}
|
|
(i.e., 4) bytes, each of which is 8-bits wide, totaling
|
|
$4\times4\times8 = 128$ bits. The bytes themselves are treated as
|
|
finite field elements in the Galois field
|
|
GF($2^8$)~\cite{wiki:galoisfield}\indGF, giving rise to the following
|
|
declarations:
|
|
\begin{code}
|
|
type GF28 = [8]
|
|
type State = [4][Nb]GF28
|
|
\end{code}
|
|
|
|
The hard-encoding of \texttt{GF28} in this specification is completely
|
|
appropriate because the construction of AES depends entirely upon the
|
|
Galois field GF($2^8$). It is conceivable that other algorithms might
|
|
be parameterized across GF($2^k$) for some $k$, in which case the
|
|
underyling type declaration would also be parameterized.
|
|
|
|
While a basic understanding Galois field operations is helpful, the
|
|
details are not essential for our modeling purposes. It suffices to
|
|
note that GF($2^8$) has precisely 256 elements, each of which is a
|
|
polynomial\indPoly of maximum degree 7, where the coefficients are
|
|
either 0 or 1. The numbers from 0 to 255 (i.e., all possible 8-bit
|
|
values) are in one-to-one correspondence with these polynomials. The
|
|
coefficients of the polynomial come from the successive bits of the
|
|
number, and vice versa. For instance the 8-bit number 87 can be
|
|
written as {\tt 0b01010111} in binary, and hence corresponds to the
|
|
polynomial $x^6 + x^4 + x^2 + x^1 + 1$. Similarly, the polynomial $x^4
|
|
+ x^3 + x$ corresponds to the number {\tt 0b00011010}, i.e., 26. We
|
|
can also compute this value by evaluating the polynomial for $x=2$,
|
|
obtaining $2^4+2^3+2 = 16+8+2 = 26$.
|
|
|
|
Cryptol allows you to write polynomials in GF($2^n$), for arbitrary
|
|
$n$, using the following notation:
|
|
\begin{Verbatim}
|
|
Cryptol> <| x^^6 + x^^4 + x^^2 + x^^1 + 1 |>
|
|
87
|
|
Cryptol> 0b1010111
|
|
87
|
|
Cryptol> <| x^^4 + x^^3 + x |>
|
|
26
|
|
Cryptol> 0b11010
|
|
26
|
|
\end{Verbatim}
|
|
|
|
A polynomial is similar to a decimal representation of a number,
|
|
albeit in a more suggestive syntax. Like with a decimal, the Cryptol
|
|
type system will default the type to be the smallest number of bits
|
|
required to represent the polynomial, but it may be expanded to more
|
|
bits if an expression requires it.
|
|
|
|
\paragraph*{Addition and Subtraction} Given two polynomials,
|
|
adding\indAddPoly and subtracting\indSubPoly them in a Galois field
|
|
GF($2^n$) results in a new polynomial where terms with the same power
|
|
cancel each other out. When interpreted as a word, both addition and
|
|
subtraction amount to a simple exclusive-or operation. Cryptol's
|
|
\Verb|^|\indXOr operator captures this idiom concisely:
|
|
\begin{Verbatim}
|
|
Cryptol> (<| x^^4 + x^^2 |> ^ <| x^^5 + x^^2 + 1 |>) == \
|
|
<| x^^5 + x^^4 + 1 |>
|
|
True
|
|
\end{Verbatim}
|
|
Note that the term $x^2$ cancels since it appears in both
|
|
polynomials. Also note the parentheses are required due to the
|
|
precedence of {\tt ==} vs. {\tt \Verb|^|}.
|
|
\begin{Exercise}\label{ex:gf:0}
|
|
Adding a polynomial to itself in GF($2^n$) will always yield {\tt 0}
|
|
since all the terms will cancel each other. Write and prove a
|
|
theorem {\tt polySelfAdd} over {\tt GF28} to state this fact.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gf:0}
|
|
\todo[inline]{Recheck use of parentheses in property after \ticket{208}
|
|
is closed.}
|
|
\begin{code}
|
|
polySelfAdd: GF28 -> Bit
|
|
property polySelfAdd x = (x ^ x) == zero
|
|
\end{code}
|
|
We have:\indSBV
|
|
\begin{Verbatim}
|
|
Cryptol> :prove polySelfAdd
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
\unparagraph While adding two polynomials does not warrant a separate
|
|
function, we will need a version that can add a sequence of
|
|
polynomials:
|
|
\begin{Exercise}\label{ex:gf:add0}
|
|
Define a function
|
|
\begin{code}
|
|
gf28Add : {n} (fin n) => [n]GF28 -> GF28
|
|
\end{code}
|
|
that adds all the elements given.
|
|
\lhint{Use a fold, see Pg.~\pageref{par:fold}.}\indFold
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gf:add0}
|
|
\begin{code}
|
|
gf28Add ps = sums ! 0
|
|
where sums = [zero] # [ p ^ s | p <- ps
|
|
| s <- sums
|
|
]
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\paragraph*{Multiplication} Multiplication GF($2^n$) follows the usual
|
|
polynomial multiplication algorithm, where we multiply the first
|
|
polynomial with each term of the second, and add all the partial sums
|
|
(i.e., compute their exclusive-or). While this operation can be
|
|
programmed explicitly, Cryptol does provide the primitive {\tt
|
|
pmult}\indMulPoly for this purpose:
|
|
\begin{Verbatim}
|
|
Cryptol> pmult <| x^^3 + x^^2 + x + 1 |> <| x^^2 + x + 1 |>
|
|
45
|
|
Cryptol> <| x^^5 + x^^3 + x^^2 + 1 |>
|
|
45
|
|
\end{Verbatim}
|
|
|
|
\begin{Exercise}\label{ex:gf:1}
|
|
Multiply the polynomials $x^3 +x^2+x+1$ and $x^2+x+1$ by hand in
|
|
GF($2^8$) and show that the result is indeed $x^5+x^3+x^2+1$, (i.e.,
|
|
$45$), justifying Cryptol's result above.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gf:1}
|
|
\todo[inline]{Figure out outdent on this answer if this todo note is
|
|
removed.} We first compute the results of multiplying our first
|
|
polynomial ($x^3 +x^2+x+1$) with each term in the second polynomial
|
|
($x^2+x+1$) separately:
|
|
\begin{eqnarray*}
|
|
(x^3 + x^2 + x + 1) \times x^2 &=& x^5 + x^4 + x^3 + x^2 \\
|
|
(x^3 + x^2 + x + 1) \times x\;\; &=& x^4 + x^3 + x^2 + x \\
|
|
(x^3 + x^2 + x + 1) \times 1\;\; &=& x^3 + x^2 + x\;\; + 1
|
|
\end{eqnarray*}
|
|
We now add the resulting polynomials, remembering the adding the same
|
|
powered terms cancel each other out. For instance, we have two
|
|
instances each of $x^4$, $x^2$, and $x$, which all get canceled. We
|
|
have three instances each of $x^3$ and $x^2$, so they survive the
|
|
addition, etc. After the addition, we are left with the polynomial
|
|
$x^5 + x^3 + x^2 + 1$, which can be interpreted as {\tt 0b00101101},
|
|
i.e., $45$.
|
|
\end{Answer}
|
|
|
|
\paragraph*{Reduction} If you multiply two polynomials with degrees $m$
|
|
and $n$, you will get a new polynomial of degree $m+n$. As we
|
|
mentioned before, the polynomials in GF($2^8$)\indGF have degree at
|
|
most 7. Obviously, $m+n$ can be larger than $7$ when we multiply to
|
|
elements of GF($2^8$). So we have to find a way to map the resulting
|
|
larger-degree polynomial back to an element of GF($2^8$). This is done
|
|
by reduction, or modulus, with respect to an {\em irreducible
|
|
polynomial}\indIrredPoly. The AES algorithm uses the following
|
|
polynomial for this purpose:
|
|
\begin{code}
|
|
irreducible = <| x^^8 + x^^4 + x^^3 + x + 1 |>
|
|
\end{code}
|
|
(Recall in the introduction of this chapter our warning about magic!)
|
|
|
|
Note that {\tt irreducible} is {\em not} an element of GF($2^8$),
|
|
since it has degree 8. However we can use this polynomial to define
|
|
the multiplication routine itself, which uses Cryptol's {\tt
|
|
pmod}\indModPoly (polynomial modulus) function, as follows:
|
|
\begin{code}
|
|
gf28Mult : (GF28, GF28) -> GF28
|
|
gf28Mult (x, y) = pmod (pmult x y) irreducible
|
|
\end{code}
|
|
Polynomial modulus and division operations follow the usual schoolbook
|
|
algorithm for long-division---a fairly laborious process in itself,
|
|
but it is well studied in mathematics~\cite{wiki:polydiv}. Luckily for
|
|
us, Cryptol's {\tt pdiv}\indDivPoly and {\tt pmod}\indModPoly
|
|
functions implement these operations, saving us the programming task.
|
|
|
|
\begin{Exercise}\label{ex:pdivmod}
|
|
Divide $x^5 + x^3 + 1$ by $x^3 + x^2 + 1$ by hand, finding the
|
|
quotient and the remainder. Check your answer with Cryptol's {\tt
|
|
pmod} and {\tt pdiv} functions.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:pdivmod}
|
|
The long division algorithm is laborious, but not particularly hard:
|
|
\[
|
|
\renewcommand\arraystretch{1.2}
|
|
\begin{array}{*1r @{\hskip\arraycolsep}c@{\hskip\arraycolsep} *6r}
|
|
% & && & 1 & 1 & 0 \\
|
|
& & & & & x^2 &+ x & \\
|
|
\cline{2-8}
|
|
x^3 + x^2 + 1 &\big)& x^5 & & + x^3 & & & + 1\\
|
|
& & x^5 & + x^4 & & + x^2 & & \\
|
|
\cline{3-6}
|
|
& & & x^4 & + x^3 & + x^2 & & + 1 \\
|
|
& & & x^4 & + x^3 & & + x& \\
|
|
\cline{4-8}
|
|
& & & & & x^2 & + x & + 1 \\
|
|
\end{array}
|
|
\]
|
|
Therefore, the quotient is $x^2+x$ and the remainder is $x^2+x+1$. We
|
|
can verify this easily with Cryptol:
|
|
\begin{Verbatim}
|
|
Cryptol> pdiv <| x^^5 + x^^3 + 1 |> <| x^^3 + x^^2 + 1 |> == \
|
|
<| x^^2 + x |>
|
|
True
|
|
Cryptol> pmod <| x^^5 + x^^3 + 1 |> <| x^^3 + x^^2 + 1 |> == \
|
|
<| x^^2 + x + 1 |>
|
|
True
|
|
\end{Verbatim}
|
|
Another way to check your result would be to multiply the quotient by
|
|
the divisor and add the remainder, and check that it gives us
|
|
precisely the polynomial we started with:
|
|
\begin{Verbatim}
|
|
Cryptol> pmult <| x^^2 + x |> <| x^^3 + x^^2 + 1 |> \
|
|
^ <| x^^2 + x + 1 |>
|
|
0x29
|
|
Cryptol> <| x^^5 + x^^3 + 1 |>
|
|
0x29
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:gf:2}
|
|
Write and prove theorems showing that {\tt gf28Mult} (i) has the
|
|
polynomial 1 as its unit, (ii) is commutative, and (iii) is
|
|
associative.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gf:2}
|
|
\fvset{fontsize=\footnotesize}
|
|
\begin{code}
|
|
property gf28MultUnit x = gf28Mult(x, 1) == x
|
|
property gf28MultCommutative x y = gf28Mult(x, y) == gf28Mult(y, x)
|
|
property gf28MultAssociative x y z = gf28Mult(x, gf28Mult(y, z))
|
|
== gf28Mult(gf28Mult(x, y), z)
|
|
\end{code}
|
|
It turns out that proving the unit and commutativity are fairly
|
|
trivial:\indCmdProve
|
|
\begin{Verbatim}
|
|
Cryptol> :prove gf28MultUnit
|
|
Q.E.D.
|
|
Cryptol> :prove gf28MultCommutative
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
But {\tt aesMultAssociative} takes much longer! We show the results of
|
|
{\tt :check} below:\indCmdCheck
|
|
\begin{Verbatim}
|
|
Cryptol> :check gf28MultAssociative
|
|
Checking case 1000 of 1000 (100.00%)
|
|
1000 tests passed OK
|
|
\end{Verbatim}
|
|
%% [Coverage: 5.96e-3%. (1000/16777216)]
|
|
Note that the coverage is pretty small (on the order of $0.006\%$)
|
|
in this case. Proving associativity of multiplication algorithms using
|
|
SAT/SMT based technologies is a notoriously hard task~\cite[Section
|
|
6.3.1]{DecisionProcedures2008}. If you have the time, you can let
|
|
Cryptol run long enough to complete the {\tt :prove
|
|
gf28MultAssociative} command, however.
|
|
\end{Answer}
|
|
|
|
%=====================================================================
|
|
% \section{The {\ttfamily{\textbf SubBytes}} transformation}
|
|
% \label{aes:subbytes}
|
|
\sectionWithAnswers{The {\ttfamily{\textbf SubBytes}} transformation}{aes:subbytes}
|
|
|
|
\todo[inline]{Introduce a figure here, perhaps lifted from the
|
|
standard, to better explain this stage of the algorithm.}
|
|
|
|
Recall that the state in AES\indAES is a $4\times4$ matrix of
|
|
bytes. As part of its operation, AES\indAES performs the so called
|
|
{\tt SubBytes} transformation\indAESSbox~\cite[Section 5.1.1]{aes},
|
|
substituting each byte in the state with another element. Given an $x
|
|
\in \mbox{GF}(2^8)$,\indGF the substitution for $x$ is computed as
|
|
follows:
|
|
\begin{enumerate}
|
|
\item Compute the multiplicative inverse of $x$ in GF$(2^8)$\indGF,
|
|
call it $b$. If $x$ is 0, then take 0 as the result.
|
|
\item Replace bits in $b$ as follows: Each bit $b_i$ becomes $b_i
|
|
\oplus b_{i+4\imod{8}} \oplus b_{i+5\imod{8}} \oplus
|
|
b_{i+6\imod{8}} \oplus b_{i+7\imod{8}} \oplus c_i$. Here $\oplus$
|
|
is exclusive-or and $c$ is {\tt 0x63}.
|
|
\end{enumerate}
|
|
|
|
\paragraph*{Computing the multiplicative inverse} It turns out that the
|
|
inverse of any non-zero $x$ in GF$(2^8)$ can be computed by raising
|
|
$x$ to the power 254, i.e., multiplying $x$ by itself 254
|
|
times. (Mathematically, GF$(2^8)$\indGF is a cyclic group such that
|
|
$x^{255}$ is always $1$ for any $x$, making $x^{254}$ the
|
|
multiplicative inverse of $x$.)
|
|
|
|
\begin{Exercise}\label{ex:gfmi:0}
|
|
Write a function
|
|
\begin{code}
|
|
gf28Pow : (GF28, [8]) -> GF28
|
|
\end{code}
|
|
such that the call {\tt gf28Pow (n, k)} returns $n^k$ using {\tt
|
|
gf28Mult} as the multiplication operator. \lhint{Use the fact that
|
|
$x^0 = 1$, $x^{2n} = (x^n)^2$, and $x^{2n+1} = x \times (x^n)^2$ to
|
|
speed up the exponentiation.}
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gfmi:0}
|
|
\begin{code}
|
|
gf28Pow (n, k) = pow k
|
|
where sq x = gf28Mult (x, x)
|
|
odd x = x ! 0
|
|
pow i = if i == 0 then 1
|
|
else if odd i
|
|
then gf28Mult (n, sq (pow (i >> 1)))
|
|
else sq (pow (i >> 1))
|
|
\end{code}
|
|
Here is a version that follows the stream-recursion pattern:
|
|
\begin{code}
|
|
gf28Pow' : (GF28, [8]) -> GF28
|
|
gf28Pow' (n, k) = pows ! 0
|
|
where pows = [1] # [ if bit then gf28Mult (n, sq x)
|
|
else sq x
|
|
| x <- pows
|
|
| bit <- k
|
|
]
|
|
sq x = gf28Mult (x, x)
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:gfmi:01}
|
|
Write a function
|
|
\begin{code}
|
|
gf28Inverse : GF28 -> GF28
|
|
\end{code}
|
|
to compute the multiplicative inverse of a given element by raising it
|
|
to the power $254$. Note that {\tt gf28Inverse} must map $0$ to
|
|
$0$. Do you have to do anything special to make sure this happens?
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gfmi:01}
|
|
\begin{code}
|
|
gf28Inverse x = gf28Pow (x, 254)
|
|
\end{code}
|
|
We do not have to do anything special about $0$, since our {\tt
|
|
gf28Inverse} function yields $0$ in that case:
|
|
\begin{Verbatim}
|
|
Cryptol> gf28Inverse 0
|
|
0x00
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:gfmi:1}
|
|
Write and prove a property {\tt gf28InverseCorrect}, ensuring that
|
|
{\tt gf28Inverse x} does indeed return the multiplicative inverse of
|
|
{\tt x}. Do you have to do anything special when {\tt x} is $0$?
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:gfmi:1}
|
|
Since $0$ does not have a multiplicative inverse, we have to write a
|
|
conditional property:
|
|
\begin{code}
|
|
property gf28InverseCorrect x =
|
|
if x == 0 then x' == 0 else gf28Mult(x, x') == 1
|
|
where x' = gf28Inverse x
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove gf28InverseCorrect
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\paragraph*{Transforming the result} As we mentioned above, the
|
|
AES\indAES specification asks us to transform each bit $b_i$ according
|
|
to the following transformation:
|
|
$$
|
|
b_i \oplus b_{i+4\imod{8}} \oplus b_{i+5\imod{8}} \oplus b_{i+6\imod{8}} \oplus b_{i+7\imod{8}} \oplus c_i
|
|
$$
|
|
For instance, bit $b_5$ becomes $b_5 \oplus b_1 \oplus b_2 \oplus b_3
|
|
\oplus c_5$. When interpreted at the word level, this basically
|
|
amounts to computing:
|
|
$$
|
|
b \oplus (b \ggg 4) \oplus (b \ggg 5) \oplus (b \ggg 6) \oplus (b \ggg 7) \oplus c
|
|
$$
|
|
by aligning the corresponding bits in the word representation.
|
|
|
|
\todo[inline]{Justify/prove the equivalence of these two terms.
|
|
Perhaps add a property to prove it in Cryptol?}
|
|
|
|
\begin{Exercise}\label{ex:aessbytes:0}
|
|
Write a function
|
|
\begin{code}
|
|
xformByte : GF28 -> GF28
|
|
\end{code}
|
|
that computes the above described transformation.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aessbytes:0}
|
|
\begin{code}
|
|
xformByte b = gf28Add [b, (b >>> 4), (b >>> 5),
|
|
(b >>> 6), (b >>> 7), c]
|
|
where c = 0x63
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\todo[inline]{Remind the reader of the context again before putting
|
|
everything together.}
|
|
|
|
\paragraph*{Putting it together} Armed with {\tt gf28Inverse} and {\tt
|
|
xformByte}, we can easily code the function that transforms a single
|
|
byte as follows:
|
|
\begin{code}
|
|
SubByte : GF28 -> GF28
|
|
SubByte b = xformByte (gf28Inverse b)
|
|
\end{code}
|
|
AES's {\tt SubBytes}\indAES transformation merely applies this
|
|
function to each byte of the state:
|
|
\begin{code}
|
|
SubBytes : State -> State
|
|
SubBytes state = [ [ SubByte b | b <- row ] | row <- state ]
|
|
\end{code}
|
|
|
|
\paragraph*{Table lookup} Our definition of the {\tt SubByte} function
|
|
above follows how the designers of AES came up with the substitution
|
|
maps, i.e., it is a {\em reference} implementation. For efficiency
|
|
purposes, however, we might prefer a version that simply performs a
|
|
look-up in a table. Notice that the type of {\tt SubByte} is {\tt GF28
|
|
-> GF28}, i.e., it takes a value between 0 and 255. Therefore, we
|
|
can make a table containing the precomputed values for all possible
|
|
256 inputs, and simply perform a table look-up instead of computing
|
|
these values each time we need it. In fact, Figure~7 on page 16 of
|
|
the AES\indAES standard lists these precomputed values for
|
|
us~\cite[Section 5.1.1]{aes}. We capture this table below in Cryptol:
|
|
|
|
\vspace{0.25cm}
|
|
\begin{minipage}{\textwidth}
|
|
{\footnotesize
|
|
\begin{code}
|
|
sbox : [256]GF28
|
|
sbox = [
|
|
0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67,
|
|
0x2b, 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59,
|
|
0x47, 0xf0, 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7,
|
|
0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1,
|
|
0x71, 0xd8, 0x31, 0x15, 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05,
|
|
0x9a, 0x07, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, 0x09, 0x83,
|
|
0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6, 0xb3, 0x29,
|
|
0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b,
|
|
0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, 0xd0, 0xef, 0xaa,
|
|
0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02, 0x7f, 0x50, 0x3c,
|
|
0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc,
|
|
0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0x0c, 0x13, 0xec,
|
|
0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19,
|
|
0x73, 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee,
|
|
0xb8, 0x14, 0xde, 0x5e, 0x0b, 0xdb, 0xe0, 0x32, 0x3a, 0x0a, 0x49,
|
|
0x06, 0x24, 0x5c, 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79,
|
|
0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4,
|
|
0xea, 0x65, 0x7a, 0xae, 0x08, 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6,
|
|
0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, 0x70,
|
|
0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, 0x61, 0x35, 0x57, 0xb9,
|
|
0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e,
|
|
0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, 0x8c, 0xa1,
|
|
0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0x0f, 0xb0,
|
|
0x54, 0xbb, 0x16]
|
|
\end{code}
|
|
}
|
|
\end{minipage}
|
|
|
|
\noindent With this definition of {\tt sbox}, the look up variants of
|
|
{\tt SubByte} and {\tt SubBytes} becomes:\label{aes:subbytetl}
|
|
\begin{code}
|
|
SubByte' : GF28 -> GF28
|
|
SubByte' x = sbox @ x
|
|
|
|
SubBytes' : State -> State
|
|
SubBytes' state = [ [ SubByte' b | b <- row ] | row <- state ]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:sbox}
|
|
Write and prove a property stating that {\tt SubByte} and
|
|
{\tt SubByte'} are equivalent.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:sbox}
|
|
\begin{code}
|
|
property SubByteCorrect x = SubByte x == SubByte' x
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove SubByteCorrect
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\note{The {\tt SubByte'} and {\tt SubBytes'} versions are going to be
|
|
more efficient for execution, naturally. We should emphasize that
|
|
this mode of development is quite common in modern cryptography.
|
|
Ciphers are typically designed using ideas from mathematics, often
|
|
requiring complicated algorithms. To speed things up, however,
|
|
implementors use clever optimization tricks, convert functions to
|
|
look-up tables using precomputed values, etc.
|
|
|
|
What Cryptol allows us to do is to write the algorithms using both
|
|
styles, and then formally show that they are indeed equivalent, as
|
|
you did in Exercise~\ref{ex:sbox} above. This mode of
|
|
high-assurance development makes sure that we have not made any
|
|
cut-and-paste errors when we wrote down the numbers in our {\tt
|
|
sbox} table. Equivalently, our proof also establishes that the
|
|
official specification~\cite[Section 5.1.1]{aes} got its own table
|
|
correct!}
|
|
|
|
\todo[inline]{Revisit a commented-out exercise here after \ticket{210}
|
|
on \texttt{:safe} is addressed.}
|
|
% \begin{Exercise}\label{ex:sbox:safe}
|
|
% The implementation of {\tt SubByte'} uses indexing ({\tt
|
|
% @})\indIndex, and hence we might be worried about
|
|
% index-out-of-bounds errors. In this case the table has precisely
|
|
% 256 elements and we are indexing with a byte, so the indexing must
|
|
% be safe. Use Cryptol's {\tt :safe} command\indCmdSafe to verify
|
|
% this claim.
|
|
% \end{Exercise}
|
|
% \begin{Answer}\ansref{ex:sbox:safe}
|
|
% \begin{Verbatim}
|
|
% Cryptol> :safe SubByte'
|
|
% "SubByte'" is safe; no safety violations exist.
|
|
% \end{Verbatim}
|
|
% \end{Answer}
|
|
|
|
%=====================================================================
|
|
% \section{The {\ttfamily{\textbf ShiftRows}} transformation}
|
|
% \label{aes:shiftrows}
|
|
\sectionWithAnswers{The {\ttfamily{\textbf ShiftRows}} transformation}{aes:shiftrows}
|
|
|
|
\todo[inline]{Need a figure here to get reader in the right frame of
|
|
mind.}
|
|
|
|
The second transformation AES\indAES utilizes is the {\tt
|
|
ShiftRows}\indAESShiftRows operation~\cite[Section 5.1.2]{aes}. This
|
|
operation treats the {\tt State} as a $4\times4$ matrix, and rotates
|
|
the last three row to the left by the amounts 1, 2, and 3;
|
|
respectively. Implementing {\tt ShiftRows} in Cryptol is trivial,
|
|
using the \verb+<<<+\indRotLeft operator:
|
|
\begin{code}
|
|
ShiftRows : State -> State
|
|
ShiftRows state = [ row <<< shiftAmount | row <- state
|
|
| shiftAmount <- [0 .. 3]
|
|
]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:aesshiftrows:0}
|
|
Can you transform a state back into itself by repeated applications
|
|
of {\tt ShiftRows}? How many times would you need to shift? Verify
|
|
your answer by writing and proving a corresponding Cryptol property.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesshiftrows:0}
|
|
Consider what happens after 4 calls to {\tt ShiftRows}. The first
|
|
row will stay the same, since it never moves. The second row moves
|
|
one position each time, and hence it will move 4 positions at the
|
|
end, restoring it back to its original starting
|
|
configuration. Similarly, row 3 will rotate $2\times4=8$ times,
|
|
again restoring it. Finally, row 4 rotates 3 times each for a total
|
|
of $3\times4 = 12$ times, cycling back to its starting
|
|
position. Hence, every 4th rotation will restore the entire state
|
|
back. We can verify this in Cryptol by the following property:
|
|
\begin{code}
|
|
shiftRow4RestoresBack : State -> Bit
|
|
property shiftRow4RestoresBack state = states @ 4 == state
|
|
where states = [state] # [ ShiftRows s | s <- states ]
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove shiftRow4RestoresBack
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
Of course, any multiple of 4 would have the same effect.
|
|
\end{Answer}
|
|
|
|
%=====================================================================
|
|
% \section{The {\ttfamily{\textbf MixColumns}} transformation}
|
|
% \label{sec:aesmixcolumns}
|
|
\sectionWithAnswers{The {\ttfamily{\textbf MixColumns}} transformation}{sec:aesmixcolumns}
|
|
|
|
The third major transformation AES\indAES performs is the {\tt
|
|
MixColumns}\indAESMixColumns operation~\cite[Section 5.1.3]{aes}.
|
|
In this transformation, the {\tt State} is viewed as a $4\times4$
|
|
matrix, and each successive column of it is replaced by the result of
|
|
multiplying it by the matrix:
|
|
$$
|
|
\left[ \begin{array}{cccc} 2 & 3 & 1 & 1 \\ 1 & 2 & 3 & 1 \\ 1 & 1 & 2 & 3 \\ 3 & 1 & 1 & 2 \end{array} \right]
|
|
$$
|
|
|
|
As you might recall from linear algebra, given two \emph{compatible}
|
|
matrices $A$ and $B$, the $ij$'th element of $A\times B$ is the
|
|
dot-product of the $i$'th row of $A$ and the $j$'th column of $B$.
|
|
(By \emph{compatible} we mean the number of columns of $A$ must be the
|
|
same as the number of rows of $B$. All our matrices are $4\times 4$,
|
|
so they are always compatible.) The dot-product is defined as
|
|
multiplying the corresponding elements of two same-length vectors and
|
|
adding the results together. The only difference here is that we use
|
|
the functions {\tt gf28Add} and {\tt gf28Mult} for addition and
|
|
multiplication respectively. We will develop this algorithm in the
|
|
following sequence of exercises.
|
|
|
|
\begin{Exercise}\label{ex:aesmc:0}
|
|
Write a function {\tt gf28DotProduct} with the signature:
|
|
\begin{code}
|
|
gf28DotProduct : {n} (fin n) => ([n]GF28, [n]GF28) -> GF28
|
|
\end{code}
|
|
such that {\tt gf28DotProduct} returns the dot-product of two length $n$ vectors of GF($2^8$) elements.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesmc:0}
|
|
\begin{code}
|
|
gf28DotProduct (xs, ys) = gf28Add [ gf28Mult (x, y) | x <- xs
|
|
| y <- ys
|
|
]
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\todo[inline]{Check correct with theorem support.}
|
|
|
|
\begin{Exercise}\label{ex:aesmc:1}
|
|
Write properties stating that the dot-matrix operation {\tt
|
|
gf28DotProduct} is commutative and distributive over vector
|
|
addition:
|
|
\begin{eqnarray*}
|
|
a \cdot b &=& b \cdot a \\
|
|
a \cdot (b + c) &=& a\cdot b + a\cdot b
|
|
\end{eqnarray*}
|
|
Addition over vectors is defined element-wise. Prove the commutativity
|
|
property for vectors of length 10. Distributivity will take much
|
|
longer, so you might want to do a {\tt :check} on it.\indCmdCheck
|
|
\todo[inline]{Explain why distributivity is so much harder to prove.}
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesmc:1}
|
|
\fvset{fontsize=\small}
|
|
\begin{code}
|
|
property DPComm a b = gf28DotProduct (a, b) == gf28DotProduct (b, a)
|
|
property DPDist a b c = gf28DotProduct (a, vectorAdd(b, c)) ==
|
|
gf28Add [ab, ac]
|
|
where ab = gf28DotProduct (a, b)
|
|
ac = gf28DotProduct (a, c)
|
|
vectorAdd (xs, ys) = [ gf28Add [x, y] | x <- xs
|
|
| y <- ys
|
|
]
|
|
\end{code}
|
|
We have:
|
|
\fvset{fontsize=\normalsize}
|
|
\begin{Verbatim}
|
|
AES> :prove DPComm : [10]GF28 -> [10]GF28 -> Bit
|
|
Q.E.D.
|
|
AES> :check DPDist : [10]GF28 -> [10]GF28 -> [10]GF28 -> Bit
|
|
Using random testing.
|
|
passed 1000 tests.
|
|
\end{Verbatim}
|
|
\todo[inline]{Revisit after \ticket{207} is addressed.}
|
|
% \verb+[Coverage: 0.00%. (1000/17668470647783843295832975007429185158274838968756189 ...58121606201292619776)]+
|
|
% The coverage number on {\tt DPDist} is a clear indication on how
|
|
% wild these property can get fairly quickly. (The total number of
|
|
% cases is $2^{3*10*8} = 2^{240}$, which is a truly enormous number!)
|
|
You might be surprised that the total number of cases for this
|
|
property is $2^{3*10*8} = 2^{240}$---a truly ginormous number!
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:aesmc:2}
|
|
Write a function
|
|
\begin{code}
|
|
gf28VectorMult : {n, m, k} (fin n) => ([n]GF28, [m][n]GF28) ->
|
|
[m]GF28
|
|
\end{code}
|
|
computing the dot-product of its first argument with each of the $m$
|
|
rows of the second argument, returning the resulting values as a
|
|
vector of $m$ elements.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesmc:2}
|
|
\begin{code}
|
|
gf28VectorMult (v, ms) = [ gf28DotProduct(v, m) | m <- ms ]
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:aesmc:3}
|
|
Write a function
|
|
\begin{code}
|
|
gf28MatrixMult : {n, m, k} (fin m) => ([n][m]GF28, [m][k]GF28)
|
|
-> [n][k]GF28
|
|
\end{code}
|
|
which multiplies the given matrices in GF($2^8$)\indGF.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesmc:3}
|
|
We simply need to call {\tt gfVectorMult} of the previous exercise
|
|
on every row of the first matrix, after transposing the second
|
|
matrix to make sure columns are properly aligned. We
|
|
have:\indTranspose
|
|
\begin{code}
|
|
gf28MatrixMult (xss, yss) = [ gf28VectorMult(xs, yss')
|
|
| xs <- xss ]
|
|
where yss' = transpose yss
|
|
\end{code}
|
|
\end{Answer}
|
|
|
|
\todo[inline]{Remind the reader of the context again before jumping
|
|
into \texttt{MixColumns}.}
|
|
|
|
\unparagraph Now that we have the matrix multiplication machinery
|
|
built, we can code {\tt MixColumns} fairly easily. Following the
|
|
description in the AES\indAES standard~\cite[Section 5.3.1]{aes}, all
|
|
we have to do is to multiply the matrix we have seen at the beginning
|
|
of this section with the state:
|
|
|
|
\begin{code}
|
|
MixColumns : State -> State
|
|
MixColumns state = gf28MatrixMult (m, state)
|
|
where m = [[2, 3, 1, 1],
|
|
[1, 2, 3, 1],
|
|
[1, 1, 2, 3],
|
|
[3, 1, 1, 2]]
|
|
\end{code}
|
|
|
|
Note that Cryptol makes no built-in assumption about row- or
|
|
column-ordering of multidimensional matrices. Of course, given
|
|
Cryptol's concrete syntax, it makes little sense to do anything but
|
|
row-based ordering.
|
|
|
|
%=====================================================================
|
|
% \section{Key expansion}
|
|
% \label{aes:keyexpansion}
|
|
\sectionWithAnswers{Key expansion}{aes:keyexpansion}
|
|
|
|
\todo[inline]{Will we push the pipeline of verification all the way
|
|
through? In particular, I'm interested in the equivalence proofs
|
|
between the various ticked and unticked versions of functions and
|
|
their composition.}
|
|
|
|
Recall from Section~\ref{sec:aesparams} that AES takes 128, 192, or
|
|
256 bit keys. The key is not used as-is, however. Instead, AES\indAES
|
|
expands the key into a number of round keys, called the {\em key
|
|
schedule}. Construction of the key schedule relies on a number of
|
|
auxiliary definitions, as we shall see shortly.
|
|
|
|
\paragraph*{Round constants} AES\indAES standard refers to the constant
|
|
array {\tt Rcon} used during key expansion. For each {\tt i}, {\tt
|
|
Rcon[i]} contains 4 words, the last three being 0~\cite[Section
|
|
5.2]{aes}. The first element is given by $x^{i-1}$, where
|
|
exponentiation is done using the {\tt gf28Pow} function you have
|
|
defined in Exercise~\ref{aes:subbytes}-\ref{ex:gfmi:0}. In Cryptol,
|
|
it is easiest to define {\tt Rcon} as a function:
|
|
\begin{code}
|
|
Rcon : [8] -> [4]GF28
|
|
Rcon i = [(gf28Pow (<| x |>, i-1)), 0, 0, 0]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:aeskerc:0}
|
|
By definition, AES\indAES only calls {\tt Rcon} with the parameters
|
|
ranging from 1--10. Based on this, create a table-lookup version
|
|
\begin{code}
|
|
Rcon' : [8] -> [4]GF28
|
|
\end{code}
|
|
that simply performs a look-up instead. \lhint{Use Cryptol to find out
|
|
what the elements of your table should be.}
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aeskerc:0}
|
|
Finding out the elements is easy:
|
|
\begin{Verbatim}
|
|
Cryptol> [ (Rcon i)@0 | i <- [1 .. 10] ]
|
|
[1, 2, 4, 8, 16, 32, 64, 128, 27, 54]
|
|
\end{Verbatim}
|
|
Note that we only capture the first element in each {\tt Rcon} value,
|
|
since we know that the rest are 0. We can now use this table to define
|
|
{\tt Rcon'} as follows:
|
|
\begin{code}
|
|
Rcon' i = [(rcons @ (i-1)), 0, 0, 0]
|
|
where rcons : [10]GF28
|
|
rcons = [1, 2, 4, 8, 16, 32, 64, 128, 27, 54]
|
|
\end{code}
|
|
Note that we subtract 1 before indexing into the {\tt rcons} sequence
|
|
to get the indexing right.
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:aeskerc:1}
|
|
Write and prove a property that {\tt Rcon} and {\tt Rcon'} are
|
|
equivalent when called with numbers in the range 1--10.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aeskerc:1}
|
|
We need to write a conditional property
|
|
(Section~\ref{sec:condproof})\indThmCond. Below, we use the function
|
|
{\tt elem} you have defined in
|
|
Exercise~\ref{sec:recandrec}-\ref{ex:recfun:4:1}:\indElem
|
|
\begin{code}
|
|
property RconCorrect x = if elem(x, [1..10])
|
|
then Rcon x == Rcon' x
|
|
else True
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove RconCorrect
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\todo[inline]{Will we support the \texttt{:safe} command again? See
|
|
\ticket{210}. There is a commented-out exercise here on the safety
|
|
of \texttt{Rcon'}.}
|
|
% \begin{Exercise}\label{ex:aeskerc:2}
|
|
% Is {\tt Rcon'} safe? Verify your answer using Cryptol.
|
|
% \end{Exercise}
|
|
% \begin{Answer}\ansref{ex:aeskerc:2}
|
|
% No, {\tt Rcon'} is {\em not} safe by itself:
|
|
% \begin{Verbatim}
|
|
% Cryptol> :safe Rcon'
|
|
% *** 1 safety condition to be checked.
|
|
% *** Checking for violations of: ``..: index of symbolic-value is out of bounds
|
|
% (valid range is 0 thru 9).''
|
|
% *** Violation detected:
|
|
% Rcon' 0x00
|
|
% = ..: index of 255 is out of bounds
|
|
% (valid range is 0 thru 9).
|
|
% *** 1 problem found.
|
|
% \end{Verbatim}
|
|
% The premise of {\tt Rcon'} is that it must be called with numbers in
|
|
% the range 1--10. So, all its uses are actually safe, as we shall
|
|
% explore in Exercise~\ref{aes:keyexpansion}~\ref{ex:rconsafety}.
|
|
% \end{Answer}
|
|
|
|
\paragraph*{The {\ttfamily{\textbf SubWord}} function} AES\indAES
|
|
specification refers to a function named {\tt SubWord}~\cite[Section
|
|
5.2]{aes}, that takes a 32-bit word and applies the {\tt SubByte}
|
|
transformation from Section~\ref{aes:subbytes}. This function is
|
|
trivial to code in Cryptol:
|
|
\begin{code}
|
|
SubWord : [4]GF28 -> [4]GF28
|
|
SubWord bs = [ SubByte' b | b <- bs ]
|
|
\end{code}
|
|
Note that we have used the table-lookup version ({\tt SubByte'},
|
|
Pg~\pageref{aes:subbytetl}) above.
|
|
|
|
\paragraph*{The {\ttfamily{\textbf RotWord}} function} The last
|
|
function we need for key-expansion is named {\tt RotWord} by the
|
|
AES\indAES standard~\cite[Section 5.2]{aes}. This function merely
|
|
rotates a given word cyclically to the left:
|
|
\begin{code}
|
|
RotWord : [4]GF28 -> [4]GF28
|
|
RotWord [a0, a1, a2, a3] = [a1, a2, a3, a0]
|
|
\end{code}
|
|
We could have used \verb+<<<+\indRotLeft to implement {\tt RotWord} as
|
|
well, but the above definition textually looks exactly the one given
|
|
in the standard specification, and hence is preferable for the
|
|
purposes of clarity.
|
|
|
|
\todo[inline]{Demonstrate the use of rotate and prove the
|
|
equivalence.}
|
|
|
|
\paragraph*{The key schedule} Recall that AES\indAES operates on 128,
|
|
192, or 256 bit keys. These keys are used to construct a sequence of
|
|
so called {\em round-keys}, each of which is 128-bits wide, and is
|
|
viewed the same way as the {\tt State}:\todo[inline]{\emph{Viewed} or
|
|
\emph{used}? Isn't this type abuse?}
|
|
\begin{code}
|
|
type RoundKey = State
|
|
\end{code}
|
|
The expanded key schedule, contains {\tt Nr}+1 round-keys. (Recall
|
|
from Section~\ref{sec:aesparams} that {\tt Nr} is the number of
|
|
rounds.) It also helps to separate out the first and the last keys,
|
|
as they are used in a slightly different fashion. Based on this
|
|
discussion, we use the following Cryptol type to capture the
|
|
key-schedule:
|
|
\begin{code}
|
|
type KeySchedule = (RoundKey, [Nr-1]RoundKey, RoundKey)
|
|
\end{code}
|
|
|
|
The key schedule is computed by seeding it with the initial key and
|
|
computing the successive elements from the previous entries. In
|
|
particular, the $i$th element of the expanded key is determined as
|
|
follows, copied verbatim from the AES\indAES
|
|
specification~\cite[Figure 11; Section 5.2]{aes}:
|
|
\begin{Verbatim}
|
|
temp = w[i-1]
|
|
if (i mod Nk = 0)
|
|
temp = SubWord(RotWord(temp)) xor Rcon[i/Nk]
|
|
else if (Nk > 6 and i mod Nk = 4)
|
|
temp = SubWord(temp)
|
|
end if
|
|
w[i] = w[i-Nk] xor temp
|
|
\end{Verbatim}
|
|
In the pseudo-code, the {\tt w} array contains the expanded key. We
|
|
are computing {\tt w[i]}, using the values {\tt w[i-1]} and {\tt
|
|
w[i-Nk]}. The result is the exclusive-or of {\tt w[i-Nk]} and a
|
|
mask value, called {\tt temp} above. The mask is computed using {\tt
|
|
w[i-1]}, the {\tt Rcon} array we have seen before, the current index
|
|
{\tt i}, and {\tt Nk}. This computation is best expressed as a
|
|
function in Cryptol that we will call {\tt NextWord}. We will name
|
|
the {\tt w[i-1]} argument {\tt prev}, and the {\tt w[i-Nk]} argument
|
|
{\tt old}. Otherwise, the Cryptol code just follows the pseudo-code
|
|
above, written in a functional style to compute the mask:
|
|
\begin{code}
|
|
NextWord : ([8],[4][8],[4][8]) -> [4][8]
|
|
NextWord(i, prev, old) = old ^ mask
|
|
where mask = if i % `Nk == 0
|
|
then SubWord(RotWord(prev)) ^ Rcon' (i / `Nk)
|
|
else if (`Nk > 6) && (i % `Nk == 4)
|
|
then SubWord(prev)
|
|
else prev
|
|
\end{code}
|
|
\note{It is well worth studying the pseudo-code above and the Cryptol
|
|
equivalent to convince yourself they are expressing the same idea!}
|
|
|
|
To compute the key schedule we start with the initial key as the
|
|
seed. We then make calls to {\tt NextWord} with a sliding window of
|
|
{\tt Nk} elements, computing the subsequent elements. Let us first
|
|
write a function that will apply this algorithm to generate an
|
|
infinite regression of elements:\indDrop\indTranspose
|
|
\begin{code}
|
|
ExpandKeyForever : [Nk][4][8] -> [inf]RoundKey
|
|
ExpandKeyForever seed = [ transpose g | g <- groupBy`{4} keyWS ]
|
|
where keyWS = seed # [ NextWord(i, prev, old)
|
|
| i <- [`Nk ...]
|
|
| prev <- drop`{Nk-1} keyWS
|
|
| old <- keyWS
|
|
]
|
|
\end{code}
|
|
Note how {\tt prev} tracks the previous 32-bits of the expanded key
|
|
(by dropping the first {\tt Nk-1} elements), while {\tt old} tracks
|
|
the {\tt i-Nk}'th recurrence for {\tt keyWS}. Once we have the
|
|
infinite expansion, it is easy to extract just the amount we need by
|
|
using number of rounds ({\tt Nr}) as our guide:\indIndex\indIndexs
|
|
\begin{code}
|
|
ExpandKey : [AESKeySize] -> KeySchedule
|
|
ExpandKey key = (keys @ 0, keys @@ [1 .. (Nr - 1)], keys @ `Nr)
|
|
where seed : [Nk][4][8]
|
|
seed = split (split key)
|
|
keys = ExpandKeyForever seed
|
|
\end{code}
|
|
The call {\tt split key} chops {\tt AESKey} into {\tt [Nk*4][8]}, and
|
|
the outer call to {\tt split} further constructs the {\tt [Nk][4][8]}
|
|
elements.
|
|
|
|
\paragraph*{Testing {\ttfamily{\textbf ExpandKey}}} The completion of
|
|
{\tt ExpandKey} is an important milestone in our AES\indAES
|
|
development, and it is worth testing it before we proceed. The AES
|
|
specification has example key-expansions that we can use. The
|
|
following function will be handy in viewing the output correctly
|
|
aligned:
|
|
\begin{code}
|
|
fromKS : KeySchedule -> [Nr+1][4][32]
|
|
fromKS (f, ms, l) = [ formKeyWords (transpose k)
|
|
| k <- [f] # ms # [l]
|
|
]
|
|
where formKeyWords bbs = [ join bs | bs <- bbs ]
|
|
\end{code}
|
|
Here is the example from Appendix~A.1 of the AES\indAES
|
|
specification~\cite{aes}:
|
|
\begin{Verbatim}
|
|
Cryptol> fromKS (ExpandKey 0x2b7e151628aed2a6abf7158809cf4f3c)
|
|
[[0x2b7e1516, 0x28aed2a6, 0xabf71588, 0x09cf4f3c],
|
|
[0xa0fafe17, 0x88542cb1, 0x23a33939, 0x2a6c7605],
|
|
[0xf2c295f2, 0x7a96b943, 0x5935807a, 0x7359f67f],
|
|
[0x3d80477d, 0x4716fe3e, 0x1e237e44, 0x6d7a883b],
|
|
[0xef44a541, 0xa8525b7f, 0xb671253b, 0xdb0bad00],
|
|
[0xd4d1c6f8, 0x7c839d87, 0xcaf2b8bc, 0x11f915bc],
|
|
[0x6d88a37a, 0x110b3efd, 0xdbf98641, 0xca0093fd],
|
|
[0x4e54f70e, 0x5f5fc9f3, 0x84a64fb2, 0x4ea6dc4f],
|
|
[0xead27321, 0xb58dbad2, 0x312bf560, 0x7f8d292f],
|
|
[0xac7766f3, 0x19fadc21, 0x28d12941, 0x575c006e],
|
|
[0xd014f9a8, 0xc9ee2589, 0xe13f0cc8, 0xb6630ca6]]
|
|
\end{Verbatim}
|
|
As you can verify this output matches the last column of the table in
|
|
Appendix~A.1 of the reference specification for AES.\indAES
|
|
|
|
\todo[inline]{A theorem for safety follows that we can re-introduce
|
|
after \ticket{210} is closed.}
|
|
% \begin{Exercise}\label{ex:rconsafety}
|
|
% Let us revisit Exercise~\ref{aes:keyexpansion}~\ref{ex:aeskerc:2},
|
|
% where we have seen that {\tt Rcon'} is not safe. However, the
|
|
% promise was that it would only be called with numbers ranging
|
|
% 1--10, thus all its uses would be fine. Verify this claim by
|
|
% proving {\tt ExpandKey} itself is safe. How about {\tt NextWord}?
|
|
% Is it safe?
|
|
% \end{Exercise}
|
|
% \begin{Answer}\ansref{ex:rconsafety}
|
|
% Let us first check {\tt NextWord}. We have:
|
|
% \begin{Verbatim}
|
|
% Cryptol> :safe NextWord
|
|
% *** 1 safety condition to be checked.
|
|
% *** Checking for violations of: ``..: index of symbolic-value is out of bounds
|
|
% (valid range is 0 thru 9).''
|
|
% *** Violation detected:
|
|
% NextWord (0x00, [0x00 0x00 0x00 0x00], [0x00 0x00 0x00 0x00])
|
|
% = ..: index of 255 is out of bounds
|
|
% (valid range is 0 thru 9).
|
|
% *** 1 problem found.
|
|
% \end{Verbatim}
|
|
% This is indeed to be expected, since {\tt NextWord} simply uses its
|
|
% {\tt i} argument in the call to {\tt Rcon' (i/Nk)}, and there is no
|
|
% guarantee that this {\tt i} will not be 0. However, the use from
|
|
% within {\tt ExpandKey} is perfectly fine:
|
|
% \begin{Verbatim}
|
|
% Cryptol> :safe ExpandKey
|
|
% "ExpandKey" is safe; no safety violations exist.
|
|
% \end{Verbatim}
|
|
% formally ensuring that the use of {\tt Rcon'} via {\tt NextWord} in
|
|
% {\tt ExpandKey} is indeed safe.
|
|
% \end{Answer}
|
|
|
|
%=====================================================================
|
|
\section{The {\ttfamily{\textbf AddRoundKey}} transformation}
|
|
\label{sec:ttfam-addr-transf}
|
|
|
|
{\tt AddRoundKey} is the simplest of all the transformations in
|
|
AES~\cite[Section 5.1.4]{aes}.\indAES It merely amounts to the
|
|
exclusive-or of the state and the current round key:\indXOr
|
|
\begin{code}
|
|
AddRoundKey : (RoundKey, State) -> State
|
|
AddRoundKey (rk, s) = rk ^ s
|
|
\end{code}
|
|
Notice that Cryptol's {\tt \Verb|^|} operator applies structurally to
|
|
arbitrary shapes, computing the exclusive-or element-wise.
|
|
|
|
%=====================================================================
|
|
% \section{AES encryption}
|
|
% \label{sec:aes:encryption}
|
|
\sectionWithAnswers{AES encryption}{sec:aes:encryption}
|
|
|
|
We now have all the necessary machinery to perform AES\indAES
|
|
encryption.
|
|
|
|
\paragraph*{AES rounds} As mentioned before, AES performs encryption in
|
|
rounds. Each round consists of performing {\tt SubBytes}
|
|
(Section~\ref{aes:subbytes}), {\tt ShiftRows}
|
|
(Section~\ref{aes:shiftrows}), and {\tt MixColumns}
|
|
(Section~\ref{sec:aesmixcolumns}). Before finishing up, each round
|
|
also adds the current round key to the state~\cite[Section
|
|
5.1]{aes}. The Cryptol code for the rounds is fairly trivial:
|
|
\begin{code}
|
|
AESRound : (RoundKey, State) -> State
|
|
AESRound (rk, s) = AddRoundKey (rk,
|
|
MixColumns (ShiftRows (SubBytes s)))
|
|
\end{code}
|
|
|
|
\paragraph*{The final round} The last round of AES is slightly
|
|
different than the others. It omits the {\tt MixColumns}
|
|
transformation:
|
|
\begin{code}
|
|
AESFinalRound : (RoundKey, State) -> State
|
|
AESFinalRound (rk, s) = AddRoundKey (rk, ShiftRows (SubBytes s))
|
|
\end{code}
|
|
|
|
\paragraph*{Forming the input/output blocks} Recall that AES processes
|
|
input in blocks of 128-bits, producing 128-bits of output, regardless
|
|
of the key size. We will need two helper functions to convert 128-bit
|
|
messages to and from AES states. Conversion from a message to a state
|
|
is easy to define:
|
|
\begin{code}
|
|
msgToState : [128] -> State
|
|
msgToState msg = transpose (split (split msg))
|
|
\end{code}
|
|
The first call to {\tt split} gives us 4 32-bit words, which we again
|
|
split into bytes. We then form the AES state by transposing the
|
|
resulting matrix. In the other direction, we simply transpose the
|
|
state and perform the necessary {\tt
|
|
join}'s:\indTranspose\indJoin\indReverse\indSplit
|
|
\begin{code}
|
|
stateToMsg : State -> [128]
|
|
stateToMsg st = join (join (transpose st))
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:aes:enc0}
|
|
Write and prove a pair of properties stating that {\tt msgToState}
|
|
and {\tt stateToMsg} are inverses of each other.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aes:enc0}
|
|
\begin{code}
|
|
property msgToStateToMsg msg = stateToMsg(msgToState(msg)) == msg
|
|
property stToMsgToSt s = msgToState(stateToMsg s) == s
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove msgToStateToMsg
|
|
Q.E.D.
|
|
Cryptol> :prove stToMsgToSt
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\paragraph*{Putting it together} To encrypt, AES merely expands the
|
|
given key and calls the round-functions. The starting state ({\tt
|
|
state0} below) is constructed by adding the first round key to the
|
|
input. We then run all the middle rounds using a simple comprehension,
|
|
and finish up by applying the last round~\cite[Figure 5, Section
|
|
5.1]{aes}:\indRIndex
|
|
\begin{code}
|
|
aesEncrypt : ([128], [AESKeySize]) -> [128]
|
|
aesEncrypt (pt, key) = stateToMsg (AESFinalRound (kFinal, rounds ! 0))
|
|
where (kInit, ks, kFinal) = ExpandKey key
|
|
state0 = AddRoundKey(kInit, msgToState pt)
|
|
rounds = [state0] # [ AESRound (rk, s) | rk <- ks
|
|
| s <- rounds
|
|
]
|
|
\end{code}
|
|
|
|
\paragraph*{Testing} We can now run some test vectors. Note that, just
|
|
because a handful of test vectors pass, we cannot claim that our
|
|
implementation of AES is correct.
|
|
|
|
The first example comes from Appendix~B of the AES\indAES
|
|
standard:~\cite{aes}:
|
|
\begin{Verbatim}
|
|
Cryptol> aesEncrypt (0x3243f6a8885a308d313198a2e0370734, \
|
|
0x2b7e151628aed2a6abf7158809cf4f3c)
|
|
0x3925841d02dc09fbdc118597196a0b32
|
|
\end{Verbatim}
|
|
which is what the standard asserts to be the answer. (Note that you
|
|
have to read the final box in Appendix~B column-wise!) The second
|
|
example comes from Appendix~C.1:
|
|
\begin{Verbatim}
|
|
Cryptol> aesEncrypt (0x00112233445566778899aabbccddeeff, \
|
|
0x000102030405060708090a0b0c0d0e0f)
|
|
0x69c4e0d86a7b0430d8cdb78070b4c55a
|
|
\end{Verbatim}
|
|
Again, the result agrees with the standard.
|
|
|
|
\commentout{
|
|
\begin{code}
|
|
aesEncSanityCheck = (aesEncrypt(0x3243f6a8885a308d313198a2e0370734, 0x2b7e151628aed2a6abf7158809cf4f3c) == 0x3925841d02dc09fbdc118597196a0b32) && (aesEncrypt(0x00112233445566778899aabbccddeeff, 0x000102030405060708090a0b0c0d0e0f) == 0x69c4e0d86a7b0430d8cdb78070b4c55a)
|
|
\end{code}
|
|
}
|
|
|
|
\commentout{
|
|
192 and 256 bit tests:
|
|
\begin{Verbatim}
|
|
aesEncrypt(0x00112233445566778899aabbccddeeff, 0x000102030405060708090a0b0c0d0e0f1011121314151617) == 0xdda97ca4864cdfe06eaf70a0ec0d7191
|
|
aesEncrypt(0x00112233445566778899aabbccddeeff, 0x000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f) == 0x8ea2b7ca516745bfeafc49904b496089
|
|
\end{Verbatim}
|
|
}
|
|
|
|
\paragraph*{Other key sizes} Our development of AES has been key-size
|
|
agnostic, relying on the definition of the parameter {\tt Nk} (See
|
|
Section~\ref{sec:aesparams}.) To obtain AES192, all we need is to set
|
|
{\tt Nk} to be 6, no additional code change is needed. Similarly, we
|
|
merely need to set {\tt Nk} to be 8 for AES256.
|
|
|
|
\begin{Exercise}\label{ex:aes192}
|
|
By setting {\tt Nk} to be 6 and 8 respectively, try the test vectors
|
|
given in Appendices C.2 and C.3 of the AES\indAES
|
|
standard~\cite{aes}.
|
|
\end{Exercise}
|
|
|
|
%=====================================================================
|
|
% \section{Decryption}
|
|
% \label{sec:aesdecryption}
|
|
\sectionWithAnswers{Decryption}{sec:aesdecryption}
|
|
|
|
AES decryption is fairly similar to encryption, except it uses inverse
|
|
transformations~\cite[Figure 12, Section 5.3]{aes}. Armed with all the
|
|
machinery we have built so far, the inverse transformations is
|
|
relatively easy to define.
|
|
|
|
%~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
\subsection{The {\ttfamily{\textbf InvSubBytes}} transformation}
|
|
\label{sec:ttfam-invs-transf}
|
|
|
|
The {\tt InvSubBytes} transformation reverses the {\tt SubBytes}
|
|
transformation of Section~\ref{aes:subbytes}. As with {\tt SubBytes},
|
|
we have a choice to either do a table lookup implementation, or follow
|
|
the mathematical description. We will do the former in these examples;
|
|
you are welcome to do the latter on your own and prove the equivalence
|
|
of the two versions. To do so, we need to invert the transformation
|
|
given by:
|
|
$$
|
|
b \oplus b \ggg 4 \oplus b \ggg 5 \oplus b \ggg 6 \oplus b \ggg 7 \oplus c
|
|
$$
|
|
where $c$ is {\tt 0x63}. It turns out that the inverse of this
|
|
transformation can be computed by
|
|
$$
|
|
b \ggg 2 \oplus b \ggg 5 \oplus b \ggg 7 \oplus d
|
|
$$
|
|
where $d$ is {\tt 0x05}. It is easy to code this inverse transform in
|
|
Cryptol:
|
|
|
|
\begin{code}
|
|
xformByte' : GF28 -> GF28
|
|
xformByte' b = gf28Add [(b >>> 2), (b >>> 5), (b >>> 7), d]
|
|
where d = 0x05
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:invsb:1}
|
|
Write and prove a Cryptol property stating that {\tt xformByte'} is
|
|
the inverse of the function {\tt xformByte} that you have defined in
|
|
Exercise~\ref{aes:subbytes}~\ref{ex:aessbytes:0}.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:invsb:1}
|
|
\begin{code}
|
|
property xformByteInverse x = xformByte' (xformByte x) == x
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove xformByteInverse
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\unparagraph We can now define the inverse s-box
|
|
transform,\indAESInvSbox using the multiplicative inverse function
|
|
{\tt gf28Inverse} you have defined in
|
|
Exercise~\ref{aes:subbytes}~\ref{ex:gfmi:01}:
|
|
\begin{code}
|
|
InvSubByte : GF28 -> GF28
|
|
InvSubByte b = gf28Inverse (xformByte' b)
|
|
|
|
InvSubBytes : State -> State
|
|
InvSubBytes state =[ [ InvSubByte b | b <- row ]
|
|
| row <- state
|
|
]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:invsb:2}
|
|
Write and prove a Cryptol property showing that {\tt InvSubByte}
|
|
reverses {\tt SubByte}.\indAESSbox
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:invsb:2}
|
|
\begin{code}
|
|
property sboxInverse s = InvSubBytes (SubBytes s) == s
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove xformByteInverse
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\begin{Exercise}\label{ex:invsb:3}
|
|
The AES specification provides an inverse s-box table~\cite[Figure
|
|
14, Section 5.3.2]{aes}. Write a Cryptol function {\tt InvSubBytes'}
|
|
using the table lookup technique. Make sure your implementation is
|
|
correct (i.e., equivalent to {\tt InvSubBytes}) by writing and
|
|
proving a corresponding property.
|
|
\end{Exercise}
|
|
|
|
\subsection{The {\ttfamily{\textbf InvShiftRows}} transformation}
|
|
\label{sec:ttfam-invsh-transf}
|
|
|
|
The {\tt InvShiftRows} transformation\indAESInvShiftRows simply
|
|
reverses the {\tt ShiftRows}\indAESShiftRows transformation from
|
|
Section~\ref{aes:shiftrows}:
|
|
|
|
\begin{code}
|
|
InvShiftRows : State -> State
|
|
InvShiftRows state = [ row >>> shiftAmount
|
|
| row <- state
|
|
| shiftAmount <- [0 .. 3]
|
|
]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:aesisr:0}
|
|
Write and prove a property stating that {\tt InvShiftRows} is the
|
|
inverse of {\tt ShiftRows}.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesisr:0}
|
|
\begin{code}
|
|
property shiftRowsInverse s = InvShiftRows (ShiftRows s) == s
|
|
\end{code}
|
|
We have:
|
|
\begin{Verbatim}
|
|
Cryptol> :prove shiftRowsInverse
|
|
Q.E.D.
|
|
\end{Verbatim}
|
|
\end{Answer}
|
|
|
|
\subsection{The {\ttfamily{\textbf InvMixColumns}} transformation}
|
|
\label{sec:ttfam-invm-transf}
|
|
|
|
Recall from Section~\ref{sec:aesmixcolumns} that {\tt
|
|
MixColumns}\indAESMixColumns amounts to matrix multiplication in
|
|
GF($2^8$).\indGF The inverse transform turns out to be the same,
|
|
except with a different matrix:\indAESInvMixColumns
|
|
|
|
\begin{code}
|
|
InvMixColumns : State -> State
|
|
InvMixColumns state = gf28MatrixMult (m, state)
|
|
where m = [[0x0e, 0x0b, 0x0d, 0x09],
|
|
[0x09, 0x0e, 0x0b, 0x0d],
|
|
[0x0d, 0x09, 0x0e, 0x0b],
|
|
[0x0b, 0x0d, 0x09, 0x0e]]
|
|
\end{code}
|
|
|
|
\begin{Exercise}\label{ex:aesimc:0}
|
|
Write and prove a property stating that {\tt InvMixColumns} is the
|
|
inverse of {\tt MixColumns}.
|
|
\end{Exercise}
|
|
\begin{Answer}\ansref{ex:aesimc:0}
|
|
\begin{code}
|
|
property mixColumnsInverse s = InvMixColumns (MixColumns s) == s
|
|
\end{code}
|
|
Unlike others, this property is harder to prove automatically and will
|
|
take much longer. Below we show the {\tt :check} results instead:
|
|
\begin{Verbatim}
|
|
Cryptol> :check mixColumnsInverse
|
|
Checking case 1000 of 1000 (100.00%)
|
|
1000 tests passed OK
|
|
\end{Verbatim}
|
|
|
|
\todo[inline]{Reflect upon coverage, in relation to \ticket{207}.}
|
|
% \verb+[Coverage: 0.00%. (1000/340282366920938463463374607431768211456)]+.
|
|
% The coverage information is an indication of the magnitude of the
|
|
% problem we are dealing with.}
|
|
\end{Answer}
|
|
|
|
%=====================================================================
|
|
\section{The inverse cipher}
|
|
\label{sec:inverse-cipher}
|
|
|
|
We now also have all the ingrediants to encode AES decryption.
|
|
Following Figure~12 (Section 5.3) of the AES\indAES
|
|
standard~\cite{aes}: {\small
|
|
\begin{code}
|
|
AESInvRound : (RoundKey, State) -> State
|
|
AESInvRound (rk, s) =
|
|
InvMixColumns (AddRoundKey (rk, InvSubBytes (InvShiftRows s)))
|
|
|
|
AESFinalInvRound : (RoundKey, State) -> State
|
|
AESFinalInvRound (rk, s) = AddRoundKey (rk, InvSubBytes (InvShiftRows s))
|
|
|
|
aesDecrypt : ([128], [AESKeySize]) -> [128]
|
|
aesDecrypt (ct, key) = stateToMsg (AESFinalInvRound (kFinal, rounds ! 0))
|
|
where
|
|
(kFinal, ks, kInit) = ExpandKey key
|
|
state0 = AddRoundKey(kInit, msgToState ct)
|
|
rounds = [state0] # [ AESInvRound (rk, s)
|
|
| rk <- reverse ks
|
|
| s <- rounds
|
|
]
|
|
\end{code}
|
|
}
|
|
|
|
Note how we use the results of {\tt ExpandedKey}, by carefully naming
|
|
the first and last round keys and using the middle-keys in reverse.
|
|
|
|
\paragraph*{Testing} Let us repeat the tests for AES encryption. Again,
|
|
the first example comes from Appendix~B of the AES\indAES
|
|
standard~\cite{aes}:
|
|
\begin{Verbatim}
|
|
Cryptol> aesDecrypt (0x3925841d02dc09fbdc118597196a0b32, \
|
|
0x2b7e151628aed2a6abf7158809cf4f3c)
|
|
0x3243f6a8885a308d313198a2e0370734
|
|
\end{Verbatim}
|
|
which agrees with the original value. The second example comes from
|
|
Appendix~C.1:
|
|
\begin{Verbatim}
|
|
Cryptol> aesDecrypt (0x69c4e0d86a7b0430d8cdb78070b4c55a, \
|
|
0x000102030405060708090a0b0c0d0e0f)
|
|
0x00112233445566778899aabbccddeeff
|
|
\end{Verbatim}
|
|
Again, the result agrees with the standard.
|
|
|
|
\commentout{
|
|
\begin{code}
|
|
aesDecSanityCheck = (aesDecrypt(0x3925841d02dc09fbdc118597196a0b32, 0x2b7e151628aed2a6abf7158809cf4f3c) == 0x3243f6a8885a308d313198a2e0370734) && (aesDecrypt(0x69c4e0d86a7b0430d8cdb78070b4c55a, 0x000102030405060708090a0b0c0d0e0f) == 0x00112233445566778899aabbccddeeff)
|
|
\end{code}
|
|
}
|
|
|
|
\commentout{
|
|
192 and 256 bit tests:
|
|
\begin{Verbatim}
|
|
aesDecrypt(0xdda97ca4864cdfe06eaf70a0ec0d7191, 0x000102030405060708090a0b0c0d0e0f1011121314151617) == 0x00112233445566778899aabbccddeeff
|
|
aesDecrypt(0x8ea2b7ca516745bfeafc49904b496089, 0x000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f) == 0x00112233445566778899aabbccddeeff
|
|
\end{Verbatim}
|
|
}
|
|
|
|
\paragraph*{Other key sizes} Similar to encryption, all we need to
|
|
obtain AES192 decryption is to set {\tt Nk} to be 6 in
|
|
Section~\ref{sec:aesparams}. Setting {\tt Nk} to 8 will
|
|
correspondingly give us AES256.
|
|
|
|
\paragraph*{The code} You can see all the Cryptol code for AES in
|
|
Appendix~\ref{app:aes}.
|
|
|
|
%=====================================================================
|
|
\section{Correctness}
|
|
\label{sec:aescorrectattempt}
|
|
|
|
While test vectors do provide good evidence of AES\indAES working
|
|
correctly, they do not provide a proof that we have implemented the
|
|
standard faithfully. In fact, for a block-cipher like AES, it is not
|
|
possible to state what correctness would mean. Tweaking some
|
|
parameters, or changing the s-box appropriately can give us a brand
|
|
new cipher. And it would be impossible to tell this new cipher apart
|
|
from AES aside from running it against published test vectors.
|
|
|
|
\todo[inline]{Is this claim about correctness really true?!}
|
|
|
|
What we can do, however, is gain assurance that our implementation
|
|
demonstrably has the desired properties. We have done this throughout
|
|
this chapter by stating and proving a number of properties about AES
|
|
and its constituent parts. The Cryptol methodology allows us to
|
|
construct the code together with expected properties, allowing
|
|
high-assurance development. We conclude this chapter with one final
|
|
property, stating that {\tt aesEncrypt} and {\tt aesDecrypt} do indeed
|
|
form an encryption-decryption pair:
|
|
|
|
\begin{code}
|
|
property AESCorrect msg key = aesDecrypt (aesEncrypt (msg, key), key) == msg
|
|
\end{code}
|
|
|
|
Can we hope to automatically prove this theorem? For 128-bit AES, the
|
|
state space for this theorem has $2^{256}$ elements. It would be
|
|
naive to expect that we can prove this theorem by a push-button tool
|
|
very quickly.\footnote{Note that, for a general algorithm with this
|
|
large a state space, it is entirely possible to perform automatic
|
|
verification using modern solvers, but if one carefully reflects
|
|
upon the nature of cryptographic functions, it becomes clear why it
|
|
should \emph{not} be the case here.} We can however, gain some
|
|
assurance by running it through the {\tt :check} command:\indCmdCheck
|
|
|
|
\begin{Verbatim}
|
|
Checking case 1000 of 1000 (100.00%)
|
|
1000 tests passed OK
|
|
\end{Verbatim}
|
|
% [Coverage: 0.00%. (1000/11579208923731619542357098500868790785326998466564
|
|
% ...0564039457584007913129639936)]
|
|
You will notice that even running quick-check will take a while for
|
|
the above theorem, and the total state space for this function means
|
|
that we have not even scratched the surface! That said, being able to
|
|
specify these properties together with very high level code is what
|
|
distinguishes Cryptol from other languages when it comes to
|
|
cryptographic algorithm programming.
|
|
|
|
\todo[inline]{This is highly dissatisfying!}
|
|
|
|
\todo[inline]{Add some text here back when we improve proof
|
|
capabilities?}
|
|
%% Furthermore, Cryptol has some further proof tools in its arsenal
|
|
%% that will actually let us complete this proof in a reasonable
|
|
%% amount of time, as we shall see in Section~\ref{sec:proveaes}.
|
|
\commentout{
|
|
\begin{code}
|
|
sanityCheck = (aesEncrypt (0x3243f6a8885a308d313198a2e0370734, 0x2b7e151628aed2a6abf7158809cf4f3c)
|
|
== 0x3925841d02dc09fbdc118597196a0b32)
|
|
&&
|
|
(aesEncrypt (0x00112233445566778899aabbccddeeff, 0x000102030405060708090a0b0c0d0e0f)
|
|
== 0x69c4e0d86a7b0430d8cdb78070b4c55a)
|
|
\end{code}
|
|
}
|
|
|
|
\todo[inline]{Need a punchier conclusion to the chapter, since it ends
|
|
the book, or add a new conclusion.}
|
|
|
|
%%% Local Variables:
|
|
%%% mode: latex
|
|
%%% TeX-master: "../main/Cryptol"
|
|
%%% End:
|