Preserve cache on reverting the snapshot (#2488)
* Preserve cache on reverting the snapshot * Renamed merge_with into replace_with * Renamed and documented snapshotting methods
This commit is contained in:
parent
e380955c34
commit
6c1b2fbed5
@ -265,7 +265,7 @@ impl<'a> Executive<'a> {
|
|||||||
let cost = self.engine.cost_of_builtin(¶ms.code_address, data);
|
let cost = self.engine.cost_of_builtin(¶ms.code_address, data);
|
||||||
if cost <= params.gas {
|
if cost <= params.gas {
|
||||||
self.engine.execute_builtin(¶ms.code_address, data, &mut output);
|
self.engine.execute_builtin(¶ms.code_address, data, &mut output);
|
||||||
self.state.clear_snapshot();
|
self.state.discard_snapshot();
|
||||||
|
|
||||||
// trace only top level calls to builtins to avoid DDoS attacks
|
// trace only top level calls to builtins to avoid DDoS attacks
|
||||||
if self.depth == 0 {
|
if self.depth == 0 {
|
||||||
@ -285,7 +285,7 @@ impl<'a> Executive<'a> {
|
|||||||
Ok(params.gas - cost)
|
Ok(params.gas - cost)
|
||||||
} else {
|
} else {
|
||||||
// just drain the whole gas
|
// just drain the whole gas
|
||||||
self.state.revert_snapshot();
|
self.state.revert_to_snapshot();
|
||||||
|
|
||||||
tracer.trace_failed_call(trace_info, vec![], evm::Error::OutOfGas.into());
|
tracer.trace_failed_call(trace_info, vec![], evm::Error::OutOfGas.into());
|
||||||
|
|
||||||
@ -331,7 +331,7 @@ impl<'a> Executive<'a> {
|
|||||||
res
|
res
|
||||||
} else {
|
} else {
|
||||||
// otherwise it's just a basic transaction, only do tracing, if necessary.
|
// otherwise it's just a basic transaction, only do tracing, if necessary.
|
||||||
self.state.clear_snapshot();
|
self.state.discard_snapshot();
|
||||||
|
|
||||||
tracer.trace_call(trace_info, U256::zero(), trace_output, vec![]);
|
tracer.trace_call(trace_info, U256::zero(), trace_output, vec![]);
|
||||||
Ok(params.gas)
|
Ok(params.gas)
|
||||||
@ -473,10 +473,10 @@ impl<'a> Executive<'a> {
|
|||||||
| Err(evm::Error::BadInstruction {.. })
|
| Err(evm::Error::BadInstruction {.. })
|
||||||
| Err(evm::Error::StackUnderflow {..})
|
| Err(evm::Error::StackUnderflow {..})
|
||||||
| Err(evm::Error::OutOfStack {..}) => {
|
| Err(evm::Error::OutOfStack {..}) => {
|
||||||
self.state.revert_snapshot();
|
self.state.revert_to_snapshot();
|
||||||
},
|
},
|
||||||
Ok(_) | Err(evm::Error::Internal) => {
|
Ok(_) | Err(evm::Error::Internal) => {
|
||||||
self.state.clear_snapshot();
|
self.state.discard_snapshot();
|
||||||
substate.accrue(un_substate);
|
substate.accrue(un_substate);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -394,7 +394,7 @@ impl Account {
|
|||||||
/// Replace self with the data from other account merging storage cache.
|
/// Replace self with the data from other account merging storage cache.
|
||||||
/// Basic account data and all modifications are overwritten
|
/// Basic account data and all modifications are overwritten
|
||||||
/// with new values.
|
/// with new values.
|
||||||
pub fn merge_with(&mut self, other: Account) {
|
pub fn overwrite_with(&mut self, other: Account) {
|
||||||
self.balance = other.balance;
|
self.balance = other.balance;
|
||||||
self.nonce = other.nonce;
|
self.nonce = other.nonce;
|
||||||
self.storage_root = other.storage_root;
|
self.storage_root = other.storage_root;
|
||||||
|
@ -93,7 +93,7 @@ impl AccountEntry {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a new account entry and mark it as dirty
|
// Create a new account entry and mark it as dirty.
|
||||||
fn new_dirty(account: Option<Account>) -> AccountEntry {
|
fn new_dirty(account: Option<Account>) -> AccountEntry {
|
||||||
AccountEntry {
|
AccountEntry {
|
||||||
account: account,
|
account: account,
|
||||||
@ -101,13 +101,27 @@ impl AccountEntry {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a new account entry and mark it as clean
|
// Create a new account entry and mark it as clean.
|
||||||
fn new_clean(account: Option<Account>) -> AccountEntry {
|
fn new_clean(account: Option<Account>) -> AccountEntry {
|
||||||
AccountEntry {
|
AccountEntry {
|
||||||
account: account,
|
account: account,
|
||||||
state: AccountState::Clean,
|
state: AccountState::Clean,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Replace data with another entry but preserve storage cache.
|
||||||
|
fn overwrite_with(&mut self, other: AccountEntry) {
|
||||||
|
self.state = other.state;
|
||||||
|
match other.account {
|
||||||
|
Some(acc) => match self.account {
|
||||||
|
Some(ref mut ours) => {
|
||||||
|
ours.overwrite_with(acc);
|
||||||
|
},
|
||||||
|
None => {},
|
||||||
|
},
|
||||||
|
None => self.account = None,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Representation of the entire state of all accounts in the system.
|
/// Representation of the entire state of all accounts in the system.
|
||||||
@ -142,10 +156,24 @@ impl AccountEntry {
|
|||||||
///
|
///
|
||||||
/// Upon destruction all the local cache data merged into the global cache.
|
/// Upon destruction all the local cache data merged into the global cache.
|
||||||
/// The merge might be rejected if current state is non-canonical.
|
/// The merge might be rejected if current state is non-canonical.
|
||||||
|
///
|
||||||
|
/// State snapshotting.
|
||||||
|
///
|
||||||
|
/// A new snapshot can be created with `snapshot()`. Snapshots can be
|
||||||
|
/// created in a hierarchy.
|
||||||
|
/// When a snapshot is active all changes are applied directly into
|
||||||
|
/// `cache` and the original value is copied into an active snapshot.
|
||||||
|
/// Reverting a snapshot with `revert_to_snapshot` involves copying
|
||||||
|
/// original values from the latest snapshot back into `cache`. The code
|
||||||
|
/// takes care not to overwrite cached storage while doing that.
|
||||||
|
/// Snapshot can be discateded with `discard_snapshot`. All of the orignal
|
||||||
|
/// backed-up values are moved into a parent snapshot (if any).
|
||||||
|
///
|
||||||
pub struct State {
|
pub struct State {
|
||||||
db: StateDB,
|
db: StateDB,
|
||||||
root: H256,
|
root: H256,
|
||||||
cache: RefCell<HashMap<Address, AccountEntry>>,
|
cache: RefCell<HashMap<Address, AccountEntry>>,
|
||||||
|
// The original account is preserved in
|
||||||
snapshots: RefCell<Vec<HashMap<Address, Option<AccountEntry>>>>,
|
snapshots: RefCell<Vec<HashMap<Address, Option<AccountEntry>>>>,
|
||||||
account_start_nonce: U256,
|
account_start_nonce: U256,
|
||||||
factories: Factories,
|
factories: Factories,
|
||||||
@ -199,31 +227,44 @@ impl State {
|
|||||||
Ok(state)
|
Ok(state)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a recoverable snaphot of this state
|
/// Create a recoverable snaphot of this state.
|
||||||
pub fn snapshot(&mut self) {
|
pub fn snapshot(&mut self) {
|
||||||
self.snapshots.borrow_mut().push(HashMap::new());
|
self.snapshots.borrow_mut().push(HashMap::new());
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Merge last snapshot with previous
|
/// Merge last snapshot with previous.
|
||||||
pub fn clear_snapshot(&mut self) {
|
pub fn discard_snapshot(&mut self) {
|
||||||
// merge with previous snapshot
|
// merge with previous snapshot
|
||||||
let last = self.snapshots.borrow_mut().pop();
|
let last = self.snapshots.borrow_mut().pop();
|
||||||
if let Some(mut snapshot) = last {
|
if let Some(mut snapshot) = last {
|
||||||
if let Some(ref mut prev) = self.snapshots.borrow_mut().last_mut() {
|
if let Some(ref mut prev) = self.snapshots.borrow_mut().last_mut() {
|
||||||
|
if prev.is_empty() {
|
||||||
|
**prev = snapshot;
|
||||||
|
} else {
|
||||||
for (k, v) in snapshot.drain() {
|
for (k, v) in snapshot.drain() {
|
||||||
prev.entry(k).or_insert(v);
|
prev.entry(k).or_insert(v);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Revert to snapshot
|
/// Revert to the last snapshot and discard it.
|
||||||
pub fn revert_snapshot(&mut self) {
|
pub fn revert_to_snapshot(&mut self) {
|
||||||
if let Some(mut snapshot) = self.snapshots.borrow_mut().pop() {
|
if let Some(mut snapshot) = self.snapshots.borrow_mut().pop() {
|
||||||
for (k, v) in snapshot.drain() {
|
for (k, v) in snapshot.drain() {
|
||||||
match v {
|
match v {
|
||||||
Some(v) => {
|
Some(v) => {
|
||||||
self.cache.borrow_mut().insert(k, v);
|
match self.cache.borrow_mut().entry(k) {
|
||||||
|
Entry::Occupied(mut e) => {
|
||||||
|
// Merge snapshotted changes back into the main account
|
||||||
|
// storage preserving the cache.
|
||||||
|
e.get_mut().overwrite_with(v);
|
||||||
|
},
|
||||||
|
Entry::Vacant(e) => {
|
||||||
|
e.insert(v);
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
None => {
|
None => {
|
||||||
match self.cache.borrow_mut().entry(k) {
|
match self.cache.borrow_mut().entry(k) {
|
||||||
@ -1717,12 +1758,12 @@ fn snapshot_basic() {
|
|||||||
state.snapshot();
|
state.snapshot();
|
||||||
state.add_balance(&a, &U256::from(69u64));
|
state.add_balance(&a, &U256::from(69u64));
|
||||||
assert_eq!(state.balance(&a), U256::from(69u64));
|
assert_eq!(state.balance(&a), U256::from(69u64));
|
||||||
state.clear_snapshot();
|
state.discard_snapshot();
|
||||||
assert_eq!(state.balance(&a), U256::from(69u64));
|
assert_eq!(state.balance(&a), U256::from(69u64));
|
||||||
state.snapshot();
|
state.snapshot();
|
||||||
state.add_balance(&a, &U256::from(1u64));
|
state.add_balance(&a, &U256::from(1u64));
|
||||||
assert_eq!(state.balance(&a), U256::from(70u64));
|
assert_eq!(state.balance(&a), U256::from(70u64));
|
||||||
state.revert_snapshot();
|
state.revert_to_snapshot();
|
||||||
assert_eq!(state.balance(&a), U256::from(69u64));
|
assert_eq!(state.balance(&a), U256::from(69u64));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1735,9 +1776,9 @@ fn snapshot_nested() {
|
|||||||
state.snapshot();
|
state.snapshot();
|
||||||
state.add_balance(&a, &U256::from(69u64));
|
state.add_balance(&a, &U256::from(69u64));
|
||||||
assert_eq!(state.balance(&a), U256::from(69u64));
|
assert_eq!(state.balance(&a), U256::from(69u64));
|
||||||
state.clear_snapshot();
|
state.discard_snapshot();
|
||||||
assert_eq!(state.balance(&a), U256::from(69u64));
|
assert_eq!(state.balance(&a), U256::from(69u64));
|
||||||
state.revert_snapshot();
|
state.revert_to_snapshot();
|
||||||
assert_eq!(state.balance(&a), U256::from(0));
|
assert_eq!(state.balance(&a), U256::from(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -191,7 +191,7 @@ impl StateDB {
|
|||||||
for (address, account) in self.cache_overlay.drain(..) {
|
for (address, account) in self.cache_overlay.drain(..) {
|
||||||
if let Some(&mut Some(ref mut existing)) = cache.accounts.get_mut(&address) {
|
if let Some(&mut Some(ref mut existing)) = cache.accounts.get_mut(&address) {
|
||||||
if let Some(new) = account {
|
if let Some(new) = account {
|
||||||
existing.merge_with(new);
|
existing.overwrite_with(new);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user