mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-24 19:12:38 +00:00 
			
		
		
		
	 a02be97fda
			
		
	
	a02be97fda
	
	
	
		
			
			handling (only when nvs encryption is enabled)
    * NVS Encryption will now be turned on by default with flash encryption
    * Updated the flash encryption example to shocase NVS encryption
    along with information on how to configure and use NVS encryption
    * Updated respective test case
    * Added two partition tables for NVS encryption
     i) Table 1- Single factory app, no OTA, encrypted NVS
     ii) Table 2- Factory app, Two OTA, encrypted NVS
		
	
		
			
				
	
	
		
			149 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| | Supported Targets | ESP32 |
 | |
| | ----------------- | ----- |
 | |
| 
 | |
| # Flash Encryption
 | |
| 
 | |
| The example checks if the flash encryption feature is enabled/disabled and if enabled prints the flash encryption mode (DEVELOPMENT / RELEASE) and FLASH_CRYPT_CNT eFuse value.
 | |
| 
 | |
| The example also demonstrates writing and reading encrypted partitions in flash.
 | |
| 
 | |
| ## How to use example
 | |
| 
 | |
| ### Hardware Required
 | |
| 
 | |
| ### Configure the project
 | |
| 
 | |
| ```
 | |
| idf.py menuconfig
 | |
| ```
 | |
| #### Configuration for flash encryption
 | |
| * Enable the flash encryption mode (Development or Release) under Security Features. Default usage mode is Development (recommended during test and development phase).
 | |
| 
 | |
| Note: After enabling flash encryption, the bootloader size increases, which means that the offset of the partition table must be changed to 0x9000 from 0x8000 to prevent the bootloader from overlapping with the partition table. In this example, the default offset of the partition table is 0x9000.
 | |
| 
 | |
| For better security, the NVS encryption is enabled by default when the flash encryption is enabled. If you choose to disable the NVS encryption, you can skip the NVS configuration step given below.
 | |
| 
 | |
| #### Configuration for NVS encryption
 | |
