2016-11-17 16:36:10 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include "unity.h"
|
|
|
|
#include "nvs.h"
|
|
|
|
#include "nvs_flash.h"
|
2017-03-14 21:39:44 +08:00
|
|
|
#include "esp_partition.h"
|
2018-07-02 16:40:43 +05:30
|
|
|
#include "esp_flash_encrypt.h"
|
2017-03-14 21:39:44 +08:00
|
|
|
#include "esp_log.h"
|
2016-11-17 16:36:10 +08:00
|
|
|
#include <string.h>
|
2018-08-09 10:35:19 +03:00
|
|
|
#include "esp_system.h"
|
2016-11-17 16:36:10 +08:00
|
|
|
|
2018-07-02 16:40:43 +05:30
|
|
|
#ifdef CONFIG_NVS_ENCRYPTION
|
|
|
|
#include "mbedtls/aes.h"
|
|
|
|
#endif
|
|
|
|
|
2017-03-14 21:39:44 +08:00
|
|
|
static const char* TAG = "test_nvs";
|
2016-11-17 16:36:10 +08:00
|
|
|
|
2020-04-27 08:51:31 +08:00
|
|
|
TEST_CASE("Partition name no longer than 16 characters", "[nvs]")
|
|
|
|
{
|
|
|
|
const char *TOO_LONG_NAME = "0123456789abcdefg";
|
|
|
|
|
|
|
|
TEST_ESP_ERR(ESP_ERR_INVALID_ARG, nvs_flash_init_partition(TOO_LONG_NAME));
|
|
|
|
|
|
|
|
nvs_flash_deinit_partition(TOO_LONG_NAME); // just in case
|
|
|
|
}
|
|
|
|
|
2020-03-05 19:02:26 +08:00
|
|
|
TEST_CASE("flash erase deinitializes initialized partition", "[nvs]")
|
2020-02-14 15:09:22 +01:00
|
|
|
{
|
2020-03-05 19:02:26 +08:00
|
|
|
nvs_handle_t handle;
|
2020-02-14 15:09:22 +01:00
|
|
|
esp_err_t err = nvs_flash_init();
|
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
2020-03-05 19:02:26 +08:00
|
|
|
nvs_flash_erase();
|
2020-02-14 15:09:22 +01:00
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
2020-04-27 08:51:31 +08:00
|
|
|
TEST_ESP_OK( err );
|
2020-02-14 15:09:22 +01:00
|
|
|
|
2020-03-05 19:02:26 +08:00
|
|
|
TEST_ESP_OK(nvs_flash_init());
|
|
|
|
TEST_ESP_OK(nvs_open("uninit_ns", NVS_READWRITE, &handle));
|
|
|
|
nvs_close(handle);
|
|
|
|
TEST_ESP_OK(nvs_flash_erase());
|
|
|
|
|
|
|
|
// exptected: no partition is initialized since nvs_flash_erase() deinitialized the partition again
|
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_NOT_INITIALIZED, nvs_open("uninit_ns", NVS_READWRITE, &handle));
|
2020-02-14 15:09:22 +01:00
|
|
|
|
2020-03-05 19:02:26 +08:00
|
|
|
// just to be sure it's deinitialized in case of error and not affecting other tests
|
2020-02-14 15:09:22 +01:00
|
|
|
nvs_flash_deinit();
|
|
|
|
}
|
|
|
|
|
2019-11-25 11:32:40 +08:00
|
|
|
// test could have different output on host tests
|
|
|
|
TEST_CASE("nvs deinit with open handle", "[nvs]")
|
|
|
|
{
|
|
|
|
nvs_handle_t handle_1;
|
|
|
|
esp_err_t err = nvs_flash_init();
|
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
|
|
|
ESP_LOGW(TAG, "nvs_flash_init failed (0x%x), erasing partition and retrying", err);
|
2020-03-05 19:02:26 +08:00
|
|
|
ESP_ERROR_CHECK(nvs_flash_erase());
|
2019-11-25 11:32:40 +08:00
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
|
|
|
ESP_ERROR_CHECK( err );
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_open("deinit_ns", NVS_READWRITE, &handle_1));
|
|
|
|
nvs_flash_deinit();
|
|
|
|
}
|
|
|
|
|
2016-11-17 16:36:10 +08:00
|
|
|
TEST_CASE("various nvs tests", "[nvs]")
|
|
|
|
{
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_1;
|
2017-03-14 21:39:44 +08:00
|
|
|
esp_err_t err = nvs_flash_init();
|
2018-07-25 20:41:09 +05:30
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
2017-03-14 21:39:44 +08:00
|
|
|
ESP_LOGW(TAG, "nvs_flash_init failed (0x%x), erasing partition and retrying", err);
|
2020-03-05 19:02:26 +08:00
|
|
|
ESP_ERROR_CHECK(nvs_flash_erase());
|
2017-03-14 21:39:44 +08:00
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
|
|
|
ESP_ERROR_CHECK( err );
|
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_NOT_FOUND, nvs_open("test_namespace1", NVS_READONLY, &handle_1));
|
2016-11-17 16:36:10 +08:00
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_INVALID_HANDLE, nvs_set_i32(handle_1, "foo", 0x12345678));
|
2016-11-23 11:29:36 +08:00
|
|
|
nvs_close(handle_1);
|
2016-11-17 16:36:10 +08:00
|
|
|
|
2016-11-23 11:29:36 +08:00
|
|
|
TEST_ESP_OK(nvs_open("test_namespace2", NVS_READWRITE, &handle_1));
|
|
|
|
TEST_ESP_OK(nvs_erase_all(handle_1));
|
2016-11-17 16:36:10 +08:00
|
|
|
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;
|
2016-11-23 11:29:36 +08:00
|
|
|
TEST_ESP_OK(nvs_open("test_namespace3", NVS_READWRITE, &handle_2));
|
|
|
|
TEST_ESP_OK(nvs_erase_all(handle_2));
|
2016-11-17 16:36:10 +08:00
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_2, "foo", 0x3456789a));
|
2016-11-23 11:29:36 +08:00
|
|
|
const char* str = "value 0123456789abcdef0123456789abcdef";
|
2016-11-17 16:36:10 +08:00
|
|
|
TEST_ESP_OK(nvs_set_str(handle_2, "key", str));
|
|
|
|
|
|
|
|
int32_t v1;
|
|
|
|
TEST_ESP_OK(nvs_get_i32(handle_1, "foo", &v1));
|
|
|
|
TEST_ASSERT_EQUAL_INT32(0x23456789, v1);
|
|
|
|
|
|
|
|
int32_t v2;
|
|
|
|
TEST_ESP_OK(nvs_get_i32(handle_2, "foo", &v2));
|
|
|
|
TEST_ASSERT_EQUAL_INT32(0x3456789a, v2);
|
|
|
|
|
|
|
|
char buf[strlen(str) + 1];
|
|
|
|
size_t buf_len = sizeof(buf);
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_get_str(handle_2, "key", buf, &buf_len));
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT32(0, strcmp(buf, str));
|
2016-11-23 11:29:36 +08:00
|
|
|
|
|
|
|
nvs_close(handle_1);
|
2017-09-18 22:30:21 +08:00
|
|
|
|
|
|
|
// check that deinit does not leak memory if some handles are still open
|
|
|
|
nvs_flash_deinit();
|
|
|
|
|
2016-11-23 11:29:36 +08:00
|
|
|
nvs_close(handle_2);
|
2016-11-17 16:36:10 +08:00
|
|
|
}
|
2018-02-20 12:11:56 +05:00
|
|
|
|
|
|
|
TEST_CASE("calculate used and free space", "[nvs]")
|
|
|
|
{
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_INVALID_ARG, nvs_get_stats(NULL, NULL));
|
2018-02-20 12:11:56 +05:00
|
|
|
nvs_stats_t stat1;
|
|
|
|
nvs_stats_t stat2;
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_NOT_INITIALIZED, nvs_get_stats(NULL, &stat1));
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries == 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 0);
|
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle = 0;
|
2018-02-20 12:11:56 +05:00
|
|
|
size_t h_count_entries;
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_INVALID_HANDLE, nvs_get_used_entry_count(handle, &h_count_entries));
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ASSERT_TRUE(h_count_entries == 0);
|
|
|
|
|
|
|
|
esp_err_t err = nvs_flash_init();
|
2018-07-25 20:41:09 +05:30
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
2018-02-20 12:11:56 +05:00
|
|
|
ESP_LOGW(TAG, "nvs_flash_init failed (0x%x), erasing partition and retrying", err);
|
|
|
|
const esp_partition_t* nvs_partition = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS, NULL);
|
|
|
|
assert(nvs_partition && "partition table must have an NVS partition");
|
2020-03-05 19:02:26 +08:00
|
|
|
ESP_ERROR_CHECK(nvs_flash_erase());
|
2018-02-20 12:11:56 +05:00
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
|
|
|
ESP_ERROR_CHECK( err );
|
|
|
|
|
|
|
|
// erase if have any namespace
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
|
|
|
if(stat1.namespace_count != 0) {
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
2020-02-14 15:09:22 +01:00
|
|
|
TEST_ESP_OK(nvs_flash_erase());
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ESP_OK(nvs_flash_init());
|
|
|
|
}
|
|
|
|
|
|
|
|
// after erase. empty partition
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries != 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries != 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 0);
|
|
|
|
|
|
|
|
// create namespace test_k1
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_1;
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ESP_OK(nvs_open("test_k1", NVS_READWRITE, &handle_1));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat2));
|
|
|
|
TEST_ASSERT_TRUE(stat2.free_entries + 1 == stat1.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.namespace_count == 1);
|
|
|
|
TEST_ASSERT_TRUE(stat2.total_entries == stat1.total_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.used_entries == 1);
|
|
|
|
|
|
|
|
// create pair key-value com
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "com", 0x12345678));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 1 == stat2.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 1);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == stat2.total_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 2);
|
|
|
|
|
|
|
|
// change value in com
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "com", 0x01234567));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat2));
|
|
|
|
TEST_ASSERT_TRUE(stat2.free_entries == stat1.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.namespace_count == 1);
|
|
|
|
TEST_ASSERT_TRUE(stat2.total_entries != 0);
|
|
|
|
TEST_ASSERT_TRUE(stat2.used_entries == 2);
|
|
|
|
|
|
|
|
// create pair key-value ru
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "ru", 0x00FF00FF));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 1 == stat2.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 1);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries != 0);
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 3);
|
|
|
|
|
|
|
|
// amount valid pair in namespace 1
|
|
|
|
size_t h1_count_entries;
|
|
|
|
TEST_ESP_OK(nvs_get_used_entry_count(handle_1, &h1_count_entries));
|
|
|
|
TEST_ASSERT_TRUE(h1_count_entries == 2);
|
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_2;
|
2018-02-20 12:11:56 +05:00
|
|
|
// create namespace test_k2
|
|
|
|
TEST_ESP_OK(nvs_open("test_k2", NVS_READWRITE, &handle_2));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat2));
|
|
|
|
TEST_ASSERT_TRUE(stat2.free_entries + 1 == stat1.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.namespace_count == 2);
|
|
|
|
TEST_ASSERT_TRUE(stat2.total_entries == stat1.total_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.used_entries == 4);
|
|
|
|
|
|
|
|
// create pair key-value
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_2, "su1", 0x00000001));
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_2, "su2", 0x00000002));
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_2, "sus", 0x00000003));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 3 == stat2.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 2);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == stat2.total_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 7);
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == (stat1.used_entries + stat1.free_entries));
|
|
|
|
|
|
|
|
// amount valid pair in namespace 2
|
|
|
|
size_t h2_count_entries;
|
|
|
|
TEST_ESP_OK(nvs_get_used_entry_count(handle_2, &h2_count_entries));
|
|
|
|
TEST_ASSERT_TRUE(h2_count_entries == 3);
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == (h1_count_entries + h2_count_entries + stat1.namespace_count));
|
|
|
|
|
|
|
|
nvs_close(handle_1);
|
|
|
|
nvs_close(handle_2);
|
|
|
|
|
|
|
|
size_t temp = h2_count_entries;
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_INVALID_HANDLE, nvs_get_used_entry_count(handle_1, &h2_count_entries));
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ASSERT_TRUE(h2_count_entries == 0);
|
|
|
|
h2_count_entries = temp;
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_INVALID_ARG, nvs_get_used_entry_count(handle_1, NULL));
|
2018-02-20 12:11:56 +05:00
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_3;
|
2018-02-20 12:11:56 +05:00
|
|
|
// create namespace test_k3
|
|
|
|
TEST_ESP_OK(nvs_open("test_k3", NVS_READWRITE, &handle_3));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat2));
|
|
|
|
TEST_ASSERT_TRUE(stat2.free_entries + 1 == stat1.free_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.namespace_count == 3);
|
|
|
|
TEST_ASSERT_TRUE(stat2.total_entries == stat1.total_entries);
|
|
|
|
TEST_ASSERT_TRUE(stat2.used_entries == 8);
|
|
|
|
|
|
|
|
// create pair blobs
|
|
|
|
uint32_t blob[12];
|
|
|
|
TEST_ESP_OK(nvs_set_blob(handle_3, "bl1", &blob, sizeof(blob)));
|
|
|
|
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
|
2018-04-02 16:14:59 +05:30
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 4 == stat2.free_entries);
|
2018-02-20 12:11:56 +05:00
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 3);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == stat2.total_entries);
|
2018-04-02 16:14:59 +05:30
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 12);
|
2018-02-20 12:11:56 +05:00
|
|
|
|
|
|
|
// amount valid pair in namespace 2
|
|
|
|
size_t h3_count_entries;
|
|
|
|
TEST_ESP_OK(nvs_get_used_entry_count(handle_3, &h3_count_entries));
|
2018-04-02 16:14:59 +05:30
|
|
|
TEST_ASSERT_TRUE(h3_count_entries == 4);
|
2018-02-20 12:11:56 +05:00
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == (h1_count_entries + h2_count_entries + h3_count_entries + stat1.namespace_count));
|
|
|
|
|
|
|
|
nvs_close(handle_3);
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
2020-02-14 15:09:22 +01:00
|
|
|
TEST_ESP_OK(nvs_flash_erase());
|
2018-02-20 12:11:56 +05:00
|
|
|
}
|
2018-08-09 10:35:19 +03:00
|
|
|
|
|
|
|
TEST_CASE("check for memory leaks in nvs_set_blob", "[nvs]")
|
|
|
|
{
|
|
|
|
esp_err_t err = nvs_flash_init();
|
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
|
|
|
ESP_ERROR_CHECK(nvs_flash_erase());
|
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
|
|
|
TEST_ESP_OK( err );
|
|
|
|
|
|
|
|
for (int i = 0; i < 500; ++i) {
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t my_handle;
|
2018-08-09 10:35:19 +03:00
|
|
|
uint8_t key[20] = {0};
|
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_OK( nvs_open("leak_check_ns", NVS_READWRITE, &my_handle) );
|
2018-08-09 10:35:19 +03:00
|
|
|
TEST_ESP_OK( nvs_set_blob(my_handle, "key", key, sizeof(key)) );
|
|
|
|
TEST_ESP_OK( nvs_commit(my_handle) );
|
|
|
|
nvs_close(my_handle);
|
|
|
|
printf("%d\n", esp_get_free_heap_size());
|
|
|
|
}
|
|
|
|
|
|
|
|
nvs_flash_deinit();
|
|
|
|
printf("%d\n", esp_get_free_heap_size());
|
|
|
|
/* heap leaks will be checked in unity_platform.c */
|
|
|
|
}
|
2018-07-02 16:40:43 +05:30
|
|
|
|
|
|
|
#ifdef CONFIG_NVS_ENCRYPTION
|
|
|
|
TEST_CASE("check underlying xts code for 32-byte size sector encryption", "[nvs]")
|
|
|
|
{
|
2019-11-06 17:21:37 +08:00
|
|
|
uint8_t eky_hex[2 * NVS_KEY_SIZE] = { 0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11,
|
2018-07-02 16:40:43 +05:30
|
|
|
0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11,
|
|
|
|
0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11,
|
|
|
|
0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11,
|
|
|
|
/* Tweak key below*/
|
2019-11-06 17:21:37 +08:00
|
|
|
0x22,0x22,0x22,0x22,0x22,0x22,0x22,0x22,
|
2018-07-02 16:40:43 +05:30
|
|
|
0x22,0x22,0x22,0x22,0x22,0x22,0x22,0x22,
|
|
|
|
0x22,0x22,0x22,0x22,0x22,0x22,0x22,0x22,
|
|
|
|
0x22,0x22,0x22,0x22,0x22,0x22,0x22,0x22 };
|
|
|
|
|
|
|
|
uint8_t ba_hex[16] = { 0x33,0x33,0x33,0x33,0x33,0x00,0x00,0x00,
|
|
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 };
|
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
uint8_t ptxt_hex[32] = { 0x44,0x44,0x44,0x44,0x44,0x44,0x44,0x44,
|
2018-07-02 16:40:43 +05:30
|
|
|
0x44,0x44,0x44,0x44,0x44,0x44,0x44,0x44,
|
|
|
|
0x44,0x44,0x44,0x44,0x44,0x44,0x44,0x44,
|
|
|
|
0x44,0x44,0x44,0x44,0x44,0x44,0x44,0x44 };
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t ctxt_hex[32] = { 0xe6,0x22,0x33,0x4f,0x18,0x4b,0xbc,0xe1,
|
|
|
|
0x29,0xa2,0x5b,0x2a,0xc7,0x6b,0x3d,0x92,
|
|
|
|
0xab,0xf9,0x8e,0x22,0xdf,0x5b,0xdd,0x15,
|
|
|
|
0xaf,0x47,0x1f,0x3d,0xb8,0x94,0x6a,0x85 };
|
|
|
|
|
|
|
|
mbedtls_aes_xts_context ectx[1];
|
|
|
|
mbedtls_aes_xts_context dctx[1];
|
|
|
|
|
|
|
|
mbedtls_aes_xts_init(ectx);
|
|
|
|
mbedtls_aes_xts_init(dctx);
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(!mbedtls_aes_xts_setkey_enc(ectx, eky_hex, 2 * NVS_KEY_SIZE * 8));
|
|
|
|
TEST_ASSERT_TRUE(!mbedtls_aes_xts_setkey_enc(dctx, eky_hex, 2 * NVS_KEY_SIZE * 8));
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(!mbedtls_aes_crypt_xts(ectx, MBEDTLS_AES_ENCRYPT, 32, ba_hex, ptxt_hex, ptxt_hex));
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(!memcmp(ptxt_hex, ctxt_hex, 32));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("Check nvs key partition APIs (read and generate keys)", "[nvs]")
|
|
|
|
{
|
|
|
|
nvs_sec_cfg_t cfg, cfg2;
|
2019-11-06 17:21:37 +08:00
|
|
|
|
2018-07-02 16:40:43 +05:30
|
|
|
const esp_partition_t* key_part = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS_KEYS, NULL);
|
|
|
|
|
|
|
|
if (!esp_flash_encryption_enabled()) {
|
|
|
|
TEST_IGNORE_MESSAGE("flash encryption disabled, skipping nvs_key partition related tests");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_ESP_OK(esp_partition_erase_range(key_part, 0, key_part->size));
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_KEYS_NOT_INITIALIZED, nvs_flash_read_security_cfg(key_part, &cfg));
|
|
|
|
|
2018-07-02 16:40:43 +05:30
|
|
|
TEST_ESP_OK(nvs_flash_generate_keys(key_part, &cfg));
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_flash_read_security_cfg(key_part, &cfg2));
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(!memcmp(&cfg, &cfg2, sizeof(nvs_sec_cfg_t)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_CASE("test nvs apis with encryption enabled", "[nvs]")
|
|
|
|
{
|
|
|
|
if (!esp_flash_encryption_enabled()) {
|
|
|
|
TEST_IGNORE_MESSAGE("flash encryption disabled, skipping nvs_api tests with encryption enabled");
|
|
|
|
}
|
|
|
|
const esp_partition_t* key_part = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS_KEYS, NULL);
|
2019-11-06 17:21:37 +08:00
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
assert(key_part && "partition table must have an NVS Key partition");
|
2018-07-02 16:40:43 +05:30
|
|
|
|
|
|
|
const esp_partition_t* nvs_partition = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS, NULL);
|
|
|
|
assert(nvs_partition && "partition table must have an NVS partition");
|
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
ESP_ERROR_CHECK( esp_partition_erase_range(key_part, 0, key_part->size) );
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
bool done = false;
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
do {
|
|
|
|
ESP_ERROR_CHECK( esp_partition_erase_range(nvs_partition, 0, nvs_partition->size) );
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
nvs_sec_cfg_t cfg;
|
|
|
|
esp_err_t err = nvs_flash_read_security_cfg(key_part, &cfg);
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
if(err == ESP_ERR_NVS_KEYS_NOT_INITIALIZED) {
|
|
|
|
uint8_t value[4096] = {[0 ... 4095] = 0xff};
|
|
|
|
TEST_ESP_OK(esp_partition_write(key_part, 0, value, sizeof(value)));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_KEYS_NOT_INITIALIZED, nvs_flash_read_security_cfg(key_part, &cfg));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
TEST_ESP_OK(nvs_flash_generate_keys(key_part, &cfg));
|
|
|
|
} else {
|
|
|
|
/* Second time key_partition exists already*/
|
|
|
|
ESP_ERROR_CHECK(err);
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
TEST_ESP_OK(nvs_flash_secure_init(&cfg));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_1;
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-11-06 17:21:37 +08:00
|
|
|
TEST_ESP_ERR(ESP_ERR_NVS_NOT_FOUND, nvs_open("namespace1", NVS_READONLY, &handle_1));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle_1));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "foo", 0x12345678));
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "foo", 0x23456789));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle_2;
|
2019-02-01 13:05:48 +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));
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
int32_t v1;
|
|
|
|
TEST_ESP_OK(nvs_get_i32(handle_1, "foo", &v1));
|
|
|
|
TEST_ASSERT_TRUE(0x23456789 == v1);
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
int32_t v2;
|
|
|
|
TEST_ESP_OK(nvs_get_i32(handle_2, "foo", &v2));
|
|
|
|
TEST_ASSERT_TRUE(0x3456789a == v2);
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
char buf[strlen(str) + 1];
|
|
|
|
size_t buf_len = sizeof(buf);
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
size_t buf_len_needed;
|
|
|
|
TEST_ESP_OK(nvs_get_str(handle_2, "key", NULL, &buf_len_needed));
|
|
|
|
TEST_ASSERT_TRUE(buf_len_needed == buf_len);
|
2018-07-02 16:40:43 +05:30
|
|
|
|
2019-02-01 13:05:48 +05:30
|
|
|
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));
|
|
|
|
TEST_ASSERT_TRUE(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));
|
|
|
|
TEST_ASSERT_TRUE(buf_len_long == buf_len);
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_get_str(handle_2, "key", buf, &buf_len));
|
|
|
|
|
|
|
|
TEST_ASSERT_TRUE(0 == strcmp(buf, str));
|
|
|
|
|
|
|
|
nvs_close(handle_1);
|
|
|
|
nvs_close(handle_2);
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
|
|
|
} while(!done);
|
2018-07-02 16:40:43 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("test nvs apis for nvs partition generator utility with encryption enabled", "[nvs_part_gen]")
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!esp_flash_encryption_enabled()) {
|
|
|
|
TEST_IGNORE_MESSAGE("flash encryption disabled, skipping nvs_api tests with encryption enabled");
|
|
|
|
}
|
|
|
|
|
2019-04-01 15:13:55 +01:00
|
|
|
nvs_handle_t handle;
|
2018-07-02 16:40:43 +05:30
|
|
|
nvs_sec_cfg_t xts_cfg;
|
|
|
|
|
|
|
|
extern const char nvs_key_start[] asm("_binary_encryption_keys_bin_start");
|
|
|
|
extern const char nvs_key_end[] asm("_binary_encryption_keys_bin_end");
|
|
|
|
|
|
|
|
extern const char nvs_data_start[] asm("_binary_partition_encrypted_bin_start");
|
|
|
|
|
|
|
|
extern const char sample_bin_start[] asm("_binary_sample_bin_start");
|
|
|
|
|
|
|
|
const esp_partition_t* key_part = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS_KEYS, NULL);
|
|
|
|
|
|
|
|
const esp_partition_t* nvs_part = esp_partition_find_first(
|
|
|
|
ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_NVS, NULL);
|
|
|
|
|
|
|
|
assert(key_part && "partition table must have a KEY partition");
|
|
|
|
TEST_ASSERT_TRUE((nvs_key_end - nvs_key_start - 1) == SPI_FLASH_SEC_SIZE);
|
|
|
|
|
|
|
|
assert(nvs_part && "partition table must have an NVS partition");
|
|
|
|
printf("\n nvs_part size:%d\n", nvs_part->size);
|
|
|
|
|
|
|
|
ESP_ERROR_CHECK(esp_partition_erase_range(key_part, 0, key_part->size));
|
|
|
|
ESP_ERROR_CHECK( esp_partition_erase_range(nvs_part, 0, nvs_part->size) );
|
|
|
|
|
|
|
|
for (int i = 0; i < key_part->size; i+= SPI_FLASH_SEC_SIZE) {
|
|
|
|
ESP_ERROR_CHECK( esp_partition_write(key_part, i, nvs_key_start + i, SPI_FLASH_SEC_SIZE) );
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < nvs_part->size; i+= SPI_FLASH_SEC_SIZE) {
|
2020-04-27 08:51:31 +08:00
|
|
|
ESP_ERROR_CHECK( esp_partition_write(nvs_part, i, nvs_data_start + i, SPI_FLASH_SEC_SIZE) );
|
2018-07-02 16:40:43 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t err = nvs_flash_read_security_cfg(key_part, &xts_cfg);
|
|
|
|
ESP_ERROR_CHECK(err);
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_flash_secure_init(&xts_cfg));
|
|
|
|
|
|
|
|
TEST_ESP_OK(nvs_open("dummyNamespace", NVS_READONLY, &handle));
|
|
|
|
uint8_t u8v;
|
|
|
|
TEST_ESP_OK( nvs_get_u8(handle, "dummyU8Key", &u8v));
|
|
|
|
TEST_ASSERT_TRUE(u8v == 127);
|
|
|
|
int8_t i8v;
|
|
|
|
TEST_ESP_OK( nvs_get_i8(handle, "dummyI8Key", &i8v));
|
|
|
|
TEST_ASSERT_TRUE(i8v == -128);
|
|
|
|
uint16_t u16v;
|
|
|
|
TEST_ESP_OK( nvs_get_u16(handle, "dummyU16Key", &u16v));
|
|
|
|
TEST_ASSERT_TRUE(u16v == 32768);
|
|
|
|
uint32_t u32v;
|
|
|
|
TEST_ESP_OK( nvs_get_u32(handle, "dummyU32Key", &u32v));
|
|
|
|
TEST_ASSERT_TRUE(u32v == 4294967295);
|
|
|
|
int32_t i32v;
|
|
|
|
TEST_ESP_OK( nvs_get_i32(handle, "dummyI32Key", &i32v));
|
|
|
|
TEST_ASSERT_TRUE(i32v == -2147483648);
|
|
|
|
|
|
|
|
char buf[64] = {0};
|
|
|
|
size_t buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_str(handle, "dummyStringKey", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(strncmp(buf, "0A:0B:0C:0D:0E:0F", buflen) == 0);
|
|
|
|
|
|
|
|
uint8_t hexdata[] = {0x01, 0x02, 0x03, 0xab, 0xcd, 0xef};
|
|
|
|
buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_blob(handle, "dummyHex2BinKey", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(buf, hexdata, buflen) == 0);
|
|
|
|
|
|
|
|
uint8_t base64data[] = {'1', '2', '3', 'a', 'b', 'c'};
|
|
|
|
buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_blob(handle, "dummyBase64Key", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(buf, base64data, buflen) == 0);
|
|
|
|
|
|
|
|
uint8_t hexfiledata[] = {0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef};
|
|
|
|
buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_blob(handle, "hexFileKey", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(buf, hexfiledata, buflen) == 0);
|
|
|
|
|
|
|
|
uint8_t base64filedata[] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0xab, 0xcd, 0xef};
|
|
|
|
buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_blob(handle, "base64FileKey", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(buf, base64filedata, buflen) == 0);
|
|
|
|
|
|
|
|
uint8_t strfiledata[64] = "abcdefghijklmnopqrstuvwxyz\0";
|
|
|
|
buflen = 64;
|
|
|
|
TEST_ESP_OK( nvs_get_str(handle, "stringFileKey", buf, &buflen));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(buf, strfiledata, buflen) == 0);
|
|
|
|
|
|
|
|
char bin_data[5120];
|
|
|
|
size_t bin_len = sizeof(bin_data);
|
|
|
|
TEST_ESP_OK( nvs_get_blob(handle, "binFileKey", bin_data, &bin_len));
|
|
|
|
TEST_ASSERT_TRUE(memcmp(bin_data, sample_bin_start, bin_len) == 0);
|
|
|
|
|
|
|
|
nvs_close(handle);
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|