parent
6f2e556b16
commit
12d2147efa
|
@ -20,7 +20,9 @@ use {
|
|||
pubkey::Pubkey,
|
||||
rent::Rent,
|
||||
saturating_add_assign,
|
||||
transaction_context::{InstructionAccount, TransactionAccount, TransactionContext},
|
||||
transaction_context::{
|
||||
IndexOfAccount, InstructionAccount, TransactionAccount, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::{
|
||||
alloc::Layout,
|
||||
|
@ -34,7 +36,7 @@ use {
|
|||
};
|
||||
|
||||
pub type ProcessInstructionWithContext =
|
||||
fn(usize, &mut InvokeContext) -> Result<(), InstructionError>;
|
||||
fn(IndexOfAccount, &mut InvokeContext) -> Result<(), InstructionError>;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct BuiltinProgram {
|
||||
|
@ -46,7 +48,7 @@ impl std::fmt::Debug for BuiltinProgram {
|
|||
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
||||
// These are just type aliases for work around of Debug-ing above pointers
|
||||
type ErasedProcessInstructionWithContext =
|
||||
fn(usize, &'static mut InvokeContext<'static>) -> Result<(), InstructionError>;
|
||||
fn(IndexOfAccount, &'static mut InvokeContext<'static>) -> Result<(), InstructionError>;
|
||||
|
||||
// rustc doesn't compile due to bug without this work around
|
||||
// https://github.com/rust-lang/rust/issues/50280
|
||||
|
@ -61,7 +63,7 @@ pub trait Executor: Debug + Send + Sync {
|
|||
/// Execute the program
|
||||
fn execute(
|
||||
&self,
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError>;
|
||||
}
|
||||
|
@ -292,8 +294,9 @@ impl<'a> InvokeContext<'a> {
|
|||
.feature_set
|
||||
.is_active(&enable_early_verification_of_account_modifications::id())
|
||||
{
|
||||
self.pre_accounts =
|
||||
Vec::with_capacity(instruction_context.get_number_of_instruction_accounts());
|
||||
self.pre_accounts = Vec::with_capacity(
|
||||
instruction_context.get_number_of_instruction_accounts() as usize,
|
||||
);
|
||||
for instruction_account_index in
|
||||
0..instruction_context.get_number_of_instruction_accounts()
|
||||
{
|
||||
|
@ -374,7 +377,7 @@ impl<'a> InvokeContext<'a> {
|
|||
fn verify(
|
||||
&mut self,
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
) -> Result<(), InstructionError> {
|
||||
let instruction_context = self
|
||||
.transaction_context
|
||||
|
@ -398,7 +401,7 @@ impl<'a> InvokeContext<'a> {
|
|||
for (instruction_account_index, instruction_account) in
|
||||
instruction_accounts.iter().enumerate()
|
||||
{
|
||||
if instruction_account_index != instruction_account.index_in_callee {
|
||||
if instruction_account_index as IndexOfAccount != instruction_account.index_in_callee {
|
||||
continue; // Skip duplicate account
|
||||
}
|
||||
{
|
||||
|
@ -477,7 +480,7 @@ impl<'a> InvokeContext<'a> {
|
|||
for (instruction_account_index, instruction_account) in
|
||||
instruction_accounts.iter().enumerate()
|
||||
{
|
||||
if instruction_account_index != instruction_account.index_in_callee {
|
||||
if instruction_account_index as IndexOfAccount != instruction_account.index_in_callee {
|
||||
continue; // Skip duplicate account
|
||||
}
|
||||
if instruction_account.index_in_transaction
|
||||
|
@ -575,7 +578,7 @@ impl<'a> InvokeContext<'a> {
|
|||
&mut self,
|
||||
instruction: &Instruction,
|
||||
signers: &[Pubkey],
|
||||
) -> Result<(Vec<InstructionAccount>, Vec<usize>), InstructionError> {
|
||||
) -> Result<(Vec<InstructionAccount>, Vec<IndexOfAccount>), InstructionError> {
|
||||
// Finds the index of each account in the instruction by its pubkey.
|
||||
// Then normalizes / unifies the privileges of duplicate accounts.
|
||||
// Note: This is an O(n^2) algorithm,
|
||||
|
@ -626,7 +629,7 @@ impl<'a> InvokeContext<'a> {
|
|||
deduplicated_instruction_accounts.push(InstructionAccount {
|
||||
index_in_transaction,
|
||||
index_in_caller,
|
||||
index_in_callee: instruction_account_index,
|
||||
index_in_callee: instruction_account_index as IndexOfAccount,
|
||||
is_signer: account_meta.is_signer,
|
||||
is_writable: account_meta.is_writable,
|
||||
});
|
||||
|
@ -723,7 +726,7 @@ impl<'a> InvokeContext<'a> {
|
|||
&mut self,
|
||||
instruction_data: &[u8],
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
compute_units_consumed: &mut u64,
|
||||
timings: &mut ExecuteTimings,
|
||||
) -> Result<(), InstructionError> {
|
||||
|
@ -960,7 +963,7 @@ pub struct MockInvokeContextPreparation {
|
|||
pub fn prepare_mock_invoke_context(
|
||||
transaction_accounts: Vec<TransactionAccount>,
|
||||
instruction_account_metas: Vec<AccountMeta>,
|
||||
_program_indices: &[usize],
|
||||
_program_indices: &[IndexOfAccount],
|
||||
) -> MockInvokeContextPreparation {
|
||||
let mut instruction_accounts: Vec<InstructionAccount> =
|
||||
Vec::with_capacity(instruction_account_metas.len());
|
||||
|
@ -968,7 +971,8 @@ pub fn prepare_mock_invoke_context(
|
|||
let index_in_transaction = transaction_accounts
|
||||
.iter()
|
||||
.position(|(key, _account)| *key == account_meta.pubkey)
|
||||
.unwrap_or(transaction_accounts.len());
|
||||
.unwrap_or(transaction_accounts.len())
|
||||
as IndexOfAccount;
|
||||
let index_in_callee = instruction_accounts
|
||||
.get(0..instruction_account_index)
|
||||
.unwrap()
|
||||
|
@ -976,7 +980,7 @@ pub fn prepare_mock_invoke_context(
|
|||
.position(|instruction_account| {
|
||||
instruction_account.index_in_transaction == index_in_transaction
|
||||
})
|
||||
.unwrap_or(instruction_account_index);
|
||||
.unwrap_or(instruction_account_index) as IndexOfAccount;
|
||||
instruction_accounts.push(InstructionAccount {
|
||||
index_in_transaction,
|
||||
index_in_caller: index_in_transaction,
|
||||
|
@ -1037,7 +1041,7 @@ pub fn with_mock_invoke_context<R, F: FnMut(&mut InvokeContext) -> R>(
|
|||
|
||||
pub fn mock_process_instruction(
|
||||
loader_id: &Pubkey,
|
||||
mut program_indices: Vec<usize>,
|
||||
mut program_indices: Vec<IndexOfAccount>,
|
||||
instruction_data: &[u8],
|
||||
transaction_accounts: Vec<TransactionAccount>,
|
||||
instruction_accounts: Vec<AccountMeta>,
|
||||
|
@ -1046,7 +1050,7 @@ pub fn mock_process_instruction(
|
|||
expected_result: Result<(), InstructionError>,
|
||||
process_instruction: ProcessInstructionWithContext,
|
||||
) -> Vec<AccountSharedData> {
|
||||
program_indices.insert(0, transaction_accounts.len());
|
||||
program_indices.insert(0, transaction_accounts.len() as IndexOfAccount);
|
||||
let mut preparation =
|
||||
prepare_mock_invoke_context(transaction_accounts, instruction_accounts, &program_indices);
|
||||
let processor_account = AccountSharedData::new(0, 0, &native_loader::id());
|
||||
|
@ -1117,14 +1121,14 @@ mod tests {
|
|||
fn test_program_entry_debug() {
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
Ok(())
|
||||
}
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -1144,7 +1148,7 @@ mod tests {
|
|||
|
||||
#[allow(clippy::integer_arithmetic)]
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -1256,9 +1260,9 @@ mod tests {
|
|||
AccountSharedData::new(index as u64, 1, invoke_stack.get(index).unwrap()),
|
||||
));
|
||||
instruction_accounts.push(InstructionAccount {
|
||||
index_in_transaction: index,
|
||||
index_in_caller: index,
|
||||
index_in_callee: instruction_accounts.len(),
|
||||
index_in_transaction: index as IndexOfAccount,
|
||||
index_in_caller: index as IndexOfAccount,
|
||||
index_in_callee: instruction_accounts.len() as IndexOfAccount,
|
||||
is_signer: false,
|
||||
is_writable: true,
|
||||
});
|
||||
|
@ -1269,9 +1273,9 @@ mod tests {
|
|||
AccountSharedData::new(1, 1, &solana_sdk::pubkey::Pubkey::default()),
|
||||
));
|
||||
instruction_accounts.push(InstructionAccount {
|
||||
index_in_transaction: index,
|
||||
index_in_caller: index,
|
||||
index_in_callee: index,
|
||||
index_in_transaction: index as IndexOfAccount,
|
||||
index_in_caller: index as IndexOfAccount,
|
||||
index_in_callee: index as IndexOfAccount,
|
||||
is_signer: false,
|
||||
is_writable: false,
|
||||
});
|
||||
|
@ -1291,7 +1295,11 @@ mod tests {
|
|||
.transaction_context
|
||||
.get_next_instruction_context()
|
||||
.unwrap()
|
||||
.configure(&[MAX_DEPTH + depth_reached], &instruction_accounts, &[]);
|
||||
.configure(
|
||||
&[(MAX_DEPTH + depth_reached) as IndexOfAccount],
|
||||
&instruction_accounts,
|
||||
&[],
|
||||
);
|
||||
if Err(InstructionError::CallDepth) == invoke_context.push() {
|
||||
break;
|
||||
}
|
||||
|
|
|
@ -9,7 +9,7 @@ use {
|
|||
clock::Clock, epoch_schedule::EpochSchedule, rent::Rent, slot_hashes::SlotHashes,
|
||||
stake_history::StakeHistory, Sysvar, SysvarId,
|
||||
},
|
||||
transaction_context::{InstructionContext, TransactionContext},
|
||||
transaction_context::{IndexOfAccount, InstructionContext, TransactionContext},
|
||||
},
|
||||
std::sync::Arc,
|
||||
};
|
||||
|
@ -183,7 +183,7 @@ pub mod get_sysvar_with_account_check {
|
|||
fn check_sysvar_account<S: Sysvar>(
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<(), InstructionError> {
|
||||
let index_in_transaction = instruction_context
|
||||
.get_index_of_instruction_account_in_transaction(instruction_account_index)?;
|
||||
|
@ -196,7 +196,7 @@ pub mod get_sysvar_with_account_check {
|
|||
pub fn clock(
|
||||
invoke_context: &InvokeContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Arc<Clock>, InstructionError> {
|
||||
check_sysvar_account::<Clock>(
|
||||
invoke_context.transaction_context,
|
||||
|
@ -209,7 +209,7 @@ pub mod get_sysvar_with_account_check {
|
|||
pub fn rent(
|
||||
invoke_context: &InvokeContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Arc<Rent>, InstructionError> {
|
||||
check_sysvar_account::<Rent>(
|
||||
invoke_context.transaction_context,
|
||||
|
@ -222,7 +222,7 @@ pub mod get_sysvar_with_account_check {
|
|||
pub fn slot_hashes(
|
||||
invoke_context: &InvokeContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Arc<SlotHashes>, InstructionError> {
|
||||
check_sysvar_account::<SlotHashes>(
|
||||
invoke_context.transaction_context,
|
||||
|
@ -236,7 +236,7 @@ pub mod get_sysvar_with_account_check {
|
|||
pub fn recent_blockhashes(
|
||||
invoke_context: &InvokeContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Arc<RecentBlockhashes>, InstructionError> {
|
||||
check_sysvar_account::<RecentBlockhashes>(
|
||||
invoke_context.transaction_context,
|
||||
|
@ -249,7 +249,7 @@ pub mod get_sysvar_with_account_check {
|
|||
pub fn stake_history(
|
||||
invoke_context: &InvokeContext,
|
||||
instruction_context: &InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Arc<StakeHistory>, InstructionError> {
|
||||
check_sysvar_account::<StakeHistory>(
|
||||
invoke_context.transaction_context,
|
||||
|
|
|
@ -62,6 +62,7 @@ use {
|
|||
pub use {
|
||||
solana_banks_client::{BanksClient, BanksClientError},
|
||||
solana_program_runtime::invoke_context::InvokeContext,
|
||||
solana_sdk::transaction_context::IndexOfAccount,
|
||||
};
|
||||
|
||||
pub mod programs;
|
||||
|
@ -94,7 +95,7 @@ fn get_invoke_context<'a, 'b>() -> &'a mut InvokeContext<'b> {
|
|||
|
||||
pub fn builtin_process_instruction(
|
||||
process_instruction: solana_sdk::entrypoint::ProcessInstruction,
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
set_invoke_context(invoke_context);
|
||||
|
@ -113,7 +114,7 @@ pub fn builtin_process_instruction(
|
|||
);
|
||||
|
||||
// Copy indices_in_instruction into a HashSet to ensure there are no duplicates
|
||||
let deduplicated_indices: HashSet<usize> = instruction_account_indices.collect();
|
||||
let deduplicated_indices: HashSet<IndexOfAccount> = instruction_account_indices.collect();
|
||||
|
||||
// Serialize entrypoint parameters with BPF ABI
|
||||
let (mut parameter_bytes, _account_lengths) = serialize_parameters(
|
||||
|
@ -177,7 +178,7 @@ pub fn builtin_process_instruction(
|
|||
macro_rules! processor {
|
||||
($process_instruction:expr) => {
|
||||
Some(
|
||||
|first_instruction_account: usize,
|
||||
|first_instruction_account: $crate::IndexOfAccount,
|
||||
invoke_context: &mut solana_program_test::InvokeContext| {
|
||||
$crate::builtin_process_instruction(
|
||||
$process_instruction,
|
||||
|
|
|
@ -14,12 +14,13 @@ use {
|
|||
program_utils::limited_deserialize,
|
||||
pubkey::{Pubkey, PUBKEY_BYTES},
|
||||
system_instruction,
|
||||
transaction_context::IndexOfAccount,
|
||||
},
|
||||
std::convert::TryFrom,
|
||||
};
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
|
|
@ -10,7 +10,7 @@ use {
|
|||
account::{Account, AccountSharedData},
|
||||
bpf_loader,
|
||||
sysvar::rent::Rent,
|
||||
transaction_context::{InstructionAccount, TransactionContext},
|
||||
transaction_context::{IndexOfAccount, InstructionAccount, TransactionContext},
|
||||
},
|
||||
test::Bencher,
|
||||
};
|
||||
|
@ -94,9 +94,9 @@ fn create_inputs() -> TransactionContext {
|
|||
.enumerate()
|
||||
.map(
|
||||
|(instruction_account_index, index_in_transaction)| InstructionAccount {
|
||||
index_in_caller: instruction_account_index,
|
||||
index_in_caller: instruction_account_index as IndexOfAccount,
|
||||
index_in_transaction,
|
||||
index_in_callee: instruction_account_index,
|
||||
index_in_callee: instruction_account_index as IndexOfAccount,
|
||||
is_signer: false,
|
||||
is_writable: instruction_account_index >= 4,
|
||||
},
|
||||
|
|
|
@ -54,7 +54,9 @@ use {
|
|||
pubkey::Pubkey,
|
||||
saturating_add_assign,
|
||||
system_instruction::{self, MAX_PERMITTED_DATA_LENGTH},
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::{cell::RefCell, fmt::Debug, rc::Rc, sync::Arc},
|
||||
thiserror::Error,
|
||||
|
@ -109,8 +111,8 @@ mod executor_metrics {
|
|||
|
||||
fn get_index_in_transaction(
|
||||
instruction_context: &InstructionContext,
|
||||
index_in_instruction: usize,
|
||||
) -> Result<usize, InstructionError> {
|
||||
index_in_instruction: IndexOfAccount,
|
||||
) -> Result<IndexOfAccount, InstructionError> {
|
||||
if index_in_instruction < instruction_context.get_number_of_program_accounts() {
|
||||
instruction_context.get_index_of_program_account_in_transaction(index_in_instruction)
|
||||
} else {
|
||||
|
@ -124,7 +126,7 @@ fn get_index_in_transaction(
|
|||
fn try_borrow_account<'a>(
|
||||
transaction_context: &'a TransactionContext,
|
||||
instruction_context: &'a InstructionContext,
|
||||
index_in_instruction: usize,
|
||||
index_in_instruction: IndexOfAccount,
|
||||
) -> Result<BorrowedAccount<'a>, InstructionError> {
|
||||
if index_in_instruction < instruction_context.get_number_of_program_accounts() {
|
||||
instruction_context.try_borrow_program_account(transaction_context, index_in_instruction)
|
||||
|
@ -138,7 +140,7 @@ fn try_borrow_account<'a>(
|
|||
}
|
||||
|
||||
pub fn create_executor(
|
||||
programdata_account_index: usize,
|
||||
programdata_account_index: IndexOfAccount,
|
||||
programdata_offset: usize,
|
||||
invoke_context: &mut InvokeContext,
|
||||
use_jit: bool,
|
||||
|
@ -247,7 +249,7 @@ pub fn create_executor(
|
|||
}
|
||||
|
||||
fn write_program_data(
|
||||
program_account_index: usize,
|
||||
program_account_index: IndexOfAccount,
|
||||
program_data_offset: usize,
|
||||
bytes: &[u8],
|
||||
invoke_context: &mut InvokeContext,
|
||||
|
@ -305,21 +307,21 @@ pub fn create_vm<'a, 'b>(
|
|||
}
|
||||
|
||||
pub fn process_instruction(
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
process_instruction_common(first_instruction_account, invoke_context, false)
|
||||
}
|
||||
|
||||
pub fn process_instruction_jit(
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
process_instruction_common(first_instruction_account, invoke_context, true)
|
||||
}
|
||||
|
||||
fn process_instruction_common(
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
use_jit: bool,
|
||||
) -> Result<(), InstructionError> {
|
||||
|
@ -467,7 +469,7 @@ fn process_instruction_common(
|
|||
}
|
||||
|
||||
fn process_loader_upgradeable_instruction(
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
use_jit: bool,
|
||||
) -> Result<(), InstructionError> {
|
||||
|
@ -1052,11 +1054,11 @@ fn process_loader_upgradeable_instruction(
|
|||
return Err(InstructionError::InvalidInstructionData);
|
||||
}
|
||||
|
||||
const PROGRAM_DATA_ACCOUNT_INDEX: usize = 0;
|
||||
const PROGRAM_DATA_ACCOUNT_INDEX: IndexOfAccount = 0;
|
||||
#[allow(dead_code)]
|
||||
// System program is only required when a CPI is performed
|
||||
const OPTIONAL_SYSTEM_PROGRAM_ACCOUNT_INDEX: usize = 1;
|
||||
const OPTIONAL_PAYER_ACCOUNT_INDEX: usize = 2;
|
||||
const OPTIONAL_SYSTEM_PROGRAM_ACCOUNT_INDEX: IndexOfAccount = 1;
|
||||
const OPTIONAL_PAYER_ACCOUNT_INDEX: IndexOfAccount = 2;
|
||||
|
||||
let programdata_account = instruction_context
|
||||
.try_borrow_instruction_account(transaction_context, PROGRAM_DATA_ACCOUNT_INDEX)?;
|
||||
|
@ -1174,7 +1176,7 @@ fn common_close_account(
|
|||
}
|
||||
|
||||
fn process_loader_instruction(
|
||||
first_instruction_account: usize,
|
||||
first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
use_jit: bool,
|
||||
) -> Result<(), InstructionError> {
|
||||
|
@ -1269,7 +1271,7 @@ impl Debug for BpfExecutor {
|
|||
impl Executor for BpfExecutor {
|
||||
fn execute(
|
||||
&self,
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let log_collector = invoke_context.get_log_collector();
|
||||
|
@ -1445,7 +1447,7 @@ mod tests {
|
|||
|
||||
fn process_instruction(
|
||||
loader_id: &Pubkey,
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
instruction_data: &[u8],
|
||||
transaction_accounts: Vec<(Pubkey, AccountSharedData)>,
|
||||
instruction_accounts: Vec<AccountMeta>,
|
||||
|
@ -1734,7 +1736,7 @@ mod tests {
|
|||
None,
|
||||
None,
|
||||
Err(InstructionError::ProgramFailedToComplete),
|
||||
|first_instruction_account: usize, invoke_context: &mut InvokeContext| {
|
||||
|first_instruction_account: IndexOfAccount, invoke_context: &mut InvokeContext| {
|
||||
invoke_context
|
||||
.get_compute_meter()
|
||||
.borrow_mut()
|
||||
|
|
|
@ -9,7 +9,7 @@ use {
|
|||
instruction::InstructionError,
|
||||
pubkey::Pubkey,
|
||||
system_instruction::MAX_PERMITTED_DATA_LENGTH,
|
||||
transaction_context::{InstructionContext, TransactionContext},
|
||||
transaction_context::{IndexOfAccount, InstructionContext, TransactionContext},
|
||||
},
|
||||
std::{io::prelude::*, mem::size_of},
|
||||
};
|
||||
|
@ -24,7 +24,7 @@ pub fn serialize_parameters(
|
|||
should_cap_ix_accounts: bool,
|
||||
) -> Result<(AlignedMemory<HOST_ALIGN>, Vec<usize>), InstructionError> {
|
||||
let num_ix_accounts = instruction_context.get_number_of_instruction_accounts();
|
||||
if should_cap_ix_accounts && num_ix_accounts > usize::from(MAX_INSTRUCTION_ACCOUNTS) {
|
||||
if should_cap_ix_accounts && num_ix_accounts > MAX_INSTRUCTION_ACCOUNTS as IndexOfAccount {
|
||||
return Err(InstructionError::MaxAccountsExceeded);
|
||||
}
|
||||
|
||||
|
@ -722,7 +722,7 @@ mod tests {
|
|||
{
|
||||
let account = invoke_context
|
||||
.transaction_context
|
||||
.get_account_at_index(index_in_transaction)
|
||||
.get_account_at_index(index_in_transaction as IndexOfAccount)
|
||||
.unwrap()
|
||||
.borrow();
|
||||
assert_eq!(&*account, original_account);
|
||||
|
@ -781,7 +781,7 @@ mod tests {
|
|||
{
|
||||
let account = invoke_context
|
||||
.transaction_context
|
||||
.get_account_at_index(index_in_transaction)
|
||||
.get_account_at_index(index_in_transaction as IndexOfAccount)
|
||||
.unwrap()
|
||||
.borrow();
|
||||
assert_eq!(&*account, original_account);
|
||||
|
|
|
@ -16,7 +16,7 @@ struct CallerAccount<'a> {
|
|||
executable: bool,
|
||||
rent_epoch: u64,
|
||||
}
|
||||
type TranslatedAccounts<'a> = Vec<(usize, Option<CallerAccount<'a>>)>;
|
||||
type TranslatedAccounts<'a> = Vec<(IndexOfAccount, Option<CallerAccount<'a>>)>;
|
||||
|
||||
/// Implemented by language specific data structure translators
|
||||
trait SyscallInvokeSigned<'a, 'b> {
|
||||
|
@ -30,7 +30,7 @@ trait SyscallInvokeSigned<'a, 'b> {
|
|||
fn translate_accounts<'c>(
|
||||
&'c self,
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
account_infos_addr: u64,
|
||||
account_infos_len: u64,
|
||||
memory_mapping: &mut MemoryMapping,
|
||||
|
@ -130,7 +130,7 @@ impl<'a, 'b> SyscallInvokeSigned<'a, 'b> for SyscallInvokeSignedRust<'a, 'b> {
|
|||
fn translate_accounts<'c>(
|
||||
&'c self,
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
account_infos_addr: u64,
|
||||
account_infos_len: u64,
|
||||
memory_mapping: &mut MemoryMapping,
|
||||
|
@ -447,7 +447,7 @@ impl<'a, 'b> SyscallInvokeSigned<'a, 'b> for SyscallInvokeSignedC<'a, 'b> {
|
|||
fn translate_accounts<'c>(
|
||||
&'c self,
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
account_infos_addr: u64,
|
||||
account_infos_len: u64,
|
||||
memory_mapping: &mut MemoryMapping,
|
||||
|
@ -605,7 +605,7 @@ impl<'a, 'b> SyscallInvokeSigned<'a, 'b> for SyscallInvokeSignedC<'a, 'b> {
|
|||
|
||||
fn get_translated_accounts<'a, T, F>(
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
program_indices: &[usize],
|
||||
program_indices: &[IndexOfAccount],
|
||||
account_info_keys: &[&Pubkey],
|
||||
account_infos: &[T],
|
||||
invoke_context: &mut InvokeContext,
|
||||
|
@ -632,7 +632,7 @@ where
|
|||
|
||||
for (instruction_account_index, instruction_account) in instruction_accounts.iter().enumerate()
|
||||
{
|
||||
if instruction_account_index != instruction_account.index_in_callee {
|
||||
if instruction_account_index as IndexOfAccount != instruction_account.index_in_callee {
|
||||
continue; // Skip duplicate account
|
||||
}
|
||||
let mut callee_account = instruction_context
|
||||
|
@ -724,7 +724,7 @@ where
|
|||
.get_orig_account_lengths()
|
||||
.map_err(SyscallError::InstructionError)?;
|
||||
caller_account.original_data_len = *orig_data_lens
|
||||
.get(instruction_account.index_in_caller)
|
||||
.get(instruction_account.index_in_caller as usize)
|
||||
.ok_or_else(|| {
|
||||
ic_msg!(
|
||||
invoke_context,
|
||||
|
|
|
@ -53,7 +53,7 @@ use {
|
|||
Secp256k1RecoverError, SECP256K1_PUBLIC_KEY_LENGTH, SECP256K1_SIGNATURE_LENGTH,
|
||||
},
|
||||
sysvar::{Sysvar, SysvarId},
|
||||
transaction_context::{InstructionAccount, TransactionContextAttribute},
|
||||
transaction_context::{IndexOfAccount, InstructionAccount, TransactionContextAttribute},
|
||||
},
|
||||
std::{
|
||||
alloc::Layout,
|
||||
|
@ -1925,7 +1925,7 @@ declare_syscall!(
|
|||
instruction_context
|
||||
.try_borrow_instruction_account(
|
||||
transaction_context,
|
||||
update.instruction_account_index as usize,
|
||||
update.instruction_account_index,
|
||||
)
|
||||
.map_err(SyscallError::InstructionError),
|
||||
result
|
||||
|
@ -3253,7 +3253,7 @@ mod tests {
|
|||
}
|
||||
if stack_height > transaction_context.get_instruction_context_stack_height() {
|
||||
let instruction_accounts = [InstructionAccount {
|
||||
index_in_transaction: index_in_trace.saturating_add(1),
|
||||
index_in_transaction: index_in_trace.saturating_add(1) as IndexOfAccount,
|
||||
index_in_caller: 0, // This is incorrect / inconsistent but not required
|
||||
index_in_callee: 0,
|
||||
is_signer: false,
|
||||
|
|
|
@ -1,10 +1,10 @@
|
|||
use {
|
||||
solana_program_runtime::invoke_context::InvokeContext,
|
||||
solana_sdk::instruction::InstructionError,
|
||||
solana_sdk::{instruction::InstructionError, transaction_context::IndexOfAccount},
|
||||
};
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
// Do nothing, compute budget instructions handled by the runtime
|
||||
|
|
|
@ -6,13 +6,13 @@ use {
|
|||
solana_program_runtime::{ic_msg, invoke_context::InvokeContext},
|
||||
solana_sdk::{
|
||||
feature_set, instruction::InstructionError, program_utils::limited_deserialize,
|
||||
pubkey::Pubkey,
|
||||
pubkey::Pubkey, transaction_context::IndexOfAccount,
|
||||
},
|
||||
std::collections::BTreeSet,
|
||||
};
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -61,7 +61,7 @@ pub fn process_instruction(
|
|||
counter += 1;
|
||||
if signer != config_account_key {
|
||||
let signer_account = instruction_context
|
||||
.try_borrow_instruction_account(transaction_context, counter)
|
||||
.try_borrow_instruction_account(transaction_context, counter as IndexOfAccount)
|
||||
.map_err(|_| {
|
||||
ic_msg!(
|
||||
invoke_context,
|
||||
|
|
|
@ -21,14 +21,14 @@ use {
|
|||
state::{Authorized, Lockup},
|
||||
},
|
||||
sysvar::clock::Clock,
|
||||
transaction_context::{InstructionContext, TransactionContext},
|
||||
transaction_context::{IndexOfAccount, InstructionContext, TransactionContext},
|
||||
},
|
||||
};
|
||||
|
||||
fn get_optional_pubkey<'a>(
|
||||
transaction_context: &'a TransactionContext,
|
||||
instruction_context: &'a InstructionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
should_be_signer: bool,
|
||||
) -> Result<Option<&'a Pubkey>, InstructionError> {
|
||||
Ok(
|
||||
|
@ -52,7 +52,7 @@ fn get_optional_pubkey<'a>(
|
|||
}
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
|
|
@ -28,7 +28,9 @@ use {
|
|||
tools::{acceptable_reference_epoch_credits, eligible_for_deactivate_delinquent},
|
||||
},
|
||||
stake_history::{StakeHistory, StakeHistoryEntry},
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
solana_vote_program::vote_state::{self, VoteState, VoteStateVersions},
|
||||
std::{collections::HashSet, convert::TryFrom},
|
||||
|
@ -539,7 +541,7 @@ pub fn authorize_with_seed(
|
|||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account: &mut BorrowedAccount,
|
||||
authority_base_index: usize,
|
||||
authority_base_index: IndexOfAccount,
|
||||
authority_seed: &str,
|
||||
authority_owner: &Pubkey,
|
||||
new_authority: &Pubkey,
|
||||
|
@ -576,8 +578,8 @@ pub fn delegate(
|
|||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account_index: usize,
|
||||
vote_account_index: usize,
|
||||
stake_account_index: IndexOfAccount,
|
||||
vote_account_index: IndexOfAccount,
|
||||
clock: &Clock,
|
||||
stake_history: &StakeHistory,
|
||||
config: &Config,
|
||||
|
@ -667,9 +669,9 @@ pub fn split(
|
|||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account_index: usize,
|
||||
stake_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
split_index: usize,
|
||||
split_index: IndexOfAccount,
|
||||
signers: &HashSet<Pubkey>,
|
||||
) -> Result<(), InstructionError> {
|
||||
let split =
|
||||
|
@ -813,8 +815,8 @@ pub fn merge(
|
|||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account_index: usize,
|
||||
source_account_index: usize,
|
||||
stake_account_index: IndexOfAccount,
|
||||
source_account_index: IndexOfAccount,
|
||||
clock: &Clock,
|
||||
stake_history: &StakeHistory,
|
||||
signers: &HashSet<Pubkey>,
|
||||
|
@ -879,8 +881,8 @@ pub fn redelegate(
|
|||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account: &mut BorrowedAccount,
|
||||
uninitialized_stake_account_index: usize,
|
||||
vote_account_index: usize,
|
||||
uninitialized_stake_account_index: IndexOfAccount,
|
||||
vote_account_index: IndexOfAccount,
|
||||
config: &Config,
|
||||
signers: &HashSet<Pubkey>,
|
||||
) -> Result<(), InstructionError> {
|
||||
|
@ -999,13 +1001,13 @@ pub fn redelegate(
|
|||
pub fn withdraw(
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account_index: usize,
|
||||
stake_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
to_index: usize,
|
||||
to_index: IndexOfAccount,
|
||||
clock: &Clock,
|
||||
stake_history: &StakeHistory,
|
||||
withdraw_authority_index: usize,
|
||||
custodian_index: Option<usize>,
|
||||
withdraw_authority_index: IndexOfAccount,
|
||||
custodian_index: Option<IndexOfAccount>,
|
||||
feature_set: &FeatureSet,
|
||||
) -> Result<(), InstructionError> {
|
||||
let withdraw_authority_pubkey = transaction_context.get_key_of_account_at_index(
|
||||
|
@ -1113,8 +1115,8 @@ pub(crate) fn deactivate_delinquent(
|
|||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
stake_account: &mut BorrowedAccount,
|
||||
delinquent_vote_account_index: usize,
|
||||
reference_vote_account_index: usize,
|
||||
delinquent_vote_account_index: IndexOfAccount,
|
||||
reference_vote_account_index: IndexOfAccount,
|
||||
current_epoch: Epoch,
|
||||
) -> Result<(), InstructionError> {
|
||||
let delinquent_vote_account_pubkey = transaction_context.get_key_of_account_at_index(
|
||||
|
@ -1205,8 +1207,8 @@ fn validate_split_amount(
|
|||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
source_account_index: usize,
|
||||
destination_account_index: usize,
|
||||
source_account_index: IndexOfAccount,
|
||||
destination_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
source_meta: &Meta,
|
||||
source_stake: Option<&Stake>,
|
||||
|
|
|
@ -11,7 +11,9 @@ use {
|
|||
pubkey::Pubkey,
|
||||
slot_hashes::{SlotHashes, MAX_ENTRIES},
|
||||
sysvar,
|
||||
transaction_context::{InstructionAccount, TransactionAccount, TransactionContext},
|
||||
transaction_context::{
|
||||
IndexOfAccount, InstructionAccount, TransactionAccount, TransactionContext,
|
||||
},
|
||||
},
|
||||
solana_vote_program::{
|
||||
vote_instruction::VoteInstruction,
|
||||
|
@ -82,7 +84,7 @@ fn create_accounts() -> (
|
|||
];
|
||||
let mut instruction_accounts = (0..4)
|
||||
.map(|index_in_callee| InstructionAccount {
|
||||
index_in_transaction: 1usize.saturating_add(index_in_callee),
|
||||
index_in_transaction: (1 as IndexOfAccount).saturating_add(index_in_callee),
|
||||
index_in_caller: index_in_callee,
|
||||
index_in_callee,
|
||||
is_signer: false,
|
||||
|
|
|
@ -12,7 +12,9 @@ use {
|
|||
instruction::InstructionError,
|
||||
program_utils::limited_deserialize,
|
||||
pubkey::Pubkey,
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::collections::HashSet,
|
||||
};
|
||||
|
@ -56,7 +58,7 @@ fn process_authorize_with_seed_instruction(
|
|||
}
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
|
|
@ -16,7 +16,9 @@ use {
|
|||
rent::Rent,
|
||||
slot_hashes::SlotHash,
|
||||
sysvar::clock::Clock,
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::{
|
||||
cmp::Ordering,
|
||||
|
@ -825,9 +827,9 @@ fn verify_authorized_signer<S: std::hash::BuildHasher>(
|
|||
pub fn withdraw<S: std::hash::BuildHasher>(
|
||||
transaction_context: &TransactionContext,
|
||||
instruction_context: &InstructionContext,
|
||||
vote_account_index: usize,
|
||||
vote_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
to_account_index: usize,
|
||||
to_account_index: IndexOfAccount,
|
||||
signers: &HashSet<Pubkey, S>,
|
||||
rent_sysvar: &Rent,
|
||||
clock: Option<&Clock>,
|
||||
|
|
|
@ -3,7 +3,10 @@
|
|||
use {
|
||||
bytemuck::Pod,
|
||||
solana_program_runtime::{ic_msg, invoke_context::InvokeContext},
|
||||
solana_sdk::instruction::{InstructionError, TRANSACTION_LEVEL_STACK_HEIGHT},
|
||||
solana_sdk::{
|
||||
instruction::{InstructionError, TRANSACTION_LEVEL_STACK_HEIGHT},
|
||||
transaction_context::IndexOfAccount,
|
||||
},
|
||||
solana_zk_token_sdk::zk_token_proof_instruction::*,
|
||||
std::result::Result,
|
||||
};
|
||||
|
@ -26,7 +29,7 @@ fn verify<T: Pod + Verifiable>(invoke_context: &mut InvokeContext) -> Result<(),
|
|||
}
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
if invoke_context.get_stack_height() != TRANSACTION_LEVEL_STACK_HEIGHT {
|
||||
|
|
|
@ -20,6 +20,7 @@ use {
|
|||
pubkey::Pubkey,
|
||||
signature::{Keypair, Signer},
|
||||
transaction::Transaction,
|
||||
transaction_context::IndexOfAccount,
|
||||
},
|
||||
std::{sync::Arc, thread::sleep, time::Duration},
|
||||
test::Bencher,
|
||||
|
@ -37,7 +38,7 @@ const NOOP_PROGRAM_ID: [u8; 32] = [
|
|||
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
|
|
@ -5,7 +5,7 @@ use {
|
|||
pubkey::Pubkey,
|
||||
rent::Rent,
|
||||
transaction::{Result, TransactionError},
|
||||
transaction_context::TransactionContext,
|
||||
transaction_context::{IndexOfAccount, TransactionContext},
|
||||
},
|
||||
};
|
||||
|
||||
|
@ -88,7 +88,7 @@ pub(crate) fn check_rent_state(
|
|||
pre_rent_state: Option<&RentState>,
|
||||
post_rent_state: Option<&RentState>,
|
||||
transaction_context: &TransactionContext,
|
||||
index: usize,
|
||||
index: IndexOfAccount,
|
||||
include_account_index_in_err: bool,
|
||||
prevent_crediting_accounts_that_end_rent_paying: bool,
|
||||
) -> Result<()> {
|
||||
|
@ -116,7 +116,7 @@ pub(crate) fn check_rent_state_with_account(
|
|||
post_rent_state: &RentState,
|
||||
address: &Pubkey,
|
||||
account_state: &AccountSharedData,
|
||||
account_index: Option<usize>,
|
||||
account_index: Option<IndexOfAccount>,
|
||||
prevent_crediting_accounts_that_end_rent_paying: bool,
|
||||
) -> Result<()> {
|
||||
submit_rent_state_metrics(pre_rent_state, post_rent_state);
|
||||
|
|
|
@ -53,7 +53,7 @@ use {
|
|||
system_program,
|
||||
sysvar::{self, epoch_schedule::EpochSchedule, instructions::construct_instructions_data},
|
||||
transaction::{Result, SanitizedTransaction, TransactionAccountLocks, TransactionError},
|
||||
transaction_context::TransactionAccount,
|
||||
transaction_context::{IndexOfAccount, TransactionAccount},
|
||||
},
|
||||
std::{
|
||||
cmp::Reverse,
|
||||
|
@ -125,7 +125,7 @@ pub struct Accounts {
|
|||
|
||||
// for the load instructions
|
||||
pub type TransactionRent = u64;
|
||||
pub type TransactionProgramIndices = Vec<Vec<usize>>;
|
||||
pub type TransactionProgramIndices = Vec<Vec<IndexOfAccount>>;
|
||||
#[derive(PartialEq, Eq, Debug, Clone)]
|
||||
pub struct LoadedTransaction {
|
||||
pub accounts: Vec<TransactionAccount>,
|
||||
|
@ -313,7 +313,7 @@ impl Accounts {
|
|||
Self::validate_fee_payer(
|
||||
key,
|
||||
&mut account,
|
||||
i,
|
||||
i as IndexOfAccount,
|
||||
error_counters,
|
||||
rent_collector,
|
||||
feature_set,
|
||||
|
@ -379,11 +379,11 @@ impl Accounts {
|
|||
self.load_executable_accounts(
|
||||
ancestors,
|
||||
&mut accounts,
|
||||
instruction.program_id_index as usize,
|
||||
instruction.program_id_index as IndexOfAccount,
|
||||
error_counters,
|
||||
)
|
||||
})
|
||||
.collect::<Result<Vec<Vec<usize>>>>()?;
|
||||
.collect::<Result<Vec<Vec<IndexOfAccount>>>>()?;
|
||||
|
||||
Ok(LoadedTransaction {
|
||||
accounts,
|
||||
|
@ -401,7 +401,7 @@ impl Accounts {
|
|||
fn validate_fee_payer(
|
||||
payer_address: &Pubkey,
|
||||
payer_account: &mut AccountSharedData,
|
||||
payer_index: usize,
|
||||
payer_index: IndexOfAccount,
|
||||
error_counters: &mut TransactionErrorMetrics,
|
||||
rent_collector: &RentCollector,
|
||||
feature_set: &FeatureSet,
|
||||
|
@ -450,11 +450,11 @@ impl Accounts {
|
|||
&self,
|
||||
ancestors: &Ancestors,
|
||||
accounts: &mut Vec<TransactionAccount>,
|
||||
mut program_account_index: usize,
|
||||
mut program_account_index: IndexOfAccount,
|
||||
error_counters: &mut TransactionErrorMetrics,
|
||||
) -> Result<Vec<usize>> {
|
||||
) -> Result<Vec<IndexOfAccount>> {
|
||||
let mut account_indices = Vec::new();
|
||||
let mut program_id = match accounts.get(program_account_index) {
|
||||
let mut program_id = match accounts.get(program_account_index as usize) {
|
||||
Some(program_account) => program_account.0,
|
||||
None => {
|
||||
error_counters.account_not_found += 1;
|
||||
|
@ -474,7 +474,7 @@ impl Accounts {
|
|||
.load_with_fixed_root(ancestors, &program_id)
|
||||
{
|
||||
Some((program_account, _)) => {
|
||||
let account_index = accounts.len();
|
||||
let account_index = accounts.len() as IndexOfAccount;
|
||||
accounts.push((program_id, program_account));
|
||||
account_index
|
||||
}
|
||||
|
@ -483,7 +483,7 @@ impl Accounts {
|
|||
return Err(TransactionError::ProgramAccountNotFound);
|
||||
}
|
||||
};
|
||||
let program = &accounts[program_account_index].1;
|
||||
let program = &accounts[program_account_index as usize].1;
|
||||
if !program.executable() {
|
||||
error_counters.invalid_program_for_execution += 1;
|
||||
return Err(TransactionError::InvalidProgramForExecution);
|
||||
|
@ -503,7 +503,7 @@ impl Accounts {
|
|||
.load_with_fixed_root(ancestors, &programdata_address)
|
||||
{
|
||||
Some((programdata_account, _)) => {
|
||||
let account_index = accounts.len();
|
||||
let account_index = accounts.len() as IndexOfAccount;
|
||||
accounts.push((programdata_address, programdata_account));
|
||||
account_index
|
||||
}
|
||||
|
@ -2028,11 +2028,11 @@ mod tests {
|
|||
for (i, program_index) in program_indices.iter().enumerate() {
|
||||
// +1 to skip first not loader account
|
||||
assert_eq!(
|
||||
loaded_transaction.accounts[*program_index].0,
|
||||
loaded_transaction.accounts[*program_index as usize].0,
|
||||
accounts[i + 1].0
|
||||
);
|
||||
assert_eq!(
|
||||
loaded_transaction.accounts[*program_index].1,
|
||||
loaded_transaction.accounts[*program_index as usize].1,
|
||||
accounts[i + 1].1
|
||||
);
|
||||
}
|
||||
|
@ -2262,7 +2262,10 @@ mod tests {
|
|||
assert_eq!(loaded_accounts.len(), 1);
|
||||
let result = loaded_accounts[0].0.as_ref().unwrap();
|
||||
assert_eq!(result.accounts[..2], accounts[..2]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][0]], accounts[2]);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][0] as usize],
|
||||
accounts[2]
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -2345,7 +2348,10 @@ mod tests {
|
|||
assert_eq!(loaded_accounts.len(), 1);
|
||||
let result = loaded_accounts[0].0.as_ref().unwrap();
|
||||
assert_eq!(result.accounts[..2], accounts[..2]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][0]], accounts[5]);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][0] as usize],
|
||||
accounts[5]
|
||||
);
|
||||
|
||||
// Solution 2: mark programdata as readonly
|
||||
message.account_keys = vec![key0, key1, key2]; // revert key change
|
||||
|
@ -2357,9 +2363,18 @@ mod tests {
|
|||
assert_eq!(loaded_accounts.len(), 1);
|
||||
let result = loaded_accounts[0].0.as_ref().unwrap();
|
||||
assert_eq!(result.accounts[..2], accounts[..2]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][0]], accounts[5]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][1]], accounts[4]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][2]], accounts[3]);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][0] as usize],
|
||||
accounts[5]
|
||||
);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][1] as usize],
|
||||
accounts[4]
|
||||
);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][2] as usize],
|
||||
accounts[3]
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -2428,7 +2443,7 @@ mod tests {
|
|||
let result = loaded_accounts[0].0.as_ref().unwrap();
|
||||
assert_eq!(result.accounts[..2], accounts_with_upgradeable_loader[..2]);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][0]],
|
||||
result.accounts[result.program_indices[0][0] as usize],
|
||||
accounts_with_upgradeable_loader[2]
|
||||
);
|
||||
|
||||
|
@ -2442,7 +2457,10 @@ mod tests {
|
|||
assert_eq!(loaded_accounts.len(), 1);
|
||||
let result = loaded_accounts[0].0.as_ref().unwrap();
|
||||
assert_eq!(result.accounts[..2], accounts[..2]);
|
||||
assert_eq!(result.accounts[result.program_indices[0][0]], accounts[2]);
|
||||
assert_eq!(
|
||||
result.accounts[result.program_indices[0][0] as usize],
|
||||
accounts[2]
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
|
|
@ -8052,6 +8052,7 @@ pub(crate) mod tests {
|
|||
system_program,
|
||||
timing::duration_as_s,
|
||||
transaction::MAX_TX_ACCOUNT_LOCKS,
|
||||
transaction_context::IndexOfAccount,
|
||||
},
|
||||
solana_vote_program::{
|
||||
vote_instruction,
|
||||
|
@ -9061,7 +9062,7 @@ pub(crate) mod tests {
|
|||
}
|
||||
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -12670,7 +12671,7 @@ pub(crate) mod tests {
|
|||
Pubkey::new(&[42u8; 32])
|
||||
}
|
||||
fn mock_vote_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -12729,7 +12730,7 @@ pub(crate) mod tests {
|
|||
let mut bank = Bank::new_for_tests(&genesis_config);
|
||||
|
||||
fn mock_vote_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Err(InstructionError::Custom(42))
|
||||
|
@ -12777,7 +12778,7 @@ pub(crate) mod tests {
|
|||
let mut bank = create_simple_test_bank(500);
|
||||
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Err(InstructionError::Custom(42))
|
||||
|
@ -14051,7 +14052,7 @@ pub(crate) mod tests {
|
|||
let mut bank = Bank::new_for_tests(&genesis_config);
|
||||
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -14111,7 +14112,7 @@ pub(crate) mod tests {
|
|||
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -14333,7 +14334,7 @@ pub(crate) mod tests {
|
|||
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_ok_vote_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -14581,7 +14582,7 @@ pub(crate) mod tests {
|
|||
#[test]
|
||||
fn test_same_program_id_uses_unqiue_executable_accounts() {
|
||||
fn nested_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -14851,7 +14852,7 @@ pub(crate) mod tests {
|
|||
fn test_add_builtin_no_overwrite() {
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -14887,7 +14888,7 @@ pub(crate) mod tests {
|
|||
fn test_add_builtin_loader_no_overwrite() {
|
||||
#[allow(clippy::unnecessary_wraps)]
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -15207,7 +15208,7 @@ pub(crate) mod tests {
|
|||
impl Executor for TestExecutor {
|
||||
fn execute(
|
||||
&self,
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
Ok(())
|
||||
|
@ -17016,7 +17017,7 @@ pub(crate) mod tests {
|
|||
|
||||
let mock_program_id = Pubkey::new(&[2u8; 32]);
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let mock_program_id = Pubkey::new(&[2u8; 32]);
|
||||
|
@ -17214,7 +17215,7 @@ pub(crate) mod tests {
|
|||
let mut bank = Bank::new_for_tests(&genesis_config);
|
||||
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -17424,7 +17425,7 @@ pub(crate) mod tests {
|
|||
let mut bank = Bank::new_for_tests(&genesis_config);
|
||||
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
let compute_budget = invoke_context.get_compute_budget();
|
||||
|
@ -17468,7 +17469,7 @@ pub(crate) mod tests {
|
|||
let mut bank = Bank::new_for_tests(&genesis_config);
|
||||
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
let compute_budget = invoke_context.get_compute_budget();
|
||||
|
@ -17519,7 +17520,7 @@ pub(crate) mod tests {
|
|||
.unwrap();
|
||||
|
||||
fn mock_ix_processor(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> std::result::Result<(), InstructionError> {
|
||||
let compute_budget = invoke_context.get_compute_budget();
|
||||
|
@ -18023,7 +18024,7 @@ pub(crate) mod tests {
|
|||
}
|
||||
|
||||
fn mock_transfer_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -18878,7 +18879,7 @@ pub(crate) mod tests {
|
|||
}
|
||||
|
||||
fn mock_realloc_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> result::Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
|
|
@ -4,8 +4,12 @@ use {
|
|||
bank::Bank,
|
||||
},
|
||||
solana_sdk::{
|
||||
account::ReadableAccount, feature_set, message::SanitizedMessage, native_loader,
|
||||
transaction::Result, transaction_context::TransactionContext,
|
||||
account::ReadableAccount,
|
||||
feature_set,
|
||||
message::SanitizedMessage,
|
||||
native_loader,
|
||||
transaction::Result,
|
||||
transaction_context::{IndexOfAccount, TransactionContext},
|
||||
},
|
||||
};
|
||||
|
||||
|
@ -22,7 +26,9 @@ impl Bank {
|
|||
(0..message.account_keys().len())
|
||||
.map(|i| {
|
||||
let rent_state = if message.is_writable(i) {
|
||||
let state = if let Ok(account) = transaction_context.get_account_at_index(i) {
|
||||
let state = if let Ok(account) =
|
||||
transaction_context.get_account_at_index(i as IndexOfAccount)
|
||||
{
|
||||
let account = account.borrow();
|
||||
|
||||
// Native programs appear to be RentPaying because they carry low lamport
|
||||
|
@ -68,7 +74,7 @@ impl Bank {
|
|||
pre_state_info.rent_state.as_ref(),
|
||||
post_state_info.rent_state.as_ref(),
|
||||
transaction_context,
|
||||
i,
|
||||
i as IndexOfAccount,
|
||||
include_account_index_in_err,
|
||||
prevent_crediting_accounts_that_end_rent_paying,
|
||||
)?;
|
||||
|
|
|
@ -18,7 +18,7 @@ use {
|
|||
saturating_add_assign,
|
||||
sysvar::instructions,
|
||||
transaction::TransactionError,
|
||||
transaction_context::{InstructionAccount, TransactionContext},
|
||||
transaction_context::{IndexOfAccount, InstructionAccount, TransactionContext},
|
||||
},
|
||||
std::{borrow::Cow, cell::RefCell, rc::Rc, sync::Arc},
|
||||
};
|
||||
|
@ -52,7 +52,7 @@ impl MessageProcessor {
|
|||
pub fn process_message(
|
||||
builtin_programs: &[BuiltinProgram],
|
||||
message: &SanitizedMessage,
|
||||
program_indices: &[Vec<usize>],
|
||||
program_indices: &[Vec<IndexOfAccount>],
|
||||
transaction_context: &mut TransactionContext,
|
||||
rent: Rent,
|
||||
log_collector: Option<Rc<RefCell<LogCollector>>>,
|
||||
|
@ -116,11 +116,12 @@ impl MessageProcessor {
|
|||
.ok_or(TransactionError::InvalidAccountIndex)?
|
||||
.iter()
|
||||
.position(|account_index| account_index == index_in_transaction)
|
||||
.unwrap_or(instruction_account_index);
|
||||
.unwrap_or(instruction_account_index)
|
||||
as IndexOfAccount;
|
||||
let index_in_transaction = *index_in_transaction as usize;
|
||||
instruction_accounts.push(InstructionAccount {
|
||||
index_in_transaction,
|
||||
index_in_caller: index_in_transaction,
|
||||
index_in_transaction: index_in_transaction as IndexOfAccount,
|
||||
index_in_caller: index_in_transaction as IndexOfAccount,
|
||||
index_in_callee,
|
||||
is_signer: message.is_signer(index_in_transaction),
|
||||
is_writable: message.is_writable(index_in_transaction),
|
||||
|
@ -216,7 +217,7 @@ mod tests {
|
|||
}
|
||||
|
||||
fn mock_system_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -428,7 +429,7 @@ mod tests {
|
|||
}
|
||||
|
||||
fn mock_system_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -642,7 +643,7 @@ mod tests {
|
|||
fn test_precompile() {
|
||||
let mock_program_id = Pubkey::new_unique();
|
||||
fn mock_process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
_invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
Err(InstructionError::Custom(0xbabb1e))
|
||||
|
|
|
@ -11,7 +11,9 @@ use {
|
|||
pubkey::Pubkey,
|
||||
system_instruction::{nonce_to_instruction_error, NonceError},
|
||||
sysvar::rent::Rent,
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::collections::HashSet,
|
||||
};
|
||||
|
@ -79,9 +81,9 @@ pub fn advance_nonce_account(
|
|||
}
|
||||
|
||||
pub fn withdraw_nonce_account(
|
||||
from_account_index: usize,
|
||||
from_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
to_account_index: usize,
|
||||
to_account_index: IndexOfAccount,
|
||||
rent: &Rent,
|
||||
signers: &HashSet<Pubkey>,
|
||||
invoke_context: &InvokeContext,
|
||||
|
@ -282,8 +284,8 @@ mod test {
|
|||
},
|
||||
};
|
||||
|
||||
pub const NONCE_ACCOUNT_INDEX: usize = 0;
|
||||
pub const WITHDRAW_TO_ACCOUNT_INDEX: usize = 1;
|
||||
pub const NONCE_ACCOUNT_INDEX: IndexOfAccount = 0;
|
||||
pub const WITHDRAW_TO_ACCOUNT_INDEX: IndexOfAccount = 1;
|
||||
|
||||
macro_rules! push_instruction_context {
|
||||
($invoke_context:expr, $transaction_context:ident, $instruction_context:ident, $instruction_accounts:ident) => {
|
||||
|
|
|
@ -19,7 +19,9 @@ use {
|
|||
NonceError, SystemError, SystemInstruction, MAX_PERMITTED_DATA_LENGTH,
|
||||
},
|
||||
system_program,
|
||||
transaction_context::{BorrowedAccount, InstructionContext, TransactionContext},
|
||||
transaction_context::{
|
||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||
},
|
||||
},
|
||||
std::collections::HashSet,
|
||||
};
|
||||
|
@ -145,8 +147,8 @@ fn allocate_and_assign(
|
|||
|
||||
#[allow(clippy::too_many_arguments)]
|
||||
fn create_account(
|
||||
from_account_index: usize,
|
||||
to_account_index: usize,
|
||||
from_account_index: IndexOfAccount,
|
||||
to_account_index: IndexOfAccount,
|
||||
to_address: &Address,
|
||||
lamports: u64,
|
||||
space: u64,
|
||||
|
@ -182,8 +184,8 @@ fn create_account(
|
|||
}
|
||||
|
||||
fn transfer_verified(
|
||||
from_account_index: usize,
|
||||
to_account_index: usize,
|
||||
from_account_index: IndexOfAccount,
|
||||
to_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
|
@ -214,8 +216,8 @@ fn transfer_verified(
|
|||
}
|
||||
|
||||
fn transfer(
|
||||
from_account_index: usize,
|
||||
to_account_index: usize,
|
||||
from_account_index: IndexOfAccount,
|
||||
to_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
|
@ -252,11 +254,11 @@ fn transfer(
|
|||
}
|
||||
|
||||
fn transfer_with_seed(
|
||||
from_account_index: usize,
|
||||
from_base_account_index: usize,
|
||||
from_account_index: IndexOfAccount,
|
||||
from_base_account_index: IndexOfAccount,
|
||||
from_seed: &str,
|
||||
from_owner: &Pubkey,
|
||||
to_account_index: usize,
|
||||
to_account_index: IndexOfAccount,
|
||||
lamports: u64,
|
||||
invoke_context: &InvokeContext,
|
||||
transaction_context: &TransactionContext,
|
||||
|
@ -314,7 +316,7 @@ fn transfer_with_seed(
|
|||
}
|
||||
|
||||
pub fn process_instruction(
|
||||
_first_instruction_account: usize,
|
||||
_first_instruction_account: IndexOfAccount,
|
||||
invoke_context: &mut InvokeContext,
|
||||
) -> Result<(), InstructionError> {
|
||||
let transaction_context = &invoke_context.transaction_context;
|
||||
|
@ -1865,7 +1867,7 @@ mod tests {
|
|||
},
|
||||
],
|
||||
Ok(()),
|
||||
|first_instruction_account: usize, invoke_context: &mut InvokeContext| {
|
||||
|first_instruction_account: IndexOfAccount, invoke_context: &mut InvokeContext| {
|
||||
invoke_context.blockhash = hash(&serialize(&0).unwrap());
|
||||
super::process_instruction(first_instruction_account, invoke_context)
|
||||
},
|
||||
|
@ -2223,7 +2225,7 @@ mod tests {
|
|||
},
|
||||
],
|
||||
Err(NonceError::NoRecentBlockhashes.into()),
|
||||
|first_instruction_account: usize, invoke_context: &mut InvokeContext| {
|
||||
|first_instruction_account: IndexOfAccount, invoke_context: &mut InvokeContext| {
|
||||
invoke_context.blockhash = hash(&serialize(&0).unwrap());
|
||||
super::process_instruction(first_instruction_account, invoke_context)
|
||||
},
|
||||
|
|
|
@ -77,21 +77,24 @@ pub enum TransactionContextAttribute {
|
|||
InstructionAccountIsWritable,
|
||||
}
|
||||
|
||||
/// Index of an account inside of the TransactionContext or an InstructionContext.
|
||||
pub type IndexOfAccount = u16;
|
||||
|
||||
/// Contains account meta data which varies between instruction.
|
||||
///
|
||||
/// It also contains indices to other structures for faster lookup.
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct InstructionAccount {
|
||||
/// Points to the account and its key in the `TransactionContext`
|
||||
pub index_in_transaction: usize,
|
||||
pub index_in_transaction: IndexOfAccount,
|
||||
/// Points to the first occurrence in the parent `InstructionContext`
|
||||
///
|
||||
/// This excludes the program accounts.
|
||||
pub index_in_caller: usize,
|
||||
pub index_in_caller: IndexOfAccount,
|
||||
/// Points to the first occurrence in the current `InstructionContext`
|
||||
///
|
||||
/// This excludes the program accounts.
|
||||
pub index_in_callee: usize,
|
||||
pub index_in_callee: IndexOfAccount,
|
||||
/// Is this account supposed to sign
|
||||
pub is_signer: bool,
|
||||
/// Is this account allowed to become writable
|
||||
|
@ -169,17 +172,17 @@ impl TransactionContext {
|
|||
}
|
||||
|
||||
/// Returns the total number of accounts loaded in this Transaction
|
||||
pub fn get_number_of_accounts(&self) -> usize {
|
||||
self.accounts.len()
|
||||
pub fn get_number_of_accounts(&self) -> IndexOfAccount {
|
||||
self.accounts.len() as IndexOfAccount
|
||||
}
|
||||
|
||||
/// Searches for an account by its key
|
||||
pub fn get_key_of_account_at_index(
|
||||
&self,
|
||||
index_in_transaction: usize,
|
||||
index_in_transaction: IndexOfAccount,
|
||||
) -> Result<&Pubkey, InstructionError> {
|
||||
self.account_keys
|
||||
.get(index_in_transaction)
|
||||
.get(index_in_transaction as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)
|
||||
}
|
||||
|
||||
|
@ -187,21 +190,27 @@ impl TransactionContext {
|
|||
#[cfg(not(target_os = "solana"))]
|
||||
pub fn get_account_at_index(
|
||||
&self,
|
||||
index_in_transaction: usize,
|
||||
index_in_transaction: IndexOfAccount,
|
||||
) -> Result<&RefCell<AccountSharedData>, InstructionError> {
|
||||
self.accounts
|
||||
.get(index_in_transaction)
|
||||
.get(index_in_transaction as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)
|
||||
}
|
||||
|
||||
/// Searches for an account by its key
|
||||
pub fn find_index_of_account(&self, pubkey: &Pubkey) -> Option<usize> {
|
||||
self.account_keys.iter().position(|key| key == pubkey)
|
||||
pub fn find_index_of_account(&self, pubkey: &Pubkey) -> Option<IndexOfAccount> {
|
||||
self.account_keys
|
||||
.iter()
|
||||
.position(|key| key == pubkey)
|
||||
.map(|index| index as IndexOfAccount)
|
||||
}
|
||||
|
||||
/// Searches for a program account by its key
|
||||
pub fn find_index_of_program_account(&self, pubkey: &Pubkey) -> Option<usize> {
|
||||
self.account_keys.iter().rposition(|key| key == pubkey)
|
||||
pub fn find_index_of_program_account(&self, pubkey: &Pubkey) -> Option<IndexOfAccount> {
|
||||
self.account_keys
|
||||
.iter()
|
||||
.rposition(|key| key == pubkey)
|
||||
.map(|index| index as IndexOfAccount)
|
||||
}
|
||||
|
||||
/// Returns the instruction trace length.
|
||||
|
@ -416,7 +425,7 @@ pub struct TransactionReturnData {
|
|||
pub struct InstructionContext {
|
||||
nesting_level: usize,
|
||||
instruction_accounts_lamport_sum: u128,
|
||||
program_accounts: Vec<usize>,
|
||||
program_accounts: Vec<IndexOfAccount>,
|
||||
instruction_accounts: Vec<InstructionAccount>,
|
||||
instruction_data: Vec<u8>,
|
||||
}
|
||||
|
@ -426,7 +435,7 @@ impl InstructionContext {
|
|||
#[cfg(not(target_os = "solana"))]
|
||||
pub fn configure(
|
||||
&mut self,
|
||||
program_accounts: &[usize],
|
||||
program_accounts: &[IndexOfAccount],
|
||||
instruction_accounts: &[InstructionAccount],
|
||||
instruction_data: &[u8],
|
||||
) {
|
||||
|
@ -448,19 +457,19 @@ impl InstructionContext {
|
|||
}
|
||||
|
||||
/// Number of program accounts
|
||||
pub fn get_number_of_program_accounts(&self) -> usize {
|
||||
self.program_accounts.len()
|
||||
pub fn get_number_of_program_accounts(&self) -> IndexOfAccount {
|
||||
self.program_accounts.len() as IndexOfAccount
|
||||
}
|
||||
|
||||
/// Number of accounts in this Instruction (without program accounts)
|
||||
pub fn get_number_of_instruction_accounts(&self) -> usize {
|
||||
self.instruction_accounts.len()
|
||||
pub fn get_number_of_instruction_accounts(&self) -> IndexOfAccount {
|
||||
self.instruction_accounts.len() as IndexOfAccount
|
||||
}
|
||||
|
||||
/// Assert that enough account were supplied to this Instruction
|
||||
pub fn check_number_of_instruction_accounts(
|
||||
&self,
|
||||
expected_at_least: usize,
|
||||
expected_at_least: IndexOfAccount,
|
||||
) -> Result<(), InstructionError> {
|
||||
if self.get_number_of_instruction_accounts() < expected_at_least {
|
||||
Err(InstructionError::NotEnoughAccountKeys)
|
||||
|
@ -479,12 +488,16 @@ impl InstructionContext {
|
|||
&self,
|
||||
transaction_context: &TransactionContext,
|
||||
pubkey: &Pubkey,
|
||||
) -> Option<usize> {
|
||||
) -> Option<IndexOfAccount> {
|
||||
self.program_accounts
|
||||
.iter()
|
||||
.position(|index_in_transaction| {
|
||||
transaction_context.account_keys.get(*index_in_transaction) == Some(pubkey)
|
||||
transaction_context
|
||||
.account_keys
|
||||
.get(*index_in_transaction as usize)
|
||||
== Some(pubkey)
|
||||
})
|
||||
.map(|index| index as IndexOfAccount)
|
||||
}
|
||||
|
||||
/// Searches for an instruction account by its key
|
||||
|
@ -492,49 +505,50 @@ impl InstructionContext {
|
|||
&self,
|
||||
transaction_context: &TransactionContext,
|
||||
pubkey: &Pubkey,
|
||||
) -> Option<usize> {
|
||||
) -> Option<IndexOfAccount> {
|
||||
self.instruction_accounts
|
||||
.iter()
|
||||
.position(|instruction_account| {
|
||||
transaction_context
|
||||
.account_keys
|
||||
.get(instruction_account.index_in_transaction)
|
||||
.get(instruction_account.index_in_transaction as usize)
|
||||
== Some(pubkey)
|
||||
})
|
||||
.map(|index| index as IndexOfAccount)
|
||||
}
|
||||
|
||||
/// Translates the given instruction wide program_account_index into a transaction wide index
|
||||
pub fn get_index_of_program_account_in_transaction(
|
||||
&self,
|
||||
program_account_index: usize,
|
||||
) -> Result<usize, InstructionError> {
|
||||
program_account_index: IndexOfAccount,
|
||||
) -> Result<IndexOfAccount, InstructionError> {
|
||||
Ok(*self
|
||||
.program_accounts
|
||||
.get(program_account_index)
|
||||
.get(program_account_index as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)?)
|
||||
}
|
||||
|
||||
/// Translates the given instruction wide instruction_account_index into a transaction wide index
|
||||
pub fn get_index_of_instruction_account_in_transaction(
|
||||
&self,
|
||||
instruction_account_index: usize,
|
||||
) -> Result<usize, InstructionError> {
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<IndexOfAccount, InstructionError> {
|
||||
Ok(self
|
||||
.instruction_accounts
|
||||
.get(instruction_account_index)
|
||||
.get(instruction_account_index as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)?
|
||||
.index_in_transaction)
|
||||
.index_in_transaction as IndexOfAccount)
|
||||
}
|
||||
|
||||
/// Returns `Some(instruction_account_index)` if this is a duplicate
|
||||
/// and `None` if it is the first account with this key
|
||||
pub fn is_instruction_account_duplicate(
|
||||
&self,
|
||||
instruction_account_index: usize,
|
||||
) -> Result<Option<usize>, InstructionError> {
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<Option<IndexOfAccount>, InstructionError> {
|
||||
let index_in_callee = self
|
||||
.instruction_accounts
|
||||
.get(instruction_account_index)
|
||||
.get(instruction_account_index as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)?
|
||||
.index_in_callee;
|
||||
Ok(if index_in_callee == instruction_account_index {
|
||||
|
@ -560,12 +574,12 @@ impl InstructionContext {
|
|||
fn try_borrow_account<'a, 'b: 'a>(
|
||||
&'a self,
|
||||
transaction_context: &'b TransactionContext,
|
||||
index_in_transaction: usize,
|
||||
index_in_instruction: usize,
|
||||
index_in_transaction: IndexOfAccount,
|
||||
index_in_instruction: IndexOfAccount,
|
||||
) -> Result<BorrowedAccount<'a>, InstructionError> {
|
||||
let account = transaction_context
|
||||
.accounts
|
||||
.get(index_in_transaction)
|
||||
.get(index_in_transaction as usize)
|
||||
.ok_or(InstructionError::MissingAccount)?
|
||||
.try_borrow_mut()
|
||||
.map_err(|_| InstructionError::AccountBorrowFailed)?;
|
||||
|
@ -595,7 +609,7 @@ impl InstructionContext {
|
|||
pub fn try_borrow_program_account<'a, 'b: 'a>(
|
||||
&'a self,
|
||||
transaction_context: &'b TransactionContext,
|
||||
program_account_index: usize,
|
||||
program_account_index: IndexOfAccount,
|
||||
) -> Result<BorrowedAccount<'a>, InstructionError> {
|
||||
let index_in_transaction =
|
||||
self.get_index_of_program_account_in_transaction(program_account_index)?;
|
||||
|
@ -610,7 +624,7 @@ impl InstructionContext {
|
|||
pub fn try_borrow_instruction_account<'a, 'b: 'a>(
|
||||
&'a self,
|
||||
transaction_context: &'b TransactionContext,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<BorrowedAccount<'a>, InstructionError> {
|
||||
let index_in_transaction =
|
||||
self.get_index_of_instruction_account_in_transaction(instruction_account_index)?;
|
||||
|
@ -625,11 +639,11 @@ impl InstructionContext {
|
|||
/// Returns whether an instruction account is a signer
|
||||
pub fn is_instruction_account_signer(
|
||||
&self,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<bool, InstructionError> {
|
||||
Ok(self
|
||||
.instruction_accounts
|
||||
.get(instruction_account_index)
|
||||
.get(instruction_account_index as usize)
|
||||
.ok_or(InstructionError::MissingAccount)?
|
||||
.is_signer)
|
||||
}
|
||||
|
@ -637,11 +651,11 @@ impl InstructionContext {
|
|||
/// Returns whether an instruction account is writable
|
||||
pub fn is_instruction_account_writable(
|
||||
&self,
|
||||
instruction_account_index: usize,
|
||||
instruction_account_index: IndexOfAccount,
|
||||
) -> Result<bool, InstructionError> {
|
||||
Ok(self
|
||||
.instruction_accounts
|
||||
.get(instruction_account_index)
|
||||
.get(instruction_account_index as usize)
|
||||
.ok_or(InstructionError::MissingAccount)?
|
||||
.is_writable)
|
||||
}
|
||||
|
@ -669,14 +683,14 @@ impl InstructionContext {
|
|||
pub struct BorrowedAccount<'a> {
|
||||
transaction_context: &'a TransactionContext,
|
||||
instruction_context: &'a InstructionContext,
|
||||
index_in_transaction: usize,
|
||||
index_in_instruction: usize,
|
||||
index_in_transaction: IndexOfAccount,
|
||||
index_in_instruction: IndexOfAccount,
|
||||
account: RefMut<'a, AccountSharedData>,
|
||||
}
|
||||
|
||||
impl<'a> BorrowedAccount<'a> {
|
||||
/// Returns the index of this account (transaction wide)
|
||||
pub fn get_index_in_transaction(&self) -> usize {
|
||||
pub fn get_index_in_transaction(&self) -> IndexOfAccount {
|
||||
self.index_in_transaction
|
||||
}
|
||||
|
||||
|
@ -1043,7 +1057,7 @@ impl<'a> BorrowedAccount<'a> {
|
|||
.account_touched_flags
|
||||
.try_borrow_mut()
|
||||
.map_err(|_| InstructionError::GenericError)?
|
||||
.get_mut(self.index_in_transaction)
|
||||
.get_mut(self.index_in_transaction as usize)
|
||||
.ok_or(InstructionError::NotEnoughAccountKeys)? = true;
|
||||
}
|
||||
Ok(())
|
||||
|
|
Loading…
Reference in New Issue