Appendices¶
Modular arithmetic¶
Modular arithmetic is used for many public key cryptosystems, including public-key encryption algorithms like RSA and key exchange protocols like Diffie-Hellman.
Modular arithmetic is something most people actually already understand, they just don’t know it’s called that. We can illustrate the principles of modular arithmetic using a clock.
For simplicity’s sake, our demonstration 12-hour clock only shows hours, not minutes or seconds. Also unlike real clocks, the hour hand is never halfway in between two hours: it always shows an exact hour, such as 2 or 9.
Addition and subtraction¶
It obviously makes sense to add hours on our clock: if it’s 2 o’clock now, and you’d like to know what time it is five hours from now, you can add 5, and end up with 7, as you can see in Figure 11.
Similarly, we can subtract times. If it’s 10 o’clock now, and you’d like to know what time it was two hours ago, you subtract 2 and end up with 8.
The “weird” part is when you cross the boundary at 12. As far as the clock is concerned, there’s no real difference between 12 and 0. If it’s 10 o’clock now, it’ll be 2 o’clock in four hours. If it’s 2 o’clock now, it was 9 o’clock five hours ago.
This is an example of what’s called “modular arithmetic”. The modulus, in this case, is 12. We can write the above equations as:
In these equations, the is an operator, giving the remainder after division. When we are dealing with modular arithmetic, where all operations are affected by the modulus instead of a simple single operation, we’ll instead write at the end of the equation and use an sign instead of an equals sign ():
This is read as “ten plus four is equivalent to two, modulo twelve” and “two minus five is equivalent to nine, modulo twelve”. That might seem like a trivial notational hack now, but the difference will become apparent once we start applying tricks for doing more complex modular computations, like multiplication and exponentiation.
In general, we call two numbers equivalent modulo some modulus if dividing them by the modulus leaves the same remainder. We can illustrate this with our previous examples: leaves a remainder of 2 when divided by 12, so it is equivalent to 2 modulo 12. For negative numbers, we’ll always use positive remainders. For example, . This is exactly the way a clock works as well: if it’s 2 o’clock now, then five hours ago was “nine o’clock”, not “minus three o’clock”.
Prime numbers¶
Prime numbers are wonderful kinds of numbers that come back in many branches of mathematics. Anything I say about them probably won’t do them justice; but we’re in a practical book about applied cryptography, so we’ll only see a few properties.
A prime number is a number that is divisible only by two numbers: 1 and itself. For example, 3 is a prime number, but 4 is not, because it can be divided by 2.
Any number can be written as a product of prime factors: a bunch of prime numbers multiplied together. That product is called a prime factorization. For example, 30 can be factorized into 2, 3 and 5:
Sometimes, a prime number will occur more than once in a factorization. For example, the factorization of 360 has 2 in it three times, and three in it twice:
The factorization of any prime number is just that prime number itself.
Modern mathematics no longer considers 1 to be a prime number, even though it is only divisible by 1 and itself (1 again). Under this convention, every number not only has a factorization, but that factorization is unique. Otherwise, 4 could be factored not only as , but also as , , and so on. The uniqueness of factorization helps in some important proofs in number theory.
Also, 0 is not a prime number, as it is divisible by many numbers: all numbers except 0 itself.
Two numbers are called coprime when their greatest common divisor is 1, or, to put it in another way, they don’t share any prime factors. Since the only prime factor a prime has is itself, that means that all prime numbers are also coprime. More generally, a prime is coprime to any number that isn’t a multiple of that prime.
Multiplication¶
You might remember you were first taught multiplication as repeated addition:
Modular multiplication is no different. You can compute modular multiplication by adding the numbers together, and taking the modulus whenever the sum gets larger than the modulus. You can also just do regular multiplication, and then take the modulus at the end.
Division and modular inverses¶
Division is defined as the inverse of multiplication. So, , then .
For example, ; so: . This is because , which leaves a remainder of 2 when divided by 7.
Usually, instead of using division directly, we’ll multiply using something called a modular inverse. The modular inverse of is a number, that when you multiply it with , you get 1. This is just like the inverse of a number in regular arithmetic: .
Like in regular arithmetic, not all numbers have modular inverses. This is the equivalent of dividing by zero in regular arithmetic.
There are two algorithms that are used to compute modular inverses: the extended Euclidean algorithm, and with the help of Euler’s theorem.
The extended Euclidean theorem¶
TODO: explain, and how you can get modular inverses with it
Using Euler’s theorem¶
Euler’s theorem states that if two numbers and are coprime, then:
In that equation, is Euler’s totient function, which counts the amount of numbers that are coprime to (and less than or equal to) its argument. As an example, the totient of 10 is 4, as 1, 3, 7, and 9 do not have common prime factors with 10.
We can use Euler’s theorem to find the multiplicative inverse of . If we just multiply both sides of the equation by , we get:
That gives us a direct formula for computing . Unfortunately, this is still generally less interesting than using the extended Euclidean algorithm, for two reasons:
It requires computing the totient function, which is harder than running the extended Euclidean algorithm in the first place, unless you happen to know the prime factors of .
Modular exponentiation is computationally expensive.
One exception to that rule is for prime moduli. Since a prime is coprime to every other number, and since there are numbers smaller than , . So, for a prime modulus, the modular inverse of is simply:
This still requires us to be able to efficiently raise to a power using modular arithmetic. We’ll discuss how you can do that efficiently in the next section.
Exponentiation¶
Like multiplication is taught as repeated addition, exponentiation can be thought of as repeated multiplication:
As with multiplication, it’s possible to compute modular exponentiation by performing regular exponentiation, and then taking the modulus at the end. However, this is very inefficient, particularly for large : the product quickly becomes far too large.
Fortunately, it is possible to compute modular exponentiation much more efficiently. This is done by splitting the problem up into smaller sub-problems. For example, instead of computing directly you could split it up:
is something you can compute on your hands: start at 2, which is , and then keep multiplying by two. Every time you multiply by two, the exponent goes up by 1, so by the time you’ve counted all your fingers (assuming you have ten of them), you’re done. The result is 1024. So:
Exponentiation by squaring¶
A particularly efficient way to do it on computers is splitting the
exponent up into a sum of powers of two. This is called exponentiation
by squaring, or sometimes also binary exponentiation. Suppose we want to
compute . First, we split up 209 into a sum of
powers of two. This process is essentially just writing 209 down in
binary: 0b11010001
. That’s very practical if the computation is
being performed by a computer, because that’s typically how the computer
had the number stored in the first place.
We use that expansion into a sum of powers of two to rewrite the equation:
Now, we need to compute those individual powers of 3: 1, 16, 64 and 128. A nice property of this algorithm is that we don’t actually have to compute the big powers separately from scratch. We can use previously computed smaller powers to compute the larger ones. For example, we need both and , but you can write the former in terms of the latter:
Let’s compute all the powers of 3 we need. For sake of brevity, we won’t write these out entirely, but remember that all tricks we’ve already seen to compute these still apply:
Filling these back in to our old equation:
This trick is particularly interesting when the exponent is a very large number. That is the case in many cryptographic applications. For example, in RSA decryption, the exponent is the private key , which is usually more than a thousand bits long. Keep in mind that this method will still leak timing information, so it’s only suitable for offline computation. Modular exponentiation can also be computed using a technique called a Montgomery ladder, which we’ll see in the next section.
Many programming languages provide access to specific modular
exponentiation functions. For example, in Python, pow(e, x, m)
performs efficient modular exponentiation. However, the expression
(e ** x) % m
will still use the inefficient method.
Montgomery ladder exponentiation¶
As we mentioned before, the exponentiation by squaring algorithm is simple and fast, but the time it takes to complete depends on the value of the exponent. That’s bad, because the exponent is usually a secret value, such as a Diffie-Hellman secret or the private exponent in RSA.
The Montgomery ladder is an algorithm that resolves this by guaranteeing the same number of operations irrespective of the particular value of the exponent. It was originally applied for efficient scalar multiplications over elliptic curves, but the mathematics works for many other systems: specifically, for any abelian group. [JY02]
Deriving the ladder¶
This is an optional, in-depth section. It almost certainly won’t help you write better software, so feel free to skip it. It is only here to satisfy your inner geek’s curiosity.
This section involves a good deal of arithmetic tricks. You might want to get out some paper and pencil to follow along.
Like with exponentiation by squaring, we start by looking at the binary expansion of the exponent . Generally, any can be written as a sum () of some powers of two (). If appears in the binary expansion, we’ll say that ; if it doesn’t, we’ll say that . That gives us:
That definition might look scary, but all you’re really doing here is
defining as bit of at position . The sum
goes over all the bits: if is bits long, and we
start indexing at 0, the index of the highest bit is , and
the index of the lowest bit is 0. For example, the binary expansion of
the number 6 is 0b110
. That number is three bits long, so
. So:
So, .
The next few steps don’t make a lot of sense until you see them come together at the end, so bear with me and check that the math works out. We’ll define a related sum, :
For example, (still with ) becomes:
Essentially, is just shifted to the right by bits. Shifting to the right by one bit is the same thing as flooring division by two, just like right-shifting by a decimal digit is the same thing as flooring division by 10. For example: 73, shifted one decimal digit to the right is 7; 0b101 (5) shifted one binary digit (bit) to the right is 0b10 (2). Analogously, shifting left is the inverse operation, and is equivalent to multiplying by two.
Next, we’ll perform a little arithmetical hocus pocus. First of all:
While you can verify this arithmetically, the easiest way to check this is to think of it in terms of right and left shifts. If you shift to the right by positions, that
You can visually verify that is indeed , shifted one to the left (which is the same thing as multiplying by two), plus that one bit that “fell off” when shifting right. is the last bit of ; in this case it happens to be 1, but it could equally well have been 0.
We define another very simple function :
Starting from our previous result:
We can combine these to produce an inductive way to compute and :
Remember that we’re doing this to compute . Let’s write the exponentiation out:
Remember that is right-shifted by bits, so is shifted right by 0 bits, or just itself. That means , the number we’re trying to compute, is the same thing as . By starting at ( raised to the power of the leftmost bit of ) and iteratively making our way down to , we have an elegant inductive method for computing based on two simple recursive rules.
The important part about this algorithm is the constant number of operations. If , computing involves one squaring and involves one multiplication; if , it’s the other way around. No matter what any of the bits of are, you need one squaring operation and one multiplication per bit.
Implementing the Montgomery ladder in Python¶
The Python implementation of this algorithm, applied to modular exponentiation, is surprisingly terse:
def montgomery(x, exponent, modulus):
x1, x2 = x, x ** 2
high_bit, *remaining_bits = bits(exponent)
for bit in remaining_bits:
if bit == 0:
x2 = x1 * x2
x1 = x1 ** 2
else:
x1 = x1 * x2
x2 = x2 ** 2
x1, x2 = x1 % modulus, x2 % modulus
return x1
This code block doesn’t show the definition of bits
: it produces the
binary expansion of its argument. Python doesn’t provide that by
default; bin
is close, but that produces a string: bin(100)
evaluates to 0b1100100
. The a, *b = bits(...)
construct assigns
the first item in bits(...)
to a
, and all remaining bits to
b
, effectively just skipping the first bit.
The important thing to note here is that no matter what the particular value of the exponent is, there is one squaring, one multiplication, and one modulo operation per bit. Keep in mind that this doesn’t necessarily make the entire algorithm take constant time, because the individual squaring and multiplication operations are not necessarily constant time.
Discrete logarithm¶
Just like subtraction is the inverse of addition, and division is the inverse of multiplication, logarithms are the inverse of exponentiation. In regular arithmetic, , if . This is pronounced “ raised to the power is ”, and “the logarithm of with respect to is ”. The equivalent of this in modular arithmetic is called a “discrete logarithm”.
As with division, if you start from the definition as the inverse of a different operator, it’s easy to come up with examples. For example, since , we can define . Unlike modular inverses, computing discrete logarithms is generally hard. There is no formal proof that computing discrete logarithms is intrinsically complex; we just haven’t found any efficient algorithms to do it. Because this field has gotten extensive research and we still don’t have very fast general algorithms, we consider it safe to base the security of protocols on the assumption that computing discrete logs is hard.
There is one theoretical algorithm for computing discrete logarithms efficiently. However, it requires a quantum computer, which is a fundamentally different kind of computer from the classical computers we use today. While we can build such computers, we can only build very small ones. The limited size of our quantum computers strongly limits which problems we can solve. So far, they’re much more in the realm of the kind of arithmetic a child can do in their head, than ousting the top of the line classical computers from the performance throne.
The complexity of computing discrete logarithms, together with the relative simplicity of computing its inverse, modular exponentiation, is the basis for many public key cryptosystems. Common examples include the RSA encryption primitive, and the Diffie-Hellman key exchange protocol.
While cryptosystems based on the discrete logarithm problem are currently considered secure with appropriate parameter choices, there are certainly ways that could change in the future. For example:
Theoretical breakthroughs in number theory could make discrete logarithms significantly easier to compute than we currently think.
Technological breakthroughs in quantum computing could lead to large enough quantum computers.
Technological breakthroughs in classical computing as well as the continuous gradual increases in performance and decreases in cost could increase the size of some problems that can be tackled using classical computers.
Discrete logarithm computation is tightly linked to the problem of number factorization. They are still areas of active mathematical research; the links between the two problems are still not thoroughly understood. That said, there are many similarities between the two:
Both are believed to be hard to compute on classical computers, but neither has a proof of that fact.
They can both be efficiently computed on quantum computers using Shor’s algorithm.
Mathematical advances in one are typically quickly turned into mathematical advances in the other.
Multiplicative order¶
Given integer and positive integer with gcd, the multiplicative order of is the smallest positive integer such that .
Elliptic curves¶
Like modular arithmetic, elliptic curve arithmetic is used for many public key cryptosystems. Many cryptosystems that traditionally work with modular arithmetic, such as Diffie-Hellman and DSA, have an elliptic curve counterpart.
Elliptic curves are curves with the following form:
This is called the “short Weierstrass form”, and is the most common form when talking about elliptic curves in general. There are several other forms which mostly have applications in cryptography, notably the Edwards form:
We can define addition of points on the curve.
TODO: Move the Abelian group thing somewhere else, since it applies to our fields thing as well
All of this put together form something called an Abelian group. That’s a scary-sounding mathematical term that almost everyone already understands the basics of. Specifically, if you know how to add integers () together, you already know an Abelian group. An Abelian group satisfies five properties:
If and are members of the Abelian group and is the operator, then is also a member of that Abelian group. Indeed, any two integers added together always get you another integer. This property is called closure, or, we say that the group is closed under addition (or whatever the name is of the operation we’ve defined).
If , and are members of the Abelian group, the order of operations doesn’t matter; to put it differently: we can move the brackets around. In equation form: . Indeed, the order in which you add integers together doesn’t matter; they will always sum up to the same value. This property is called associativity, and the group is said to be associative.
There’s exactly one identity element , for which . For integer addition, that’s zero: for all a.
For each element , there’s exactly one inverse element , for which , where is the identity element. Indeed, for integer addition, for all a.
The order of elements doesn’t matter for the result of the operation. For all elements , . This is known as commutativity, and the group is said to be commutative.
The first four properties are called group properties and make something a group; the last property is what makes a group Abelian.
We can see that our elliptic curve, with the point at infinity and the addition operator, forms an Abelian group:
If and are two points on the elliptic curve, then is also always a point on the curve.
If , , and are all points on the curve, then , so the elliptic curve is associative.
There’s an identity element, our point at infinity . For all points on the curve , .
Each element has an inverse element. This is easiest explained visually TODO: Explain visually
The order of operations doesn’t matter, for all on the curve.
The elliptic curve discrete log problem¶
TODO: explain fully
As with the regular discrete log problem, the elliptic curve discrete log problem doesn’t actually have a formal proof that the operation is “hard” to perform: we just know that there is no publicly available algorithm to do it efficiently. It’s possible, however unlikely, that someone has a magical algorithm that makes the problem easy, and that would break elliptic curve cryptography completely. It’s far more likely that we will see a stream of continuous improvements, which coupled with increased computing power eventually eat away at the security of the algorithm.