Skip to content

Commit

Permalink
Merge f0fa4ee into b0eaa9e
Browse files Browse the repository at this point in the history
  • Loading branch information
nkysg authored Apr 11, 2022
2 parents b0eaa9e + f0fa4ee commit 460ef73
Show file tree
Hide file tree
Showing 6 changed files with 66 additions and 39 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion contrib-contracts/src/starcoin_merkle_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ fn test_starcoin_merkle() -> Result<()> {
}

{
// change to previout state root.
// change to previous state root.
let old_chain_state = chain_state.fork_at(state_root);
// let state_root = chain_state.state_root();
let _expected_root = MoveValue::vector_u8(state_root.to_vec());
Expand Down
5 changes: 0 additions & 5 deletions executor/src/block_executor.rs
Original file line number Diff line number Diff line change
Expand Up @@ -63,11 +63,6 @@ pub fn block_execute(
let txn_state_root = chain_state
.commit()
.map_err(BlockExecutorError::BlockChainStateErr)?;
//every transaction's state tree root and tree nodes should save to storage
//TODO merge database flush.
chain_state
.flush()
.map_err(BlockExecutorError::BlockChainStateErr)?;

executed_data.txn_infos.push(TransactionInfo::new(
txn_hash,
Expand Down
1 change: 1 addition & 0 deletions state/state-tree/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ serde = { version = "1.0.130" }
forkable-jellyfish-merkle = { path = "../../commons/forkable-jellyfish-merkle"}
starcoin-state-store-api = {path = "../state-store-api"}
bcs-ext = { package="bcs-ext", path = "../../commons/bcs_ext" }
logger = { path = "../../commons/logger", package="starcoin-logger"}

[dev-dependencies]
starcoin-config= { path = "../../config"}
Expand Down
61 changes: 46 additions & 15 deletions state/state-tree/src/state_tree.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ use forkable_jellyfish_merkle::proof::SparseMerkleProof;
use forkable_jellyfish_merkle::{
JellyfishMerkleTree, RawKey, StaleNodeIndex, TreeReader, TreeUpdateBatch,
};
use logger::prelude::debug;
use parking_lot::{Mutex, RwLock};
use starcoin_crypto::hash::*;
use starcoin_state_store_api::*;
Expand All @@ -23,7 +24,8 @@ use std::sync::Arc;
#[derive(Clone)]
pub struct StateCache<K: RawKey> {
root_hash: HashValue,
change_set: TreeUpdateBatch<K>,
change_set_list: Vec<(HashValue, TreeUpdateBatch<K>)>,
split_off_idx: Option<usize>,
}

impl<K> StateCache<K>
Expand All @@ -33,17 +35,22 @@ where
pub fn new(initial_root: HashValue) -> Self {
Self {
root_hash: initial_root,
change_set: TreeUpdateBatch::default(),
change_set_list: Vec::new(),
split_off_idx: None,
}
}

fn reset(&mut self, root_hash: HashValue) {
self.root_hash = root_hash;
self.change_set = TreeUpdateBatch::default();
self.change_set_list = if let Some(split_idx) = self.split_off_idx {
self.change_set_list.split_off(split_idx)
} else {
Vec::new()
};
}

fn add_changeset(&mut self, root_hash: HashValue, cs: TreeUpdateBatch<K>) {
let cur_change_set = &mut self.change_set;
let mut cur_change_set = TreeUpdateBatch::default();
let mut cs_num_stale_leaves = cs.num_stale_leaves;
for stale_node in cs.stale_node_index_batch.iter() {
match cur_change_set.node_batch.remove(&stale_node.node_key) {
Expand All @@ -70,7 +77,8 @@ where
cur_change_set.num_new_leaves += 1;
}
}

self.change_set_list
.push((root_hash, cur_change_set));
self.root_hash = root_hash;
}
}
Expand Down Expand Up @@ -196,11 +204,25 @@ where

/// commit the state change into underline storage.
pub fn flush(&self) -> Result<()> {
let (root_hash, change_sets) = self.change_sets();
let change_set_list = {
let mut cache_guard = self.cache.lock();
cache_guard.split_off_idx = Some(cache_guard.change_set_list.len());
cache_guard.change_set_list.clone()
};

debug!("change_set_list len {}", change_set_list.len());
// when self::commit call self::updates(&self, updates: Vec<(K, Option<Blob>)>)
// the param updates is empty cause this situation
if change_set_list.is_empty() {
return Ok(());
}
let mut root_hash = HashValue::default();
let mut node_map = BTreeMap::new();
for (nk, n) in change_sets.node_batch.into_iter() {
node_map.insert(nk, n.try_into()?);
for (hash, change_sets) in change_set_list.into_iter() {
for (nk, n) in change_sets.node_batch.into_iter() {
node_map.insert(nk, n.try_into()?);
}
root_hash = hash;
}
self.storage.write_nodes(node_map)?;
// and then advance the storage root hash
Expand Down Expand Up @@ -259,9 +281,6 @@ where
};
let tree = JellyfishMerkleTree::new(&reader);
let (new_state_root, change_set) = tree.updates(Some(cur_root_hash), updates)?;
// cache.root_hashes.push(new_state_root);
// cache.change_sets.push(change_set);
// cache.root_hash = new_state_root;
cache.add_changeset(new_state_root, change_set);
Ok(new_state_root)
}
Expand Down Expand Up @@ -296,10 +315,18 @@ where
// }

/// get all changes so far based on initial root_hash.
/*
pub fn change_sets(&self) -> (HashValue, TreeUpdateBatch<K>) {
let cache_guard = self.cache.lock();
(cache_guard.root_hash, cache_guard.change_set.clone())
} */

/// get last changes root_hash
pub fn last_change_sets(&self) -> Option<(HashValue, TreeUpdateBatch<K>)> {
let cache_gurad = self.cache.lock();
cache_gurad.change_set_list.last().cloned()
}

// TODO: to keep atomic with other commit.
// TODO: think about the WriteBatch trait position.
// pub fn save<T>(&self, batch: &mut T) -> Result<()>
Expand Down Expand Up @@ -327,8 +354,10 @@ where
if node_key == &*SPARSE_MERKLE_PLACEHOLDER_HASH {
return Ok(Some(Node::new_null()));
}
if let Some(n) = self.cache.change_set.node_batch.get(node_key).cloned() {
return Ok(Some(n));
for change_set in self.cache.change_set_list.iter().rev() {
if let Some(n) = change_set.1.node_batch.get(node_key).cloned() {
return Ok(Some(n));
}
}
match self.store.get(node_key) {
Ok(Some(n)) => Ok(Some(n.try_into()?)),
Expand All @@ -350,8 +379,10 @@ where
if node_key == &*SPARSE_MERKLE_PLACEHOLDER_HASH {
return Ok(Some(Node::new_null()));
}
if let Some(n) = self.cache.change_set.node_batch.get(node_key).cloned() {
return Ok(Some(n));
for change_set in self.cache.change_set_list.iter().rev() {
if let Some(n) = change_set.1.node_batch.get(node_key).cloned() {
return Ok(Some(n));
}
}
match self.store.get(node_key) {
Ok(Some(n)) => Ok(Some(n.try_into()?)),
Expand Down
35 changes: 17 additions & 18 deletions state/state-tree/src/state_tree_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ pub fn test_put_blob() -> Result<()> {
assert_eq!(state.get(&account1)?, Some(vec![0, 0, 0]));
assert_eq!(state.get(&update_nibble(&hash_value, 0, 8))?, None);

let (root, updates) = state.change_sets();
let (root, updates) = state.last_change_sets().unwrap();
assert_eq!(root, new_root_hash);
assert_eq!(updates.num_stale_leaves, 0);
assert_eq!(updates.num_new_leaves, 1);
Expand All @@ -56,25 +56,25 @@ pub fn test_put_blob() -> Result<()> {
let new_root_hash = state.commit()?;
assert_eq!(state.root_hash(), new_root_hash);
assert_eq!(state.get(&account2)?, Some(vec![0, 0, 0]));
let (root, updates) = state.change_sets();
let (root, updates) = state.last_change_sets().unwrap();
assert_eq!(root, new_root_hash);
assert_eq!(updates.num_stale_leaves, 0);
assert_eq!(updates.num_new_leaves, 2);
assert_eq!(updates.node_batch.len(), 3);
assert_eq!(updates.stale_node_index_batch.len(), 1);
assert_eq!(updates.num_new_leaves, 1);
assert_eq!(updates.node_batch.len(), 2);
assert_eq!(updates.stale_node_index_batch.len(), 0);

// modify existed account
state.put(account1, vec![1, 1, 1]);
assert_eq!(state.get(&account1)?, Some(vec![1, 1, 1]));
let new_root_hash = state.commit()?;
assert_eq!(state.root_hash(), new_root_hash);
assert_eq!(state.get(&account1)?, Some(vec![1, 1, 1]));
let (root, updates) = state.change_sets();
let (root, updates) = state.last_change_sets().unwrap();
assert_eq!(root, new_root_hash);
assert_eq!(updates.num_stale_leaves, 0);
assert_eq!(updates.num_new_leaves, 2);
assert_eq!(updates.node_batch.len(), 3);
assert_eq!(updates.stale_node_index_batch.len(), 1);
assert_eq!(updates.num_stale_leaves, 1);
assert_eq!(updates.num_new_leaves, 1);
assert_eq!(updates.node_batch.len(), 2);
assert_eq!(updates.stale_node_index_batch.len(), 2);

let account3 = update_nibble(&account1, 2, 3);
for (k, v) in vec![(account1, vec![1, 1, 0]), (account3, vec![0, 0, 0])] {
Expand All @@ -90,11 +90,11 @@ pub fn test_put_blob() -> Result<()> {
assert_eq!(state.get(&account2)?, Some(vec![0, 0, 0]));
assert_eq!(state.get(&account3)?, Some(vec![0, 0, 0]));

let (_, updates) = state.change_sets();
assert_eq!(updates.num_stale_leaves, 0);
assert_eq!(updates.num_new_leaves, 3);
assert_eq!(updates.node_batch.len(), 6);
assert_eq!(updates.stale_node_index_batch.len(), 1);
let (_, updates) = state.last_change_sets().unwrap();
assert_eq!(updates.num_stale_leaves, 1);
assert_eq!(updates.num_new_leaves, 2);
assert_eq!(updates.node_batch.len(), 5);
assert_eq!(updates.stale_node_index_batch.len(), 2);
Ok(())
}

Expand Down Expand Up @@ -228,7 +228,7 @@ pub fn test_state_multi_commit_missing_node() -> Result<()> {
let state = StateTree::new(Arc::new(storage.clone()), None);
let hash_value1 = HashValueKey(HashValue::random());
let value1 = vec![1u8, 2u8];
state.put(hash_value1, value1);
state.put(hash_value1, value1.clone());
state.commit()?;
let root_hash1 = state.root_hash();
let hash_value2 = HashValueKey(HashValue::random());
Expand All @@ -240,8 +240,7 @@ pub fn test_state_multi_commit_missing_node() -> Result<()> {
state.flush()?;
let root_hash2 = state.root_hash();
let state1 = StateTree::new(Arc::new(storage.clone()), Some(root_hash1));
let result = state1.get(&hash_value1);
assert!(result.is_err(), "Missing node at HashValue");
assert_eq!(state1.get(&hash_value1)?, Some(value1));

let state2 = StateTree::new(Arc::new(storage), Some(root_hash2));
assert_eq!(state2.get(&hash_value1)?, Some(value12));
Expand Down

0 comments on commit 460ef73

Please sign in to comment.