reth_trie_common/
input.rs

1use crate::{
2    prefix_set::TriePrefixSetsMut,
3    updates::{TrieUpdates, TrieUpdatesSorted},
4    HashedPostState, HashedPostStateSorted,
5};
6use alloc::sync::Arc;
7
8/// Inputs for trie-related computations.
9#[derive(Default, Debug, Clone)]
10pub struct TrieInput {
11    /// The collection of cached in-memory intermediate trie nodes that
12    /// can be reused for computation.
13    pub nodes: TrieUpdates,
14    /// The in-memory overlay hashed state.
15    pub state: HashedPostState,
16    /// The collection of prefix sets for the computation. Since the prefix sets _always_
17    /// invalidate the in-memory nodes, not all keys from `self.state` might be present here,
18    /// if we have cached nodes for them.
19    pub prefix_sets: TriePrefixSetsMut,
20}
21
22impl TrieInput {
23    /// Create new trie input.
24    pub const fn new(
25        nodes: TrieUpdates,
26        state: HashedPostState,
27        prefix_sets: TriePrefixSetsMut,
28    ) -> Self {
29        Self { nodes, state, prefix_sets }
30    }
31
32    /// Create new trie input from in-memory state. The prefix sets will be constructed and
33    /// set automatically.
34    pub fn from_state(state: HashedPostState) -> Self {
35        let prefix_sets = state.construct_prefix_sets();
36        Self { nodes: TrieUpdates::default(), state, prefix_sets }
37    }
38
39    /// Create new trie input from the provided blocks, from oldest to newest. See the documentation
40    /// for [`Self::extend_with_blocks`] for details.
41    pub fn from_blocks<'a>(
42        blocks: impl IntoIterator<Item = (&'a HashedPostState, &'a TrieUpdates)>,
43    ) -> Self {
44        let mut input = Self::default();
45        input.extend_with_blocks(blocks);
46        input
47    }
48
49    /// Create new trie input from the provided sorted blocks, from oldest to newest.
50    /// Converts sorted types to unsorted for aggregation.
51    pub fn from_blocks_sorted<'a>(
52        blocks: impl IntoIterator<Item = (&'a HashedPostStateSorted, &'a TrieUpdatesSorted)>,
53    ) -> Self {
54        let mut input = Self::default();
55        for (hashed_state, trie_updates) in blocks {
56            // Extend directly from sorted types, avoiding intermediate HashMap allocations
57            input.nodes.extend_from_sorted(trie_updates);
58            input.state.extend_from_sorted(hashed_state);
59        }
60        input
61    }
62
63    /// Extend the trie input with the provided blocks, from oldest to newest.
64    pub fn extend_with_blocks<'a>(
65        &mut self,
66        blocks: impl IntoIterator<Item = (&'a HashedPostState, &'a TrieUpdates)>,
67    ) {
68        for (hashed_state, trie_updates) in blocks {
69            self.append_cached_ref(trie_updates, hashed_state);
70        }
71    }
72
73    /// Prepend another trie input to the current one.
74    pub fn prepend_self(&mut self, mut other: Self) {
75        core::mem::swap(&mut self.nodes, &mut other.nodes);
76        self.nodes.extend(other.nodes);
77        core::mem::swap(&mut self.state, &mut other.state);
78        self.state.extend(other.state);
79        // No need to swap prefix sets, as they will be sorted and deduplicated.
80        self.prefix_sets.extend(other.prefix_sets);
81    }
82
83    /// Prepend state to the input and extend the prefix sets.
84    pub fn prepend(&mut self, mut state: HashedPostState) {
85        self.prefix_sets.extend(state.construct_prefix_sets());
86        core::mem::swap(&mut self.state, &mut state);
87        self.state.extend(state);
88    }
89
90    /// Prepend intermediate nodes and state to the input.
91    /// Prefix sets for incoming state will be ignored.
92    pub fn prepend_cached(&mut self, mut nodes: TrieUpdates, mut state: HashedPostState) {
93        core::mem::swap(&mut self.nodes, &mut nodes);
94        self.nodes.extend(nodes);
95        core::mem::swap(&mut self.state, &mut state);
96        self.state.extend(state);
97    }
98
99    /// Append state to the input and extend the prefix sets.
100    pub fn append(&mut self, state: HashedPostState) {
101        self.prefix_sets.extend(state.construct_prefix_sets());
102        self.state.extend(state);
103    }
104
105    /// Append state to the input by reference and extend the prefix sets.
106    pub fn append_ref(&mut self, state: &HashedPostState) {
107        self.prefix_sets.extend(state.construct_prefix_sets());
108        self.state.extend_ref(state);
109    }
110
111    /// Append intermediate nodes and state to the input.
112    /// Prefix sets for incoming state will be ignored.
113    pub fn append_cached(&mut self, nodes: TrieUpdates, state: HashedPostState) {
114        self.nodes.extend(nodes);
115        self.state.extend(state);
116    }
117
118    /// Append intermediate nodes and state to the input by reference.
119    /// Prefix sets for incoming state will be ignored.
120    pub fn append_cached_ref(&mut self, nodes: &TrieUpdates, state: &HashedPostState) {
121        self.nodes.extend_ref(nodes);
122        self.state.extend_ref(state);
123    }
124
125    /// This method clears the trie input nodes, state, and prefix sets.
126    pub fn clear(&mut self) {
127        self.nodes.clear();
128        self.state.clear();
129        self.prefix_sets.clear();
130    }
131
132    /// This method returns a cleared version of this trie input.
133    pub fn cleared(mut self) -> Self {
134        self.clear();
135        self
136    }
137}
138
139/// Sorted variant of [`TrieInput`] for efficient proof generation.
140///
141/// This type holds sorted versions of trie data structures, which eliminates the need
142/// for expensive sorting operations during multiproof generation.
143#[derive(Default, Debug, Clone)]
144pub struct TrieInputSorted {
145    /// Sorted cached in-memory intermediate trie nodes.
146    pub nodes: Arc<TrieUpdatesSorted>,
147    /// Sorted in-memory overlay hashed state.
148    pub state: Arc<HashedPostStateSorted>,
149    /// Prefix sets for computation.
150    pub prefix_sets: TriePrefixSetsMut,
151}
152
153impl TrieInputSorted {
154    /// Create new sorted trie input.
155    pub const fn new(
156        nodes: Arc<TrieUpdatesSorted>,
157        state: Arc<HashedPostStateSorted>,
158        prefix_sets: TriePrefixSetsMut,
159    ) -> Self {
160        Self { nodes, state, prefix_sets }
161    }
162
163    /// Create from unsorted [`TrieInput`] by sorting.
164    pub fn from_unsorted(input: TrieInput) -> Self {
165        Self {
166            nodes: Arc::new(input.nodes.into_sorted()),
167            state: Arc::new(input.state.into_sorted()),
168            prefix_sets: input.prefix_sets,
169        }
170    }
171}
172
173#[cfg(test)]
174mod tests {}