/* * This file is part of the Pico Keys SDK distribution (https://github.com/polhenarejos/pico-keys-sdk). * Copyright (c) 2022 Pol Henarejos. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, version 3. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "file.h" #include "pico_keys.h" #include #include "otp.h" #ifdef PICO_RP2350 #include "pico/bootrom.h" #include "hardware/structs/otp.h" #include "hardware/regs/otp_data.h" #endif #include "random.h" #include "mbedtls/ecdsa.h" #ifdef PICO_RP2350 static bool is_empty_buffer(const uint8_t *buffer, uint16_t buffer_len) { for (int i = 0; i < buffer_len; i++) { if (buffer[i] != 0x00) { return false; } } return true; } static int otp_write_data_mode(uint16_t row, uint8_t *data, uint16_t len, bool is_ecc) { otp_cmd_t cmd = { .flags = row | (is_ecc ? OTP_CMD_ECC_BITS : 0) | OTP_CMD_WRITE_BITS }; uint32_t ret = rom_func_otp_access(data, len, cmd); if (ret) { printf("OTP Write failed with error: %ld\n", ret); } return ret; } int otp_write_data(uint16_t row, uint8_t *data, uint16_t len) { return otp_write_data_mode(row, data, len, true); } int otp_write_data_raw(uint16_t row, uint8_t *data, uint16_t len) { return otp_write_data_mode(row, data, len, false); } uint8_t* otp_buffer(uint16_t row) { volatile uint32_t *p = ((uint32_t *)(OTP_DATA_BASE + (row*2))); return (uint8_t *)p; } uint8_t* otp_buffer_raw(uint16_t row) { volatile uint32_t *p = ((uint32_t *)(OTP_DATA_RAW_BASE + (row*4))); return (uint8_t *)p; } bool is_empty_otp_buffer(uint16_t row, uint16_t len) { return is_empty_buffer(otp_buffer(row), len); } static bool is_otp_locked_page(uint8_t page) { volatile uint32_t *p = ((uint32_t *)(OTP_DATA_BASE + ((OTP_DATA_PAGE0_LOCK0_ROW + page*2)*2))); return ((p[0] & 0xFFFF0000) == 0x3C3C0000 && (p[1] & 0xFF) == 0x3C); } static void otp_lock_page(uint8_t page) { if (!is_otp_locked_page(page)) { uint32_t value = 0x3c3c3c; otp_write_data_raw(OTP_DATA_PAGE0_LOCK0_ROW + page*2 + 1, (uint8_t *)&value, sizeof(value)); } otp_hw->sw_lock[page] = 0b1100; } #endif #ifdef ESP_PLATFORM uint8_t _otp_key_1[32] = {0}; uint8_t _otp_key_2[32] = {0}; esp_err_t read_key_from_efuse(esp_efuse_block_t block, uint8_t *key, size_t key_len) { const esp_efuse_desc_t **key_desc = esp_efuse_get_key(block); if (!key_desc) { return ESP_FAIL; } return esp_efuse_read_field_blob(key_desc, key, key_len * 8); } #endif const uint8_t *otp_key_1 = NULL; const uint8_t *otp_key_2 = NULL; #ifdef PICO_RP2350 typedef int otp_ret_t; #define OTP_WRITE(ROW, DATA, LEN) otp_write_data(ROW, DATA, LEN) #define OTP_READ(ROW, PTR) do { PTR = otp_buffer(ROW); } while(0) #define OTP_EMTPY(ROW, LEN) is_empty_otp_buffer(ROW, LEN) #elif defined(ESP_PLATFORM) typedef esp_err_t otp_ret_t; #define OTP_WRITE(ROW, DATA, LEN) esp_efuse_write_key(ROW, ESP_EFUSE_KEY_PURPOSE_USER, DATA, LEN); #define OTP_READ(ROW, PTR) do { \ esp_err_t ret = read_key_from_efuse(ROW, _##PTR, sizeof(_##PTR)); \ if (ret != ESP_OK) { printf("Error reading OTP key 1 [%d]\n", ret); } \ PTR = _##PTR; } while(0) #define OTP_EMTPY(ROW, LEN) esp_efuse_key_block_unused(ROW) #endif #ifndef SECURE_BOOT_BOOTKEY_INDEX #define SECURE_BOOT_BOOTKEY_INDEX 0 #endif int otp_enable_secure_boot(uint8_t bootkey, bool secure_lock) { int ret = 0; #ifdef PICO_RP2350 uint8_t BOOTKEY[] = "\xe1\xd1\x6b\xa7\x64\xab\xd7\x12\xd4\xef\x6e\x3e\xdd\x74\x4e\xd5\x63\x8c\x26\xb\x77\x1c\xf9\x81\x51\x11\xb\xaf\xac\x9b\xc8\x71"; if (is_empty_otp_buffer(OTP_DATA_BOOTKEY0_0_ROW + 0x10*bootkey, 32)) { PICOKEY_CHECK(otp_write_data(OTP_DATA_BOOTKEY0_0_ROW + 0x10*bootkey, BOOTKEY, sizeof(BOOTKEY))); } uint8_t *boot_flags1 = otp_buffer_raw(OTP_DATA_BOOT_FLAGS1_ROW); uint8_t flagsb1[] = { boot_flags1[0] | (1 << (bootkey + OTP_DATA_BOOT_FLAGS1_KEY_VALID_LSB)), boot_flags1[1], boot_flags1[2], 0x00 }; if (secure_lock) { flagsb1[1] |= ((OTP_DATA_BOOT_FLAGS1_KEY_INVALID_BITS >> OTP_DATA_BOOT_FLAGS1_KEY_INVALID_LSB) & (~(1 << bootkey))); } PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_BOOT_FLAGS1_ROW, flagsb1, sizeof(flagsb1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_BOOT_FLAGS1_R1_ROW, flagsb1, sizeof(flagsb1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_BOOT_FLAGS1_R2_ROW, flagsb1, sizeof(flagsb1))); uint8_t *crit1 = otp_buffer_raw(OTP_DATA_CRIT1_ROW); uint8_t flagsc1[] = { crit1[0] | (1 << OTP_DATA_CRIT1_SECURE_BOOT_ENABLE_LSB), crit1[1], crit1[2], 0x00 }; if (secure_lock) { flagsc1[0] |= (1 << OTP_DATA_CRIT1_DEBUG_DISABLE_LSB); flagsc1[0] |= (1 << OTP_DATA_CRIT1_GLITCH_DETECTOR_ENABLE_LSB); flagsc1[0] |= (3 << OTP_DATA_CRIT1_GLITCH_DETECTOR_SENS_LSB); } PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R1_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R2_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R3_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R4_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R5_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R6_ROW, flagsc1, sizeof(flagsc1))); PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_CRIT1_R7_ROW, flagsc1, sizeof(flagsc1))); if (secure_lock) { uint8_t *page1 = otp_buffer_raw(OTP_DATA_PAGE1_LOCK1_ROW); uint8_t page1v = page1[0] | (OTP_DATA_PAGE1_LOCK1_LOCK_BL_VALUE_READ_ONLY << OTP_DATA_PAGE1_LOCK1_LOCK_BL_LSB); uint8_t flagsp1[] = { page1v, page1v, page1v, 0x00 }; PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_PAGE1_LOCK1_ROW, flagsp1, sizeof(flagsp1))); uint8_t *page2 = otp_buffer_raw(OTP_DATA_PAGE2_LOCK1_ROW); uint8_t page2v = page2[0] | (OTP_DATA_PAGE2_LOCK1_LOCK_BL_VALUE_READ_ONLY << OTP_DATA_PAGE2_LOCK1_LOCK_BL_LSB); uint8_t flagsp2[] = { page2v, page2v, page2v, 0x00 }; PICOKEY_CHECK(otp_write_data_raw(OTP_DATA_PAGE2_LOCK1_ROW, flagsp2, sizeof(flagsp2))); } #endif // PICO_RP2350 err: if (ret != PICOKEY_OK) { return ret; } return PICOKEY_OK; } void init_otp_files() { #if defined(PICO_RP2350) || defined(ESP_PLATFORM) otp_ret_t ret = 0; uint16_t write_otp[2] = {0xFFFF, 0xFFFF}; if (OTP_EMTPY(OTP_KEY_1, 32)) { uint8_t mkek[32] = {0}; random_gen(NULL, mkek, sizeof(mkek)); ret = OTP_WRITE(OTP_KEY_1, mkek, sizeof(mkek)); if (ret != 0) { printf("Error writing OTP key 1 [%d]\n", ret); } write_otp[0] = OTP_KEY_1; } OTP_READ(OTP_KEY_1, otp_key_1); if (OTP_EMTPY(OTP_KEY_2, 32)) { mbedtls_ecdsa_context ecdsa; size_t olen = 0; uint8_t pkey[MBEDTLS_ECP_MAX_BYTES]; while (olen != 32) { mbedtls_ecdsa_init(&ecdsa); mbedtls_ecp_group_id ec_id = MBEDTLS_ECP_DP_SECP256K1; mbedtls_ecdsa_genkey(&ecdsa, ec_id, random_gen, NULL); mbedtls_ecp_write_key_ext(&ecdsa, &olen, pkey, sizeof(pkey)); mbedtls_ecdsa_free(&ecdsa); } ret = OTP_WRITE(OTP_KEY_2, pkey, olen); if (ret != 0) { printf("Error writing OTP key 2 [%d]\n", ret); } write_otp[1] = OTP_KEY_2; } OTP_READ(OTP_KEY_2, otp_key_2); for (int i = 0; i < sizeof(write_otp)/sizeof(uint16_t); i++) { if (write_otp[i] != 0xFFFF) { #if defined(PICO_RP2350) otp_lock_page(write_otp[i] >> 6); #elif defined(ESP_PLATFORM) ret = esp_efuse_set_key_dis_write(write_otp[i]); if (ret != ESP_OK) { printf("Error setting OTP key %d to read only [%d]\n", i, ret); } ret = esp_efuse_set_keypurpose_dis_write(write_otp[i]); if (ret != ESP_OK) { printf("Error setting OTP key %d purpose to read only [%d]\n", i, ret); } #endif } } #endif // PICO_RP2350 || ESP_PLATFORM }