Skip to content

Commit

Permalink
Instruction sysvar fixes, additions (solana-labs#20958)
Browse files Browse the repository at this point in the history
  • Loading branch information
jackcmay authored and pull[bot] committed Apr 4, 2023
1 parent 30df88f commit 28044fa
Show file tree
Hide file tree
Showing 4 changed files with 224 additions and 21 deletions.
6 changes: 2 additions & 4 deletions programs/bpf/rust/instruction_introspection/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -36,11 +36,9 @@ fn process_instruction(
let instruction = instructions::load_instruction_at_checked(
secp_instruction_index as usize,
instruction_accounts,
)
.map_err(|_| ProgramError::InvalidAccountData)?;
)?;

let current_instruction =
instructions::load_current_index(&instruction_accounts.try_borrow_data()?);
let current_instruction = instructions::load_current_index_checked(instruction_accounts)?;
let my_index = instruction_data[1] as u16;
assert_eq!(current_instruction, my_index);

Expand Down
2 changes: 1 addition & 1 deletion programs/bpf/rust/sysvar/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ pub fn process_instruction(
msg!("Instructions identifier:");
sysvar::instructions::id().log();
assert_eq!(*accounts[4].owner, sysvar::id());
let index = instructions::load_current_index(&accounts[4].try_borrow_data()?);
let index = instructions::load_current_index_checked(&accounts[4])?;
let instruction = instructions::load_instruction_at_checked(index as usize, &accounts[4])?;
assert_eq!(0, index);
assert_eq!(
Expand Down
12 changes: 6 additions & 6 deletions programs/bpf/tests/programs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1388,17 +1388,17 @@ fn assert_instruction_count() {
programs.extend_from_slice(&[
("solana_bpf_rust_128bit", 584),
("solana_bpf_rust_alloc", 7388),
("solana_bpf_rust_custom_heap", 512),
("solana_bpf_rust_custom_heap", 535),
("solana_bpf_rust_dep_crate", 47),
("solana_bpf_rust_external_spend", 483),
("solana_bpf_rust_external_spend", 506),
("solana_bpf_rust_iter", 824),
("solana_bpf_rust_many_args", 941),
("solana_bpf_rust_mem", 3083),
("solana_bpf_rust_mem", 3085),
("solana_bpf_rust_membuiltins", 3976),
("solana_bpf_rust_noop", 457),
("solana_bpf_rust_noop", 480),
("solana_bpf_rust_param_passing", 146),
("solana_bpf_rust_rand", 464),
("solana_bpf_rust_sanity", 1714),
("solana_bpf_rust_rand", 487),
("solana_bpf_rust_sanity", 1716),
("solana_bpf_rust_secp256k1_recover", 25216),
("solana_bpf_rust_sha", 30704),
]);
Expand Down
225 changes: 215 additions & 10 deletions sdk/program/src/sysvar/instructions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,21 +24,43 @@ pub fn construct_instructions_data(
data
}

/// Load the current instruction's index from the Instructions Sysvar data
/// Load the current `Instruction`'s index in the currently executing
/// `Transaction` from the Instructions Sysvar data
#[deprecated(
since = "1.8.0",
note = "Unsafe because the sysvar accounts address is not checked, please use `load_current_index_checked` instead"
)]
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)
}

/// Store the current instruction's index in the Instructions Sysvar data
/// Load the current `Instruction`'s index in the currently executing
/// `Transaction`
pub fn load_current_index_checked(
instruction_sysvar_account_info: &AccountInfo,
) -> Result<u16, ProgramError> {
if !check_id(instruction_sysvar_account_info.key) {
return Err(ProgramError::UnsupportedSysvar);
}

let instruction_sysvar = instruction_sysvar_account_info.try_borrow_data()?;
let mut instr_fixed_data = [0u8; 2];
let len = instruction_sysvar.len();
instr_fixed_data.copy_from_slice(&instruction_sysvar[len - 2..len]);
Ok(u16::from_le_bytes(instr_fixed_data))
}

/// Store the current `Instruction`'s index in the Instructions Sysvar 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());
}

/// Load an instruction at the specified index
/// Load an `Instruction` in the currently executing `Transaction` 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"
Expand All @@ -47,7 +69,8 @@ pub fn load_instruction_at(index: usize, data: &[u8]) -> Result<Instruction, San
crate::message::Message::deserialize_instruction(index, data)
}

/// Load an instruction at the specified index
/// Load an `Instruction` in the currently executing `Transaction` at the
/// specified index
pub fn load_instruction_at_checked(
index: usize,
instruction_sysvar_account_info: &AccountInfo,
Expand All @@ -65,6 +88,34 @@ pub fn load_instruction_at_checked(
})
}

/// Returns the `Instruction` relative to the current `Instruction` in the
/// currently executing `Transaction`
pub fn get_instruction_relative(
index_relative_to_current: i64,
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.data.borrow();
#[allow(deprecated)]
let current_index = load_current_index(&instruction_sysvar) as i64;
let index = current_index.saturating_add(index_relative_to_current);
if index < 0 {
return Err(ProgramError::InvalidArgument);
}
#[allow(deprecated)]
load_instruction_at(
current_index.saturating_add(index_relative_to_current) as usize,
&instruction_sysvar,
)
.map_err(|err| match err {
SanitizeError::IndexOutOfBounds => ProgramError::InvalidArgument,
_ => ProgramError::InvalidInstructionData,
})
}

