Skip to main content

reth_engine_tree/tree/
trie_updates.rs

1use alloy_primitives::{
2    map::{B256Map, HashMap},
3    B256,
4};
5use reth_db::DatabaseError;
6use reth_trie::{
7    trie_cursor::{TrieCursor, TrieCursorFactory},
8    updates::{StorageTrieUpdates, TrieUpdates},
9    BranchNodeCompact, Nibbles,
10};
11use std::collections::BTreeSet;
12use tracing::warn;
13
14#[derive(Debug)]
15struct EntryDiff<T> {
16    task: T,
17    regular: T,
18    database: T,
19}
20
21#[derive(Debug, Default)]
22struct TrieUpdatesDiff {
23    account_nodes: HashMap<Nibbles, EntryDiff<Option<BranchNodeCompact>>>,
24    removed_nodes: HashMap<Nibbles, EntryDiff<bool>>,
25    storage_tries: B256Map<StorageTrieUpdatesDiff>,
26}
27
28impl TrieUpdatesDiff {
29    fn has_differences(&self) -> bool {
30        !self.account_nodes.is_empty() ||
31            !self.removed_nodes.is_empty() ||
32            !self.storage_tries.is_empty()
33    }
34
35    pub(super) fn log_differences(mut self) {
36        if self.has_differences() {
37            for (path, EntryDiff { task, regular, database }) in &mut self.account_nodes {
38                warn!(target: "engine::tree", ?path, ?task, ?regular, ?database, "Difference in account trie updates");
39            }
40
41            for (
42                path,
43                EntryDiff {
44                    task: task_removed,
45                    regular: regular_removed,
46                    database: database_not_exists,
47                },
48            ) in &self.removed_nodes
49            {
50                warn!(target: "engine::tree", ?path, ?task_removed, ?regular_removed, ?database_not_exists, "Difference in removed account trie nodes");
51            }
52
53            for (address, storage_diff) in self.storage_tries {
54                storage_diff.log_differences(address);
55            }
56        }
57    }
58}
59
60#[derive(Debug, Default)]
61struct StorageTrieUpdatesDiff {
62    is_deleted: Option<EntryDiff<bool>>,
63    storage_nodes: HashMap<Nibbles, EntryDiff<Option<BranchNodeCompact>>>,
64    removed_nodes: HashMap<Nibbles, EntryDiff<bool>>,
65}
66
67impl StorageTrieUpdatesDiff {
68    fn has_differences(&self) -> bool {
69        self.is_deleted.is_some() ||
70            !self.storage_nodes.is_empty() ||
71            !self.removed_nodes.is_empty()
72    }
73
74    fn log_differences(&self, address: B256) {
75        if let Some(EntryDiff {
76            task: task_deleted,
77            regular: regular_deleted,
78            database: database_not_exists,
79        }) = self.is_deleted
80        {
81            warn!(target: "engine::tree", ?address, ?task_deleted, ?regular_deleted, ?database_not_exists, "Difference in storage trie deletion");
82        }
83
84        for (path, EntryDiff { task, regular, database }) in &self.storage_nodes {
85            warn!(target: "engine::tree", ?address, ?path, ?task, ?regular, ?database, "Difference in storage trie updates");
86        }
87
88        for (
89            path,
90            EntryDiff {
91                task: task_removed,
92                regular: regular_removed,
93                database: database_not_exists,
94            },
95        ) in &self.removed_nodes
96        {
97            warn!(target: "engine::tree", ?address, ?path, ?task_removed, ?regular_removed, ?database_not_exists, "Difference in removed storage trie nodes");
98        }
99    }
100}
101
102/// Compares the trie updates from state root task, regular state root calculation and database,
103/// and logs the differences if there's any.
104pub(crate) fn compare_trie_updates(
105    trie_cursor_factory: impl TrieCursorFactory,
106    task: TrieUpdates,
107    regular: TrieUpdates,
108) -> Result<(), DatabaseError> {
109    let mut task = adjust_trie_updates(task);
110    let mut regular = adjust_trie_updates(regular);
111
112    let mut diff = TrieUpdatesDiff::default();
113
114    // compare account nodes
115    let mut account_trie_cursor = trie_cursor_factory.account_trie_cursor()?;
116    for key in task
117        .account_nodes
118        .keys()
119        .chain(regular.account_nodes.keys())
120        .copied()
121        .collect::<BTreeSet<_>>()
122    {
123        let (task, regular) = (task.account_nodes.remove(&key), regular.account_nodes.remove(&key));
124        let database = account_trie_cursor.seek_exact(key)?.map(|x| x.1);
125
126        if !branch_nodes_equal(task.as_ref(), regular.as_ref(), database.as_ref())? {
127            diff.account_nodes.insert(key, EntryDiff { task, regular, database });
128        }
129    }
130
131    // compare removed nodes
132    let mut account_trie_cursor = trie_cursor_factory.account_trie_cursor()?;
133    for key in task
134        .removed_nodes
135        .iter()
136        .chain(regular.removed_nodes.iter())
137        .copied()
138        .collect::<BTreeSet<_>>()
139    {
140        let (task_removed, regular_removed) =
141            (task.removed_nodes.contains(&key), regular.removed_nodes.contains(&key));
142        let database_not_exists = account_trie_cursor.seek_exact(key)?.is_none();
143        // If the deletion is a no-op, meaning that the entry is not in the
144        // database, do not add it to the diff.
145        if task_removed != regular_removed && !database_not_exists {
146            diff.removed_nodes.insert(
147                key,
148                EntryDiff {
149                    task: task_removed,
150                    regular: regular_removed,
151                    database: database_not_exists,
152                },
153            );
154        }
155    }
156
157    // compare storage tries
158    for key in task
159        .storage_tries
160        .keys()
161        .chain(regular.storage_tries.keys())
162        .copied()
163        .collect::<BTreeSet<_>>()
164    {
165        let (mut task, mut regular) =
166            (task.storage_tries.remove(&key), regular.storage_tries.remove(&key));
167        if task != regular {
168            #[expect(clippy::or_fun_call)]
169            let storage_diff = compare_storage_trie_updates(
170                || trie_cursor_factory.storage_trie_cursor(key),
171                // Compare non-existent storage tries as empty.
172                task.as_mut().unwrap_or(&mut Default::default()),
173                regular.as_mut().unwrap_or(&mut Default::default()),
174            )?;
175            if storage_diff.has_differences() {
176                diff.storage_tries.insert(key, storage_diff);
177            }
178        }
179    }
180
181    // log differences
182    diff.log_differences();
183
184    Ok(())
185}
186
187fn compare_storage_trie_updates<C: TrieCursor>(
188    trie_cursor: impl Fn() -> Result<C, DatabaseError>,
189    task: &mut StorageTrieUpdates,
190    regular: &mut StorageTrieUpdates,
191) -> Result<StorageTrieUpdatesDiff, DatabaseError> {
192    // Check if the storage trie exists by seeking to the first entry
193    let database_not_exists = trie_cursor()?.seek(Nibbles::default())?.is_none();
194    let mut diff = StorageTrieUpdatesDiff {
195        // If the deletion is a no-op, meaning that the entry is not in the
196        // database, do not add it to the diff.
197        is_deleted: (task.is_deleted != regular.is_deleted && !database_not_exists).then_some(
198            EntryDiff {
199                task: task.is_deleted,
200                regular: regular.is_deleted,
201                database: database_not_exists,
202            },
203        ),
204        ..Default::default()
205    };
206
207    // compare storage nodes
208    let mut storage_trie_cursor = trie_cursor()?;
209    for key in task
210        .storage_nodes
211        .keys()
212        .chain(regular.storage_nodes.keys())
213        .copied()
214        .collect::<BTreeSet<_>>()
215    {
216        let (task, regular) = (task.storage_nodes.remove(&key), regular.storage_nodes.remove(&key));
217        let database = storage_trie_cursor.seek_exact(key)?.map(|x| x.1);
218        if !branch_nodes_equal(task.as_ref(), regular.as_ref(), database.as_ref())? {
219            diff.storage_nodes.insert(key, EntryDiff { task, regular, database });
220        }
221    }
222
223    // compare removed nodes
224    let mut storage_trie_cursor = trie_cursor()?;
225    for key in
226        task.removed_nodes.iter().chain(regular.removed_nodes.iter()).collect::<BTreeSet<_>>()
227    {
228        let (task_removed, regular_removed) =
229            (task.removed_nodes.contains(key), regular.removed_nodes.contains(key));
230        if task_removed == regular_removed {
231            continue;
232        }
233        let database_not_exists = storage_trie_cursor.seek_exact(*key)?.map(|x| x.1).is_none();
234        // If the deletion is a no-op, meaning that the entry is not in the
235        // database, do not add it to the diff.
236        if !database_not_exists {
237            diff.removed_nodes.insert(
238                *key,
239                EntryDiff {
240                    task: task_removed,
241                    regular: regular_removed,
242                    database: database_not_exists,
243                },
244            );
245        }
246    }
247
248    Ok(diff)
249}
250
251/// Filters the removed nodes of both account trie updates and storage trie updates, so that they
252/// don't include those nodes that were also updated.
253fn adjust_trie_updates(trie_updates: TrieUpdates) -> TrieUpdates {
254    TrieUpdates {
255        removed_nodes: trie_updates
256            .removed_nodes
257            .into_iter()
258            .filter(|key| !trie_updates.account_nodes.contains_key(key))
259            .collect(),
260        storage_tries: trie_updates
261            .storage_tries
262            .into_iter()
263            .map(|(address, updates)| {
264                (
265                    address,
266                    StorageTrieUpdates {
267                        removed_nodes: updates
268                            .removed_nodes
269                            .into_iter()
270                            .filter(|key| !updates.storage_nodes.contains_key(key))
271                            .collect(),
272                        ..updates
273                    },
274                )
275            })
276            .collect(),
277        ..trie_updates
278    }
279}
280
281/// Compares the branch nodes from state root task and regular state root calculation.
282///
283/// If one of the branch nodes is [`None`], it means it's not updated and the other is compared to
284/// the branch node from the database.
285///
286/// Returns `true` if they are equal.
287fn branch_nodes_equal(
288    task: Option<&BranchNodeCompact>,
289    regular: Option<&BranchNodeCompact>,
290    database: Option<&BranchNodeCompact>,
291) -> Result<bool, DatabaseError> {
292    Ok(match (task, regular) {
293        (Some(task), Some(regular)) => {
294            task.state_mask == regular.state_mask &&
295                task.tree_mask == regular.tree_mask &&
296                task.hash_mask == regular.hash_mask &&
297                task.hashes == regular.hashes &&
298                task.root_hash == regular.root_hash
299        }
300        (None, None) => true,
301        _ => {
302            if task.is_some() {
303                task == database
304            } else {
305                regular == database
306            }
307        }
308    })
309}