nvs: remove search cache at page level

Since read cache was introduced at page level, search cache became
useless in terms of reducing the number of flash read operations.
In addition to that, search cache used an assumption that if pointers to
keys are identical, the keys are also identical, which was proven wrong
by applications which generate key names dynamically.

This change removes CachedFindInfo, and all its uses. This is done at
expense of a small extra number of CPU operations (looking up a value in
the read cache is slightly more expensive) but no extra flash read
operations.

Ref TW12505
Ref https://github.com/espressif/arduino-esp32/issues/365
This commit is contained in:
Ivan Grokhotkov 2017-05-12 12:18:08 +08:00
parent 15a6145961
commit bf01525fc1
3 changed files with 21 additions and 59 deletions

View File

@ -296,9 +296,6 @@ esp_err_t Page::eraseItem(uint8_t nsIndex, ItemType datatype, const char* key)
if (rc != ESP_OK) { if (rc != ESP_OK) {
return rc; return rc;
} }
if (CachedFindInfo(nsIndex, datatype, key) == mFindInfo) {
invalidateCache();
}
return eraseEntryAndSpan(index); return eraseEntryAndSpan(index);
} }
@ -386,10 +383,6 @@ esp_err_t Page::moveItem(Page& other)
return ESP_ERR_NVS_NOT_FOUND; return ESP_ERR_NVS_NOT_FOUND;
} }
if (mFindInfo.itemIndex() == mFirstUsedEntry) {
invalidateCache();
}
if (other.mState == PageState::UNINITIALIZED) { if (other.mState == PageState::UNINITIALIZED) {
auto err = other.initialize(); auto err = other.initialize();
if (err != ESP_OK) { if (err != ESP_OK) {
@ -608,7 +601,6 @@ esp_err_t Page::initialize()
mNextFreeEntry = 0; mNextFreeEntry = 0;
std::fill_n(mEntryTable.data(), mEntryTable.byteSize() / sizeof(uint32_t), 0xffffffff); std::fill_n(mEntryTable.data(), mEntryTable.byteSize() / sizeof(uint32_t), 0xffffffff);
invalidateCache();
return ESP_OK; return ESP_OK;
} }
@ -685,11 +677,6 @@ esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key, si
return ESP_ERR_NVS_NOT_FOUND; return ESP_ERR_NVS_NOT_FOUND;
} }
CachedFindInfo findInfo(nsIndex, datatype, key);
if (mFindInfo == findInfo) {
findBeginIndex = mFindInfo.itemIndex();
}
size_t start = mFirstUsedEntry; size_t start = mFirstUsedEntry;
if (findBeginIndex > mFirstUsedEntry && findBeginIndex < ENTRY_COUNT) { if (findBeginIndex > mFirstUsedEntry && findBeginIndex < ENTRY_COUNT) {
start = findBeginIndex; start = findBeginIndex;
@ -745,8 +732,6 @@ esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key, si
} }
itemIndex = i; itemIndex = i;
findInfo.setItemIndex(static_cast<uint32_t>(itemIndex));
mFindInfo = findInfo;
return ESP_OK; return ESP_OK;
} }
@ -806,12 +791,6 @@ esp_err_t Page::markFull()
return alterPageState(PageState::FULL); return alterPageState(PageState::FULL);
} }
void Page::invalidateCache()
{
mFindInfo = CachedFindInfo();
}
const char* Page::pageStateToName(PageState ps) const char* Page::pageStateToName(PageState ps)
{ {
switch (ps) { switch (ps) {

View File

@ -28,38 +28,6 @@
namespace nvs namespace nvs
{ {
class CachedFindInfo
{
public:
CachedFindInfo() { }
CachedFindInfo(uint8_t nsIndex, ItemType type, const char* key) :
mKeyPtr(key),
mNsIndex(nsIndex),
mType(type)
{
}
bool operator==(const CachedFindInfo& other) const
{
return mKeyPtr != nullptr && mKeyPtr == other.mKeyPtr && mType == other.mType && mNsIndex == other.mNsIndex;
}
void setItemIndex(uint32_t index)
{
mItemIndex = index;
}
uint32_t itemIndex() const
{
return mItemIndex;
}
protected:
uint32_t mItemIndex = 0;
const char* mKeyPtr = nullptr;
uint8_t mNsIndex = 0;
ItemType mType;
};
class Page : public intrusive_list_node<Page> class Page : public intrusive_list_node<Page>
{ {
@ -161,8 +129,6 @@ public:
esp_err_t erase(); esp_err_t erase();
void invalidateCache();
void debugDump() const; void debugDump() const;
protected: protected:
@ -235,7 +201,6 @@ protected:
uint16_t mUsedEntryCount = 0; uint16_t mUsedEntryCount = 0;
uint16_t mErasedEntryCount = 0; uint16_t mErasedEntryCount = 0;
CachedFindInfo mFindInfo;
HashList mHashList; HashList mHashList;
static const uint32_t HEADER_OFFSET = 0; static const uint32_t HEADER_OFFSET = 0;

View File

@ -18,6 +18,9 @@
#include <sstream> #include <sstream>
#include <iostream> #include <iostream>
#define TEST_ESP_ERR(rc, res) CHECK((rc) == (res))
#define TEST_ESP_OK(rc) CHECK((rc) == ESP_OK)
using namespace std; using namespace std;
using namespace nvs; using namespace nvs;
@ -209,6 +212,24 @@ TEST_CASE("can write and read variable length data", "[nvs]")
CHECK(memcmp(buf, str, strlen(str)) == 0); CHECK(memcmp(buf, str, strlen(str)) == 0);
} }
TEST_CASE("different key names are distinguished even if the pointer is the same", "[nvs]")
{
SpiFlashEmulator emu(1);
Page page;
TEST_ESP_OK(page.load(0));
TEST_ESP_OK(page.writeItem(1, "i1", 1));
TEST_ESP_OK(page.writeItem(1, "i2", 2));
int32_t value;
char keyname[10] = {0};
for (int i = 0; i < 2; ++i) {
strncpy(keyname, "i1", sizeof(keyname) - 1);
TEST_ESP_OK(page.readItem(1, keyname, value));
CHECK(value == 1);
strncpy(keyname, "i2", sizeof(keyname) - 1);
TEST_ESP_OK(page.readItem(1, keyname, value));
CHECK(value == 2);
}
}
TEST_CASE("can init PageManager in empty flash", "[nvs]") TEST_CASE("can init PageManager in empty flash", "[nvs]")
{ {
@ -428,9 +449,6 @@ TEST_CASE("can erase items", "[nvs]")
CHECK(storage.readItem(3, "key00222", val) == ESP_ERR_NVS_NOT_FOUND); CHECK(storage.readItem(3, "key00222", val) == ESP_ERR_NVS_NOT_FOUND);
} }
#define TEST_ESP_ERR(rc, res) CHECK((rc) == (res))
#define TEST_ESP_OK(rc) CHECK((rc) == ESP_OK)
TEST_CASE("nvs api tests", "[nvs]") TEST_CASE("nvs api tests", "[nvs]")
{ {
SpiFlashEmulator emu(10); SpiFlashEmulator emu(10);