//! @brief Solana Native program entry point #[cfg(RUSTC_WITH_SPECIALIZATION)] use crate::abi_example::AbiExample; use crate::{ account::Account, account::KeyedAccount, instruction::CompiledInstruction, instruction::InstructionError, message::Message, pubkey::Pubkey, }; use std::{cell::RefCell, rc::Rc, sync::Arc}; // Prototype of a native program entry point /// /// program_id: Program ID of the currently executing program /// keyed_accounts: Accounts passed as part of the instruction /// instruction_data: Instruction data pub type ProgramEntrypoint = unsafe extern "C" fn( program_id: &Pubkey, keyed_accounts: &[KeyedAccount], instruction_data: &[u8], ) -> Result<(), InstructionError>; // 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>; #[rustversion::since(1.46.0)] #[macro_export] macro_rules! declare_name { ($name:ident) => { #[macro_export] macro_rules! $name { () => { // Subtle: // The outer `declare_name!` macro may be expanded in another // crate, causing the macro `$name!` to be defined in that // crate. We want to emit a call to `$crate::id()`, and have // `$crate` be resolved in the crate where `$name!` gets defined, // *not* in this crate (where `declare_name! is defined). // // When a macro_rules! macro gets expanded, any $crate tokens // in its output will be 'marked' with the crate they were expanded // from. This includes nested macros like our macro `$name` - even // though it looks like a separate macro, Rust considers it to be // just another part of the output of `declare_program!`. // // We pass `$name` as the second argument to tell `respan!` to // apply use the `Span` of `$name` when resolving `$crate::id`. // This causes `$crate` to behave as though it was written // at the same location as the `$name` value passed // to `declare_name!` (e.g. the 'foo' in // `declare_name(foo)` // // See the `respan!` macro for more details. // This should use `crate::respan!` once // https://github.com/rust-lang/rust/pull/72121 is merged: // see https://github.com/solana-labs/solana/issues/10933. // For now, we need to use `::solana_sdk` // // `respan!` respans the path `$crate::id`, which we then call (hence the extra // parens) ( stringify!($name).to_string(), ::solana_sdk::respan!($crate::id, $name)(), ) }; } }; } #[rustversion::not(since(1.46.0))] #[macro_export] macro_rules! declare_name { ($name:ident) => { #[macro_export] macro_rules! $name { () => { (stringify!($name).to_string(), $crate::id()) }; } }; } /// Convenience macro to declare a native program /// /// bs58_string: bs58 string representation the program's id /// name: Name of the program /// entrypoint: Program's entrypoint, must be of `type Entrypoint` /// id: Path to the program id access function, used if this macro is not /// called in `src/lib` /// /// # Examples /// /// ``` /// use std::str::FromStr; /// # // wrapper is used so that the macro invocation occurs in the item position /// # // rather than in the statement position which isn't allowed. /// # mod item_wrapper { /// use solana_sdk::account::KeyedAccount; /// use solana_sdk::instruction::InstructionError; /// use solana_sdk::pubkey::Pubkey; /// use solana_sdk::declare_program; /// /// fn my_process_instruction( /// program_id: &Pubkey, /// keyed_accounts: &[KeyedAccount], /// instruction_data: &[u8], /// ) -> Result<(), InstructionError> { /// // Process an instruction /// Ok(()) /// } /// /// declare_program!( /// "My11111111111111111111111111111111111111111", /// solana_my_program, /// my_process_instruction /// ); /// /// # } /// # use solana_sdk::pubkey::Pubkey; /// # use item_wrapper::id; /// let my_id = Pubkey::from_str("My11111111111111111111111111111111111111111").unwrap(); /// assert_eq!(id(), my_id); /// ``` /// ``` /// use std::str::FromStr; /// # // wrapper is used so that the macro invocation occurs in the item position /// # // rather than in the statement position which isn't allowed. /// # mod item_wrapper { /// use solana_sdk::account::KeyedAccount; /// use solana_sdk::instruction::InstructionError; /// use solana_sdk::pubkey::Pubkey; /// use solana_sdk::declare_program; /// /// fn my_process_instruction( /// program_id: &Pubkey, /// keyed_accounts: &[KeyedAccount], /// instruction_data: &[u8], /// ) -> Result<(), InstructionError> { /// // Process an instruction /// Ok(()) /// } /// /// declare_program!( /// solana_sdk::system_program::ID, /// solana_my_program, /// my_process_instruction /// ); /// # } /// /// # use item_wrapper::id; /// assert_eq!(id(), solana_sdk::system_program::ID); /// ``` #[macro_export] macro_rules! declare_program( ($bs58_string:expr, $name:ident, $entrypoint:expr) => ( $crate::declare_id!($bs58_string); $crate::declare_name!($name); #[no_mangle] pub extern "C" fn $name( program_id: &$crate::pubkey::Pubkey, keyed_accounts: &[$crate::account::KeyedAccount], instruction_data: &[u8], ) -> Result<(), $crate::instruction::InstructionError> { $entrypoint(program_id, keyed_accounts, instruction_data) } ) ); pub type ProcessInstruction = fn(&Pubkey, &[KeyedAccount], &[u8]) -> Result<(), InstructionError>; pub type ProcessInstructionWithContext = fn(&Pubkey, &[KeyedAccount], &[u8], &mut dyn InvokeContext) -> Result<(), InstructionError>; // These are just type aliases for work around of Debug-ing above function pointers pub type ErasedProcessInstructionWithContext = fn( &'static Pubkey, &'static [KeyedAccount<'static>], &'static [u8], &'static mut dyn InvokeContext, ) -> Result<(), InstructionError>; pub type ErasedProcessInstruction = fn( &'static Pubkey, &'static [KeyedAccount<'static>], &'static [u8], ) -> 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); /// Verify and update PreAccount state based on program execution fn verify_and_update( &mut self, message: &Message, instruction: &CompiledInstruction, accounts: &[Rc>], ) -> 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 fn get_programs(&self) -> &[(Pubkey, ProcessInstruction)]; /// Get this invocation's logger fn get_logger(&self) -> Rc>; /// Are cross program invocations supported fn is_cross_program_supported(&self) -> bool; /// Get this invocation's compute budget fn get_compute_budget(&self) -> ComputeBudget; /// Get this invocation's compute meter fn get_compute_meter(&self) -> Rc>; /// Loaders may need to do work in order to execute a program. Cache /// the work that can be re-used across executions fn add_executor(&mut self, pubkey: &Pubkey, executor: Arc); /// Get the completed loader work that can be re-used across executions fn get_executor(&mut self, pubkey: &Pubkey) -> Option>; } #[derive(Clone, Copy, Debug)] pub struct ComputeBudget { /// 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, /// Number of compute units consumed by an invoke call (not including the cost incured by /// the called program) pub invoke_units: u64, /// Maximum cross-program invocation depth allowed including the orignal caller pub max_invoke_depth: usize, } impl Default for ComputeBudget { fn default() -> Self { // Tuned for ~1ms ComputeBudget { max_units: 200_000, log_units: 100, log_64_units: 100, create_program_address_units: 1500, invoke_units: 1000, max_invoke_depth: 2, } } } /// 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; /// Log a message fn log(&mut self, message: &str); } /// Program executor pub trait Executor: Send + Sync { /// Execute the program fn execute( &self, program_id: &Pubkey, keyed_accounts: &[KeyedAccount], instruction_data: &[u8], invoke_context: &mut dyn InvokeContext, ) -> Result<(), InstructionError>; } #[cfg(RUSTC_WITH_SPECIALIZATION)] impl AbiExample for Arc { fn example() -> Self { struct ExampleExecutor {} impl Executor for ExampleExecutor { fn execute( &self, _program_id: &Pubkey, _keyed_accounts: &[KeyedAccount], _instruction_data: &[u8], _invoke_context: &mut dyn InvokeContext, ) -> std::result::Result<(), InstructionError> { Ok(()) } } Arc::new(ExampleExecutor {}) } }