1
6
7
8
9
10
11
12
15
16
17
18
19
20
21
22
23
24
44
45
56
57
58
59
60
61
62
63
67
68
91
92
93
94
95
96
104
105
108
109
110
111
114
115
116
119
120
121
122
123
124
125
126
127
128
131
132
133
134
135
136
137
140
141
142
145
146
147
148
149
157
158
161
162
163
166
167
168
169
170
171
172
173
174
175
178
179
182
183
184
187
188
189
190
191
192
193
194
195
196
197
198
199
200
203
204
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
232
233
234
235
236
237
238
239
240
241
242
243
244
/* ... */
#include "nvs_pagemanager.hpp"
namespace nvs
{
esp_err_t PageManager::load(Partition *partition, uint32_t baseSector, uint32_t sectorCount)
{
if (partition == nullptr) {
return ESP_ERR_INVALID_ARG;
}{...}
mBaseSector = baseSector;
mPageCount = sectorCount;
mPageList.clear();
mFreePageList.clear();
mPages.reset(new (nothrow) Page[sectorCount]);
if (!mPages) return ESP_ERR_NO_MEM;
for (uint32_t i = 0; i < sectorCount; ++i) {
auto err = mPages[i].load(partition, baseSector + i);
if (err != ESP_OK) {
return err;
}{...}
uint32_t seqNumber;
if (mPages[i].getSeqNumber(seqNumber) != ESP_OK) {
mFreePageList.push_back(&mPages[i]);
}{...} else {
auto pos = std::find_if(std::begin(mPageList), std::end(mPageList), [=](const Page& page) -> bool {
uint32_t otherSeqNumber;
return page.getSeqNumber(otherSeqNumber) == ESP_OK && otherSeqNumber > seqNumber;
}{...});
if (pos == mPageList.end()) {
mPageList.push_back(&mPages[i]);
}{...} else {
mPageList.insert(pos, &mPages[i]);
}{...}
}{...}
}{...}
if (mPageList.empty()) {
mSeqNumber = 0;
return activatePage();
}{...} else {
uint32_t lastSeqNo;
auto err = mPageList.back().getSeqNumber(lastSeqNo);
if (err != ESP_OK) {
return err;
}{...}
mSeqNumber = lastSeqNo + 1;
}{...}
Page& lastPage = back();
size_t lastItemIndex = SIZE_MAX;
Item item;
size_t itemIndex = 0;
while (lastPage.findItem(Page::NS_ANY, ItemType::ANY, nullptr, itemIndex, item) == ESP_OK) {
itemIndex += item.span;
lastItemIndex = itemIndex;
}{...}
if (lastItemIndex != SIZE_MAX) {
auto last = PageManager::TPageListIterator(&lastPage);
TPageListIterator it;
for (it = begin(); it != last; ++it) {
if ((it->state() != Page::PageState::FREEING) &&
(it->eraseItem(item.nsIndex, item.datatype, item.key, item.chunkIndex) == ESP_OK)) {
break;
}{...}
}{...}
if ((it == last) && (item.datatype == ItemType::BLOB_IDX)) {
/* ... */
for (it = begin(); it != last; ++it) {
if ((it->state() != Page::PageState::FREEING) &&
(it->eraseItem(item.nsIndex, ItemType::BLOB, item.key, item.chunkIndex) == ESP_OK)) {
break;
}{...}
}{...}
}{...}
}{...}
for (auto it = begin(); it!= end(); ++it) {
if (it->state() == Page::PageState::FREEING) {
Page* newPage = &mPageList.back();
if (newPage->state() == Page::PageState::ACTIVE) {
auto err = newPage->erase();
if (err != ESP_OK) {
return err;
}{...}
mPageList.erase(newPage);
mFreePageList.push_back(newPage);
}{...}
auto err = activatePage();
if (err != ESP_OK) {
return err;
}{...}
newPage = &mPageList.back();
err = it->copyItems(*newPage);
if (err != ESP_OK && err != ESP_ERR_NVS_NOT_FOUND) {
return err;
}{...}
err = it->erase();
if (err != ESP_OK) {
return err;
}{...}
Page* p = static_cast<Page*>(it);
mPageList.erase(it);
mFreePageList.push_back(p);
break;
}{...}
}{...}
if (mFreePageList.empty()) {
return ESP_ERR_NVS_NO_FREE_PAGES;
}{...}
return ESP_OK;
}{ ... }
esp_err_t PageManager::requestNewPage()
{
if (mFreePageList.empty()) {
return ESP_ERR_NVS_INVALID_STATE;
}{...}
if (mFreePageList.size() >= 2) {
return activatePage();
}{...}
TPageListIterator maxUnusedItemsPageIt;
size_t maxUnusedItems = 0;
for (auto it = begin(); it != end(); ++it) {
auto unused = Page::ENTRY_COUNT - it->getUsedEntryCount();
if (unused > maxUnusedItems) {
maxUnusedItemsPageIt = it;
maxUnusedItems = unused;
}{...}
}{...}
if (maxUnusedItems == 0) {
return ESP_ERR_NVS_NOT_ENOUGH_SPACE;
}{...}
esp_err_t err = activatePage();
if (err != ESP_OK) {
return err;
}{...}
Page* newPage = &mPageList.back();
Page* erasedPage = maxUnusedItemsPageIt;
#ifndef NDEBUG
size_t usedEntries = erasedPage->getUsedEntryCount();
#endif
err = erasedPage->markFreeing();
if (err != ESP_OK) {
return err;
}{...}
err = erasedPage->copyItems(*newPage);
if (err != ESP_OK && err != ESP_ERR_NVS_NOT_FOUND) {
return err;
}{...}
err = erasedPage->erase();
if (err != ESP_OK) {
return err;
}{...}
#ifndef NDEBUG
NVS_ASSERT_OR_RETURN(usedEntries == newPage->getUsedEntryCount(), ESP_FAIL);
#endif
mPageList.erase(maxUnusedItemsPageIt);
mFreePageList.push_back(erasedPage);
return ESP_OK;
}{ ... }
esp_err_t PageManager::activatePage()
{
if (mFreePageList.empty()) {
return ESP_ERR_NVS_NOT_ENOUGH_SPACE;
}{...}
Page* p = &mFreePageList.front();
if (p->state() == Page::PageState::CORRUPT) {
auto err = p->erase();
if (err != ESP_OK) {
return err;
}{...}
}{...}
mFreePageList.pop_front();
mPageList.push_back(p);
p->setSeqNumber(mSeqNumber);
++mSeqNumber;
return ESP_OK;
}{ ... }
esp_err_t PageManager::fillStats(nvs_stats_t& nvsStats)
{
nvsStats.used_entries = 0;
nvsStats.free_entries = 0;
nvsStats.available_entries = 0;
nvsStats.total_entries = 0;
esp_err_t err = ESP_OK;
for (auto p = mPageList.begin(); p != mPageList.end(); ++p) {
err = p->calcEntries(nvsStats);
if (err != ESP_OK) {
return err;
}{...}
}{...}
nvsStats.total_entries += mFreePageList.size() * Page::ENTRY_COUNT;
nvsStats.free_entries += mFreePageList.size() * Page::ENTRY_COUNT;
nvsStats.available_entries = (nvsStats.free_entries >= Page::ENTRY_COUNT) ? nvsStats.free_entries - Page::ENTRY_COUNT : 0;
return err;
}{ ... }
}{...}