2022-05-11 18:49:55 +08:00
/*
* SPDX - FileCopyrightText : 2015 - 2022 Espressif Systems ( Shanghai ) CO LTD
*
* SPDX - License - Identifier : Apache - 2.0
*/
2016-08-17 23:08:22 +08:00
# include "catch.hpp"
# include "nvs.hpp"
2016-11-15 18:24:56 +08:00
# include "nvs_test_api.h"
2019-06-21 20:31:22 +08:00
# include "sdkconfig.h"
2016-08-17 23:08:22 +08:00
# include "spi_flash_emulation.h"
2020-04-27 08:51:31 +08:00
# include "nvs_partition_manager.hpp"
# include "nvs_partition.hpp"
# include "mbedtls/aes.h"
2016-08-17 23:08:22 +08:00
# include <sstream>
# include <iostream>
2018-04-02 16:14:59 +05:30
# include <fstream>
2018-11-28 10:56:06 +05:30
# include <dirent.h>
2018-04-19 05:48:05 +05:30
# include <unistd.h>
# include <sys/wait.h>
2018-11-28 10:56:06 +05:30
# include <string.h>
2018-11-05 09:03:04 +01:00
# include <string>
2020-04-27 08:51:31 +08:00
# include "test_fixtures.hpp"
2017-05-12 12:18:08 +08:00
# define TEST_ESP_ERR(rc, res) CHECK((rc) == (res))
# define TEST_ESP_OK(rc) CHECK((rc) == ESP_OK)
2016-08-17 23:08:22 +08:00
stringstream s_perf ;
2022-10-18 15:06:10 +02:00
void dumpBytes ( const uint8_t * data , size_t count )
2016-08-17 23:08:22 +08:00
{
for ( uint32_t i = 0 ; i < count ; + + i ) {
if ( i % 32 = = 0 ) {
printf ( " %08x " , i ) ;
}
printf ( " %02x " , data [ i ] ) ;
if ( ( i + 1 ) % 32 = = 0 ) {
printf ( " \n " ) ;
}
}
}
2017-05-31 12:59:24 +08:00
2018-03-12 06:35:21 +08:00
TEST_CASE ( " Page handles invalid CRC of variable length items " , " [nvs][cur] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 4 ) ;
2018-03-12 06:35:21 +08:00
{
2022-10-18 15:06:10 +02:00
nvs : : Page page ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( page . load ( & f . part , 0 ) ) ;
2018-03-12 06:35:21 +08:00
char buf [ 128 ] = { 0 } ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( page . writeItem ( 1 , nvs : : ItemType : : BLOB , " 1 " , buf , sizeof ( buf ) ) ) ;
2018-03-12 06:35:21 +08:00
}
// corrupt header of the item (64 is the offset of the first item in page)
uint32_t overwrite_buf = 0 ;
2020-04-27 08:51:31 +08:00
f . emu . write ( 64 , & overwrite_buf , 4 ) ;
2018-03-12 06:35:21 +08:00
// load page again
{
2022-10-18 15:06:10 +02:00
nvs : : Page page ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( page . load ( & f . part , 0 ) ) ;
2018-03-12 06:35:21 +08:00
}
}
2019-12-06 16:16:07 +08:00
TEST_CASE ( " namespace name is deep copy " , " [nvs] " )
{
char ns_name [ 16 ] ;
strcpy ( ns_name , " const_name " ) ;
nvs_handle_t handle_1 ;
nvs_handle_t handle_2 ;
const uint32_t NVS_FLASH_SECTOR = 6 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3 ;
2022-10-18 15:06:10 +02:00
PartitionEmulationFixture f ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR_COUNT_MIN ) ;
2020-04-27 08:51:31 +08:00
f . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2019-12-06 16:16:07 +08:00
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2019-12-06 16:16:07 +08:00
TEST_ESP_OK ( nvs_open ( " const_name " , NVS_READWRITE , & handle_1 ) ) ;
strcpy ( ns_name , " just_kidding " ) ;
CHECK ( nvs_open ( " just_kidding " , NVS_READONLY , & handle_2 ) = = ESP_ERR_NVS_NOT_FOUND ) ;
nvs_close ( handle_1 ) ;
nvs_close ( handle_2 ) ;
nvs_flash_deinit_partition ( NVS_DEFAULT_PART_NAME ) ;
}
2019-04-15 14:06:57 -05:00
TEST_CASE ( " writing the identical content does not write or erase " , " [nvs] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 20 ) ;
2019-04-15 14:06:57 -05:00
const uint32_t NVS_FLASH_SECTOR = 5 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 10 ;
2020-04-27 08:51:31 +08:00
f . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2019-04-15 14:06:57 -05:00
nvs_handle misc_handle ;
TEST_ESP_OK ( nvs_open ( " test " , NVS_READWRITE , & misc_handle ) ) ;
// Test writing a u8 twice, then changing it
nvs_set_u8 ( misc_handle , " test_u8 " , 8 ) ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_u8 ( misc_handle , " test_u8 " , 8 ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) = = 0 ) ;
CHECK ( f . emu . getEraseOps ( ) = = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_u8 ( misc_handle , " test_u8 " , 9 ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) ! = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
2019-04-15 14:06:57 -05:00
// Test writing a string twice, then changing it
static const char * test [ 2 ] = { " Hello world. " , " Hello world! " } ;
nvs_set_str ( misc_handle , " test_str " , test [ 0 ] ) ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_str ( misc_handle , " test_str " , test [ 0 ] ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) = = 0 ) ;
CHECK ( f . emu . getEraseOps ( ) = = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_str ( misc_handle , " test_str " , test [ 1 ] ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) ! = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
2019-04-15 14:06:57 -05:00
// Test writing a multi-page blob, then changing it
2022-10-18 15:06:10 +02:00
uint8_t blob [ nvs : : Page : : CHUNK_MAX_SIZE * 3 ] = { 0 } ;
2019-04-15 14:06:57 -05:00
memset ( blob , 1 , sizeof ( blob ) ) ;
nvs_set_blob ( misc_handle , " test_blob " , blob , sizeof ( blob ) ) ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_blob ( misc_handle , " test_blob " , blob , sizeof ( blob ) ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) = = 0 ) ;
CHECK ( f . emu . getEraseOps ( ) = = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
2019-04-15 14:06:57 -05:00
blob [ sizeof ( blob ) - 1 ] + + ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2019-04-15 14:06:57 -05:00
nvs_set_blob ( misc_handle , " test_blob " , blob , sizeof ( blob ) ) ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . getWriteOps ( ) ! = 0 ) ;
CHECK ( f . emu . getReadOps ( ) ! = 0 ) ;
TEST_ESP_OK ( nvs_flash_deinit_partition ( NVS_DEFAULT_PART_NAME ) ) ;
2019-04-15 14:06:57 -05:00
}
2016-08-17 23:08:22 +08:00
TEST_CASE ( " can init storage from flash with random contents " , " [nvs] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 10 ) ;
f . emu . randomize ( 42 ) ;
2018-07-02 16:40:43 +05:30
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2016-08-17 23:08:22 +08:00
const uint32_t NVS_FLASH_SECTOR = 5 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3 ;
2020-04-27 08:51:31 +08:00
f . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
TEST_ESP_OK ( nvs_open ( " nvs.net80211 " , NVS_READWRITE , & handle ) ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
uint8_t opmode = 2 ;
if ( nvs_get_u8 ( handle , " wifi.opmode " , & opmode ) ! = ESP_OK ) {
TEST_ESP_OK ( nvs_set_u8 ( handle , " wifi.opmode " , opmode ) ) ;
}
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( NVS_DEFAULT_PART_NAME ) ) ;
2016-08-17 23:08:22 +08:00
}
2018-04-16 11:51:43 +08:00
TEST_CASE ( " nvs api tests, starting with random data in flash " , " [nvs][long] " )
2016-08-17 23:08:22 +08:00
{
2018-04-16 15:02:11 +08:00
const size_t testIters = 3000 ;
int lastPercent = - 1 ;
for ( size_t count = 0 ; count < testIters ; + + count ) {
int percentDone = ( int ) ( count * 100 / testIters ) ;
if ( percentDone ! = lastPercent ) {
lastPercent = percentDone ;
printf ( " %d%% \n " , percentDone ) ;
}
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 10 ) ;
f . emu . randomize ( static_cast < uint32_t > ( count ) ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
const uint32_t NVS_FLASH_SECTOR = 6 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3 ;
2020-04-27 08:51:31 +08:00
f . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2018-07-02 16:40:43 +05:30
2019-04-01 15:13:55 +01:00
nvs_handle_t handle_1 ;
2016-08-17 23:08:22 +08:00
TEST_ESP_ERR ( nvs_open ( " namespace1 " , NVS_READONLY , & handle_1 ) , ESP_ERR_NVS_NOT_FOUND ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
TEST_ESP_OK ( nvs_open ( " namespace1 " , NVS_READWRITE , & handle_1 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_1 , " foo " , 0x12345678 ) ) ;
for ( size_t i = 0 ; i < 500 ; + + i ) {
2019-04-01 15:13:55 +01:00
nvs_handle_t handle_2 ;
2016-08-17 23:08:22 +08:00
TEST_ESP_OK ( nvs_open ( " namespace2 " , NVS_READWRITE , & handle_2 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_1 , " foo " , 0x23456789 % ( i + 1 ) ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_2 , " foo " , static_cast < int32_t > ( i ) ) ) ;
2022-10-18 15:06:10 +02:00
const char * str = " value 0123456789abcdef0123456789abcdef %09d " ;
2016-08-17 23:08:22 +08:00
char str_buf [ 128 ] ;
snprintf ( str_buf , sizeof ( str_buf ) , str , i + count * 1024 ) ;
TEST_ESP_OK ( nvs_set_str ( handle_2 , " key " , str_buf ) ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
int32_t v1 ;
TEST_ESP_OK ( nvs_get_i32 ( handle_1 , " foo " , & v1 ) ) ;
CHECK ( 0x23456789 % ( i + 1 ) = = v1 ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
int32_t v2 ;
TEST_ESP_OK ( nvs_get_i32 ( handle_2 , " foo " , & v2 ) ) ;
CHECK ( static_cast < int32_t > ( i ) = = v2 ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
char buf [ 128 ] ;
size_t buf_len = sizeof ( buf ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
TEST_ESP_OK ( nvs_get_str ( handle_2 , " key " , buf , & buf_len ) ) ;
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
CHECK ( 0 = = strcmp ( buf , str_buf ) ) ;
nvs_close ( handle_2 ) ;
}
nvs_close ( handle_1 ) ;
2020-04-27 08:51:31 +08:00
2020-12-17 19:30:19 +11:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( f . part . get_partition_name ( ) ) ) ;
}
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
2017-08-21 15:26:16 +05:30
extern " C " void nvs_dump ( const char * partName ) ;
2016-08-22 12:11:32 +08:00
class RandomTest {
2018-11-28 10:56:06 +05:30
2018-04-02 16:14:59 +05:30
static const size_t nKeys = 11 ;
2016-08-17 23:08:22 +08:00
int32_t v1 = 0 , v2 = 0 ;
uint64_t v3 = 0 , v4 = 0 ;
2016-08-22 12:11:32 +08:00
static const size_t strBufLen = 1024 ;
2022-10-18 15:06:10 +02:00
static const size_t smallBlobLen = nvs : : Page : : CHUNK_MAX_SIZE / 3 ;
static const size_t largeBlobLen = nvs : : Page : : CHUNK_MAX_SIZE * 3 ;
2016-08-17 23:08:22 +08:00
char v5 [ strBufLen ] , v6 [ strBufLen ] , v7 [ strBufLen ] , v8 [ strBufLen ] , v9 [ strBufLen ] ;
2018-04-02 16:14:59 +05:30
uint8_t v10 [ smallBlobLen ] , v11 [ largeBlobLen ] ;
2016-08-17 23:08:22 +08:00
bool written [ nKeys ] ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
public :
RandomTest ( )
{
std : : fill_n ( written , nKeys , false ) ;
}
template < typename TGen >
2022-10-18 15:06:10 +02:00
esp_err_t doRandomThings ( nvs_handle_t handle , TGen gen , size_t & count )
{
2018-11-28 10:56:06 +05:30
2022-10-18 15:06:10 +02:00
const char * keys [ ] = { " foo " , " bar " , " longkey_0123456 " , " another key " , " param1 " , " param2 " , " param3 " , " param4 " , " param5 " , " singlepage " , " multipage " } ;
const nvs : : ItemType types [ ] = { nvs : : ItemType : : I32 , nvs : : ItemType : : I32 , nvs : : ItemType : : U64 , nvs : : ItemType : : U64 , nvs : : ItemType : : SZ , nvs : : ItemType : : SZ , nvs : : ItemType : : SZ , nvs : : ItemType : : SZ , nvs : : ItemType : : SZ , nvs : : ItemType : : BLOB , nvs : : ItemType : : BLOB } ;
2018-11-28 10:56:06 +05:30
2022-10-18 15:06:10 +02:00
void * values [ ] = { & v1 , & v2 , & v3 , & v4 , & v5 , & v6 , & v7 , & v8 , & v9 , & v10 , & v11 } ;
2018-11-28 10:56:06 +05:30
2016-08-22 12:11:32 +08:00
const size_t nKeys = sizeof ( keys ) / sizeof ( keys [ 0 ] ) ;
static_assert ( nKeys = = sizeof ( types ) / sizeof ( types [ 0 ] ) , " " ) ;
static_assert ( nKeys = = sizeof ( values ) / sizeof ( values [ 0 ] ) , " " ) ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
auto randomRead = [ & ] ( size_t index ) - > esp_err_t {
2022-10-18 15:06:10 +02:00
switch ( types [ index ] )
{
case nvs : : ItemType : : I32 : {
int32_t val ;
auto err = nvs_get_i32 ( handle , keys [ index ] , & val ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( ! written [ index ] ) {
REQUIRE ( err = = ESP_ERR_NVS_NOT_FOUND ) ;
} else {
REQUIRE ( err = = ESP_OK ) ;
REQUIRE ( val = = * reinterpret_cast < int32_t * > ( values [ index ] ) ) ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
break ;
}
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
case nvs : : ItemType : : U64 : {
uint64_t val ;
auto err = nvs_get_u64 ( handle , keys [ index ] , & val ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( ! written [ index ] ) {
REQUIRE ( err = = ESP_ERR_NVS_NOT_FOUND ) ;
} else {
REQUIRE ( err = = ESP_OK ) ;
REQUIRE ( val = = * reinterpret_cast < uint64_t * > ( values [ index ] ) ) ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
break ;
}
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
case nvs : : ItemType : : SZ : {
char buf [ strBufLen ] ;
size_t len = strBufLen ;
auto err = nvs_get_str ( handle , keys [ index ] , buf , & len ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
if ( ! written [ index ] ) {
REQUIRE ( err = = ESP_ERR_NVS_NOT_FOUND ) ;
} else {
REQUIRE ( err = = ESP_OK ) ;
REQUIRE ( strncmp ( buf , reinterpret_cast < const char * > ( values [ index ] ) , strBufLen ) = = 0 ) ;
}
break ;
}
case nvs : : ItemType : : BLOB : {
uint32_t blobBufLen = 0 ;
if ( strncmp ( keys [ index ] , " singlepage " , sizeof ( " singlepage " ) ) = = 0 ) {
blobBufLen = smallBlobLen ;
} else {
blobBufLen = largeBlobLen ;
2018-04-02 16:14:59 +05:30
}
2022-10-18 15:06:10 +02:00
uint8_t buf [ blobBufLen ] ;
memset ( buf , 0 , blobBufLen ) ;
2018-04-02 16:14:59 +05:30
2022-10-18 15:06:10 +02:00
size_t len = blobBufLen ;
auto err = nvs_get_blob ( handle , keys [ index ] , buf , & len ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( ! written [ index ] ) {
REQUIRE ( err = = ESP_ERR_NVS_NOT_FOUND ) ;
} else {
REQUIRE ( err = = ESP_OK ) ;
REQUIRE ( memcmp ( buf , reinterpret_cast < const uint8_t * > ( values [ index ] ) , blobBufLen ) = = 0 ) ;
}
break ;
}
2018-04-02 16:14:59 +05:30
2022-10-18 15:06:10 +02:00
default :
assert ( 0 ) ;
2016-08-17 23:08:22 +08:00
}
2016-08-22 12:11:32 +08:00
return ESP_OK ;
} ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
auto randomWrite = [ & ] ( size_t index ) - > esp_err_t {
2022-10-18 15:06:10 +02:00
switch ( types [ index ] )
{
case nvs : : ItemType : : I32 : {
int32_t val = static_cast < int32_t > ( gen ( ) ) ;
auto err = nvs_set_i32 ( handle , keys [ index ] , val ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( err = = ESP_ERR_NVS_REMOVE_FAILED ) {
2016-08-22 12:11:32 +08:00
written [ index ] = true ;
2022-10-18 15:06:10 +02:00
* reinterpret_cast < int32_t * > ( values [ index ] ) = val ;
return ESP_ERR_FLASH_OP_FAIL ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
REQUIRE ( err = = ESP_OK ) ;
written [ index ] = true ;
* reinterpret_cast < int32_t * > ( values [ index ] ) = val ;
break ;
}
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
case nvs : : ItemType : : U64 : {
uint64_t val = static_cast < uint64_t > ( gen ( ) ) ;
auto err = nvs_set_u64 ( handle , keys [ index ] , val ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( err = = ESP_ERR_NVS_REMOVE_FAILED ) {
2016-08-22 12:11:32 +08:00
written [ index ] = true ;
2022-10-18 15:06:10 +02:00
* reinterpret_cast < uint64_t * > ( values [ index ] ) = val ;
return ESP_ERR_FLASH_OP_FAIL ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
REQUIRE ( err = = ESP_OK ) ;
written [ index ] = true ;
* reinterpret_cast < uint64_t * > ( values [ index ] ) = val ;
break ;
}
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
case nvs : : ItemType : : SZ : {
char buf [ strBufLen ] ;
size_t len = strBufLen ;
size_t strLen = gen ( ) % ( strBufLen - 1 ) ;
std : : generate_n ( buf , strLen , [ & ] ( ) - > char {
const char c = static_cast < char > ( gen ( ) % 127 ) ;
return ( c < 32 ) ? 32 : c ;
} ) ;
buf [ strLen ] = 0 ;
auto err = nvs_set_str ( handle , keys [ index ] , buf ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( err = = ESP_ERR_NVS_REMOVE_FAILED ) {
2016-08-22 12:11:32 +08:00
written [ index ] = true ;
2022-10-18 15:06:10 +02:00
strncpy ( reinterpret_cast < char * > ( values [ index ] ) , buf , strBufLen ) ;
return ESP_ERR_FLASH_OP_FAIL ;
2016-08-17 23:08:22 +08:00
}
2022-10-18 15:06:10 +02:00
REQUIRE ( err = = ESP_OK ) ;
written [ index ] = true ;
strncpy ( reinterpret_cast < char * > ( values [ index ] ) , buf , strBufLen ) ;
break ;
}
2018-04-02 16:14:59 +05:30
2022-10-18 15:06:10 +02:00
case nvs : : ItemType : : BLOB : {
uint32_t blobBufLen = 0 ;
if ( strncmp ( keys [ index ] , " singlepage " , sizeof ( " singlepage " ) ) = = 0 ) {
blobBufLen = smallBlobLen ;
} else {
blobBufLen = largeBlobLen ;
}
uint8_t buf [ blobBufLen ] ;
memset ( buf , 0 , blobBufLen ) ;
size_t blobLen = gen ( ) % blobBufLen ;
std : : generate_n ( buf , blobLen , [ & ] ( ) - > uint8_t {
return static_cast < uint8_t > ( gen ( ) % 256 ) ;
} ) ;
auto err = nvs_set_blob ( handle , keys [ index ] , buf , blobLen ) ;
if ( err = = ESP_ERR_FLASH_OP_FAIL ) {
return err ;
}
if ( err = = ESP_ERR_NVS_REMOVE_FAILED ) {
2018-04-02 16:14:59 +05:30
written [ index ] = true ;
2022-10-18 15:06:10 +02:00
memcpy ( reinterpret_cast < uint8_t * > ( values [ index ] ) , buf , blobBufLen ) ;
return ESP_ERR_FLASH_OP_FAIL ;
2018-04-02 16:14:59 +05:30
}
2022-10-18 15:06:10 +02:00
REQUIRE ( err = = ESP_OK ) ;
written [ index ] = true ;
memcpy ( reinterpret_cast < char * > ( values [ index ] ) , buf , blobBufLen ) ;
break ;
}
2018-04-02 16:14:59 +05:30
2022-10-18 15:06:10 +02:00
default :
assert ( 0 ) ;
2016-08-17 23:08:22 +08:00
}
2016-08-22 12:11:32 +08:00
return ESP_OK ;
} ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
for ( ; count ! = 0 ; - - count ) {
2018-04-02 16:14:59 +05:30
size_t index = gen ( ) % ( nKeys ) ;
2016-08-22 12:11:32 +08:00
switch ( gen ( ) % 3 ) {
2022-10-18 15:06:10 +02:00
case 0 : // read, 1/3
if ( randomRead ( index ) = = ESP_ERR_FLASH_OP_FAIL ) {
return ESP_ERR_FLASH_OP_FAIL ;
}
break ;
default : // write, 2/3
if ( randomWrite ( index ) = = ESP_ERR_FLASH_OP_FAIL ) {
return ESP_ERR_FLASH_OP_FAIL ;
}
break ;
2016-08-17 23:08:22 +08:00
}
}
return ESP_OK ;
}
2016-08-22 12:11:32 +08:00
2022-10-18 15:06:10 +02:00
esp_err_t handleExternalWriteAtIndex ( uint8_t index , const void * value , const size_t len )
{
if ( index = = 9 ) { /* This is only done for small-page blobs for now*/
if ( len > smallBlobLen ) {
2018-04-02 16:14:59 +05:30
return ESP_FAIL ;
}
memcpy ( v10 , value , len ) ;
written [ index ] = true ;
return ESP_OK ;
} else {
return ESP_FAIL ;
}
2018-11-28 10:56:06 +05:30
}
2018-04-02 16:14:59 +05:30
} ;
2016-08-17 23:08:22 +08:00
2018-04-16 11:51:43 +08:00
TEST_CASE ( " test recovery from sudden poweroff " , " [long][nvs][recovery][monkey] " )
2016-08-22 12:11:32 +08:00
{
std : : random_device rd ;
std : : mt19937 gen ( rd ( ) ) ;
uint32_t seed = 3 ;
gen . seed ( seed ) ;
const size_t iter_count = 2000 ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
size_t totalOps = 0 ;
int lastPercent = - 1 ;
2016-10-31 19:17:25 +08:00
for ( uint32_t errDelay = 0 ; ; + + errDelay ) {
2016-08-22 12:11:32 +08:00
INFO ( errDelay ) ;
2020-12-17 19:30:19 +11:00
PartitionEmulationFixture f ( 0 , 10 ) ;
const uint32_t NVS_FLASH_SECTOR = 2 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 8 ;
f . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2020-04-27 08:51:31 +08:00
f . emu . randomize ( seed ) ;
f . emu . clearStats ( ) ;
f . emu . failAfter ( errDelay ) ;
2016-08-22 12:11:32 +08:00
RandomTest test ;
2018-07-02 16:40:43 +05:30
2016-08-22 12:11:32 +08:00
if ( totalOps ! = 0 ) {
int percent = errDelay * 100 / totalOps ;
2016-10-31 19:17:25 +08:00
if ( percent > lastPercent ) {
2016-08-22 12:11:32 +08:00
printf ( " %d/%d (%d%%) \r \n " , errDelay , static_cast < int > ( totalOps ) , percent ) ;
lastPercent = percent ;
}
}
2018-07-02 16:40:43 +05:30
2016-08-17 23:08:22 +08:00
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2016-08-22 12:11:32 +08:00
size_t count = iter_count ;
2016-10-31 19:17:25 +08:00
2022-10-18 15:06:10 +02:00
if ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
2020-04-27 08:51:31 +08:00
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) = = ESP_OK ) {
2020-12-17 19:30:19 +11:00
auto res = ESP_ERR_FLASH_OP_FAIL ;
2016-10-31 19:17:25 +08:00
if ( nvs_open ( " namespace1 " , NVS_READWRITE , & handle ) = = ESP_OK ) {
2020-12-17 19:30:19 +11:00
res = test . doRandomThings ( handle , gen , count ) ;
2016-10-31 19:17:25 +08:00
nvs_close ( handle ) ;
}
2020-12-17 19:30:19 +11:00
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( NVS_DEFAULT_PART_NAME ) ) ;
2020-12-17 19:30:19 +11:00
if ( res ! = ESP_ERR_FLASH_OP_FAIL ) {
// This means we got to the end without an error due to f.emu.failAfter(), therefore errDelay
// is high enough that we're not triggering it any more, therefore we're done
break ;
}
2016-08-22 12:11:32 +08:00
}
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part ,
NVS_FLASH_SECTOR ,
NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2016-08-22 12:11:32 +08:00
TEST_ESP_OK ( nvs_open ( " namespace1 " , NVS_READWRITE , & handle ) ) ;
auto res = test . doRandomThings ( handle , gen , count ) ;
if ( res ! = ESP_OK ) {
2017-08-21 15:26:16 +05:30
nvs_dump ( NVS_DEFAULT_PART_NAME ) ;
2016-08-22 12:11:32 +08:00
CHECK ( 0 ) ;
}
nvs_close ( handle ) ;
2020-04-27 08:51:31 +08:00
totalOps = f . emu . getEraseOps ( ) + f . emu . getWriteBytes ( ) / 4 ;
TEST_ESP_OK ( nvs_flash_deinit_partition ( NVS_DEFAULT_PART_NAME ) ) ;
2016-08-22 12:11:32 +08:00
}
2016-08-17 23:08:22 +08:00
}
2020-12-17 19:30:19 +11:00
2016-10-31 19:48:28 +08:00
TEST_CASE ( " duplicate items are removed " , " [nvs][dupes] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2016-10-31 19:48:28 +08:00
{
// create one item
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & f . part , 0 ) ;
2016-10-31 19:48:28 +08:00
p . writeItem < uint8_t > ( 1 , " opmode " , 3 ) ;
}
{
2016-10-31 21:10:47 +08:00
// add another two without deleting the first one
2022-10-18 15:06:10 +02:00
nvs : : Item item ( 1 , nvs : : ItemType : : U8 , 1 , " opmode " ) ;
2016-10-31 19:48:28 +08:00
item . data [ 0 ] = 2 ;
item . crc32 = item . calculateCrc32 ( ) ;
2022-10-18 15:06:10 +02:00
f . emu . write ( 3 * 32 , reinterpret_cast < const uint32_t * > ( & item ) , sizeof ( item ) ) ;
f . emu . write ( 4 * 32 , reinterpret_cast < const uint32_t * > ( & item ) , sizeof ( item ) ) ;
2016-10-31 21:10:47 +08:00
uint32_t mask = 0xFFFFFFEA ;
2020-04-27 08:51:31 +08:00
f . emu . write ( 32 , & mask , 4 ) ;
2016-10-31 19:48:28 +08:00
}
{
// load page and check that second item persists
2020-04-27 08:51:31 +08:00
nvs : : Storage s ( & f . part ) ;
2016-10-31 21:10:47 +08:00
s . init ( 0 , 3 ) ;
2016-10-31 19:48:28 +08:00
uint8_t val ;
2016-10-31 21:10:47 +08:00
ESP_ERROR_CHECK ( s . readItem ( 1 , " opmode " , val ) ) ;
2016-10-31 19:48:28 +08:00
CHECK ( val = = 2 ) ;
2016-10-31 21:10:47 +08:00
}
{
2022-10-18 15:06:10 +02:00
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & f . part , 0 ) ;
2016-10-31 21:10:47 +08:00
CHECK ( p . getErasedEntryCount ( ) = = 2 ) ;
2016-10-31 19:48:28 +08:00
CHECK ( p . getUsedEntryCount ( ) = = 1 ) ;
}
}
TEST_CASE ( " recovery after failure to write data " , " [nvs] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2016-10-31 21:10:47 +08:00
const char str [ ] = " value 0123456789abcdef012345678value 0123456789abcdef012345678 " ;
2016-10-31 19:48:28 +08:00
2022-10-18 15:06:10 +02:00
// make flash write fail exactly in nvs::Page::writeEntryData
2020-04-27 08:51:31 +08:00
f . emu . failAfter ( 17 ) ;
2016-10-31 19:48:28 +08:00
{
2022-10-18 15:06:10 +02:00
nvs : : Storage storage ( & f . part ) ;
2016-10-31 19:48:28 +08:00
TEST_ESP_OK ( storage . init ( 0 , 3 ) ) ;
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_ERR ( storage . writeItem ( 1 , nvs : : ItemType : : SZ , " key " , str , strlen ( str ) ) , ESP_ERR_FLASH_OP_FAIL ) ;
2018-07-02 16:40:43 +05:30
2016-10-31 19:48:28 +08:00
// check that repeated operations cause an error
2022-10-18 15:06:10 +02:00
TEST_ESP_ERR ( storage . writeItem ( 1 , nvs : : ItemType : : SZ , " key " , str , strlen ( str ) ) , ESP_ERR_NVS_INVALID_STATE ) ;
2018-07-02 16:40:43 +05:30
2016-10-31 19:48:28 +08:00
uint8_t val ;
2022-10-18 15:06:10 +02:00
TEST_ESP_ERR ( storage . readItem ( 1 , nvs : : ItemType : : U8 , " key " , & val , sizeof ( val ) ) , ESP_ERR_NVS_NOT_FOUND ) ;
2016-10-31 19:48:28 +08:00
}
{
// load page and check that data was erased
2022-10-18 15:06:10 +02:00
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & f . part , 0 ) ;
2016-10-31 19:48:28 +08:00
CHECK ( p . getErasedEntryCount ( ) = = 3 ) ;
CHECK ( p . getUsedEntryCount ( ) = = 0 ) ;
2018-07-02 16:40:43 +05:30
2016-10-31 19:48:28 +08:00
// try to write again
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( p . writeItem ( 1 , nvs : : ItemType : : SZ , " key " , str , strlen ( str ) ) ) ;
2016-10-31 19:48:28 +08:00
}
}
2017-09-04 11:54:18 +08:00
TEST_CASE ( " crc errors in item header are handled " , " [nvs] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2022-10-18 15:06:10 +02:00
nvs : : Storage storage ( & f . part ) ;
2017-09-04 11:54:18 +08:00
// prepare some data
TEST_ESP_OK ( storage . init ( 0 , 3 ) ) ;
TEST_ESP_OK ( storage . writeItem ( 0 , " ns1 " , static_cast < uint8_t > ( 1 ) ) ) ;
TEST_ESP_OK ( storage . writeItem ( 1 , " value1 " , static_cast < uint32_t > ( 1 ) ) ) ;
TEST_ESP_OK ( storage . writeItem ( 1 , " value2 " , static_cast < uint32_t > ( 2 ) ) ) ;
2018-07-02 16:40:43 +05:30
2017-09-04 11:54:18 +08:00
// corrupt item header
uint32_t val = 0 ;
2020-04-27 08:51:31 +08:00
f . emu . write ( 32 * 3 , & val , 4 ) ;
2018-07-02 16:40:43 +05:30
2017-09-04 11:54:18 +08:00
// check that storage can recover
TEST_ESP_OK ( storage . init ( 0 , 3 ) ) ;
TEST_ESP_OK ( storage . readItem ( 1 , " value2 " , val ) ) ;
CHECK ( val = = 2 ) ;
// check that the corrupted item is no longer present
TEST_ESP_ERR ( ESP_ERR_NVS_NOT_FOUND , storage . readItem ( 1 , " value1 " , val ) ) ;
2018-07-02 16:40:43 +05:30
2017-09-04 11:54:18 +08:00
// add more items to make the page full
2022-10-18 15:06:10 +02:00
for ( size_t i = 0 ; i < nvs : : Page : : ENTRY_COUNT ; + + i ) {
char item_name [ nvs : : Item : : MAX_KEY_LENGTH + 1 ] ;
2018-02-20 12:11:56 +05:00
snprintf ( item_name , sizeof ( item_name ) , " item_%ld " , ( long int ) i ) ;
2017-09-04 11:54:18 +08:00
TEST_ESP_OK ( storage . writeItem ( 1 , item_name , static_cast < uint32_t > ( i ) ) ) ;
}
// corrupt another item on the full page
val = 0 ;
2020-04-27 08:51:31 +08:00
f . emu . write ( 32 * 4 , & val , 4 ) ;
2018-07-02 16:40:43 +05:30
2017-09-04 11:54:18 +08:00
// check that storage can recover
TEST_ESP_OK ( storage . init ( 0 , 3 ) ) ;
// check that the corrupted item is no longer present
TEST_ESP_ERR ( ESP_ERR_NVS_NOT_FOUND , storage . readItem ( 1 , " value2 " , val ) ) ;
}
2016-10-31 21:10:47 +08:00
TEST_CASE ( " crc error in variable length item is handled " , " [nvs] " )
{
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2016-10-31 21:10:47 +08:00
const uint64_t before_val = 0xbef04e ;
const uint64_t after_val = 0xaf7e4 ;
// write some data
{
2022-10-18 15:06:10 +02:00
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & f . part , 0 ) ;
2016-10-31 21:10:47 +08:00
TEST_ESP_OK ( p . writeItem < uint64_t > ( 0 , " before " , before_val ) ) ;
2022-10-18 15:06:10 +02:00
const char * str = " foobar " ;
TEST_ESP_OK ( p . writeItem ( 0 , nvs : : ItemType : : SZ , " key " , str , strlen ( str ) ) ) ;
2016-10-31 21:10:47 +08:00
TEST_ESP_OK ( p . writeItem < uint64_t > ( 0 , " after " , after_val ) ) ;
}
// corrupt some data
uint32_t w ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . read ( & w , 32 * 3 + 8 , sizeof ( w ) ) ) ;
2016-10-31 21:10:47 +08:00
w & = 0xf000000f ;
2020-04-27 08:51:31 +08:00
CHECK ( f . emu . write ( 32 * 3 + 8 , & w , sizeof ( w ) ) ) ;
2016-10-31 21:10:47 +08:00
// load and check
{
2022-10-18 15:06:10 +02:00
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & f . part , 0 ) ;
2016-10-31 21:10:47 +08:00
CHECK ( p . getUsedEntryCount ( ) = = 2 ) ;
CHECK ( p . getErasedEntryCount ( ) = = 2 ) ;
uint64_t val ;
TEST_ESP_OK ( p . readItem < uint64_t > ( 0 , " before " , val ) ) ;
CHECK ( val = = before_val ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_ERR ( p . findItem ( 0 , nvs : : ItemType : : SZ , " key " ) , ESP_ERR_NVS_NOT_FOUND ) ;
2016-10-31 21:10:47 +08:00
TEST_ESP_OK ( p . readItem < uint64_t > ( 0 , " after " , val ) ) ;
CHECK ( val = = after_val ) ;
}
}
2017-08-21 15:26:16 +05:30
TEST_CASE ( " multiple partitions access check " , " [nvs] " )
{
SpiFlashEmulator emu ( 10 ) ;
2020-04-27 08:51:31 +08:00
PartitionEmulation p0 ( & emu , 0 * SPI_FLASH_SEC_SIZE , 5 * SPI_FLASH_SEC_SIZE , " nvs1 " ) ;
PartitionEmulation p1 ( & emu , 5 * SPI_FLASH_SEC_SIZE , 5 * SPI_FLASH_SEC_SIZE , " nvs2 " ) ;
2018-04-13 14:49:05 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & p0 , 0 , 5 ) ) ;
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & p1 , 5 , 5 ) ) ;
nvs_handle_t handle1 , handle2 ;
TEST_ESP_OK ( nvs_open_from_partition ( " nvs1 " , " test " , NVS_READWRITE , & handle1 ) ) ;
TEST_ESP_OK ( nvs_open_from_partition ( " nvs2 " , " test " , NVS_READWRITE , & handle2 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle1 , " foo " , 0xdeadbeef ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle2 , " foo " , 0xcafebabe ) ) ;
int32_t v1 , v2 ;
TEST_ESP_OK ( nvs_get_i32 ( handle1 , " foo " , & v1 ) ) ;
TEST_ESP_OK ( nvs_get_i32 ( handle2 , " foo " , & v2 ) ) ;
CHECK ( v1 = = 0xdeadbeef ) ;
CHECK ( v2 = = 0xcafebabe ) ;
2020-04-27 08:51:31 +08:00
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( p0 . get_partition_name ( ) ) ) ;
TEST_ESP_OK ( nvs_flash_deinit_partition ( p1 . get_partition_name ( ) ) ) ;
2018-02-20 12:11:56 +05:00
}
2018-04-16 12:50:57 +08:00
2022-10-18 15:06:10 +02:00
// leaks memory
2018-04-13 12:29:06 +05:30
TEST_CASE ( " Recovery from power-off when the entry being erased is not on active page " , " [nvs] " )
{
2022-10-18 15:06:10 +02:00
const size_t blob_size = nvs : : Page : : CHUNK_MAX_SIZE / 2 ;
2018-04-13 12:29:06 +05:30
size_t read_size = blob_size ;
uint8_t blob [ blob_size ] = { 0x11 } ;
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part , 0 , 3 ) ) ;
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2018-04-13 12:29:06 +05:30
TEST_ESP_OK ( nvs_open ( " test " , NVS_READWRITE , & handle ) ) ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2022-10-18 15:06:10 +02:00
f . emu . failAfter ( nvs : : Page : : CHUNK_MAX_SIZE / 4 + 75 ) ;
2018-04-13 12:29:06 +05:30
TEST_ESP_OK ( nvs_set_blob ( handle , " 1a " , blob , blob_size ) ) ;
TEST_ESP_OK ( nvs_set_blob ( handle , " 1b " , blob , blob_size ) ) ;
TEST_ESP_ERR ( nvs_erase_key ( handle , " 1a " ) , ESP_ERR_FLASH_OP_FAIL ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part , 0 , 3 ) ) ;
2018-04-13 12:29:06 +05:30
/* Check 1a is erased fully*/
TEST_ESP_ERR ( nvs_get_blob ( handle , " 1a " , blob , & read_size ) , ESP_ERR_NVS_NOT_FOUND ) ;
2018-04-16 12:50:57 +08:00
2018-04-13 12:29:06 +05:30
/* Check 2b is still accessible*/
TEST_ESP_OK ( nvs_get_blob ( handle , " 1b " , blob , & read_size ) ) ;
2018-04-16 12:50:57 +08:00
2018-04-13 12:29:06 +05:30
nvs_close ( handle ) ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( f . part . get_partition_name ( ) ) ) ;
2018-04-13 12:29:06 +05:30
}
2018-04-16 12:50:57 +08:00
2022-10-18 15:06:10 +02:00
// leaks memory
2018-04-27 12:40:29 +05:30
TEST_CASE ( " Recovery from power-off when page is being freed. " , " [nvs] " )
{
2022-10-18 15:06:10 +02:00
const size_t blob_size = ( nvs : : Page : : ENTRY_COUNT - 3 ) * nvs : : Page : : ENTRY_SIZE ;
size_t read_size = blob_size / 2 ;
2018-04-27 12:40:29 +05:30
uint8_t blob [ blob_size ] = { 0 } ;
2020-04-27 08:51:31 +08:00
PartitionEmulationFixture f ( 0 , 3 ) ;
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part , 0 , 3 ) ) ;
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2018-04-27 12:40:29 +05:30
TEST_ESP_OK ( nvs_open ( " test " , NVS_READWRITE , & handle ) ) ;
// Fill first page
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs_set_blob ( handle , " 1a " , blob , blob_size / 3 ) ) ;
TEST_ESP_OK ( nvs_set_blob ( handle , " 1b " , blob , blob_size / 3 ) ) ;
TEST_ESP_OK ( nvs_set_blob ( handle , " 1c " , blob , blob_size / 4 ) ) ;
2018-04-27 12:40:29 +05:30
// Fill second page
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs_set_blob ( handle , " 2a " , blob , blob_size / 2 ) ) ;
TEST_ESP_OK ( nvs_set_blob ( handle , " 2b " , blob , blob_size / 2 ) ) ;
2018-04-27 12:40:29 +05:30
TEST_ESP_OK ( nvs_erase_key ( handle , " 1c " ) ) ;
2020-04-27 08:51:31 +08:00
f . emu . clearStats ( ) ;
2022-10-18 15:06:10 +02:00
f . emu . failAfter ( 6 * nvs : : Page : : ENTRY_COUNT ) ;
TEST_ESP_ERR ( nvs_set_blob ( handle , " 1d " , blob , blob_size / 4 ) , ESP_ERR_FLASH_OP_FAIL ) ;
2018-04-27 12:40:29 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & f . part , 0 , 3 ) ) ;
2018-04-27 12:40:29 +05:30
2022-10-18 15:06:10 +02:00
read_size = blob_size / 3 ;
2018-04-27 12:40:29 +05:30
TEST_ESP_OK ( nvs_get_blob ( handle , " 1a " , blob , & read_size ) ) ;
TEST_ESP_OK ( nvs_get_blob ( handle , " 1b " , blob , & read_size ) ) ;
2022-10-18 15:06:10 +02:00
read_size = blob_size / 4 ;
2018-04-27 12:40:29 +05:30
TEST_ESP_ERR ( nvs_get_blob ( handle , " 1c " , blob , & read_size ) , ESP_ERR_NVS_NOT_FOUND ) ;
TEST_ESP_ERR ( nvs_get_blob ( handle , " 1d " , blob , & read_size ) , ESP_ERR_NVS_NOT_FOUND ) ;
2022-10-18 15:06:10 +02:00
read_size = blob_size / 2 ;
2018-04-27 12:40:29 +05:30
TEST_ESP_OK ( nvs_get_blob ( handle , " 2a " , blob , & read_size ) ) ;
TEST_ESP_OK ( nvs_get_blob ( handle , " 2b " , blob , & read_size ) ) ;
TEST_ESP_OK ( nvs_commit ( handle ) ) ;
nvs_close ( handle ) ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( f . part . get_partition_name ( ) ) ) ;
2018-04-27 12:40:29 +05:30
}
2018-04-02 16:14:59 +05:30
TEST_CASE ( " Check that NVS supports old blob format without blob index " , " [nvs] " )
{
SpiFlashEmulator emu ( " ../nvs_partition_generator/part_old_blob_format.bin " ) ;
2020-04-27 08:51:31 +08:00
PartitionEmulation part ( & emu , 0 , 2 * SPI_FLASH_SEC_SIZE , " test " ) ;
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2018-04-02 16:14:59 +05:30
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( & part , 0 , 2 ) ) ;
2019-11-26 14:43:30 +08:00
TEST_ESP_OK ( nvs_open_from_partition ( " test " , " dummyNamespace " , NVS_READWRITE , & handle ) ) ;
2018-11-28 10:56:06 +05:30
2018-04-02 16:14:59 +05:30
char buf [ 64 ] = { 0 } ;
size_t buflen = 64 ;
uint8_t hexdata [ ] = { 0x01 , 0x02 , 0x03 , 0xab , 0xcd , 0xef } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyHex2BinKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , hexdata , buflen ) = = 0 ) ;
buflen = 64 ;
uint8_t base64data [ ] = { ' 1 ' , ' 2 ' , ' 3 ' , ' a ' , ' b ' , ' c ' } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyBase64Key " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , base64data , buflen ) = = 0 ) ;
2022-10-18 15:06:10 +02:00
nvs : : Page p ;
2020-04-27 08:51:31 +08:00
p . load ( & part , 0 ) ;
2018-04-02 16:14:59 +05:30
/* Check that item is stored in old format without blob index*/
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( p . findItem ( 1 , nvs : : ItemType : : BLOB , " dummyHex2BinKey " ) ) ;
2018-04-02 16:14:59 +05:30
/* Modify the blob so that it is stored in the new format*/
hexdata [ 0 ] = hexdata [ 1 ] = hexdata [ 2 ] = 0x99 ;
TEST_ESP_OK ( nvs_set_blob ( handle , " dummyHex2BinKey " , hexdata , sizeof ( hexdata ) ) ) ;
2022-10-18 15:06:10 +02:00
nvs : : Page p2 ;
2020-04-27 08:51:31 +08:00
p2 . load ( & part , 0 ) ;
2018-04-02 16:14:59 +05:30
/* Check the type of the blob. Expect type mismatch since the blob is stored in new format*/
2022-10-18 15:06:10 +02:00
TEST_ESP_ERR ( p2 . findItem ( 1 , nvs : : ItemType : : BLOB , " dummyHex2BinKey " ) , ESP_ERR_NVS_TYPE_MISMATCH ) ;
2018-04-02 16:14:59 +05:30
/* Check that index is present for the modified blob according to new format*/
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( p2 . findItem ( 1 , nvs : : ItemType : : BLOB_IDX , " dummyHex2BinKey " ) ) ;
2018-04-02 16:14:59 +05:30
/* Read the blob in new format and check the contents*/
2018-11-28 10:56:06 +05:30
buflen = 64 ;
2018-04-02 16:14:59 +05:30
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyBase64Key " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , base64data , buflen ) = = 0 ) ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( part . get_partition_name ( ) ) ) ;
2018-04-02 16:14:59 +05:30
}
2020-04-27 08:51:31 +08:00
static void check_nvs_part_gen_args ( SpiFlashEmulator * spi_flash_emulator ,
2022-10-18 15:06:10 +02:00
char const * part_name ,
int size ,
char const * filename ,
bool is_encr ,
nvs_sec_cfg_t * xts_cfg )
2018-04-19 05:48:05 +05:30
{
2019-04-01 15:13:55 +01:00
nvs_handle_t handle ;
2018-11-28 10:56:06 +05:30
2020-04-27 08:51:31 +08:00
esp_partition_t esp_part ;
esp_part . encrypted = false ; // we're not testing generic flash encryption here, only the legacy NVS encryption
esp_part . address = 0 ;
esp_part . size = size * SPI_FLASH_SEC_SIZE ;
strncpy ( esp_part . label , part_name , PART_NAME_MAX_SIZE ) ;
2022-11-23 14:40:38 +01:00
unique_ptr < nvs : : Partition > part ;
2020-04-27 08:51:31 +08:00
if ( is_encr ) {
2022-11-23 14:40:38 +01:00
nvs : : NVSEncryptedPartition * enc_part = new ( std : : nothrow ) nvs : : NVSEncryptedPartition ( & esp_part ) ;
REQUIRE ( enc_part ! = nullptr ) ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( enc_part - > init ( xts_cfg ) ) ;
part . reset ( enc_part ) ;
} else {
part . reset ( new PartitionEmulation ( spi_flash_emulator , 0 , size , part_name ) ) ;
}
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( part . get ( ) , 0 , size ) ) ;
2018-11-28 10:56:06 +05:30
2018-10-17 17:08:32 +05:30
TEST_ESP_OK ( nvs_open_from_partition ( part_name , " dummyNamespace " , NVS_READONLY , & handle ) ) ;
2018-04-19 05:48:05 +05:30
uint8_t u8v ;
TEST_ESP_OK ( nvs_get_u8 ( handle , " dummyU8Key " , & u8v ) ) ;
CHECK ( u8v = = 127 ) ;
int8_t i8v ;
TEST_ESP_OK ( nvs_get_i8 ( handle , " dummyI8Key " , & i8v ) ) ;
CHECK ( i8v = = - 128 ) ;
uint16_t u16v ;
TEST_ESP_OK ( nvs_get_u16 ( handle , " dummyU16Key " , & u16v ) ) ;
CHECK ( u16v = = 32768 ) ;
uint32_t u32v ;
TEST_ESP_OK ( nvs_get_u32 ( handle , " dummyU32Key " , & u32v ) ) ;
CHECK ( u32v = = 4294967295 ) ;
int32_t i32v ;
TEST_ESP_OK ( nvs_get_i32 ( handle , " dummyI32Key " , & i32v ) ) ;
CHECK ( i32v = = - 2147483648 ) ;
2018-04-02 16:14:59 +05:30
2022-06-02 15:29:35 +02:00
char string_buf [ 256 ] ;
const char test_str [ ] = " Lorem ipsum dolor sit amet, consectetur adipiscing elit. \n "
" Fusce quis risus justo. \n "
" Suspendisse egestas in nisi sit amet auctor. \n "
" Pellentesque rhoncus dictum sodales. \n "
" In justo erat, viverra at interdum eget, interdum vel dui. " ;
size_t str_len = sizeof ( test_str ) ;
TEST_ESP_OK ( nvs_get_str ( handle , " dummyStringKey " , string_buf , & str_len ) ) ;
CHECK ( strncmp ( string_buf , test_str , str_len ) = = 0 ) ;
char buf [ 64 ] = { 0 } ;
uint8_t hexdata [ ] = { 0x01 , 0x02 , 0x03 , 0xab , 0xcd , 0xef } ;
size_t buflen = 64 ;
int j ;
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyHex2BinKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , hexdata , buflen ) = = 0 ) ;
uint8_t base64data [ ] = { ' 1 ' , ' 2 ' , ' 3 ' , ' a ' , ' b ' , ' c ' } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyBase64Key " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , base64data , buflen ) = = 0 ) ;
buflen = 64 ;
uint8_t hexfiledata [ ] = { 0x01 , 0x23 , 0x45 , 0x67 , 0x89 , 0xab , 0xcd , 0xef } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " hexFileKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , hexfiledata , buflen ) = = 0 ) ;
buflen = 64 ;
uint8_t strfiledata [ 64 ] = " abcdefghijklmnopqrstuvwxyz \0 " ;
TEST_ESP_OK ( nvs_get_str ( handle , " stringFileKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , strfiledata , buflen ) = = 0 ) ;
char bin_data [ 5200 ] ;
size_t bin_len = sizeof ( bin_data ) ;
char binfiledata [ 5200 ] ;
ifstream file ;
file . open ( filename ) ;
file . read ( binfiledata , 5200 ) ;
TEST_ESP_OK ( nvs_get_blob ( handle , " binFileKey " , bin_data , & bin_len ) ) ;
CHECK ( memcmp ( bin_data , binfiledata , bin_len ) = = 0 ) ;
file . close ( ) ;
nvs_close ( handle ) ;
TEST_ESP_OK ( nvs_flash_deinit_partition ( part_name ) ) ;
}
static void check_nvs_part_gen_args_mfg ( SpiFlashEmulator * spi_flash_emulator ,
2022-10-18 15:06:10 +02:00
char const * part_name ,
int size ,
char const * filename ,
bool is_encr ,
nvs_sec_cfg_t * xts_cfg )
2022-06-02 15:29:35 +02:00
{
nvs_handle_t handle ;
esp_partition_t esp_part ;
esp_part . encrypted = false ; // we're not testing generic flash encryption here, only the legacy NVS encryption
esp_part . address = 0 ;
esp_part . size = size * SPI_FLASH_SEC_SIZE ;
strncpy ( esp_part . label , part_name , PART_NAME_MAX_SIZE ) ;
2022-11-23 14:40:38 +01:00
unique_ptr < nvs : : Partition > part ;
2022-06-02 15:29:35 +02:00
if ( is_encr ) {
2022-11-23 14:40:38 +01:00
nvs : : NVSEncryptedPartition * enc_part = new ( std : : nothrow ) nvs : : NVSEncryptedPartition ( & esp_part ) ;
REQUIRE ( enc_part ! = nullptr ) ;
2022-06-02 15:29:35 +02:00
TEST_ESP_OK ( enc_part - > init ( xts_cfg ) ) ;
part . reset ( enc_part ) ;
} else {
part . reset ( new PartitionEmulation ( spi_flash_emulator , 0 , size , part_name ) ) ;
}
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - > init_custom ( part . get ( ) , 0 , size ) ) ;
2022-06-02 15:29:35 +02:00
TEST_ESP_OK ( nvs_open_from_partition ( part_name , " dummyNamespace " , NVS_READONLY , & handle ) ) ;
uint8_t u8v ;
TEST_ESP_OK ( nvs_get_u8 ( handle , " dummyU8Key " , & u8v ) ) ;
CHECK ( u8v = = 127 ) ;
int8_t i8v ;
TEST_ESP_OK ( nvs_get_i8 ( handle , " dummyI8Key " , & i8v ) ) ;
CHECK ( i8v = = - 128 ) ;
uint16_t u16v ;
TEST_ESP_OK ( nvs_get_u16 ( handle , " dummyU16Key " , & u16v ) ) ;
CHECK ( u16v = = 32768 ) ;
uint32_t u32v ;
TEST_ESP_OK ( nvs_get_u32 ( handle , " dummyU32Key " , & u32v ) ) ;
CHECK ( u32v = = 4294967295 ) ;
int32_t i32v ;
TEST_ESP_OK ( nvs_get_i32 ( handle , " dummyI32Key " , & i32v ) ) ;
CHECK ( i32v = = - 2147483648 ) ;
2018-04-19 05:48:05 +05:30
char buf [ 64 ] = { 0 } ;
size_t buflen = 64 ;
TEST_ESP_OK ( nvs_get_str ( handle , " dummyStringKey " , buf , & buflen ) ) ;
CHECK ( strncmp ( buf , " 0A:0B:0C:0D:0E:0F " , buflen ) = = 0 ) ;
2018-04-02 16:14:59 +05:30
2018-04-19 05:48:05 +05:30
uint8_t hexdata [ ] = { 0x01 , 0x02 , 0x03 , 0xab , 0xcd , 0xef } ;
2018-04-02 16:14:59 +05:30
buflen = 64 ;
int j ;
2018-04-19 05:48:05 +05:30
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyHex2BinKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , hexdata , buflen ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
2018-06-14 16:27:37 +05:30
uint8_t base64data [ ] = { ' 1 ' , ' 2 ' , ' 3 ' , ' a ' , ' b ' , ' c ' } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " dummyBase64Key " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , base64data , buflen ) = = 0 ) ;
2018-04-02 16:14:59 +05:30
buflen = 64 ;
uint8_t hexfiledata [ ] = { 0x01 , 0x23 , 0x45 , 0x67 , 0x89 , 0xab , 0xcd , 0xef } ;
TEST_ESP_OK ( nvs_get_blob ( handle , " hexFileKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , hexfiledata , buflen ) = = 0 ) ;
buflen = 64 ;
uint8_t strfiledata [ 64 ] = " abcdefghijklmnopqrstuvwxyz \0 " ;
TEST_ESP_OK ( nvs_get_str ( handle , " stringFileKey " , buf , & buflen ) ) ;
CHECK ( memcmp ( buf , strfiledata , buflen ) = = 0 ) ;
char bin_data [ 5200 ] ;
size_t bin_len = sizeof ( bin_data ) ;
char binfiledata [ 5200 ] ;
ifstream file ;
2018-10-17 17:08:32 +05:30
file . open ( filename ) ;
2018-04-02 16:14:59 +05:30
file . read ( binfiledata , 5200 ) ;
TEST_ESP_OK ( nvs_get_blob ( handle , " binFileKey " , bin_data , & bin_len ) ) ;
CHECK ( memcmp ( bin_data , binfiledata , bin_len ) = = 0 ) ;
file . close ( ) ;
2018-11-28 10:56:06 +05:30
2018-07-02 16:40:43 +05:30
nvs_close ( handle ) ;
2020-04-27 08:51:31 +08:00
TEST_ESP_OK ( nvs_flash_deinit_partition ( part_name ) ) ;
2018-04-19 05:48:05 +05:30
}
2018-10-17 17:08:32 +05:30
TEST_CASE ( " check and read data from partition generated via partition generation utility with multipage blob support disabled " , " [nvs_part_gen] " )
2018-09-12 17:47:24 +05:30
{
2018-11-28 10:56:06 +05:30
int status ;
2018-09-12 17:47:24 +05:30
int childpid = fork ( ) ;
if ( childpid = = 0 ) {
2018-11-28 10:56:06 +05:30
exit ( execlp ( " cp " , " cp " ,
" -rf " ,
" ../nvs_partition_generator/testdata " ,
2022-10-18 15:06:10 +02:00
" . " , NULL ) ) ;
2018-09-12 17:47:24 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
CHECK ( WEXITSTATUS ( status ) ! = - 1 ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/nvs_partition_gen.py " ,
" generate " ,
" ../nvs_partition_generator/sample_singlepage_blob.csv " ,
" partition_single_page.bin " ,
" 0x3000 " ,
" --version " ,
" 1 " ,
" --outdir " ,
" ../nvs_partition_generator " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
int status ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-09-12 17:47:24 +05:30
}
2018-10-17 17:08:32 +05:30
SpiFlashEmulator emu ( " ../nvs_partition_generator/partition_single_page.bin " ) ;
2018-11-28 10:56:06 +05:30
2020-04-27 08:51:31 +08:00
check_nvs_part_gen_args ( & emu , " test " , 3 , " ../nvs_partition_generator/testdata/sample_singlepage_blob.bin " , false , NULL ) ;
2018-09-12 17:47:24 +05:30
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
2018-11-28 10:56:06 +05:30
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
" -rf " ,
2022-10-18 15:06:10 +02:00
" testdata " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
}
2018-09-12 17:47:24 +05:30
2018-10-17 17:08:32 +05:30
TEST_CASE ( " check and read data from partition generated via partition generation utility with multipage blob support enabled " , " [nvs_part_gen] " )
{
2018-11-28 10:56:06 +05:30
int status ;
2018-10-17 17:08:32 +05:30
int childpid = fork ( ) ;
if ( childpid = = 0 ) {
2018-11-28 10:56:06 +05:30
exit ( execlp ( " cp " , " cp " ,
" -rf " ,
" ../nvs_partition_generator/testdata " ,
2022-10-18 15:06:10 +02:00
" . " , NULL ) ) ;
2018-10-17 17:08:32 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/nvs_partition_gen.py " ,
" generate " ,
" ../nvs_partition_generator/sample_multipage_blob.csv " ,
" partition_multipage_blob.bin " ,
" 0x4000 " ,
" --version " ,
" 2 " ,
" --outdir " ,
" ../nvs_partition_generator " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-10-17 17:08:32 +05:30
}
2018-09-12 17:47:24 +05:30
2018-10-17 17:08:32 +05:30
SpiFlashEmulator emu ( " ../nvs_partition_generator/partition_multipage_blob.bin " ) ;
2018-11-28 10:56:06 +05:30
2022-10-18 15:06:10 +02:00
check_nvs_part_gen_args ( & emu , " test " , 4 , " ../nvs_partition_generator/testdata/sample_multipage_blob.bin " , false , NULL ) ;
2018-09-12 17:47:24 +05:30
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
2018-11-28 10:56:06 +05:30
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
" -rf " ,
2022-10-18 15:06:10 +02:00
" testdata " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-09-12 17:47:24 +05:30
}
2018-10-29 13:19:20 +05:30
TEST_CASE ( " check and read data from partition generated via manufacturing utility with multipage blob support disabled " , " [mfg_gen] " )
{
int childpid = fork ( ) ;
int status ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
2019-06-21 20:31:22 +08:00
" rm -rf ../../../tools/mass_mfg/host_test && \
cp - rf . . / . . / . . / tools / mass_mfg / testdata mfg_testdata & & \
cp - rf . . / nvs_partition_generator / testdata . & & \
mkdir - p . . / . . / . . / tools / mass_mfg / host_test " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/samples/sample_config.csv " ,
" ../../../tools/mass_mfg/samples/sample_values_singlepage_blob.csv " ,
" Test " ,
" 0x3000 " ,
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 1 " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2022-06-30 17:53:12 +02:00
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
" generate " ,
" ../../../tools/mass_mfg/host_test/csv/Test-1.csv " ,
" ../nvs_partition_generator/Test-1-partition.bin " ,
" 0x3000 " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 1 " , NULL ) ) ;
2022-06-30 17:53:12 +02:00
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
}
}
}
SpiFlashEmulator emu1 ( " ../../../tools/mass_mfg/host_test/bin/Test-1.bin " ) ;
check_nvs_part_gen_args_mfg ( & emu1 , " test " , 3 , " mfg_testdata/sample_singlepage_blob.bin " , false , NULL ) ;
SpiFlashEmulator emu2 ( " ../nvs_partition_generator/Test-1-partition.bin " ) ;
check_nvs_part_gen_args_mfg ( & emu2 , " test " , 3 , " testdata/sample_singlepage_blob.bin " , false , NULL ) ;
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
rm - rf mfg_testdata | \
2022-10-18 15:06:10 +02:00
rm - rf testdata " , NULL));
2022-06-30 17:53:12 +02:00
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
}
}
TEST_CASE ( " check and read data from partition generated via manufacturing utility with blank lines in csv files and multipage blob support disabled " , " [mfg_gen] " )
{
int childpid = fork ( ) ;
int status ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test && \
cp - rf . . / . . / . . / tools / mass_mfg / testdata mfg_testdata & & \
cp - rf . . / nvs_partition_generator / testdata . & & \
mkdir - p . . / . . / . . / tools / mass_mfg / host_test " , NULL));
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
" generate " ,
" ../../../tools/mass_mfg/samples/sample_config_blank_lines.csv " ,
" ../../../tools/mass_mfg/samples/sample_values_singlepage_blob_blank_lines.csv " ,
" Test " ,
" 0x3000 " ,
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 1 " , NULL ) ) ;
2022-06-30 17:53:12 +02:00
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/host_test/csv/Test-1.csv " ,
" ../nvs_partition_generator/Test-1-partition.bin " ,
" 0x3000 " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 1 " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
}
SpiFlashEmulator emu1 ( " ../../../tools/mass_mfg/host_test/bin/Test-1.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu1 , " test " , 3 , " mfg_testdata/sample_singlepage_blob.bin " , false , NULL ) ;
2018-10-29 13:19:20 +05:30
SpiFlashEmulator emu2 ( " ../nvs_partition_generator/Test-1-partition.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu2 , " test " , 3 , " testdata/sample_singlepage_blob.bin " , false , NULL ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
rm - rf mfg_testdata | \
2022-10-18 15:06:10 +02:00
rm - rf testdata " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
TEST_CASE ( " check and read data from partition generated via manufacturing utility with multipage blob support enabled " , " [mfg_gen] " )
{
int childpid = fork ( ) ;
int status ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
cp - rf . . / . . / . . / tools / mass_mfg / testdata mfg_testdata | \
cp - rf . . / nvs_partition_generator / testdata . | \
2022-10-18 15:06:10 +02:00
mkdir - p . . / . . / . . / tools / mass_mfg / host_test " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/samples/sample_config.csv " ,
" ../../../tools/mass_mfg/samples/sample_values_multipage_blob.csv " ,
" Test " ,
" 0x4000 " ,
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 2 " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/host_test/csv/Test-1.csv " ,
" ../nvs_partition_generator/Test-1-partition.bin " ,
" 0x4000 " ,
" --version " ,
2022-10-18 15:06:10 +02:00
" 2 " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
}
SpiFlashEmulator emu1 ( " ../../../tools/mass_mfg/host_test/bin/Test-1.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu1 , " test " , 4 , " mfg_testdata/sample_multipage_blob.bin " , false , NULL ) ;
2018-10-29 13:19:20 +05:30
SpiFlashEmulator emu2 ( " ../nvs_partition_generator/Test-1-partition.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu2 , " test " , 4 , " testdata/sample_multipage_blob.bin " , false , NULL ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
rm - rf mfg_testdata | \
2022-10-18 15:06:10 +02:00
rm - rf testdata " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
2018-07-02 16:40:43 +05:30
# if CONFIG_NVS_ENCRYPTION
TEST_CASE ( " check underlying xts code for 32-byte size sector encryption " , " [nvs] " )
{
auto toHex = [ ] ( char ch ) {
if ( ch > = ' 0 ' & & ch < = ' 9 ' )
return ch - ' 0 ' ;
else if ( ch > = ' a ' & & ch < = ' f ' )
return ch - ' a ' + 10 ;
else if ( ch > = ' A ' & & ch < = ' F ' )
return ch - ' A ' + 10 ;
else
return 0 ;
} ;
auto toHexByte = [ toHex ] ( char * c ) {
return 16 * toHex ( c [ 0 ] ) + toHex ( c [ 1 ] ) ;
} ;
auto toHexStream = [ toHexByte ] ( char * src , uint8_t * dest ) {
uint32_t cnt = 0 ;
char * p = src ;
while ( * p ! = ' \0 ' & & * ( p + 1 ) ! = ' \0 ' )
{
dest [ cnt + + ] = toHexByte ( p ) ; p + = 2 ;
}
} ;
uint8_t eky_hex [ 2 * NVS_KEY_SIZE ] ;
2022-10-18 15:06:10 +02:00
uint8_t ptxt_hex [ nvs : : Page : : ENTRY_SIZE ] , ctxt_hex [ nvs : : Page : : ENTRY_SIZE ] , ba_hex [ 16 ] ;
2018-07-02 16:40:43 +05:30
mbedtls_aes_xts_context ectx [ 1 ] ;
mbedtls_aes_xts_context dctx [ 1 ] ;
char eky [ ] [ 2 * NVS_KEY_SIZE + 1 ] = {
" 0000000000000000000000000000000000000000000000000000000000000000 " ,
" 1111111111111111111111111111111111111111111111111111111111111111 "
} ;
char tky [ ] [ 2 * NVS_KEY_SIZE + 1 ] = {
" 0000000000000000000000000000000000000000000000000000000000000000 " ,
" 2222222222222222222222222222222222222222222222222222222222222222 "
} ;
char blk_addr [ ] [ 2 * 16 + 1 ] = {
" 00000000000000000000000000000000 " ,
" 33333333330000000000000000000000 "
} ;
2022-10-18 15:06:10 +02:00
char ptxt [ ] [ 2 * nvs : : Page : : ENTRY_SIZE + 1 ] = {
2018-07-02 16:40:43 +05:30
" 0000000000000000000000000000000000000000000000000000000000000000 " ,
" 4444444444444444444444444444444444444444444444444444444444444444 "
} ;
2022-10-18 15:06:10 +02:00
char ctxt [ ] [ 2 * nvs : : Page : : ENTRY_SIZE + 1 ] = {
2018-07-02 16:40:43 +05:30
" d456b4fc2e620bba6ffbed27b956c9543454dd49ebd8d8ee6f94b65cbe158f73 " ,
" e622334f184bbce129a25b2ac76b3d92abf98e22df5bdd15af471f3db8946a85 "
} ;
mbedtls_aes_xts_init ( ectx ) ;
mbedtls_aes_xts_init ( dctx ) ;
for ( uint8_t cnt = 0 ; cnt < sizeof ( eky ) / sizeof ( eky [ 0 ] ) ; cnt + + ) {
toHexStream ( eky [ cnt ] , eky_hex ) ;
toHexStream ( tky [ cnt ] , & eky_hex [ NVS_KEY_SIZE ] ) ;
toHexStream ( ptxt [ cnt ] , ptxt_hex ) ;
toHexStream ( ctxt [ cnt ] , ctxt_hex ) ;
toHexStream ( blk_addr [ cnt ] , ba_hex ) ;
CHECK ( ! mbedtls_aes_xts_setkey_enc ( ectx , eky_hex , 2 * NVS_KEY_SIZE * 8 ) ) ;
CHECK ( ! mbedtls_aes_xts_setkey_enc ( dctx , eky_hex , 2 * NVS_KEY_SIZE * 8 ) ) ;
2022-10-18 15:06:10 +02:00
CHECK ( ! mbedtls_aes_crypt_xts ( ectx , MBEDTLS_AES_ENCRYPT , nvs : : Page : : ENTRY_SIZE , ba_hex , ptxt_hex , ptxt_hex ) ) ;
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
CHECK ( ! memcmp ( ptxt_hex , ctxt_hex , nvs : : Page : : ENTRY_SIZE ) ) ;
2018-07-02 16:40:43 +05:30
}
}
TEST_CASE ( " test nvs apis with encryption enabled " , " [nvs] " )
{
2019-04-01 15:13:55 +01:00
nvs_handle_t handle_1 ;
2018-07-02 16:40:43 +05:30
const uint32_t NVS_FLASH_SECTOR = 6 ;
const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3 ;
nvs_sec_cfg_t xts_cfg ;
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
xts_cfg . eky [ count ] = 0x11 ;
xts_cfg . tky [ count ] = 0x22 ;
}
2020-04-27 08:51:31 +08:00
EncryptedPartitionFixture fixture ( & xts_cfg , NVS_FLASH_SECTOR , NVS_FLASH_SECTOR_COUNT_MIN ) ;
fixture . emu . randomize ( 100 ) ;
fixture . emu . setBounds ( NVS_FLASH_SECTOR , NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ) ;
2018-07-02 16:40:43 +05:30
for ( uint16_t i = NVS_FLASH_SECTOR ; i < NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN ; + + i ) {
2020-04-27 08:51:31 +08:00
fixture . emu . erase ( i ) ;
2018-07-02 16:40:43 +05:30
}
2022-10-18 15:06:10 +02:00
TEST_ESP_OK ( nvs : : NVSPartitionManager : : get_instance ( ) - >
2020-04-27 08:51:31 +08:00
init_custom ( & fixture . part , NVS_FLASH_SECTOR , NVS_FLASH_SECTOR_COUNT_MIN ) ) ;
2018-07-02 16:40:43 +05:30
TEST_ESP_ERR ( nvs_open ( " namespace1 " , NVS_READONLY , & handle_1 ) , ESP_ERR_NVS_NOT_FOUND ) ;
TEST_ESP_OK ( nvs_open ( " namespace1 " , NVS_READWRITE , & handle_1 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_1 , " foo " , 0x12345678 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_1 , " foo " , 0x23456789 ) ) ;
2019-04-01 15:13:55 +01:00
nvs_handle_t handle_2 ;
2018-07-02 16:40:43 +05:30
TEST_ESP_OK ( nvs_open ( " namespace2 " , NVS_READWRITE , & handle_2 ) ) ;
TEST_ESP_OK ( nvs_set_i32 ( handle_2 , " foo " , 0x3456789a ) ) ;
const char * str = " value 0123456789abcdef0123456789abcdef " ;
TEST_ESP_OK ( nvs_set_str ( handle_2 , " key " , str ) ) ;
int32_t v1 ;
TEST_ESP_OK ( nvs_get_i32 ( handle_1 , " foo " , & v1 ) ) ;
CHECK ( 0x23456789 = = v1 ) ;
int32_t v2 ;
TEST_ESP_OK ( nvs_get_i32 ( handle_2 , " foo " , & v2 ) ) ;
CHECK ( 0x3456789a = = v2 ) ;
char buf [ strlen ( str ) + 1 ] ;
size_t buf_len = sizeof ( buf ) ;
size_t buf_len_needed ;
TEST_ESP_OK ( nvs_get_str ( handle_2 , " key " , NULL , & buf_len_needed ) ) ;
CHECK ( buf_len_needed = = buf_len ) ;
size_t buf_len_short = buf_len - 1 ;
TEST_ESP_ERR ( ESP_ERR_NVS_INVALID_LENGTH , nvs_get_str ( handle_2 , " key " , buf , & buf_len_short ) ) ;
CHECK ( buf_len_short = = buf_len ) ;
size_t buf_len_long = buf_len + 1 ;
TEST_ESP_OK ( nvs_get_str ( handle_2 , " key " , buf , & buf_len_long ) ) ;
CHECK ( buf_len_long = = buf_len ) ;
TEST_ESP_OK ( nvs_get_str ( handle_2 , " key " , buf , & buf_len ) ) ;
CHECK ( 0 = = strcmp ( buf , str ) ) ;
nvs_close ( handle_1 ) ;
nvs_close ( handle_2 ) ;
TEST_ESP_OK ( nvs_flash_deinit ( ) ) ;
}
TEST_CASE ( " test nvs apis for nvs partition generator utility with encryption enabled " , " [nvs_part_gen] " )
{
2018-11-28 10:56:06 +05:30
int status ;
2018-07-02 16:40:43 +05:30
int childpid = fork ( ) ;
if ( childpid = = 0 ) {
2018-11-28 10:56:06 +05:30
exit ( execlp ( " cp " , " cp " ,
" -rf " ,
" ../nvs_partition_generator/testdata " ,
2022-10-18 15:06:10 +02:00
" . " , NULL ) ) ;
2018-07-02 16:40:43 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/nvs_partition_gen.py " ,
" encrypt " ,
" ../nvs_partition_generator/sample_multipage_blob.csv " ,
" partition_encrypted.bin " ,
" 0x4000 " ,
" --inputkey " ,
" ../nvs_partition_generator/testdata/sample_encryption_keys.bin " ,
" --outdir " ,
" ../nvs_partition_generator " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-07-02 16:40:43 +05:30
}
SpiFlashEmulator emu ( " ../nvs_partition_generator/partition_encrypted.bin " ) ;
2018-11-28 10:56:06 +05:30
2018-10-17 17:08:32 +05:30
nvs_sec_cfg_t cfg ;
2022-10-18 15:06:10 +02:00
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
2018-10-17 17:08:32 +05:30
cfg . eky [ count ] = 0x11 ;
cfg . tky [ count ] = 0x22 ;
}
2018-07-02 16:40:43 +05:30
2020-04-27 08:51:31 +08:00
check_nvs_part_gen_args ( & emu , NVS_DEFAULT_PART_NAME , 4 , " ../nvs_partition_generator/testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
" -rf " ,
2022-10-18 15:06:10 +02:00
" testdata " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-10-17 17:08:32 +05:30
}
2020-07-20 10:19:12 +02:00
TEST_CASE ( " test decrypt functionality for encrypted data " , " [nvs_part_gen] " )
{
//retrieving the temporary test data
int status = system ( " cp -rf ../nvs_partition_generator/testdata . " ) ;
CHECK ( status = = 0 ) ;
//encoding data from sample_multipage_blob.csv
status = system ( " python ../nvs_partition_generator/nvs_partition_gen.py generate ../nvs_partition_generator/sample_multipage_blob.csv partition_encoded.bin 0x5000 --outdir ../nvs_partition_generator " ) ;
CHECK ( status = = 0 ) ;
//encrypting data from sample_multipage_blob.csv
status = system ( " python ../nvs_partition_generator/nvs_partition_gen.py encrypt ../nvs_partition_generator/sample_multipage_blob.csv partition_encrypted.bin 0x5000 --inputkey ../nvs_partition_generator/testdata/sample_encryption_keys.bin --outdir ../nvs_partition_generator " ) ;
CHECK ( status = = 0 ) ;
//decrypting data from partition_encrypted.bin
status = system ( " python ../nvs_partition_generator/nvs_partition_gen.py decrypt ../nvs_partition_generator/partition_encrypted.bin ../nvs_partition_generator/testdata/sample_encryption_keys.bin ../nvs_partition_generator/partition_decrypted.bin " ) ;
CHECK ( status = = 0 ) ;
status = system ( " diff ../nvs_partition_generator/partition_decrypted.bin ../nvs_partition_generator/partition_encoded.bin " ) ;
CHECK ( status = = 0 ) ;
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
//cleaning up the temporary test data
status = system ( " rm -rf testdata " ) ;
CHECK ( status = = 0 ) ;
}
2018-10-17 17:08:32 +05:30
TEST_CASE ( " test nvs apis for nvs partition generator utility with encryption enabled using keygen " , " [nvs_part_gen] " )
{
int childpid = fork ( ) ;
2018-11-21 13:40:24 +05:30
int status ;
2018-10-17 17:08:32 +05:30
2018-11-28 10:56:06 +05:30
if ( childpid = = 0 ) {
exit ( execlp ( " cp " , " cp " ,
" -rf " ,
" ../nvs_partition_generator/testdata " ,
2022-10-18 15:06:10 +02:00
" . " , NULL ) ) ;
2018-10-17 17:08:32 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
" -rf " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/keys " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
2019-05-07 15:36:02 +05:30
" encrypt " ,
2018-11-28 10:56:06 +05:30
" ../nvs_partition_generator/sample_multipage_blob.csv " ,
2019-05-07 15:36:02 +05:30
" partition_encrypted_using_keygen.bin " ,
2018-11-28 10:56:06 +05:30
" 0x4000 " ,
" --keygen " ,
2019-05-07 15:36:02 +05:30
" --outdir " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
}
2018-10-17 17:08:32 +05:30
}
2018-11-28 10:56:06 +05:30
DIR * dir ;
struct dirent * file ;
char * filename ;
char * files ;
char * file_ext ;
2019-05-07 15:36:02 +05:30
dir = opendir ( " ../nvs_partition_generator/keys " ) ;
2022-10-18 15:06:10 +02:00
while ( ( file = readdir ( dir ) ) ! = NULL ) {
2018-11-28 10:56:06 +05:30
filename = file - > d_name ;
files = strrchr ( filename , ' . ' ) ;
2022-10-18 15:06:10 +02:00
if ( files ! = NULL ) {
file_ext = files + 1 ;
if ( strncmp ( file_ext , " bin " , 3 ) = = 0 ) {
2018-11-28 10:56:06 +05:30
break ;
}
}
}
2019-05-07 15:36:02 +05:30
std : : string encr_file = std : : string ( " ../nvs_partition_generator/keys/ " ) + std : : string ( filename ) ;
2018-10-17 17:08:32 +05:30
SpiFlashEmulator emu ( " ../nvs_partition_generator/partition_encrypted_using_keygen.bin " ) ;
2018-11-28 10:56:06 +05:30
2018-10-17 17:08:32 +05:30
char buffer [ 64 ] ;
FILE * fp ;
2022-10-18 15:06:10 +02:00
fp = fopen ( encr_file . c_str ( ) , " rb " ) ;
fread ( buffer , sizeof ( buffer ) , 1 , fp ) ;
2018-10-17 17:08:32 +05:30
fclose ( fp ) ;
nvs_sec_cfg_t cfg ;
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
2018-10-17 17:08:32 +05:30
cfg . eky [ count ] = buffer [ count ] & 255 ;
2022-10-18 15:06:10 +02:00
cfg . tky [ count ] = buffer [ count + 32 ] & 255 ;
2018-07-02 16:40:43 +05:30
}
2020-04-27 08:51:31 +08:00
check_nvs_part_gen_args ( & emu , NVS_DEFAULT_PART_NAME , 4 , " ../nvs_partition_generator/testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
}
2018-07-02 16:40:43 +05:30
2019-05-07 15:36:02 +05:30
TEST_CASE ( " test nvs apis for nvs partition generator utility with encryption enabled using inputkey " , " [nvs_part_gen] " )
2018-10-17 17:08:32 +05:30
{
int childpid = fork ( ) ;
2018-11-21 13:40:24 +05:30
int status ;
2018-11-28 10:56:06 +05:30
DIR * dir ;
struct dirent * file ;
char * filename ;
char * files ;
char * file_ext ;
2019-05-07 15:36:02 +05:30
dir = opendir ( " ../nvs_partition_generator/keys " ) ;
2022-10-18 15:06:10 +02:00
while ( ( file = readdir ( dir ) ) ! = NULL ) {
2018-11-28 10:56:06 +05:30
filename = file - > d_name ;
files = strrchr ( filename , ' . ' ) ;
2022-10-18 15:06:10 +02:00
if ( files ! = NULL ) {
file_ext = files + 1 ;
if ( strncmp ( file_ext , " bin " , 3 ) = = 0 ) {
2018-11-28 10:56:06 +05:30
break ;
}
}
}
2019-05-07 15:36:02 +05:30
std : : string encr_file = std : : string ( " ../nvs_partition_generator/keys/ " ) + std : : string ( filename ) ;
2018-11-28 10:56:06 +05:30
2022-10-18 15:06:10 +02:00
if ( childpid = = 0 ) {
2018-10-17 17:08:32 +05:30
exit ( execlp ( " python " , " python " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/nvs_partition_gen.py " ,
" encrypt " ,
" ../nvs_partition_generator/sample_multipage_blob.csv " ,
" partition_encrypted_using_keyfile.bin " ,
" 0x4000 " ,
" --inputkey " ,
encr_file . c_str ( ) ,
" --outdir " ,
" ../nvs_partition_generator " , NULL ) ) ;
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-17 17:08:32 +05:30
}
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
SpiFlashEmulator emu ( " ../nvs_partition_generator/partition_encrypted_using_keyfile.bin " ) ;
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
char buffer [ 64 ] ;
FILE * fp ;
2018-07-02 16:40:43 +05:30
2022-10-18 15:06:10 +02:00
fp = fopen ( encr_file . c_str ( ) , " rb " ) ;
fread ( buffer , sizeof ( buffer ) , 1 , fp ) ;
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
fclose ( fp ) ;
2018-07-02 16:40:43 +05:30
2018-10-17 17:08:32 +05:30
nvs_sec_cfg_t cfg ;
2022-10-18 15:06:10 +02:00
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
2018-10-17 17:08:32 +05:30
cfg . eky [ count ] = buffer [ count ] & 255 ;
2022-10-18 15:06:10 +02:00
cfg . tky [ count ] = buffer [ count + 32 ] & 255 ;
2018-10-17 17:08:32 +05:30
}
2020-04-27 08:51:31 +08:00
check_nvs_part_gen_args ( & emu , NVS_DEFAULT_PART_NAME , 4 , " ../nvs_partition_generator/testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-07-02 16:40:43 +05:30
2018-11-21 13:40:24 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
2018-11-28 10:56:06 +05:30
" -rf " ,
2022-10-18 15:06:10 +02:00
" ../nvs_partition_generator/keys " , NULL ) ) ;
2018-11-21 13:40:24 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-21 13:40:24 +05:30
2018-11-28 10:56:06 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " rm " , " rm " ,
" -rf " ,
2022-10-18 15:06:10 +02:00
" testdata " , NULL ) ) ;
2018-11-28 10:56:06 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-11-28 10:56:06 +05:30
}
2018-11-21 13:40:24 +05:30
}
2018-10-29 13:19:20 +05:30
}
2019-06-20 18:47:59 +05:30
TEST_CASE ( " check and read data from partition generated via manufacturing utility with encryption enabled using sample inputkey " , " [mfg_gen] " )
2018-10-29 13:19:20 +05:30
{
int childpid = fork ( ) ;
int status ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
cp - rf . . / . . / . . / tools / mass_mfg / testdata mfg_testdata | \
cp - rf . . / nvs_partition_generator / testdata . | \
2022-10-18 15:06:10 +02:00
mkdir - p . . / . . / . . / tools / mass_mfg / host_test " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/samples/sample_config.csv " ,
" ../../../tools/mass_mfg/samples/sample_values_multipage_blob.csv " ,
" Test " ,
" 0x4000 " ,
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --version " ,
2019-06-20 18:47:59 +05:30
" 2 " ,
" --inputkey " ,
2022-10-18 15:06:10 +02:00
" mfg_testdata/sample_encryption_keys.bin " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
2019-06-20 18:47:59 +05:30
" encrypt " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/host_test/csv/Test-1.csv " ,
" ../nvs_partition_generator/Test-1-partition-encrypted.bin " ,
" 0x4000 " ,
" --version " ,
2019-06-20 18:47:59 +05:30
" 2 " ,
" --inputkey " ,
2022-10-18 15:06:10 +02:00
" testdata/sample_encryption_keys.bin " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
}
SpiFlashEmulator emu1 ( " ../../../tools/mass_mfg/host_test/bin/Test-1.bin " ) ;
nvs_sec_cfg_t cfg ;
2022-10-18 15:06:10 +02:00
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
2018-10-29 13:19:20 +05:30
cfg . eky [ count ] = 0x11 ;
cfg . tky [ count ] = 0x22 ;
}
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu1 , NVS_DEFAULT_PART_NAME , 4 , " mfg_testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-10-29 13:19:20 +05:30
SpiFlashEmulator emu2 ( " ../nvs_partition_generator/Test-1-partition-encrypted.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu2 , NVS_DEFAULT_PART_NAME , 4 , " testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
rm - rf mfg_testdata | \
2022-10-18 15:06:10 +02:00
rm - rf testdata " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
TEST_CASE ( " check and read data from partition generated via manufacturing utility with encryption enabled using new generated key " , " [mfg_gen] " )
{
int childpid = fork ( ) ;
int status ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf ../../../tools/mass_mfg/host_test | \
cp - rf . . / . . / . . / tools / mass_mfg / testdata mfg_testdata | \
cp - rf . . / nvs_partition_generator / testdata . | \
2022-10-18 15:06:10 +02:00
mkdir - p . . / . . / . . / tools / mass_mfg / host_test " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate-key " ,
2018-10-29 13:19:20 +05:30
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --keyfile " ,
2022-10-18 15:06:10 +02:00
" encr_keys_host_test.bin " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../../../tools/mass_mfg/mfg_gen.py " ,
2019-06-20 18:47:59 +05:30
" generate " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/samples/sample_config.csv " ,
" ../../../tools/mass_mfg/samples/sample_values_multipage_blob.csv " ,
" Test " ,
" 0x4000 " ,
" --outdir " ,
" ../../../tools/mass_mfg/host_test " ,
" --version " ,
2019-06-20 18:47:59 +05:30
" 2 " ,
" --inputkey " ,
2022-10-18 15:06:10 +02:00
" ../../../tools/mass_mfg/host_test/keys/encr_keys_host_test.bin " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " python " , " python " ,
" ../nvs_partition_generator/nvs_partition_gen.py " ,
2019-06-20 18:47:59 +05:30
" encrypt " ,
2018-10-29 13:19:20 +05:30
" ../../../tools/mass_mfg/host_test/csv/Test-1.csv " ,
" ../nvs_partition_generator/Test-1-partition-encrypted.bin " ,
" 0x4000 " ,
" --version " ,
2019-06-20 18:47:59 +05:30
" 2 " ,
" --inputkey " ,
2022-10-18 15:06:10 +02:00
" ../../../tools/mass_mfg/host_test/keys/encr_keys_host_test.bin " , NULL ) ) ;
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
}
}
}
SpiFlashEmulator emu1 ( " ../../../tools/mass_mfg/host_test/bin/Test-1.bin " ) ;
char buffer [ 64 ] ;
FILE * fp ;
2022-10-18 15:06:10 +02:00
fp = fopen ( " ../../../tools/mass_mfg/host_test/keys/encr_keys_host_test.bin " , " rb " ) ;
fread ( buffer , sizeof ( buffer ) , 1 , fp ) ;
2018-10-29 13:19:20 +05:30
fclose ( fp ) ;
nvs_sec_cfg_t cfg ;
2022-10-18 15:06:10 +02:00
for ( int count = 0 ; count < NVS_KEY_SIZE ; count + + ) {
2018-10-29 13:19:20 +05:30
cfg . eky [ count ] = buffer [ count ] & 255 ;
2022-10-18 15:06:10 +02:00
cfg . tky [ count ] = buffer [ count + 32 ] & 255 ;
2018-10-29 13:19:20 +05:30
}
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu1 , NVS_DEFAULT_PART_NAME , 4 , " mfg_testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-10-29 13:19:20 +05:30
SpiFlashEmulator emu2 ( " ../nvs_partition_generator/Test-1-partition-encrypted.bin " ) ;
2022-06-02 15:29:35 +02:00
check_nvs_part_gen_args_mfg ( & emu2 , NVS_DEFAULT_PART_NAME , 4 , " testdata/sample_multipage_blob.bin " , true , & cfg ) ;
2018-10-29 13:19:20 +05:30
childpid = fork ( ) ;
if ( childpid = = 0 ) {
exit ( execlp ( " bash " , " bash " ,
" -c " ,
" rm -rf keys | \
rm - rf mfg_testdata | \
rm - rf testdata | \
2022-10-18 15:06:10 +02:00
rm - rf . . / . . / . . / tools / mass_mfg / host_test " , NULL));
2018-10-29 13:19:20 +05:30
} else {
CHECK ( childpid > 0 ) ;
waitpid ( childpid , & status , 0 ) ;
2020-11-11 10:35:24 +11:00
CHECK ( WEXITSTATUS ( status ) = = 0 ) ;
2018-10-29 13:19:20 +05:30
}
2018-07-02 16:40:43 +05:30
}
# endif
/* Add new tests above */
/* This test has to be the final one */
2018-09-12 17:47:24 +05:30
2018-04-16 12:50:57 +08:00
TEST_CASE ( " dump all performance data " , " [nvs] " )
{
std : : cout < < " ==================== " < < std : : endl < < " Dumping benchmarks " < < std : : endl ;
std : : cout < < s_perf . str ( ) < < std : : endl ;
std : : cout < < " ==================== " < < std : : endl ;
}