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
102pub(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 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 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 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 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 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 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 let database_not_exists = trie_cursor()?.seek(Nibbles::default())?.is_none();
194 let mut diff = StorageTrieUpdatesDiff {
195 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 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 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 !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
251fn 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
281fn 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}