revm/db/states/transition_account.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
use super::{AccountRevert, BundleAccount, StorageWithOriginalValues};
use crate::db::AccountStatus;
use revm_interpreter::primitives::{hash_map, AccountInfo, Bytecode, B256, I256, U256};
/// Account Created when EVM state is merged to cache state.
/// And it is sent to Block state.
///
/// It is used when block state gets merged to bundle state to
/// create needed Reverts.
#[derive(Clone, Debug, PartialEq, Eq, Default)]
pub struct TransitionAccount {
pub info: Option<AccountInfo>,
pub status: AccountStatus,
/// Previous account info is needed for account that got initially loaded.
/// Initially loaded account are not present inside bundle and are needed
/// to generate Reverts.
pub previous_info: Option<AccountInfo>,
/// Mostly needed when previous status Loaded/LoadedEmpty.
pub previous_status: AccountStatus,
/// Storage contains both old and new account
pub storage: StorageWithOriginalValues,
/// If there is transition that clears the storage we should mark it here and
/// delete all storages in BundleState. This flag is needed if we have transition
/// between Destroyed states from DestroyedChanged-> DestroyedAgain-> DestroyedChanged
/// in the end transition that we would have would be `DestroyedChanged->DestroyedChanged`
/// and with only that info we couldn't decide what to do.
pub storage_was_destroyed: bool,
}
impl TransitionAccount {
/// Create new LoadedEmpty account.
pub fn new_empty_eip161(storage: StorageWithOriginalValues) -> Self {
Self {
info: Some(AccountInfo::default()),
status: AccountStatus::InMemoryChange,
previous_info: None,
previous_status: AccountStatus::LoadedNotExisting,
storage,
storage_was_destroyed: false,
}
}
/// Return new contract bytecode if it is changed or newly created.
pub fn has_new_contract(&self) -> Option<(B256, &Bytecode)> {
let present_new_codehash = self.info.as_ref().map(|info| &info.code_hash);
let previous_codehash = self.previous_info.as_ref().map(|info| &info.code_hash);
if present_new_codehash != previous_codehash {
return self
.info
.as_ref()
.and_then(|info| info.code.as_ref().map(|c| (info.code_hash, c)));
}
None
}
/// Return the balance of account before transition.
pub fn previous_balance(&self) -> U256 {
self.previous_info
.as_ref()
.map(|info| info.balance)
.unwrap_or_default()
}
/// Return the balance of account after transition.
pub fn current_balance(&self) -> U256 {
self.info
.as_ref()
.map(|info| info.balance)
.unwrap_or_default()
}
/// Calculate the change in account's balance for this transition.
/// Returns `None` if delta does not fit in [I256].
pub fn balance_delta(&self) -> Option<I256> {
let previous_balance = self.previous_balance();
let current_balance = self.current_balance();
let delta = I256::try_from(previous_balance.abs_diff(current_balance)).ok()?;
if current_balance >= previous_balance {
Some(delta)
} else {
delta.checked_neg()
}
}
/// Update new values of transition. Don't override old values.
/// Both account info and old storages need to be left intact.
pub fn update(&mut self, other: Self) {
self.info.clone_from(&other.info);
self.status = other.status;
// if transition is from some to destroyed drop the storage.
// This need to be done here as it is one increment of the state.
if matches!(
other.status,
AccountStatus::Destroyed | AccountStatus::DestroyedAgain
) {
self.storage = other.storage;
self.storage_was_destroyed = true;
} else {
// update changed values to this transition.
for (key, slot) in other.storage.into_iter() {
match self.storage.entry(key) {
hash_map::Entry::Vacant(entry) => {
entry.insert(slot);
}
hash_map::Entry::Occupied(mut entry) => {
let value = entry.get_mut();
// if new value is same as original value. Remove storage entry.
if value.original_value() == slot.present_value() {
entry.remove();
} else {
// if value is different, update transition present value;
value.present_value = slot.present_value;
}
}
}
}
}
}
/// Consume Self and create account revert from it.
pub fn create_revert(self) -> Option<AccountRevert> {
let mut previous_account = self.original_bundle_account();
previous_account.update_and_create_revert(self)
}
/// Present bundle account
pub fn present_bundle_account(&self) -> BundleAccount {
BundleAccount {
info: self.info.clone(),
original_info: self.previous_info.clone(),
storage: self.storage.clone(),
status: self.status,
}
}
/// Original bundle account
fn original_bundle_account(&self) -> BundleAccount {
BundleAccount {
info: self.previous_info.clone(),
original_info: self.previous_info.clone(),
storage: StorageWithOriginalValues::default(),
status: self.previous_status,
}
}
}