Flash Encryption
This is a quick start guide to ESP32-C6’s flash encryption feature. Using application code as an example, it demonstrates how to test and verify flash encryption operations during development and production.
Introduction
Flash encryption is intended for encrypting the contents of the ESP32-C6’s off-chip flash memory. Once this feature is enabled, firmware is flashed as plaintext, and then the data is encrypted in place on the first boot. As a result, physical readout of flash will not be sufficient to recover most flash contents.
With flash encryption enabled, the following types of data are encrypted by default:
Firmware bootloader
Partition Table
All “app” type partitions
Other types of data can be encrypted conditionally:
Any partition marked with the
encrypted
flag in the partition table. For details, see Encrypted Partition Flag.Secure Boot bootloader digest if Secure Boot is enabled (see below).
Important
For production use, flash encryption should be enabled in the “Release” mode only.
Important
Enabling flash encryption limits the options for further updates of ESP32-C6. Before using this feature, read the document and make sure to understand the implications.
Relevant eFuses
The flash encryption operation is controlled by various eFuses available on ESP32-C6. The list of eFuses and their descriptions is given in the table below. The names in eFuse column are also used by espefuse.py tool. For usage in the eFuse API, modify the name by adding ESP_EFUSE_
, for example: esp_efuse_read_field_bit(ESP_EFUSE_DISABLE_DL_ENCRYPT).
eFuse |
Description |
Bit Depth |
|
AES key storage. N is between 0 and 5. |
256 bit key block |
|
Control the purpose of eFuse block |
4 |
|
If set, disable flash encryption when in download bootmodes. |
1 |
|
Enable encryption and decryption, when an SPI boot mode is set. Feature is enabled if 1 or 3 bits are set in the eFuse, disabled otherwise. |
3 |
Note
R/W access control is available for all the eFuse bits listed in the table above.
The default value of these bits is 0 afer manufacturing.
Read and write access to eFuse bits is controlled by appropriate fields in the registers WR_DIS
and RD_DIS
. For more information on ESP32-C6 eFuses, see eFuse manager. To change protection bits of eFuse field using espefuse.py, use these two commands: read_protect_efuse and write_protect_efuse. Example espefuse.py write_protect_efuse DISABLE_DL_ENCRYPT
.
Flash Encryption Process
Assuming that the eFuse values are in their default states and the firmware bootloader is compiled to support flash encryption, the flash encryption process executes as shown below:
On the first power-on reset, all data in flash is un-encrypted (plaintext). The ROM bootloader loads the firmware bootloader.
Firmware bootloader reads the
SPI_BOOT_CRYPT_CNT
eFuse value (0b000
). Since the value is0
(even number of bits set), it configures and enables the flash encryption block. For more information on the flash encryption block, see ESP32-C6 Technical Reference Manual.Firmware bootloader uses RNG (random) module to generate an 256 bit key and then writes it into BLOCK_KEYN eFuse. The software also updates the
KEY_PURPOSE_N
for the block where the key is stored. The key cannot be accessed via software as the write and read protection bits for BLOCK_KEYN eFuse are set.KEY_PURPOSE_N
field is write-protected as well. The flash encryption is completely conducted by hardware, and the key cannot be accessed via software.Flash encryption block encrypts the flash contents - the firmware bootloader, applications and partitions marked as
encrypted
. Encrypting in-place can take time, up to a minute for large partitions.Firmware bootloader sets the first available bit in
SPI_BOOT_CRYPT_CNT
(0b001) to mark the flash contents as encrypted. Odd number of bits is set.For Development Mode, the firmware bootloader allows the UART bootloader to re-flash encrypted binaries. Also, the
SPI_BOOT_CRYPT_CNT
eFuse bits are NOT write-protected. In addition, the firmware bootloader by default sets the eFuse bitsDIS_DOWNLOAD_ICACHE
,DIS_PAD_JTAG
,DIS_USB_JTAG
andDIS_LEGACY_SPI_BOOT
.For Release Mode, the firmware bootloader sets all the eFuse bits set under development mode as well as
DIS_DOWNLOAD_MANUAL_ENCRYPT
. It also write-protects theSPI_BOOT_CRYPT_CNT
eFuse bits. To modify this behavior, see Enabling UART Bootloader Encryption/Decryption.The device is then rebooted to start executing the encrypted image. The firmware bootloader calls the flash decryption block to decrypt the flash contents and then loads the decrypted contents into IRAM.
During the development stage, there is a frequent need to program different plaintext flash images and test the flash encryption process. This requires that Firmware Download mode is able to load new plaintext images as many times as it might be needed. However, during manufacturing or production stages, Firmware Download mode should not be allowed to access flash contents for security reasons.
Hence, two different flash encryption configurations were created: for development and for production. For details on these configurations, see Section Flash Encryption Configuration.
Flash Encryption Configuration
The following flash encryption modes are available:
Development Mode - recommended for use only during development. In this mode, it is still possible to flash new plaintext firmware to the device, and the bootloader will transparently encrypt this firmware using the key stored in hardware. This allows, indirectly, to read out the plaintext of the firmware in flash.
Release Mode - recommended for manufacturing and production. In this mode, flashing plaintext firmware to the device without knowing the encryption key is no longer possible.
This section provides information on the mentioned flash encryption modes and step by step instructions on how to use them.
Development Mode
During development, you can encrypt flash using either an ESP32-C6 generated key or external host-generated key.
Using ESP32-C6 Generated Key
Development mode allows you to download multiple plaintext images using Firmware Download mode.
To test flash encryption process, take the following steps:
Ensure that you have an ESP32-C6 device with default flash encryption eFuse settings as shown in Relevant eFuses.
See how to check ESP32-C6 Flash Encryption Status.
In Project Configuration Menu, do the following:
Select encryption mode (Development mode by default).
Select UART ROM download mode (enabled by default).
Save the configuration and exit.
Enabling flash encryption will increase the size of bootloader, which might require updating partition table offset. See Bootloader Size.
Run the command given below to build and flash the complete images.
idf.py flash monitorNote
This command does not include any user files which should be written to the partitions on the flash memory. Please write them manually before running this command otherwise the files should be encrypted separately before writing.
This command will write to flash memory unencrypted images: the firmware bootloader, the partition table and applications. Once the flashing is complete, ESP32-C6 will reset. On the next boot, the firmware bootloader encrypts: the firmware bootloader, application partitions and partitions marked as
encrypted
then resets. Encrypting in-place can take time, up to a minute for large partitions. After that, the application is decrypted at runtime and executed.
A sample output of the first ESP32-C6 boot after enabling flash encryption is given below:
rst:0x1 (POWERON),boot:0xc (SPI_FAST_FLASH_BOOT)
SPIWP:0xee
mode:DIO, clock div:2
load:0x4086c410,len:0xd5c
load:0x4086e610,len:0x4584
load:0x40875888,len:0x2bac
entry 0x4086c410
I (25) boot: ESP-IDF v5.1-dev-4270-g4bff4ed6e5-dirty 2nd stage bootloader
I (25) boot: compile time Mar 27 2023 16:48:49
I (27) boot: chip revision: v0.0
I (30) boot.esp32c6: SPI Speed : 40MHz
I (35) boot.esp32c6: SPI Mode : DIO
I (40) boot.esp32c6: SPI Flash Size : 2MB
I (44) boot: Enabling RNG early entropy source...
W (50) bootloader_random: bootloader_random_enable() has not been implemented yet
I (58) boot: Partition Table:
I (62) boot: ## Label Usage Type ST Offset Length
I (69) boot: 0 nvs WiFi data 01 02 0000a000 00006000
I (76) boot: 1 storage Unknown data 01 ff 00010000 00001000
I (84) boot: 2 factory factory app 00 00 00020000 00100000
I (91) boot: 3 nvs_key NVS keys 01 04 00120000 00001000
I (99) boot: 4 custom_nvs WiFi data 01 02 00121000 00006000
I (106) boot: End of partition table
I (110) esp_image: segment 0: paddr=00020020 vaddr=42018020 size=090e8h ( 37096) map
I (126) esp_image: segment 1: paddr=00029110 vaddr=40800000 size=06f08h ( 28424) load
I (134) esp_image: segment 2: paddr=00030020 vaddr=42000020 size=12fd8h ( 77784) map
I (151) esp_image: segment 3: paddr=00043000 vaddr=40806f08 size=03c00h ( 15360) load
I (158) boot: Loaded app from partition at offset 0x20000
I (158) boot: Checking flash encryption...
I (160) efuse: Batch mode of writing fields is enabled
I (165) flash_encrypt: Generating new flash encryption key...
I (174) efuse: Writing EFUSE_BLK_KEY0 with purpose 4
W (178) flash_encrypt: Not disabling UART bootloader encryption
I (184) flash_encrypt: Disable UART bootloader cache...
I (190) flash_encrypt: Disable JTAG...
I (197) efuse: BURN BLOCK4
I (204) efuse: BURN BLOCK4 - OK (write block == read block)
I (206) efuse: BURN BLOCK0
I (212) efuse: BURN BLOCK0 - OK (all write block bits are set)
I (216) efuse: Batch mode. Prepared fields are committed
I (222) esp_image: segment 0: paddr=00000020 vaddr=4086c410 size=00d5ch ( 3420)
I (231) esp_image: segment 1: paddr=00000d84 vaddr=4086e610 size=04584h ( 17796)
I (240) esp_image: segment 2: paddr=00005310 vaddr=40875888 size=02bach ( 11180)
I (632) flash_encrypt: bootloader encrypted successfully
I (679) flash_encrypt: partition table encrypted and loaded successfully
I (680) flash_encrypt: Encrypting partition 1 at offset 0x10000 (length 0x1000)...
I (732) flash_encrypt: Done encrypting
I (732) esp_image: segment 0: paddr=00020020 vaddr=42018020 size=090e8h ( 37096) map
I (741) esp_image: segment 1: paddr=00029110 vaddr=40800000 size=06f08h ( 28424)
I (747) esp_image: segment 2: paddr=00030020 vaddr=42000020 size=12fd8h ( 77784) map
I (765) esp_image: segment 3: paddr=00043000 vaddr=40806f08 size=03c00h ( 15360)
I (769) flash_encrypt: Encrypting partition 2 at offset 0x20000 (length 0x100000)...
I (13025) flash_encrypt: Done encrypting
I (13025) flash_encrypt: Encrypting partition 3 at offset 0x120000 (length 0x1000)...
I (13074) flash_encrypt: Done encrypting
I (13075) efuse: BURN BLOCK0
I (13077) efuse: BURN BLOCK0 - OK (all write block bits are set)
I (13078) flash_encrypt: Flash encryption completed
I (13083) boot: Resetting with flash encryption enabled...
A sample output of subsequent ESP32-C6 boots just mentions that flash encryption is already enabled:
rst:0x3 (LP_SW_HPSYS),boot:0xc (SPI_FAST_FLASH_BOOT)
Saved PC:0x4001974a
SPIWP:0xee
mode:DIO, clock div:2
load:0x4086c410,len:0xd5c
load:0x4086e610,len:0x4584
load:0x40875888,len:0x2bac
entry 0x4086c410
I (24) boot: ESP-IDF v5.1-dev-4270-g4bff4ed6e5-dirty 2nd stage bootloader
I (24) boot: compile time Mar 27 2023 16:48:49
I (25) boot: chip revision: v0.0
I (29) boot.esp32c6: SPI Speed : 40MHz
I (34) boot.esp32c6: SPI Mode : DIO
I (39) boot.esp32c6: SPI Flash Size : 2MB
I (43) boot: Enabling RNG early entropy source...
W (49) bootloader_random: bootloader_random_enable() has not been implemented yet
I (57) boot: Partition Table:
I (60) boot: ## Label Usage Type ST Offset Length
I (68) boot: 0 nvs WiFi data 01 02 0000a000 00006000
I (75) boot: 1 storage Unknown data 01 ff 00010000 00001000
I (83) boot: 2 factory factory app 00 00 00020000 00100000
I (90) boot: 3 nvs_key NVS keys 01 04 00120000 00001000
I (98) boot: 4 custom_nvs WiFi data 01 02 00121000 00006000
I (105) boot: End of partition table
I (109) esp_image: segment 0: paddr=00020020 vaddr=42018020 size=090e8h ( 37096) map
I (126) esp_image: segment 1: paddr=00029110 vaddr=40800000 size=06f08h ( 28424) load
I (134) esp_image: segment 2: paddr=00030020 vaddr=42000020 size=12fd8h ( 77784) map
I (152) esp_image: segment 3: paddr=00043000 vaddr=40806f08 size=03c00h ( 15360) load
I (159) boot: Loaded app from partition at offset 0x20000
I (159) boot: Checking flash encryption...
I (160) flash_encrypt: flash encryption is enabled (1 plaintext flashes left)
I (168) boot: Disabling RNG early entropy source...
W (173) bootloader_random: bootloader_random_enable() has not been implemented yet
I (193) cpu_start: Pro cpu up.
W (202) clk: esp_perip_clk_init() has not been implemented yet
I (208) cpu_start: Pro cpu start user code
I (209) cpu_start: cpu freq: 160000000 Hz
I (209) cpu_start: Application information:
I (211) cpu_start: Project name: flash_encryption
I (217) cpu_start: App version: v5.1-dev-4270-g4bff4ed6e5-dirty
I (224) cpu_start: Compile time: Mar 27 2023 16:49:00
I (230) cpu_start: ELF file SHA256: df1dd35054510e16...
I (236) cpu_start: ESP-IDF: v5.1-dev-4270-g4bff4ed6e5-dirty
I (243) cpu_start: Min chip rev: v0.0
I (248) cpu_start: Max chip rev: v0.99
I (253) cpu_start: Chip rev: v0.0
I (258) heap_init: Initializing. RAM available for dynamic allocation:
I (265) heap_init: At 4080B9E0 len 00070C30 (451 KiB): D/IRAM
I (271) heap_init: At 4087C610 len 00002F54 (11 KiB): STACK/DIRAM
I (278) heap_init: At 50000010 len 00003FF0 (15 KiB): RTCRAM
I (285) spi_flash: detected chip: generic
I (289) spi_flash: flash io: dio
W (293) spi_flash: Detected size(4096k) larger than the size in the binary image header(2048k). Using the size in the binary image header.
W (306) flash_encrypt: Flash encryption mode is DEVELOPMENT (not secure)
I (314) sleep: Configure to isolate all GPIO pins in sleep state
I (320) sleep: Enable automatic switching of GPIO sleep configuration
I (327) coexist: coex firmware version: 5315623
I (333) coexist: coexist rom version 5b8dcfa
I (338) app_start: Starting scheduler on CPU0
I (342) main_task: Started on CPU0
I (342) main_task: Calling app_main()
Example to check Flash Encryption status
This is esp32c6 chip with 1 CPU core(s), WiFi/BLE, silicon revision v0.0, 2MB external flash
FLASH_CRYPT_CNT eFuse value is 1
Flash encryption feature is enabled in DEVELOPMENT mode
At this stage, if you need to update and re-flash binaries, see Re-flashing Updated Partitions.
Using Host Generated Key
It is possible to pre-generate a flash encryption key on the host computer and burn it into the eFuse. This allows you to pre-encrypt data on the host and flash already encrypted data without needing a plaintext flash update. This feature can be used in both Development Mode and Release Mode. Without a pre-generated key, data is flashed in plaintext and then ESP32-C6 encrypts the data in-place.
Note
This option is not recommended for production, unless a separate key is generated for each individual device.
To use a host generated key, take the following steps:
Ensure that you have an ESP32-C6 device with default flash encryption eFuse settings as shown in Relevant eFuses.
See how to check ESP32-C6 Flash Encryption Status.
Generate a random key by running:
espsecure.py generate_flash_encryption_key my_flash_encryption_key.bin
Before the first encrypted boot, burn the key into your device’s eFuse using the command below. This action can be done only once.
espefuse.py --port PORT burn_key BLOCK my_flash_encryption_key.bin XTS_AES_128_KEYwhere
BLOCK
is a free keyblock betweenBLOCK_KEY0
andBLOCK_KEY5
.If the key is not burned and the device is started after enabling flash encryption, the ESP32-C6 will generate a random key that software cannot access or modify.
In Project Configuration Menu, do the following:
Select encryption mode (Development mode by default)
Save the configuration and exit.
Enabling flash encryption will increase the size of bootloader, which might require updating partition table offset. See Bootloader Size.
Run the command given below to build and flash the complete images.
idf.py flash monitorNote
This command does not include any user files which should be written to the partitions on the flash memory. Please write them manually before running this command otherwise the files should be encrypted separately before writing.
This command will write to flash memory unencrypted images: the firmware bootloader, the partition table and applications. Once the flashing is complete, ESP32-C6 will reset. On the next boot, the firmware bootloader encrypts: the firmware bootloader, application partitions and partitions marked as
encrypted
then resets. Encrypting in-place can take time, up to a minute for large partitions. After that, the application is decrypted at runtime and executed.
If using Development Mode, then the easiest way to update and re-flash binaries is Re-flashing Updated Partitions.
If using Release Mode, then it is possible to pre-encrypt the binaries on the host and then flash them as ciphertext. See Manually Encrypting Files.
Re-flashing Updated Partitions
If you update your application code (done in plaintext) and want to re-flash it, you will need to encrypt it before flashing. To encrypt the application and flash it in one step, run:
idf.py encrypted-app-flash monitor
If all partitions needs to be updated in encrypted format, run:
idf.py encrypted-flash monitor
Release Mode
In Release mode, UART bootloader cannot perform flash encryption operations. New plaintext images can ONLY be downloaded using the over-the-air (OTA) scheme which will encrypt the plaintext image before writing to flash.
To use this mode, take the following steps:
Ensure that you have an ESP32-C6 device with default flash encryption eFuse settings as shown in Relevant eFuses.
See how to check ESP32-C6 Flash Encryption Status.
In Project Configuration Menu, do the following:
Select Release mode (Note that once Release mode is selected, the
EFUSE_DIS_DOWNLOAD_MANUAL_ENCRYPT
eFuse bit will be burned to disable flash encryption hardware in ROM Download Mode.)Select UART ROM download mode (Permanently switch to Secure mode (recommended)). This is the default option, and is recommended. It is also possible to change this configuration setting to permanently disable UART ROM download mode, if this mode is not needed.
Save the configuration and exit.
Enabling flash encryption will increase the size of bootloader, which might require updating partition table offset. See Bootloader Size.
Run the command given below to build and flash the complete images.
idf.py flash monitorNote
This command does not include any user files which should be written to the partitions on the flash memory. Please write them manually before running this command otherwise the files should be encrypted separately before writing.
This command will write to flash memory unencrypted images: the firmware bootloader, the partition table and applications. Once the flashing is complete, ESP32-C6 will reset. On the next boot, the firmware bootloader encrypts: the firmware bootloader, application partitions and partitions marked as
encrypted
then resets. Encrypting in-place can take time, up to a minute for large partitions. After that, the application is decrypted at runtime and executed.
Once the flash encryption is enabled in Release mode, the bootloader will write-protect the SPI_BOOT_CRYPT_CNT
eFuse.
For subsequent plaintext field updates, use OTA scheme.
Note
If you have pre-generated the flash encryption key and stored a copy, and the UART download mode is not permanently disabled via CONFIG_SECURE_UART_ROM_DL_MODE , then it is possible to update the flash locally by pre-encrypting the files and then flashing the ciphertext. See Manually Encrypting Files.
Best Practices
When using Flash Encryption in production:
Do not reuse the same flash encryption key between multiple devices. This means that an attacker who copies encrypted data from one device cannot transfer it to a second device.
The UART ROM Download Mode should be disabled entirely if it is not needed, or permanently set to “Secure Download Mode” otherwise. Secure Download Mode permanently limits the available commands to updating SPI config, changing baud rate, basic flash write, and returning a summary of the currently enabled security features with the get_security_info command. The default behaviour is to set Secure Download Mode on first boot in Release mode. To disable Download Mode entirely, select CONFIG_SECURE_UART_ROM_DL_MODE to “Permanently disable ROM Download Mode (recommended)” or call
esp_efuse_disable_rom_download_mode()
at runtime.Enable Secure Boot as an extra layer of protection, and to prevent an attacker from selectively corrupting any part of the flash before boot.
Possible Failures
Once flash encryption is enabled, the SPI_BOOT_CRYPT_CNT
eFuse value will have an odd number of bits set. It means that all the partitions marked with the encryption flag are expected to contain encrypted ciphertext. Below are the three typical failure cases if the ESP32-C6 is erroneously loaded with plaintext data:
If the bootloader partition is re-flashed with a plaintext firmware bootloader image, the ROM bootloader will fail to load the firmware bootloader resulting in the following failure:
rst:0x3 (SW_RESET),boot:0x13 (SPI_FAST_FLASH_BOOT)
invalid header: 0xb414f76b
invalid header: 0xb414f76b
invalid header: 0xb414f76b
invalid header: 0xb414f76b
invalid header: 0xb414f76b
invalid header: 0xb414f76b
invalid header: 0xb414f76b
Note
The value of invalid header will be different for every application.
Note
This error also appears if the flash contents are erased or corrupted.
If the firmware bootloader is encrypted, but the partition table is re-flashed with a plaintext partition table image, the bootloader will fail to read the partition table resulting in the following failure:
rst:0x3 (SW_RESET),boot:0x13 (SPI_FAST_FLASH_BOOT) configsip: 0, SPIWP:0xee clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00 mode:DIO, clock div:2 load:0x3fff0018,len:4 load:0x3fff001c,len:10464 ho 0 tail 12 room 4 load:0x40078000,len:19168 load:0x40080400,len:6664 entry 0x40080764 I (60) boot: ESP-IDF v4.0-dev-763-g2c55fae6c-dirty 2nd stage bootloader I (60) boot: compile time 19:15:54 I (62) boot: Enabling RNG early entropy source... I (67) boot: SPI Speed : 40MHz I (72) boot: SPI Mode : DIO I (76) boot: SPI Flash Size : 4MB E (80) flash_parts: partition 0 invalid magic number 0x94f6 E (86) boot: Failed to verify partition table E (91) boot: load partition table error!
If the bootloader and partition table are encrypted, but the application is re-flashed with a plaintext application image, the bootloader will fail to load the application resulting in the following failure:
rst:0x3 (SW_RESET),boot:0x13 (SPI_FAST_FLASH_BOOT) configsip: 0, SPIWP:0xee clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00 mode:DIO, clock div:2 load:0x3fff0018,len:4 load:0x3fff001c,len:8452 load:0x40078000,len:13616 load:0x40080400,len:6664 entry 0x40080764 I (56) boot: ESP-IDF v4.0-dev-850-gc4447462d-dirty 2nd stage bootloader I (56) boot: compile time 15:37:14 I (58) boot: Enabling RNG early entropy source... I (64) boot: SPI Speed : 40MHz I (68) boot: SPI Mode : DIO I (72) boot: SPI Flash Size : 4MB I (76) boot: Partition Table: I (79) boot: ## Label Usage Type ST Offset Length I (87) boot: 0 nvs WiFi data 01 02 0000a000 00006000 I (94) boot: 1 phy_init RF data 01 01 00010000 00001000 I (102) boot: 2 factory factory app 00 00 00020000 00100000 I (109) boot: End of partition table E (113) esp_image: image at 0x20000 has invalid magic byte W (120) esp_image: image at 0x20000 has invalid SPI mode 108 W (126) esp_image: image at 0x20000 has invalid SPI size 11 E (132) boot: Factory app partition is not bootable E (138) boot: No bootable app partitions in the partition table
ESP32-C6 Flash Encryption Status
Ensure that you have an ESP32-C6 device with default flash encryption eFuse settings as shown in Relevant eFuses.
To check if flash encryption on your ESP32-C6 device is enabled, do one of the following:
flash the application example security/flash_encryption onto your device. This application prints the
SPI_BOOT_CRYPT_CNT
eFuse value and if flash encryption is enabled or disabled.Find the serial port name under which your ESP32-C6 device is connected, replace
PORT
with your port name in the following command, and run it:espefuse.py -p PORT summary
Reading and Writing Data in Encrypted Flash
ESP32-C6 application code can check if flash encryption is currently enabled by calling esp_flash_encryption_enabled()
. Also, a device can identify the flash encryption mode by calling esp_get_flash_encryption_mode()
.
Once flash encryption is enabled, be more careful with accessing flash contents from code.
Scope of Flash Encryption
Whenever the SPI_BOOT_CRYPT_CNT
eFuse is set to a value with an odd number of bits, all flash content accessed via the MMU’s flash cache is transparently decrypted. It includes:
Executable application code in flash (IROM).
All read-only data stored in flash (DROM).
Any data accessed via
spi_flash_mmap()
.The firmware bootloader image when it is read by the ROM bootloader.
Important
The MMU flash cache unconditionally decrypts all existing data. Data which is stored unencrypted in flash memory will also be “transparently decrypted” via the flash cache and will appear to software as random garbage.
Reading from Encrypted Flash
To read data without using a flash cache MMU mapping, you can use the partition read function esp_partition_read()
. This function will only decrypt data when it is read from an encrypted partition. Data read from unencrypted partitions will not be decrypted. In this way, software can access encrypted and non-encrypted flash in the same way.
You can also use the following SPI flash API functions:
esp_flash_read()
to read raw (encrypted) data which will not be decryptedesp_flash_read_encrypted()
to read and decrypt data
Data stored using the Non-Volatile Storage (NVS) API is always stored and read decrypted from the perspective of flash encryption. It is up to the library to provide encryption feature if required. Refer to NVS Encryption for more details.
Writing to Encrypted Flash
It is recommended to use the partition write function esp_partition_write()
. This function will only encrypt data when it is written to an encrypted partition. Data written to unencrypted partitions will not be encrypted. In this way, software can access encrypted and non-encrypted flash in the same way.
You can also pre-encrypt and write data using the function esp_flash_write_encrypted()
Also, the following ROM function exist but not supported in esp-idf applications:
esp_rom_spiflash_write_encrypted
pre-encrypts and writes data to flashSPIWrite
writes unencrypted data to flash
Since data is encrypted in blocks, the minimum write size for encrypted data is 16 bytes and the alignment is also 16 bytes.
Updating Encrypted Flash
OTA Updates
OTA updates to encrypted partitions will automatically write encrypted data if the function esp_partition_write()
is used.
Before building the application image for OTA updating of an already encrypted device, enable the option Enable flash encryption on boot in project configuration menu.
For general information about ESP-IDF OTA updates, please refer to OTA
Updating Encrypted Flash via Serial
Flashing an encrypted device via serial bootloader requires that the serial bootloader download interface has not been permanently disabled via eFuse.
In Development Mode, the recommended method is Re-flashing Updated Partitions.
In Release Mode, if a copy of the same key stored in eFuse is available on the host then it’s possible to pre-encrypt files on the host and then flash them. See Manually Encrypting Files.
Disabling Flash Encryption
If flash encryption was enabled accidentally, flashing of plaintext data will soft-brick the ESP32-C6. The device will reboot continuously, printing the error flash read err, 1000
or invalid header: 0xXXXXXX
.
For flash encryption in Development mode, encryption can be disabled by burning the SPI_BOOT_CRYPT_CNT
eFuse. It can only be done one time per chip by taking the following steps:
In Project Configuration Menu, disable Enable flash encryption on boot, then save and exit.
Open project configuration menu again and double-check that you have disabled this option! If this option is left enabled, the bootloader will immediately re-enable encryption when it boots.
With flash encryption disabled, build and flash the new bootloader and application by running
idf.py flash
.Use
espefuse.py
(incomponents/esptool_py/esptool
) to disable theSPI_BOOT_CRYPT_CNT
by running:
espefuse.py burn_efuse SPI_BOOT_CRYPT_CNT
Reset the ESP32-C6. Flash encryption will be disabled, and the bootloader will boot as usual.
Key Points About Flash Encryption
Flash memory contents is encrypted using XTS-AES-128. The flash encryption key is 256 bits and stored in one
BLOCK_KEYN
eFuse internal to the chip and, by default, is protected from software access.Flash access is transparent via the flash cache mapping feature of ESP32-C6 - any flash regions which are mapped to the address space will be transparently decrypted when read.
Some data partitions might need to remain unencrypted for ease of access or might require the use of flash-friendly update algorithms which are ineffective if the data is encrypted. NVS partitions for non-volatile storage cannot be encrypted since the NVS library is not directly compatible with flash encryption. For details, refer to NVS Encryption.
If flash encryption might be used in future, the programmer must keep it in mind and take certain precautions when writing code that uses encrypted flash.
If secure boot is enabled, re-flashing the bootloader of an encrypted device requires a “Re-flashable” secure boot digest (see Flash Encryption and Secure Boot).
Enabling flash encryption will increase the size of bootloader, which might require updating partition table offset. See Bootloader Size.
Important
Do not interrupt power to the ESP32-C6 while the first boot encryption pass is running. If power is interrupted, the flash contents will be corrupted and will require flashing with unencrypted data again. In this case, re-flashing will not count towards the flashing limit.
Limitations of Flash Encryption
Flash encryption protects firmware against unauthorised readout and modification. It is important to understand the limitations of the flash encryption feature:
Flash encryption is only as strong as the key. For this reason, we recommend keys are generated on the device during first boot (default behaviour). If generating keys off-device, ensure proper procedure is followed and don’t share the same key between all production devices.
Not all data is stored encrypted. If storing data on flash, check if the method you are using (library, API, etc.) supports flash encryption.
Flash encryption does not prevent an attacker from understanding the high-level layout of the flash. This is because the same AES key is used for every pair of adjacent 16 byte AES blocks. When these adjacent 16 byte blocks contain identical content (such as empty or padding areas), these blocks will encrypt to produce matching pairs of encrypted blocks. This may allow an attacker to make high-level comparisons between encrypted devices (i.e. to tell if two devices are probably running the same firmware version).
Flash encryption alone may not prevent an attacker from modifying the firmware of the device. To prevent unauthorised firmware from running on the device, use flash encryption in combination with Secure Boot.
Flash Encryption and Secure Boot
It is recommended to use flash encryption in combination with Secure Boot. However, if Secure Boot is enabled, additional restrictions apply to device re-flashing:
OTA Updates are not restricted, provided that the new app is signed correctly with the Secure Boot signing key.
Advanced Features
The following section covers advanced features of flash encryption.
Encrypted Partition Flag
Some partitions are encrypted by default. Other partitions can be marked in the partition table description as requiring encryption by adding the flag encrypted
to the partitions’ flag field. As a result, data in these marked partitions will be treated as encrypted in the same manner as an app partition.
# Name, Type, SubType, Offset, Size, Flags
nvs, data, nvs, 0x9000, 0x6000
phy_init, data, phy, 0xf000, 0x1000
factory, app, factory, 0x10000, 1M
secret_data, 0x40, 0x01, 0x20000, 256K, encrypted
For details on partition table description, see partition table.
Further information about encryption of partitions:
Default partition tables do not include any encrypted data partitions.
With flash encryption enabled, the
app
partition is always treated as encrypted and does not require marking.If flash encryption is not enabled, the flag “encrypted” has no effect.
You can also consider protecting
phy_init
data from physical access, readout, or modification, by marking the optionalphy
partition with the flagencrypted
.The
nvs
partition cannot be encrypted, because the NVS library is not directly compatible with flash encryption.
Enabling UART Bootloader Encryption/Decryption
On the first boot, the flash encryption process burns by default the following eFuses:
DIS_DOWNLOAD_MANUAL_ENCRYPT
which disables flash encryption operation when running in UART bootloader boot mode.DIS_DOWNLOAD_ICACHE
which disables the entire MMU flash cache when running in UART bootloader mode.DIS_PAD_JTAG
andDIS_USB_JTAG
which disables JTAG.DIS_DIRECT_BOOT
(old nameDIS_LEGACY_SPI_BOOT
) which disables direct boot mode
However, before the first boot you can choose to keep any of these features enabled by burning only selected eFuses and write-protect the rest of eFuses with unset value 0. For example:
espefuse.py --port PORT burn_efuse DIS_DOWNLOAD_MANUAL_ENCRYPT
espefuse.py --port PORT write_protect_efuse DIS_DOWNLOAD_MANUAL_ENCRYPT
Note
Set all appropriate bits before write-protecting!
Write protection of all the three eFuses is controlled by one bit. It means that write-protecting one eFuse bit will inevitably write-protect all unset eFuse bits.
Write protecting these eFuses to keep them unset is not currently very useful, as esptool.py
does not support reading encrypted flash.
JTAG Debugging
By default, when Flash Encryption is enabled (in either Development or Release mode) then JTAG debugging is disabled via eFuse. The bootloader does this on first boot, at the same time it enables flash encryption.
See JTAG with Flash Encryption or Secure Boot for more information about using JTAG Debugging with Flash Encryption.
Manually Encrypting Files
Manually encrypting or decrypting files requires the flash encryption key to be pre-burned in eFuse (see Using Host Generated Key) and a copy to be kept on the host. If the flash encryption is configured in Development Mode then it’s not necessary to keep a copy of the key or follow these steps, the simpler Re-flashing Updated Partitions steps can be used.
The key file should be a single raw binary file (example: key.bin
).
For example, these are the steps to encrypt the file build/my-app.bin
to flash at offset 0x10000. Run espsecure.py as follows:
espsecure.py encrypt_flash_data --aes_xts --keyfile /path/to/key.bin --address 0x10000 --output my-app-ciphertext.bin build/my-app.bin
The file my-app-ciphertext.bin
can then be flashed to offset 0x10000 using esptool.py
. To see all of the command line options recommended for esptool.py
, see the output printed when idf.py build
succeeds.
Note
If the flashed ciphertext file is not recognized by the ESP32-C6 when it boots, check that the keys match and that the command line arguments match exactly, including the correct offset.
The command espsecure.py decrypt_flash_data
can be used with the same options (and different input/output files), to decrypt ciphertext flash contents or a previously encrypted file.
Technical Details
The following sections provide some reference information about the operation of flash encryption.
Flash Encryption Algorithm
ESP32-C6 use the XTS-AES block chiper mode with 256 bit size for flash encryption.
XTS-AES is a block chiper mode specifically designed for disc encryption and addresses the weaknesses other potential modes (e.g. AES-CTR) have for this use case. A detailed description of the XTS-AES algorithm can be found in IEEE Std 1619-2007.
The flash encryption key is stored in one
BLOCK_KEYN
eFuse and, by default, is protected from further writes or software readout.To see the full flash encryption algorithm implemented in Python, refer to the _flash_encryption_operation() function in the
espsecure.py
source code.