From e01b629b7d9614e915a2651c6c14839e7db58ba7 Mon Sep 17 00:00:00 2001 From: Dion Dokter Date: Mon, 12 Feb 2024 16:21:37 +0100 Subject: [PATCH 1/3] Implement map remove feature --- src/item.rs | 6 +++++ src/map.rs | 74 ++++++++++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 79 insertions(+), 1 deletion(-) diff --git a/src/item.rs b/src/item.rs index 6b45374..dcc60e8 100644 --- a/src/item.rs +++ b/src/item.rs @@ -47,6 +47,9 @@ impl ItemHeader { const LENGTH_FIELD: Range = 4..6; const LENGTH_CRC_FIELD: Range = 6..8; + /// Read the header from the flash at the given address. + /// + /// If the item doesn't exist or doesn't fit between the address and the end address, none is returned. pub async fn read_new( flash: &mut S, address: u32, @@ -160,6 +163,7 @@ impl ItemHeader { }) } + /// Erase this item by setting the crc to none and overwriting the header with it pub async fn erase_data( mut self, flash: &mut S, @@ -173,10 +177,12 @@ impl ItemHeader { Ok(self) } + /// Get the address of the start of the data for this item pub fn data_address(address: u32) -> u32 { address + round_up_to_alignment::(Self::LENGTH as u32) } + /// Get the location of the next item in flash pub fn next_item_address(&self, address: u32) -> u32 { let data_address = ItemHeader::data_address::(address); data_address + round_up_to_alignment::(self.length as u32) diff --git a/src/map.rs b/src/map.rs index a404f07..c868d61 100644 --- a/src/map.rs +++ b/src/map.rs @@ -115,9 +115,14 @@ //! # }); //! ``` +use embedded_storage_async::nor_flash::MultiwriteNorFlash; + use crate::item::{find_next_free_item_spot, Item, ItemHeader, ItemIter}; -use self::cache::{KeyCacheImpl, PrivateKeyCacheImpl}; +use self::{ + cache::{KeyCacheImpl, PrivateKeyCacheImpl}, + item::ItemHeaderIter, +}; use super::*; @@ -476,6 +481,73 @@ pub async fn store_item( } } +pub async fn remove_item( + flash: &mut S, + flash_range: Range, + mut cache: impl KeyCacheImpl, + data_buffer: &mut [u8], + search_key: I::Key, +) -> Result<(), MapError> { + // Search for the last used page. We're gonna erase from the one after this first. + // If we get an early shutoff or cancellation, this will make it so that we don't return + // an old version of the key on the next fetch. + let last_used_page = find_first_page( + flash, + flash_range.clone(), + &mut cache, + 0, + PageState::PartialOpen, + ) + .await? + .unwrap_or_default(); + + // Go through all the pages + for page_index in get_pages::( + flash_range.clone(), + next_page::(flash_range.clone(), last_used_page), + ) { + if get_page_state(flash, flash_range.clone(), &mut cache, page_index) + .await? + .is_open() + { + // This page is open, we don't have to check it + continue; + } + + let page_data_start_address = + calculate_page_address::(flash_range.clone(), page_index) + S::WORD_SIZE as u32; + let page_data_end_address = + calculate_page_end_address::(flash_range.clone(), page_index) - S::WORD_SIZE as u32; + + // Go through all items on the page + let mut item_headers = ItemHeaderIter::new(page_data_start_address, page_data_end_address); + + while let (Some(item_header), item_address) = item_headers.next(flash).await? { + let item = item_header + .read_item(flash, data_buffer, item_address, page_data_end_address) + .await?; + + match item { + item::MaybeItem::Corrupted(_, _) => continue, + item::MaybeItem::Erased(_, _) => continue, + item::MaybeItem::Present(item) => { + let item_key = I::deserialize_key_only(item.data()).map_err(MapError::Item)?; + + // If this item has the same key as the key we're trying to erase, then erase the item. + // But keep going! We need to erase everything. + if item_key == search_key { + item.header + .erase_data(flash, flash_range.clone(), &mut cache, item_address) + .await?; + } + } + } + } + } + + Ok(()) +} + /// A way of serializing and deserializing items in the storage. /// /// Serialized items must not be 0 bytes and may not be longer than [u16::MAX]. From 08d3d1b11a7fd161ad12fd49dbc7a88c0c6643e5 Mon Sep 17 00:00:00 2001 From: Dion Dokter Date: Tue, 13 Feb 2024 11:28:59 +0100 Subject: [PATCH 2/3] Add docs, test and fuzz --- fuzz/fuzz_targets/map.rs | 63 +++++++++++++++++++++++++- src/map.rs | 97 ++++++++++++++++++++++++++++++++++++++++ 2 files changed, 159 insertions(+), 1 deletion(-) diff --git a/fuzz/fuzz_targets/map.rs b/fuzz/fuzz_targets/map.rs index 79609bd..4b98f51 100644 --- a/fuzz/fuzz_targets/map.rs +++ b/fuzz/fuzz_targets/map.rs @@ -34,6 +34,7 @@ struct Input { enum Op { Store(StoreOp), Fetch(u8), + Remove(u8), } #[derive(Arbitrary, Debug, Clone)] @@ -107,7 +108,11 @@ enum CacheType { fn fuzz(ops: Input, mut cache: impl KeyCacheImpl) { let mut flash = MockFlashBase::::new( - WriteCountCheck::OnceOnly, + if ops.ops.iter().any(|op| matches!(op, Op::Remove(_))) { + WriteCountCheck::Twice + } else { + WriteCountCheck::OnceOnly + }, Some(ops.fuel as u32), true, ); @@ -243,6 +248,62 @@ fn fuzz(ops: Input, mut cache: impl KeyCacheImpl) { Err(e) => panic!("{e:?}"), } } + Op::Remove(key) => { + match block_on(sequential_storage::map::remove_item::( + &mut flash, + FLASH_RANGE, + &mut cache, + &mut buf.0, + key, + )) { + Ok(()) => { + map.remove(&key); + } + Err(MapError::Storage { + value: MockFlashError::EarlyShutoff(_), + backtrace: _backtrace, + }) => { + match block_on(sequential_storage::map::fetch_item::( + &mut flash, + FLASH_RANGE, + &mut cache, + &mut buf.0, + key, + )) { + Ok(Some(_)) => { + #[cfg(fuzzing_repro)] + eprintln!("Early shutoff when removing item {key}! Originated from:\n{_backtrace:#}"); + } + _ => { + // Could not fetch the item we stored... + #[cfg(fuzzing_repro)] + eprintln!("Early shutoff when removing item {key}! (but it still removed fully). Originated from:\n{_backtrace:#}"); + // Even though we got a shutoff, it still managed to store well + map.remove(&key); + } + } + } + Err(MapError::Corrupted { + backtrace: _backtrace, + }) if !corruption_repaired => { + #[cfg(fuzzing_repro)] + eprintln!( + "### Encountered curruption while fetching! Repairing now. Originated from:\n{_backtrace:#}" + ); + + block_on(sequential_storage::map::try_repair::( + &mut flash, + FLASH_RANGE, + &mut cache, + &mut buf.0, + )) + .unwrap(); + corruption_repaired = true; + retry = true; + } + Err(e) => panic!("{e:?}"), + } + } } } } diff --git a/src/map.rs b/src/map.rs index c868d61..c0c34bc 100644 --- a/src/map.rs +++ b/src/map.rs @@ -481,6 +481,15 @@ pub async fn store_item( } } +/// Fully remove an item. Additional calls to fetch with the same key will return None until +/// a new one is stored again. +/// +///
+/// This is really slow! +/// +/// All items in flash have to be read and deserialized to find the items with the key. +/// This is unlikely to be cached well. +///
pub async fn remove_item( flash: &mut S, flash_range: Range, @@ -488,6 +497,8 @@ pub async fn remove_item( data_buffer: &mut [u8], search_key: I::Key, ) -> Result<(), MapError> { + cache.notice_key_erased(&search_key); + // Search for the last used page. We're gonna erase from the one after this first. // If we get an early shutoff or cancellation, this will make it so that we don't return // an old version of the key on the next fetch. @@ -545,6 +556,9 @@ pub async fn remove_item( } } + // We're done, we now know the cache is in a good state + cache.unmark_dirty(); + Ok(()) } @@ -1206,4 +1220,87 @@ mod tests { assert_eq!(item.value, vec![i as u8; LENGHT_PER_KEY[i]]); } } + + #[test] + async fn remove_items() { + let mut flash = mock_flash::MockFlashBase::<4, 1, 4096>::new( + mock_flash::WriteCountCheck::Twice, + None, + true, + ); + let mut data_buffer = AlignedBuf([0; 128]); + const FLASH_RANGE: Range = 0x0000..0x4000; + + // Add some data to flash + for j in 0..10 { + for i in 0..24 { + let item = MockStorageItem { + key: i as u8, + value: vec![i as u8; j + 2], + }; + + store_item::<_, _>( + &mut flash, + FLASH_RANGE, + cache::NoCache::new(), + &mut data_buffer, + &item, + ) + .await + .unwrap(); + } + } + + for j in (0..24).rev() { + // Are all things still in flash that we expect? + for i in 0..=j { + assert!(fetch_item::( + &mut flash, + FLASH_RANGE, + cache::NoCache::new(), + &mut data_buffer, + i + ) + .await + .unwrap() + .is_some()); + } + + // Remove the item + remove_item::( + &mut flash, + FLASH_RANGE, + cache::NoCache::new(), + &mut data_buffer, + j, + ) + .await + .unwrap(); + + // Are all things still in flash that we expect? + for i in 0..j { + assert!(fetch_item::( + &mut flash, + FLASH_RANGE, + cache::NoCache::new(), + &mut data_buffer, + i + ) + .await + .unwrap() + .is_some()); + } + + assert!(fetch_item::( + &mut flash, + FLASH_RANGE, + cache::NoCache::new(), + &mut data_buffer, + j + ) + .await + .unwrap() + .is_none()); + } + } } From 88db2a3a3f80088bb74e4decb20e9e84d9ceeb70 Mon Sep 17 00:00:00 2001 From: Dion Dokter Date: Tue, 13 Feb 2024 11:33:42 +0100 Subject: [PATCH 3/3] Prepare for release --- CHANGELOG.md | 4 ++++ Cargo.toml | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 64c7de2..fe364af 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,10 @@ ## Unreleased +## 0.9.1 13-02-24 + +- Added `remove_item` to map + ## 0.9.0 11-02-24 - *Breaking:* Storage item key must now also be clone diff --git a/Cargo.toml b/Cargo.toml index 6d19880..255da40 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "sequential-storage" -version = "0.9.0" +version = "0.9.1" edition = "2021" license = "MIT OR Apache-2.0" description = "A crate for storing data in flash with minimal erase cycles."