2021-11-06 17:23:21 +08:00
/*
2024-02-22 18:05:47 +08:00
* SPDX - FileCopyrightText : 2015 - 2024 Espressif Systems ( Shanghai ) CO LTD
2021-11-06 17:23:21 +08:00
*
* SPDX - License - Identifier : Apache - 2.0
*/
2019-01-08 18:29:25 +08:00
# include <stdlib.h>
# include <stdio.h>
# include <sys/param.h>
# include <string.h>
2022-04-01 16:53:40 +08:00
# include "esp_memory_utils.h"
2019-01-08 18:29:25 +08:00
# include "spi_flash_chip_driver.h"
# include "memspi_host_driver.h"
# include "esp_log.h"
# include "sdkconfig.h"
2019-09-12 02:41:00 +08:00
# include "esp_flash_internal.h"
2020-12-18 12:57:55 +08:00
# include "spi_flash_defs.h"
2022-06-27 15:24:07 +08:00
# include "spi_flash_mmap.h"
2021-07-24 13:28:25 +02:00
# include "esp_rom_caps.h"
2021-09-28 14:12:56 +08:00
# include "esp_rom_spiflash.h"
2022-08-23 12:31:53 +08:00
# include "esp_private/esp_clk.h"
# include "esp_spi_flash_counters.h"
2021-02-25 12:25:38 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
# include "esp_crypto_lock.h" // for locking flash encryption peripheral
# endif //CONFIG_IDF_TARGET_ESP32S2
2020-09-03 18:17:24 +08:00
2023-03-02 16:10:02 +08:00
DRAM_ATTR static const char TAG [ ] = " spi_flash " ;
2019-01-08 18:29:25 +08:00
2020-06-04 14:51:34 +05:30
# ifdef CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
# define MAX_WRITE_CHUNK CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE /* write in chunks */
# else
2019-01-08 18:29:25 +08:00
# define MAX_WRITE_CHUNK 8192 /* write in chunks */
2020-06-04 14:51:34 +05:30
# endif // CONFIG_SPI_FLASH_WRITE_CHUNK_SIZE
2019-01-08 18:29:25 +08:00
# define MAX_READ_CHUNK 16384
2023-03-02 16:10:02 +08:00
# define VERIFY_BUF_LEN 64
2019-01-08 18:29:25 +08:00
# ifdef CONFIG_SPI_FLASH_DANGEROUS_WRITE_ABORTS
# define UNSAFE_WRITE_ADDRESS abort()
# else
# define UNSAFE_WRITE_ADDRESS return ESP_ERR_INVALID_ARG
# endif
/* CHECK_WRITE_ADDRESS macro to fail writes which land in the
bootloader , partition table , or running application region .
*/
2023-07-17 11:59:28 +02:00
# define CHECK_WRITE_ADDRESS(CHIP, ADDR, SIZE) do { \
if ( CHIP & & CHIP - > os_func - > region_protected ) { \
esp_err_t ret = CHIP - > os_func - > region_protected ( CHIP - > os_func_data , ADDR , SIZE ) ; \
if ( ret = = ESP_ERR_NOT_ALLOWED ) { \
return ret ; /* ESP_ERR_NOT_ALLOWED from read-only partition check */ \
} else if ( ret ! = ESP_OK ) { \
UNSAFE_WRITE_ADDRESS ; /* FAILS or ABORTS */ \
} \
} \
2019-01-08 18:29:25 +08:00
} while ( 0 )
2022-10-21 18:28:54 +08:00
/* Convenience macro for beginning of all API functions.
* Check the return value of ` rom_spiflash_api_funcs - > chip_check ` is correct ,
* and the chip supports the operation in question .
*/
2023-07-17 11:59:28 +02:00
# define VERIFY_CHIP_OP(op) do { \
2022-10-21 18:28:54 +08:00
if ( err ! = ESP_OK ) return err ; \
2023-07-17 11:59:28 +02:00
if ( chip - > chip_drv - > op = = NULL ) { \
return ESP_ERR_FLASH_UNSUPPORTED_CHIP ; \
} \
2022-10-21 18:28:54 +08:00
} while ( 0 )
2022-08-23 12:31:53 +08:00
# if CONFIG_SPI_FLASH_ENABLE_COUNTERS
static esp_flash_counters_t esp_flash_stats ;
# define COUNTER_START() uint32_t ts_begin = esp_cpu_get_cycle_count()
# define COUNTER_STOP(counter) \
do { \
esp_flash_stats . counter . count + + ; \
esp_flash_stats . counter . time + = ( esp_cpu_get_cycle_count ( ) - ts_begin ) / ( esp_clk_cpu_freq ( ) / 1000000 ) ; \
} while ( 0 )
# define COUNTER_ADD_BYTES(counter, size) \
do { \
esp_flash_stats . counter . bytes + = size ; \
} while ( 0 )
const esp_flash_counters_t * esp_flash_get_counters ( void )
{
return & esp_flash_stats ;
}
void esp_flash_reset_counters ( void )
{
memset ( & esp_flash_stats , 0 , sizeof ( esp_flash_stats ) ) ;
}
void esp_flash_dump_counters ( FILE * stream )
{
if ( stream ! = NULL ) {
fprintf ( stream , " read: count=%8ld time=%8ldus bytes=%8ld \n " , esp_flash_stats . read . count , esp_flash_stats . read . time , esp_flash_stats . read . bytes ) ;
fprintf ( stream , " write: count=%8ld time=%8ldus bytes=%8ld \n " , esp_flash_stats . write . count , esp_flash_stats . write . time , esp_flash_stats . write . bytes ) ;
fprintf ( stream , " erase: count=%8ld time=%8ldus bytes=%8ld \n " , esp_flash_stats . erase . count , esp_flash_stats . erase . time , esp_flash_stats . erase . bytes ) ;
}
}
const spi_flash_counters_t * spi_flash_get_counters ( void )
{
return ( spi_flash_counters_t * ) esp_flash_get_counters ( ) ;
}
void spi_flash_reset_counters ( void )
{
esp_flash_reset_counters ( ) ;
}
void spi_flash_dump_counters ( void )
{
esp_flash_dump_counters ( stdout ) ;
}
# else
# define COUNTER_START()
# define COUNTER_STOP(counter)
# define COUNTER_ADD_BYTES(counter, size)
# endif //CONFIG_SPI_FLASH_ENABLE_COUNTERS
2021-09-01 15:58:15 +08:00
# define IO_STR_LEN 10
2019-01-08 18:29:25 +08:00
static const char io_mode_str [ ] [ IO_STR_LEN ] = {
" slowrd " ,
" fastrd " ,
" dout " ,
" dio " ,
" qout " ,
" qio " ,
2021-09-01 15:58:15 +08:00
[ 6 . . . 15 ] = " not used " , // reserved io mode for future, not used currently.
" opi_str " ,
" opi_dtr " ,
2019-01-08 18:29:25 +08:00
} ;
2021-09-01 15:58:15 +08:00
_Static_assert ( sizeof ( io_mode_str ) / IO_STR_LEN = = SPI_FLASH_READ_MODE_MAX , " the io_mode_str should be consistent with the esp_flash_io_mode_t defined in spi_flash_types.h " ) ;
2019-01-08 18:29:25 +08:00
2019-09-10 00:56:46 +08:00
esp_err_t esp_flash_read_chip_id ( esp_flash_t * chip , uint32_t * flash_id ) ;
2019-01-08 18:29:25 +08:00
2024-02-22 18:05:47 +08:00
# if !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2020-04-30 10:37:35 +08:00
static esp_err_t spiflash_start_default ( esp_flash_t * chip ) ;
static esp_err_t spiflash_end_default ( esp_flash_t * chip , esp_err_t err ) ;
static esp_err_t check_chip_pointer_default ( esp_flash_t * * inout_chip ) ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
static esp_err_t flash_end_flush_cache ( esp_flash_t * chip , esp_err_t err , bool bus_acquired , uint32_t address , uint32_t length ) ;
2024-02-22 18:05:47 +08:00
# endif // !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2020-04-30 10:37:35 +08:00
typedef struct {
esp_err_t ( * start ) ( esp_flash_t * chip ) ;
esp_err_t ( * end ) ( esp_flash_t * chip , esp_err_t err ) ;
esp_err_t ( * chip_check ) ( esp_flash_t * * inout_chip ) ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
esp_err_t ( * flash_end_flush_cache ) ( esp_flash_t * chip , esp_err_t err , bool bus_acquired , uint32_t address , uint32_t length ) ;
2020-04-30 10:37:35 +08:00
} rom_spiflash_api_func_t ;
2024-02-22 18:05:47 +08:00
# if !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2020-04-30 10:37:35 +08:00
// These functions can be placed in the ROM. For now we use the code in IDF.
DRAM_ATTR static rom_spiflash_api_func_t default_spiflash_rom_api = {
. start = spiflash_start_default ,
. end = spiflash_end_default ,
. chip_check = check_chip_pointer_default ,
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
. flash_end_flush_cache = flash_end_flush_cache ,
2020-04-30 10:37:35 +08:00
} ;
DRAM_ATTR rom_spiflash_api_func_t * rom_spiflash_api_funcs = & default_spiflash_rom_api ;
2020-12-16 14:50:13 +11:00
# else
extern rom_spiflash_api_func_t * esp_flash_api_funcs ;
# define rom_spiflash_api_funcs esp_flash_api_funcs
2024-02-22 18:05:47 +08:00
# endif // !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2020-04-30 10:37:35 +08:00
2019-01-08 18:29:25 +08:00
/* Static function to notify OS of a new SPI flash operation.
If returns an error result , caller must abort . If returns ESP_OK , caller must
2020-04-30 10:37:35 +08:00
call rom_spiflash_api_funcs - > end ( ) before returning .
2019-01-08 18:29:25 +08:00
*/
2024-02-22 18:05:47 +08:00
# if !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2020-04-30 10:37:35 +08:00
static esp_err_t IRAM_ATTR spiflash_start_default ( esp_flash_t * chip )
2019-01-08 18:29:25 +08:00
{
if ( chip - > os_func ! = NULL & & chip - > os_func - > start ! = NULL ) {
esp_err_t err = chip - > os_func - > start ( chip - > os_func_data ) ;
if ( err ! = ESP_OK ) {
return err ;
}
}
2020-05-07 14:46:41 +08:00
chip - > host - > driver - > dev_config ( chip - > host ) ;
2019-01-08 18:29:25 +08:00
return ESP_OK ;
}
/* Static function to notify OS that SPI flash operation is complete.
*/
2020-04-30 10:37:35 +08:00
static esp_err_t IRAM_ATTR spiflash_end_default ( esp_flash_t * chip , esp_err_t err )
2019-01-08 18:29:25 +08:00
{
if ( chip - > os_func ! = NULL
& & chip - > os_func - > end ! = NULL ) {
esp_err_t end_err = chip - > os_func - > end ( chip - > os_func_data ) ;
if ( err = = ESP_OK ) {
err = end_err ; // Only return the 'end' error if we haven't already failed
}
}
return err ;
}
2020-04-30 10:37:35 +08:00
// check that the 'chip' parameter is properly initialised
2023-03-20 12:44:31 +08:00
static IRAM_ATTR esp_err_t check_chip_pointer_default ( esp_flash_t * * inout_chip )
2020-04-30 10:37:35 +08:00
{
esp_flash_t * chip = * inout_chip ;
if ( chip = = NULL ) {
chip = esp_flash_default_chip ;
}
* inout_chip = chip ;
if ( chip = = NULL | | ! esp_flash_chip_driver_initialized ( chip ) ) {
return ESP_ERR_FLASH_NOT_INITIALISED ;
}
return ESP_OK ;
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
static IRAM_ATTR esp_err_t flash_end_flush_cache ( esp_flash_t * chip , esp_err_t err , bool bus_acquired , uint32_t address , uint32_t length )
{
if ( ! bus_acquired ) {
// Try to acquire the bus again to flush the cache before exit.
esp_err_t acquire_err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( acquire_err ! = ESP_OK ) {
return ( err = = ESP_OK ) ? acquire_err : err ;
}
}
if ( chip - > host - > driver - > flush_cache ) {
esp_err_t flush_err = chip - > host - > driver - > flush_cache ( chip - > host , address , length ) ;
if ( err = = ESP_OK ) {
err = flush_err ;
}
}
return rom_spiflash_api_funcs - > end ( chip , err ) ;
}
2024-02-22 18:05:47 +08:00
# endif // !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2019-01-08 18:29:25 +08:00
/* Top-level API functions, calling into chip_drv functions via chip->drv */
static esp_err_t detect_spi_flash_chip ( esp_flash_t * chip ) ;
2023-03-20 12:44:31 +08:00
bool IRAM_ATTR esp_flash_chip_driver_initialized ( const esp_flash_t * chip )
2019-01-08 18:29:25 +08:00
{
if ( ! chip - > chip_drv ) return false ;
return true ;
}
esp_err_t IRAM_ATTR esp_flash_init ( esp_flash_t * chip )
{
2020-07-27 03:13:07 +08:00
// Chip init flow
// 1. Read chip id
// 2. (optional) Detect chip vendor
// 3. Get basic parameters of the chip (size, dummy count, etc.)
// 4. Init chip into desired mode (without breaking the cache!)
2019-01-08 18:29:25 +08:00
esp_err_t err = ESP_OK ;
2020-05-07 14:46:41 +08:00
if ( chip = = NULL | | chip - > host = = NULL | | chip - > host - > driver = = NULL | |
( ( memspi_host_inst_t * ) chip - > host ) - > spi = = NULL ) {
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
2019-09-10 00:56:46 +08:00
//read chip id
uint32_t flash_id ;
int retries = 10 ;
do {
err = esp_flash_read_chip_id ( chip , & flash_id ) ;
} while ( err = = ESP_ERR_FLASH_NOT_INITIALISED & & retries - - > 0 ) ;
if ( err ! = ESP_OK ) {
return err ;
}
chip - > chip_id = flash_id ;
2019-01-08 18:29:25 +08:00
if ( ! esp_flash_chip_driver_initialized ( chip ) ) {
// Detect chip_drv
err = detect_spi_flash_chip ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
}
// Detect flash size
uint32_t size ;
2022-10-21 18:28:54 +08:00
err = esp_flash_get_physical_size ( chip , & size ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
ESP_LOGE ( TAG , " failed to get chip size " ) ;
return err ;
}
2021-05-18 12:05:41 +08:00
if ( chip - > chip_drv - > get_chip_caps = = NULL ) {
// chip caps get failed, pass the flash capability check.
ESP_EARLY_LOGW ( TAG , " get_chip_caps function pointer hasn't been initialized " ) ;
} else {
if ( ( ( chip - > chip_drv - > get_chip_caps ( chip ) & SPI_FLASH_CHIP_CAP_32MB_SUPPORT ) = = 0 ) & & ( size > ( 16 * 1024 * 1024 ) ) ) {
ESP_EARLY_LOGW ( TAG , " Detected flash size > 16 MB, but access beyond 16 MB is not supported for this flash model yet. " ) ;
size = ( 16 * 1024 * 1024 ) ;
}
}
2019-01-08 18:29:25 +08:00
ESP_LOGI ( TAG , " flash io: %s " , io_mode_str [ chip - > read_mode ] ) ;
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
if ( err = = ESP_OK ) {
// Try to set the flash mode to whatever default mode was chosen
2019-09-05 13:11:36 +08:00
err = chip - > chip_drv - > set_io_mode ( chip ) ;
if ( err = = ESP_ERR_FLASH_NO_RESPONSE & & ! esp_flash_is_quad_mode ( chip ) ) {
//some chips (e.g. Winbond) don't support to clear QE, treat as success
err = ESP_OK ;
}
2019-01-08 18:29:25 +08:00
}
// Done: all fields on 'chip' are initialised
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
2021-09-01 15:58:15 +08:00
// Note: This function is only used for internal. Only call this function to initialize the main flash.
// (flash chip on SPI1 CS0)
esp_err_t IRAM_ATTR esp_flash_init_main ( esp_flash_t * chip )
{
// Chip init flow
// 1. Read chip id
// 2. (optional) Detect chip vendor
// 3. Get basic parameters of the chip (size, dummy count, etc.)
// 4. Init chip into desired mode (without breaking the cache!)
esp_err_t err = ESP_OK ;
2021-12-07 13:21:08 +05:30
bool octal_mode ;
2021-09-01 15:58:15 +08:00
if ( chip = = NULL | | chip - > host = = NULL | | chip - > host - > driver = = NULL | |
( ( memspi_host_inst_t * ) chip - > host ) - > spi = = NULL ) {
return ESP_ERR_INVALID_ARG ;
}
2021-12-07 13:21:08 +05:30
octal_mode = ( chip - > read_mode > = SPI_FLASH_OPI_FLAG ) ;
2021-09-01 15:58:15 +08:00
//read chip id
// This can indicate the MSPI support OPI, if the flash works on MSPI in OPI mode, we directly bypass read id.
uint32_t flash_id = 0 ;
if ( octal_mode ) {
// bypass the reading but get the flash_id from the ROM variable, to avoid resetting the chip to QSPI mode and read the ID again
flash_id = g_rom_flashchip . device_id ;
} else {
int retries = 10 ;
do {
err = esp_flash_read_chip_id ( chip , & flash_id ) ;
} while ( err = = ESP_ERR_FLASH_NOT_INITIALISED & & retries - - > 0 ) ;
}
if ( err ! = ESP_OK ) {
return err ;
}
chip - > chip_id = flash_id ;
if ( ! esp_flash_chip_driver_initialized ( chip ) ) {
// Detect chip_drv
err = detect_spi_flash_chip ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
}
// Detect flash size
uint32_t size ;
2022-10-21 18:28:54 +08:00
err = esp_flash_get_physical_size ( chip , & size ) ;
2021-09-01 15:58:15 +08:00
if ( err ! = ESP_OK ) {
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGE ( TAG , " failed to get chip size " ) ;
2021-09-01 15:58:15 +08:00
return err ;
}
if ( chip - > chip_drv - > get_chip_caps = = NULL ) {
// chip caps get failed, pass the flash capability check.
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGW ( TAG , " get_chip_caps function pointer hasn't been initialized " ) ;
2021-09-01 15:58:15 +08:00
} else {
if ( ( ( chip - > chip_drv - > get_chip_caps ( chip ) & SPI_FLASH_CHIP_CAP_32MB_SUPPORT ) = = 0 ) & & ( size > ( 16 * 1024 * 1024 ) ) ) {
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGW ( TAG , " Detected flash size > 16 MB, but access beyond 16 MB is not supported for this flash model yet. " ) ;
2021-09-01 15:58:15 +08:00
size = ( 16 * 1024 * 1024 ) ;
}
}
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGI ( TAG , " flash io: %s " , io_mode_str [ chip - > read_mode ] ) ;
2021-09-01 15:58:15 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
if ( err = = ESP_OK & & ! octal_mode ) {
// Try to set the flash mode to whatever default mode was chosen
err = chip - > chip_drv - > set_io_mode ( chip ) ;
if ( err = = ESP_ERR_FLASH_NO_RESPONSE & & ! esp_flash_is_quad_mode ( chip ) ) {
//some chips (e.g. Winbond) don't support to clear QE, treat as success
err = ESP_OK ;
}
}
// Done: all fields on 'chip' are initialised
return rom_spiflash_api_funcs - > end ( chip , err ) ;
}
2020-07-27 03:13:07 +08:00
static esp_err_t IRAM_ATTR read_id_core ( esp_flash_t * chip , uint32_t * out_id , bool sanity_check )
2019-09-05 13:11:36 +08:00
{
2020-07-27 03:13:07 +08:00
bool installed = esp_flash_chip_driver_initialized ( chip ) ;
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-09-05 13:11:36 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
2020-07-27 03:13:07 +08:00
esp_err_t ( * read_id_func ) ( void * , uint32_t * ) ;
void * read_id_arg ;
if ( installed & & chip - > chip_drv - > read_id ) {
read_id_func = ( void * ) chip - > chip_drv - > read_id ;
read_id_arg = ( void * ) chip ;
} else {
//default option if the chip is not detected/chosen yet.
read_id_func = ( void * ) chip - > host - > driver - > read_id ;
read_id_arg = ( void * ) chip - > host ;
}
// Inner function fails if it sees all-ones or all-zeroes.
err = read_id_func ( read_id_arg , out_id ) ;
2019-09-05 13:11:36 +08:00
2020-07-27 03:13:07 +08:00
if ( sanity_check & & err = = ESP_OK ) {
// Send RDID command twice, check for a matching result and retry in case we just powered on
2019-09-05 13:11:36 +08:00
uint32_t new_id ;
2020-07-27 03:13:07 +08:00
err = read_id_func ( read_id_arg , & new_id ) ;
if ( err = = ESP_OK & & ( new_id ! = * out_id ) ) {
2019-09-05 13:11:36 +08:00
err = ESP_ERR_FLASH_NOT_INITIALISED ;
}
}
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-09-05 13:11:36 +08:00
}
2020-07-27 03:13:07 +08:00
// Faster version with sanity check.
// Called in esp_flash_init and unit test (though not public)
esp_err_t esp_flash_read_chip_id ( esp_flash_t * chip , uint32_t * out_id )
{
return read_id_core ( chip , out_id , true ) ;
}
2020-12-16 14:50:13 +11:00
# ifndef CONFIG_SPI_FLASH_ROM_IMPL
2020-07-27 03:13:07 +08:00
esp_err_t esp_flash_read_id ( esp_flash_t * chip , uint32_t * out_id )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
//Accept uninitialized chip when reading chip id
if ( err ! = ESP_OK & & ! ( err = = ESP_ERR_FLASH_NOT_INITIALISED & & chip ! = NULL ) ) return err ;
if ( out_id = = NULL ) return ESP_ERR_INVALID_ARG ;
return read_id_core ( chip , out_id , false ) ;
}
2020-12-16 14:50:13 +11:00
# endif //CONFIG_SPI_FLASH_ROM_IMPL
2020-07-27 03:13:07 +08:00
2020-11-27 19:09:40 +08:00
static esp_err_t IRAM_ATTR NOINLINE_ATTR read_unique_id ( esp_flash_t * chip , uint64_t * out_uid )
{
esp_err_t err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
err = chip - > chip_drv - > read_unique_id ( chip , out_uid ) ;
return rom_spiflash_api_funcs - > end ( chip , err ) ;
}
esp_err_t esp_flash_read_unique_chip_id ( esp_flash_t * chip , uint64_t * out_uid )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
2021-05-18 12:05:41 +08:00
if ( chip - > chip_drv - > get_chip_caps = = NULL ) {
// chip caps get failed, pass the flash capability check.
ESP_EARLY_LOGW ( TAG , " get_chip_caps function pointer hasn't been initialized " ) ;
} else {
if ( ( chip - > chip_drv - > get_chip_caps ( chip ) & SPI_FLASH_CHIP_CAP_UNIQUE_ID ) = = 0 ) {
ESP_EARLY_LOGE ( TAG , " chip %s doesn't support reading unique id " , chip - > chip_drv - > name ) ;
return ESP_ERR_NOT_SUPPORTED ;
}
}
2020-11-27 19:09:40 +08:00
if ( out_uid = = NULL ) {
return ESP_ERR_INVALID_ARG ;
} ;
return read_unique_id ( chip , out_uid ) ;
}
2019-01-08 18:29:25 +08:00
static esp_err_t IRAM_ATTR detect_spi_flash_chip ( esp_flash_t * chip )
{
esp_err_t err ;
2019-09-10 00:56:46 +08:00
uint32_t flash_id = chip - > chip_id ;
2019-01-08 18:29:25 +08:00
// Detect the chip and set the chip_drv structure for it
const spi_flash_chip_t * * drivers = esp_flash_registered_chips ;
while ( * drivers ! = NULL & & ! esp_flash_chip_driver_initialized ( chip ) ) {
chip - > chip_drv = * drivers ;
// start/end SPI operation each time, for multitasking
// and also so esp_flash_registered_flash_drivers can live in flash
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGD ( TAG , " trying chip: %s " , chip - > chip_drv - > name ) ;
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
if ( chip - > chip_drv - > probe ( chip , flash_id ) ! = ESP_OK ) {
chip - > chip_drv = NULL ;
}
// if probe succeeded, chip->drv stays set
drivers + + ;
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
}
if ( ! esp_flash_chip_driver_initialized ( chip ) ) {
return ESP_ERR_NOT_FOUND ;
}
2022-06-08 19:19:44 +08:00
ESP_EARLY_LOGI ( TAG , " detected chip: %s " , chip - > chip_drv - > name ) ;
2019-01-08 18:29:25 +08:00
return ESP_OK ;
}
2022-10-21 18:28:54 +08:00
esp_err_t IRAM_ATTR esp_flash_get_physical_size ( esp_flash_t * chip , uint32_t * flash_size )
2019-01-08 18:29:25 +08:00
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
2022-10-21 18:28:54 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
2020-04-30 10:37:35 +08:00
VERIFY_CHIP_OP ( detect_size ) ;
2022-10-21 18:28:54 +08:00
if ( flash_size = = NULL ) {
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
uint32_t detect_size ;
err = chip - > chip_drv - > detect_size ( chip , & detect_size ) ;
if ( err = = ESP_OK ) {
2022-10-21 18:28:54 +08:00
if ( chip - > size = = 0 ) {
// chip->size will not be changed if detected, it will always be equal to configured flash size.
chip - > size = detect_size ;
}
* flash_size = detect_size ;
2019-01-08 18:29:25 +08:00
}
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
2022-10-21 18:28:54 +08:00
# ifndef CONFIG_SPI_FLASH_ROM_IMPL
/* Return true if regions 'a' and 'b' overlap at all, based on their start offsets and lengths. */
inline static bool regions_overlap ( uint32_t a_start , uint32_t a_len , uint32_t b_start , uint32_t b_len ) ;
esp_err_t IRAM_ATTR esp_flash_get_size ( esp_flash_t * chip , uint32_t * out_size )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
if ( out_size = = NULL ) {
return ESP_ERR_INVALID_ARG ;
}
if ( chip - > size ! = 0 ) {
* out_size = chip - > size ;
return ESP_OK ;
}
//Return flash chip physical size, when this API is called before flash initialisation,
//After initialization will return available size.
return esp_flash_get_physical_size ( chip , out_size ) ;
}
2019-01-08 18:29:25 +08:00
esp_err_t IRAM_ATTR esp_flash_erase_chip ( esp_flash_t * chip )
{
2022-10-13 14:48:47 +02:00
esp_err_t err = ESP_OK ;
uint32_t size = 0 ;
err = esp_flash_get_size ( chip , & size ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
2022-10-13 14:48:47 +02:00
ESP_LOGE ( TAG , " esp_flash_get_size failed, flash error code: %d " , err ) ;
2019-01-08 18:29:25 +08:00
return err ;
}
2022-10-13 14:48:47 +02:00
err = esp_flash_erase_region ( chip , 0 , size ) ;
return err ;
2019-01-08 18:29:25 +08:00
}
esp_err_t IRAM_ATTR esp_flash_erase_region ( esp_flash_t * chip , uint32_t start , uint32_t len )
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( erase_sector ) ;
VERIFY_CHIP_OP ( erase_block ) ;
2019-01-08 18:29:25 +08:00
CHECK_WRITE_ADDRESS ( chip , start , len ) ;
2020-04-30 10:37:35 +08:00
2019-01-08 18:29:25 +08:00
uint32_t block_erase_size = chip - > chip_drv - > erase_block = = NULL ? 0 : chip - > chip_drv - > block_erase_size ;
uint32_t sector_size = chip - > chip_drv - > sector_size ;
2022-08-23 12:31:53 +08:00
COUNTER_START ( ) ;
2019-01-08 18:29:25 +08:00
if ( sector_size = = 0 | | ( block_erase_size % sector_size ) ! = 0 ) {
return ESP_ERR_FLASH_NOT_INITIALISED ;
}
if ( start > chip - > size | | start + len > chip - > size ) {
return ESP_ERR_INVALID_ARG ;
}
if ( ( start % chip - > chip_drv - > sector_size ) ! = 0 | | ( len % chip - > chip_drv - > sector_size ) ! = 0 ) {
// Can only erase multiples of the sector size, starting at sector boundary
return ESP_ERR_INVALID_ARG ;
}
2021-07-23 02:35:27 -07:00
if ( len = = 0 ) {
return ESP_OK ;
}
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
err = ESP_OK ;
2019-01-08 18:29:25 +08:00
// Check for write protected regions overlapping the erase region
2019-08-02 13:04:48 +08:00
if ( chip - > chip_drv - > get_protected_regions ! = NULL & &
chip - > chip_drv - > num_protectable_regions > 0 ) {
2019-09-10 14:34:06 +08:00
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-09-10 14:34:06 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
2019-01-08 18:29:25 +08:00
uint64_t protected = 0 ;
err = chip - > chip_drv - > get_protected_regions ( chip , & protected ) ;
if ( err = = ESP_OK & & protected ! = 0 ) {
for ( int i = 0 ; i < chip - > chip_drv - > num_protectable_regions & & err = = ESP_OK ; i + + ) {
const esp_flash_region_t * region = & chip - > chip_drv - > protectable_regions [ i ] ;
if ( ( protected & BIT64 ( i ) )
& & regions_overlap ( start , len , region - > offset , region - > size ) ) {
err = ESP_ERR_FLASH_PROTECTED ;
}
}
}
2019-09-10 14:34:06 +08:00
// Don't lock the SPI flash for the entire erase, as this may be very long
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
uint32_t erase_addr = start ;
uint32_t len_remain = len ;
// Indicate whether the bus is acquired by the driver, needs to be released before return
bool bus_acquired = false ;
while ( 1 ) {
//check before the operation, in case this is called too close to the last operation
2020-12-16 14:50:13 +11:00
if ( chip - > chip_drv - > yield ) {
err = chip - > chip_drv - > yield ( chip , 0 ) ;
if ( err ! = ESP_OK ) {
return err ;
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
}
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
break ;
2019-01-08 18:29:25 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
bus_acquired = true ;
2019-01-08 18:29:25 +08:00
2020-04-24 02:13:17 +08:00
# ifndef CONFIG_SPI_FLASH_BYPASS_BLOCK_ERASE
2019-01-08 18:29:25 +08:00
// If possible erase an entire multi-sector block
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
if ( block_erase_size > 0 & & len_remain > = block_erase_size & & ( erase_addr % block_erase_size ) = = 0 ) {
err = chip - > chip_drv - > erase_block ( chip , erase_addr ) ;
erase_addr + = block_erase_size ;
len_remain - = block_erase_size ;
2022-08-23 12:31:53 +08:00
COUNTER_ADD_BYTES ( erase , block_erase_size ) ;
2020-04-24 02:13:17 +08:00
} else
# endif
{
2019-01-08 18:29:25 +08:00
// Otherwise erase individual sector only
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
err = chip - > chip_drv - > erase_sector ( chip , erase_addr ) ;
erase_addr + = sector_size ;
len_remain - = sector_size ;
2022-08-23 12:31:53 +08:00
COUNTER_ADD_BYTES ( erase , sector_size ) ;
2019-01-08 18:29:25 +08:00
}
2021-07-23 02:35:27 -07:00
assert ( len_remain < len ) ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
if ( err ! = ESP_OK | | len_remain = = 0 ) {
// On ESP32, the cache re-enable is in the end() function, while flush_cache should
// happen when the cache is still disabled on ESP32. Break before the end() function and
// do end() later
assert ( bus_acquired ) ;
break ;
}
2020-04-02 17:29:21 +08:00
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
err = rom_spiflash_api_funcs - > end ( chip , ESP_OK ) ;
if ( err ! = ESP_OK ) {
break ;
2020-04-02 17:29:21 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
bus_acquired = false ;
2019-01-08 18:29:25 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( erase ) ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
return rom_spiflash_api_funcs - > flash_end_flush_cache ( chip , err , bus_acquired , start , len ) ;
2019-01-08 18:29:25 +08:00
}
2021-07-24 13:28:25 +02:00
# endif // !CONFIG_SPI_FLASH_ROM_IMPL
# if defined(CONFIG_SPI_FLASH_ROM_IMPL) && ESP_ROM_HAS_ERASE_0_REGION_BUG
/* ROM esp_flash_erase_region implementation doesn't handle 0 erase size correctly.
* Check the size and call ROM function instead of overriding it completely .
* The behavior is slightly different from esp_flash_erase_region above , thought :
* here the check for 0 size is done first , but in esp_flash_erase_region the check is
* done after the other arguments are checked .
*/
extern esp_err_t rom_esp_flash_erase_region ( esp_flash_t * chip , uint32_t start , uint32_t len ) ;
esp_err_t IRAM_ATTR esp_flash_erase_region ( esp_flash_t * chip , uint32_t start , uint32_t len )
{
if ( len = = 0 ) {
return ESP_OK ;
}
return rom_esp_flash_erase_region ( chip , start , len ) ;
}
# endif // defined(CONFIG_SPI_FLASH_ROM_IMPL) && ESP_ROM_HAS_ERASE_0_REGION_BUG
# ifndef CONFIG_SPI_FLASH_ROM_IMPL
2019-08-02 13:04:48 +08:00
esp_err_t IRAM_ATTR esp_flash_get_chip_write_protect ( esp_flash_t * chip , bool * out_write_protected )
2019-01-08 18:29:25 +08:00
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( get_chip_write_protect ) ;
2019-08-02 13:04:48 +08:00
if ( out_write_protected = = NULL ) {
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
2019-08-02 13:04:48 +08:00
err = chip - > chip_drv - > get_chip_write_protect ( chip , out_write_protected ) ;
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
2019-06-19 20:35:55 +08:00
esp_err_t IRAM_ATTR esp_flash_set_chip_write_protect ( esp_flash_t * chip , bool write_protect )
2019-01-08 18:29:25 +08:00
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( set_chip_write_protect ) ;
2019-01-08 18:29:25 +08:00
//TODO: skip writing if already locked or unlocked
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
2019-06-19 20:35:55 +08:00
err = chip - > chip_drv - > set_chip_write_protect ( chip , write_protect ) ;
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
2019-06-19 20:35:55 +08:00
esp_err_t esp_flash_get_protectable_regions ( const esp_flash_t * chip , const esp_flash_region_t * * out_regions , uint32_t * out_num_regions )
2019-01-08 18:29:25 +08:00
{
2019-06-19 20:35:55 +08:00
if ( out_num_regions ! = NULL ) {
* out_num_regions = 0 ; // In case caller doesn't check result
2019-01-08 18:29:25 +08:00
}
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( ( esp_flash_t * * ) & chip ) ;
VERIFY_CHIP_OP ( get_protected_regions ) ;
2019-01-08 18:29:25 +08:00
2019-06-19 20:35:55 +08:00
if ( out_regions = = NULL | | out_num_regions = = NULL ) {
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
2019-06-19 20:35:55 +08:00
* out_num_regions = chip - > chip_drv - > num_protectable_regions ;
* out_regions = chip - > chip_drv - > protectable_regions ;
2019-01-08 18:29:25 +08:00
return ESP_OK ;
}
static esp_err_t find_region ( const esp_flash_t * chip , const esp_flash_region_t * region , uint8_t * index )
{
if ( region = = NULL ) {
return ESP_ERR_INVALID_ARG ;
}
for ( * index = 0 ; * index < chip - > chip_drv - > num_protectable_regions ; ( * index ) + + ) {
if ( memcmp ( & chip - > chip_drv - > protectable_regions [ * index ] ,
region , sizeof ( esp_flash_region_t ) ) = = 0 ) {
return ESP_OK ;
}
}
return ESP_ERR_NOT_FOUND ;
}
2019-06-19 16:37:11 +08:00
esp_err_t IRAM_ATTR esp_flash_get_protected_region ( esp_flash_t * chip , const esp_flash_region_t * region , bool * out_protected )
2019-01-08 18:29:25 +08:00
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( get_protected_regions ) ;
2019-01-08 18:29:25 +08:00
2019-06-19 16:37:11 +08:00
if ( out_protected = = NULL ) {
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
uint8_t index ;
2020-04-30 10:37:35 +08:00
err = find_region ( chip , region , & index ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
uint64_t protection_mask = 0 ;
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
err = chip - > chip_drv - > get_protected_regions ( chip , & protection_mask ) ;
if ( err = = ESP_OK ) {
2019-06-19 16:37:11 +08:00
* out_protected = protection_mask & ( 1LL < < index ) ;
2019-01-08 18:29:25 +08:00
}
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
2019-06-19 16:37:11 +08:00
esp_err_t IRAM_ATTR esp_flash_set_protected_region ( esp_flash_t * chip , const esp_flash_region_t * region , bool protect )
2019-01-08 18:29:25 +08:00
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( set_protected_regions ) ;
2019-01-08 18:29:25 +08:00
uint8_t index ;
2020-04-30 10:37:35 +08:00
err = find_region ( chip , region , & index ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
uint64_t protection_mask = 0 ;
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
return err ;
}
err = chip - > chip_drv - > get_protected_regions ( chip , & protection_mask ) ;
if ( err = = ESP_OK ) {
2019-06-19 16:37:11 +08:00
if ( protect ) {
2019-01-08 18:29:25 +08:00
protection_mask | = ( 1LL < < index ) ;
} else {
protection_mask & = ~ ( 1LL < < index ) ;
}
err = chip - > chip_drv - > set_protected_regions ( chip , protection_mask ) ;
}
2020-04-30 10:37:35 +08:00
return rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
}
esp_err_t IRAM_ATTR esp_flash_read ( esp_flash_t * chip , void * buffer , uint32_t address , uint32_t length )
{
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( read ) ;
2019-01-08 18:29:25 +08:00
if ( buffer = = NULL | | address > chip - > size | | address + length > chip - > size ) {
return ESP_ERR_INVALID_ARG ;
}
2021-07-23 02:35:27 -07:00
if ( length = = 0 ) {
return ESP_OK ;
}
2019-01-08 18:29:25 +08:00
//when the cache is disabled, only the DRAM can be read, check whether we need to receive in another buffer in DRAM.
2022-04-01 16:53:40 +08:00
bool direct_read = false ;
//If the buffer is internal already, it's ok to use it directly
direct_read | = esp_ptr_in_dram ( buffer ) ;
//If not, we need to check if the HW support direct write
direct_read | = chip - > host - > driver - > supports_direct_read ( chip - > host , buffer ) ;
2019-01-08 18:29:25 +08:00
uint8_t * temp_buffer = NULL ;
2020-02-27 11:57:00 +11:00
//each time, we at most read this length
//after that, we release the lock to allow some other operations
size_t read_chunk_size = MIN ( MAX_READ_CHUNK , length ) ;
2019-01-08 18:29:25 +08:00
if ( ! direct_read ) {
2020-04-30 10:37:35 +08:00
size_t actual_len = 0 ;
2020-07-29 11:39:56 +02:00
if ( chip - > os_func - > get_temp_buffer ! = NULL ) {
temp_buffer = chip - > os_func - > get_temp_buffer ( chip - > os_func_data , read_chunk_size , & actual_len ) ;
read_chunk_size = actual_len ;
}
2020-02-25 14:56:13 +11:00
if ( temp_buffer = = NULL ) {
return ESP_ERR_NO_MEM ;
}
2019-01-08 18:29:25 +08:00
}
2022-08-23 12:31:53 +08:00
COUNTER_START ( ) ;
2020-04-30 10:37:35 +08:00
err = ESP_OK ;
2019-01-08 18:29:25 +08:00
do {
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
break ;
}
//if required (dma buffer allocated), read to the buffer instead of the original buffer
uint8_t * buffer_to_read = ( temp_buffer ) ? temp_buffer : buffer ;
2020-02-27 11:57:00 +11:00
// Length we will read this iteration is either the chunk size or the remaining length, whichever is smaller
size_t length_to_read = MIN ( read_chunk_size , length ) ;
2019-01-08 18:29:25 +08:00
if ( err = = ESP_OK ) {
err = chip - > chip_drv - > read ( chip , buffer_to_read , address , length_to_read ) ;
}
if ( err ! = ESP_OK ) {
2020-04-30 10:37:35 +08:00
rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
break ;
}
//even if this is failed, the data is still valid, copy before quit
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > end ( chip , err ) ;
2019-01-08 18:29:25 +08:00
//copy back to the original buffer
if ( temp_buffer ) {
memcpy ( buffer , temp_buffer , length_to_read ) ;
}
address + = length_to_read ;
length - = length_to_read ;
2020-04-30 10:37:35 +08:00
buffer = ( void * ) ( ( intptr_t ) buffer + length_to_read ) ;
2022-08-23 12:31:53 +08:00
COUNTER_ADD_BYTES ( read , length_to_read ) ;
2019-01-08 18:29:25 +08:00
} while ( err = = ESP_OK & & length > 0 ) ;
2020-07-29 11:39:56 +02:00
if ( chip - > os_func - > release_temp_buffer ! = NULL ) {
chip - > os_func - > release_temp_buffer ( chip - > os_func_data , temp_buffer ) ;
}
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( read ) ;
2019-01-08 18:29:25 +08:00
return err ;
}
2023-03-02 16:10:02 +08:00
# 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
2019-01-08 18:29:25 +08:00
esp_err_t IRAM_ATTR esp_flash_write ( esp_flash_t * chip , const void * buffer , uint32_t address , uint32_t length )
{
2023-03-02 16:10:02 +08:00
esp_err_t ret = ESP_FAIL ;
2022-09-13 10:26:59 +08:00
# if CONFIG_SPI_FLASH_VERIFY_WRITE
2023-03-02 16:10:02 +08:00
//used for verify write
bool is_encrypted = false ;
2022-09-13 10:26:59 +08:00
# endif //CONFIG_SPI_FLASH_VERIFY_WRITE
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( write ) ;
2019-01-08 18:29:25 +08:00
CHECK_WRITE_ADDRESS ( chip , address , length ) ;
if ( buffer = = NULL | | address > chip - > size | | address + length > chip - > size ) {
return ESP_ERR_INVALID_ARG ;
}
2021-07-23 02:35:27 -07:00
if ( length = = 0 ) {
return ESP_OK ;
}
2019-01-08 18:29:25 +08:00
//when the cache is disabled, only the DRAM can be read, check whether we need to copy the data first
2022-04-01 16:53:40 +08:00
bool direct_write = false ;
//If the buffer is internal already, it's ok to write it directly
direct_write | = esp_ptr_in_dram ( buffer ) ;
//If not, we need to check if the HW support direct write
direct_write | = chip - > host - > driver - > supports_direct_write ( chip - > host , buffer ) ;
2019-01-08 18:29:25 +08:00
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
// Indicate whether the bus is acquired by the driver, needs to be released before return
bool bus_acquired = false ;
2020-04-30 10:37:35 +08:00
err = ESP_OK ;
2022-08-23 12:31:53 +08:00
COUNTER_START ( ) ;
2019-01-08 18:29:25 +08:00
/* Write output in chunks, either by buffering on stack or
by artificially cutting into MAX_WRITE_CHUNK parts ( in an OS
environment , this prevents writing from causing interrupt or higher priority task
starvation . ) */
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
uint32_t write_addr = address ;
uint32_t len_remain = length ;
while ( 1 ) {
2019-01-08 18:29:25 +08:00
uint32_t write_len ;
const void * write_buf ;
2020-08-05 13:37:42 +08:00
uint32_t temp_buf [ 8 ] ;
2019-01-08 18:29:25 +08:00
if ( direct_write ) {
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
write_len = MIN ( len_remain , MAX_WRITE_CHUNK ) ;
2019-01-08 18:29:25 +08:00
write_buf = buffer ;
} else {
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
write_len = MIN ( len_remain , sizeof ( temp_buf ) ) ;
2020-08-05 13:37:42 +08:00
memcpy ( temp_buf , buffer , write_len ) ;
write_buf = temp_buf ;
2019-01-08 18:29:25 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
//check before the operation, in case this is called too close to the last operation
2020-12-16 14:50:13 +11:00
if ( chip - > chip_drv - > yield ) {
err = chip - > chip_drv - > yield ( chip , 0 ) ;
if ( err ! = ESP_OK ) {
return err ;
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
}
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > start ( chip ) ;
2019-01-08 18:29:25 +08:00
if ( err ! = ESP_OK ) {
2023-03-02 16:10:02 +08:00
goto restore_cache ;
2019-01-08 18:29:25 +08:00
}
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
bus_acquired = true ;
2019-01-08 18:29:25 +08:00
2023-03-02 16:10:02 +08:00
# 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
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
err = chip - > chip_drv - > write ( chip , write_buf , write_addr , write_len ) ;
len_remain - = write_len ;
2021-07-23 02:35:27 -07:00
assert ( len_remain < length ) ;
2022-08-23 12:31:53 +08:00
COUNTER_ADD_BYTES ( write , write_len ) ;
2019-01-08 18:29:25 +08:00
2023-03-02 16:10:02 +08:00
if ( err ! = ESP_OK ) {
//Error happens, we end flash operation. Re-enable cache and flush it
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
assert ( bus_acquired ) ;
2023-03-02 16:10:02 +08:00
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
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
break ;
}
2019-01-08 18:29:25 +08:00
2020-04-30 10:37:35 +08:00
err = rom_spiflash_api_funcs - > end ( chip , err ) ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
if ( err ! = ESP_OK ) {
2023-03-02 16:10:02 +08:00
goto restore_cache ;
esp_flash: refactor to support various type of yield
There is a periodically yield in the esp_flash driver, to ensure the
cache will not be disabled for too long on ESP32.
On ESP32-S2 and later, we need to support more different kind of yield:
1. polling conditions, including timeout, SW read request, etc.
2. wait for events, including HW done/error/auto-suspend, timeout
semaphore, etc.
The check_yield() and yield() is separated into two parts, because we
may need to insert suspend, etc. between them.
2020-09-11 18:20:08 +08:00
}
bus_acquired = false ;
write_addr + = write_len ;
buffer = ( void * ) ( ( intptr_t ) buffer + write_len ) ;
}
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( write ) ;
2022-09-13 10:26:59 +08:00
err = rom_spiflash_api_funcs - > flash_end_flush_cache ( chip , err , bus_acquired , address , length ) ;
2023-03-02 16:10:02 +08:00
return err ;
2022-09-13 10:26:59 +08:00
2023-03-02 16:10:02 +08:00
restore_cache :
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( write ) ;
2023-03-02 16:10:02 +08:00
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 " ) ;
}
2022-09-13 10:26:59 +08:00
return err ;
2019-01-08 18:29:25 +08:00
}
2024-02-22 18:05:47 +08:00
inline static IRAM_ATTR bool regions_overlap ( uint32_t a_start , uint32_t a_len , uint32_t b_start , uint32_t b_len )
{
uint32_t a_end = a_start + a_len ;
uint32_t b_end = b_start + b_len ;
return ( a_end > b_start & & b_end > a_start ) ;
}
esp_err_t IRAM_ATTR esp_flash_read_encrypted ( esp_flash_t * chip , uint32_t address , void * out_buffer , uint32_t length )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
if ( err ! = ESP_OK ) return err ;
if ( address + length > g_rom_flashchip . chip_size ) {
return ESP_ERR_INVALID_SIZE ;
}
if ( length = = 0 ) {
return ESP_OK ;
}
if ( out_buffer = = NULL ) {
return ESP_ERR_INVALID_ARG ;
}
COUNTER_START ( ) ;
const uint8_t * map ;
spi_flash_mmap_handle_t map_handle ;
size_t map_src = address & ~ ( SPI_FLASH_MMU_PAGE_SIZE - 1 ) ;
size_t map_size = length + ( address - map_src ) ;
err = spi_flash_mmap ( map_src , map_size , SPI_FLASH_MMAP_DATA , ( const void * * ) & map , & map_handle ) ;
if ( err ! = ESP_OK ) {
return err ;
}
memcpy ( out_buffer , map + ( address - map_src ) , length ) ;
spi_flash_munmap ( map_handle ) ;
COUNTER_ADD_BYTES ( read , length ) ;
COUNTER_STOP ( read ) ;
return err ;
}
// test only, non-public
IRAM_ATTR esp_err_t esp_flash_get_io_mode ( esp_flash_t * chip , bool * qe )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( get_io_mode ) ;
esp_flash_io_mode_t io_mode ;
err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
err = chip - > chip_drv - > get_io_mode ( chip , & io_mode ) ;
err = rom_spiflash_api_funcs - > end ( chip , err ) ;
if ( err = = ESP_OK ) {
* qe = ( io_mode = = SPI_FLASH_QOUT ) ;
}
return err ;
}
IRAM_ATTR esp_err_t esp_flash_set_io_mode ( esp_flash_t * chip , bool qe )
{
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
VERIFY_CHIP_OP ( set_io_mode ) ;
chip - > read_mode = ( qe ? SPI_FLASH_QOUT : SPI_FLASH_SLOWRD ) ;
err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( err ! = ESP_OK ) {
return err ;
}
err = chip - > chip_drv - > set_io_mode ( chip ) ;
return rom_spiflash_api_funcs - > end ( chip , err ) ;
}
# endif //CONFIG_SPI_FLASH_ROM_IMPL
# if !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
// use `esp_flash_write_encrypted` ROM version not in C3 and S3
2019-01-08 18:29:25 +08:00
esp_err_t IRAM_ATTR esp_flash_write_encrypted ( esp_flash_t * chip , uint32_t address , const void * buffer , uint32_t length )
{
2023-03-02 16:11:54 +08:00
esp_err_t ret = ESP_FAIL ;
2022-09-13 10:26:59 +08:00
# if CONFIG_SPI_FLASH_VERIFY_WRITE
2023-03-02 16:11:54 +08:00
//used for verify write
bool is_encrypted = true ;
2022-09-13 10:26:59 +08:00
# endif //CONFIG_SPI_FLASH_VERIFY_WRITE
2020-04-30 10:37:35 +08:00
esp_err_t err = rom_spiflash_api_funcs - > chip_check ( & chip ) ;
2024-02-14 13:45:11 +05:30
VERIFY_CHIP_OP ( write ) ;
2021-02-25 12:25:38 +08:00
// Flash encryption only support on main flash.
if ( chip ! = esp_flash_default_chip ) {
return ESP_ERR_NOT_SUPPORTED ;
}
2024-02-14 13:45:11 +05:30
CHECK_WRITE_ADDRESS ( chip , address , length ) ;
2021-02-25 12:25:38 +08:00
if ( buffer = = NULL | | address + length > chip - > size ) {
return ESP_ERR_INVALID_ARG ;
}
if ( ( address % 16 ) ! = 0 ) {
ESP_EARLY_LOGE ( TAG , " flash encrypted write address must be 16 bytes aligned " ) ;
2019-01-08 18:29:25 +08:00
return ESP_ERR_INVALID_ARG ;
}
2021-02-25 12:25:38 +08:00
2021-07-23 02:35:27 -07:00
if ( length = = 0 ) {
return ESP_OK ;
}
2021-02-25 12:25:38 +08:00
if ( ( length % 16 ) ! = 0 ) {
ESP_EARLY_LOGE ( TAG , " flash encrypted write length must be multiple of 16 " ) ;
return ESP_ERR_INVALID_SIZE ;
}
bool bus_acquired = false ;
const uint8_t * ssrc = ( const uint8_t * ) buffer ;
2022-08-23 12:31:53 +08:00
COUNTER_START ( ) ;
2021-02-25 12:25:38 +08:00
/* On ESP32, write_encrypted encrypts data in RAM as it writes,
so copy to a temporary buffer - 32 bytes at a time .
Each call to write_encrypted takes a 32 byte " row " of
data to encrypt , and each row is two 16 byte AES blocks
that share a key ( as derived from flash address ) .
On ESP32 - S2 and later , the temporary buffer need to be
seperated into 16 - bytes , 32 - bytes , 64 - bytes ( if supported ) .
So , on ESP32 - S2 and later , here has a totally different
data prepare implementation .
*/
uint8_t encrypt_buf [ 64 ] __attribute__ ( ( aligned ( 4 ) ) ) ;
uint32_t row_size_length ;
for ( size_t i = 0 ; i < length ; i + = row_size_length ) {
uint32_t row_addr = address + i ;
uint8_t row_size ;
uint8_t encrypt_byte ;
# if CONFIG_IDF_TARGET_ESP32
if ( i = = 0 & & ( row_addr % 32 ) ! = 0 ) {
/* writing to second block of a 32 byte row */
row_size = 16 ;
row_addr - = 16 ;
/* copy to second block in buffer */
memcpy ( encrypt_buf + 16 , ssrc + i , row_size ) ;
/* decrypt the first block from flash, will reencrypt to same bytes */
esp_flash_read_encrypted ( chip , row_addr , encrypt_buf , 16 ) ;
} else if ( length - i = = 16 ) {
/* 16 bytes left, is first block of a 32 byte row */
row_size = 16 ;
/* copy to first block in buffer */
memcpy ( encrypt_buf , ssrc + i , row_size ) ;
/* decrypt the second block from flash, will reencrypt to same bytes */
esp_flash_read_encrypted ( chip , row_addr + 16 , encrypt_buf + 16 , 16 ) ;
} else {
/* Writing a full 32 byte row (2 blocks) */
row_size = 32 ;
memcpy ( encrypt_buf , ssrc + i , row_size ) ;
}
encrypt_byte = 32 ;
row_size_length = row_size ;
# else // FOR ESP32-S2, ESP32-S3, ESP32-C3
if ( ( row_addr % 64 ) = = 0 & & ( length - i ) > = 64 & & SOC_FLASH_ENCRYPTED_XTS_AES_BLOCK_MAX = = 64 ) {
row_size = 64 ;
memcpy ( encrypt_buf , ssrc + i , row_size ) ;
} else if ( ( row_addr % 32 ) = = 0 & & ( length - i ) > = 32 ) {
row_size = 32 ;
memcpy ( encrypt_buf , ssrc + i , row_size ) ;
} else {
row_size = 16 ;
memcpy ( encrypt_buf , ssrc + i , row_size ) ;
}
encrypt_byte = row_size ;
row_size_length = row_size ;
# endif //CONFIG_IDF_TARGET_ESP32
2023-03-02 16:11:54 +08:00
# 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
2021-02-25 12:25:38 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
esp_crypto_dma_lock_acquire ( ) ;
# endif //CONFIG_IDF_TARGET_ESP32S2
err = rom_spiflash_api_funcs - > start ( chip ) ;
if ( err ! = ESP_OK ) {
# if CONFIG_IDF_TARGET_ESP32S2
esp_crypto_dma_lock_release ( ) ;
# endif //CONFIG_IDF_TARGET_ESP32S2
2023-03-02 16:11:54 +08:00
//Error happens, we end flash operation. Re-enable cache and flush it
goto restore_cache ;
2021-02-25 12:25:38 +08:00
}
bus_acquired = true ;
err = chip - > chip_drv - > write_encrypted ( chip , ( uint32_t * ) encrypt_buf , row_addr , encrypt_byte ) ;
if ( err ! = ESP_OK ) {
# if CONFIG_IDF_TARGET_ESP32S2
esp_crypto_dma_lock_release ( ) ;
# endif //CONFIG_IDF_TARGET_ESP32S2
bus_acquired = false ;
assert ( bus_acquired ) ;
2023-03-02 16:11:54 +08:00
//Error happens, we end flash operation. Re-enable cache and flush it
goto restore_cache ;
2021-02-25 12:25:38 +08:00
}
err = rom_spiflash_api_funcs - > end ( chip , ESP_OK ) ;
2022-08-23 12:31:53 +08:00
COUNTER_ADD_BYTES ( write , encrypt_byte ) ;
2021-02-25 12:25:38 +08:00
# if CONFIG_IDF_TARGET_ESP32S2
esp_crypto_dma_lock_release ( ) ;
# endif //CONFIG_IDF_TARGET_ESP32S2
if ( err ! = ESP_OK ) {
bus_acquired = false ;
2023-03-02 16:11:54 +08:00
//Error happens, we end flash operation. Re-enable cache and flush it
goto restore_cache ;
2021-02-25 12:25:38 +08:00
}
bus_acquired = false ;
2022-09-13 10:26:59 +08:00
# if CONFIG_SPI_FLASH_VERIFY_WRITE
2023-03-02 16:11:54 +08:00
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 ;
2022-09-13 10:26:59 +08:00
}
2023-03-02 16:11:54 +08:00
# endif //CONFIG_SPI_FLASH_VERIFY_WRITE
2022-09-13 10:26:59 +08:00
}
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( write ) ;
2023-03-02 16:11:54 +08:00
err = rom_spiflash_api_funcs - > flash_end_flush_cache ( chip , err , bus_acquired , address , length ) ;
return err ;
restore_cache :
2022-08-23 12:31:53 +08:00
COUNTER_STOP ( write ) ;
2023-03-02 16:11:54 +08:00
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 " ) ;
}
2022-09-13 10:26:59 +08:00
return err ;
2019-01-08 18:29:25 +08:00
}
2024-02-22 18:05:47 +08:00
# endif // !CONFIG_SPI_FLASH_ROM_IMPL || ESP_ROM_HAS_ENCRYPTED_WRITES_USING_LEGACY_DRV
2019-09-05 13:11:36 +08:00
2020-12-18 12:57:55 +08:00
//init suspend mode cmd, uses internal.
esp_err_t esp_flash_suspend_cmd_init ( esp_flash_t * chip )
{
2021-03-19 17:39:56 +08:00
ESP_EARLY_LOGW ( TAG , " Flash suspend feature is enabled " ) ;
2021-05-18 12:05:41 +08:00
if ( chip - > chip_drv - > get_chip_caps = = NULL ) {
// chip caps get failed, pass the flash capability check.
ESP_EARLY_LOGW ( TAG , " get_chip_caps function pointer hasn't been initialized " ) ;
} else {
if ( ( chip - > chip_drv - > get_chip_caps ( chip ) & SPI_FLASH_CHIP_CAP_SUSPEND ) = = 0 ) {
ESP_EARLY_LOGW ( TAG , " Suspend and resume may not supported for this flash model yet. " ) ;
}
}
2020-12-18 12:57:55 +08:00
return chip - > chip_drv - > sus_setup ( chip ) ;
}
2019-09-12 02:41:00 +08:00
esp_err_t esp_flash_app_disable_protect ( bool disable )
{
if ( disable ) {
return esp_flash_app_disable_os_functions ( esp_flash_default_chip ) ;
} else {
2020-04-09 13:30:12 +08:00
return esp_flash_app_enable_os_functions ( esp_flash_default_chip ) ;
2019-09-12 02:41:00 +08:00
}
}