1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#![allow(clippy::integer_arithmetic)]
use crate::{
account_info::AccountInfo, instruction::Instruction, program_error::ProgramError,
sanitize::SanitizeError,
};
pub struct Instructions();
crate::declare_sysvar_id!("Sysvar1nstructions1111111111111111111111111", Instructions);
pub fn load_current_index(data: &[u8]) -> u16 {
let mut instr_fixed_data = [0u8; 2];
let len = data.len();
instr_fixed_data.copy_from_slice(&data[len - 2..len]);
u16::from_le_bytes(instr_fixed_data)
}
pub fn store_current_index(data: &mut [u8], instruction_index: u16) {
let last_index = data.len() - 2;
data[last_index..last_index + 2].copy_from_slice(&instruction_index.to_le_bytes());
}
#[deprecated(
since = "1.8.0",
note = "Unsafe because the sysvar accounts address is not checked, please use `load_instruction_at_checked` instead"
)]
pub fn load_instruction_at(index: usize, data: &[u8]) -> Result<Instruction, SanitizeError> {
crate::message::Message::deserialize_instruction(index, data)
}
pub fn load_instruction_at_checked(
index: usize,
instruction_sysvar_account_info: &AccountInfo,
) -> Result<Instruction, ProgramError> {
if !check_id(instruction_sysvar_account_info.key) {
return Err(ProgramError::UnsupportedSysvar);
}
let instruction_sysvar = instruction_sysvar_account_info.try_borrow_data()?;
crate::message::Message::deserialize_instruction(index, &instruction_sysvar).map_err(|err| {
match err {
SanitizeError::IndexOutOfBounds => ProgramError::InvalidArgument,
_ => ProgramError::InvalidInstructionData,
}
})
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{instruction::AccountMeta, message::Message, pubkey::Pubkey};
#[test]
fn test_load_store_instruction() {
let mut data = [4u8; 10];
store_current_index(&mut data, 3);
assert_eq!(load_current_index(&data), 3);
assert_eq!([4u8; 8], data[0..8]);
}
#[test]
fn test_load_instruction_at_checked() {
let instruction1 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let instruction2 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let message = Message::new(
&[instruction1.clone(), instruction2.clone()],
Some(&Pubkey::new_unique()),
);
let key = id();
let mut lamports = 0;
let mut data = message.serialize_instructions(true);
data.resize(data.len() + 2, 0);
let owner = crate::sysvar::id();
let account_info = AccountInfo::new(
&key,
false,
false,
&mut lamports,
&mut data,
&owner,
false,
0,
);
assert_eq!(
instruction1,
load_instruction_at_checked(0, &account_info).unwrap()
);
assert_eq!(
instruction2,
load_instruction_at_checked(1, &account_info).unwrap()
);
assert_eq!(
Err(ProgramError::InvalidArgument),
load_instruction_at_checked(2, &account_info)
);
}
}