This is part 4 in a multi-part report describing how we implemented Quantum Key Distribution (QKD) in OpenSSL as part of the pan-European quantum Internet hackathon in Delft on 5 and 6 November 2019. See the main page of this report for the other parts. -- Bruno Rijsman
The work that we did to add Quantum Key Distribution (QKD) support to OpenSSL was based on the OpenSSL integration challenge that was designed by Wojciech Kozlowski for the Pan-European Quantum Internet Hackathon hosted at QuTech at the Delft University of Technology on 5 and 6 November, 2019.
The challenge consists of two parts:
-
Add QKD support to OpenSSL by invoking the ETSI QKD API. Here OpenSSL becomes a consumer of the QKD API. The hackathon organizers had provided a mock implementation of the QKD API for testing purposes.
-
Implement a specific QKD protocol, namely Bennett and Brassard 1984 (BB84) , on top of the SimulaQron quantum network simulator. Here the BB84 implementation becomes a provider of the QKD API.
The end-goal of the challenge is to use an off-the-shelf browser (e.g. Firefox) and connect it to a secure HTTPS website hosted on an off-the-shelf web server (e.g. Apache), while using the BB84 quantum key distribution algorithm as the key agreement protocol (running a SimulaQron simulated quantum network), instead of the classical Diffie-Hellman protocol that is normally used in classical networks.
During the hackathon, the team that I was a part of worked on part 1 of the challenge. This page describes that work (which was successfully completed). Another team worked on part 2 of the challenge, which they partially completed. The end-goal (i.e. the integration between part 1 and 2) was not yet achieved. It is my intention to complete the work for part 2, to complete the integration, and to update this document once that work is completed.
There are two approaches to adding support for QKD in OpenSSL using the ETSI QKD API:
- Hacking the existing engine-based extension mechanism for Diffie-Hellman.
- Introducing QKD as a new first-class key exchange protocol.
OpenSSL has a mechanism, called "engines", that allows third parties (such as myself) to add extensions to OpenSSL. These extensions can be implemented as dynamic libraries (.so files on Linux or .dylib files on macOS) which allows the extensions to be dynamically loaded into OpenSSL without make any changes to the source code of OpenSSL itself. The OpenSSL configuration file controls which extensions are loaded into OpenSSL.
The main purpose of OpenSSL engines to allow time-consuming cryptographic operations to be offloaded from the default software implementation in OpenSSL into special-purpose crypto acceleration hardware. OpenSSL has made some a-priori decisions about which operations make sense to be offloaded. The OpenSSL engine framework provides APIs to allow those specific operations, and only those specific operations, to be offloaded.
The engine APIs allow the dynamically loaded engine to register a callback function that OpenSSL should call whenever a particular expensive operations needs to be performed. This registered function is called instead of the default software implementation in OpenSSL, and it is expected to implement the same function in a more efficient manner on special-purpose hardware.
In our current implementation of QKD support in OpenSSL, we have used the engine mechanism in a creative way to avoid having to change the OpenSSL source code (see approach 2 below). A less charitable way of saying it is that our current implementation is a hack.
Our QKD engine overloaded two callback functions that were really intended to implement hardware acceleration for the Diffie-Hellman (DH) key exchange algorithm:
-
The Diffie-Hellman
compute_key
engine callback is really intended to choose a Diffie-Hellman private key and to compute the corresponding public key, using the negotiated Diffie-Hellman g and p parameters. We have hacked this callback and we have hijacked the public key to instead communicate an ETSI QKDkey_handle
from the QKD server to the QKD client. -
The Diffie-Hellman
generate_key
engine callback is really intended to generate a Diffie-Hellman shared secret using the end-point's own private key, the peer's public key, and the negotiated Diffie-Hellman g and p parameters. We have hacked this callback to instead retrieve the shared secret from the QKD key management layer using the ETSI QKD GET_KEY API.
The following diagram illustrates this first approach of hacking the existing engine-based extension mechanism for Diffie-Hellman. A more detailed description of the actual code is given below.
As a result of how engines are implemented in OpenSSL (see the description above), OpenSSL engines have their limitations. They can only be used to accelerate a pre-determined set of operations in existing crypto algorithms. They cannot be used to introduce, for example, a completely new key exchange algorithms such as QKD.
We got away with introducing QKD support without changing the OpenSSL source code by "hacking" the Diffie-Hellman engine as summarized above. Admittedly, this is not the proper way to implement QKD. The resulting code is ugly and fragile (see the challenges section below for more details)). The proper thing to do would have been to change the OpenSSL code to introduce QKD as a first-class abstraction in OpenSSL. We would still need an engine (for QKD, not for "fake" Diffie-Hellman) because the QKD provider is typically some external device reached through the ETSI API.
Who knows, maybe sometime in the future in some remote South-America town, I will find myself with some spare time on my hands, do a "proper" implementation, and update this repository and report accordingly.
The plan of attack for the hackathon was to split the work across two teams:
-
One team would work on implementing an engine for OpenSSL to consume the ETSI QKD API for doing an QKD-based key exchange in OpenSSL. This was the team that I was a part of.
-
Another team would work on implementing the ETSI QKD API by implementing the BB84 QKD protocol on top of the SimulaQron quantum network simulator using the SimulaQron CQC API. (CQC is simply the name of the C-language binding of the API that is used to talk to SimulaQron).
Thus, whereas team 1 is a consumer of the ETSI QKD API, team 2 is a provider of the ETSI QKD API.
To allow the two teams to work in parallel, team 1 would initially use a "mock" implementation of the ETSI QKD API. This is shown on the left side in the diagram below. In this mock implementation there is a "fake" ETSI QKD API provider that simply allocates a random shared secret and send it "in the clear" over a TCP connection to the other side.
Once both team 1 and team 2 are finished, the two components (the consumer and the provider) can be integrated together as show on the right side in the diagram below.
At this point in time, we have only completed the left side of the diagram. The right side, i.e. the integration, has not yet been achieved. But I plan to complete that integration in the future and update this repository and this report accordingly.
This section describes in detail how we "hacked" the existing engine-based extension mechanism for Diffie-Hellman to add support for QKD in OpenSSL on top of the ETSI QKD API.
The following resources are helpful for understanding how engines work in OpenSSL in general and for understanding the code in this repository in specific:
- OpenSSL wiki main page
- OpenSSL wiki libcrypto main page
- OpenSSL wiki Diffie-Hellman
- OpenSSL wiki example ECDH engine
- OpenSSL wiki SSL and TLS tutorial
- OpenSSL man page for engines
- OpenSSL man page for DH_generate_key engine callback
- OpenSSL man page for DH_compute_key engine callback
- Gost-engine/engine GitHub repo containing an OpenSSL engine implementation
The following diagram shows how the ETSI QKD API is intended to be used (see section 6.1.1. of the ETSI API for details):
Notice that the server side uses the API in a slightly different way than the client. For example, the server calls QKD_OPEN with a NULL key_handle (which causes a new key_handle to be allocated) whereas the client calls QKD_OPEN with the key_handle that was allocated on the server side. However, the DH callbacks in the engine have no way of finding out whether they are being called on the server side or on the client side. For this reason we ended up having to create two separate engines: one for the server side and a different one for the client side.
The following table summarizes how the server maps OpenSSL engine DH callbacks to ETSI QKD API calls:
Engine callback | Intended purpose | Actual implementation |
---|---|---|
init | Initialize the engine | Call QKD_init |
generate_key | Generate the DH private key and derive the DH public key from it | Use fixed value for the DH private key. Call QKD_OPEN and use returned key_handle as DH public key. |
compute_key | Compute the shared_secret given the remote peer's public key | Retrieve our own (i.e. the server's) DH public key as the key_handle. Call QKD_CONNECT_BLOCKING. Call QKD_GET_KEY and use the returned QKD key as the DH shared secret. Call QKD_CLOSE. |
The following table summarizes how the client maps OpenSSL engine DH callbacks to ETSI QKD API calls:
Engine callback | Intended purpose | Actual implementation |
---|---|---|
init | Initialize the engine | Call QKD_init |
generate_key | Generate the DH private key and derive the DH public key from it | Use fixed values for the DH private key and the DH public key. |
compute_key | Compute the shared_secret given the remote peer's public key | Use the peer's (i.e. the server's) DH public key as the key_handle. Call QKD_OPEN. Call QKD_CONNECT_BLOCKING. Call QKD_GET_KEY and use the return QKD key as the DH shared secret. Call QKD_CLOSE. |
The following files contain our implementation of the OpenSSL engine for QKD:
qkd_engine_common.c
andqkd_engine_common.h
contain the engine code that is common to the server and the client.qkd_engine_server.c
contains the code that is unique to the server (it implements the server mapping table shown above).qkd_engine_client.c
contains the code that is unique to the client (it implements the client mapping table shown above).
Note that this engine code doesn't know or care whether the QKD implementation is (a) a mock QKD implementation or (b) an implementation of real BB84 QKD running on a simulated quantum network using SimulaQron or even (c) implementation of real QKD running over a real quantum network running some commercially available real QKD device.
This sections describes in details on we created a "mock" (i.e. fake) implementation of the ETSI QKD API that allows us to test OpenSSL QKD without using any quantum network (neither a real quantum network nor a simulated quantum network).
Behavior of the mock ETSI QKD API on the server side:
ETSI QKD API | Mock implementation |
---|---|
QKD_INIT | Listen for incoming mock QKD connection. |
QKD_OPEN | Allocate and remember a new key_handle. |
QKD_CONNECT_BLOCKING | Accept incoming QKD connection from client. Receive client key_handle over QKD connection. Verify that client key_handle is same as server key_handle. |
QKD_CONNECT_NONBLOCK | Not implemented yet. |
QKD_CONNECT_GET_KEY | Receive the shared_secret from the client over the QKD connection. |
QKD_CONNECT_CLOSE | Close the QKD connection. |
Behavior of the mock ETSI QKD API on the client side:
ETSI QKD API | Mock implementation |
---|---|
QKD_INIT | - |
QKD_OPEN | Remember the provided key_handle. |
QKD_CONNECT_BLOCKING | Create outgoing QKD connection to server. Send key_handle over QKD connection to the server.. |
QKD_CONNECT_NONBLOCK | Not implemented yet. |
QKD_CONNECT_GET_KEY | Allocate a random shared_secret. Send the shared_secret over the QKD connection to the server. |
QKD_CONNECT_CLOSE | Close the QKD connection. |
The exchange of key handles over the QKD connection is done only for verification / debugging reasons. We just want to make sure that the client is connected to the right server and vice versa.
(*) See the challenges section for an explanation why the client side choses the shared secret and send it to the server instead of vice versa, what would have seemed more natural.
Note that the mock QKD protocol is asymmetric. One side generates the shared secret and provides it to the other side. Once again, we see that the mock API needs to know whether it is running on the server side or on the client side. The current implementation does this by assuming that the server will pass a NULL destination to the QKD_OPEN call ("accept incoming QKD sessions from any client) whereas the client will pass a non-NULL destination to the QKD_OPEN call ("create a QKD session to a specific server").
The file qkd_api.h
contains the interface definition of the ETSI QKD API. This file remains the same, regardless of whether the underlying API implementation is (a) a mock API or (b) a real BB84 QKD implementation running on a simulated quantum network using SimulaQron or (c) a real QKD implementation running on a real quantum network using real commercially available QKD devices or (d) anything else.
The file qkd_api_mock.c
implements the ETSI QKD API as defined in the qkd_api.h
header file. It maps the API functions to the behavior described in the two table above.
Both the qkd_api.h
and qkd_api_mock.c
file in this repository are based on the original mock API that was provided by the hackathon organizers.
There are a few remaining major TODO items in the mock API implementation:
-
Support the server and the client running on different computers.
-
On the server, support more than one concurrent client session.
-
Add support for QKD_CONNECT_NONBLOCK.
We encountered the following challenges:
-
The engine's behavior is different on the server and client side, but the engine callback functions have no way of finding out whether they are running on the server or the client. Solution: we created two separate engines, one for the server and one for the client.
-
During the implementation of the mock API we ran into an interesting deadlock problem, namely:
- In an earlier attempted implementation, it used to be the server who chose the random shared secret and sent it over the TCP connection to the client, by doing a blocking write on the TCP socket.
- This turned out to cause a deadlock because the OpenSSL implementation of the client side is as follows
- The client receives a Server Hello Message, which contains the Diffie-Hellman parameters and public key from the server.
- The client chooses its own Diffie-Hellman private key, using the received Diffie-Hellman parameters.
- The client calls generate_key to compute the shared secret based on (a) the received Diffie-Hellman paramters, (b) the received server's public key, and (c) it's own, i.e. the client's private key.
- Then, and only then, does the client send the Client Key Exchange message to the server, which contains the client's public key.
- The server can only call generate_key after it has received the Client Key Exchange message.
- Now, if the server chooses the key, then the client will do a blocking receive in step 3 (c) to receive the shared secret from the server, but that message will never come because it can only be sent in step 5 (e).
- We fixed this by reversing who chooses the shared secret (the key): in the current code the client choses the shared secret and sends it to the server, instead of the other way around (the latter causes the deadlock).
- This design is fragile because the mock API makes assumptions about the order in which the API functions are called.
- This, in turn, is a side-effect of abusing the OpenSSL Diffie-Hellman callbacks to "hack QKD into OpenSSL". If and when we introduce QKD as a first-class key exchange mechanism with its own APIs and its own engine, then we will design those QKD APIs and callbacks in a matter that maps better to the QKD use cases.
The code in the repository as been tested on Apple macOS and on Ubuntu Linux 18.04 LTS in an Amazon Web Services (AWS) instance.
The instructions for installing, building, and running the code on Ubuntu Linux 18.04 LTS in an AWS instance as as follows:
1.1. Launch an Ubuntu Linux 18.04 LTS instance in AWS using the default parameters. A t2.tiny instance is sufficient.
1.2. SSH login to the instance (the default username for AWS Ubuntu instances is "ubuntu").
ssh -i <your-private-key-file.pem> ubuntu@<ip-address-of-your-aws-instance>
1.3. Install the build essentials:
sudo apt-get update
sudo apt-get install -y build-essential
1.4. Install Python:
sudo apt-get install -y python
Installing tshark is kind of a hassle. We can't use apt-get
because it installs version 2.x of tshark; we need version 3.x of wireshark.
These steps also install wireshark, which we don't need for the Makefile, but it is handy to have it if you are running on a desktop Ubuntu.
Thanks to Consent Factory for describing the installation steps.
2.1. Install the dependencies for tshark (this is all one long line):
sudo apt-get install -y build-essential pkg-config ninja-build bison flex qt5-default qttools5-dev-tools qtcreator ninja-build libpcap-dev cmake libglib2.0-dev libgcrypt20-dev qttools5-dev qtmultimedia5-dev libqt5svg5-dev
2.2. Create the Wireshark group and add yourself to the group:
sudo groupadd -g 62 wireshark
sudo usermod -a -G wireshark ubuntu
2.3. Grab and extract the wireshark version 3.0.6 tarball:
cd ~
wget https://www.wireshark.org/download/src/all-versions/wireshark-3.0.6.tar.xz
2.4. Extract the tarball and move into the directory:
tar xf wireshark-3.0.6.tar.xz
cd wireshark-3.0.6
2.5. Build the tshark and wireshark applications. This will take a whopping 20 minutes or so, so grab a cup of coffee...
mkdir build &&
cd build &&
cmake -DCMAKE_INSTALL_PREFIX=/usr \
-DCMAKE_BUILD_TYPE=Release \
-DCMAKE_INSTALL_DOCDIR=/usr/share/doc/wireshark-3.0.6 \
-G Ninja \
.. &&
ninja
2.6. Install the built wireshark application:
sudo ninja install &&
sudo install -v -m755 -d /usr/share/doc/wireshark-3.0.6 &&
sudo install -v -m644 ../README.linux ../doc/README.* ../doc/{*.pod,randpkt.txt} \
/usr/share/doc/wireshark-3.0.6 &&
pushd /usr/share/doc/wireshark-3.0.6 &&
for FILENAME in ../../wireshark/*.html; do
sudo ln -s -v -f $FILENAME .
done &&
popd
unset FILENAME
2.7. Secure the application directories:
sudo chown -v ubuntu:wireshark /usr/bin/{tshark,dumpcap} &&
sudo chmod -v 6550 /usr/bin/{tshark,dumpcap}
2.8. Allow tshark to capture interface traffic. Answer "yes" to the pop-up question "Should non-superusers be able to capture packets?"
sudo dpkg-reconfigure wireshark-common
Technically, we don't need to install the OpenSSL source code. The OpenSSL engine mechanism allows us to dynamically load an engine shared library into the OpenSSL binary that is already installed on the instance. Still, we install the OpenSSL source code since reading the source code is essential to understanding how OpenSSL works, and because we will end up changing the OpenSSL source code if and when we introduce QKD as a first-class key exchange protocol at some point in the future.
3.1. Clone OpenSSL from GitHub into the home directory of your AWS instance:
cd ~
git clone https://github.com/openssl/openssl.git
3.2. Configure and build the cloned OpenSSL code. Expect the make
step to take about 5 minutes. Expect the make test
step to take a bit more than 2 minutes. A few test cases (md2, rc5, gost, etc.) will be skipped, but all other test cases should report "ok". Do not run make install
(we don't want to overwrite the default OpenSSL library).
cd ~/openssl
./config
make
make test
4.1. Clone openssl-qkd from GitHub into the home directory of your AWS instance:
cd ~
git clone https://github.com/brunorijsman/openssl-qkd.git
4.2. Build the openssl-qkd code:
cd ~/openssl-qkd
make
4.3. Run the openssl-qkd test:
cd ~/openssl-qkd
make test
You should see the following at the end out the output:
Checking tshark output for correct Diffie-Helman exchange... OK
Congratulations! You have successfully run a (mock) QKD key exchange between an HTTP client and an HTTP server using OpenSSL and the ETSI QKD API!
Note: currently the test sometimes fails with the following error message. This is a known issue that will be fixed (hopefully) soon. Just run the test again.
Checking tshark output for correct Diffie-Helman exchange... Did not match pattern #0: Connection establish request \(SYN\): server port 44330
The unit test / demo involves the following components:
-
The OpenSSL demo HTTPS web server (
openssl s_server
). It waits for incoming HTTPS GET requests and produces a web page with some OpenSSL statistics in response. In our unit test we use an OpenSSL configuration file that dynamically loads the server-side mock QKD engine. -
The OpenSSL demo HTTPS web client (
openssl s_client
). It sends an HTTPS GET requests to the web server and prints the response. In our unit test we use an OpenSSL configuration file that dynamically loads the client-side mock QKD engine. -
A tshark process that captures all network packets between the client and the server. It produces a tshark.out file which contains the decoded traffic in human-readable form. Our unit test script parses this tshark.out file to check whether the expected TLS handshake messages are actually observed. It also produces a binary tshak.pcap file which can be loaded into WireShark for interactive investigation of the TLS exchange.
The unit test / demo can be run using make test
or invoking the run_mock_test.sh
shell script. Either way, it does the following:
-
Stop any HTTPS server process that might still be running in the background from a previous run. (Script
stop_server.sh
) -
Start a new HTTPS server process in the background, listening for incoming HTTPS GET requests. (Script
start_server.sh
) -
Start a tshark process in the background to capture the traffic between the client and server. (Script
start_tshark.sh
) -
Run a HTTP client process to send a single HTTPS GET request to the server and print the response. (Script
run_client.sh
) -
Stop the tshark process running in the background. (Script
stop_tshark.sh
) -
Stop the HTTPS server process running in the background. (Script
stop_server.sh
) -
Analyze the captured and decoded traffic file
tshark.out
and verify that the expected TLS message were exchanged. (Scriptcheck_shark.py
)