2023-09-26 11:13:57 -04:00
|
|
|
use anyhow::{anyhow, Result};
|
2023-05-11 14:56:10 +02:00
|
|
|
use eth_trie_utils::nibbles::Nibbles;
|
|
|
|
|
use eth_trie_utils::partial_trie::{HashedPartialTrie, PartialTrie};
|
2024-01-09 11:15:53 +01:00
|
|
|
use ethereum_types::{BigEndianHash, H256, U512};
|
|
|
|
|
use rand::random;
|
2023-05-11 14:56:10 +02:00
|
|
|
|
|
|
|
|
use crate::cpu::kernel::aggregator::KERNEL;
|
|
|
|
|
use crate::cpu::kernel::constants::global_metadata::GlobalMetadata;
|
|
|
|
|
use crate::cpu::kernel::interpreter::Interpreter;
|
2023-12-07 12:08:47 -05:00
|
|
|
use crate::cpu::kernel::tests::account_code::initialize_mpts;
|
2023-05-11 14:56:10 +02:00
|
|
|
use crate::cpu::kernel::tests::mpt::{nibbles_64, test_account_1_rlp, test_account_2};
|
2023-12-07 12:08:47 -05:00
|
|
|
use crate::generation::mpt::AccountRlp;
|
2023-05-11 14:56:10 +02:00
|
|
|
use crate::generation::TrieInputs;
|
|
|
|
|
use crate::Node;
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn mpt_delete_empty() -> Result<()> {
|
|
|
|
|
test_state_trie(Default::default(), nibbles_64(0xABC), test_account_2())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn mpt_delete_leaf_nonoverlapping_keys() -> Result<()> {
|
|
|
|
|
let state_trie = Node::Leaf {
|
|
|
|
|
nibbles: nibbles_64(0xABC),
|
|
|
|
|
value: test_account_1_rlp(),
|
|
|
|
|
}
|
|
|
|
|
.into();
|
|
|
|
|
test_state_trie(state_trie, nibbles_64(0x123), test_account_2())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn mpt_delete_leaf_overlapping_keys() -> Result<()> {
|
|
|
|
|
let state_trie = Node::Leaf {
|
|
|
|
|
nibbles: nibbles_64(0xABC),
|
|
|
|
|
value: test_account_1_rlp(),
|
|
|
|
|
}
|
|
|
|
|
.into();
|
|
|
|
|
test_state_trie(state_trie, nibbles_64(0xADE), test_account_2())
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-11 18:57:17 +02:00
|
|
|
#[test]
|
|
|
|
|
fn mpt_delete_branch_into_hash() -> Result<()> {
|
|
|
|
|
let hash = Node::Hash(H256::random());
|
|
|
|
|
let state_trie = Node::Extension {
|
|
|
|
|
nibbles: nibbles_64(0xADF),
|
|
|
|
|
child: hash.into(),
|
|
|
|
|
}
|
|
|
|
|
.into();
|
|
|
|
|
test_state_trie(state_trie, nibbles_64(0xADE), test_account_2())
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 11:15:53 +01:00
|
|
|
#[test]
|
|
|
|
|
fn test_after_mpt_delete_extension_branch() -> Result<()> {
|
|
|
|
|
let hash = Node::Hash(H256::random());
|
|
|
|
|
let branch = Node::Branch {
|
|
|
|
|
children: std::array::from_fn(|i| {
|
|
|
|
|
if i == 0 {
|
|
|
|
|
Node::Empty.into()
|
|
|
|
|
} else {
|
|
|
|
|
hash.clone().into()
|
|
|
|
|
}
|
|
|
|
|
}),
|
|
|
|
|
value: vec![],
|
|
|
|
|
};
|
|
|
|
|
let nibbles = Nibbles::from_bytes_be(&random::<[u8; 5]>()).unwrap();
|
|
|
|
|
let state_trie = Node::Extension {
|
|
|
|
|
nibbles,
|
|
|
|
|
child: branch.into(),
|
|
|
|
|
}
|
|
|
|
|
.into();
|
|
|
|
|
let key = nibbles.merge_nibbles(&Nibbles {
|
|
|
|
|
packed: U512::zero(),
|
|
|
|
|
count: 64 - nibbles.count,
|
|
|
|
|
});
|
|
|
|
|
test_state_trie(state_trie, key, test_account_2())
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-11 14:56:10 +02:00
|
|
|
/// Note: The account's storage_root is ignored, as we can't insert a new storage_root without the
|
|
|
|
|
/// accompanying trie data. An empty trie's storage_root is used instead.
|
|
|
|
|
fn test_state_trie(
|
|
|
|
|
state_trie: HashedPartialTrie,
|
|
|
|
|
k: Nibbles,
|
|
|
|
|
mut account: AccountRlp,
|
|
|
|
|
) -> Result<()> {
|
|
|
|
|
assert_eq!(k.count, 64);
|
|
|
|
|
|
|
|
|
|
// Ignore any storage_root; see documentation note.
|
|
|
|
|
account.storage_root = HashedPartialTrie::from(Node::Empty).hash();
|
|
|
|
|
|
|
|
|
|
let trie_inputs = TrieInputs {
|
|
|
|
|
state_trie: state_trie.clone(),
|
|
|
|
|
transactions_trie: Default::default(),
|
|
|
|
|
receipts_trie: Default::default(),
|
|
|
|
|
storage_tries: vec![],
|
|
|
|
|
};
|
|
|
|
|
let mpt_insert_state_trie = KERNEL.global_labels["mpt_insert_state_trie"];
|
|
|
|
|
let mpt_delete = KERNEL.global_labels["mpt_delete"];
|
|
|
|
|
let mpt_hash_state_trie = KERNEL.global_labels["mpt_hash_state_trie"];
|
|
|
|
|
|
2023-12-07 12:08:47 -05:00
|
|
|
let initial_stack = vec![];
|
|
|
|
|
let mut interpreter = Interpreter::new_with_kernel(0, initial_stack);
|
|
|
|
|
|
|
|
|
|
initialize_mpts(&mut interpreter, &trie_inputs);
|
2023-05-11 14:56:10 +02:00
|
|
|
assert_eq!(interpreter.stack(), vec![]);
|
|
|
|
|
|
|
|
|
|
// Next, execute mpt_insert_state_trie.
|
|
|
|
|
interpreter.generation_state.registers.program_counter = mpt_insert_state_trie;
|
|
|
|
|
let trie_data = interpreter.get_trie_data_mut();
|
|
|
|
|
if trie_data.is_empty() {
|
|
|
|
|
// In the assembly we skip over 0, knowing trie_data[0] = 0 by default.
|
|
|
|
|
// Since we don't explicitly set it to 0, we need to do so here.
|
|
|
|
|
trie_data.push(0.into());
|
|
|
|
|
}
|
|
|
|
|
let value_ptr = trie_data.len();
|
|
|
|
|
trie_data.push(account.nonce);
|
|
|
|
|
trie_data.push(account.balance);
|
|
|
|
|
// In memory, storage_root gets interpreted as a pointer to a storage trie,
|
|
|
|
|
// so we have to ensure the pointer is valid. It's easiest to set it to 0,
|
|
|
|
|
// which works as an empty node, since trie_data[0] = 0 = MPT_TYPE_EMPTY.
|
|
|
|
|
trie_data.push(H256::zero().into_uint());
|
|
|
|
|
trie_data.push(account.code_hash.into_uint());
|
|
|
|
|
let trie_data_len = trie_data.len().into();
|
|
|
|
|
interpreter.set_global_metadata_field(GlobalMetadata::TrieDataSize, trie_data_len);
|
2023-12-18 19:11:16 +01:00
|
|
|
interpreter
|
|
|
|
|
.push(0xDEADBEEFu32.into())
|
|
|
|
|
.expect("The stack should not overflow");
|
|
|
|
|
interpreter
|
|
|
|
|
.push(value_ptr.into())
|
|
|
|
|
.expect("The stack should not overflow"); // value_ptr
|
|
|
|
|
interpreter
|
|
|
|
|
.push(k.try_into_u256().unwrap())
|
|
|
|
|
.expect("The stack should not overflow"); // key
|
2023-05-11 14:56:10 +02:00
|
|
|
interpreter.run()?;
|
|
|
|
|
assert_eq!(
|
|
|
|
|
interpreter.stack().len(),
|
|
|
|
|
0,
|
|
|
|
|
"Expected empty stack after insert, found {:?}",
|
|
|
|
|
interpreter.stack()
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Next, execute mpt_delete, deleting the account we just inserted.
|
|
|
|
|
let state_trie_ptr = interpreter.get_global_metadata_field(GlobalMetadata::StateTrieRoot);
|
|
|
|
|
interpreter.generation_state.registers.program_counter = mpt_delete;
|
2023-12-18 19:11:16 +01:00
|
|
|
interpreter
|
|
|
|
|
.push(0xDEADBEEFu32.into())
|
|
|
|
|
.expect("The stack should not overflow");
|
|
|
|
|
interpreter
|
|
|
|
|
.push(k.try_into_u256().unwrap())
|
|
|
|
|
.expect("The stack should not overflow");
|
|
|
|
|
interpreter
|
|
|
|
|
.push(64.into())
|
|
|
|
|
.expect("The stack should not overflow");
|
|
|
|
|
interpreter
|
|
|
|
|
.push(state_trie_ptr)
|
|
|
|
|
.expect("The stack should not overflow");
|
2023-05-11 14:56:10 +02:00
|
|
|
interpreter.run()?;
|
2023-12-18 19:11:16 +01:00
|
|
|
let state_trie_ptr = interpreter.pop().expect("The stack should not be empty");
|
2023-05-11 14:56:10 +02:00
|
|
|
interpreter.set_global_metadata_field(GlobalMetadata::StateTrieRoot, state_trie_ptr);
|
|
|
|
|
|
|
|
|
|
// Now, execute mpt_hash_state_trie.
|
|
|
|
|
interpreter.generation_state.registers.program_counter = mpt_hash_state_trie;
|
2023-12-18 19:11:16 +01:00
|
|
|
interpreter
|
|
|
|
|
.push(0xDEADBEEFu32.into())
|
|
|
|
|
.expect("The stack should not overflow");
|
|
|
|
|
interpreter
|
|
|
|
|
.push(1.into()) // Initial length of the trie data segment, unused.
|
|
|
|
|
.expect("The stack should not overflow");
|
2023-05-11 14:56:10 +02:00
|
|
|
interpreter.run()?;
|
|
|
|
|
|
2023-12-18 19:11:16 +01:00
|
|
|
let state_trie_hash =
|
|
|
|
|
H256::from_uint(&interpreter.pop().expect("The stack should not be empty"));
|
2023-05-11 14:56:10 +02:00
|
|
|
let expected_state_trie_hash = state_trie.hash();
|
|
|
|
|
assert_eq!(state_trie_hash, expected_state_trie_hash);
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|