Skip to content

Commit

Permalink
replace the page arrays with vecmaps
Browse files Browse the repository at this point in the history
  • Loading branch information
tqn committed Aug 31, 2024
1 parent 8d3f249 commit 24146f5
Showing 1 changed file with 31 additions and 26 deletions.
57 changes: 31 additions & 26 deletions crates/core/executor/src/mmu.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ use std::mem::size_of;

use serde::{Deserialize, Serialize};
use serde_with::serde_as;
use vec_map::VecMap;

use crate::events::MemoryRecord;

Expand All @@ -14,11 +15,11 @@ pub const MAX_PAGE_COUNT: usize = 1 << (u32::BITS as usize - LOG_BLOCK_LEN);

#[serde_as]
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct Page(#[serde_as(as = "Box<[_; BLOCK_LEN]>")] Box<[Option<MemoryRecord>; BLOCK_LEN]>);
pub struct Page(VecMap<MemoryRecord>);

impl Default for Page {
fn default() -> Self {
Self(Box::new([None; BLOCK_LEN]))
Self(VecMap::with_capacity(BLOCK_LEN))
}
}

Expand All @@ -45,35 +46,29 @@ pub mod vecmap_mmu {

pub fn get(&self, index: usize) -> Option<&MemoryRecord> {
let (upper, lower) = Self::split_index(index);
self.page_table.get(upper)?.0[lower].as_ref()
self.page_table.get(upper)?.0.get(lower)
}

pub fn get_mut(&mut self, index: usize) -> Option<&mut MemoryRecord> {
let (upper, lower) = Self::split_index(index);
self.page_table.get_mut(upper)?.0[lower].as_mut()
self.page_table.get_mut(upper)?.0.get_mut(lower)
}

pub fn insert(&mut self, index: usize, value: MemoryRecord) {
pub fn insert(&mut self, index: usize, value: MemoryRecord) -> Option<MemoryRecord> {
let (upper, lower) = Self::split_index(index);
if let Some(block) = self.page_table.get_mut(upper) {
block.0[lower] = Some(value);
} else {
let mut new_block = Page::default();
new_block.0[lower] = Some(value);
self.page_table.insert(upper, new_block);
}
self.page_table.entry(upper).or_insert_with(Page::default).0.insert(lower, value)
}

pub fn remove(&mut self, index: usize) -> Option<MemoryRecord> {
let (upper, lower) = Self::split_index(index);
take(&mut self.page_table.get_mut(upper)?.0[lower])
self.page_table.get_mut(upper)?.0.remove(lower)
}

pub fn entry(&mut self, index: usize) -> Entry<'_> {
let (upper, lower) = Self::split_index(index);
let page_table_entry = self.page_table.entry(upper);
if let vec_map::Entry::Occupied(occ_entry) = page_table_entry {
if occ_entry.get().0[lower].is_some() {
if occ_entry.get().0.contains_key(lower) {
Entry::Occupied(OccupiedEntry { lower, page_table_occupied_entry: occ_entry })
} else {
Entry::Vacant(VacantEntry {
Expand All @@ -87,12 +82,9 @@ pub mod vecmap_mmu {
}

pub fn keys(&self) -> impl Iterator<Item = usize> + '_ {
self.page_table.iter().flat_map(|(upper, page)| {
page.0
.iter()
.enumerate()
.filter_map(move |(lower, record)| record.is_some().then_some(upper + lower))
})
self.page_table
.iter()
.flat_map(|(upper, page)| page.0.iter().map(move |(lower, _)| upper + lower))
}

#[inline]
Expand All @@ -114,8 +106,21 @@ pub mod vecmap_mmu {
impl<'a> VacantEntry<'a> {
pub fn insert(self, value: MemoryRecord) -> &'a mut MemoryRecord {
// By construction, the slot in the page is `None`.
self.page_table_entry.or_insert_with(Default::default).0[self.index & BLOCK_MASK]
.insert(value)
match self
.page_table_entry
.or_insert_with(Default::default)
.0
.entry(self.index & BLOCK_MASK)
{
vec_map::Entry::Vacant(entry) => entry.insert(value),
vec_map::Entry::Occupied(entry) => {
panic!(
"entry at {} should be vacant, but found {:?}",
self.index,
entry.into_mut()
)
}
}
}

pub fn into_key(self) -> usize {
Expand All @@ -134,23 +139,23 @@ pub mod vecmap_mmu {

impl<'a> OccupiedEntry<'a> {
pub fn get(&self) -> &MemoryRecord {
self.page_table_occupied_entry.get().0[self.lower].as_ref().unwrap()
self.page_table_occupied_entry.get().0.get(self.lower).unwrap()
}

pub fn get_mut(&mut self) -> &mut MemoryRecord {
self.page_table_occupied_entry.get_mut().0[self.lower].as_mut().unwrap()
self.page_table_occupied_entry.get_mut().0.get_mut(self.lower).unwrap()
}

pub fn insert(&mut self, value: MemoryRecord) -> MemoryRecord {
replace(self.get_mut(), value)
}

pub fn into_mut(self) -> &'a mut MemoryRecord {
self.page_table_occupied_entry.into_mut().0[self.lower].as_mut().unwrap()
self.page_table_occupied_entry.into_mut().0.get_mut(self.lower).unwrap()
}

pub fn remove(mut self) -> MemoryRecord {
self.page_table_occupied_entry.get_mut().0[self.lower].take().unwrap()
self.page_table_occupied_entry.get_mut().0.remove(self.lower).unwrap()
}
}

Expand Down

0 comments on commit 24146f5

Please sign in to comment.