| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 |
- // Copyright 2017-2018 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 "esp_efuse.h"
- #include "esp_efuse_utility.h"
- #include "soc/efuse_periph.h"
- #include "assert.h"
- #include "sdkconfig.h"
- #include "esp_efuse_table.h"
- const static char *TAG = "efuse";
- #if defined(BOOTLOADER_BUILD)
- #define EFUSE_LOCK_ACQUIRE()
- #define EFUSE_LOCK_RELEASE()
- #define EFUSE_LOCK_ACQUIRE_RECURSIVE()
- #define EFUSE_LOCK_RELEASE_RECURSIVE()
- #else
- #include <sys/lock.h>
- static _lock_t s_efuse_lock;
- #define EFUSE_LOCK_ACQUIRE() _lock_acquire(&s_efuse_lock)
- #define EFUSE_LOCK_RELEASE() _lock_release(&s_efuse_lock)
- #define EFUSE_LOCK_ACQUIRE_RECURSIVE() _lock_acquire_recursive(&s_efuse_lock)
- #define EFUSE_LOCK_RELEASE_RECURSIVE() _lock_release_recursive(&s_efuse_lock)
- #endif
- static bool s_batch_writing_mode = false;
- // Public API functions
- // read value from EFUSE, writing it into an array
- esp_err_t esp_efuse_read_field_blob(const esp_efuse_desc_t* field[], void* dst, size_t dst_size_bits)
- {
- EFUSE_LOCK_ACQUIRE();
- esp_err_t err = ESP_OK;
- if (field == NULL || dst == NULL || dst_size_bits == 0) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- memset((uint8_t *)dst, 0, esp_efuse_utility_get_number_of_items(dst_size_bits, 8));
- err = esp_efuse_utility_process(field, dst, dst_size_bits, esp_efuse_utility_fill_buff);
- }
- EFUSE_LOCK_RELEASE();
- return err;
- }
- bool esp_efuse_read_field_bit(const esp_efuse_desc_t *field[])
- {
- uint8_t value = 0;
- esp_err_t err = esp_efuse_read_field_blob(field, &value, 1);
- assert(err == ESP_OK);
- return (err == ESP_OK) && value;
- }
- // read number of bits programmed as "1" in the particular field
- esp_err_t esp_efuse_read_field_cnt(const esp_efuse_desc_t* field[], size_t* out_cnt)
- {
- EFUSE_LOCK_ACQUIRE();
- esp_err_t err = ESP_OK;
- if (field == NULL || out_cnt == NULL) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- *out_cnt = 0;
- err = esp_efuse_utility_process(field, out_cnt, 0, esp_efuse_utility_count_once);
- }
- EFUSE_LOCK_RELEASE();
- return err;
- }
- // write array to EFUSE
- esp_err_t esp_efuse_write_field_blob(const esp_efuse_desc_t* field[], const void* src, size_t src_size_bits)
- {
- EFUSE_LOCK_ACQUIRE_RECURSIVE();
- esp_err_t err = ESP_OK;
- if (field == NULL || src == NULL || src_size_bits == 0) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- if (s_batch_writing_mode == false) {
- esp_efuse_utility_reset();
- }
- err = esp_efuse_utility_process(field, (void*)src, src_size_bits, esp_efuse_utility_write_blob);
- if (s_batch_writing_mode == false) {
- if (err == ESP_OK) {
- err = esp_efuse_utility_apply_new_coding_scheme();
- if (err == ESP_OK) {
- esp_efuse_utility_burn_efuses();
- }
- }
- esp_efuse_utility_reset();
- }
- }
- EFUSE_LOCK_RELEASE_RECURSIVE();
- return err;
- }
- // program cnt bits to "1"
- esp_err_t esp_efuse_write_field_cnt(const esp_efuse_desc_t* field[], size_t cnt)
- {
- EFUSE_LOCK_ACQUIRE_RECURSIVE();
- esp_err_t err = ESP_OK;
- if (field == NULL || cnt == 0) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- if (s_batch_writing_mode == false) {
- esp_efuse_utility_reset();
- }
- err = esp_efuse_utility_process(field, &cnt, 0, esp_efuse_utility_write_cnt);
- if (cnt != 0) {
- ESP_LOGE(TAG, "The required number of bits can not be set. [Not set %d]", cnt);
- err = ESP_ERR_EFUSE_CNT_IS_FULL;
- }
- if (err == ESP_OK_EFUSE_CNT) {
- err = ESP_OK;
- }
- if (s_batch_writing_mode == false) {
- if (err == ESP_OK) {
- err = esp_efuse_utility_apply_new_coding_scheme();
- if (err == ESP_OK) {
- esp_efuse_utility_burn_efuses();
- }
- }
- esp_efuse_utility_reset();
- }
- }
- EFUSE_LOCK_RELEASE_RECURSIVE();
- return err;
- }
- esp_err_t esp_efuse_write_field_bit(const esp_efuse_desc_t* field[])
- {
- esp_err_t err;
- uint8_t existing = 0;
- const uint8_t one = 1;
- if (field == NULL || field[0]->bit_count != 1) {
- return ESP_ERR_INVALID_ARG;
- }
- /* Check existing value. esp_efuse_write_field_blob() also checks this, but will log an error */
- err = esp_efuse_read_field_blob(field, &existing, 1);
- if (err != ESP_OK || existing) {
- return err; // Error reading, or the bit is already written and we can no-op this
- }
- return esp_efuse_write_field_blob(field, &one, 1);
- }
- // get the length of the field in bits
- int esp_efuse_get_field_size(const esp_efuse_desc_t* field[])
- {
- int bits_counter = 0;
- if (field != NULL) {
- int i = 0;
- while (field[i] != NULL) {
- bits_counter += field[i]->bit_count;
- ++i;
- }
- }
- return bits_counter;
- }
- // reading efuse register.
- uint32_t esp_efuse_read_reg(esp_efuse_block_t blk, unsigned int num_reg)
- {
- EFUSE_LOCK_ACQUIRE();
- uint32_t ret_val = esp_efuse_utility_read_reg(blk, num_reg);
- EFUSE_LOCK_RELEASE();
- return ret_val;
- }
- // writing efuse register.
- esp_err_t esp_efuse_write_reg(esp_efuse_block_t blk, unsigned int num_reg, uint32_t val)
- {
- EFUSE_LOCK_ACQUIRE_RECURSIVE();
- if (s_batch_writing_mode == false) {
- esp_efuse_utility_reset();
- }
- esp_err_t err = esp_efuse_utility_write_reg(blk, num_reg, val);
- if (s_batch_writing_mode == false) {
- if (err == ESP_OK) {
- err = esp_efuse_utility_apply_new_coding_scheme();
- if (err == ESP_OK) {
- esp_efuse_utility_burn_efuses();
- }
- }
- esp_efuse_utility_reset();
- }
- EFUSE_LOCK_RELEASE_RECURSIVE();
- return err;
- }
- // This function reads the key from the efuse block, starting at the offset and the required size.
- esp_err_t esp_efuse_read_block(esp_efuse_block_t blk, void* dst_key, size_t offset_in_bits, size_t size_bits)
- {
- esp_err_t err = ESP_OK;
- if (blk == EFUSE_BLK0 || blk >= EFUSE_BLK_MAX || dst_key == NULL || size_bits == 0) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- const esp_efuse_desc_t field_desc[] = {
- {blk, offset_in_bits, size_bits},
- };
- const esp_efuse_desc_t* field[] = {
- &field_desc[0],
- NULL
- };
- err = esp_efuse_read_field_blob(field, dst_key, size_bits);
- }
- return err;
- }
- // This function writes the key from the efuse block, starting at the offset and the required size.
- esp_err_t esp_efuse_write_block(esp_efuse_block_t blk, const void* src_key, size_t offset_in_bits, size_t size_bits)
- {
- esp_err_t err = ESP_OK;
- if (blk == EFUSE_BLK0 || blk >= EFUSE_BLK_MAX || src_key == NULL || size_bits == 0) {
- err = ESP_ERR_INVALID_ARG;
- } else {
- const esp_efuse_desc_t field_desc[] = {
- {blk, offset_in_bits, size_bits},
- };
- const esp_efuse_desc_t* field[] = {
- &field_desc[0],
- NULL
- };
- err = esp_efuse_write_field_blob(field, src_key, size_bits);
- }
- return err;
- }
- esp_err_t esp_efuse_batch_write_begin(void)
- {
- EFUSE_LOCK_ACQUIRE();
- s_batch_writing_mode = true;
- esp_efuse_utility_reset();
- ESP_LOGI(TAG, "Batch mode of writing fields is enabled");
- return ESP_OK;
- }
- esp_err_t esp_efuse_batch_write_cancel(void)
- {
- if (s_batch_writing_mode == true) {
- s_batch_writing_mode = false;
- esp_efuse_utility_reset();
- ESP_LOGI(TAG, "Batch mode of writing fields is disabled");
- EFUSE_LOCK_RELEASE();
- return ESP_OK;
- } else {
- return ESP_ERR_INVALID_STATE;
- }
- }
- esp_err_t esp_efuse_batch_write_commit(void)
- {
- if (s_batch_writing_mode == false) {
- return ESP_ERR_INVALID_STATE;
- } else {
- esp_err_t err = esp_efuse_utility_apply_new_coding_scheme();
- if (err == ESP_OK) {
- esp_efuse_utility_burn_efuses();
- }
- esp_efuse_batch_write_cancel();
- return err;
- }
- }
|