| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | use solana_sdk::{
 | 
					
						
							| 
									
										
										
										
											2020-10-27 05:43:14 +00:00
										 |  |  |     account::Account,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |     feature_set::{
 | 
					
						
							|  |  |  |         bpf_compute_budget_balancing, max_invoke_depth_4, max_program_call_depth_64,
 | 
					
						
							|  |  |  |         pubkey_log_syscall_enabled, FeatureSet,
 | 
					
						
							|  |  |  |     },
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     instruction::{CompiledInstruction, Instruction, InstructionError},
 | 
					
						
							| 
									
										
										
										
											2020-10-27 05:43:14 +00:00
										 |  |  |     keyed_account::KeyedAccount,
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     message::Message,
 | 
					
						
							|  |  |  |     pubkey::Pubkey,
 | 
					
						
							|  |  |  | };
 | 
					
						
							| 
									
										
										
										
											2020-10-20 17:20:40 +00:00
										 |  |  | use std::{cell::RefCell, fmt::Debug, rc::Rc, sync::Arc};
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Prototype of a native loader entry point
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | /// program_id: Program ID of the currently executing program
 | 
					
						
							|  |  |  | /// keyed_accounts: Accounts passed as part of the instruction
 | 
					
						
							|  |  |  | /// instruction_data: Instruction data
 | 
					
						
							|  |  |  | /// invoke_context: Invocation context
 | 
					
						
							|  |  |  | pub type LoaderEntrypoint = unsafe extern "C" fn(
 | 
					
						
							|  |  |  |     program_id: &Pubkey,
 | 
					
						
							|  |  |  |     keyed_accounts: &[KeyedAccount],
 | 
					
						
							|  |  |  |     instruction_data: &[u8],
 | 
					
						
							|  |  |  |     invoke_context: &dyn InvokeContext,
 | 
					
						
							|  |  |  | ) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub type ProcessInstructionWithContext =
 | 
					
						
							|  |  |  |     fn(&Pubkey, &[KeyedAccount], &[u8], &mut dyn InvokeContext) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Invocation context passed to loaders
 | 
					
						
							|  |  |  | pub trait InvokeContext {
 | 
					
						
							|  |  |  |     /// Push a program ID on to the invocation stack
 | 
					
						
							|  |  |  |     fn push(&mut self, key: &Pubkey) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  |     /// Pop a program ID off of the invocation stack
 | 
					
						
							|  |  |  |     fn pop(&mut self);
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  |     /// Current depth of the invocation stake
 | 
					
						
							|  |  |  |     fn invoke_depth(&self) -> usize;
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Verify and update PreAccount state based on program execution
 | 
					
						
							|  |  |  |     fn verify_and_update(
 | 
					
						
							|  |  |  |         &mut self,
 | 
					
						
							|  |  |  |         message: &Message,
 | 
					
						
							|  |  |  |         instruction: &CompiledInstruction,
 | 
					
						
							|  |  |  |         accounts: &[Rc<RefCell<Account>>],
 | 
					
						
							|  |  |  |     ) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  |     /// Get the program ID of the currently executing program
 | 
					
						
							|  |  |  |     fn get_caller(&self) -> Result<&Pubkey, InstructionError>;
 | 
					
						
							|  |  |  |     /// Get a list of built-in programs
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |     fn get_programs(&self) -> &[(Pubkey, ProcessInstructionWithContext)];
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Get this invocation's logger
 | 
					
						
							|  |  |  |     fn get_logger(&self) -> Rc<RefCell<dyn Logger>>;
 | 
					
						
							|  |  |  |     /// Get this invocation's compute budget
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |     fn get_bpf_compute_budget(&self) -> &BpfComputeBudget;
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Get this invocation's compute meter
 | 
					
						
							|  |  |  |     fn get_compute_meter(&self) -> Rc<RefCell<dyn ComputeMeter>>;
 | 
					
						
							|  |  |  |     /// Loaders may need to do work in order to execute a program.  Cache
 | 
					
						
							|  |  |  |     /// the work that can be re-used across executions
 | 
					
						
							| 
									
										
										
										
											2020-10-30 10:35:59 +00:00
										 |  |  |     fn add_executor(&self, pubkey: &Pubkey, executor: Arc<dyn Executor>);
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Get the completed loader work that can be re-used across executions
 | 
					
						
							| 
									
										
										
										
											2020-10-30 10:35:59 +00:00
										 |  |  |     fn get_executor(&self, pubkey: &Pubkey) -> Option<Arc<dyn Executor>>;
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Record invoked instruction
 | 
					
						
							|  |  |  |     fn record_instruction(&self, instruction: &Instruction);
 | 
					
						
							| 
									
										
										
										
											2020-09-29 14:36:30 -07:00
										 |  |  |     /// Get the bank's active feature set
 | 
					
						
							|  |  |  |     fn is_feature_active(&self, feature_id: &Pubkey) -> bool;
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  | #[derive(Clone, Copy, Debug, AbiExample)]
 | 
					
						
							|  |  |  | pub struct BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Number of compute units that an instruction is allowed.  Compute units
 | 
					
						
							|  |  |  |     /// are consumed by program execution, resources they use, etc...
 | 
					
						
							|  |  |  |     pub max_units: u64,
 | 
					
						
							|  |  |  |     /// Number of compute units consumed by a log call
 | 
					
						
							|  |  |  |     pub log_units: u64,
 | 
					
						
							|  |  |  |     /// Number of compute units consumed by a log_u64 call
 | 
					
						
							|  |  |  |     pub log_64_units: u64,
 | 
					
						
							|  |  |  |     /// Number of compute units consumed by a create_program_address call
 | 
					
						
							|  |  |  |     pub create_program_address_units: u64,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |     /// Number of compute units consumed by an invoke call (not including the cost incurred by
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// the called program)
 | 
					
						
							|  |  |  |     pub invoke_units: u64,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |     /// Maximum cross-program invocation depth allowed including the original caller
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     pub max_invoke_depth: usize,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |     /// Base number of compute units consumed to call SHA256
 | 
					
						
							| 
									
										
										
										
											2020-09-29 23:29:20 -07:00
										 |  |  |     pub sha256_base_cost: u64,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |     /// Incremental number of units consumed by SHA256 (based on bytes)
 | 
					
						
							| 
									
										
										
										
											2020-09-29 23:29:20 -07:00
										 |  |  |     pub sha256_byte_cost: u64,
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |     /// Maximum BPF to BPF call depth
 | 
					
						
							|  |  |  |     pub max_call_depth: usize,
 | 
					
						
							|  |  |  |     /// Size of a stack frame in bytes, must match the size specified in the LLVM BPF backend
 | 
					
						
							|  |  |  |     pub stack_frame_size: usize,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |     /// Number of compute units consumed by logging a `Pubkey`
 | 
					
						
							|  |  |  |     pub log_pubkey_units: u64,
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  | impl Default for BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     fn default() -> Self {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |         Self::new(&FeatureSet::all_enabled())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  | impl BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |     pub fn new(feature_set: &FeatureSet) -> Self {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         let mut bpf_compute_budget =
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |         // Original
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |             max_units: 100_000,
 | 
					
						
							|  |  |  |             log_units: 0,
 | 
					
						
							|  |  |  |             log_64_units: 0,
 | 
					
						
							|  |  |  |             create_program_address_units: 0,
 | 
					
						
							|  |  |  |             invoke_units: 0,
 | 
					
						
							|  |  |  |             max_invoke_depth: 1,
 | 
					
						
							| 
									
										
										
										
											2020-09-29 23:29:20 -07:00
										 |  |  |             sha256_base_cost: 85,
 | 
					
						
							|  |  |  |             sha256_byte_cost: 1,
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |             max_call_depth: 20,
 | 
					
						
							|  |  |  |             stack_frame_size: 4_096,
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |             log_pubkey_units: 0,
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |         };
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         if feature_set.is_active(&bpf_compute_budget_balancing::id()) {
 | 
					
						
							|  |  |  |             bpf_compute_budget = BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |                 max_units: 200_000,
 | 
					
						
							|  |  |  |                 log_units: 100,
 | 
					
						
							|  |  |  |                 log_64_units: 100,
 | 
					
						
							|  |  |  |                 create_program_address_units: 1500,
 | 
					
						
							|  |  |  |                 invoke_units: 1000,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |                 ..bpf_compute_budget
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |             };
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |         if feature_set.is_active(&max_invoke_depth_4::id()) {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |             bpf_compute_budget = BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |                 max_invoke_depth: 4,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |                 ..bpf_compute_budget
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |             };
 | 
					
						
							| 
									
										
										
										
											2020-10-09 18:49:44 +00:00
										 |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |         if feature_set.is_active(&max_program_call_depth_64::id()) {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |             bpf_compute_budget = BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |                 max_call_depth: 64,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |                 ..bpf_compute_budget
 | 
					
						
							| 
									
										
										
										
											2020-10-09 22:08:01 +00:00
										 |  |  |             };
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |         if feature_set.is_active(&pubkey_log_syscall_enabled::id()) {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |             bpf_compute_budget = BpfComputeBudget {
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |                 log_pubkey_units: 100,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |                 ..bpf_compute_budget
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:40:20 +00:00
										 |  |  |             };
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         bpf_compute_budget
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Compute meter
 | 
					
						
							|  |  |  | pub trait ComputeMeter {
 | 
					
						
							|  |  |  |     /// Consume compute units
 | 
					
						
							|  |  |  |     fn consume(&mut self, amount: u64) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  |     /// Get the number of remaining compute units
 | 
					
						
							|  |  |  |     fn get_remaining(&self) -> u64;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Log messages
 | 
					
						
							|  |  |  | pub trait Logger {
 | 
					
						
							|  |  |  |     fn log_enabled(&self) -> bool;
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Log a message.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// Unless explicitly stated, log messages are not considered stable and may change in the
 | 
					
						
							|  |  |  |     /// future as necessary
 | 
					
						
							| 
									
										
										
										
											2020-10-30 10:35:59 +00:00
										 |  |  |     fn log(&self, message: &str);
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  | ///
 | 
					
						
							|  |  |  | /// Stable program log messages
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | /// The format of these log messages should not be modified to avoid breaking downstream consumers
 | 
					
						
							|  |  |  | /// of program logging
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | pub mod stable_log {
 | 
					
						
							|  |  |  |     use super::*;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Log a program invoke.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The general form is:
 | 
					
						
							|  |  |  |     ///     "Program <address> invoke [<depth>]"
 | 
					
						
							|  |  |  |     pub fn program_invoke(
 | 
					
						
							|  |  |  |         logger: &Rc<RefCell<dyn Logger>>,
 | 
					
						
							|  |  |  |         program_id: &Pubkey,
 | 
					
						
							|  |  |  |         invoke_depth: usize,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							|  |  |  |         if let Ok(logger) = logger.try_borrow_mut() {
 | 
					
						
							|  |  |  |             if logger.log_enabled() {
 | 
					
						
							|  |  |  |                 logger.log(&format!("Program {} invoke [{}]", program_id, invoke_depth));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Log a message from the program itself.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The general form is:
 | 
					
						
							|  |  |  |     ///     "Program log: <program-generated output>"
 | 
					
						
							|  |  |  |     /// That is, any program-generated output is guaranteed to be prefixed by "Program log: "
 | 
					
						
							|  |  |  |     pub fn program_log(logger: &Rc<RefCell<dyn Logger>>, message: &str) {
 | 
					
						
							|  |  |  |         if let Ok(logger) = logger.try_borrow_mut() {
 | 
					
						
							|  |  |  |             if logger.log_enabled() {
 | 
					
						
							|  |  |  |                 logger.log(&format!("Program log: {}", message))
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Log successful program execution.
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The general form is:
 | 
					
						
							|  |  |  |     ///     "Program <address> success"
 | 
					
						
							|  |  |  |     pub fn program_success(logger: &Rc<RefCell<dyn Logger>>, program_id: &Pubkey) {
 | 
					
						
							|  |  |  |         if let Ok(logger) = logger.try_borrow_mut() {
 | 
					
						
							|  |  |  |             if logger.log_enabled() {
 | 
					
						
							|  |  |  |                 logger.log(&format!("Program {} success", program_id));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Log program execution failure
 | 
					
						
							|  |  |  |     ///
 | 
					
						
							|  |  |  |     /// The general form is:
 | 
					
						
							|  |  |  |     ///     "Program <address> failed: <program error details>"
 | 
					
						
							|  |  |  |     pub fn program_failure(
 | 
					
						
							|  |  |  |         logger: &Rc<RefCell<dyn Logger>>,
 | 
					
						
							|  |  |  |         program_id: &Pubkey,
 | 
					
						
							|  |  |  |         err: &InstructionError,
 | 
					
						
							|  |  |  |     ) {
 | 
					
						
							|  |  |  |         if let Ok(logger) = logger.try_borrow_mut() {
 | 
					
						
							|  |  |  |             if logger.log_enabled() {
 | 
					
						
							|  |  |  |                 logger.log(&format!("Program {} failed: {}", program_id, err));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  | /// Program executor
 | 
					
						
							| 
									
										
										
										
											2020-10-20 17:20:40 +00:00
										 |  |  | pub trait Executor: Debug + Send + Sync {
 | 
					
						
							| 
									
										
										
										
											2020-09-29 01:36:46 -07:00
										 |  |  |     /// Execute the program
 | 
					
						
							|  |  |  |     fn execute(
 | 
					
						
							|  |  |  |         &self,
 | 
					
						
							|  |  |  |         program_id: &Pubkey,
 | 
					
						
							|  |  |  |         keyed_accounts: &[KeyedAccount],
 | 
					
						
							|  |  |  |         instruction_data: &[u8],
 | 
					
						
							|  |  |  |         invoke_context: &mut dyn InvokeContext,
 | 
					
						
							|  |  |  |     ) -> Result<(), InstructionError>;
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #[derive(Debug, Default, Clone)]
 | 
					
						
							|  |  |  | pub struct MockComputeMeter {
 | 
					
						
							|  |  |  |     pub remaining: u64,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | impl ComputeMeter for MockComputeMeter {
 | 
					
						
							|  |  |  |     fn consume(&mut self, amount: u64) -> Result<(), InstructionError> {
 | 
					
						
							|  |  |  |         let exceeded = self.remaining < amount;
 | 
					
						
							|  |  |  |         self.remaining = self.remaining.saturating_sub(amount);
 | 
					
						
							|  |  |  |         if exceeded {
 | 
					
						
							|  |  |  |             return Err(InstructionError::ComputationalBudgetExceeded);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn get_remaining(&self) -> u64 {
 | 
					
						
							|  |  |  |         self.remaining
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Debug, Default, Clone)]
 | 
					
						
							|  |  |  | pub struct MockLogger {
 | 
					
						
							|  |  |  |     pub log: Rc<RefCell<Vec<String>>>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | impl Logger for MockLogger {
 | 
					
						
							|  |  |  |     fn log_enabled(&self) -> bool {
 | 
					
						
							|  |  |  |         true
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 10:35:59 +00:00
										 |  |  |     fn log(&self, message: &str) {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         self.log.borrow_mut().push(message.to_string());
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct MockInvokeContext {
 | 
					
						
							|  |  |  |     pub key: Pubkey,
 | 
					
						
							|  |  |  |     pub logger: MockLogger,
 | 
					
						
							|  |  |  |     pub bpf_compute_budget: BpfComputeBudget,
 | 
					
						
							|  |  |  |     pub compute_meter: MockComputeMeter,
 | 
					
						
							| 
									
										
										
										
											2020-11-01 05:43:43 +00:00
										 |  |  |     pub programs: Vec<(Pubkey, ProcessInstructionWithContext)>,
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  |     invoke_depth: usize,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | impl Default for MockInvokeContext {
 | 
					
						
							|  |  |  |     fn default() -> Self {
 | 
					
						
							|  |  |  |         MockInvokeContext {
 | 
					
						
							|  |  |  |             key: Pubkey::default(),
 | 
					
						
							|  |  |  |             logger: MockLogger::default(),
 | 
					
						
							|  |  |  |             bpf_compute_budget: BpfComputeBudget::default(),
 | 
					
						
							|  |  |  |             compute_meter: MockComputeMeter {
 | 
					
						
							|  |  |  |                 remaining: std::i64::MAX as u64,
 | 
					
						
							|  |  |  |             },
 | 
					
						
							| 
									
										
										
										
											2020-11-01 05:43:43 +00:00
										 |  |  |             programs: vec![],
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  |             invoke_depth: 0,
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | impl InvokeContext for MockInvokeContext {
 | 
					
						
							|  |  |  |     fn push(&mut self, _key: &Pubkey) -> Result<(), InstructionError> {
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  |         self.invoke_depth += 1;
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-11-12 12:44:37 -08:00
										 |  |  |     fn pop(&mut self) {
 | 
					
						
							|  |  |  |         self.invoke_depth -= 1;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn invoke_depth(&self) -> usize {
 | 
					
						
							|  |  |  |         self.invoke_depth
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |     fn verify_and_update(
 | 
					
						
							|  |  |  |         &mut self,
 | 
					
						
							|  |  |  |         _message: &Message,
 | 
					
						
							|  |  |  |         _instruction: &CompiledInstruction,
 | 
					
						
							|  |  |  |         _accounts: &[Rc<RefCell<Account>>],
 | 
					
						
							|  |  |  |     ) -> Result<(), InstructionError> {
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn get_caller(&self) -> Result<&Pubkey, InstructionError> {
 | 
					
						
							|  |  |  |         Ok(&self.key)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn get_programs(&self) -> &[(Pubkey, ProcessInstructionWithContext)] {
 | 
					
						
							| 
									
										
										
										
											2020-11-01 05:43:43 +00:00
										 |  |  |         &self.programs
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |     }
 | 
					
						
							|  |  |  |     fn get_logger(&self) -> Rc<RefCell<dyn Logger>> {
 | 
					
						
							|  |  |  |         Rc::new(RefCell::new(self.logger.clone()))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn get_bpf_compute_budget(&self) -> &BpfComputeBudget {
 | 
					
						
							|  |  |  |         &self.bpf_compute_budget
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn get_compute_meter(&self) -> Rc<RefCell<dyn ComputeMeter>> {
 | 
					
						
							|  |  |  |         Rc::new(RefCell::new(self.compute_meter.clone()))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-10-30 10:35:59 +00:00
										 |  |  |     fn add_executor(&self, _pubkey: &Pubkey, _executor: Arc<dyn Executor>) {}
 | 
					
						
							|  |  |  |     fn get_executor(&self, _pubkey: &Pubkey) -> Option<Arc<dyn Executor>> {
 | 
					
						
							| 
									
										
										
										
											2020-10-30 07:47:17 +00:00
										 |  |  |         None
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     fn record_instruction(&self, _instruction: &Instruction) {}
 | 
					
						
							|  |  |  |     fn is_feature_active(&self, _feature_id: &Pubkey) -> bool {
 | 
					
						
							|  |  |  |         true
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 |