solana_program/message/versions/v0/
loaded.rs1use {
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#[derive(Debug, Clone)]
12pub struct LoadedMessage<'a> {
13 pub message: Cow<'a, v0::Message>,
15 pub loaded_addresses: Cow<'a, LoadedAddresses>,
17 pub is_writable_account_cache: Vec<bool>,
20}
21
22#[derive(Clone, Default, Debug, PartialEq, Eq, Serialize, Deserialize)]
25pub struct LoadedAddresses {
26 pub writable: Vec<Pubkey>,
28 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 pub fn is_empty(&self) -> bool {
48 self.len() == 0
49 }
50
51 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 pub fn account_keys(&self) -> AccountKeys {
93 AccountKeys::new(&self.message.account_keys, Some(&self.loaded_addresses))
94 }
95
96 pub fn static_account_keys(&self) -> &[Pubkey] {
98 &self.message.account_keys
99 }
100
101 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 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 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 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 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}