2016-11-17 03:36:10 -05: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 09:39:44 -04:00
|
|
|
#include "esp_partition.h"
|
|
|
|
#include "esp_log.h"
|
2016-11-17 03:36:10 -05:00
|
|
|
#include <string.h>
|
2018-04-02 06:44:59 -04:00
|
|
|
#include "esp_system.h"
|
2016-11-17 03:36:10 -05:00
|
|
|
|
2017-03-14 09:39:44 -04:00
|
|
|
static const char* TAG = "test_nvs";
|
2016-11-17 03:36:10 -05:00
|
|
|
|
|
|
|
TEST_CASE("various nvs tests", "[nvs]")
|
|
|
|
{
|
|
|
|
nvs_handle handle_1;
|
2017-03-14 09:39:44 -04:00
|
|
|
esp_err_t err = nvs_flash_init();
|
2018-07-25 09:25:05 -04:00
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
2017-03-14 09:39:44 -04: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");
|
|
|
|
ESP_ERROR_CHECK( esp_partition_erase_range(nvs_partition, 0, nvs_partition->size) );
|
|
|
|
err = nvs_flash_init();
|
|
|
|
}
|
|
|
|
ESP_ERROR_CHECK( err );
|
|
|
|
|
2016-11-22 22:29:36 -05:00
|
|
|
TEST_ESP_ERR(nvs_open("test_namespace1", NVS_READONLY, &handle_1), ESP_ERR_NVS_NOT_FOUND);
|
2016-11-17 03:36:10 -05:00
|
|
|
|
2016-11-22 22:29:36 -05:00
|
|
|
TEST_ESP_ERR(nvs_set_i32(handle_1, "foo", 0x12345678), ESP_ERR_NVS_INVALID_HANDLE);
|
|
|
|
nvs_close(handle_1);
|
2016-11-17 03:36:10 -05:00
|
|
|
|
2016-11-22 22:29:36 -05:00
|
|
|
TEST_ESP_OK(nvs_open("test_namespace2", NVS_READWRITE, &handle_1));
|
|
|
|
TEST_ESP_OK(nvs_erase_all(handle_1));
|
2016-11-17 03:36:10 -05:00
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "foo", 0x12345678));
|
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_1, "foo", 0x23456789));
|
|
|
|
|
|
|
|
nvs_handle handle_2;
|
2016-11-22 22:29:36 -05:00
|
|
|
TEST_ESP_OK(nvs_open("test_namespace3", NVS_READWRITE, &handle_2));
|
|
|
|
TEST_ESP_OK(nvs_erase_all(handle_2));
|
2016-11-17 03:36:10 -05:00
|
|
|
TEST_ESP_OK(nvs_set_i32(handle_2, "foo", 0x3456789a));
|
2016-11-22 22:29:36 -05:00
|
|
|
const char* str = "value 0123456789abcdef0123456789abcdef";
|
2016-11-17 03:36:10 -05: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-22 22:29:36 -05:00
|
|
|
|
|
|
|
nvs_close(handle_1);
|
2017-09-18 10:30:21 -04:00
|
|
|
|
|
|
|
// check that deinit does not leak memory if some handles are still open
|
|
|
|
nvs_flash_deinit();
|
|
|
|
|
2016-11-22 22:29:36 -05:00
|
|
|
nvs_close(handle_2);
|
2016-11-17 03:36:10 -05:00
|
|
|
}
|
2018-02-20 02:11:56 -05:00
|
|
|
|
|
|
|
TEST_CASE("calculate used and free space", "[nvs]")
|
|
|
|
{
|
|
|
|
TEST_ESP_ERR(nvs_get_stats(NULL, NULL), ESP_ERR_INVALID_ARG);
|
|
|
|
nvs_stats_t stat1;
|
|
|
|
nvs_stats_t stat2;
|
2018-04-16 05:41:34 -04:00
|
|
|
TEST_ESP_ERR(nvs_get_stats(NULL, &stat1), ESP_ERR_NVS_NOT_INITIALIZED);
|
2018-02-20 02: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);
|
|
|
|
|
|
|
|
nvs_handle handle = 0;
|
|
|
|
size_t h_count_entries;
|
|
|
|
TEST_ESP_ERR(nvs_get_used_entry_count(handle, &h_count_entries), ESP_ERR_NVS_INVALID_HANDLE);
|
|
|
|
TEST_ASSERT_TRUE(h_count_entries == 0);
|
|
|
|
|
|
|
|
esp_err_t err = nvs_flash_init();
|
2018-07-25 09:25:05 -04:00
|
|
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
2018-02-20 02: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");
|
|
|
|
ESP_ERROR_CHECK( esp_partition_erase_range(nvs_partition, 0, nvs_partition->size) );
|
|
|
|
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_erase());
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
|
|
|
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
|
|
|
|
nvs_handle handle_1;
|
|
|
|
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);
|
|
|
|
|
|
|
|
nvs_handle handle_2;
|
|
|
|
// 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;
|
|
|
|
TEST_ESP_ERR(nvs_get_used_entry_count(handle_1, &h2_count_entries), ESP_ERR_NVS_INVALID_HANDLE);
|
|
|
|
TEST_ASSERT_TRUE(h2_count_entries == 0);
|
|
|
|
h2_count_entries = temp;
|
|
|
|
TEST_ESP_ERR(nvs_get_used_entry_count(handle_1, NULL), ESP_ERR_INVALID_ARG);
|
|
|
|
|
|
|
|
nvs_handle handle_3;
|
|
|
|
// 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 06:44:59 -04:00
|
|
|
#ifdef CONFIG_MP_BLOB_SUPPORT
|
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 4 == stat2.free_entries);
|
|
|
|
#else
|
2018-02-20 02:11:56 -05:00
|
|
|
TEST_ASSERT_TRUE(stat1.free_entries + 3 == stat2.free_entries);
|
2018-04-02 06:44:59 -04:00
|
|
|
#endif
|
|
|
|
|
2018-02-20 02:11:56 -05:00
|
|
|
TEST_ASSERT_TRUE(stat1.namespace_count == 3);
|
|
|
|
TEST_ASSERT_TRUE(stat1.total_entries == stat2.total_entries);
|
2018-04-02 06:44:59 -04:00
|
|
|
#ifdef CONFIG_MP_BLOB_SUPPORT
|
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 12);
|
|
|
|
#else
|
2018-02-20 02:11:56 -05:00
|
|
|
TEST_ASSERT_TRUE(stat1.used_entries == 11);
|
2018-04-02 06:44:59 -04:00
|
|
|
#endif
|
2018-02-20 02: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 06:44:59 -04:00
|
|
|
#ifdef CONFIG_MP_BLOB_SUPPORT
|
|
|
|
TEST_ASSERT_TRUE(h3_count_entries == 4);
|
|
|
|
#else
|
2018-02-20 02:11:56 -05:00
|
|
|
TEST_ASSERT_TRUE(h3_count_entries == 3);
|
2018-04-02 06:44:59 -04:00
|
|
|
#endif
|
2018-02-20 02: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_erase());
|
|
|
|
TEST_ESP_OK(nvs_flash_deinit());
|
|
|
|
}
|
2018-04-02 06:44:59 -04: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) {
|
|
|
|
nvs_handle my_handle;
|
|
|
|
uint8_t key[20] = {0};
|
|
|
|
|
|
|
|
TEST_ESP_OK( nvs_open("test_namespace", NVS_READWRITE, &my_handle) );
|
|
|
|
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 */
|
|
|
|
}
|