* add checked instructions sysvar api (#20790)
(cherry picked from commit a8098f37d0
)
# Conflicts:
# programs/bpf/rust/sysvar/src/lib.rs
# runtime/src/accounts.rs
* resolve conflicts
Co-authored-by: Jack May <jack@solana.com>
This commit is contained in:
@ -26,9 +26,9 @@ fn process_instruction(
|
||||
return Err(ProgramError::InvalidAccountData);
|
||||
}
|
||||
|
||||
let instruction = instructions::load_instruction_at(
|
||||
let instruction = instructions::load_instruction_at_checked(
|
||||
secp_instruction_index as usize,
|
||||
&instruction_accounts.try_borrow_data()?,
|
||||
instruction_accounts,
|
||||
)
|
||||
.map_err(|_| ProgramError::InvalidAccountData)?;
|
||||
|
||||
|
@ -6,6 +6,7 @@ use solana_program::{
|
||||
entrypoint,
|
||||
entrypoint::ProgramResult,
|
||||
fee_calculator::FeeCalculator,
|
||||
instruction::{AccountMeta, Instruction},
|
||||
msg,
|
||||
program_error::ProgramError,
|
||||
pubkey::Pubkey,
|
||||
@ -19,7 +20,7 @@ use solana_program::{
|
||||
entrypoint!(process_instruction);
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
pub fn process_instruction(
|
||||
_program_id: &Pubkey,
|
||||
program_id: &Pubkey,
|
||||
accounts: &[AccountInfo],
|
||||
_instruction_data: &[u8],
|
||||
) -> ProgramResult {
|
||||
@ -57,8 +58,30 @@ pub fn process_instruction(
|
||||
// Instructions
|
||||
msg!("Instructions identifier:");
|
||||
sysvar::instructions::id().log();
|
||||
assert_eq!(*accounts[4].owner, sysvar::id());
|
||||
let index = instructions::load_current_index(&accounts[5].try_borrow_data()?);
|
||||
let instruction = instructions::load_instruction_at_checked(index as usize, &accounts[5])?;
|
||||
assert_eq!(0, index);
|
||||
assert_eq!(
|
||||
instruction,
|
||||
Instruction::new_with_bytes(
|
||||
*program_id,
|
||||
&[] as &[u8],
|
||||
vec![
|
||||
AccountMeta::new(*accounts[0].key, true),
|
||||
AccountMeta::new(*accounts[1].key, false),
|
||||
AccountMeta::new_readonly(*accounts[2].key, false),
|
||||
AccountMeta::new_readonly(*accounts[3].key, false),
|
||||
AccountMeta::new_readonly(*accounts[4].key, false),
|
||||
AccountMeta::new_readonly(*accounts[5].key, false),
|
||||
AccountMeta::new_readonly(*accounts[6].key, false),
|
||||
AccountMeta::new_readonly(*accounts[7].key, false),
|
||||
AccountMeta::new_readonly(*accounts[8].key, false),
|
||||
AccountMeta::new_readonly(*accounts[9].key, false),
|
||||
AccountMeta::new_readonly(*accounts[10].key, false),
|
||||
],
|
||||
)
|
||||
);
|
||||
|
||||
// Recent Blockhashes
|
||||
{
|
||||
|
@ -49,6 +49,7 @@ fn bench_manual_instruction_deserialize(b: &mut Bencher) {
|
||||
let serialized = message.serialize_instructions(DEMOTE_PROGRAM_WRITE_LOCKS);
|
||||
b.iter(|| {
|
||||
for i in 0..instructions.len() {
|
||||
#[allow(deprecated)]
|
||||
test::black_box(instructions::load_instruction_at(i, &serialized).unwrap());
|
||||
}
|
||||
});
|
||||
@ -60,6 +61,7 @@ fn bench_manual_instruction_deserialize_single(b: &mut Bencher) {
|
||||
let message = Message::new(&instructions, None);
|
||||
let serialized = message.serialize_instructions(DEMOTE_PROGRAM_WRITE_LOCKS);
|
||||
b.iter(|| {
|
||||
#[allow(deprecated)]
|
||||
test::black_box(instructions::load_instruction_at(3, &serialized).unwrap());
|
||||
});
|
||||
}
|
||||
|
@ -1,7 +1,10 @@
|
||||
#![allow(clippy::integer_arithmetic)]
|
||||
//! This account contains the serialized transaction instructions
|
||||
|
||||
use crate::{instruction::Instruction, sanitize::SanitizeError};
|
||||
use crate::{
|
||||
account_info::AccountInfo, instruction::Instruction, program_error::ProgramError,
|
||||
sanitize::SanitizeError,
|
||||
};
|
||||
|
||||
// Instructions Sysvar, dummy type, use the associated helpers instead of the Sysvar trait
|
||||
pub struct Instructions();
|
||||
@ -23,13 +26,36 @@ pub fn store_current_index(data: &mut [u8], instruction_index: u16) {
|
||||
}
|
||||
|
||||
/// Load an instruction at the specified index
|
||||
#[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)
|
||||
}
|
||||
|
||||
/// Load an instruction at the specified index
|
||||
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() {
|
||||
@ -38,4 +64,51 @@ mod tests {
|
||||
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)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user