mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
245 lines
6.0 KiB
C++
245 lines
6.0 KiB
C++
/*
|
|
* SPDX-FileCopyrightText: 2015-2023 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include "esp_partition.h"
|
|
#include "nvs_partition_manager.hpp"
|
|
#include "nvs_partition_lookup.hpp"
|
|
#include "nvs_internal.h"
|
|
|
|
#ifndef LINUX_TARGET
|
|
#include "nvs_encrypted_partition.hpp"
|
|
#endif // ! LINUX_TARGET
|
|
|
|
namespace nvs {
|
|
|
|
NVSPartitionManager* NVSPartitionManager::instance = nullptr;
|
|
|
|
NVSPartitionManager* NVSPartitionManager::get_instance()
|
|
{
|
|
if (!instance) {
|
|
instance = new (std::nothrow) NVSPartitionManager();
|
|
}
|
|
|
|
return instance;
|
|
}
|
|
|
|
#ifdef ESP_PLATFORM
|
|
esp_err_t NVSPartitionManager::init_partition(const char *partition_label)
|
|
{
|
|
if (strlen(partition_label) > NVS_PART_NAME_MAX_SIZE) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
uint32_t size;
|
|
Storage* mStorage;
|
|
|
|
mStorage = lookup_storage_from_name(partition_label);
|
|
if (mStorage) {
|
|
return ESP_OK;
|
|
}
|
|
|
|
NVS_ASSERT_OR_RETURN(SPI_FLASH_SEC_SIZE != 0, ESP_FAIL);
|
|
|
|
NVSPartition *p = nullptr;
|
|
esp_err_t result = partition_lookup::lookup_nvs_partition(partition_label, &p);
|
|
|
|
if (result != ESP_OK) {
|
|
goto error;
|
|
}
|
|
|
|
size = p->get_size();
|
|
|
|
result = init_custom(p, 0, size / SPI_FLASH_SEC_SIZE);
|
|
if (result != ESP_OK) {
|
|
goto error;
|
|
}
|
|
|
|
nvs_partition_list.push_back(p);
|
|
|
|
return ESP_OK;
|
|
|
|
error:
|
|
delete p;
|
|
return result;
|
|
}
|
|
#endif // ESP_PLATFORM
|
|
|
|
esp_err_t NVSPartitionManager::init_custom(Partition *partition, uint32_t baseSector, uint32_t sectorCount)
|
|
{
|
|
Storage* new_storage = nullptr;
|
|
Storage* storage = lookup_storage_from_name(partition->get_partition_name());
|
|
if (storage == nullptr) {
|
|
new_storage = new (std::nothrow) Storage(partition);
|
|
|
|
if (new_storage == nullptr) {
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
storage = new_storage;
|
|
} else {
|
|
// if storage was initialized already, we don't need partition and hence delete it
|
|
for (auto it = nvs_partition_list.begin(); it != nvs_partition_list.end(); ++it) {
|
|
if (partition == it) {
|
|
nvs_partition_list.erase(it);
|
|
delete partition;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
esp_err_t err = storage->init(baseSector, sectorCount);
|
|
if (new_storage != nullptr) {
|
|
if (err == ESP_OK) {
|
|
nvs_storage_list.push_back(new_storage);
|
|
} else {
|
|
delete new_storage;
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
#ifdef ESP_PLATFORM
|
|
esp_err_t NVSPartitionManager::secure_init_partition(const char *part_name, nvs_sec_cfg_t* cfg)
|
|
{
|
|
if (strlen(part_name) > NVS_PART_NAME_MAX_SIZE) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
Storage* mStorage;
|
|
|
|
mStorage = lookup_storage_from_name(part_name);
|
|
if (mStorage != nullptr) {
|
|
return ESP_OK;
|
|
}
|
|
|
|
NVSPartition *p;
|
|
esp_err_t result;
|
|
if (cfg != nullptr) {
|
|
result = partition_lookup::lookup_nvs_encrypted_partition(part_name, cfg, &p);
|
|
} else {
|
|
result = partition_lookup::lookup_nvs_partition(part_name, &p);
|
|
}
|
|
|
|
if (result != ESP_OK) {
|
|
return result;
|
|
}
|
|
|
|
uint32_t size = p->get_size();
|
|
|
|
result = init_custom(p, 0, size / SPI_FLASH_SEC_SIZE);
|
|
if (result != ESP_OK) {
|
|
delete p;
|
|
return result;
|
|
}
|
|
|
|
nvs_partition_list.push_back(p);
|
|
|
|
return ESP_OK;
|
|
}
|
|
#endif // ESP_PLATFORM
|
|
|
|
esp_err_t NVSPartitionManager::deinit_partition(const char *partition_label)
|
|
{
|
|
Storage* storage = lookup_storage_from_name(partition_label);
|
|
if (!storage) {
|
|
return ESP_ERR_NVS_NOT_INITIALIZED;
|
|
}
|
|
|
|
/* Clean up handles related to the storage being deinitialized */
|
|
for (auto it = nvs_handles.begin(); it != nvs_handles.end(); ++it) {
|
|
if (it->mStoragePtr == storage) {
|
|
it->valid = false;
|
|
nvs_handles.erase(it);
|
|
}
|
|
}
|
|
|
|
/* Finally delete the storage and its partition */
|
|
nvs_storage_list.erase(storage);
|
|
delete storage;
|
|
|
|
for (auto it = nvs_partition_list.begin(); it != nvs_partition_list.end(); ++it) {
|
|
if (strcmp(it->get_partition_name(), partition_label) == 0) {
|
|
NVSPartition *p = it;
|
|
nvs_partition_list.erase(it);
|
|
delete p;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t NVSPartitionManager::open_handle(const char *part_name,
|
|
const char *ns_name,
|
|
nvs_open_mode_t open_mode,
|
|
NVSHandleSimple** handle)
|
|
{
|
|
uint8_t nsIndex;
|
|
Storage* sHandle;
|
|
|
|
if (handle == nullptr) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (nvs_storage_list.empty()) {
|
|
return ESP_ERR_NVS_NOT_INITIALIZED;
|
|
}
|
|
|
|
sHandle = lookup_storage_from_name(part_name);
|
|
if (sHandle == nullptr) {
|
|
return ESP_ERR_NVS_PART_NOT_FOUND;
|
|
}
|
|
|
|
if (open_mode == NVS_READWRITE && const_cast<Partition*>(sHandle->getPart())->get_readonly()) {
|
|
return ESP_ERR_NOT_ALLOWED;
|
|
}
|
|
|
|
|
|
esp_err_t err = sHandle->createOrOpenNamespace(ns_name, open_mode == NVS_READWRITE, nsIndex);
|
|
if (err != ESP_OK) {
|
|
return err;
|
|
}
|
|
|
|
NVSHandleSimple* new_handle = new (std::nothrow) NVSHandleSimple(open_mode==NVS_READONLY, nsIndex, sHandle);
|
|
if (new_handle == nullptr) {
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
nvs_handles.push_back(new_handle);
|
|
|
|
*handle = new_handle;
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t NVSPartitionManager::close_handle(NVSHandleSimple* handle) {
|
|
for (auto it = nvs_handles.begin(); it != nvs_handles.end(); ++it) {
|
|
if (it == intrusive_list<NVSHandleSimple>::iterator(handle)) {
|
|
nvs_handles.erase(it);
|
|
return ESP_OK;
|
|
}
|
|
}
|
|
|
|
return ESP_ERR_NVS_INVALID_HANDLE;
|
|
}
|
|
|
|
size_t NVSPartitionManager::open_handles_size()
|
|
{
|
|
return nvs_handles.size();
|
|
}
|
|
|
|
Storage* NVSPartitionManager::lookup_storage_from_name(const char* name)
|
|
{
|
|
auto it = find_if(begin(nvs_storage_list), end(nvs_storage_list), [=](Storage& e) -> bool {
|
|
return (strcmp(e.getPartName(), name) == 0);
|
|
});
|
|
|
|
if (it == end(nvs_storage_list)) {
|
|
return nullptr;
|
|
}
|
|
return it;
|
|
}
|
|
|
|
} // nvs
|