Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implementation of EIP-1153: Transient Storage using Disk Persistence and Lifecycle Management #1588

Merged
merged 52 commits into from
Jan 14, 2025
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
52 commits
Select commit Hold shift + click to select a range
801dc0c
initial implementation of EIP-1153: Transient Storage - TODOs remain …
snissn Nov 20, 2024
ef679de
fix transient state error message
snissn Nov 21, 2024
a98c058
remove pub visibility from transient_data_lifespan. TODO still is com…
snissn Nov 21, 2024
9f3e90d
further implementation of lifecycle management for transient data, TO…
snissn Nov 21, 2024
2ee1d9d
update evm contract State to use transient_data a composite of transi…
snissn Dec 2, 2024
31180a1
cargo fmt
snissn Dec 2, 2024
56d2458
update comments on tests
snissn Dec 2, 2024
312299f
add tstorage test contract files
snissn Dec 3, 2024
757131f
use a non optimized delete method to clear transient storage. TODO up…
snissn Dec 3, 2024
eb894ae
failing tests that need to be investigated
snissn Dec 3, 2024
3dfe377
add test confirming transient storage is cleared
snissn Dec 4, 2024
ab808a2
more test debugging
snissn Dec 4, 2024
20aefb5
cargo fmt
snissn Dec 4, 2024
99c1e6b
fix cargo check nit
snissn Dec 4, 2024
2b811de
simplify transient test
snissn Dec 7, 2024
89311f9
support nested and reentry in transient storage in integration tests
snissn Dec 7, 2024
af6bb99
progress on fixing removing tranisent storage lifespan from system, b…
snissn Dec 7, 2024
661755a
Merge branch 'master' into feat/transient_storage
snissn Dec 7, 2024
d93167b
use clear instead of loop and delete
snissn Dec 7, 2024
0ceb2d8
Merge branch 'master' into mikers/feat-tstore-reload-stuck2
snissn Dec 7, 2024
250d133
use kamt clear in reload
snissn Dec 7, 2024
3052b34
merge
snissn Dec 7, 2024
e614a36
unused var
snissn Dec 7, 2024
978b49d
remove Options from transient data in state and handle None better to…
snissn Dec 10, 2024
a47c21e
add current_transient_data_lifespan to state and refactor get_current…
snissn Dec 10, 2024
e26e519
fmt
snissn Dec 10, 2024
4feec0f
set in call to true because transient data calls message() which has …
snissn Dec 10, 2024
3d5deba
Update actors/evm/src/interpreter/system.rs
snissn Dec 10, 2024
22963df
Revert "Update actors/evm/src/interpreter/system.rs"
snissn Dec 10, 2024
41c41de
Revert "Revert "Update actors/evm/src/interpreter/system.rs""
snissn Dec 11, 2024
837a003
get get_current_transient_data_lifespan throw an error if it cannot f…
snissn Dec 11, 2024
3c6d9a7
change transient data origin from id to Address
snissn Dec 12, 2024
037939a
Revert "change transient data origin from id to Address"
snissn Dec 12, 2024
dddf19c
remove test: origin is always an ID address
snissn Dec 12, 2024
646ac96
Merge branch 'master' into feat/transient_storage
snissn Dec 12, 2024
074e16f
remove unused import
snissn Dec 12, 2024
7a99e10
Merge branch 'feat/transient_storage' of github.com:filecoin-project/…
snissn Dec 12, 2024
fb8025d
remove duplicate comment
snissn Dec 13, 2024
2c8febf
use self.current_transient_data_lifespan
snissn Dec 13, 2024
f1ac779
get_current_transient_data_lifespan does not need to be public
snissn Dec 13, 2024
7c6ba37
remove match from get_current_transient_data_lifespan and use an unwr…
snissn Dec 13, 2024
a005cf7
remove error case from new
snissn Dec 13, 2024
8eed0c9
remove allow non snake case from transient storage tests
snissn Dec 13, 2024
133f7b5
Update actors/evm/src/interpreter/system.rs
snissn Dec 13, 2024
1b57940
Update actors/evm/tests/basic.rs
snissn Dec 13, 2024
685fd63
Update actors/evm/tests/contracts/TransientStorageTest.sol
snissn Dec 13, 2024
ff09cfa
Update actors/evm/tests/contracts/TransientStorageTest.sol
snissn Dec 13, 2024
b96dbe5
Update actors/evm/tests/contracts/TransientStorageTest.sol
snissn Dec 13, 2024
b7ac7ba
remove now unused solidity test contracts that have previously been r…
snissn Dec 14, 2024
a43f34a
Merge branch 'feat/transient_storage' of github.com:filecoin-project/…
snissn Dec 14, 2024
260f401
forge fmt and solc recompile
snissn Dec 14, 2024
a2e4b37
add comments to transient storage test file with test information
snissn Dec 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions actors/evm/src/interpreter/execution.rs
Original file line number Diff line number Diff line change
Expand Up @@ -163,6 +163,8 @@ pub mod opcodes {
0x59: MSIZE,
0x5a: GAS,
0x5b: JUMPDEST,
0x5c: TLOAD,
0x5d: TSTORE,
0x5e: MCOPY,
0x5F: PUSH0,
0x60: PUSH1,
Expand Down Expand Up @@ -400,6 +402,8 @@ mod tests {
MSTORE8,
SLOAD,
SSTORE,
TLOAD,
TSTORE,
LOG0,
LOG1,
LOG2,
Expand Down
2 changes: 2 additions & 0 deletions actors/evm/src/interpreter/instructions/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -338,6 +338,8 @@ def_stdproc! { MSTORE(a, b) => memory::mstore }
def_stdproc! { MSTORE8(a, b) => memory::mstore8 }
def_stdfun! { SLOAD(a) => storage::sload }
def_stdproc! { SSTORE(a, b) => storage::sstore }
def_stdfun! { TLOAD(a) => storage::tload }
def_stdproc! { TSTORE(a, b) => storage::tstore }
def_stdfun! { MSIZE() => memory::msize }
def_stdfun! { GAS() => context::gas }
def_stdlog! { LOG0(0, ()) }
Expand Down
73 changes: 73 additions & 0 deletions actors/evm/src/interpreter/instructions/storage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,30 @@ pub fn sstore(
system.set_storage(key, value)
}

#[inline]
pub fn tload(
_state: &mut ExecutionState,
system: &mut System<impl Runtime>,
location: U256,
) -> Result<U256, ActorError> {
// get from storage and place on stack
system.get_transient_storage(location)
}

#[inline]
pub fn tstore(
_state: &mut ExecutionState,
system: &mut System<impl Runtime>,
key: U256,
value: U256,
) -> Result<(), ActorError> {
if system.readonly {
return Err(ActorError::read_only("store called while read-only".into()));
}

system.set_transient_storage(key, value)
}

#[cfg(test)]
mod tests {
use fil_actors_evm_shared::uints::U256;
Expand Down Expand Up @@ -82,4 +106,53 @@ mod tests {
assert_eq!(m.system.get_storage(U256::from(0)).unwrap(), U256::from(0x42));
};
}

// TODO test transient storage lifecycle

#[test]
fn test_tload() {
// happy path
evm_unit_test! {
(m) {
TLOAD;
}
m.system.set_transient_storage(U256::from(0), U256::from(0x42)).unwrap();
m.state.stack.push(U256::from(0)).unwrap();
let result = m.step();
assert!(result.is_ok(), "execution step failed");
assert_eq!(m.state.stack.len(), 1);
assert_eq!(m.state.stack.pop().unwrap(), U256::from(0x42));
};
}

#[test]
fn test_tload_oob() {
// oob access -- it is a zero
evm_unit_test! {
(m) {
TLOAD;
}
m.state.stack.push(U256::from(1234)).unwrap();
let result = m.step();
assert!(result.is_ok(), "execution step failed");
assert_eq!(m.state.stack.len(), 1);
assert_eq!(m.state.stack.pop().unwrap(), U256::from(0));
};
}

#[test]
fn test_tstore() {
evm_unit_test! {
(m) {
TSTORE;
}

m.state.stack.push(U256::from(0x42)).unwrap();
m.state.stack.push(U256::from(0)).unwrap();
let result = m.step();
assert!(result.is_ok(), "execution step failed");
assert_eq!(m.state.stack.len(), 0);
assert_eq!(m.system.get_transient_storage(U256::from(0)).unwrap(), U256::from(0x42));
};
}
}
61 changes: 59 additions & 2 deletions actors/evm/src/interpreter/system.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ use fvm_shared::error::{ErrorNumber, ExitCode};
use fvm_shared::sys::SendFlags;
use fvm_shared::{MethodNum, Response, IPLD_RAW, METHOD_SEND};

use crate::state::{State, Tombstone};
use crate::state::{State, Tombstone, TransientDataLifespan};
use crate::BytecodeHash;

use cid::Cid;
Expand Down Expand Up @@ -91,6 +91,11 @@ pub struct System<'r, RT: Runtime> {
bytecode: Option<EvmBytecode>,
/// The contract's EVM storage slots.
slots: StateKamt<RT::Blockstore>,

/// The contract's EVM transient storage slots.
transient_slots: StateKamt<RT::Blockstore>,
pub(crate) transient_data_lifespan: Option<TransientDataLifespan>,
snissn marked this conversation as resolved.
Show resolved Hide resolved

/// The contracts "nonce" (incremented when creating new actors).
pub(crate) nonce: u64,
/// The last saved state root. None if the current state hasn't been saved yet.
Expand All @@ -111,9 +116,12 @@ impl<'r, RT: Runtime> System<'r, RT> {
RT::Blockstore: Clone,
{
let store = rt.store().clone();
let transient_store = rt.store().clone();
Self {
rt,
slots: StateKamt::new_with_config(store, KAMT_CONFIG.clone()),
transient_slots: StateKamt::new_with_config(transient_store, KAMT_CONFIG.clone()),
transient_data_lifespan: None,
nonce: 1,
saved_state_root: None,
bytecode: None,
Expand Down Expand Up @@ -164,6 +172,7 @@ impl<'r, RT: Runtime> System<'r, RT> {
RT::Blockstore: Clone,
{
let store = rt.store().clone();
let transient_store = rt.store().clone();
let state_root = rt.get_state_root()?;
let state: State = store
.get_cbor(&state_root)
Expand All @@ -182,6 +191,13 @@ impl<'r, RT: Runtime> System<'r, RT> {
rt,
slots: StateKamt::load_with_config(&state.contract_state, store, KAMT_CONFIG.clone())
.context_code(ExitCode::USR_ILLEGAL_STATE, "state not in blockstore")?,
transient_slots: StateKamt::load_with_config(
&state.transient_state,
transient_store,
KAMT_CONFIG.clone(),
)
.context_code(ExitCode::USR_ILLEGAL_STATE, "state not in blockstore")?,
Stebalien marked this conversation as resolved.
Show resolved Hide resolved
transient_data_lifespan: state.transient_data_lifespan,
nonce: state.nonce,
saved_state_root: Some(state_root),
bytecode: Some(EvmBytecode::new(state.bytecode, state.bytecode_hash)),
Expand Down Expand Up @@ -255,7 +271,7 @@ impl<'r, RT: Runtime> System<'r, RT> {
Ok(result.map_err(|e| e.0))
}

/// Flush the actor state (bytecode, nonce, and slots).
/// Flush the actor state (bytecode, nonce, transient data and slots).
pub fn flush(&mut self) -> Result<(), ActorError> {
if self.saved_state_root.is_some() {
return Ok(());
Expand All @@ -281,6 +297,11 @@ impl<'r, RT: Runtime> System<'r, RT> {
ExitCode::USR_ILLEGAL_STATE,
"failed to flush contract state",
)?,
transient_state: self.transient_slots.flush().context_code(
ExitCode::USR_ILLEGAL_STATE,
"failed to flush contract state",
)?,
transient_data_lifespan: self.transient_data_lifespan,
nonce: self.nonce,
tombstone: self.tombstone,
},
Expand Down Expand Up @@ -314,6 +335,10 @@ impl<'r, RT: Runtime> System<'r, RT> {
self.slots
.set_root(&state.contract_state)
.context_code(ExitCode::USR_ILLEGAL_STATE, "state not in blockstore")?;
self.transient_slots
Stebalien marked this conversation as resolved.
Show resolved Hide resolved
.set_root(&state.transient_state)
.context_code(ExitCode::USR_ILLEGAL_STATE, "transient_state not in blockstore")?;
self.transient_data_lifespan = state.transient_data_lifespan;
self.nonce = state.nonce;
self.saved_state_root = Some(root);
self.bytecode = Some(EvmBytecode::new(state.bytecode, state.bytecode_hash));
Expand Down Expand Up @@ -384,6 +409,38 @@ impl<'r, RT: Runtime> System<'r, RT> {
};
Ok(())
}
///
/// Get value of a storage key.
pub fn get_transient_storage(&mut self, key: U256) -> Result<U256, ActorError> {
//TODO check tombstone for liveliness of data
Ok(self
.transient_slots
.get(&key)
.context_code(ExitCode::USR_ILLEGAL_STATE, "failed to clear storage slot")?
.cloned()
.unwrap_or_default())
}

/// Set value of a storage key.
pub fn set_transient_storage(&mut self, key: U256, value: U256) -> Result<(), ActorError> {
//TODO check tombstone for liveliness of data
let changed = if value.is_zero() {
self.transient_slots
.delete(&key)
.map(|v| v.is_some())
.context_code(ExitCode::USR_ILLEGAL_STATE, "failed to clear storage slot")?
} else {
self.transient_slots
.set(key, value)
.map(|v| v != Some(value))
.context_code(ExitCode::USR_ILLEGAL_STATE, "failed to update storage slot")?
};

if changed {
self.saved_state_root = None; // dirty.
};
Ok(())
}

/// Resolve the address to the ethereum equivalent, if possible.
///
Expand Down
14 changes: 14 additions & 0 deletions actors/evm/src/state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,15 @@ pub struct Tombstone {
pub nonce: u64,
}

/// A structure representing the transient data lifespan.
#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize_tuple, Deserialize_tuple)]
pub struct TransientDataLifespan {
/// The origin actor ID associated with the transient data.
pub origin: ActorID,
/// A unique nonce identifying the transaction.
pub nonce: u64,
}

/// A Keccak256 digest of EVM bytecode.
#[derive(Deserialize, Serialize, Clone, Copy, Eq, PartialEq)]
#[serde(transparent)]
Expand Down Expand Up @@ -106,6 +115,11 @@ pub struct State {
/// KAMT<U256, U256>
pub contract_state: Cid,

/// The EVM contract state diciontary that represents the transient storage
pub transient_state: Cid,
/// The nonce and actor id that represents the lifespan of the transient storage data
pub transient_data_lifespan: Option<TransientDataLifespan>,
Stebalien marked this conversation as resolved.
Show resolved Hide resolved

/// The EVM nonce used to track how many times CREATE or CREATE2 have been called.
pub nonce: u64,

Expand Down
Loading