solana_message/versions/v0/
loaded.rs1#[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#[derive(Debug, Clone, Eq, PartialEq)]
13pub struct LoadedMessage<'a> {
14 pub message: Cow<'a, v0::Message>,
16 pub loaded_addresses: Cow<'a, LoadedAddresses>,
18 pub is_writable_account_cache: Vec<bool>,
21}
22
23#[derive(Clone, Default, Debug, PartialEq, Eq)]
26#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
27pub struct LoadedAddresses {
28 pub writable: Vec<Address>,
30 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 pub fn is_empty(&self) -> bool {
50 self.len() == 0
51 }
52
53 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 pub fn account_keys(&self) -> AccountKeys<'_> {
104 AccountKeys::new(&self.message.account_keys, Some(&self.loaded_addresses))
105 }
106
107 pub fn static_account_keys(&self) -> &[Address] {
109 &self.message.account_keys
110 }
111
112 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 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 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 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 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}