From 038457a59499df48764a7b4dc0356d897c7b32ba Mon Sep 17 00:00:00 2001 From: Robin Salen <30937548+Nashtare@users.noreply.github.com> Date: Fri, 4 Oct 2024 15:17:08 -0400 Subject: [PATCH] fix: missing code read in state write (#699) --- trace_decoder/src/core.rs | 39 +++++++++++++++++++++++++++++++++------ 1 file changed, 33 insertions(+), 6 deletions(-) diff --git a/trace_decoder/src/core.rs b/trace_decoder/src/core.rs index eb598b6ae..c5aa890da 100644 --- a/trace_decoder/src/core.rs +++ b/trace_decoder/src/core.rs @@ -40,6 +40,12 @@ pub fn entrypoint( code_db, txn_info, } = trace; + + let fatal_missing_code = match trie_pre_images { + BlockTraceTriePreImages::Separate(_) => FatalMissingCode(true), + BlockTraceTriePreImages::Combined(_) => FatalMissingCode(false), + }; + let (state, storage, mut code) = start(trie_pre_images)?; code.extend(code_db); @@ -68,6 +74,7 @@ pub fn entrypoint( &b_meta, ger_data, withdrawals, + fatal_missing_code, observer, )?; @@ -270,6 +277,14 @@ pub struct IntraBlockTries { pub receipt: ReceiptTrie, } +/// Hacky handling of possibly missing contract bytecode in `Hash2Code` inner +/// map. +/// Allows incomplete payloads fetched with the zero tracer to skip these +/// silently. +// TODO(Nashtare): https://github.com/0xPolygonZero/zk_evm/issues/700 +#[derive(Copy, Clone)] +pub struct FatalMissingCode(pub bool); + /// Does the main work mentioned in the [module documentation](super). #[allow(clippy::too_many_arguments)] fn middle( @@ -285,6 +300,7 @@ fn middle( ger_data: Option<(H256, H256)>, // added to final batch mut withdrawals: Vec<(Address, U256)>, + fatal_missing_code: FatalMissingCode, // called with the untrimmed tries after each batch observer: &mut impl Observer, ) -> anyhow::Result>> { @@ -433,7 +449,21 @@ fn middle( acct.code_hash = code_usage .map(|it| match it { ContractCodeUsage::Read(hash) => { - batch_contract_code.insert(code.get(hash)?); + // TODO(Nashtare): https://github.com/0xPolygonZero/zk_evm/issues/700 + // This is a bug in the zero tracer, which shouldn't be giving us + // this read at all. Workaround for now. + match (fatal_missing_code, code.get(hash)) { + (FatalMissingCode(true), None) => { + bail!("no code for hash {hash:x}") + } + (_, Some(byte_code)) => { + batch_contract_code.insert(byte_code); + } + (_, None) => { + log::warn!("no code for {hash:x}") + } + } + anyhow::Ok(hash) } ContractCodeUsage::Write(bytes) => { @@ -767,11 +797,8 @@ impl Hash2Code { this.insert(vec![]); this } - pub fn get(&mut self, hash: H256) -> anyhow::Result> { - match self.inner.get(&hash) { - Some(code) => Ok(code.clone()), - None => bail!("no code for hash {:x}", hash), - } + pub fn get(&mut self, hash: H256) -> Option> { + self.inner.get(&hash).cloned() } pub fn insert(&mut self, code: Vec) { self.inner.insert(keccak_hash::keccak(&code), code);