This change removes the earlier limitation of 1984 bytes for storing data-blobs.
Blobs larger than the sector size are split and stored on multiple sectors.
For this purpose, two new datatypes (multi-page index and multi-page data) are
added for entries stored in the sectors. The underlying read, write, erase and find
operations are modified to support these large blobs. The change is transparent
to users of the library and no special APIs need to be used to store these large
blobs.
- variable length binary data (blob)
.. note::
- String and blob values are currently limited to 1984 bytes. For strings, this includes the null terminator.
+ String values are currently limited to 4000 bytes. This includes the null terminator. Blob values are limited to 508000 bytes or (97.6% of the partition size - 4000) bytes whichever is lower.
Additional types, such as ``float`` and ``double`` may be added later.
Structure of entry
^^^^^^^^^^^^^^^^^^
-For values of primitive types (currently integers from 1 to 8 bytes long), entry holds one key-value pair. For string and blob types, entry holds part of the whole key-value pair. In case when a key-value pair spans multiple entries, all entries are stored in the same page.
+For values of primitive types (currently integers from 1 to 8 bytes long), entry holds one key-value pair. For string and blob types, entry holds part of the whole key-value pair. For strings, in case when a key-value pair spans multiple entries, all entries are stored in the same page. Blobs are allowed to span over multiple pages by dividing them into smaller chunks. For the purpose tracking these chunks, an additional fixed length metadata entry is stored called "blob index" entry. Earlier format of blobs are still supported (can be read and modified). However, once the blobs are modified, they are stored using the new format.
::
- +--------+----------+----------+---------+-----------+---------------+----------+
- | NS (1) | Type (1) | Span (1) | Rsv (1) | CRC32 (4) | Key (16) | Data (8) |
- +--------+----------+----------+---------+-----------+---------------+----------+
+ +--------+----------+----------+----------------+-----------+---------------+----------+
+ | NS (1) | Type (1) | Span (1) | ChunkIndex (1) | CRC32 (4) | Key (16) | Data (8) |
+ +--------+----------+----------+----------------+-----------+---------------+----------+
- +--------------------------------+
- +-> Fixed length: | Data (8) |
- | +--------------------------------+
- Data format ---+
- | +----------+---------+-----------+
- +-> Variable length: | Size (2) | Rsv (2) | CRC32 (4) |
- +----------+---------+-----------+
+ Primitive +--------------------------------+
+ +--------> | Data (8) |
+ | Types +--------------------------------+
+ +-> Fixed length --
+ | | +---------+--------------+---------------+-------+
+ | +--------> | Size(4) | ChunkCount(1)| ChunkStart(1) | Rsv(2)|
+ Data format ---+ Blob Index +---------+--------------+---------------+-------+
+ |
+ | +----------+---------+-----------+
+ +-> Variable length --> | Size (2) | Rsv (2) | CRC32 (4) |
+ (Strings, Blob Data) +----------+---------+-----------+
Individual fields in entry structure have the following meanings:
Span
Number of entries used by this key-value pair. For integer types, this is equal to 1. For strings and blobs this depends on value length.
-Rsv
- Unused field, should be ``0xff``.
+ChunkIndex
+ Used to store index of the blob-data chunk for blob types. For other types, this should be ``0xff``.
CRC32
Checksum calculated over all the bytes in this entry, except for the CRC32 field itself.
Zero-terminated ASCII string containing key name. Maximum string length is 15 bytes, excluding zero terminator.
Data
- For integer types, this field contains the value itself. If the value itself is shorter than 8 bytes it is padded to the right, with unused bytes filled with ``0xff``. For string and blob values, these 8 bytes hold additional data about the value, described next:
+ For integer types, this field contains the value itself. If the value itself is shorter than 8 bytes it is padded to the right, with unused bytes filled with ``0xff``.
-Size
- (Only for strings and blobs.) Size, in bytes, of actual data. For strings, this includes zero terminator.
+ For "blob index" entry, these 8 bytes hold the following information about data-chunks:
-CRC32
- (Only for strings and blobs.) Checksum calculated over all bytes of data.
+ - Size
+ (Only for blob index.) Size, in bytes, of complete blob data.
+
+ - ChunkCount
+ (Only for blob index.) Total number of blob-data chunks into which the blob was divided during storage.
+
+ - ChunkStart
+ (Only for blob index.) ChunkIndex of the first blob-data chunk of this blob. Subsequent chunks have chunkIndex incrementely allocated (step of 1).
+
+ For string and blob data chunks, these 8 bytes hold additional data about the value, described next:
+
+ - Size
+ (Only for strings and blobs.) Size, in bytes, of actual data. For strings, this includes zero terminator.
+
+ - CRC32
+ (Only for strings and blobs.) Checksum calculated over all bytes of data.
Variable length values (strings and blobs) are written into subsequent entries, 32 bytes per entry. `Span` field of the first entry indicates how many entries are used.
To reduce the number of reads performed from flash memory, each member of Page class maintains a list of pairs: (item index; item hash). This list makes searches much quicker. Instead of iterating over all entries, reading them from flash one at a time, ``Page::findItem`` first performs search for item hash in the hash list. This gives the item index within the page, if such an item exists. Due to a hash collision it is possible that a different item will be found. This is handled by falling back to iteration over items in flash.
-Each node in hash list contains a 24-bit hash and 8-bit item index. Hash is calculated based on item namespace and key name. CRC32 is used for calculation, result is truncated to 24 bits. To reduce overhead of storing 32-bit entries in a linked list, list is implemented as a doubly-linked list of arrays. Each array holds 29 entries, for the total size of 128 bytes, together with linked list pointers and 32-bit count field. Minimal amount of extra RAM useage per page is therefore 128 bytes, maximum is 640 bytes.
+Each node in hash list contains a 24-bit hash and 8-bit item index. Hash is calculated based on item namespace, key name and ChunkIndex. CRC32 is used for calculation, result is truncated to 24 bits. To reduce overhead of storing 32-bit entries in a linked list, list is implemented as a doubly-linked list of arrays. Each array holds 29 entries, for the total size of 128 bytes, together with linked list pointers and 32-bit count field. Minimal amount of extra RAM useage per page is therefore 128 bytes, maximum is 640 bytes.
class Page(object):
PAGE_PARAMS = {
"max_size": 4096,
- "max_blob_size": 1984,
+ "max_blob_size": 4000,
"max_entries": 126
}
I32 = 0x14
SZ = 0x21
BLOB = 0x41
+ BLOB_DATA = 0x42
+ BLOB_IDX = 0x48
# Few Page constants
HEADER_SIZE = 32
BITMAPARRAY_SIZE_IN_BYTES = 32
FIRST_ENTRY_OFFSET = 64
SINGLE_ENTRY_SIZE = 32
+ CHUNK_ANY = 0xFF
+ ACTIVE = 0xFFFFFFFE
+ FULL = 0xFFFFFFFC
def __init__(self, page_num):
self.entry_num = 0
self.set_header(page_num)
def set_header(self, page_num):
+ global page_header
+
# set page state to active
page_header= bytearray(b'\xff')*32
- page_state_active_seq = 0xFFFFFFFE
+ page_state_active_seq = Page.ACTIVE
page_header[0:4] = struct.pack('<I', page_state_active_seq)
# set page sequence number
page_header[4:8] = struct.pack('<I', page_num)
start_idx = data_offset
end_idx = data_offset + len(data)
self.page_buf[start_idx:end_idx] = data
-
# Set bitmap array for entries in current page
for i in range(0, entrycount):
self.write_bitmaparray()
self.entry_num += 1
+ def set_crc_header(self, entry_struct):
+ crc_data = bytearray(28)
+ crc_data[0:4] = entry_struct[0:4]
+ crc_data[4:28] = entry_struct[8:32]
+ crc = zlib.crc32(buffer(crc_data), 0xFFFFFFFF)
+ entry_struct[4:8] = struct.pack('<I', crc & 0xFFFFFFFF)
+ return entry_struct
+
+ def write_varlen_binary_data(self, entry_struct, ns_index, key, data, data_size, total_entry_count, nvs_obj):
+ chunk_start = 0
+ chunk_count = 0
+ chunk_index = Page.CHUNK_ANY
+ offset = 0
+ remaining_size = data_size
+ tailroom = None
+
+ while True:
+ chunk_size = 0
+
+ # Get the size available in current page
+ if self.entry_num < (Page.PAGE_PARAMS["max_entries"] - 1):
+ tailroom = (Page.PAGE_PARAMS["max_entries"] - self.entry_num - 1) * Page.SINGLE_ENTRY_SIZE
+
+ # Split the binary data into two and store a chunk of available size onto curr page
+ if tailroom < remaining_size:
+ chunk_size = tailroom
+ else:
+ chunk_size = remaining_size
+
+ remaining_size = remaining_size - chunk_size
+
+ # Change type of data to BLOB_DATA
+ entry_struct[1] = Page.BLOB_DATA
+
+ # Calculate no. of entries data chunk will require
+ datachunk_rounded_size = (chunk_size + 31) & ~31
+ datachunk_entry_count = datachunk_rounded_size / 32
+ datachunk_total_entry_count = datachunk_entry_count + 1 # +1 for the entry header
+
+ # Set Span
+ entry_struct[2] = datachunk_total_entry_count
+
+ # Update the chunkIndex
+ chunk_index = chunk_start + chunk_count
+ entry_struct[3] = chunk_index
+
+ # Set data chunk
+ data_chunk = data[offset:offset + chunk_size]
+
+ # Compute CRC of data chunk
+ entry_struct[24:26] = struct.pack('<H', chunk_size)
+ crc = zlib.crc32(data_chunk, 0xFFFFFFFF)
+ entry_struct[28:32] = struct.pack('<I', crc & 0xFFFFFFFF)
+
+ # compute crc of entry header
+ entry_struct = self.set_crc_header(entry_struct)
+
+ # write entry header
+ self.write_entry_to_buf(entry_struct, 1)
+ # write actual data
+ self.write_entry_to_buf(data_chunk, datachunk_entry_count)
+
+ chunk_count = chunk_count + 1
+
+ if remaining_size or (tailroom - chunk_size) < Page.SINGLE_ENTRY_SIZE:
+ if page_header[0:4] != Page.FULL:
+ page_state_full_seq = Page.FULL
+ page_header[0:4] = struct.pack('<I', page_state_full_seq)
+ nvs_obj.create_new_page()
+ self = nvs_obj.cur_page
+
+ offset = offset + chunk_size
+
+ # All chunks are stored, now store the index
+ if not remaining_size:
+ # change type of data to BLOB_IDX
+ entry_struct[1] = Page.BLOB_IDX
+
+ # Set Span
+ entry_struct[2] = 1
+
+ # Update the chunkIndex
+ chunk_index = Page.CHUNK_ANY
+ entry_struct[3] = chunk_index
+
+ entry_struct[24:28] = struct.pack('<I', data_size)
+ entry_struct[28] = chunk_count
+ entry_struct[29] = chunk_start
+
+ # compute crc of entry header
+ entry_struct = self.set_crc_header(entry_struct)
+
+ # write entry header
+ self.write_entry_to_buf(entry_struct, 1)
+ break
+
+
+
+ return entry_struct
+
+
+
"""
Low-level function to write variable length data into page buffer. Data should be formatted
according to encoding specified.
"""
- def write_varlen_data(self, key, data, encoding, ns_index):
+ def write_varlen_data(self, key, data, encoding, ns_index, nvs_obj):
+ # Set size of data
datalen = len(data)
- if datalen > Page.PAGE_PARAMS["max_blob_size"]:
- raise InputError("%s: Size exceeds max allowed length." % key)
+
+ if encoding == "string":
+ if datalen > Page.PAGE_PARAMS["max_blob_size"]:
+ raise InputError("%s: Size exceeds max allowed length." % key)
# Calculate no. of entries data will require
rounded_size = (datalen + 31) & ~31
total_entry_count = data_entry_count + 1 # +1 for the entry header
# Check if page is already full and new page is needed to be created right away
- if (self.entry_num + total_entry_count) >= Page.PAGE_PARAMS["max_entries"]:
- raise PageFullError()
+ if encoding == "string":
+ if (self.entry_num + total_entry_count) >= Page.PAGE_PARAMS["max_entries"]:
+ raise PageFullError()
# Entry header
entry_struct = bytearray('\xff')*32
- entry_struct[0] = ns_index # namespace index
- entry_struct[2] = data_entry_count + 1 # Span
+ # Set Namespace Index
+ entry_struct[0] = ns_index
+ # Set Span
+ if encoding == "string":
+ entry_struct[2] = data_entry_count + 1
+ # Set Chunk Index
+ chunk_index = Page.CHUNK_ANY
+ entry_struct[3] = chunk_index
# set key
key_array = bytearray('\x00')*16
elif encoding in ["hex2bin", "binary", "base64"]:
entry_struct[1] = Page.BLOB
- # compute CRC of data
- entry_struct[24:26] = struct.pack('<H', datalen)
- crc = zlib.crc32(data, 0xFFFFFFFF)
- entry_struct[28:32] = struct.pack('<I', crc & 0xFFFFFFFF)
+ if encoding == "binary" or encoding == "hex2bin" or encoding == "base64":
+ entry_struct = self.write_varlen_binary_data(entry_struct,ns_index,key,data,\
+ datalen,total_entry_count,nvs_obj)
+ else:
+ # compute CRC of data
+ entry_struct[24:26] = struct.pack('<H', datalen)
+ crc = zlib.crc32(data, 0xFFFFFFFF)
+ entry_struct[28:32] = struct.pack('<I', crc & 0xFFFFFFFF)
- # compute crc of entry header
- crc_data = bytearray(28)
- crc_data[0:4] = entry_struct[0:4]
- crc_data[4:28] = entry_struct[8:32]
- crc = zlib.crc32(buffer(crc_data), 0xFFFFFFFF)
- entry_struct[4:8] = struct.pack('<I', crc & 0xFFFFFFFF)
+ # compute crc of entry header
+ entry_struct = self.set_crc_header(entry_struct)
+
+ # write entry header
+ self.write_entry_to_buf(entry_struct, 1)
+ # write actual data
+ self.write_entry_to_buf(data, data_entry_count)
- # write entry header
- self.write_entry_to_buf(entry_struct, 1)
- # write actual data
- self.write_entry_to_buf(data, data_entry_count)
""" Low-level function to write data of primitive type into page buffer. """
def write_primitive_data(self, key, data, encoding, ns_index):
entry_struct = bytearray('\xff')*32
entry_struct[0] = ns_index # namespace index
entry_struct[2] = 0x01 # Span
+ chunk_index = Page.CHUNK_ANY
+ entry_struct[3] = chunk_index
# write key
key_array = bytearray('\x00')*16
primitive_encodings = ["u8", "i8", "u16", "u32", "i32"]
if encoding in varlen_encodings:
try:
- self.cur_page.write_varlen_data(key, value, encoding, self.namespace_idx)
+ self.cur_page.write_varlen_data(key, value, encoding, self.namespace_idx, self)
except PageFullError:
new_page = self.create_new_page()
- new_page.write_varlen_data(key, value, encoding, self.namespace_idx)
+ new_page.write_varlen_data(key, value, encoding, self.namespace_idx, self)
pass
elif encoding in primitive_encodings:
try:
-\ 1\ 2\ 3\ 4\ 5\ 6\a\b «Íï
\ No newline at end of file
+start0000000000000000000000start0123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef00000000000000000123456789abcdef0000000000000000end00000000000000000000000000end
\ No newline at end of file
{
return mSize == 0;
}
-
-
+
void clear()
{
while (mFirst) {
}
}
+ void clearAndFreeNodes()
+ {
+ while (mFirst) {
+ auto tmp = mFirst;
+ erase(mFirst);
+ delete tmp;
+ }
+ }
+
+
protected:
T* mFirst = nullptr;
T* mLast = nullptr;
return ESP_OK;
}
-
+
esp_err_t Page::writeEntryData(const uint8_t* data, size_t size)
{
assert(size % ENTRY_SIZE == 0);
assert(mNextFreeEntry != INVALID_ENTRY);
assert(mFirstUsedEntry != INVALID_ENTRY);
const uint16_t count = size / ENTRY_SIZE;
-
+
const uint8_t* buf = data;
-
+
#ifdef ESP_PLATFORM
/* On the ESP32, data can come from DROM, which is not accessible by spi_flash_write
* function. To work around this, we copy the data to heap if it came from DROM.
return ESP_OK;
}
-esp_err_t Page::writeItem(uint8_t nsIndex, ItemType datatype, const char* key, const void* data, size_t dataSize)
+esp_err_t Page::writeItem(uint8_t nsIndex, ItemType datatype, const char* key, const void* data, size_t dataSize, uint8_t chunkIdx)
{
Item item;
esp_err_t err;
-
+
if (mState == PageState::INVALID) {
return ESP_ERR_NVS_INVALID_STATE;
}
-
+
if (mState == PageState::UNINITIALIZED) {
err = initialize();
if (err != ESP_OK) {
if (keySize > Item::MAX_KEY_LENGTH) {
return ESP_ERR_NVS_KEY_TOO_LONG;
}
-
- if (dataSize > Page::BLOB_MAX_SIZE) {
+
+ if (dataSize > Page::CHUNK_MAX_SIZE) {
return ESP_ERR_NVS_VALUE_TOO_LONG;
}
size_t totalSize = ENTRY_SIZE;
size_t entriesCount = 1;
- if (datatype == ItemType::SZ || datatype == ItemType::BLOB) {
+ if (isVariableLengthType(datatype)) {
size_t roundedSize = (dataSize + ENTRY_SIZE - 1) & ~(ENTRY_SIZE - 1);
totalSize += roundedSize;
entriesCount += roundedSize / ENTRY_SIZE;
}
// primitive types should fit into one entry
- assert(totalSize == ENTRY_SIZE || datatype == ItemType::BLOB || datatype == ItemType::SZ);
+ assert(totalSize == ENTRY_SIZE ||
+ isVariableLengthType(datatype));
if (mNextFreeEntry == INVALID_ENTRY || mNextFreeEntry + entriesCount > ENTRY_COUNT) {
// page will not fit this amount of data
// write first item
size_t span = (totalSize + ENTRY_SIZE - 1) / ENTRY_SIZE;
- item = Item(nsIndex, datatype, span, key);
+ item = Item(nsIndex, datatype, span, key, chunkIdx);
mHashList.insert(item, mNextFreeEntry);
- if (datatype != ItemType::SZ && datatype != ItemType::BLOB) {
+ if (!isVariableLengthType(datatype)) {
memcpy(item.data, data, dataSize);
item.crc32 = item.calculateCrc32();
err = writeEntry(item);
const uint8_t* src = reinterpret_cast<const uint8_t*>(data);
item.varLength.dataCrc32 = Item::calculateCrc32(src, dataSize);
item.varLength.dataSize = dataSize;
- item.varLength.reserved2 = 0xffff;
+ item.varLength.reserved = 0xffff;
item.crc32 = item.calculateCrc32();
err = writeEntry(item);
if (err != ESP_OK) {
return err;
}
}
-
+
size_t tail = dataSize - left;
if (tail > 0) {
std::fill_n(item.rawData, ENTRY_SIZE / 4, 0xffffffff);
return err;
}
}
-
+
}
return ESP_OK;
}
-esp_err_t Page::readItem(uint8_t nsIndex, ItemType datatype, const char* key, void* data, size_t dataSize)
+esp_err_t Page::readItem(uint8_t nsIndex, ItemType datatype, const char* key, void* data, size_t dataSize, uint8_t chunkIdx, VerOffset chunkStart)
{
size_t index = 0;
Item item;
-
+
if (mState == PageState::INVALID) {
return ESP_ERR_NVS_INVALID_STATE;
}
-
- esp_err_t rc = findItem(nsIndex, datatype, key, index, item);
+
+ esp_err_t rc = findItem(nsIndex, datatype, key, index, item, chunkIdx, chunkStart);
if (rc != ESP_OK) {
return rc;
}
- if (datatype != ItemType::SZ && datatype != ItemType::BLOB) {
+ if (!isVariableLengthType(datatype)) {
if (dataSize != getAlignmentForType(datatype)) {
return ESP_ERR_NVS_TYPE_MISMATCH;
}
return ESP_OK;
}
-esp_err_t Page::eraseItem(uint8_t nsIndex, ItemType datatype, const char* key)
+esp_err_t Page::eraseItem(uint8_t nsIndex, ItemType datatype, const char* key, uint8_t chunkIdx, VerOffset chunkStart)
{
size_t index = 0;
Item item;
- esp_err_t rc = findItem(nsIndex, datatype, key, index, item);
+ esp_err_t rc = findItem(nsIndex, datatype, key, index, item, chunkIdx, chunkStart);
if (rc != ESP_OK) {
return rc;
}
return eraseEntryAndSpan(index);
}
-esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key)
+esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key, uint8_t chunkIdx, VerOffset chunkStart)
{
size_t index = 0;
Item item;
- return findItem(nsIndex, datatype, key, index, item);
+ return findItem(nsIndex, datatype, key, index, item, chunkIdx, chunkStart);
}
esp_err_t Page::eraseEntryAndSpan(size_t index)
mState = PageState::INVALID;
return err;
}
-
+
if (item.crc32 != item.calculateCrc32()) {
err = eraseEntryAndSpan(i);
if (err != ESP_OK) {
}
mHashList.insert(item, i);
-
+
// search for potential duplicate item
size_t duplicateIndex = mHashList.find(0, item);
-
- if (item.datatype == ItemType::BLOB || item.datatype == ItemType::SZ) {
+
+ if (isVariableLengthType(item.datatype)) {
span = item.span;
bool needErase = false;
for (size_t j = i; j < i + span; ++j) {
continue;
}
}
-
+
+ /* Note that logic for duplicate detections works fine even
+ * when old-format blob is present along with new-format blob-index
+ * for same key on active page. Since datatype is not used in hash calculation,
+ * old-format blob will be removed.*/
if (duplicateIndex < i) {
eraseEntryAndSpan(duplicateIndex);
}
size_t span = item.span;
- if (item.datatype == ItemType::BLOB || item.datatype == ItemType::SZ) {
+ if (isVariableLengthType(item.datatype)) {
for (size_t j = i + 1; j < i + span; ++j) {
if (mEntryTable.get(j) != EntryState::WRITTEN) {
eraseEntryAndSpan(i);
return ESP_OK;
}
-esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key, size_t &itemIndex, Item& item)
+esp_err_t Page::findItem(uint8_t nsIndex, ItemType datatype, const char* key, size_t &itemIndex, Item& item, uint8_t chunkIdx, VerOffset chunkStart)
{
if (mState == PageState::CORRUPT || mState == PageState::INVALID || mState == PageState::UNINITIALIZED) {
return ESP_ERR_NVS_NOT_FOUND;
}
-
+
size_t findBeginIndex = itemIndex;
if (findBeginIndex >= ENTRY_COUNT) {
return ESP_ERR_NVS_NOT_FOUND;
}
if (nsIndex != NS_ANY && datatype != ItemType::ANY && key != NULL) {
- size_t cachedIndex = mHashList.find(start, Item(nsIndex, datatype, 0, key));
+ size_t cachedIndex = mHashList.find(start, Item(nsIndex, datatype, 0, key, chunkIdx));
if (cachedIndex < ENTRY_COUNT) {
start = cachedIndex;
} else {
continue;
}
- if (item.datatype == ItemType::BLOB || item.datatype == ItemType::SZ) {
+ if (isVariableLengthType(item.datatype)) {
next = i + item.span;
}
if (key != nullptr && strncmp(key, item.key, Item::MAX_KEY_LENGTH) != 0) {
continue;
}
+ /* For blob data, chunkIndex should match*/
+ if (chunkIdx != CHUNK_ANY
+ && datatype == ItemType::BLOB_DATA
+ && item.chunkIndex != chunkIdx) {
+ continue;
+ }
+ /* Blob-index will match the <ns,key> with blob data.
+ * Skip data chunks when searching for blob index*/
+ if (datatype == ItemType::BLOB_IDX
+ && item.chunkIndex != CHUNK_ANY) {
+ continue;
+ }
+ /* Match the version for blob-index*/
+ if (datatype == ItemType::BLOB_IDX
+ && chunkStart != VerOffset::VER_ANY
+ && item.blobIndex.chunkStart != chunkStart) {
+ continue;
+ }
+
if (datatype != ItemType::ANY && item.datatype != datatype) {
+ if (key == nullptr && nsIndex == NS_ANY && chunkIdx == CHUNK_ANY) {
+ continue; // continue for bruteforce search on blob indices.
+ }
return ESP_ERR_NVS_TYPE_MISMATCH;
}
}
return alterPageState(PageState::FULL);
}
-
+
+size_t Page::getVarDataTailroom() const
+{
+ if (mState == PageState::UNINITIALIZED) {
+ return CHUNK_MAX_SIZE;
+ } else if (mState == PageState::FULL) {
+ return 0;
+ }
+ /* Skip one entry for header*/
+ return ((mNextFreeEntry < (ENTRY_COUNT-1)) ? ((ENTRY_COUNT - mNextFreeEntry - 1) * ENTRY_SIZE): 0);
+}
+
const char* Page::pageStateToName(PageState ps)
{
switch (ps) {
case PageState::CORRUPT:
return "CORRUPT";
-
+
case PageState::ACTIVE:
return "ACTIVE";
-
+
case PageState::FREEING:
return "FREEING";
-
+
case PageState::FULL:
return "FULL";
-
+
case PageState::INVALID:
return "INVALID";
-
+
case PageState::UNINITIALIZED:
return "UNINITIALIZED";
-
+
default:
assert(0 && "invalid state value");
return "";
Item item;
readEntry(i, item);
if (skip == 0) {
- printf("W ns=%2u type=%2u span=%3u key=\"%s\" len=%d\n", item.nsIndex, static_cast<unsigned>(item.datatype), item.span, item.key, (item.span != 1)?((int)item.varLength.dataSize):-1);
+ printf("W ns=%2u type=%2u span=%3u key=\"%s\" chunkIdx=%d len=%d\n", item.nsIndex, static_cast<unsigned>(item.datatype), item.span, item.key, item.chunkIndex, (item.span != 1)?((int)item.varLength.dataSize):-1);
if (item.span > 0 && item.span <= ENTRY_COUNT - i) {
skip = item.span - 1;
} else {
static const size_t ENTRY_SIZE = 32;
static const size_t ENTRY_COUNT = 126;
static const uint32_t INVALID_ENTRY = 0xffffffff;
-
- static const size_t BLOB_MAX_SIZE = ENTRY_SIZE * (ENTRY_COUNT / 2 - 1);
+
+ static const size_t CHUNK_MAX_SIZE = ENTRY_SIZE * (ENTRY_COUNT - 1);
static const uint8_t NS_INDEX = 0;
static const uint8_t NS_ANY = 255;
+ static const uint8_t CHUNK_ANY = Item::CHUNK_ANY;
+
enum class PageState : uint32_t {
// All bits set, default state after flash erase. Page has not been initialized yet.
UNINITIALIZED = 0xffffffff,
esp_err_t setSeqNumber(uint32_t seqNumber);
- esp_err_t writeItem(uint8_t nsIndex, ItemType datatype, const char* key, const void* data, size_t dataSize);
+ esp_err_t writeItem(uint8_t nsIndex, ItemType datatype, const char* key, const void* data, size_t dataSize, uint8_t chunkIdx = CHUNK_ANY);
- esp_err_t readItem(uint8_t nsIndex, ItemType datatype, const char* key, void* data, size_t dataSize);
+ esp_err_t readItem(uint8_t nsIndex, ItemType datatype, const char* key, void* data, size_t dataSize, uint8_t chunkIdx = CHUNK_ANY, VerOffset chunkStart = VerOffset::VER_ANY);
- esp_err_t eraseItem(uint8_t nsIndex, ItemType datatype, const char* key);
+ esp_err_t eraseItem(uint8_t nsIndex, ItemType datatype, const char* key, uint8_t chunkIdx = CHUNK_ANY, VerOffset chunkStart = VerOffset::VER_ANY);
- esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key);
+ esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key, uint8_t chunkIdx = CHUNK_ANY, VerOffset chunkStart = VerOffset::VER_ANY);
- esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key, size_t &itemIndex, Item& item);
+ esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key, size_t &itemIndex, Item& item, uint8_t chunkIdx = CHUNK_ANY, VerOffset chunkStart = VerOffset::VER_ANY);
template<typename T>
esp_err_t writeItem(uint8_t nsIndex, const char* key, const T& value)
{
return mErasedEntryCount;
}
-
+ size_t getVarDataTailroom() const ;
esp_err_t markFull();
if (lastItemIndex != SIZE_MAX) {
auto last = PageManager::TPageListIterator(&lastPage);
- for (auto it = begin(); it != last; ++it) {
+ TPageListIterator it;
+
+ for (it = begin(); it != last; ++it) {
if ((it->state() != Page::PageState::FREEING) &&
- (it->eraseItem(item.nsIndex, item.datatype, item.key) == ESP_OK)) {
+ (it->eraseItem(item.nsIndex, item.datatype, item.key, item.chunkIndex) == ESP_OK)) {
break;
}
}
+ if ((it == last) && (item.datatype == ItemType::BLOB_IDX)) {
+ /* Rare case in which the blob was stored using old format, but power went just after writing
+ * blob index during modification. Loop again and delete the old version blob*/
+ 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;
+ }
+ }
+ }
}
// check if power went out while page was being freed
return mPageList.back();
}
+ uint32_t getPageCount() {
+ return mPageCount;
+ }
+
esp_err_t requestNewPage();
esp_err_t fillStats(nvs_stats_t& nvsStats);
void Storage::clearNamespaces()
{
- for (auto it = std::begin(mNamespaces); it != std::end(mNamespaces); ) {
- auto tmp = it;
- ++it;
- mNamespaces.erase(tmp);
- delete static_cast<NamespaceEntry*>(tmp);
+ mNamespaces.clearAndFreeNodes();
+}
+
+void Storage::populateBlobIndices(TBlobIndexList& blobIdxList)
+{
+ for (auto it = mPageManager.begin(); it != mPageManager.end(); ++it) {
+ Page& p = *it;
+ size_t itemIndex = 0;
+ Item item;
+
+ /* If the power went off just after writing a blob index, the duplicate detection
+ * logic in pagemanager will remove the earlier index. So we should never find a
+ * duplicate index at this point */
+
+ while (p.findItem(Page::NS_ANY, ItemType::BLOB_IDX, nullptr, itemIndex, item) == ESP_OK) {
+ BlobIndexNode* entry = new BlobIndexNode;
+
+ item.getKey(entry->key, sizeof(entry->key) - 1);
+ entry->nsIndex = item.nsIndex;
+ entry->chunkStart = item.blobIndex.chunkStart;
+ entry->chunkCount = item.blobIndex.chunkCount;
+
+ blobIdxList.push_back(entry);
+ itemIndex += item.span;
+ }
+ }
+}
+
+void Storage::eraseOrphanDataBlobs(TBlobIndexList& blobIdxList)
+{
+ for (auto it = mPageManager.begin(); it != mPageManager.end(); ++it) {
+ Page& p = *it;
+ size_t itemIndex = 0;
+ Item item;
+ /* Chunks with same <ns,key> and with chunkIndex in the following ranges
+ * belong to same family.
+ * 1) VER_0_OFFSET <= chunkIndex < VER_1_OFFSET-1 => Version0 chunks
+ * 2) VER_1_OFFSET <= chunkIndex < VER_ANY => Version1 chunks
+ */
+ while (p.findItem(Page::NS_ANY, ItemType::BLOB_DATA, nullptr, itemIndex, item) == ESP_OK) {
+
+ auto iter = std::find_if(blobIdxList.begin(),
+ blobIdxList.end(),
+ [=] (const BlobIndexNode& e) -> bool
+ {return (strncmp(item.key, e.key, sizeof(e.key) - 1) == 0)
+ && (item.nsIndex == e.nsIndex)
+ && (item.chunkIndex >= static_cast<uint8_t> (e.chunkStart))
+ && (item.chunkIndex < static_cast<uint8_t> (e.chunkStart) + e.chunkCount);});
+ if (iter == std::end(blobIdxList)) {
+ p.eraseItem(item.nsIndex, item.datatype, item.key, item.chunkIndex);
+ }
+ itemIndex += item.span;
+ }
}
}
mNamespaceUsage.set(0, true);
mNamespaceUsage.set(255, true);
mState = StorageState::ACTIVE;
+
+ // Populate list of multi-page index entries.
+ TBlobIndexList blobIdxList;
+ populateBlobIndices(blobIdxList);
+
+ // Remove the entries for which there is no parent multi-page index.
+ eraseOrphanDataBlobs(blobIdxList);
+
+ // Purge the blob index list
+ blobIdxList.clearAndFreeNodes();
+
#ifndef ESP_PLATFORM
debugCheck();
#endif
return mState == StorageState::ACTIVE;
}
-esp_err_t Storage::findItem(uint8_t nsIndex, ItemType datatype, const char* key, Page* &page, Item& item)
+esp_err_t Storage::findItem(uint8_t nsIndex, ItemType datatype, const char* key, Page* &page, Item& item, uint8_t chunkIdx, VerOffset chunkStart)
{
for (auto it = std::begin(mPageManager); it != std::end(mPageManager); ++it) {
size_t itemIndex = 0;
- auto err = it->findItem(nsIndex, datatype, key, itemIndex, item);
+ auto err = it->findItem(nsIndex, datatype, key, itemIndex, item, chunkIdx, chunkStart);
if (err == ESP_OK) {
page = it;
return ESP_OK;
return ESP_ERR_NVS_NOT_FOUND;
}
+esp_err_t Storage::writeMultiPageBlob(uint8_t nsIndex, const char* key, const void* data, size_t dataSize, VerOffset chunkStart)
+{
+ uint8_t chunkCount = 0;
+ TUsedPageList usedPages;
+ size_t remainingSize = dataSize;
+ size_t offset=0;
+ esp_err_t err = ESP_OK;
+
+ /* Check how much maximum data can be accommodated**/
+ uint32_t max_pages = mPageManager.getPageCount() - 1;
+
+ if(max_pages > (Page::CHUNK_ANY-1)/2) {
+ max_pages = (Page::CHUNK_ANY-1)/2;
+ }
+
+ if (dataSize > max_pages * Page::CHUNK_MAX_SIZE) {
+ return ESP_ERR_NVS_VALUE_TOO_LONG;
+ }
+
+ do {
+ Page& page = getCurrentPage();
+ size_t tailroom = page.getVarDataTailroom();
+ size_t chunkSize =0;
+ if (!chunkCount && tailroom < dataSize && tailroom < Page::CHUNK_MAX_SIZE/10) {
+ /** This is the first chunk and tailroom is too small ***/
+ if (page.state() != Page::PageState::FULL) {
+ err = page.markFull();
+ if (err != ESP_OK) {
+ return err;
+ }
+ }
+ err = mPageManager.requestNewPage();
+ if (err != ESP_OK) {
+ return err;
+ } else {
+ continue;
+ }
+ } else if (!tailroom) {
+ err = ESP_ERR_NVS_NOT_ENOUGH_SPACE;
+ break;
+ }
+
+ /* Split the blob into two and store the chunk of available size onto the current page */
+ assert(tailroom!=0);
+ chunkSize = (remainingSize > tailroom)? tailroom : remainingSize;
+ remainingSize -= chunkSize;
+
+ err = page.writeItem(nsIndex, ItemType::BLOB_DATA, key,
+ static_cast<const uint8_t*> (data) + offset, chunkSize, static_cast<uint8_t> (chunkStart) + chunkCount);
+ chunkCount++;
+ assert(err != ESP_ERR_NVS_PAGE_FULL);
+ if (err != ESP_OK) {
+ break;
+ } else {
+ UsedPageNode* node = new UsedPageNode();
+ node->mPage = &page;
+ usedPages.push_back(node);
+ if (remainingSize || (tailroom - chunkSize) < Page::ENTRY_SIZE) {
+ if (page.state() != Page::PageState::FULL) {
+ err = page.markFull();
+ if (err != ESP_OK) {
+ break;
+ }
+ }
+ err = mPageManager.requestNewPage();
+ if (err != ESP_OK) {
+ break;
+ }
+ }
+ }
+ offset += chunkSize;
+ if (!remainingSize) {
+ /* All pages are stored. Now store the index.*/
+ Item item;
+ item.blobIndex.dataSize = dataSize;
+ item.blobIndex.chunkCount = chunkCount;
+ item.blobIndex.chunkStart = chunkStart;
+
+ err = getCurrentPage().writeItem(nsIndex, ItemType::BLOB_IDX, key, item.data, sizeof(item.data));
+ assert(err != ESP_ERR_NVS_PAGE_FULL);
+ break;
+ }
+ } while (1);
+
+ if (err != ESP_OK) {
+ /* Anything failed, then we should erase all the written chunks*/
+ int ii=0;
+ for (auto it = std::begin(usedPages); it != std::end(usedPages); it++) {
+ it->mPage->eraseItem(nsIndex, ItemType::BLOB_DATA, key, ii++);
+ }
+ usedPages.clearAndFreeNodes();
+ return err;
+ }
+ return ESP_OK;
+}
+
esp_err_t Storage::writeItem(uint8_t nsIndex, ItemType datatype, const char* key, const void* data, size_t dataSize)
{
if (mState != StorageState::ACTIVE) {
Page* findPage = nullptr;
Item item;
- auto err = findItem(nsIndex, datatype, key, findPage, item);
+
+ esp_err_t err;
+ if (datatype == ItemType::BLOB) {
+ err = findItem(nsIndex, ItemType::BLOB_IDX, key, findPage, item);
+ } else {
+ err = findItem(nsIndex, datatype, key, findPage, item);
+ }
+
+
if (err != ESP_OK && err != ESP_ERR_NVS_NOT_FOUND) {
return err;
}
- Page& page = getCurrentPage();
- err = page.writeItem(nsIndex, datatype, key, data, dataSize);
- if (err == ESP_ERR_NVS_PAGE_FULL) {
- if (page.state() != Page::PageState::FULL) {
- err = page.markFull();
- if (err != ESP_OK) {
- return err;
+ if (datatype == ItemType::BLOB) {
+ VerOffset prevStart, nextStart;
+ prevStart = nextStart = VerOffset::VER_0_OFFSET;
+ if (findPage) {
+ if (findPage->state() == Page::PageState::UNINITIALIZED ||
+ findPage->state() == Page::PageState::INVALID) {
+ ESP_ERROR_CHECK(findItem(nsIndex, datatype, key, findPage, item));
}
+ /* Get the version of the previous index with same <ns,key> */
+ prevStart = item.blobIndex.chunkStart;
+ assert(prevStart == VerOffset::VER_0_OFFSET || prevStart == VerOffset::VER_1_OFFSET);
+
+ /* Toggle the version by changing the offset */
+ nextStart
+ = (prevStart == VerOffset::VER_1_OFFSET) ? VerOffset::VER_0_OFFSET : VerOffset::VER_1_OFFSET;
+ }
+ /* Write the blob with new version*/
+ err = writeMultiPageBlob(nsIndex, key, data, dataSize, nextStart);
+ if (err == ESP_ERR_NVS_PAGE_FULL) {
+ return ESP_ERR_NVS_NOT_ENOUGH_SPACE;
}
- err = mPageManager.requestNewPage();
if (err != ESP_OK) {
return err;
}
- err = getCurrentPage().writeItem(nsIndex, datatype, key, data, dataSize);
- if (err == ESP_ERR_NVS_PAGE_FULL) {
- return ESP_ERR_NVS_NOT_ENOUGH_SPACE;
+ if (findPage) {
+ /* Erase the blob with earlier version*/
+ err = eraseMultiPageBlob(nsIndex, key, prevStart);
+
+ if (err == ESP_ERR_FLASH_OP_FAIL) {
+ return ESP_ERR_NVS_REMOVE_FAILED;
+ }
+ if (err != ESP_OK) {
+ return err;
+ }
+
+ findPage = nullptr;
+ } else {
+ /* Support for earlier versions where BLOBS were stored without index */
+ err = findItem(nsIndex, datatype, key, findPage, item);
+ if (err != ESP_OK && err != ESP_ERR_NVS_NOT_FOUND) {
+ return err;
+ }
}
- if (err != ESP_OK) {
+ } else {
+
+ Page& page = getCurrentPage();
+ err = page.writeItem(nsIndex, datatype, key, data, dataSize);
+ if (err == ESP_ERR_NVS_PAGE_FULL) {
+ if (page.state() != Page::PageState::FULL) {
+ err = page.markFull();
+ if (err != ESP_OK) {
+ return err;
+ }
+ }
+ err = mPageManager.requestNewPage();
+ if (err != ESP_OK) {
+ return err;
+ }
+
+ err = getCurrentPage().writeItem(nsIndex, datatype, key, data, dataSize);
+ if (err == ESP_ERR_NVS_PAGE_FULL) {
+ return ESP_ERR_NVS_NOT_ENOUGH_SPACE;
+ }
+ if (err != ESP_OK) {
+ return err;
+ }
+ } else if (err != ESP_OK) {
return err;
}
- } else if (err != ESP_OK) {
- return err;
}
if (findPage) {
if (findPage->state() == Page::PageState::UNINITIALIZED ||
findPage->state() == Page::PageState::INVALID) {
- ESP_ERROR_CHECK( findItem(nsIndex, datatype, key, findPage, item) );
+ ESP_ERROR_CHECK(findItem(nsIndex, datatype, key, findPage, item));
}
err = findPage->eraseItem(nsIndex, datatype, key);
if (err == ESP_ERR_FLASH_OP_FAIL) {
return ESP_OK;
}
+esp_err_t Storage::readMultiPageBlob(uint8_t nsIndex, const char* key, void* data, size_t dataSize)
+{
+ Item item;
+ Page* findPage = nullptr;
+
+ /* First read the blob index */
+ auto err = findItem(nsIndex, ItemType::BLOB_IDX, key, findPage, item);
+ if (err != ESP_OK) {
+ return err;
+ }
+
+ uint8_t chunkCount = item.blobIndex.chunkCount;
+ VerOffset chunkStart = item.blobIndex.chunkStart;
+ size_t readSize = item.blobIndex.dataSize;
+ size_t offset = 0;
+
+ assert(dataSize == readSize);
+
+ /* Now read corresponding chunks */
+ for (uint8_t chunkNum = 0; chunkNum < chunkCount; chunkNum++) {
+ err = findItem(nsIndex, ItemType::BLOB_DATA, key, findPage, item, static_cast<uint8_t> (chunkStart) + chunkNum);
+ if (err != ESP_OK) {
+ if (err == ESP_ERR_NVS_NOT_FOUND) {
+ break;
+ }
+ return err;
+ }
+ err = findPage->readItem(nsIndex, ItemType::BLOB_DATA, key, static_cast<uint8_t*>(data) + offset, item.varLength.dataSize, static_cast<uint8_t> (chunkStart) + chunkNum);
+ if (err != ESP_OK) {
+ return err;
+ }
+ assert(static_cast<uint8_t> (chunkStart) + chunkNum == item.chunkIndex);
+ offset += item.varLength.dataSize;
+ }
+ if (err == ESP_OK) {
+ assert(offset == dataSize);
+ }
+ if (err == ESP_ERR_NVS_NOT_FOUND) {
+ eraseMultiPageBlob(nsIndex, key); // cleanup if a chunk is not found
+ }
+ return err;
+}
+
esp_err_t Storage::readItem(uint8_t nsIndex, ItemType datatype, const char* key, void* data, size_t dataSize)
{
if (mState != StorageState::ACTIVE) {
Item item;
Page* findPage = nullptr;
+ if (datatype == ItemType::BLOB) {
+ auto err = readMultiPageBlob(nsIndex, key, data, dataSize);
+ if (err != ESP_ERR_NVS_NOT_FOUND) {
+ return err;
+ } // else check if the blob is stored with earlier version format without index
+ }
+
auto err = findItem(nsIndex, datatype, key, findPage, item);
if (err != ESP_OK) {
return err;
}
-
return findPage->readItem(nsIndex, datatype, key, data, dataSize);
+
+}
+
+esp_err_t Storage::eraseMultiPageBlob(uint8_t nsIndex, const char* key, VerOffset chunkStart)
+{
+ if (mState != StorageState::ACTIVE) {
+ return ESP_ERR_NVS_NOT_INITIALIZED;
+ }
+ Item item;
+ Page* findPage = nullptr;
+
+ auto err = findItem(nsIndex, ItemType::BLOB_IDX, key, findPage, item, Page::CHUNK_ANY, chunkStart);
+ if (err != ESP_OK) {
+ return err;
+ }
+ /* Erase the index first and make children blobs orphan*/
+ err = findPage->eraseItem(nsIndex, ItemType::BLOB_IDX, key, Page::CHUNK_ANY, chunkStart);
+ if (err != ESP_OK) {
+ return err;
+ }
+
+ uint8_t chunkCount = item.blobIndex.chunkCount;
+
+ if (chunkStart == VerOffset::VER_ANY) {
+ chunkStart = item.blobIndex.chunkStart;
+ } else {
+ assert(chunkStart == item.blobIndex.chunkStart);
+ }
+
+ /* Now erase corresponding chunks*/
+ for (uint8_t chunkNum = 0; chunkNum < chunkCount; chunkNum++) {
+ err = findItem(nsIndex, ItemType::BLOB_DATA, key, findPage, item, static_cast<uint8_t> (chunkStart) + chunkNum);
+
+ if (err != ESP_OK && err != ESP_ERR_NVS_NOT_FOUND) {
+ return err;
+ } else if (err == ESP_ERR_NVS_NOT_FOUND) {
+ continue; // Keep erasing other chunks
+ }
+ err = findPage->eraseItem(nsIndex, ItemType::BLOB_DATA, key, static_cast<uint8_t> (chunkStart) + chunkNum);
+ if (err != ESP_OK) {
+ return err;
+ }
+
+ }
+
+ return ESP_OK;
}
esp_err_t Storage::eraseItem(uint8_t nsIndex, ItemType datatype, const char* key)
return ESP_ERR_NVS_NOT_INITIALIZED;
}
+ if (datatype == ItemType::BLOB) {
+ return eraseMultiPageBlob(nsIndex, key);
+ }
+
Item item;
Page* findPage = nullptr;
auto err = findItem(nsIndex, datatype, key, findPage, item);
Page* findPage = nullptr;
auto err = findItem(nsIndex, datatype, key, findPage, item);
if (err != ESP_OK) {
- return err;
+ if (datatype != ItemType::BLOB) {
+ return err;
+ }
+ err = findItem(nsIndex, ItemType::BLOB_IDX, key, findPage, item);
+ if (err != ESP_OK) {
+ return err;
+ }
+ dataSize = item.blobIndex.dataSize;
+ return ESP_OK;
}
dataSize = item.varLength.dataSize;
void Storage::debugCheck()
{
std::map<std::string, Page*> keys;
-
+
for (auto p = mPageManager.begin(); p != mPageManager.end(); ++p) {
size_t itemIndex = 0;
size_t usedCount = 0;
Item item;
while (p->findItem(Page::NS_ANY, ItemType::ANY, nullptr, itemIndex, item) == ESP_OK) {
std::stringstream keyrepr;
- keyrepr << static_cast<unsigned>(item.nsIndex) << "_" << static_cast<unsigned>(item.datatype) << "_" << item.key;
+ keyrepr << static_cast<unsigned>(item.nsIndex) << "_" << static_cast<unsigned>(item.datatype) << "_" << item.key <<"_"<<static_cast<unsigned>(item.chunkIndex);
std::string keystr = keyrepr.str();
if (keys.find(keystr) != std::end(keys)) {
printf("Duplicate key: %s\n", keystr.c_str());
}
usedEntries += item.span;
itemIndex += item.span;
- if(itemIndex >= it->ENTRY_COUNT) break;
+ if (itemIndex >= it->ENTRY_COUNT) break;
}
}
return ESP_OK;
typedef intrusive_list<NamespaceEntry> TNamespaces;
+ struct UsedPageNode: public intrusive_list_node<UsedPageNode> {
+ public: Page* mPage;
+ };
+
+ typedef intrusive_list<UsedPageNode> TUsedPageList;
+
+ struct BlobIndexNode: public intrusive_list_node<BlobIndexNode> {
+ public:
+ char key[Item::MAX_KEY_LENGTH + 1];
+ uint8_t nsIndex;
+ uint8_t chunkCount;
+ VerOffset chunkStart;
+ };
+
+ typedef intrusive_list<BlobIndexNode> TBlobIndexList;
+
public:
~Storage();
return mPartitionName;
}
+ esp_err_t writeMultiPageBlob(uint8_t nsIndex, const char* key, const void* data, size_t dataSize, VerOffset chunkStart);
+
+ esp_err_t readMultiPageBlob(uint8_t nsIndex, const char* key, void* data, size_t dataSize);
+
+ esp_err_t eraseMultiPageBlob(uint8_t nsIndex, const char* key, VerOffset chunkStart = VerOffset::VER_ANY);
+
void debugDump();
void debugCheck();
void clearNamespaces();
- esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key, Page* &page, Item& item);
+ void populateBlobIndices(TBlobIndexList&);
+
+ void eraseOrphanDataBlobs(TBlobIndexList&);
+
+
+ esp_err_t findItem(uint8_t nsIndex, ItemType datatype, const char* key, Page* &page, Item& item, uint8_t chunkIdx = Page::CHUNK_ANY, VerOffset chunkStart = VerOffset::VER_ANY);
protected:
const char *mPartitionName;
result = crc32_le(result, p + offsetof(Item, nsIndex),
offsetof(Item, datatype) - offsetof(Item, nsIndex));
result = crc32_le(result, p + offsetof(Item, key), sizeof(key));
+ result = crc32_le(result, p + offsetof(Item, chunkIndex), sizeof(chunkIndex));
return result;
}
I64 = 0x18,
SZ = 0x21,
BLOB = 0x41,
+ BLOB_DATA = 0x42,
+ BLOB_IDX = 0x48,
ANY = 0xff
};
+enum class VerOffset: uint8_t {
+ VER_0_OFFSET = 0x0,
+ VER_1_OFFSET = 0x80,
+ VER_ANY = 0xff,
+};
+
template<typename T, typename std::enable_if<std::is_integral<T>::value, void*>::type = nullptr>
constexpr ItemType itemTypeOf()
{
return itemTypeOf<T>();
}
+inline bool isVariableLengthType(ItemType type)
+{
+ return (type == ItemType::BLOB ||
+ type == ItemType::SZ ||
+ type == ItemType::BLOB_DATA);
+}
+
class Item
{
public:
uint8_t nsIndex;
ItemType datatype;
uint8_t span;
- uint8_t reserved;
+ uint8_t chunkIndex;
uint32_t crc32;
char key[16];
union {
struct {
uint16_t dataSize;
- uint16_t reserved2;
+ uint16_t reserved;
uint32_t dataCrc32;
} varLength;
+ struct {
+ uint32_t dataSize;
+ uint8_t chunkCount; // Number of children data blobs.
+ VerOffset chunkStart; // Offset from which the chunkIndex for children blobs starts
+ uint16_t reserved;
+ } blobIndex;
uint8_t data[8];
};
};
static const size_t MAX_KEY_LENGTH = sizeof(key) - 1;
- Item(uint8_t nsIndex, ItemType datatype, uint8_t span, const char* key_)
- : nsIndex(nsIndex), datatype(datatype), span(span), reserved(0xff)
+ // 0xff cannot be used as a valid chunkIndex for blob datatype.
+ static const uint8_t CHUNK_ANY = 0xff;
+
+
+ Item(uint8_t nsIndex, ItemType datatype, uint8_t span, const char* key_, uint8_t chunkIdx = CHUNK_ANY)
+ : nsIndex(nsIndex), datatype(datatype), span(span), chunkIndex(chunkIdx)
{
std::fill_n(reinterpret_cast<uint32_t*>(key), sizeof(key) / 4, 0xffffffff);
std::fill_n(reinterpret_cast<uint32_t*>(data), sizeof(data) / 4, 0xffffffff);
uint32_t blob[12];
TEST_ESP_OK(nvs_set_blob(handle_3, "bl1", &blob, sizeof(blob)));
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
- TEST_ASSERT_TRUE(stat1.free_entries + 3 == stat2.free_entries);
+ TEST_ASSERT_TRUE(stat1.free_entries + 4 == stat2.free_entries);
TEST_ASSERT_TRUE(stat1.namespace_count == 3);
TEST_ASSERT_TRUE(stat1.total_entries == stat2.total_entries);
- TEST_ASSERT_TRUE(stat1.used_entries == 11);
+ TEST_ASSERT_TRUE(stat1.used_entries == 12);
// amount valid pair in namespace 2
size_t h3_count_entries;
TEST_ESP_OK(nvs_get_used_entry_count(handle_3, &h3_count_entries));
- TEST_ASSERT_TRUE(h3_count_entries == 3);
+ TEST_ASSERT_TRUE(h3_count_entries == 4);
TEST_ASSERT_TRUE(stat1.used_entries == (h1_count_entries + h2_count_entries + h3_count_entries + stat1.namespace_count));
{
load(filename);
// Atleast one page should be free, hence we create mData of size of 2 sectors.
- mData.resize(2 * SPI_FLASH_SEC_SIZE / 4, 0xffffffff);
+ mData.resize(mData.size() + SPI_FLASH_SEC_SIZE / 4, 0xffffffff);
+ mUpperSectorBound = mData.size() * 4 / SPI_FLASH_SEC_SIZE;
spi_flash_emulator_set(this);
}
#include "spi_flash_emulation.h"
#include <sstream>
#include <iostream>
+#include <fstream>
#include <unistd.h>
#include <sys/wait.h>
item1.datatype = ItemType::I32;
item1.nsIndex = 1;
item1.crc32 = 0;
- item1.reserved = 0xff;
+ item1.chunkIndex = 0xff;
fill_n(item1.key, sizeof(item1.key), 0xbb);
fill_n(item1.data, sizeof(item1.data), 0xaa);
// Check that the second one is actually returned.
TEST_ESP_ERR(page.writeItem(1, ItemType::BLOB, "2", buf, Page::ENTRY_COUNT * Page::ENTRY_SIZE), ESP_ERR_NVS_VALUE_TOO_LONG);
// Should fail as well
- TEST_ESP_ERR(page.writeItem(1, ItemType::BLOB, "2", buf, Page::BLOB_MAX_SIZE + 1), ESP_ERR_NVS_VALUE_TOO_LONG);
- TEST_ESP_OK(page.writeItem(1, ItemType::BLOB, "2", buf, Page::BLOB_MAX_SIZE));
+ TEST_ESP_ERR(page.writeItem(1, ItemType::BLOB, "2", buf, Page::CHUNK_MAX_SIZE + 1), ESP_ERR_NVS_VALUE_TOO_LONG);
+ TEST_ESP_OK(page.writeItem(1, ItemType::BLOB, "2", buf, Page::CHUNK_MAX_SIZE));
}
TEST_CASE("Page handles invalid CRC of variable length items", "[nvs][cur]")
Storage storage;
CHECK(storage.init(0, 3) == ESP_OK);
int bar = 0;
- uint8_t bigdata[Page::BLOB_MAX_SIZE] = {0};
+ uint8_t bigdata[(Page::CHUNK_MAX_SIZE - Page::ENTRY_SIZE)/2] = {0};
// write one big chunk of data
ESP_ERROR_CHECK(storage.writeItem(0, ItemType::BLOB, "1", bigdata, sizeof(bigdata)));
// write another big chunk of data
nvs_close(handle_1);
}
}
-
extern "C" void nvs_dump(const char *partName);
class RandomTest {
- static const size_t nKeys = 9;
+ static const size_t nKeys = 11;
int32_t v1 = 0, v2 = 0;
uint64_t v3 = 0, v4 = 0;
static const size_t strBufLen = 1024;
+ static const size_t smallBlobLen = Page::CHUNK_MAX_SIZE / 3;
+ static const size_t largeBlobLen = Page::CHUNK_MAX_SIZE * 3;
char v5[strBufLen], v6[strBufLen], v7[strBufLen], v8[strBufLen], v9[strBufLen];
+ uint8_t v10[smallBlobLen], v11[largeBlobLen];
bool written[nKeys];
public:
template<typename TGen>
esp_err_t doRandomThings(nvs_handle handle, TGen gen, size_t& count) {
- const char* keys[] = {"foo", "bar", "longkey_0123456", "another key", "param1", "param2", "param3", "param4", "param5"};
- const ItemType types[] = {ItemType::I32, ItemType::I32, ItemType::U64, ItemType::U64, ItemType::SZ, ItemType::SZ, ItemType::SZ, ItemType::SZ, ItemType::SZ};
+ const char* keys[] = {"foo", "bar", "longkey_0123456", "another key", "param1", "param2", "param3", "param4", "param5", "singlepage", "multipage"};
+ const ItemType types[] = {ItemType::I32, ItemType::I32, ItemType::U64, ItemType::U64, ItemType::SZ, ItemType::SZ, ItemType::SZ, ItemType::SZ, ItemType::SZ, ItemType::BLOB, ItemType::BLOB};
- void* values[] = {&v1, &v2, &v3, &v4, &v5, &v6, &v7, &v8, &v9};
+ void* values[] = {&v1, &v2, &v3, &v4, &v5, &v6, &v7, &v8, &v9, &v10, &v11};
const size_t nKeys = sizeof(keys) / sizeof(keys[0]);
static_assert(nKeys == sizeof(types) / sizeof(types[0]), "");
}
break;
}
-
+
+ case ItemType::BLOB:
+ {
+ uint32_t blobBufLen = 0;
+ if(strncmp(keys[index],"singlepage", sizeof("singlepage")) == 0) {
+ blobBufLen = smallBlobLen ;
+ } else {
+ blobBufLen = largeBlobLen ;
+
+ }
+ uint8_t buf[blobBufLen];
+ memset(buf, 0, blobBufLen);
+
+ size_t len = blobBufLen;
+ auto err = nvs_get_blob(handle, keys[index], buf, &len);
+ if (err == ESP_ERR_FLASH_OP_FAIL) {
+ return err;
+ }
+ if (!written[index]) {
+ REQUIRE(err == ESP_ERR_NVS_NOT_FOUND);
+ }
+ else {
+ REQUIRE(err == ESP_OK);
+ REQUIRE(memcmp(buf, reinterpret_cast<const uint8_t*>(values[index]), blobBufLen) == 0);
+ }
+ break;
+ }
+
+
default:
assert(0);
}
strncpy(reinterpret_cast<char*>(values[index]), buf, strBufLen);
break;
}
-
+
+ case ItemType::BLOB:
+ {
+ uint32_t blobBufLen = 0;
+ if(strncmp(keys[index],"singlepage", sizeof("singlepage")) == 0) {
+ blobBufLen = smallBlobLen ;
+ } else {
+ blobBufLen = largeBlobLen ;
+ }
+ uint8_t buf[blobBufLen];
+ memset(buf, 0, blobBufLen);
+ size_t blobLen = gen() % blobBufLen;
+ std::generate_n(buf, blobLen, [&]() -> uint8_t {
+ return static_cast<uint8_t>(gen() % 256);
+ });
+
+ auto err = nvs_set_blob(handle, keys[index], buf, blobLen);
+ if (err == ESP_ERR_FLASH_OP_FAIL) {
+ return err;
+ }
+ if (err == ESP_ERR_NVS_REMOVE_FAILED) {
+ written[index] = true;
+ memcpy(reinterpret_cast<uint8_t*>(values[index]), buf, blobBufLen);
+ return ESP_ERR_FLASH_OP_FAIL;
+ }
+ REQUIRE(err == ESP_OK);
+ written[index] = true;
+ memcpy(reinterpret_cast<char*>(values[index]), buf, blobBufLen);
+ break;
+ }
+
default:
assert(0);
}
return ESP_OK;
};
-
+
for (; count != 0; --count) {
- size_t index = gen() % nKeys;
+ size_t index = gen() % (nKeys);
switch (gen() % 3) {
case 0: // read, 1/3
if (randomRead(index) == ESP_ERR_FLASH_OP_FAIL) {
}
return ESP_OK;
}
-};
+ esp_err_t handleExternalWriteAtIndex(uint8_t index, const void* value, const size_t len ) {
+ if(index == 9) { /* This is only done for small-page blobs for now*/
+ if(len > smallBlobLen) {
+ return ESP_FAIL;
+ }
+ memcpy(v10, value, len);
+ written[index] = true;
+ return ESP_OK;
+ } else {
+ return ESP_FAIL;
+ }
+ }
+};
TEST_CASE("monkey test", "[nvs][monkey]")
{
emu.randomize(seed);
emu.clearStats();
- const uint32_t NVS_FLASH_SECTOR = 6;
- const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3;
+ const uint32_t NVS_FLASH_SECTOR = 2;
+ const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 8;
emu.setBounds(NVS_FLASH_SECTOR, NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN);
TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, NVS_FLASH_SECTOR, NVS_FLASH_SECTOR_COUNT_MIN));
SpiFlashEmulator emu(10);
- const uint32_t NVS_FLASH_SECTOR = 6;
- const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 3;
+ const uint32_t NVS_FLASH_SECTOR = 2;
+ const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 8;
emu.setBounds(NVS_FLASH_SECTOR, NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN);
size_t totalOps = 0;
totalOps = emu.getEraseOps() + emu.getWriteBytes() / 4;
}
}
-
TEST_CASE("test for memory leaks in open/set", "[leaks]")
{
SpiFlashEmulator emu(10);
TEST_CASE("nvs_flash_init checks for an empty page", "[nvs]")
{
- const size_t blob_size = Page::BLOB_MAX_SIZE;
+ const size_t blob_size = Page::CHUNK_MAX_SIZE;
uint8_t blob[blob_size] = {0};
SpiFlashEmulator emu(5);
TEST_ESP_OK( nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5) );
TEST_ESP_OK( nvs_open("test", NVS_READWRITE, &handle) );
// Fill first page
TEST_ESP_OK( nvs_set_blob(handle, "1a", blob, blob_size) );
- TEST_ESP_OK( nvs_set_blob(handle, "1b", blob, blob_size) );
// Fill second page
TEST_ESP_OK( nvs_set_blob(handle, "2a", blob, blob_size) );
- TEST_ESP_OK( nvs_set_blob(handle, "2b", blob, blob_size) );
// Fill third page
TEST_ESP_OK( nvs_set_blob(handle, "3a", blob, blob_size) );
- TEST_ESP_OK( nvs_set_blob(handle, "3b", blob, blob_size) );
TEST_ESP_OK( nvs_commit(handle) );
nvs_close(handle);
// first two pages are now full, third one is writable, last two are empty
TEST_CASE("nvs page selection takes into account free entries also not just erased entries", "[nvs]")
{
- const size_t blob_size = Page::BLOB_MAX_SIZE;
+ const size_t blob_size = Page::CHUNK_MAX_SIZE/2;
uint8_t blob[blob_size] = {0};
SpiFlashEmulator emu(3);
TEST_ESP_OK( nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 3) );
uint32_t blob[12];
TEST_ESP_OK(nvs_set_blob(handle_3, "bl1", &blob, sizeof(blob)));
TEST_ESP_OK(nvs_get_stats(NULL, &stat1));
- CHECK(stat1.free_entries + 3 == stat2.free_entries);
+ CHECK(stat1.free_entries + 4 == stat2.free_entries);
CHECK(stat1.namespace_count == 3);
CHECK(stat1.total_entries == stat2.total_entries);
- CHECK(stat1.used_entries == 11);
+ CHECK(stat1.used_entries == 12);
// amount valid pair in namespace 2
size_t h3_count_entries;
TEST_ESP_OK(nvs_get_used_entry_count(handle_3, &h3_count_entries));
- CHECK(h3_count_entries == 3);
+ CHECK(h3_count_entries == 4);
CHECK(stat1.used_entries == (h1_count_entries + h2_count_entries + h3_count_entries + stat1.namespace_count));
TEST_CASE("Recovery from power-off when the entry being erased is not on active page", "[nvs]")
{
- const size_t blob_size = Page::BLOB_MAX_SIZE;
+ const size_t blob_size = Page::CHUNK_MAX_SIZE/2 ;
size_t read_size = blob_size;
uint8_t blob[blob_size] = {0x11};
SpiFlashEmulator emu(3);
TEST_ESP_OK( nvs_open("test", NVS_READWRITE, &handle) );
emu.clearStats();
- emu.failAfter(2 * Page::BLOB_MAX_SIZE/4 + 36);
+ emu.failAfter(Page::CHUNK_MAX_SIZE/4 + 75);
TEST_ESP_OK( nvs_set_blob(handle, "1a", blob, blob_size) );
TEST_ESP_OK( nvs_set_blob(handle, "1b", blob, blob_size) );
nvs_close(handle);
}
+TEST_CASE("Multi-page blobs are supported", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *2;
+ uint8_t blob[blob_size] = {0};
+ SpiFlashEmulator emu(5);
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5));
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("test", NVS_READWRITE, &handle));
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, blob_size));
+ TEST_ESP_OK(nvs_commit(handle));
+ nvs_close(handle);
+}
+
+TEST_CASE("Failures are handled while storing multi-page blobs", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *7;
+ uint8_t blob[blob_size] = {0};
+ SpiFlashEmulator emu(5);
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5));
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("test", NVS_READWRITE, &handle));
+ TEST_ESP_ERR(nvs_set_blob(handle, "abc", blob, blob_size), ESP_ERR_NVS_VALUE_TOO_LONG);
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, Page::CHUNK_MAX_SIZE*2));
+ TEST_ESP_OK(nvs_commit(handle));
+ nvs_close(handle);
+}
+
+TEST_CASE("Reading multi-page blobs", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *3;
+ uint8_t blob[blob_size];
+ uint8_t blob_read[blob_size];
+ size_t read_size = blob_size;
+ SpiFlashEmulator emu(5);
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5));
+ nvs_handle handle;
+ memset(blob, 0x11, blob_size);
+ memset(blob_read, 0xee, blob_size);
+ TEST_ESP_OK(nvs_open("readTest", NVS_READWRITE, &handle));
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, blob_size));
+ TEST_ESP_OK(nvs_get_blob(handle, "abc", blob_read, &read_size));
+ CHECK(memcmp(blob, blob_read, blob_size) == 0);
+ TEST_ESP_OK(nvs_commit(handle));
+ nvs_close(handle);
+}
+
+TEST_CASE("Modification of values for Multi-page blobs are supported", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *2;
+ uint8_t blob[blob_size] = {0};
+ uint8_t blob_read[blob_size] = {0xfe};;
+ uint8_t blob2[blob_size] = {0x11};
+ uint8_t blob3[blob_size] = {0x22};
+ uint8_t blob4[blob_size] ={ 0x33};
+ size_t read_size = blob_size;
+ SpiFlashEmulator emu(6);
+ TEST_ESP_OK( nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 6) );
+ nvs_handle handle;
+ memset(blob, 0x11, blob_size);
+ memset(blob2, 0x22, blob_size);
+ memset(blob3, 0x33, blob_size);
+ memset(blob4, 0x44, blob_size);
+ memset(blob_read, 0xff, blob_size);
+ TEST_ESP_OK( nvs_open("test", NVS_READWRITE, &handle) );
+ TEST_ESP_OK( nvs_set_blob(handle, "abc", blob, blob_size) );
+ TEST_ESP_OK( nvs_set_blob(handle, "abc", blob2, blob_size) );
+ TEST_ESP_OK( nvs_set_blob(handle, "abc", blob3, blob_size) );
+ TEST_ESP_OK( nvs_set_blob(handle, "abc", blob4, blob_size) );
+ TEST_ESP_OK( nvs_get_blob(handle, "abc", blob_read, &read_size));
+ CHECK(memcmp(blob4, blob_read, blob_size) == 0);
+ TEST_ESP_OK( nvs_commit(handle) );
+ nvs_close(handle);
+}
+
+TEST_CASE("Modification from single page blob to multi-page", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *3;
+ uint8_t blob[blob_size] = {0};
+ uint8_t blob_read[blob_size] = {0xff};
+ size_t read_size = blob_size;
+ SpiFlashEmulator emu(5);
+ TEST_ESP_OK( nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5) );
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("Test", NVS_READWRITE, &handle) );
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, Page::CHUNK_MAX_SIZE/2));
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, blob_size));
+ TEST_ESP_OK(nvs_get_blob(handle, "abc", blob_read, &read_size));
+ CHECK(memcmp(blob, blob_read, blob_size) == 0);
+ TEST_ESP_OK(nvs_commit(handle) );
+ nvs_close(handle);
+}
+
+TEST_CASE("Modification from multi-page to single page", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *3;
+ uint8_t blob[blob_size] = {0};
+ uint8_t blob_read[blob_size] = {0xff};
+ size_t read_size = blob_size;
+ SpiFlashEmulator emu(5);
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 5) );
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("Test", NVS_READWRITE, &handle) );
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, blob_size));
+ TEST_ESP_OK(nvs_set_blob(handle, "abc", blob, Page::CHUNK_MAX_SIZE/2));
+ TEST_ESP_OK(nvs_set_blob(handle, "abc2", blob, blob_size));
+ TEST_ESP_OK(nvs_get_blob(handle, "abc", blob_read, &read_size));
+ CHECK(memcmp(blob, blob_read, Page::CHUNK_MAX_SIZE) == 0);
+ TEST_ESP_OK(nvs_commit(handle) );
+ nvs_close(handle);
+}
+
+
+TEST_CASE("Check that orphaned blobs are erased during init", "[nvs]")
+{
+ const size_t blob_size = Page::CHUNK_MAX_SIZE *3 ;
+ uint8_t blob[blob_size] = {0x11};
+ uint8_t blob2[blob_size] = {0x22};
+ uint8_t blob3[blob_size] = {0x33};
+ SpiFlashEmulator emu(5);
+ Storage storage;
+
+ TEST_ESP_OK(storage.init(0, 5));
+
+ TEST_ESP_OK(storage.writeItem(1, ItemType::BLOB, "key", blob, sizeof(blob)));
+
+
+ TEST_ESP_OK(storage.init(0, 5));
+ /* Check that multi-page item is still available.**/
+ TEST_ESP_OK(storage.readItem(1, ItemType::BLOB, "key", blob, sizeof(blob)));
+
+ TEST_ESP_ERR(storage.writeItem(1, ItemType::BLOB, "key2", blob, sizeof(blob)), ESP_ERR_NVS_NOT_ENOUGH_SPACE);
+
+ Page p;
+ p.load(3); // This is where index will be placed.
+ p.erase();
+
+ TEST_ESP_OK(storage.init(0, 5));
+
+ TEST_ESP_ERR(storage.readItem(1, ItemType::BLOB, "key", blob, sizeof(blob)), ESP_ERR_NVS_NOT_FOUND);
+ TEST_ESP_OK(storage.writeItem(1, ItemType::BLOB, "key3", blob, sizeof(blob)));
+}
+
+TEST_CASE("Check that NVS supports old blob format without blob index", "[nvs]")
+{
+ SpiFlashEmulator emu("../nvs_partition_generator/part_old_blob_format.bin");
+ nvs_handle handle;
+
+ TEST_ESP_OK( nvs_flash_init_custom("test", 0, 2) );
+ TEST_ESP_OK( nvs_open_from_partition("test", "dummyNamespace", NVS_READONLY, &handle));
+
+ char buf[64] = {0};
+ size_t buflen = 64;
+ uint8_t hexdata[] = {0x01, 0x02, 0x03, 0xab, 0xcd, 0xef};
+ TEST_ESP_OK( nvs_get_blob(handle, "dummyHex2BinKey", buf, &buflen));
+ CHECK(memcmp(buf, hexdata, buflen) == 0);
+
+ buflen = 64;
+ uint8_t base64data[] = {'1', '2', '3', 'a', 'b', 'c'};
+ TEST_ESP_OK( nvs_get_blob(handle, "dummyBase64Key", buf, &buflen));
+ CHECK(memcmp(buf, base64data, buflen) == 0);
+
+ Page p;
+ p.load(0);
+
+ /* Check that item is stored in old format without blob index*/
+ TEST_ESP_OK(p.findItem(1, ItemType::BLOB, "dummyHex2BinKey"));
+
+ /* Modify the blob so that it is stored in the new format*/
+ hexdata[0] = hexdata[1] = hexdata[2] = 0x99;
+ TEST_ESP_OK(nvs_set_blob(handle, "dummyHex2BinKey", hexdata, sizeof(hexdata)));
+
+ Page p2;
+ p2.load(0);
+
+ /* Check the type of the blob. Expect type mismatch since the blob is stored in new format*/
+ TEST_ESP_ERR(p2.findItem(1, ItemType::BLOB, "dummyHex2BinKey"), ESP_ERR_NVS_TYPE_MISMATCH);
+
+ /* Check that index is present for the modified blob according to new format*/
+ TEST_ESP_OK(p2.findItem(1, ItemType::BLOB_IDX, "dummyHex2BinKey"));
+
+ /* Read the blob in new format and check the contents*/
+ buflen = 64;
+ TEST_ESP_OK( nvs_get_blob(handle, "dummyBase64Key", buf, &buflen));
+ CHECK(memcmp(buf, base64data, buflen) == 0);
+
+}
+
+TEST_CASE("monkey test with old-format blob present", "[nvs][monkey]")
+{
+ std::random_device rd;
+ std::mt19937 gen(rd());
+ uint32_t seed = 3;
+ gen.seed(seed);
+
+ SpiFlashEmulator emu(10);
+ emu.randomize(seed);
+ emu.clearStats();
+
+ const uint32_t NVS_FLASH_SECTOR = 2;
+ const uint32_t NVS_FLASH_SECTOR_COUNT_MIN = 8;
+ static const size_t smallBlobLen = Page::CHUNK_MAX_SIZE / 3;
+
+ emu.setBounds(NVS_FLASH_SECTOR, NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN);
+
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, NVS_FLASH_SECTOR, NVS_FLASH_SECTOR_COUNT_MIN));
+
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+ RandomTest test;
+
+ for ( uint8_t it = 0; it < 10; it++) {
+ size_t count = 200;
+
+ /* Erase index and chunks for the blob with "singlepage" key */
+ for (uint8_t num = NVS_FLASH_SECTOR; num < NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN; num++) {
+ Page p;
+ p.load(num);
+ p.eraseItem(1, ItemType::BLOB, "singlepage", Item::CHUNK_ANY, VerOffset::VER_ANY);
+ p.eraseItem(1, ItemType::BLOB_IDX, "singlepage", Item::CHUNK_ANY, VerOffset::VER_ANY);
+ p.eraseItem(1, ItemType::BLOB_DATA, "singlepage", Item::CHUNK_ANY, VerOffset::VER_ANY);
+ }
+
+ /* Now write "singlepage" blob in old format*/
+ for (uint8_t num = NVS_FLASH_SECTOR; num < NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN; num++) {
+ Page p;
+ p.load(num);
+ if (p.state() == Page::PageState::ACTIVE) {
+ uint8_t buf[smallBlobLen];
+ size_t blobLen = gen() % smallBlobLen;
+
+ if(blobLen > p.getVarDataTailroom()) {
+ blobLen = p.getVarDataTailroom();
+ }
+
+ std::generate_n(buf, blobLen, [&]() -> uint8_t {
+ return static_cast<uint8_t>(gen() % 256);
+ });
+
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB, "singlepage", buf, blobLen, Item::CHUNK_ANY));
+ TEST_ESP_OK(p.findItem(1, ItemType::BLOB, "singlepage"));
+ test.handleExternalWriteAtIndex(9, buf, blobLen); // This assumes "singlepage" is always at index 9
+
+ break;
+ }
+ }
+ /* Initialize again */
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, NVS_FLASH_SECTOR, NVS_FLASH_SECTOR_COUNT_MIN));
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+
+ /* Perform random things */
+ auto res = test.doRandomThings(handle, gen, count);
+ if (res != ESP_OK) {
+ nvs_dump(NVS_DEFAULT_PART_NAME);
+ CHECK(0);
+ }
+
+ /* Check that only one version is present for "singlepage". Its possible that last iteration did not write
+ * anything for "singlepage". So either old version or new version should be present.*/
+ bool oldVerPresent = false, newVerPresent = false;
+
+ for (uint8_t num = NVS_FLASH_SECTOR; num < NVS_FLASH_SECTOR + NVS_FLASH_SECTOR_COUNT_MIN; num++) {
+ Page p;
+ p.load(num);
+ if(!oldVerPresent && p.findItem(1, ItemType::BLOB, "singlepage", Item::CHUNK_ANY, VerOffset::VER_ANY) == ESP_OK) {
+ oldVerPresent = true;
+ }
+
+ if(!newVerPresent && p.findItem(1, ItemType::BLOB_IDX, "singlepage", Item::CHUNK_ANY, VerOffset::VER_ANY) == ESP_OK) {
+ newVerPresent = true;
+ }
+ }
+ CHECK(oldVerPresent != newVerPresent);
+ }
+
+ s_perf << "Monkey test: nErase=" << emu.getEraseOps() << " nWrite=" << emu.getWriteOps() << std::endl;
+}
+
+TEST_CASE("Recovery from power-off during modification of blob present in old-format (same page)", "[nvs]")
+{
+ std::random_device rd;
+ std::mt19937 gen(rd());
+ uint32_t seed = 3;
+ gen.seed(seed);
+
+ SpiFlashEmulator emu(3);
+ emu.clearStats();
+
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 3));
+
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+
+ uint8_t hexdata[] = {0x01, 0x02, 0x03, 0xab, 0xcd, 0xef};
+ uint8_t hexdata_old[] = {0x11, 0x12, 0x13, 0xbb, 0xcc, 0xee};
+ size_t buflen = sizeof(hexdata);
+ uint8_t buf[Page::CHUNK_MAX_SIZE];
+
+ /* Power-off when blob was being written on the same page where its old version in old format
+ * was present*/
+ Page p;
+ p.load(0);
+ /* Write blob in old-format*/
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB, "singlepage", hexdata_old, sizeof(hexdata_old)));
+
+ /* Write blob in new format*/
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB_DATA, "singlepage", hexdata, sizeof(hexdata), 0));
+ /* All pages are stored. Now store the index.*/
+ Item item;
+ item.blobIndex.dataSize = sizeof(hexdata);
+ item.blobIndex.chunkCount = 1;
+ item.blobIndex.chunkStart = VerOffset::VER_0_OFFSET;
+
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB_IDX, "singlepage", item.data, sizeof(item.data)));
+
+ TEST_ESP_OK(p.findItem(1, ItemType::BLOB, "singlepage"));
+
+ /* Initialize again */
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 3));
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+
+ TEST_ESP_OK( nvs_get_blob(handle, "singlepage", buf, &buflen));
+ CHECK(memcmp(buf, hexdata, buflen) == 0);
+
+ Page p2;
+ p2.load(0);
+ TEST_ESP_ERR(p2.findItem(1, ItemType::BLOB, "singlepage"), ESP_ERR_NVS_TYPE_MISMATCH);
+
+}
+
+TEST_CASE("Recovery from power-off during modification of blob present in old-format (different page)", "[nvs]")
+{
+ std::random_device rd;
+ std::mt19937 gen(rd());
+ uint32_t seed = 3;
+ gen.seed(seed);
+
+ SpiFlashEmulator emu(3);
+ emu.clearStats();
+
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 3));
+
+ nvs_handle handle;
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+
+ uint8_t hexdata[] = {0x01, 0x02, 0x03, 0xab, 0xcd, 0xef};
+ uint8_t hexdata_old[] = {0x11, 0x12, 0x13, 0xbb, 0xcc, 0xee};
+ size_t buflen = sizeof(hexdata);
+ uint8_t buf[Page::CHUNK_MAX_SIZE];
+
+
+ /* Power-off when blob was being written on the different page where its old version in old format
+ * was present*/
+ Page p;
+ p.load(0);
+ /* Write blob in old-format*/
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB, "singlepage", hexdata_old, sizeof(hexdata_old)));
+
+ /* Write blob in new format*/
+ TEST_ESP_OK(p.writeItem(1, ItemType::BLOB_DATA, "singlepage", hexdata, sizeof(hexdata), 0));
+ /* All pages are stored. Now store the index.*/
+ Item item;
+ item.blobIndex.dataSize = sizeof(hexdata);
+ item.blobIndex.chunkCount = 1;
+ item.blobIndex.chunkStart = VerOffset::VER_0_OFFSET;
+ p.markFull();
+ Page p2;
+ p2.load(1);
+ p2.setSeqNumber(1);
+
+ TEST_ESP_OK(p2.writeItem(1, ItemType::BLOB_IDX, "singlepage", item.data, sizeof(item.data)));
+
+ TEST_ESP_OK(p.findItem(1, ItemType::BLOB, "singlepage"));
+
+ /* Initialize again */
+ TEST_ESP_OK(nvs_flash_init_custom(NVS_DEFAULT_PART_NAME, 0, 3));
+ TEST_ESP_OK(nvs_open("namespace1", NVS_READWRITE, &handle));
+
+ TEST_ESP_OK( nvs_get_blob(handle, "singlepage", buf, &buflen));
+ CHECK(memcmp(buf, hexdata, buflen) == 0);
+
+ Page p3;
+ p3.load(0);
+ TEST_ESP_ERR(p3.findItem(1, ItemType::BLOB, "singlepage"), ESP_ERR_NVS_NOT_FOUND);
+}
/* Add new tests above */
/* This test has to be the final one */
{
SpiFlashEmulator emu("../nvs_partition_generator/partition.bin");
nvs_handle handle;
- TEST_ESP_OK( nvs_flash_init_custom("test", 0, 2) );
+ TEST_ESP_OK( nvs_flash_init_custom("test", 0, 3) );
TEST_ESP_OK( nvs_open_from_partition("test", "dummyNamespace", NVS_READONLY, &handle));
uint8_t u8v;
TEST_ESP_OK( nvs_get_u8(handle, "dummyU8Key", &u8v));
int32_t i32v;
TEST_ESP_OK( nvs_get_i32(handle, "dummyI32Key", &i32v));
CHECK(i32v == -2147483648);
+
char buf[64] = {0};
size_t buflen = 64;
TEST_ESP_OK( nvs_get_str(handle, "dummyStringKey", buf, &buflen));
CHECK(strncmp(buf, "0A:0B:0C:0D:0E:0F", buflen) == 0);
- buflen = 64;
+
uint8_t hexdata[] = {0x01, 0x02, 0x03, 0xab, 0xcd, 0xef};
+ buflen = 64;
+ int j;
TEST_ESP_OK( nvs_get_blob(handle, "dummyHex2BinKey", buf, &buflen));
CHECK(memcmp(buf, hexdata, buflen) == 0);
+
uint8_t base64data[] = {'1', '2', '3', 'a', 'b', 'c'};
TEST_ESP_OK( nvs_get_blob(handle, "dummyBase64Key", buf, &buflen));
CHECK(memcmp(buf, base64data, buflen) == 0);
+
+ buflen = 64;
+ uint8_t hexfiledata[] = {0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef};
+ TEST_ESP_OK( nvs_get_blob(handle, "hexFileKey", buf, &buflen));
+ CHECK(memcmp(buf, hexfiledata, buflen) == 0);
+
+ buflen = 64;
+ uint8_t strfiledata[64] = "abcdefghijklmnopqrstuvwxyz\0";
+ TEST_ESP_OK( nvs_get_str(handle, "stringFileKey", buf, &buflen));
+ CHECK(memcmp(buf, strfiledata, buflen) == 0);
+
+ char bin_data[5200];
+ size_t bin_len = sizeof(bin_data);
+ char binfiledata[5200];
+ ifstream file;
+ file.open("../nvs_partition_generator/testdata/sample.bin");
+ file.read(binfiledata,5200);
+ TEST_ESP_OK( nvs_get_blob(handle, "binFileKey", bin_data, &bin_len));
+ CHECK(memcmp(bin_data, binfiledata, bin_len) == 0);
+
+ file.close();
+
}
TEST_CASE("dump all performance data", "[nvs]")