mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
Merge branch 'feature/spi_flash_verify_write' into 'master'
spi_flash: verify_write feature, and ESP-IDF vs ESP-ROM spi flash driver difference doc Closes IDF-5624 See merge request espressif/esp-idf!19998
This commit is contained in:
commit
7f7b0a70eb
@ -3,6 +3,7 @@ menu "SPI Flash driver"
|
|||||||
|
|
||||||
config SPI_FLASH_VERIFY_WRITE
|
config SPI_FLASH_VERIFY_WRITE
|
||||||
bool "Verify SPI flash writes"
|
bool "Verify SPI flash writes"
|
||||||
|
depends on !SPI_FLASH_ROM_IMPL
|
||||||
default n
|
default n
|
||||||
help
|
help
|
||||||
If this option is enabled, any time SPI flash is written then the data will be read
|
If this option is enabled, any time SPI flash is written then the data will be read
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
#include "esp_crypto_lock.h" // for locking flash encryption peripheral
|
#include "esp_crypto_lock.h" // for locking flash encryption peripheral
|
||||||
#endif //CONFIG_IDF_TARGET_ESP32S2
|
#endif //CONFIG_IDF_TARGET_ESP32S2
|
||||||
|
|
||||||
static const char TAG[] = "spi_flash";
|
DRAM_ATTR static const char TAG[] = "spi_flash";
|
||||||
|
|
||||||
#ifdef CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
|
#ifdef CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
|
||||||
#define MAX_WRITE_CHUNK CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE /* write in chunks */
|
#define MAX_WRITE_CHUNK CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE /* write in chunks */
|
||||||
@ -32,6 +32,7 @@ static const char TAG[] = "spi_flash";
|
|||||||
#endif // CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
|
#endif // CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
|
||||||
|
|
||||||
#define MAX_READ_CHUNK 16384
|
#define MAX_READ_CHUNK 16384
|
||||||
|
#define VERIFY_BUF_LEN 64
|
||||||
|
|
||||||
|
|
||||||
#ifdef CONFIG_SPI_FLASH_DANGEROUS_WRITE_ABORTS
|
#ifdef CONFIG_SPI_FLASH_DANGEROUS_WRITE_ABORTS
|
||||||
@ -847,8 +848,91 @@ esp_err_t IRAM_ATTR esp_flash_read(esp_flash_t *chip, void *buffer, uint32_t add
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
static esp_err_t IRAM_ATTR s_check_setting_zero_to_one(esp_flash_t *chip, uint32_t verify_address, uint32_t remain_verify_len, const uint32_t *to_write_buf, bool is_encrypted)
|
||||||
|
{
|
||||||
|
esp_err_t err = ESP_FAIL;
|
||||||
|
uint8_t verify_buffer[VERIFY_BUF_LEN];
|
||||||
|
uint32_t *val_in_flash = (uint32_t *)verify_buffer;
|
||||||
|
|
||||||
|
while (remain_verify_len) {
|
||||||
|
uint32_t this_len = MIN(remain_verify_len, VERIFY_BUF_LEN);
|
||||||
|
|
||||||
|
err = chip->chip_drv->read(chip, verify_buffer, verify_address, this_len);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
ESP_DRAM_LOGE(TAG, "failed to read flash to verify if setting zero to one, err: 0x%x", err);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int r = 0; r < this_len / sizeof(uint32_t); r++) {
|
||||||
|
if (is_encrypted) {
|
||||||
|
(void)to_write_buf;
|
||||||
|
if (val_in_flash[r] != 0xFFFFFFFF) {
|
||||||
|
ESP_DRAM_LOGW(TAG, "Write at offset 0x%x but not erased (0x%08x)",
|
||||||
|
verify_address + r, val_in_flash[r]);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((val_in_flash[r] & to_write_buf[r]) != to_write_buf[r]) {
|
||||||
|
ESP_DRAM_LOGW(TAG, "Write at offset 0x%x requests 0x%08x but will write 0x%08x -> 0x%08x",
|
||||||
|
verify_address + r, to_write_buf[r], val_in_flash[r], (val_in_flash[r] & to_write_buf[r]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
remain_verify_len -= this_len;
|
||||||
|
verify_address += this_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
static esp_err_t IRAM_ATTR s_verify_write(esp_flash_t *chip, uint32_t verify_address, uint32_t remain_verify_len, const uint32_t *expected_buf, bool is_encrypted)
|
||||||
|
{
|
||||||
|
esp_err_t err = ESP_FAIL;
|
||||||
|
uint8_t verify_buffer[VERIFY_BUF_LEN];
|
||||||
|
uint32_t *val_in_flash = (uint32_t *)verify_buffer;
|
||||||
|
|
||||||
|
while (remain_verify_len) {
|
||||||
|
uint32_t this_len = MIN(remain_verify_len, VERIFY_BUF_LEN);
|
||||||
|
|
||||||
|
if (is_encrypted) {
|
||||||
|
err = esp_flash_read_encrypted(chip, verify_address, verify_buffer, this_len);
|
||||||
|
} else {
|
||||||
|
err = chip->chip_drv->read(chip, verify_buffer, verify_address, this_len);
|
||||||
|
}
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
ESP_DRAM_LOGE(TAG, "failed to read flash to verify previous write, err: 0x%x", err);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int r = 0; r < this_len / sizeof(uint32_t); r++) {
|
||||||
|
if (val_in_flash[r] != expected_buf[r]) {
|
||||||
|
#if CONFIG_SPI_FLASH_LOG_FAILED_WRITE
|
||||||
|
ESP_DRAM_LOGE(TAG, "Bad write at %d offset: 0x%x, expected: 0x%08x, readback: 0x%08x", r, verify_address + r, expected_buf[r], val_in_flash[r]);
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_LOG_FAILED_WRITE
|
||||||
|
return ESP_FAIL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
expected_buf = (uint32_t *)((void *)expected_buf + this_len);
|
||||||
|
remain_verify_len -= this_len;
|
||||||
|
verify_address += this_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
|
||||||
esp_err_t IRAM_ATTR esp_flash_write(esp_flash_t *chip, const void *buffer, uint32_t address, uint32_t length)
|
esp_err_t IRAM_ATTR esp_flash_write(esp_flash_t *chip, const void *buffer, uint32_t address, uint32_t length)
|
||||||
{
|
{
|
||||||
|
esp_err_t ret = ESP_FAIL;
|
||||||
|
#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
//used for verify write
|
||||||
|
bool is_encrypted = false;
|
||||||
|
#endif //CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
|
||||||
esp_err_t err = rom_spiflash_api_funcs->chip_check(&chip);
|
esp_err_t err = rom_spiflash_api_funcs->chip_check(&chip);
|
||||||
VERIFY_CHIP_OP(write);
|
VERIFY_CHIP_OP(write);
|
||||||
CHECK_WRITE_ADDRESS(chip, address, length);
|
CHECK_WRITE_ADDRESS(chip, address, length);
|
||||||
@ -898,25 +982,45 @@ esp_err_t IRAM_ATTR esp_flash_write(esp_flash_t *chip, const void *buffer, uint3
|
|||||||
|
|
||||||
err = rom_spiflash_api_funcs->start(chip);
|
err = rom_spiflash_api_funcs->start(chip);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
break;
|
goto restore_cache;
|
||||||
}
|
}
|
||||||
bus_acquired = true;
|
bus_acquired = true;
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
err = s_check_setting_zero_to_one(chip, write_addr, write_len, write_buf, is_encrypted);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
|
}
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
|
||||||
err = chip->chip_drv->write(chip, write_buf, write_addr, write_len);
|
err = chip->chip_drv->write(chip, write_buf, write_addr, write_len);
|
||||||
len_remain -= write_len;
|
len_remain -= write_len;
|
||||||
assert(len_remain < length);
|
assert(len_remain < length);
|
||||||
|
|
||||||
if (err != ESP_OK || len_remain == 0) {
|
if (err != ESP_OK) {
|
||||||
// On ESP32, the cache re-enable is in the end() function, while flush_cache should
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
// happen when the cache is still disabled on ESP32. Break before the end() function and
|
|
||||||
// do end() later
|
|
||||||
assert(bus_acquired);
|
assert(bus_acquired);
|
||||||
|
goto restore_cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
err = s_verify_write(chip, write_addr, write_len, write_buf, is_encrypted);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
|
}
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
|
||||||
|
|
||||||
|
if (len_remain == 0) {
|
||||||
|
//Flash operation done
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = rom_spiflash_api_funcs->end(chip, err);
|
err = rom_spiflash_api_funcs->end(chip, err);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
break;
|
goto restore_cache;
|
||||||
}
|
}
|
||||||
bus_acquired = false;
|
bus_acquired = false;
|
||||||
|
|
||||||
@ -924,11 +1028,28 @@ esp_err_t IRAM_ATTR esp_flash_write(esp_flash_t *chip, const void *buffer, uint3
|
|||||||
buffer = (void *)((intptr_t)buffer + write_len);
|
buffer = (void *)((intptr_t)buffer + write_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
return rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
err = rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
|
||||||
|
restore_cache:
|
||||||
|
|
||||||
|
ret = rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
||||||
|
if (ret != ESP_OK) {
|
||||||
|
ESP_DRAM_LOGE(TAG, "restore cache fail\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t address, const void *buffer, uint32_t length)
|
esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t address, const void *buffer, uint32_t length)
|
||||||
{
|
{
|
||||||
|
esp_err_t ret = ESP_FAIL;
|
||||||
|
#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
//used for verify write
|
||||||
|
bool is_encrypted = true;
|
||||||
|
#endif //CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
|
||||||
esp_err_t err = rom_spiflash_api_funcs->chip_check(&chip);
|
esp_err_t err = rom_spiflash_api_funcs->chip_check(&chip);
|
||||||
// Flash encryption only support on main flash.
|
// Flash encryption only support on main flash.
|
||||||
if (chip != esp_flash_default_chip) {
|
if (chip != esp_flash_default_chip) {
|
||||||
@ -1014,6 +1135,14 @@ esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t addres
|
|||||||
row_size_length = row_size;
|
row_size_length = row_size;
|
||||||
#endif //CONFIG_IDF_TARGET_ESP32
|
#endif //CONFIG_IDF_TARGET_ESP32
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
err = s_check_setting_zero_to_one(chip, row_addr, encrypt_byte, NULL, is_encrypted);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
|
}
|
||||||
|
#endif //#if CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE
|
||||||
|
|
||||||
#if CONFIG_IDF_TARGET_ESP32S2
|
#if CONFIG_IDF_TARGET_ESP32S2
|
||||||
esp_crypto_dma_lock_acquire();
|
esp_crypto_dma_lock_acquire();
|
||||||
#endif //CONFIG_IDF_TARGET_ESP32S2
|
#endif //CONFIG_IDF_TARGET_ESP32S2
|
||||||
@ -1023,7 +1152,8 @@ esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t addres
|
|||||||
#if CONFIG_IDF_TARGET_ESP32S2
|
#if CONFIG_IDF_TARGET_ESP32S2
|
||||||
esp_crypto_dma_lock_release();
|
esp_crypto_dma_lock_release();
|
||||||
#endif //CONFIG_IDF_TARGET_ESP32S2
|
#endif //CONFIG_IDF_TARGET_ESP32S2
|
||||||
break;
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
}
|
}
|
||||||
bus_acquired = true;
|
bus_acquired = true;
|
||||||
|
|
||||||
@ -1034,7 +1164,8 @@ esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t addres
|
|||||||
#endif //CONFIG_IDF_TARGET_ESP32S2
|
#endif //CONFIG_IDF_TARGET_ESP32S2
|
||||||
bus_acquired = false;
|
bus_acquired = false;
|
||||||
assert(bus_acquired);
|
assert(bus_acquired);
|
||||||
break;
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
}
|
}
|
||||||
err = rom_spiflash_api_funcs->end(chip, ESP_OK);
|
err = rom_spiflash_api_funcs->end(chip, ESP_OK);
|
||||||
#if CONFIG_IDF_TARGET_ESP32S2
|
#if CONFIG_IDF_TARGET_ESP32S2
|
||||||
@ -1042,11 +1173,32 @@ esp_err_t IRAM_ATTR esp_flash_write_encrypted(esp_flash_t *chip, uint32_t addres
|
|||||||
#endif //CONFIG_IDF_TARGET_ESP32S2
|
#endif //CONFIG_IDF_TARGET_ESP32S2
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
bus_acquired = false;
|
bus_acquired = false;
|
||||||
break;
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
}
|
}
|
||||||
bus_acquired = false;
|
bus_acquired = false;
|
||||||
|
|
||||||
|
#if CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
|
err = s_verify_write(chip, row_addr, encrypt_byte, (uint32_t *)encrypt_buf, is_encrypted);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
//Error happens, we end flash operation. Re-enable cache and flush it
|
||||||
|
goto restore_cache;
|
||||||
|
}
|
||||||
|
#endif //CONFIG_SPI_FLASH_VERIFY_WRITE
|
||||||
}
|
}
|
||||||
return rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
|
||||||
|
err = rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
|
||||||
|
restore_cache:
|
||||||
|
|
||||||
|
ret = rom_spiflash_api_funcs->flash_end_flush_cache(chip, err, bus_acquired, address, length);
|
||||||
|
if (ret != ESP_OK) {
|
||||||
|
ESP_DRAM_LOGE(TAG, "restore cache fail\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline static IRAM_ATTR bool regions_overlap(uint32_t a_start, uint32_t a_len,uint32_t b_start, uint32_t b_len)
|
inline static IRAM_ATTR bool regions_overlap(uint32_t a_start, uint32_t a_len,uint32_t b_start, uint32_t b_len)
|
||||||
|
@ -317,6 +317,7 @@ esp_err_t esp_flash_read(esp_flash_t *chip, void *buffer, uint32_t address, uint
|
|||||||
*
|
*
|
||||||
* @return
|
* @return
|
||||||
* - ESP_OK on success,
|
* - ESP_OK on success,
|
||||||
|
* - ESP_FAIL, bad write, this will be detected only when CONFIG_SPI_FLASH_VERIFY_WRITE is enabled
|
||||||
* - ESP_ERR_NOT_SUPPORTED if the chip is not able to perform the operation. This is indicated by WREN = 1 after the command is sent.
|
* - ESP_ERR_NOT_SUPPORTED if the chip is not able to perform the operation. This is indicated by WREN = 1 after the command is sent.
|
||||||
* - Other flash error code if operation failed.
|
* - Other flash error code if operation failed.
|
||||||
*/
|
*/
|
||||||
@ -333,6 +334,7 @@ esp_err_t esp_flash_write(esp_flash_t *chip, const void *buffer, uint32_t addres
|
|||||||
*
|
*
|
||||||
* @return
|
* @return
|
||||||
* - ESP_OK: on success
|
* - ESP_OK: on success
|
||||||
|
* - ESP_FAIL: bad write, this will be detected only when CONFIG_SPI_FLASH_VERIFY_WRITE is enabled
|
||||||
* - ESP_ERR_NOT_SUPPORTED: encrypted write not supported for this chip.
|
* - ESP_ERR_NOT_SUPPORTED: encrypted write not supported for this chip.
|
||||||
* - ESP_ERR_INVALID_ARG: Either the address, buffer or length is invalid.
|
* - ESP_ERR_INVALID_ARG: Either the address, buffer or length is invalid.
|
||||||
*/
|
*/
|
||||||
|
@ -411,6 +411,7 @@ TEST_CASE_MULTI_FLASH("SPI flash three byte reads/writes", test_three_byte_read_
|
|||||||
void test_erase_large_region(const esp_partition_t *part)
|
void test_erase_large_region(const esp_partition_t *part)
|
||||||
{
|
{
|
||||||
esp_flash_t* chip = part->flash_chip;
|
esp_flash_t* chip = part->flash_chip;
|
||||||
|
erase_test_region(part, 2);
|
||||||
|
|
||||||
/* Write some noise at the start and the end of the region */
|
/* Write some noise at the start and the end of the region */
|
||||||
const char *ohai = "OHAI";
|
const char *ohai = "OHAI";
|
||||||
|
@ -16,6 +16,7 @@ from pytest_embedded import Dut
|
|||||||
[
|
[
|
||||||
'release',
|
'release',
|
||||||
'flash_qio',
|
'flash_qio',
|
||||||
|
'verify'
|
||||||
],
|
],
|
||||||
indirect=True,
|
indirect=True,
|
||||||
)
|
)
|
||||||
|
@ -1,4 +1,2 @@
|
|||||||
CONFIG_ESP_TASK_WDT_EN=n
|
CONFIG_ESP_TASK_WDT_EN=n
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM=y
|
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions.csv"
|
|
||||||
CONFIG_ESPTOOLPY_FLASHMODE_QIO=y
|
CONFIG_ESPTOOLPY_FLASHMODE_QIO=y
|
||||||
|
@ -3,5 +3,3 @@ CONFIG_FREERTOS_USE_TICKLESS_IDLE=y
|
|||||||
CONFIG_COMPILER_OPTIMIZATION_SIZE=y
|
CONFIG_COMPILER_OPTIMIZATION_SIZE=y
|
||||||
CONFIG_BOOTLOADER_COMPILER_OPTIMIZATION_SIZE=y
|
CONFIG_BOOTLOADER_COMPILER_OPTIMIZATION_SIZE=y
|
||||||
CONFIG_COMPILER_OPTIMIZATION_ASSERTIONS_SILENT=y
|
CONFIG_COMPILER_OPTIMIZATION_ASSERTIONS_SILENT=y
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM=y
|
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions.csv"
|
|
||||||
|
@ -1,4 +1,2 @@
|
|||||||
CONFIG_ESP_TASK_WDT_EN=n
|
CONFIG_ESP_TASK_WDT_EN=n
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM=y
|
|
||||||
CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions.csv"
|
|
||||||
CONFIG_SPI_FLASH_ROM_IMPL=y
|
CONFIG_SPI_FLASH_ROM_IMPL=y
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
CONFIG_ESP_TASK_WDT=n
|
||||||
|
CONFIG_SPI_FLASH_VERIFY_WRITE=y
|
||||||
|
CONFIG_SPI_FLASH_LOG_FAILED_WRITE=y
|
||||||
|
CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE=y
|
@ -0,0 +1,19 @@
|
|||||||
|
CONFIG_ESP_TASK_WDT_EN=n
|
||||||
|
CONFIG_FREERTOS_USE_TICKLESS_IDLE=y
|
||||||
|
CONFIG_COMPILER_OPTIMIZATION_SIZE=y
|
||||||
|
CONFIG_BOOTLOADER_COMPILER_OPTIMIZATION_SIZE=y
|
||||||
|
CONFIG_COMPILER_OPTIMIZATION_ASSERTIONS_SILENT=y
|
||||||
|
CONFIG_PARTITION_TABLE_CUSTOM=y
|
||||||
|
CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions.csv"
|
||||||
|
CONFIG_SECURE_FLASH_ENC_ENABLED=y
|
||||||
|
CONFIG_SECURE_FLASH_ENCRYPTION_MODE_DEVELOPMENT=y
|
||||||
|
CONFIG_SECURE_BOOT_ALLOW_ROM_BASIC=y
|
||||||
|
CONFIG_SECURE_BOOT_ALLOW_JTAG=y
|
||||||
|
CONFIG_SECURE_FLASH_UART_BOOTLOADER_ALLOW_ENC=y
|
||||||
|
CONFIG_SECURE_FLASH_UART_BOOTLOADER_ALLOW_DEC=y
|
||||||
|
CONFIG_SECURE_FLASH_UART_BOOTLOADER_ALLOW_CACHE=y
|
||||||
|
CONFIG_SECURE_FLASH_REQUIRE_ALREADY_ENABLED=y
|
||||||
|
|
||||||
|
CONFIG_SPI_FLASH_VERIFY_WRITE=y
|
||||||
|
CONFIG_SPI_FLASH_LOG_FAILED_WRITE=y
|
||||||
|
CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE=y
|
@ -134,7 +134,7 @@ The following options will reduce IRAM usage of some ESP-IDF features:
|
|||||||
- Enable :ref:`CONFIG_RINGBUF_PLACE_FUNCTIONS_INTO_FLASH`. Provided these functions are not (incorrectly) used from ISRs, this option is safe to enable in all configurations.
|
- Enable :ref:`CONFIG_RINGBUF_PLACE_FUNCTIONS_INTO_FLASH`. Provided these functions are not (incorrectly) used from ISRs, this option is safe to enable in all configurations.
|
||||||
- Enable :ref:`CONFIG_RINGBUF_PLACE_ISR_FUNCTIONS_INTO_FLASH`. This option is not safe to use if the ISR ringbuf functions are used from an IRAM interrupt context, e.g. if :ref:`CONFIG_UART_ISR_IN_IRAM` is enabled. For the IDF drivers where this is the case you will get an error at run-time when installing the driver in question.
|
- Enable :ref:`CONFIG_RINGBUF_PLACE_ISR_FUNCTIONS_INTO_FLASH`. This option is not safe to use if the ISR ringbuf functions are used from an IRAM interrupt context, e.g. if :ref:`CONFIG_UART_ISR_IN_IRAM` is enabled. For the IDF drivers where this is the case you will get an error at run-time when installing the driver in question.
|
||||||
:SOC_WIFI_SUPPORTED: - Disable Wi-Fi options :ref:`CONFIG_ESP_WIFI_IRAM_OPT` and/or :ref:`CONFIG_ESP_WIFI_RX_IRAM_OPT`. Disabling these options will free available IRAM at the cost of Wi-Fi performance.
|
:SOC_WIFI_SUPPORTED: - Disable Wi-Fi options :ref:`CONFIG_ESP_WIFI_IRAM_OPT` and/or :ref:`CONFIG_ESP_WIFI_RX_IRAM_OPT`. Disabling these options will free available IRAM at the cost of Wi-Fi performance.
|
||||||
:esp32c3 or esp32s3: - :ref:`CONFIG_SPI_FLASH_ROM_IMPL` enabling this option will free some IRAM but will mean that esp_flash bugfixes and new flash chip support is not available.
|
:CONFIG_ESP_ROM_HAS_SPI_FLASH: - :ref:`CONFIG_SPI_FLASH_ROM_IMPL` enabling this option will free some IRAM but will mean that esp_flash bugfixes and new flash chip support is not available, see :doc:`/api-reference/peripherals/spi_flash/spi_flash_idf_vs_rom` for details.
|
||||||
:esp32: - :ref:`CONFIG_SPI_FLASH_ROM_DRIVER_PATCH` disabling this option will free some IRAM but is only available in some flash configurations (see the configuration item help text).
|
:esp32: - :ref:`CONFIG_SPI_FLASH_ROM_DRIVER_PATCH` disabling this option will free some IRAM but is only available in some flash configurations (see the configuration item help text).
|
||||||
:esp32: - If the application uses PSRAM and is based on ESP32 rev. 3 (ECO3), setting :ref:`CONFIG_ESP32_REV_MIN` to ``3`` will disable PSRAM bug workarounds, saving ~10kB or more of IRAM.
|
:esp32: - If the application uses PSRAM and is based on ESP32 rev. 3 (ECO3), setting :ref:`CONFIG_ESP32_REV_MIN` to ``3`` will disable PSRAM bug workarounds, saving ~10kB or more of IRAM.
|
||||||
- Disabling :ref:`CONFIG_ESP_EVENT_POST_FROM_IRAM_ISR` prevents posting ``esp_event`` events from :ref:`iram-safe-interrupt-handlers` but will save some IRAM.
|
- Disabling :ref:`CONFIG_ESP_EVENT_POST_FROM_IRAM_ISR` prevents posting ``esp_event`` events from :ref:`iram-safe-interrupt-handlers` but will save some IRAM.
|
||||||
|
@ -250,6 +250,20 @@ Additionally, all API functions are protected with a mutex (``s_flash_op_mutex``
|
|||||||
|
|
||||||
In a single core environment (:ref:`CONFIG_FREERTOS_UNICORE` enabled), you need to disable both caches, so that no inter-CPU communication can take place.
|
In a single core environment (:ref:`CONFIG_FREERTOS_UNICORE` enabled), you need to disable both caches, so that no inter-CPU communication can take place.
|
||||||
|
|
||||||
|
|
||||||
|
.. toctree::
|
||||||
|
:hidden:
|
||||||
|
|
||||||
|
spi_flash_idf_vs_rom
|
||||||
|
|
||||||
|
.. only:: CONFIG_ESP_ROM_HAS_SPI_FLASH
|
||||||
|
|
||||||
|
ESP-IDF vs Chip-ROM SPI Flash Driver
|
||||||
|
------------------------------------
|
||||||
|
|
||||||
|
Refer to :doc:`spi_flash_idf_vs_rom`.
|
||||||
|
|
||||||
|
|
||||||
API Reference - SPI Flash
|
API Reference - SPI Flash
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
@ -0,0 +1,37 @@
|
|||||||
|
SPI Flash API ESP-IDF version vs Chip-ROM version
|
||||||
|
=================================================
|
||||||
|
|
||||||
|
.. toctree::
|
||||||
|
:maxdepth: 1
|
||||||
|
|
||||||
|
There is a set of SPI Flash drivers in Chip-ROM which you can use by enabling :ref:`CONFIG_SPI_FLASH_ROM_IMPL`. Most of the ESP-IDF SPI Flash driver code are in internal RAM, therefore enabling this option will free some internal RAM usage. Note if you enable this option, this means some SPI Flash driver features and bugfixes that are done in ESP-IDF might not be included in the Chip-ROM version.
|
||||||
|
|
||||||
|
|
||||||
|
Feature Supported by ESP-IDF but not in Chip-ROM
|
||||||
|
------------------------------------------------
|
||||||
|
|
||||||
|
.. list::
|
||||||
|
|
||||||
|
- Octal Flash chip support. See :ref:`oct-flash-doc` for details.
|
||||||
|
- 32-bit-address support for GD25Q256. See :ref:`32-bit-flash-doc` for details.
|
||||||
|
- TH Flash chip support.
|
||||||
|
- Kconfig option :ref:`CONFIG_SPI_FLASH_CHECK_ERASE_TIMEOUT_DISABLED`.
|
||||||
|
- :ref:`CONFIG_SPI_FLASH_VERIFY_WRITE`, enabling this option helps you detect bad writing.
|
||||||
|
- :ref:`CONFIG_SPI_FLASH_LOG_FAILED_WRITE`, enabling this option will print the bad writing.
|
||||||
|
- :ref:`CONFIG_SPI_FLASH_WARN_SETTING_ZERO_TO_ONE`, enabling this option will check if you're writing zero to one.
|
||||||
|
:esp32h2 or esp32c6: - Flash MMAP driver isn't ready in Chip-ROM.
|
||||||
|
|
||||||
|
|
||||||
|
Bugfixes Introduced in ESP-IDF but not in Chip-ROM
|
||||||
|
--------------------------------------------------
|
||||||
|
|
||||||
|
.. list::
|
||||||
|
|
||||||
|
- Detected Flash physical size correctly, for larger than 256MBit Flash chips. (Commit ID: b4964279d44f73cce7cfd5cf684567fbdfd6fd9e)
|
||||||
|
:esp32c3: - Improved SPI1 cs setup timing, otherwise issue may happen on ZB32Q128. (Commit ID: 08f1bbe0c75382f1702e40c941e93314285105d4)
|
||||||
|
:esp32s3: - Fixed issue that 4-line Flash encryption cannot work normally when 8-line PSRAM enabled. (Commit ID: 683d92bc884e0f2a7eebea40a551cf05f0c28256)
|
||||||
|
:esp32s2: - Fixed issue that only 4MB virtual address ranges can be mapped to read-only data on Flash.
|
||||||
|
:esp32s3: - Fixed issue that only 128KB virtual address ranges can be mapped to instructions on Flash.
|
||||||
|
:esp32s3: - Fixed issue that only 16MB virtual address ranges can be mapped to read-only data on Flash.
|
||||||
|
:esp32c3: - Fixed issue that only 128KB virtual address ranges can be mapped to instructions on Flash.
|
||||||
|
:esp32c2: - Fixed issue that only at most 128KB virtual address ranges can be mapped to instructions on Flash.
|
@ -59,6 +59,8 @@ Flash Chips List:
|
|||||||
6. XMC
|
6. XMC
|
||||||
7. BOYA
|
7. BOYA
|
||||||
|
|
||||||
|
.. _hpm-doc:
|
||||||
|
|
||||||
High performance mode
|
High performance mode
|
||||||
---------------------
|
---------------------
|
||||||
|
|
||||||
@ -83,6 +85,9 @@ Flash Chips (name & ID) List:
|
|||||||
|
|
||||||
It is hard to create several strategies to cover all situations, so all flash chips using HPM need to be supported explicitly. Therefore, if you try to use a flash not listed as supported under high performance mode, it might cause some error. So, when you try to use the flash chip beyond supported list, please test properly.
|
It is hard to create several strategies to cover all situations, so all flash chips using HPM need to be supported explicitly. Therefore, if you try to use a flash not listed as supported under high performance mode, it might cause some error. So, when you try to use the flash chip beyond supported list, please test properly.
|
||||||
|
|
||||||
|
|
||||||
|
.. _oct-flash-doc:
|
||||||
|
|
||||||
OPI flash support
|
OPI flash support
|
||||||
-----------------
|
-----------------
|
||||||
|
|
||||||
@ -102,6 +107,9 @@ Flash Chips List:
|
|||||||
|
|
||||||
1. MX25UM25645G
|
1. MX25UM25645G
|
||||||
|
|
||||||
|
|
||||||
|
.. _32-bit-flash-doc:
|
||||||
|
|
||||||
32-bit Address Flash Chips
|
32-bit Address Flash Chips
|
||||||
--------------------------
|
--------------------------
|
||||||
|
|
||||||
|
@ -250,6 +250,20 @@ Flash 操作完成后,CPU A 上的函数将设置另一标志位,即 ``s_fla
|
|||||||
|
|
||||||
在单核环境中(启用 :ref:`CONFIG_FREERTOS_UNICORE`),您需要禁用上述两个 cache 以防发生 CPU 间通信。
|
在单核环境中(启用 :ref:`CONFIG_FREERTOS_UNICORE`),您需要禁用上述两个 cache 以防发生 CPU 间通信。
|
||||||
|
|
||||||
|
|
||||||
|
.. toctree::
|
||||||
|
:hidden:
|
||||||
|
|
||||||
|
spi_flash_idf_vs_rom
|
||||||
|
|
||||||
|
.. only:: CONFIG_ESP_ROM_HAS_SPI_FLASH
|
||||||
|
|
||||||
|
ESP-IDF 和 Chip-ROM 版本 SPI Flash 驱动对比
|
||||||
|
-----------------------------------------------------------------
|
||||||
|
|
||||||
|
请参考 :doc:`spi_flash_idf_vs_rom`.
|
||||||
|
|
||||||
|
|
||||||
SPI Flash API 参考
|
SPI Flash API 参考
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
@ -0,0 +1 @@
|
|||||||
|
.. include:: /../en/api-reference/peripherals/spi_flash/spi_flash_idf_vs_rom.rst
|
Loading…
x
Reference in New Issue
Block a user