Skip to main content

solana_message/versions/v0/
loaded.rs

1#[cfg(feature = "serde")]
2use serde_derive::{Deserialize, Serialize};
3use {
4    crate::v0,
5    alloc::{borrow::Cow, vec::Vec},
6    solana_address::Address,
7};
8#[cfg(feature = "std")]
9use {crate::AccountKeys, solana_sdk_ids::bpf_loader_upgradeable, std::collections::HashSet};
10
11/// Combination of a version #0 message and its loaded addresses
12#[derive(Debug, Clone, Eq, PartialEq)]
13pub struct LoadedMessage<'a> {
14    /// Message which loaded a collection of lookup table addresses
15    pub message: Cow<'a, v0::Message>,
16    /// Addresses loaded with on-chain address lookup tables
17    pub loaded_addresses: Cow<'a, LoadedAddresses>,
18    /// List of boolean with same length as account_keys(), each boolean value indicates if
19    /// corresponding account key is writable or not.
20    pub is_writable_account_cache: Vec<bool>,
21}
22
23/// Collection of addresses loaded from on-chain lookup tables, split
24/// by readonly and writable.
25#[derive(Clone, Default, Debug, PartialEq, Eq)]
26#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
27pub struct LoadedAddresses {
28    /// List of addresses for writable loaded accounts
29    pub writable: Vec<Address>,
30    /// List of addresses for read-only loaded accounts
31    pub readonly: Vec<Address>,
32}
33
34impl FromIterator<LoadedAddresses> for LoadedAddresses {
35    fn from_iter<T: IntoIterator<Item = LoadedAddresses>>(iter: T) -> Self {
36        let (writable, readonly): (Vec<Vec<Address>>, Vec<Vec<Address>>) = iter
37            .into_iter()
38            .map(|addresses| (addresses.writable, addresses.readonly))
39            .unzip();
40        LoadedAddresses {
41            writable: writable.into_iter().flatten().collect(),
42            readonly: readonly.into_iter().flatten().collect(),
43        }
44    }
45}
46
47impl LoadedAddresses {
48    /// Checks if there are no writable or readonly addresses
49    pub fn is_empty(&self) -> bool {
50        self.len() == 0
51    }
52
53    /// Combined length of loaded writable and readonly addresses
54    pub fn len(&self) -> usize {
55        self.writable.len().saturating_add(self.readonly.len())
56    }
57}
58
59#[cfg(feature = "std")]
60impl<'a> LoadedMessage<'a> {
61    pub fn new(
62        message: v0::Message,
63        loaded_addresses: LoadedAddresses,
64        reserved_account_keys: &HashSet<Address>,
65    ) -> Self {
66        let mut loaded_message = Self {
67            message: Cow::Owned(message),
68            loaded_addresses: Cow::Owned(loaded_addresses),
69            is_writable_account_cache: Vec::default(),
70        };
71        loaded_message.set_is_writable_account_cache(reserved_account_keys);
72        loaded_message
73    }
74
75    pub fn new_borrowed(
76        message: &'a v0::Message,
77        loaded_addresses: &'a LoadedAddresses,
78        reserved_account_keys: &HashSet<Address>,
79    ) -> Self {
80        let mut loaded_message = Self {
81            message: Cow::Borrowed(message),
82            loaded_addresses: Cow::Borrowed(loaded_addresses),
83            is_writable_account_cache: Vec::default(),
84        };
85        loaded_message.set_is_writable_account_cache(reserved_account_keys);
86        loaded_message
87    }
88
89    fn set_is_writable_account_cache(&mut self, reserved_account_keys: &HashSet<Address>) {
90        let is_writable_account_cache = self
91            .account_keys()
92            .iter()
93            .enumerate()
94            .map(|(i, _key)| self.is_writable_internal(i, reserved_account_keys))
95            .collect::<Vec<_>>();
96        let _ = core::mem::replace(
97            &mut self.is_writable_account_cache,
98            is_writable_account_cache,
99        );
100    }
101
102    /// Returns the full list of static and dynamic account keys that are loaded for this message.
103    pub fn account_keys(&self) -> AccountKeys<'_> {
104        AccountKeys::new(&self.message.account_keys, Some(&self.loaded_addresses))
105    }
106
107    /// Returns the list of static account keys that are loaded for this message.
108    pub fn static_account_keys(&self) -> &[Address] {
109        &self.message.account_keys
110    }
111
112    /// Returns true if any account keys are duplicates
113    pub fn has_duplicates(&self) -> bool {
114        let mut uniq = HashSet::with_capacity(self.account_keys().len());
115        self.account_keys().iter().any(|x| !uniq.insert(x))
116    }
117
118    /// Returns true if the account at the specified index was requested to be
119    /// writable.  This method should not be used directly.
120    fn is_writable_index(&self, key_index: usize) -> bool {
121        let header = &self.message.header;
122        let num_account_keys = self.message.account_keys.len();
123        let num_signed_accounts = usize::from(header.num_required_signatures);
124        if key_index >= num_account_keys {
125            let loaded_addresses_index = key_index.saturating_sub(num_account_keys);
126            loaded_addresses_index < self.loaded_addresses.writable.len()
127        } else if key_index >= num_signed_accounts {
128            let num_unsigned_accounts = num_account_keys.saturating_sub(num_signed_accounts);
129            let num_writable_unsigned_accounts = num_unsigned_accounts
130                .saturating_sub(usize::from(header.num_readonly_unsigned_accounts));
131            let unsigned_account_index = key_index.saturating_sub(num_signed_accounts);
132            unsigned_account_index < num_writable_unsigned_accounts
133        } else {
134            let num_writable_signed_accounts = num_signed_accounts
135                .saturating_sub(usize::from(header.num_readonly_signed_accounts));
136            key_index < num_writable_signed_accounts
137        }
138    }
139
140    /// Returns true if the account at the specified index was loaded as writable
141    fn is_writable_internal(
142        &self,
143        key_index: usize,
144        reserved_account_keys: &HashSet<Address>,
145    ) -> bool {
146        if self.is_writable_index(key_index) {
147            if let Some(key) = self.account_keys().get(key_index) {
148                return !(reserved_account_keys.contains(key) || self.demote_program_id(key_index));
149            }
150        }
151        false
152    }
153
154    pub fn is_writable(&self, key_index: usize) -> bool {
155        *self
156            .is_writable_account_cache
157            .get(key_index)
158            .unwrap_or(&false)
159    }
160
161    pub fn is_signer(&self, i: usize) -> bool {
162        i < self.message.header.num_required_signatures as usize
163    }
164
165    pub fn demote_program_id(&self, i: usize) -> bool {
166        self.is_key_called_as_program(i) && !self.is_upgradeable_loader_present()
167    }
168
169    /// Returns true if the account at the specified index is called as a program by an instruction
170    pub fn is_key_called_as_program(&self, key_index: usize) -> bool {
171        if let Ok(key_index) = u8::try_from(key_index) {
172            self.message
173                .instructions
174                .iter()
175                .any(|ix| ix.program_id_index == key_index)
176        } else {
177            false
178        }
179    }
180
181    /// Returns true if any account is the bpf upgradeable loader
182    pub fn is_upgradeable_loader_present(&self) -> bool {
183        self.account_keys()
184            .iter()
185            .any(|&key| key == bpf_loader_upgradeable::id())
186    }
187}
188
189#[cfg(test)]
190mod tests {
191    use {
192        super::*,
193        crate::{compiled_instruction::CompiledInstruction, MessageHeader},
194        alloc::vec,
195        itertools::Itertools,
196        solana_sdk_ids::{system_program, sysvar},
197    };
198
199    fn check_test_loaded_message() -> (LoadedMessage<'static>, [Address; 6]) {
200        let key0 = Address::new_unique();
201        let key1 = Address::new_unique();
202        let key2 = Address::new_unique();
203        let key3 = Address::new_unique();
204        let key4 = Address::new_unique();
205        let key5 = Address::new_unique();
206
207        let message = LoadedMessage::new(
208            v0::Message {
209                header: MessageHeader {
210                    num_required_signatures: 2,
211                    num_readonly_signed_accounts: 1,
212                    num_readonly_unsigned_accounts: 1,
213                },
214                account_keys: vec![key0, key1, key2, key3],
215                ..v0::Message::default()
216            },
217            LoadedAddresses {
218                writable: vec![key4],
219                readonly: vec![key5],
220            },
221            &HashSet::default(),
222        );
223
224        (message, [key0, key1, key2, key3, key4, key5])
225    }
226
227    #[test]
228    fn test_has_duplicates() {
229        let message = check_test_loaded_message().0;
230
231        assert!(!message.has_duplicates());
232    }
233
234    #[test]
235    fn test_has_duplicates_with_dupe_keys() {
236        let create_message_with_dupe_keys = |mut keys: Vec<Address>| {
237            LoadedMessage::new(
238                v0::Message {
239                    account_keys: keys.split_off(2),
240                    ..v0::Message::default()
241                },
242                LoadedAddresses {
243                    writable: keys.split_off(2),
244                    readonly: keys,
245                },
246                &HashSet::default(),
247            )
248        };
249
250        let key0 = Address::new_unique();
251        let key1 = Address::new_unique();
252        let key2 = Address::new_unique();
253        let key3 = Address::new_unique();
254        let dupe_key = Address::new_unique();
255
256        let keys = vec![key0, key1, key2, key3, dupe_key, dupe_key];
257        let keys_len = keys.len();
258        for keys in keys.into_iter().permutations(keys_len).unique() {
259            let message = create_message_with_dupe_keys(keys);
260            assert!(message.has_duplicates());
261        }
262    }
263
264    #[test]
265    fn test_is_writable_index() {
266        let message = check_test_loaded_message().0;
267
268        assert!(message.is_writable_index(0));
269        assert!(!message.is_writable_index(1));
270        assert!(message.is_writable_index(2));
271        assert!(!message.is_writable_index(3));
272        assert!(message.is_writable_index(4));
273        assert!(!message.is_writable_index(5));
274    }
275
276    #[test]
277    fn test_is_writable() {
278        let reserved_account_keys = HashSet::from_iter([sysvar::clock::id(), system_program::id()]);
279        let create_message_with_keys = |keys: Vec<Address>| {
280            LoadedMessage::new(
281                v0::Message {
282                    header: MessageHeader {
283                        num_required_signatures: 1,
284                        num_readonly_signed_accounts: 0,
285                        num_readonly_unsigned_accounts: 1,
286                    },
287                    account_keys: keys[..2].to_vec(),
288                    ..v0::Message::default()
289                },
290                LoadedAddresses {
291                    writable: keys[2..=2].to_vec(),
292                    readonly: keys[3..].to_vec(),
293                },
294                &reserved_account_keys,
295            )
296        };
297
298        let key0 = Address::new_unique();
299        let key1 = Address::new_unique();
300        let key2 = Address::new_unique();
301        {
302            let message = create_message_with_keys(vec![sysvar::clock::id(), key0, key1, key2]);
303            assert!(message.is_writable_index(0));
304            assert!(!message.is_writable(0));
305        }
306
307        {
308            let message = create_message_with_keys(vec![system_program::id(), key0, key1, key2]);
309            assert!(message.is_writable_index(0));
310            assert!(!message.is_writable(0));
311        }
312
313        {
314            let message = create_message_with_keys(vec![key0, key1, system_program::id(), key2]);
315            assert!(message.is_writable_index(2));
316            assert!(!message.is_writable(2));
317        }
318    }
319
320    #[test]
321    fn test_demote_writable_program() {
322        let key0 = Address::new_unique();
323        let key1 = Address::new_unique();
324        let key2 = Address::new_unique();
325        let message = LoadedMessage::new(
326            v0::Message {
327                header: MessageHeader {
328                    num_required_signatures: 1,
329                    num_readonly_signed_accounts: 0,
330                    num_readonly_unsigned_accounts: 0,
331                },
332                account_keys: vec![key0],
333                instructions: vec![CompiledInstruction {
334                    program_id_index: 2,
335                    accounts: vec![1],
336                    data: vec![],
337                }],
338                ..v0::Message::default()
339            },
340            LoadedAddresses {
341                writable: vec![key1, key2],
342                readonly: vec![],
343            },
344            &HashSet::default(),
345        );
346
347        assert!(message.is_writable_index(2));
348        assert!(!message.is_writable(2));
349    }
350}