solana/sdk/src/entrypoint.rs

121 lines
3.6 KiB
Rust
Raw Normal View History

2019-05-21 13:39:27 -07:00
//! @brief Solana Rust-based BPF program entrypoint and its parameter types
2019-09-06 17:32:14 -07:00
#![cfg(feature = "program")]
2019-06-20 16:07:12 -07:00
extern crate alloc;
2019-05-21 13:39:27 -07:00
use crate::{
account_info::{AccountInfo, AccountInfoMut},
pubkey::Pubkey,
};
2019-06-20 16:07:12 -07:00
use alloc::vec::Vec;
2019-05-21 13:39:27 -07:00
use core::mem::size_of;
use core::slice::{from_raw_parts, from_raw_parts_mut};
use std::cell::RefCell;
use std::rc::Rc;
2019-05-21 13:39:27 -07:00
/// User implemented program entrypoint
///
/// program_id: Program ID of the currently executing program
/// accounts: Accounts passed as part of the instruction
/// data: Instruction data
pub type ProcessInstruction =
fn(program_id: &Pubkey, accounts: &mut [AccountInfo], data: &[u8]) -> u32;
2019-05-21 13:39:27 -07:00
/// Programs indicate success with a return value of 0
pub const SUCCESS: u32 = 0;
2019-05-21 13:39:27 -07:00
/// Declare entrypoint of the program.
///
/// Deserialize the program input parameters and call
/// the user defined `ProcessInstruction`. Users must call
2019-05-21 13:39:27 -07:00
/// this function otherwise an entrypoint for
/// their program will not be created.
#[macro_export]
macro_rules! entrypoint {
($process_instruction:ident) => {
2020-01-03 09:14:28 -08:00
/// # Safety
2019-05-21 13:39:27 -07:00
#[no_mangle]
pub unsafe extern "C" fn entrypoint(input: *mut u8) -> u32 {
2019-09-06 17:32:14 -07:00
unsafe {
let (program_id, mut accounts, data) = $crate::entrypoint::deserialize(input);
$process_instruction(&program_id, &mut accounts, &data)
2019-05-21 13:39:27 -07:00
}
}
};
}
/// Deserialize the input parameters
///
/// # Safety
2019-05-21 13:39:27 -07:00
#[allow(clippy::type_complexity)]
2019-09-06 17:32:14 -07:00
pub unsafe fn deserialize<'a>(input: *mut u8) -> (&'a Pubkey, Vec<AccountInfo<'a>>, &'a [u8]) {
2019-05-21 13:39:27 -07:00
let mut offset: usize = 0;
2019-09-06 17:32:14 -07:00
// Number of accounts present
2019-05-21 13:39:27 -07:00
#[allow(clippy::cast_ptr_alignment)]
2019-09-06 17:32:14 -07:00
let num_accounts = *(input.add(offset) as *const u64) as usize;
2019-06-20 16:07:12 -07:00
offset += size_of::<u64>();
2019-05-21 13:39:27 -07:00
2019-09-06 17:32:14 -07:00
// Account Infos
2019-05-21 13:39:27 -07:00
2019-09-06 17:32:14 -07:00
let mut accounts = Vec::with_capacity(num_accounts);
for _ in 0..num_accounts {
let dup_info = *(input.add(offset) as *const u8) as usize;
offset += size_of::<u8>();
if dup_info == 0 {
let is_signer = {
#[allow(clippy::cast_ptr_alignment)]
let is_signer_val = *(input.add(offset) as *const u64);
(is_signer_val != 0)
};
offset += size_of::<u64>();
let key: &Pubkey = &*(input.add(offset) as *const Pubkey);
offset += size_of::<Pubkey>();
#[allow(clippy::cast_ptr_alignment)]
let lamports = &mut *(input.add(offset) as *mut u64);
offset += size_of::<u64>();
2019-05-21 13:39:27 -07:00
#[allow(clippy::cast_ptr_alignment)]
let data_length = *(input.add(offset) as *const u64) as usize;
offset += size_of::<u64>();
let data = { from_raw_parts_mut(input.add(offset), data_length) };
offset += data_length;
let owner: &Pubkey = &*(input.add(offset) as *const Pubkey);
offset += size_of::<Pubkey>();
let m = Rc::new(RefCell::new(AccountInfoMut { lamports, data }));
accounts.push(AccountInfo {
key,
is_signer,
m,
owner,
});
} else {
// Duplicate account, clone the original
accounts.push(accounts[dup_info].clone());
}
2019-05-21 13:39:27 -07:00
}
// Instruction data
#[allow(clippy::cast_ptr_alignment)]
let data_length = *(input.add(offset) as *const u64) as usize;
offset += size_of::<u64>();
let data = { from_raw_parts(input.add(offset), data_length) };
offset += data_length;
2019-06-20 16:07:12 -07:00
// Program Id
2019-05-21 13:39:27 -07:00
let program_id: &Pubkey = &*(input.add(offset) as *const Pubkey);
2019-05-21 13:39:27 -07:00
2019-09-06 17:32:14 -07:00
(program_id, accounts, data)
2019-05-21 13:39:27 -07:00
}