Secure Boot V2
Important
This document is about Secure Boot V2, supported on the following chips: ESP32 (ECO3 onwards), ESP32-S2, ESP32-S3, ESP32-C3 (ECO3 onwards), and ESP32-C2. Except for ESP32, it is the only supported Secure Boot scheme.
Secure Boot V2 uses ECDSA based app and bootloader verification. This document can also be used as a reference for signing apps using the ECDSA scheme without signing the bootloader.
Background
Secure Boot protects a device from running any unauthorized (i.e., unsigned) code by checking that each piece of software that is being booted is signed. On an ESP32-C2, these pieces of software include the second stage bootloader and each application binary. Note that the first stage bootloader does not require signing as it is ROM code thus cannot be changed.
A new ECC based Secure Boot verification scheme (Secure Boot V2) has been introduced on the ESP32-C2.
The Secure Boot process on the ESP32-C2 involves the following steps:
When the first stage bootloader loads the second stage bootloader, the second stage bootloader’s ECDSA signature is verified. If the verification is successful, the second stage bootloader is executed.
When the second stage bootloader loads a particular application image, the application’s ECDSA signature is verified. If the verification is successful, the application image is executed.
Advantages
The ECDSA public key is stored on the device. The corresponding ECDSA private key is kept at a secret place and is never accessed by the device.
Only one public key can be generated and stored in the chip during manufacturing.
Same image format and signature verification method is applied for applications and software bootloader.
No secrets are stored on the device. Therefore, it is immune to passive side-channel attacks (timing or power analysis, etc.)
Secure Boot V2 Process
This is an overview of the Secure Boot V2 Process. Instructions how to enable Secure Boot are supplied in section How To Enable Secure Boot V2.
Secure Boot V2 verifies the bootloader image and application binary images using a dedicated signature block. Each image has a separately generated signature block which is appended to the end of the image.
Only one signature block can be appended to the bootloader or application image in ESP32-C2
Each signature block contains a signature of the preceding image as well as the corresponding ECDSA-256 or ECDSA-192 public key. For more details about the format, refer to Signature Block Format. A digest of the ECDSA-256 or ECDSA-192 public key is stored in the eFuse.
The application image is not only verified on every boot but also on each over the air (OTA) update. If the currently selected OTA app image cannot be verified, the bootloader will fall back and look for another correctly signed application image.
The Secure Boot V2 process follows these steps:
On startup, the ROM code checks the Secure Boot V2 bit in the eFuse. If Secure Boot is disabled, a normal boot will be executed. If Secure Boot is enabled, the boot will proceed according to the following steps.
The ROM code verifies the bootloader’s signature block (Verifying a Signature Block). If this fails, the boot process will be aborted.
The ROM code verifies the bootloader image using the raw image data, its corresponding signature block(s), and the eFuse (Verifying an Image). If this fails, the boot process will be aborted.
The ROM code executes the bootloader.
The bootloader verifies the application image’s signature block (Verifying a Signature Block). If this fails, the boot process will be aborted.
The bootloader verifies the application image using the raw image data, its corresponding signature blocks and the eFuse (Verifying an Image). If this fails, the boot process will be aborted. If the verification fails but another application image is found, the bootloader will then try to verify that other image using steps 5 to 7. This repeats until a valid image is found or no other images are found.
The bootloader executes the verified application image.
Signature Block Format
The bootloader and application images are padded to the next 4096 byte boundary, thus the signature has a flash sector of its own. The signature is calculated over all bytes in the image including the padding bytes.
The content of each signature block is shown in the following table:
Offset |
Size (bytes) |
Description |
---|---|---|
0 |
1 |
Magic byte. |
1 |
1 |
Version number byte (currently 0x03). |
2 |
2 |
Padding bytes, Reserved. Should be zero. |
4 |
32 |
SHA-256 hash of only the image content, not including the signature block. |
36 |
1 |
Curve ID (1 for NIST192p curve. 2 for NIST256p curve). |
37 |
64 |
ECDSA Public key: 32 byte X coordinate followed by 32 byte Y coordinate. |
101 |
64 |
ECDSA Signature result (section 5.3.2 of RFC6090) of the image content: 32 byte R component followed by 32 byte S component. |
165 |
1031 |
Reserved. |
1196 |
4 |
CRC32 of the preceding 1196 bytes. |
1200 |
16 |
Zero padding to length 1216 bytes. |
The remainder of the signature sector is erased flash (0xFF) which allows writing other signature blocks after previous signature block.
Verifying a Signature Block
A signature block is “valid” if the first byte is 0xe7 and a valid CRC32 is stored at offset 1196. Otherwise it’s invalid.
Verifying an Image
An image is “verified” if the public key stored in any signature block is valid for this device, and if the stored signature is valid for the image data read from flash.
Compare the SHA-256 hash digest of the public key embedded in the bootloader’s signature block with the digest(s) saved in the eFuses. If public key’s hash doesn’t match any of the hashes from the eFuses, the verification fails.
Generate the application image digest and match it with the image digest in the signature block. If the digests don’t match, the verification fails.
Use the public key to verify the signature of the bootloader image, using ECDSA signature verification (section 5.3.3 of RFC6090) with the image digest calculated in step (2) for comparison.
Bootloader Size
Enabling Secure boot and/or flash encryption will increase the size of bootloader, which might require updating partition table offset. See Bootloader Size.
eFuse usage
The key(s) must be readable in order to give software access to it. If the key(s) is read-protected then the software reads the key(s) as all zeros and the signature verification process will fail, and the boot process will be aborted.
How To Enable Secure Boot V2
Open the Project Configuration Menu, in “Security features” set “Enable hardware Secure Boot in bootloader” to enable Secure Boot.
Set other menuconfig options (as desired). Then exit menuconfig and save your configuration.
The first time you run
idf.py build
, if the signing key is not found then an error message will be printed with a command to generate a signing key viaespsecure.py generate_signing_key
.
Important
A signing key generated this way will use the best random number source available to the OS and its Python installation (/dev/urandom on OSX/Linux and CryptGenRandom() on Windows). If this random number source is weak, then the private key will be weak.
Important
For production environments, we recommend generating the key pair using openssl or another industry standard encryption program. See Generating Secure Boot Signing Key for more details.
Run
idf.py bootloader
to build a Secure Boot enabled bootloader. The build output will include a prompt for a flashing command, usingesptool.py write_flash
.When you’re ready to flash the bootloader, run the specified command (you have to enter it yourself, this step is not performed by the build system) and then wait for flashing to complete.
Run
idf.py flash
to build and flash the partition table and the just-built app image. The app image will be signed using the signing key you generated in step 6.
Note
idf.py flash
doesn’t flash the bootloader if Secure Boot is enabled.
Reset the ESP32-C2 and it will boot the software bootloader you flashed. The software bootloader will enable Secure Boot on the chip, and then it verifies the app image signature and boots the app. You should watch the serial console output from the ESP32-C2 to verify that Secure Boot is enabled and no errors have occurred due to the build configuration.
Note
Secure boot won’t be enabled until after a valid partition table and app image have been flashed. This is to prevent accidents before the system is fully configured.
Note
If the ESP32-C2 is reset or powered down during the first boot, it will start the process again on the next boot.
On subsequent boots, the Secure Boot hardware will verify the software bootloader has not changed and the software bootloader will verify the signed app image (using the validated public key portion of its appended signature block).
Restrictions after Secure Boot is enabled
Any updated bootloader or app will need to be signed with a key matching the digest already stored in eFuse.
After Secure Boot is enabled, no further eFuses can be read protected. (If Flash Encryption is enabled then the bootloader will ensure that any flash encryption key generated on first boot will already be read protected.) If CONFIG_SECURE_BOOT_INSECURE is enabled then this behavior can be disabled, but this is not recommended.
Generating Secure Boot Signing Key
The build system will prompt you with a command to generate a new signing key via espsecure.py generate_signing_key
.
Select the ECDSA scheme by passing --version 2 --scheme ecdsa256
or --version 2 --scheme ecdsa192
to generate corresponding ECDSA private key
The strength of the signing key is proportional to (a) the random number source of the system, and (b) the correctness of the algorithm used. For production devices, we recommend generating signing keys from a system with a quality entropy source, and using the best available ECDSA key generation utilities.
For example, to generate a signing key using the openssl command line:
For NIST192p curve
`
openssl ecparam -name prime192v1 -genkey -noout -out my_secure_boot_signing_key.pem
`
For NIST256p curve
`
openssl ecparam -name prime256v1 -genkey -noout -out my_secure_boot_signing_key.pem
`
Remember that the strength of the Secure Boot system depends on keeping the signing key private.
Remote Signing of Images
For production builds, it can be good practice to use a remote signing server rather than have the signing key on the build machine (which is the default esp-idf Secure Boot configuration). The espsecure.py command line program can be used to sign app images & partition table data for Secure Boot, on a remote system.
To use remote signing, disable the option “Sign binaries during build”. The private signing key does not need to be present on the build system.
After the app image and partition table are built, the build system will print signing steps using espsecure.py:
espsecure.py sign_data --version 2 --keyfile PRIVATE_SIGNING_KEY BINARY_FILE
The above command appends the image signature to the existing binary. You can use the –output argument to write the signed binary to a separate file:
espsecure.py sign_data --version 2 --keyfile PRIVATE_SIGNING_KEY --output SIGNED_BINARY_FILE BINARY_FILE
Secure Boot Best Practices
Generate the signing key on a system with a quality source of entropy.
Keep the signing key private at all times. A leak of this key will compromise the Secure Boot system.
Do not allow any third party to observe any aspects of the key generation or signing process using espsecure.py. Both processes are vulnerable to timing or other side-channel attacks.
Enable all Secure Boot options in the Secure Boot Configuration. These include flash encryption, disabling of JTAG, disabling BASIC ROM interpreter, and disabling the UART bootloader encrypted flash access.
Use Secure Boot in combination with flash encryption to prevent local readout of the flash contents.
Technical Details
The following sections contain low-level reference descriptions of various Secure Boot elements:
Manual Commands
Secure boot is integrated into the esp-idf build system, so idf.py build
will sign an app image and idf.py bootloader
will produce a signed bootloader if secure signed binaries on build is enabled.
However, it is possible to use the espsecure.py
tool to make standalone signatures and digests.
To sign a binary image:
espsecure.py sign_data --version 2 --keyfile ./my_signing_key.pem --output ./image_signed.bin image-unsigned.bin
Keyfile is the PEM file containing an ECDSA-256 or ECDSA-192 private signing key.
Secure Boot & Flash Encryption
If Secure Boot is used without Flash Encryption, it is possible to launch “time-of-check to time-of-use” attack, where flash contents are swapped after the image is verified and running. Therefore, it is recommended to use both the features together.
Important
ESP32-C2 has only one eFuse key block, which is used for both keys: Secure Boot and Flash Encryption. The eFuse key block can only be burned once. Therefore these keys should be burned together at the same time. Please note that “Secure Boot” and “Flash Encryption” can not be enabled separately as subsequent writes to eFuse key block shall return an error.
Signed App Verification Without Hardware Secure Boot
The Secure Boot V2 signature of apps can be checked on OTA update, without enabling the hardware Secure Boot option. This option uses the same app signature scheme as Secure Boot V2, but unlike hardware Secure Boot it does not prevent an attacker who can write to flash from bypassing the signature protection.
This may be desirable in cases where the delay of Secure Boot verification on startup is unacceptable, and/or where the threat model does not include physical access or attackers writing to bootloader or app partitions in flash.
In this mode, the public key which is present in the signature block of the currently running app will be used to verify the signature of a newly updated app. (The signature on the running app isn’t verified during the update process, it’s assumed to be valid.) In this way the system creates a chain of trust from the running app to the newly updated app.
For this reason, it’s essential that the initial app flashed to the device is also signed. A check is run on app startup and the app will abort if no signatures are found. This is to try and prevent a situation where no update is possible. The app should have only one valid signature block in the first position. Note again that, unlike hardware Secure Boot V2, the signature of the running app isn’t verified on boot. The system only verifies a signature block in the first position and ignores any other appended signatures.
Although multiple trusted keys are supported when using hardware Secure Boot, only the first public key in the signature block is used to verify updates if signature checking without Secure Boot is configured. If multiple trusted public keys are required, it’s necessary to enable the full Secure Boot feature instead.
Note
In general, it’s recommended to use full hardware Secure Boot unless certain that this option is sufficient for application security needs.
How To Enable Signed App Verification
Open Project Configuration Menu -> Security features
Ensure App Signing Scheme is ECDSA (V2)
By default, “Sign binaries during build” will be enabled on selecting “Require signed app images” option, which will sign binary files as a part of build process. The file named in “Secure boot private signing key” will be used to sign the image.
If you disable “Sign binaries during build” option then all app binaries must be manually signed by following instructions in Remote Signing of Images.
Warning
It is very important that all apps flashed have been signed, either during the build or after the build.
Advanced Features
JTAG Debugging
By default, when Secure Boot is enabled then JTAG debugging is disabled via eFuse. The bootloader does this on first boot, at the same time it enables Secure Boot.
See JTAG with Flash Encryption or Secure Boot for more information about using JTAG Debugging with either Secure Boot or signed app verification enabled.