plonky2/evm/tests/erc721.rs

315 lines
26 KiB
Rust
Raw Normal View History

use std::str::FromStr;
use std::time::Duration;
use env_logger::{try_init_from_env, Env, DEFAULT_FILTER_ENV};
use eth_trie_utils::nibbles::Nibbles;
use eth_trie_utils::partial_trie::{HashedPartialTrie, PartialTrie};
use ethereum_types::{Address, BigEndianHash, H160, H256, U256};
use hex_literal::hex;
use keccak_hash::keccak;
use plonky2::field::goldilocks_field::GoldilocksField;
use plonky2::plonk::config::KeccakGoldilocksConfig;
use plonky2::util::timing::TimingTree;
use plonky2_evm::all_stark::AllStark;
use plonky2_evm::config::StarkConfig;
use plonky2_evm::generation::mpt::{AccountRlp, LegacyReceiptRlp, LogRlp};
use plonky2_evm::generation::{GenerationInputs, TrieInputs};
use plonky2_evm::proof::{BlockHashes, BlockMetadata, TrieRoots};
use plonky2_evm::prover::prove;
use plonky2_evm::verifier::verify_proof;
use plonky2_evm::Node;
type F = GoldilocksField;
const D: usize = 2;
type C = KeccakGoldilocksConfig;
/// Test a simple ERC721 token transfer.
/// Used the following Solidity code:
/// ```solidity
/// pragma solidity ^0.8.20;
///
/// import "@openzeppelin/contracts@5.0.1/token/ERC721/ERC721.sol";
/// import "@openzeppelin/contracts@5.0.1/access/Ownable.sol";
///
/// contract TestToken is ERC721, Ownable {
/// constructor(address initialOwner)
/// ERC721("TestToken", "TEST")
/// Ownable(initialOwner)
/// {}
///
/// function safeMint(address to, uint256 tokenId) public onlyOwner {
/// _safeMint(to, tokenId);
/// }
/// }
/// ```
///
/// The transaction calls the `safeTransferFrom` function to transfer token `1337` from address
/// `0x5B38Da6a701c568545dCfcB03FcB875f56beddC4` to address `0xAb8483F64d9C6d1EcF9b849Ae677dD3315835cb2`.
#[test]
fn test_erc721() -> anyhow::Result<()> {
init_logger();
let all_stark = AllStark::<F, D>::default();
let config = StarkConfig::standard_fast_config();
let beneficiary = hex!("deadbeefdeadbeefdeadbeefdeadbeefdeadbeef");
let owner = hex!("5B38Da6a701c568545dCfcB03FcB875f56beddC4");
let contract = hex!("f2B1114C644cBb3fF63Bf1dD284c8Cd716e95BE9");
let owner_state_key = keccak(owner);
let contract_state_key = keccak(contract);
let owner_nibbles = Nibbles::from_bytes_be(owner_state_key.as_bytes()).unwrap();
let contract_nibbles = Nibbles::from_bytes_be(contract_state_key.as_bytes()).unwrap();
let mut state_trie_before = HashedPartialTrie::from(Node::Empty);
state_trie_before.insert(owner_nibbles, rlp::encode(&owner_account()).to_vec());
state_trie_before.insert(contract_nibbles, rlp::encode(&contract_account()).to_vec());
let storage_tries = vec![(contract_state_key, contract_storage())];
let tries_before = TrieInputs {
state_trie: state_trie_before,
transactions_trie: HashedPartialTrie::from(Node::Empty),
receipts_trie: HashedPartialTrie::from(Node::Empty),
storage_tries,
};
let txn = signed_tx();
let gas_used = 58_418.into();
let contract_code = [contract_bytecode(), vec![]]
.map(|v| (keccak(v.clone()), v))
.into();
let expected_state_trie_after: HashedPartialTrie = {
let mut state_trie_after = HashedPartialTrie::from(Node::Empty);
let owner_account = owner_account();
let owner_account_after = AccountRlp {
nonce: owner_account.nonce + 1,
balance: owner_account.balance - gas_used * 0xa,
..owner_account
};
state_trie_after.insert(owner_nibbles, rlp::encode(&owner_account_after).to_vec());
let contract_account_after = AccountRlp {
storage_root: contract_storage_after().hash(),
..contract_account()
};
state_trie_after.insert(
contract_nibbles,
rlp::encode(&contract_account_after).to_vec(),
);
state_trie_after
};
let logs = vec![LogRlp {
address: H160::from_str("0xf2B1114C644cBb3fF63Bf1dD284c8Cd716e95BE9").unwrap(),
topics: vec![
H256::from_str("0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef")
.unwrap(),
H256::from_str("0x0000000000000000000000005b38da6a701c568545dcfcb03fcb875f56beddc4")
.unwrap(),
H256::from_str("0x000000000000000000000000ab8483f64d9c6d1ecf9b849ae677dd3315835cb2")
.unwrap(),
H256::from_str("0x0000000000000000000000000000000000000000000000000000000000000539")
.unwrap(),
],
data: vec![].into(),
}];
let mut bloom_bytes = [0u8; 256];
add_logs_to_bloom(&mut bloom_bytes, &logs);
let receipt_0 = LegacyReceiptRlp {
status: true,
cum_gas_used: gas_used,
bloom: bloom_bytes.to_vec().into(),
logs,
};
let mut receipts_trie = HashedPartialTrie::from(Node::Empty);
receipts_trie.insert(Nibbles::from_str("0x80").unwrap(), receipt_0.encode(0));
let transactions_trie: HashedPartialTrie = Node::Leaf {
nibbles: Nibbles::from_str("0x80").unwrap(),
value: txn.to_vec(),
}
.into();
let trie_roots_after = TrieRoots {
state_root: expected_state_trie_after.hash(),
transactions_root: transactions_trie.hash(),
receipts_root: receipts_trie.hash(),
};
let bloom = bloom_bytes
.chunks_exact(32)
.map(U256::from_big_endian)
.collect::<Vec<_>>();
let block_metadata = BlockMetadata {
block_beneficiary: Address::from(beneficiary),
block_timestamp: 0x03e8.into(),
block_number: 1.into(),
block_difficulty: 0x020000.into(),
block_random: H256::from_uint(&0x020000.into()),
block_gaslimit: 0xff112233u32.into(),
block_chain_id: 1.into(),
block_base_fee: 0xa.into(),
block_gas_used: gas_used,
block_bloom: bloom.try_into().unwrap(),
};
let inputs = GenerationInputs {
signed_txn: Some(txn.to_vec()),
withdrawals: vec![],
tries: tries_before,
trie_roots_after,
contract_code,
checkpoint_state_trie_root: HashedPartialTrie::from(Node::Empty).hash(),
block_metadata,
txn_number_before: 0.into(),
gas_used_before: 0.into(),
gas_used_after: gas_used,
block_hashes: BlockHashes {
prev_hashes: vec![H256::default(); 256],
cur_hash: H256::default(),
},
};
let mut timing = TimingTree::new("prove", log::Level::Debug);
let proof = prove::<F, C, D>(&all_stark, &config, inputs, &mut timing, None)?;
timing.filter(Duration::from_millis(100)).print();
verify_proof(&all_stark, proof, &config)
}
fn init_logger() {
let _ = try_init_from_env(Env::default().filter_or(DEFAULT_FILTER_ENV, "info"));
}
fn contract_bytecode() -> Vec<u8> {
hex!("608060405234801561000f575f80fd5b5060043610610109575f3560e01c8063715018a6116100a0578063a22cb4651161006f578063a22cb465146102a1578063b88d4fde146102bd578063c87b56dd146102d9578063e985e9c514610309578063f2fde38b1461033957610109565b8063715018a61461023f5780638da5cb5b1461024957806395d89b4114610267578063a14481941461028557610109565b806323b872dd116100dc57806323b872dd146101a757806342842e0e146101c35780636352211e146101df57806370a082311461020f57610109565b806301ffc9a71461010d57806306fdde031461013d578063081812fc1461015b578063095ea7b31461018b575b5f80fd5b61012760048036038101906101229190611855565b610355565b604051610134919061189a565b60405180910390f35b610145610436565b604051610152919061193d565b60405180910390f35b61017560048036038101906101709190611990565b6104c5565b60405161018291906119fa565b60405180910390f35b6101a560048036038101906101a09190611a3d565b6104e0565b005b6101c160048036038101906101bc9190611a7b565b6104f6565b005b6101dd60048036038101906101d89190611a7b565b6105f5565b005b6101f960048036038101906101f49190611990565b610614565b60405161020691906119fa565b60405180910390f35b61022960048036038101906102249190611acb565b610625565b6040516102369190611b05565b60405180910390f35b6102476106db565b005b6102516106ee565b60405161025e91906119fa565b60405180910390f35b61026f610716565b60405161027c919061193d565b60405180910390f35b61029f600480360381019061029a9190611a3d565b6107a6565b005b6102bb60048036038101906102b69190611b48565b6107bc565b005b6102d760048036038101906102d29190611cb2565b6107d2565b005b6102f360048036038101906102ee9190611990565b6107ef565b604051610300919061193d565b60405180910390f35b610323600480360381019061031e9190611d32565b610855565b604051610330919061189a565b60405180910390f35b610353600480360381019061034e9190611acb565b6108e3565b005b5f7f80ac58cd000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061041f57507f5b5e139f000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b8061042f575061042e82610967565b5b9050919050565b60605f805461044490611d9d565b80601f016020809104026020016040519081016040528092919081815260200182805461047090611d9d565b80156104bb5780601f10610492576101008083540402835291602001916104bb565b820191905f5260205f20905b81548152906001019060200180831161049e57829003601f168201915b5050505050905090565b5f6104cf826109d0565b506104d982610a56565b9050919050565b6104f282826104ed610a8f565b610a96565b5050565b5f73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610566575f6040517f64a0ae9200000000000000000000000000000000000000000000000000000000815260040161055d91906119fa565b60405180910390fd5b5f6105798383610574610a8f565b610aa8565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146105ef578382826040517f64283d7b0000000000000000000000000000000000000000000000000000000081526004016105e693929190611dcd565b60405180910390fd5b50505050565b61060f83838360405180602001604052805f8152506107d2565b505050565b5f61061e826109d0565b9050919050565b5f8073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610696575f6040517f89c62b6400000000000000000000000000000000000000000000000000000000815260040161068d91906119fa565b60405180910390fd5b60035f8373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020015f20549050919050565b6106e3610cb3565b6106ec5f610d3a565b565b5f60065f9054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b60606001805461072590611d9d565b80601f016020809104026020016040519081016040528092919081815260200182805461075190611d9d565b801561079c5780601f106107735761010080835404028352916020019161079c565b820191905f5260205f20905b81548152906001019060200180831161077f57829003601f168201915b5050505050905090565b6107ae610cb3565b6107b88282610dfd565b5050565b6107ce6107c7610a8f565b8383610e1a565b5050565b6107dd8484846104f6565b6107e984848484610f83565b50505050565b60606107fa826109d0565
}
fn insert_storage(trie: &mut HashedPartialTrie, slot: U256, value: U256) {
let mut bytes = [0; 32];
slot.to_big_endian(&mut bytes);
let key = keccak(bytes);
let nibbles = Nibbles::from_bytes_be(key.as_bytes()).unwrap();
let r = rlp::encode(&value);
let r = r.freeze().to_vec();
trie.insert(nibbles, r);
}
fn sd2u(s: &str) -> U256 {
U256::from_dec_str(s).unwrap()
}
fn sh2u(s: &str) -> U256 {
U256::from_str_radix(s, 16).unwrap()
}
fn contract_storage() -> HashedPartialTrie {
let mut trie = HashedPartialTrie::from(Node::Empty);
insert_storage(
&mut trie,
U256::zero(),
sh2u("0x54657374546f6b656e0000000000000000000000000000000000000000000012"),
);
insert_storage(
&mut trie,
U256::one(),
sh2u("0x5445535400000000000000000000000000000000000000000000000000000008"),
);
insert_storage(
&mut trie,
sd2u("6"),
sh2u("0x5b38da6a701c568545dcfcb03fcb875f56beddc4"),
);
insert_storage(
&mut trie,
sh2u("0x343ff8127bd64f680be4e996254dc3528603c6ecd54364b4cf956ebdd28f0028"),
sh2u("0x5b38da6a701c568545dcfcb03fcb875f56beddc4"),
);
insert_storage(
&mut trie,
sh2u("0x118c1ea466562cb796e30ef705e4db752f5c39d773d22c5efd8d46f67194e78a"),
sd2u("1"),
);
trie
}
fn contract_storage_after() -> HashedPartialTrie {
let mut trie = HashedPartialTrie::from(Node::Empty);
insert_storage(
&mut trie,
U256::zero(),
sh2u("0x54657374546f6b656e0000000000000000000000000000000000000000000012"),
);
insert_storage(
&mut trie,
U256::one(),
sh2u("0x5445535400000000000000000000000000000000000000000000000000000008"),
);
insert_storage(
&mut trie,
sd2u("6"),
sh2u("0x5b38da6a701c568545dcfcb03fcb875f56beddc4"),
);
insert_storage(
&mut trie,
sh2u("0x343ff8127bd64f680be4e996254dc3528603c6ecd54364b4cf956ebdd28f0028"),
sh2u("0xab8483f64d9c6d1ecf9b849ae677dd3315835cb2"),
);
insert_storage(
&mut trie,
sh2u("0xf3aa6a8a9f7e3707e36cc99c499a27514922afe861ec3d80a1a314409cba92f9"),
sd2u("1"),
);
trie
}
fn owner_account() -> AccountRlp {
AccountRlp {
nonce: 2.into(),
balance: 0x1000000.into(),
storage_root: HashedPartialTrie::from(Node::Empty).hash(),
code_hash: keccak([]),
}
}
fn contract_account() -> AccountRlp {
AccountRlp {
nonce: 0.into(),
balance: 0.into(),
storage_root: contract_storage().hash(),
code_hash: keccak(contract_bytecode()),
}
}
fn signed_tx() -> Vec<u8> {
hex!("f8c5020a8307a12094f2b1114c644cbb3ff63bf1dd284c8cd716e95be980b86442842e0e0000000000000000000000005b38da6a701c568545dcfcb03fcb875f56beddc4000000000000000000000000ab8483f64d9c6d1ecf9b849ae677dd3315835cb2000000000000000000000000000000000000000000000000000000000000053925a0414867f13ac63d663e84099d52c8215615666ea37c969c69aa58a0fad26a3f6ea01a7160c6274969083b2316eb8ca6011b4bf6b00972159a78bf64d06fa40c1402").into()
}
fn add_logs_to_bloom(bloom: &mut [u8; 256], logs: &Vec<LogRlp>) {
for log in logs {
add_to_bloom(bloom, log.address.as_bytes());
for topic in &log.topics {
add_to_bloom(bloom, topic.as_bytes());
}
}
}
fn add_to_bloom(bloom: &mut [u8; 256], bloom_entry: &[u8]) {
let bloom_hash = keccak(bloom_entry).to_fixed_bytes();
for idx in 0..3 {
let bit_pair = u16::from_be_bytes(bloom_hash[2 * idx..2 * (idx + 1)].try_into().unwrap());
let bit_to_set = 0x07FF - (bit_pair & 0x07FF);
let byte_index = bit_to_set / 8;
let bit_value = 1 << (7 - bit_to_set % 8);
bloom[byte_index as usize] |= bit_value;
}
}