| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407 |
- // Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- #include <string.h>
- #include "esp_attr.h"
- #include "esp_types.h"
- #include "esp_log.h"
- #include "esp32/rom/cache.h"
- #include "esp32/rom/ets_sys.h"
- #include "esp32/rom/crc.h"
- #include "soc/efuse_periph.h"
- #include "soc/rtc_periph.h"
- #include "bootloader_sha.h"
- #include "bootloader_utility.h"
- #include "sdkconfig.h"
- #include "bootloader_flash.h"
- #include "bootloader_random.h"
- #include "esp_image_format.h"
- #include "esp_secure_boot.h"
- #include "esp_flash_encrypt.h"
- #include "esp_efuse.h"
- /* The following API implementations are used only when called
- * from the bootloader code.
- */
- /* Burn values written to the efuse write registers */
- static inline void burn_efuses(void)
- {
- esp_efuse_burn_new_values();
- }
- #ifdef CONFIG_SECURE_BOOT_V1_ENABLED
- static const char *TAG = "secure_boot_v1";
- /**
- * @function : secure_boot_generate
- * @description: generate boot digest (aka "abstract") & iv
- *
- * @inputs: image_len - length of image to calculate digest for
- */
- static bool secure_boot_generate(uint32_t image_len){
- esp_err_t err;
- esp_secure_boot_iv_digest_t digest;
- const uint32_t *image;
- /* hardware secure boot engine only takes full blocks, so round up the
- image length. The additional data should all be 0xFF (or the appended SHA, if it falls in the same block).
- */
- if (image_len % sizeof(digest.iv) != 0) {
- image_len = (image_len / sizeof(digest.iv) + 1) * sizeof(digest.iv);
- }
- ets_secure_boot_start();
- ets_secure_boot_rd_iv((uint32_t *)digest.iv);
- ets_secure_boot_hash(NULL);
- /* iv stored in sec 0 */
- err = bootloader_flash_erase_sector(0);
- if (err != ESP_OK)
- {
- ESP_LOGE(TAG, "SPI erase failed: 0x%x", err);
- return false;
- }
- /* generate digest from image contents */
- image = bootloader_mmap(ESP_BOOTLOADER_OFFSET, image_len);
- if (!image) {
- ESP_LOGE(TAG, "bootloader_mmap(0x1000, 0x%x) failed", image_len);
- return false;
- }
- for (int i = 0; i < image_len; i+= sizeof(digest.iv)) {
- ets_secure_boot_hash(&image[i/sizeof(uint32_t)]);
- }
- bootloader_munmap(image);
- ets_secure_boot_obtain();
- ets_secure_boot_rd_abstract((uint32_t *)digest.digest);
- ets_secure_boot_finish();
- ESP_LOGD(TAG, "write iv+digest to flash");
- err = bootloader_flash_write(FLASH_OFFS_SECURE_BOOT_IV_DIGEST, &digest,
- sizeof(digest), esp_flash_encryption_enabled());
- if (err != ESP_OK) {
- ESP_LOGE(TAG, "SPI write failed: 0x%x", err);
- return false;
- }
- Cache_Read_Enable(0);
- return true;
- }
- esp_err_t esp_secure_boot_generate_digest(void)
- {
- esp_err_t err;
- if (esp_secure_boot_enabled()) {
- ESP_LOGI(TAG, "bootloader secure boot is already enabled."
- " No need to generate digest. continuing..");
- return ESP_OK;
- }
- uint32_t coding_scheme = REG_GET_FIELD(EFUSE_BLK0_RDATA6_REG, EFUSE_CODING_SCHEME);
- if (coding_scheme != EFUSE_CODING_SCHEME_VAL_NONE && coding_scheme != EFUSE_CODING_SCHEME_VAL_34) {
- ESP_LOGE(TAG, "Unknown/unsupported CODING_SCHEME value 0x%x", coding_scheme);
- return ESP_ERR_NOT_SUPPORTED;
- }
- /* Verify the bootloader */
- esp_image_metadata_t bootloader_data = { 0 };
- err = esp_image_verify_bootloader_data(&bootloader_data);
- if (err != ESP_OK) {
- ESP_LOGE(TAG, "bootloader image appears invalid! error %d", err);
- return err;
- }
- /* Generate secure boot key and keep in EFUSE */
- uint32_t dis_reg = REG_READ(EFUSE_BLK0_RDATA0_REG);
- bool efuse_key_read_protected = dis_reg & EFUSE_RD_DIS_BLK2;
- bool efuse_key_write_protected = dis_reg & EFUSE_WR_DIS_BLK2;
- if (efuse_key_read_protected == false
- && efuse_key_write_protected == false
- && REG_READ(EFUSE_BLK2_RDATA0_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA1_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA2_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA3_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA4_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA5_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA6_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA7_REG) == 0) {
- ESP_LOGI(TAG, "Generating new secure boot key...");
- esp_efuse_write_random_key(EFUSE_BLK2_WDATA0_REG);
- burn_efuses();
- } else {
- ESP_LOGW(TAG, "Using pre-loaded secure boot key in EFUSE block 2");
- }
- /* Generate secure boot digest using programmed key in EFUSE */
- ESP_LOGI(TAG, "Generating secure boot digest...");
- uint32_t image_len = bootloader_data.image_len;
- if(bootloader_data.image.hash_appended) {
- /* Secure boot digest doesn't cover the hash */
- image_len -= ESP_IMAGE_HASH_LEN;
- }
- if (false == secure_boot_generate(image_len)){
- ESP_LOGE(TAG, "secure boot generation failed");
- return ESP_FAIL;
- }
- ESP_LOGI(TAG, "Digest generation complete.");
- return ESP_OK;
- }
- esp_err_t esp_secure_boot_permanently_enable(void)
- {
- if (esp_secure_boot_enabled()) {
- ESP_LOGI(TAG, "bootloader secure boot is already enabled, continuing..");
- return ESP_OK;
- }
- uint32_t dis_reg = REG_READ(EFUSE_BLK0_RDATA0_REG);
- bool efuse_key_read_protected = dis_reg & EFUSE_RD_DIS_BLK2;
- bool efuse_key_write_protected = dis_reg & EFUSE_WR_DIS_BLK2;
- if (efuse_key_read_protected == false
- && efuse_key_write_protected == false) {
- ESP_LOGI(TAG, "Read & write protecting new key...");
- REG_WRITE(EFUSE_BLK0_WDATA0_REG, EFUSE_WR_DIS_BLK2 | EFUSE_RD_DIS_BLK2);
- burn_efuses();
- efuse_key_read_protected = true;
- efuse_key_write_protected = true;
- }
- if (!efuse_key_read_protected) {
- ESP_LOGE(TAG, "Pre-loaded key is not read protected. Refusing to blow secure boot efuse.");
- return ESP_ERR_INVALID_STATE;
- }
- if (!efuse_key_write_protected) {
- ESP_LOGE(TAG, "Pre-loaded key is not write protected. Refusing to blow secure boot efuse.");
- return ESP_ERR_INVALID_STATE;
- }
- ESP_LOGI(TAG, "blowing secure boot efuse...");
- ESP_LOGD(TAG, "before updating, EFUSE_BLK0_RDATA6 %x", REG_READ(EFUSE_BLK0_RDATA6_REG));
- uint32_t new_wdata6 = EFUSE_RD_ABS_DONE_0;
- #ifndef CONFIG_SECURE_BOOT_ALLOW_JTAG
- ESP_LOGI(TAG, "Disable JTAG...");
- new_wdata6 |= EFUSE_RD_DISABLE_JTAG;
- #else
- ESP_LOGW(TAG, "Not disabling JTAG - SECURITY COMPROMISED");
- #endif
- #ifndef CONFIG_SECURE_BOOT_ALLOW_ROM_BASIC
- ESP_LOGI(TAG, "Disable ROM BASIC interpreter fallback...");
- new_wdata6 |= EFUSE_RD_CONSOLE_DEBUG_DISABLE;
- #else
- ESP_LOGW(TAG, "Not disabling ROM BASIC fallback - SECURITY COMPROMISED");
- #endif
- REG_WRITE(EFUSE_BLK0_WDATA6_REG, new_wdata6);
- burn_efuses();
- uint32_t after = REG_READ(EFUSE_BLK0_RDATA6_REG);
- ESP_LOGD(TAG, "after updating, EFUSE_BLK0_RDATA6 %x", after);
- if (after & EFUSE_RD_ABS_DONE_0) {
- ESP_LOGI(TAG, "secure boot is now enabled for bootloader image");
- return ESP_OK;
- } else {
- ESP_LOGE(TAG, "secure boot not enabled for bootloader image, EFUSE_RD_ABS_DONE_0 is probably write protected!");
- return ESP_ERR_INVALID_STATE;
- }
- }
- #elif CONFIG_SECURE_BOOT_V2_ENABLED
- #define ALIGN_UP(num, align) (((num) + ((align) - 1)) & ~((align) - 1))
- static const char *TAG = "secure_boot_v2";
- #define SIG_BLOCK_MAGIC_BYTE 0xe7
- #define CRC_SIGN_BLOCK_LEN 1196
- #define SIG_BLOCK_PADDING 4096
- #define DIGEST_LEN 32
- static esp_err_t validate_signature_block(const ets_secure_boot_signature_t *sig_block, uint8_t *digest)
- {
- uint32_t crc = crc32_le(0, (uint8_t *)sig_block, CRC_SIGN_BLOCK_LEN);
- if (sig_block->block[0].magic_byte == SIG_BLOCK_MAGIC_BYTE && sig_block->block[0].block_crc == crc && !memcmp(digest, sig_block->block[0].image_digest, DIGEST_LEN)) {
- ESP_LOGI(TAG, "valid signature block found");
- return ESP_OK;
- }
- return ESP_FAIL;
- }
- static esp_err_t secure_boot_v2_digest_generate(uint32_t flash_offset, uint32_t flash_size, uint8_t *public_key_digest)
- {
- esp_err_t ret = ESP_FAIL;
- uint8_t image_digest[DIGEST_LEN] = {0};
- size_t sig_block_addr = flash_offset + ALIGN_UP(flash_size, FLASH_SECTOR_SIZE);
- ret = bootloader_sha256_flash_contents(flash_offset, sig_block_addr - flash_offset, image_digest);
- if (ret != ESP_OK) {
- ESP_LOGE(TAG, "error generating image digest, %d", ret);
- return ret;
- }
- ESP_LOGD(TAG, "reading signature block");
- const ets_secure_boot_signature_t *sig_block = bootloader_mmap(sig_block_addr, sizeof(ets_secure_boot_signature_t));
- if (sig_block == NULL) {
- ESP_LOGE(TAG, "bootloader_mmap(0x%x, 0x%x) failed", sig_block_addr, sizeof(ets_secure_boot_signature_t));
- return ret;
- }
- /* Validating Signature block */
- ret = validate_signature_block(sig_block, image_digest);
- if (ret != ESP_OK) {
- ESP_LOGE(TAG, "signature block (address 0x%x) validation failed %d", sig_block_addr, ret);
- goto done;
- }
- /* Verifying Signature block */
- uint8_t verified_digest[DIGEST_LEN] = {0};
- /* Generating the SHA of the public key components in the signature block */
- bootloader_sha256_handle_t sig_block_sha;
- sig_block_sha = bootloader_sha256_start();
- bootloader_sha256_data(sig_block_sha, &sig_block->block[0].key, sizeof(sig_block->block[0].key));
- bootloader_sha256_finish(sig_block_sha, public_key_digest);
- secure_boot_v2_status_t error;
- error = ets_secure_boot_verify_signature(sig_block, image_digest, public_key_digest, verified_digest);
- if (error != SBV2_SUCCESS) {
- ESP_LOGE(TAG, "secure boot v2 verification failed %d", error);
- ret = ESP_FAIL;
- goto done;
- } else {
- ret = ESP_OK;
- }
- done:
- bootloader_munmap(sig_block);
- return ret;
- }
- esp_err_t esp_secure_boot_v2_permanently_enable(const esp_image_metadata_t *image_data)
- {
- ESP_LOGI(TAG, "enabling secure boot v2...");
- esp_err_t ret;
- if (esp_secure_boot_enabled()) {
- ESP_LOGI(TAG, "secure boot v2 is already enabled. Continuing..");
- return ESP_OK;
- }
- uint32_t coding_scheme = REG_GET_FIELD(EFUSE_BLK0_RDATA6_REG, EFUSE_CODING_SCHEME);
- if (coding_scheme != EFUSE_CODING_SCHEME_VAL_NONE) {
- ESP_LOGE(TAG, "No coding schemes are supported in secure boot v2.(Detected scheme: 0x%x)", coding_scheme);
- return ESP_ERR_NOT_SUPPORTED;
- }
- /* Verify the bootloader */
- esp_image_metadata_t bootloader_data = { 0 };
- ret = esp_image_verify_bootloader_data(&bootloader_data);
- if (ret != ESP_OK) {
- ESP_LOGE(TAG, "bootloader image appears invalid! error %d", ret);
- return ret;
- }
- uint8_t boot_pub_key_digest[DIGEST_LEN];
- uint32_t dis_reg = REG_READ(EFUSE_BLK0_RDATA0_REG);
- bool efuse_key_read_protected = dis_reg & EFUSE_RD_DIS_BLK2;
- bool efuse_key_write_protected = dis_reg & EFUSE_WR_DIS_BLK2;
- if (efuse_key_write_protected == false
- && efuse_key_read_protected == false
- && REG_READ(EFUSE_BLK2_RDATA0_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA1_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA2_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA3_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA4_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA5_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA6_REG) == 0
- && REG_READ(EFUSE_BLK2_RDATA7_REG) == 0) {
- /* Verifies the signature block appended to the image matches with the signature block of the app to be loaded */
- ret = secure_boot_v2_digest_generate(bootloader_data.start_addr, bootloader_data.image_len - SIG_BLOCK_PADDING, boot_pub_key_digest);
- if (ret != ESP_OK) {
- ESP_LOGE(TAG, "Public key digest generation failed");
- return ret;
- }
- ESP_LOGI(TAG, "Burning public key hash to efuse.");
- uint32_t *boot_public_key_digest_ptr = (uint32_t *) boot_pub_key_digest;
- for (int i = 0; i < 8 ; i++) {
- REG_WRITE(EFUSE_BLK2_WDATA0_REG + 4 * i, boot_public_key_digest_ptr[i]);
- ESP_LOGD(TAG, "EFUSE_BLKx_WDATA%d_REG = 0x%08x", i, boot_public_key_digest_ptr[i]);
- }
- ESP_LOGI(TAG, "Write protecting public key digest...");
- REG_WRITE(EFUSE_BLK0_WDATA0_REG, EFUSE_WR_DIS_BLK2);
- efuse_key_write_protected = true;
- efuse_key_read_protected = false;
- } else {
- ESP_LOGW(TAG, "Using pre-loaded secure boot v2 public key digest in EFUSE block 2");
- }
- uint8_t app_pub_key_digest[DIGEST_LEN];
- ret = secure_boot_v2_digest_generate(image_data->start_addr, image_data->image_len - SIG_BLOCK_PADDING, app_pub_key_digest);
- if (ret != ESP_OK) {
- ESP_LOGE(TAG, "Application signature block is invalid.");
- return ret;
- }
- /* Confirming if the public key in the bootloader's signature block matches with the one in the application's signature block */
- if (memcmp(boot_pub_key_digest, app_pub_key_digest, DIGEST_LEN) != 0) {
- ESP_LOGE(TAG, "Application not signed with a valid private key.");
- return ESP_FAIL;
- }
- if (efuse_key_read_protected) {
- ESP_LOGE(TAG, "Efuse BLK2 (public key digest) is read protected. Refusing to blow secure boot efuse.");
- return ESP_ERR_INVALID_STATE;
- }
- if (!efuse_key_write_protected) {
- ESP_LOGE(TAG, "Efuse BLK2 (public key digest) is not write protected. Refusing to blow secure boot efuse.");
- return ESP_ERR_INVALID_STATE;
- }
- ESP_LOGI(TAG, "blowing secure boot efuse...");
- ESP_LOGD(TAG, "before updating, EFUSE_BLK0_RDATA6 %x", REG_READ(EFUSE_BLK0_RDATA6_REG));
- uint32_t new_wdata6 = EFUSE_RD_ABS_DONE_1;
- #ifndef CONFIG_SECURE_BOOT_ALLOW_JTAG
- ESP_LOGI(TAG, "Disable JTAG...");
- new_wdata6 |= EFUSE_RD_DISABLE_JTAG;
- #else
- ESP_LOGW(TAG, "Not disabling JTAG - SECURITY COMPROMISED");
- #endif
- #ifndef CONFIG_SECURE_BOOT_ALLOW_ROM_BASIC
- ESP_LOGI(TAG, "Disable ROM BASIC interpreter fallback...");
- new_wdata6 |= EFUSE_RD_CONSOLE_DEBUG_DISABLE;
- #else
- ESP_LOGW(TAG, "Not disabling ROM BASIC fallback - SECURITY COMPROMISED");
- #endif
- REG_WRITE(EFUSE_BLK0_WDATA6_REG, new_wdata6);
- burn_efuses();
- uint32_t after = REG_READ(EFUSE_BLK0_RDATA6_REG);
- ESP_LOGD(TAG, "after updating, EFUSE_BLK0_RDATA6 %x", after);
- if (after & EFUSE_RD_ABS_DONE_1) {
- ESP_LOGI(TAG, "secure boot v2 is now enabled.");
- return ESP_OK;
- } else {
- ESP_LOGE(TAG, " secure boot v2 not enabled, EFUSE_RD_ABS_DONE_1 is probably write protected!");
- return ESP_ERR_INVALID_STATE;
- }
- }
- #endif // CONFIG_SECURE_BOOT_V2_ENABLED
|