solana_program/message/versions/v0/
loaded.rs

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