#[cfg(test)]
mod tests {
use super::*;
Expand All @@ -75,12 +126,122 @@ mod tests {
fn test_load_store_instruction() {
let mut data = [4u8; 10];
store_current_index(&mut data, 3);
assert_eq!(load_current_index(&data), 3);
#[allow(deprecated)]
let index = load_current_index(&data);
assert_eq!(index, 3);
assert_eq!([4u8; 8], data[0..8]);
}

#[test]
fn test_load_instruction_at_checked() {
let instruction0 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let instruction1 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let sanitized_message = crate::message::SanitizedMessage::try_from(Message::new(
&[instruction0.clone(), instruction1.clone()],
Some(&Pubkey::new_unique()),
))
.unwrap();

let key = id();
let mut lamports = 0;
let mut data = construct_instructions_data(&sanitized_message, true);
let owner = crate::sysvar::id();
let mut account_info = AccountInfo::new(
&key,
false,
false,
&mut lamports,
&mut data,
&owner,
false,
0,
);

assert_eq!(
instruction0,
load_instruction_at_checked(0, &account_info).unwrap()
);
assert_eq!(
instruction1,
load_instruction_at_checked(1, &account_info).unwrap()
);
assert_eq!(
Err(ProgramError::InvalidArgument),
load_instruction_at_checked(2, &account_info)
);

let key = Pubkey::new_unique();
account_info.key = &key;
assert_eq!(
Err(ProgramError::UnsupportedSysvar),
load_instruction_at_checked(2, &account_info)
);
}

#[test]
fn test_load_current_index_checked() {
let instruction0 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let instruction1 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let sanitized_message = crate::message::SanitizedMessage::try_from(Message::new(
&[instruction0, instruction1],
Some(&Pubkey::new_unique()),
))
.unwrap();

let key = id();
let mut lamports = 0;
let mut data = construct_instructions_data(&sanitized_message, true);
store_current_index(&mut data, 1);
let owner = crate::sysvar::id();
let mut account_info = AccountInfo::new(
&key,
false,
false,
&mut lamports,
&mut data,
&owner,
false,
0,
);

assert_eq!(1, load_current_index_checked(&account_info).unwrap());
{
let mut data = account_info.try_borrow_mut_data().unwrap();
store_current_index(&mut data, 0);
}
assert_eq!(0, load_current_index_checked(&account_info).unwrap());

let key = Pubkey::new_unique();
account_info.key = &key;
assert_eq!(
Err(ProgramError::UnsupportedSysvar),
load_current_index_checked(&account_info)
);
}

#[test]
fn test_get_instruction_relative() {
let instruction0 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let instruction1 = Instruction::new_with_bincode(
Pubkey::new_unique(),
&0,
Expand All @@ -92,16 +253,21 @@ mod tests {
vec![AccountMeta::new(Pubkey::new_unique(), false)],
);
let sanitized_message = crate::message::SanitizedMessage::try_from(Message::new(
&[instruction1.clone(), instruction2.clone()],
&[
instruction0.clone(),
instruction1.clone(),
instruction2.clone(),
],
Some(&Pubkey::new_unique()),
))
.unwrap();

let key = id();
let mut lamports = 0;
let mut data = construct_instructions_data(&sanitized_message, true);
store_current_index(&mut data, 1);
let owner = crate::sysvar::id();
let account_info = AccountInfo::new(
let mut account_info = AccountInfo::new(
&key,
false,
false,
Expand All @@ -112,17 +278,56 @@ mod tests {
0,
);

assert_eq!(
Err(ProgramError::InvalidArgument),
get_instruction_relative(-2, &account_info)
);
assert_eq!(
instruction0,
get_instruction_relative(-1, &account_info).unwrap()
);
assert_eq!(
instruction1,
load_instruction_at_checked(0, &account_info).unwrap()
get_instruction_relative(0, &account_info).unwrap()
);
assert_eq!(
instruction2,
load_instruction_at_checked(1, &account_info).unwrap()
get_instruction_relative(1, &account_info).unwrap()
);
assert_eq!(
Err(ProgramError::InvalidArgument),
load_instruction_at_checked(2, &account_info)
get_instruction_relative(2, &account_info)
);
{
let mut data = account_info.try_borrow_mut_data().unwrap();
store_current_index(&mut data, 0);
}
assert_eq!(
Err(ProgramError::InvalidArgument),
get_instruction_relative(-1, &account_info)
);
assert_eq!(
instruction0,
get_instruction_relative(0, &account_info).unwrap()
);
assert_eq!(
instruction1,
get_instruction_relative(1, &account_info).unwrap()
);
assert_eq!(
instruction2,
get_instruction_relative(2, &account_info).unwrap()
);
assert_eq!(
Err(ProgramError::InvalidArgument),
get_instruction_relative(3, &account_info)
);

let key = Pubkey::new_unique();
account_info.key = &key;
assert_eq!(
Err(ProgramError::UnsupportedSysvar),
get_instruction_relative(0, &account_info)
);
}
}

0 comments on commit 28044fa

Please sign in to comment.