This project demonstrates how the TPM2.0 can be utilized as update agent in automotive (OTA) update scenarios. It is used as a “cryptographic proxy” (Security Building Block (SBB) 1) and update installation authorization coordinator (SBB2). As “cryptographic proxy”, it translates the asymmetric cryptography that is typically deployed in the vehicle environment to the symmetric cryptography that is typically used in the restricted in-vehicle network. As update installation authorization coordinator it authorizes the update installation under particular conditions, e.g., to prevent downgrade attacks or to enforce that the vehicle is in a safe state.
To show this, we create a virtual automotive network and corresponding backend server via Docker containers. The telematics control unit (TCU) is attached to the TPM via a virtualized SPI bus (vSPI), while TCU, update target ECU (ECU_U), and condition ECU (ECU_C) are connected via virtualized (Automotive) Ethernet (vAE).
graph LR
TCU[TCU<br>-Container 1-]
SWTPM[SWTPM<br>-Container 2-]
ECUU[ECU_U<br>-Container 3-]
ECUC[ECU_C<br>-Container 4-]
Backend -- Wireless --> TCU
subgraph "Vehicle"
subgraph "Update Agent"
TCU -- vSPI --> SWTPM
end
TCU -- vAE --> ECUU
TCU -- vAE --> ECUC
end
Module | Source File(s) | Description |
---|---|---|
Backend | main-backend.sh | This script is used to create and sign the update bundle and create the initial derivation secrets for TPM and ECU_U. Following the manual, all steps are executed in sequence (allcmds ). This command creates the update signing RSA keys (createbackendkeys ) with OpenSSL and creates (createupdate ) and signs (signupdate ) the update bundle. Also, it creates the derivation secret (createderivationsecret ) and the template hash (createtemplatehash ) for both TPM and ECU_U that they use in the following to derive update-specific HMAC keys. Note: The transmission of the derivation secret to the ECUs is "implicitly" modeled by copying the data directly into the corresponding Docker images |
TCU | main-tpm.sh | The TCU verifies the update bundle of the backend (verifies the signature), and, depending on a successful verification, derives a HMAC key according to the template and update hash (enforced by the TPM policy RKP). This functionality represents SBB1 (“cryptographic proxy”). Additionally, TCU coordinates the update installation by authorizing the installation only under particular conditions, e.g., to prevent downgrade attacks or to enforce that the vehicle is in a safe state. This functionality represents SBB2 (“update installation authorization coordinator”). main-tpm.sh is the main script to control the TCU functionality. First, the TPM is provisioned the derivation key with the initial authorization policy and NV index (provision ). Note: For simplicity, we also create the application policies RKP and IAP here ( createpolicies ) and sign with the backend key (signpolicies ). Normally, these steps should be done in the backend and the backend signing key should never be exposed to the vehicle. |
main_tpm.c | The main program of the TCU that is controlled by main-tpm.sh . It contains the code for provisioning the TPM (PROVISION ), creating the policies (CREATE_POLICIES ), starting the rekeying (CONDITIONAL_REKEYING ) and update installation coordination (AUTHORIZE_INSTALLATION ). |
|
fileHandler.c/.h | Auxiliary program to handle file access, e.g, write to file, append to file, read from file. | |
updateHandlerTpm.c/.h | Auxiliary program to handle TPM communication (e.g., external key loading, (derivation) key and policy creation and execution, verify signature, HMAC creation). | |
osslUtils.c | Auxiliary program to migrate OpenSSL to TPM data structures and vice versa. | |
ECU_U/ECU_C | main-ecu-u.sh/main-ecu-c.sh | ECU_U is the update target ECU that is intended to install the update and ECU_C is a ECU that will enforce restrictions to the update installation process. Both ECUs share the same code base and only the provisioning and usage step will define their role as either ECU_U or ECU_C. main-ecu-u.sh/main-ecu-c.sh is the main script the control the ECU functionality. In either case, the ECU gets provisioned with the initial derivation key (provision ). Then for ECU_U, the sbb1 procedure is started (sbb1 ). Here, ECU_U will listen for incoming update data and the TPM HMAC of the update data. After receiving update data and HMAC and storing them (main-ecu-u-update.tmp and main-ecu-u-mac-tpm.tmp), it will recompute the HMAC and compare it with the TPM HMAC. If they match, the update distribution and thus SBB1 was successful. In case of SBB2, ECU_U, sends a nonce to the TPM (sbb2 ) to check if it can install the update. The TPM will start policy processing and send a nonce to ECU_C which is started in listening mode (sbb2 ). ECU_C answers it with its provisioned key. If it is correct, the TPM can process the policy correctly and unlock the correct HMAC key to unlock the challenge from ECU_U. ECU_U verifies the challenge and, if successful, can install the update. |
main_ecu.c | The main program of ECU_U/ECU_C that is controlled by main-ecu-u.sh/main-ecu-c.sh . It contains the code for provisioning the ECU_U/ECU_C (PROVISION ). For ECU_U it contains the code for verifying the TPM HMAC in SBB1 (VERIFY_HMAC ), generating the challenge for the TPM in SBB2 (GENERATE_CHALLENGE ), and authorizing the installation in SBB2 (AUTHORIZE_INSTALLATION ). For ECU_C, it contains the code necessary in SBB2 to answer the challenge by the TPM (ANSWER_CHALLENGE ) |
|
fileHandler.c/.h | Auxiliary program to handle file access, e.g, write to file, append to file, read from file. | |
updateHandlerSw.c/.h | Auxiliary program to handle key derivation according to TPM algorithms and cryptographic operations for HMAC generation. | |
SWTPM | - | For our demo, we use a software TPM simulator. We use https://github.com/stefanberger/libtpms for the TPM software library and https://github.com/stefanberger/swtpm for the socket interface. |
To get the project working, [Docker Compose] (https://docs.docker.com/compose/) needs to be installed. The docker compose file defines all networking between the containers. These commands are the most important:
1. docker compose -f docker-compose.yml build // build all the container images
2. docker compose -f docker-compose.yml up -d // start all containers and detach
3. docker compose -f docker-compose.yml down // stop and reset all containers
Per default, the project is compiled using a software TPM (SWTPM). However, if available, also a hardware TPM can be used. To switch between both variants the TCTI needs to be specified accordingly:
- SWTPM:
char *tcti_name = "swtpm:host=10.0.0.20,port=2321"
- HWTPM:
char *tcti_name = "device:/dev/tpm0"
This script uses OpenSSL to create the asymmetric backend keys that are used to sign the update bundles. Moreover, it creates and signs the update bundle, creates the template hash (used as input for the SBB- and update-specific symmetric key derivation), the derivation secret, and the condition ECU IDs.
1. cd src
2. Create backend keys, update packages, and some auxiliary data
- bash main-backend.sh allcmds rsa
3. cd ..
Hint: To execute the whole system with ECC, replace "rsa" with "ecc" in the
bash main-backend.sh allcmds rsa
command. Depending on whether "rsa" or "ecc" is used, a corresponding flag is set, so that all following commands will automatically use the correct cryptographic algorithms and schemes.
After the backend provisioning, TCU, update target ECU (ECU_U), and condition ECU (ECU_C) are started as containers in our virtual automotive network.
1. docker compose -f docker-compose.yml build
2. docker compose -f docker-compose.yml up -d
In this step, the SSB1 (Authenticated Update Distribution) will be executed. In SBB1, the TPM is responsible for the secure transmission of the update bundle from the backend in the vehicle environment to the update target ECU(ECU_U). In this role, the TPM acts as “cryptographic proxy” that translates the asymmetric cryptography that is typically deployed in the vehicle environment to the symmetric cryptography that is typically used in the restricted in-vehicle network.
First, login into the Docker container of ECU_U and provision ECU_U with its unique derivation secret. Then start ECU_U in listening mode to wait for incoming update data.
docker exec -w /tmp/src -it update-distribution-ecuu /bin/bash
bash main-ecu-u.sh provision
bash main-ecu-u.sh sbb1
Second, login into the Docker container of the TCU and provision the TCU with its unique derivation secret, the revocation NV index, and the initial authorization policy. Then, create the update policies RKP (Rekeying Policy) and IAP (Installation Authorization Policy). Finally, start SBB1 by verifying the update bundle and conditionally deriving the HMAC key and rekeying the update bundle, which is enforced inside the TPM by processing the RKP. After successful rekeying, the TCU sends the update and HMAC to ECU_U.
docker exec -w /tmp/src -it update-distribution-tcu /bin/bash
bash main-tpm.sh provision
bash main-tpm.sh createpolicies
bash main-tpm.sh sbb1
ECU_U receives update bundle and HMAC, derives the update- and SBB-dependent HMAC key and verifies the HMAC. Upon successful verification (REKEYING SUCCESS), update is securely (unmodified) distributed from the backend via TCU (TPM) to ECU_U with our rekeying mechanism.
In this step, the SSB2 (Coordinated Update Authorization) will be executed. In SBB2, the TPM is responsible to coordinate the update installation after the update has been securely distributed with SBB1. It authorizes the update installation under particular conditions, e.g., to prevent downgrade attacks or to enforce that the vehicle is in a safe state. These conditions are requested from condition ECUs (ECU_C) in challenge-response schemes.
First, login into the Docker container of ECU_C and provision ECU_C with its unique derivation secret. Then start ECU_C in listening mode to wait for incoming challenge data.
docker exec -w /tmp/src -it update-distribution-ecuc /bin/bash
bash main-ecu-c.sh provision
bash main-ecu-c.sh sbb2
Second, also start the TCU in listening mode to wait for a challenge sent by ECU_U.
bash main-tpm.sh sbb2
Finally, on ECU_U, generate and send the challenge to TCU. Upon successful answer by the TPM, ECU_U will install the previously received (and benign) update bundle. TCU will receive the challenge and forward it to the TPM. To successfully answer the challenge, the TPM needs to authorize specific key derivation that is bound to a successful processing of IAP. IAP requires state retrieval of the condition ECU ECU_C. ECU_C answers the challenge if it is in a correct state, e.g., has enough remaining battery power, and the TPM verifies the answer. Upon successful processing of IAP, TPM authorizes key derivation and can then successfully answer the initial challenge of ECU_U. The update authorization process is successful and ECU_U can install the update (INSTALLATION SUCCESS).
bash main-ecu-u.sh sbb2