Skip to content

Commit

Permalink
[TieredStorage] Implementation of AccountIndexFormat for hot accounts (
Browse files Browse the repository at this point in the history
…#32497)

#### Summary of Changes
This PR implements AccountIndexFormat::AddressAndOffset, the index format
that will be used by the hot account storage.

#### Test Plan
Unit tests are included in this PR.
Tested via the prototype implementation of tiered-storage.
  • Loading branch information
yhchiang-sol authored Jul 15, 2023
1 parent 2a86420 commit a2d9870
Show file tree
Hide file tree
Showing 3 changed files with 161 additions and 25 deletions.
1 change: 1 addition & 0 deletions runtime/src/tiered_storage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ pub mod error;
pub mod file;
pub mod footer;
pub mod hot;
pub mod index;
pub mod meta;
pub mod mmap_utils;
pub mod readable;
Expand Down
28 changes: 3 additions & 25 deletions runtime/src/tiered_storage/footer.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
use {
crate::tiered_storage::{
error::TieredStorageError, file::TieredStorageFile, mmap_utils::get_type,
TieredStorageResult as TsResult,
error::TieredStorageError, file::TieredStorageFile, index::AccountIndexFormat,
mmap_utils::get_type, TieredStorageResult as TsResult,
},
memmap2::Mmap,
solana_sdk::{hash::Hash, pubkey::Pubkey},
Expand Down Expand Up @@ -85,28 +85,6 @@ pub enum OwnersBlockFormat {
LocalIndex = 0,
}

#[repr(u16)]
#[derive(
Clone,
Copy,
Debug,
Default,
Eq,
Hash,
PartialEq,
num_enum::IntoPrimitive,
num_enum::TryFromPrimitive,
)]
pub enum AccountIndexFormat {
// This format does not support any fast lookup.
// Any query from account hash to account meta requires linear search.
#[default]
Linear = 0,
// Similar to index, but this format also stores the offset of each account
// meta in the index block.
LinearIndex = 1,
}

#[derive(Debug, PartialEq, Eq, Clone)]
#[repr(C)]
pub struct TieredStorageFooter {
Expand Down Expand Up @@ -262,7 +240,7 @@ mod tests {
let expected_footer = TieredStorageFooter {
account_meta_format: AccountMetaFormat::Hot,
owners_block_format: OwnersBlockFormat::LocalIndex,
account_index_format: AccountIndexFormat::Linear,
account_index_format: AccountIndexFormat::AddressAndOffset,
account_block_format: AccountBlockFormat::AlignedRaw,
account_entry_count: 300,
account_meta_entry_size: 24,
Expand Down
157 changes: 157 additions & 0 deletions runtime/src/tiered_storage/index.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
use {
crate::tiered_storage::{
file::TieredStorageFile, footer::TieredStorageFooter, mmap_utils::get_type,
TieredStorageResult,
},
memmap2::Mmap,
solana_sdk::pubkey::Pubkey,
};

/// The in-memory struct for the writing index block.
/// The actual storage format of a tiered account index entry might be different
/// from this.
#[derive(Debug)]
pub struct AccountIndexWriterEntry<'a> {
pub address: &'a Pubkey,
pub block_offset: u64,
pub intra_block_offset: u64,
}

/// The index format of a tiered accounts file.
#[repr(u16)]
#[derive(
Clone,
Copy,
Debug,
Default,
Eq,
Hash,
PartialEq,
num_enum::IntoPrimitive,
num_enum::TryFromPrimitive,
)]
pub enum AccountIndexFormat {
/// This format optimizes the storage size by storing only account addresses
/// and offsets. It skips storing the size of account data by storing account
/// block entries and index block entries in the same order.
#[default]
AddressAndOffset = 0,
}

impl AccountIndexFormat {
/// Persists the specified index_entries to the specified file and returns
/// the total number of bytes written.
pub fn write_index_block(
&self,
file: &TieredStorageFile,
index_entries: &[AccountIndexWriterEntry],
) -> TieredStorageResult<usize> {
match self {
Self::AddressAndOffset => {
let mut bytes_written = 0;
for index_entry in index_entries {
bytes_written += file.write_type(index_entry.address)?;
}
for index_entry in index_entries {
bytes_written += file.write_type(&index_entry.block_offset)?;
}
Ok(bytes_written)
}
}
}

/// Returns the address of the account given the specified index.
pub fn get_account_address<'a>(
&self,
map: &'a Mmap,
footer: &TieredStorageFooter,
index: usize,
) -> TieredStorageResult<&'a Pubkey> {
let offset = match self {
Self::AddressAndOffset => {
footer.account_index_offset as usize + std::mem::size_of::<Pubkey>() * index
}
};
let (address, _) = get_type::<Pubkey>(map, offset)?;
Ok(address)
}

/// Returns the offset to the account block that contains the account
/// associated with the specified index to the index block.
pub fn get_account_block_offset(
&self,
map: &Mmap,
footer: &TieredStorageFooter,
index: usize,
) -> TieredStorageResult<u64> {
match self {
Self::AddressAndOffset => {
let offset = footer.account_index_offset as usize
+ std::mem::size_of::<Pubkey>() * footer.account_entry_count as usize
+ index * std::mem::size_of::<u64>();
let (account_block_offset, _) = get_type(map, offset)?;
Ok(*account_block_offset)
}
}
}

/// Returns the size of one index entry.
pub fn entry_size(&self) -> usize {
match self {
Self::AddressAndOffset => std::mem::size_of::<Pubkey>() + std::mem::size_of::<u64>(),
}
}
}

#[cfg(test)]
mod tests {
use {
super::*, crate::tiered_storage::file::TieredStorageFile, memmap2::MmapOptions, rand::Rng,
std::fs::OpenOptions, tempfile::TempDir,
};

#[test]
fn test_address_and_offset_indexer() {
const ENTRY_COUNT: usize = 100;
let footer = TieredStorageFooter {
account_entry_count: ENTRY_COUNT as u32,
..TieredStorageFooter::default()
};
let temp_dir = TempDir::new().unwrap();
let path = temp_dir.path().join("test_address_and_offset_indexer");
let addresses: Vec<_> = std::iter::repeat_with(Pubkey::new_unique)
.take(ENTRY_COUNT)
.collect();
let mut rng = rand::thread_rng();
let index_entries: Vec<_> = addresses
.iter()
.map(|address| AccountIndexWriterEntry {
address,
block_offset: rng.gen_range(128, 2048),
intra_block_offset: 0,
})
.collect();

{
let file = TieredStorageFile::new_writable(&path);
let indexer = AccountIndexFormat::AddressAndOffset;
indexer.write_index_block(&file, &index_entries).unwrap();
}

let indexer = AccountIndexFormat::AddressAndOffset;
let file = OpenOptions::new()
.read(true)
.create(false)
.open(&path)
.unwrap();
let map = unsafe { MmapOptions::new().map(&file).unwrap() };
for (i, index_entry) in index_entries.iter().enumerate() {
assert_eq!(
index_entry.block_offset,
indexer.get_account_block_offset(&map, &footer, i).unwrap()
);
let address = indexer.get_account_address(&map, &footer, i).unwrap();
assert_eq!(index_entry.address, address);
}
}
}

0 comments on commit a2d9870

Please sign in to comment.