| For using NVS encryption, the partition table must contain the [NVS key partition](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#nvs-key-partition). Two partition tables containing the NVS keys partition are provided for NVS encryption under the partition table option . They can be selected with the project configuration menu (`menuconfig -> Partition Table`). This particular example uses a custom partition table as it requires a `storage` partition along with the `nvs_keys` partition.
 | |
| 
 | |
| The configuration for NVS encryption involves generating the XTS encryption keys in the [NVS key partition](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#nvs-key-partition) partition. It can be done with one of the following method.
 | |
| 
 | |
| 1. Generate the XTS encryption keys on the ESP chip:
 | |
| 
 | |
|     When NVS encryption is enabled the `nvs_flash_init` API function can internally generate the XTS encryption keys on the ESP chip. The API function finds the first [NVS key partition](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#nvs-key-partition) i.e. a partition of type  `data` and subtype `nvs_keys`.
 | |
|     Then the API function automatically generates and stores the
 | |
|     nvs keys in that partition. New keys are generated and stored only when the respective key partiton is empty. (Consult the [`nvs_flash_init`](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#_CPPv414nvs_flash_initv) API documentation in the ESP-IDF programming guide for more details)
 | |
| 
 | |
| 2. Use pre-generated XTS encryption keys:
 | |
|     This method will be required by the user when the `XTS encryption keys` in [NVS key partition](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#nvs-key-partition) are not generated by the application.
 | |
|     The pre generated `Sample XTS encryption keys` can be stored on the flash with help of the following two commands
 | |
| 
 | |
|     i) Build and flash the partition table:
 | |
|     ```
 | |
|     idf.py partition_table partition_table-flash
 | |
|     ```
 | |
|     ii) Store the `sample_encryption_keys.bin` in the `nvs_key`partition (on the flash) with the help of [parttool.py](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/partition-tables.html#partition-tool-parttool-py):
 | |
|     ```
 | |
|     parttool.py --port /dev/ttyUSB0 --partition-table-offset 0x9000 write_partition --partition-name="nvs_key" --input sample_encryption_keys.bin
 | |
|     ```
 | |
|     The sample [NVS key partition](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_flash.html#nvs-key-partition) partition used in this example is generated with the help of [NVS Partition Generator Utility](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/storage/nvs_partition_gen.html#nvs-partition-generator-utility)
 | |
|     .
 | |
| 
 | |
| ### Build and Flash
 | |
| 
 | |
| When building the project and flashing it to the board FOR THE FIRST TIME after enabling flash encryption feature in menuconfig, run following command to program ESP32 and monitor the output:
 | |
| 
 | |
| ```
 | |
| idf.py -p PORT flash monitor
 | |
| ```
 | |
| 
 | |
| (To exit the serial monitor, type ``Ctrl-]``.)
 | |
| 
 | |
| See the Getting Started Guide for full steps to configure and use ESP-IDF to build projects.
 | |
| 
 | |
| When reprogramming the device subsequently use following command for encrypted write of new plaintext application:
 | |
| 
 | |
| ```
 | |
| idf.py -p PORT encrypted-app-flash monitor
 | |
| ```
 | |
| 
 | |
| Please note above command programs only the app partition. In order to reprogram all partitions (bootloader, partition table and application) in encrypted form use:
 | |
| 
 | |
| ```
 | |
| idf.py -p PORT encrypted-flash monitor
 | |
| ```
 | |
| 
 | |
| ## Example Output
 | |
| 
 | |
| When running the example without enabling flash encryption, the output would be as follows:
 | |
| 
 | |
| ```
 | |
| Example to check Flash Encryption status
 | |
| This is ESP32 chip with 2 CPU cores, WiFi/BT/BLE, silicon revision 0, 2MB external flash
 | |
| FLASH_CRYPT_CNT eFuse value is 0
 | |
| Flash encryption feature is disabled
 | |
| Erasing partition "storage" (0x1000 bytes)
 | |
| Writing data with esp_partition_write:
 | |
| I (378) example: 0x3ffb4dc0   00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  |................|
 | |
| I (378) example: 0x3ffb4dd0   10 11 12 13 14 15 16 17  18 19 1a 1b 1c 1d 1e 1f  |................|
 | |
| Reading with esp_partition_read:
 | |
| I (388) example: 0x3ffb4da0   00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  |................|
 | |
| I (398) example: 0x3ffb4db0   10 11 12 13 14 15 16 17  18 19 1a 1b 1c 1d 1e 1f  |................|
 | |
| Reading with spi_flash_read:
 | |
| I (408) example: 0x3ffb4da0   00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  |................|
 | |
| I (418) example: 0x3ffb4db0   10 11 12 13 14 15 16 17  18 19 1a 1b 1c 1d 1e 1f  |................|
 | |
| ```
 | |
| 
 | |
| After enabling flash encryption in Development mode, the output shows the process of enabling the flash encryption:
 | |
| 
 | |
| ```
 | |
| I (168) boot: Checking flash encryption...
 | |
| I (168) flash_encrypt: Generating new flash encryption key...
 | |
| I (187) flash_encrypt: Read & write protecting new key...
 | |
| I (187) flash_encrypt: Setting CRYPT_CONFIG efuse to 0xF
 | |
| W (188) flash_encrypt: Not disabling UART bootloader encryption
 | |
| I (195) flash_encrypt: Disable UART bootloader decryption...
 | |
| I (201) flash_encrypt: Disable UART bootloader MMU cache...
 | |
| I (208) flash_encrypt: Disable JTAG...
 | |
| I (212) flash_encrypt: Disable ROM BASIC interpreter fallback...
 | |
| ....
 | |
| ....
 | |
| ....
 | |
| I (13229) flash_encrypt: Flash encryption completed
 | |
| I (13229) boot: Resetting with flash encryption enabled...
 | |
| ```
 | |
| 
 | |
| Once the flash encryption is enabled the device will reset itself. At this stage the flash contents are in encrypted form. The output would be similar to:
 | |
| 
 | |
| ```
 | |
| Example to check Flash Encryption status
 | |
| This is ESP32 chip with 2 CPU cores, WiFi/BT/BLE, silicon revision 0, 4MB external flash
 | |
| FLASH_CRYPT_CNT eFuse value is 1
 | |
| Flash encryption feature is enabled in DEVELOPMENT mode
 | |
| Erasing partition "storage" (0x1000 bytes)
 | |
| Writing data with esp_partition_write:
 | |
| I (451) example: 0x3ffb4dc0   00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  |................|
 | |
| I (451) example: 0x3ffb4dd0   10 11 12 13 14 15 16 17  18 19 1a 1b 1c 1d 1e 1f  |................|
 | |
| Reading with esp_partition_read:
 | |
| I (461) example: 0x3ffb4da0   00 01 02 03 04 05 06 07  08 09 0a 0b 0c 0d 0e 0f  |................|
 | |
| I (471) example: 0x3ffb4db0   10 11 12 13 14 15 16 17  18 19 1a 1b 1c 1d 1e 1f  |................|
 | |
| Reading with spi_flash_read:
 | |
| I (491) example: 0x3ffb4b30   35 9b f2 07 b4 6d 40 89  28 b4 1e 22 98 7b 4a 36  |5....m@.(..".{J6|
 | |
| I (491) example: 0x3ffb4b40   ba 89 81 67 77 a3 60 5e  0a e7 51 01 b3 58 c2 f6  |...gw.`^..Q..X..|
 | |
| ```
 | |
| 
 | |
| If the NVS encryption is enabled, then the output will show the status of the encrypted partition as follows
 | |
| 
 | |
| ```
 | |
| I (667) example_nvs: NVS partition "nvs" is encrypted.
 | |
| ```
 | |
| ## Troubleshooting
 | |
| 
 | |
| It is also possible to use esptool.py utility to read the eFuse values and check if flash encryption is enabled or not
 | |
| 
 | |
| ```
 | |
| python $IDF_PATH/components/esptool_py/esptool/espefuse.py --port PORT summary
 | |
| ```
 | |
| 
 | |
| If FLASH_CRYPT_CNT eFuse value is non-zero flash encryption is enabled
 |