#![allow(clippy::arithmetic_side_effects)]
#![cfg_attr(docsrs, feature(doc_cfg))]
#![no_std]
#[cfg(feature = "borsh")]
use borsh::io::Error as BorshIoError;
use core::{convert::TryFrom, fmt};
#[cfg(feature = "serde")]
use serde_derive::{Deserialize, Serialize};
pub type ProgramResult = core::result::Result<(), ProgramError>;
pub const BUILTIN_BIT_SHIFT: usize = 32;
macro_rules! to_builtin {
($error:expr) => {
($error as u64) << BUILTIN_BIT_SHIFT
};
}
pub const CUSTOM_ZERO: u64 = to_builtin!(1);
pub const INVALID_ARGUMENT: u64 = to_builtin!(2);
pub const INVALID_INSTRUCTION_DATA: u64 = to_builtin!(3);
pub const INVALID_ACCOUNT_DATA: u64 = to_builtin!(4);
pub const ACCOUNT_DATA_TOO_SMALL: u64 = to_builtin!(5);
pub const INSUFFICIENT_FUNDS: u64 = to_builtin!(6);
pub const INCORRECT_PROGRAM_ID: u64 = to_builtin!(7);
pub const MISSING_REQUIRED_SIGNATURES: u64 = to_builtin!(8);
pub const ACCOUNT_ALREADY_INITIALIZED: u64 = to_builtin!(9);
pub const UNINITIALIZED_ACCOUNT: u64 = to_builtin!(10);
pub const NOT_ENOUGH_ACCOUNT_KEYS: u64 = to_builtin!(11);
pub const ACCOUNT_BORROW_FAILED: u64 = to_builtin!(12);
pub const MAX_SEED_LENGTH_EXCEEDED: u64 = to_builtin!(13);
pub const INVALID_SEEDS: u64 = to_builtin!(14);
pub const BORSH_IO_ERROR: u64 = to_builtin!(15);
pub const ACCOUNT_NOT_RENT_EXEMPT: u64 = to_builtin!(16);
pub const UNSUPPORTED_SYSVAR: u64 = to_builtin!(17);
pub const ILLEGAL_OWNER: u64 = to_builtin!(18);
pub const MAX_ACCOUNTS_DATA_ALLOCATIONS_EXCEEDED: u64 = to_builtin!(19);
pub const INVALID_ACCOUNT_DATA_REALLOC: u64 = to_builtin!(20);
pub const MAX_INSTRUCTION_TRACE_LENGTH_EXCEEDED: u64 = to_builtin!(21);
pub const BUILTIN_PROGRAMS_MUST_CONSUME_COMPUTE_UNITS: u64 = to_builtin!(22);
pub const INVALID_ACCOUNT_OWNER: u64 = to_builtin!(23);
pub const ARITHMETIC_OVERFLOW: u64 = to_builtin!(24);
pub const IMMUTABLE: u64 = to_builtin!(25);
pub const INCORRECT_AUTHORITY: u64 = to_builtin!(26);
#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum ProgramError {
Custom(u32),
InvalidArgument,
InvalidInstructionData,
InvalidAccountData,
AccountDataTooSmall,
InsufficientFunds,
IncorrectProgramId,
MissingRequiredSignature,
AccountAlreadyInitialized,
UninitializedAccount,
NotEnoughAccountKeys,
AccountBorrowFailed,
MaxSeedLengthExceeded,
InvalidSeeds,
BorshIoError,
AccountNotRentExempt,
UnsupportedSysvar,
IllegalOwner,
MaxAccountsDataAllocationsExceeded,
InvalidRealloc,
MaxInstructionTraceLengthExceeded,
BuiltinProgramsMustConsumeComputeUnits,
InvalidAccountOwner,
ArithmeticOverflow,
Immutable,
IncorrectAuthority,
}
impl core::error::Error for ProgramError {}
impl fmt::Display for ProgramError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
ProgramError::Custom(num) => write!(f,"Custom program error: {num:#x}"),
ProgramError::InvalidArgument
=> f.write_str("The arguments provided to a program instruction were invalid"),
ProgramError::InvalidInstructionData
=> f.write_str("An instruction's data contents was invalid"),
ProgramError::InvalidAccountData
=> f.write_str("An account's data contents was invalid"),
ProgramError::AccountDataTooSmall
=> f.write_str("An account's data was too small"),
ProgramError::InsufficientFunds
=> f.write_str("An account's balance was too small to complete the instruction"),
ProgramError::IncorrectProgramId
=> f.write_str("The account did not have the expected program id"),
ProgramError::MissingRequiredSignature
=> f.write_str("A signature was required but not found"),
ProgramError::AccountAlreadyInitialized
=> f.write_str("An initialize instruction was sent to an account that has already been initialized"),
ProgramError::UninitializedAccount
=> f.write_str("An attempt to operate on an account that hasn't been initialized"),
ProgramError::NotEnoughAccountKeys
=> f.write_str("The instruction expected additional account keys"),
ProgramError::AccountBorrowFailed
=> f.write_str("Failed to borrow a reference to account data, already borrowed"),
ProgramError::MaxSeedLengthExceeded
=> f.write_str("Length of the seed is too long for address generation"),
ProgramError::InvalidSeeds
=> f.write_str("Provided seeds do not result in a valid address"),
ProgramError::BorshIoError => f.write_str("IO Error"),
ProgramError::AccountNotRentExempt
=> f.write_str("An account does not have enough lamports to be rent-exempt"),
ProgramError::UnsupportedSysvar
=> f.write_str("Unsupported sysvar"),
ProgramError::IllegalOwner
=> f.write_str("Provided owner is not allowed"),
ProgramError::MaxAccountsDataAllocationsExceeded
=> f.write_str("Accounts data allocations exceeded the maximum allowed per transaction"),
ProgramError::InvalidRealloc
=> f.write_str("Account data reallocation was invalid"),
ProgramError::MaxInstructionTraceLengthExceeded
=> f.write_str("Instruction trace length exceeded the maximum allowed per transaction"),
ProgramError::BuiltinProgramsMustConsumeComputeUnits
=> f.write_str("Builtin programs must consume compute units"),
ProgramError::InvalidAccountOwner
=> f.write_str("Invalid account owner"),
ProgramError::ArithmeticOverflow
=> f.write_str("Program arithmetic overflowed"),
ProgramError::Immutable
=> f.write_str("Account is immutable"),
ProgramError::IncorrectAuthority
=> f.write_str("Incorrect authority provided"),
}
}
}
pub trait ToStr {
fn to_str(&self) -> &'static str;
}
impl ProgramError {
pub fn to_str<E>(&self) -> &'static str
where
E: 'static + ToStr + TryFrom<u32>,
{
match self {
Self::Custom(error) => {
if let Ok(custom_error) = E::try_from(*error) {
custom_error.to_str()
} else {
"Error: Unknown"
}
}
Self::InvalidArgument => "Error: InvalidArgument",
Self::InvalidInstructionData => "Error: InvalidInstructionData",
Self::InvalidAccountData => "Error: InvalidAccountData",
Self::AccountDataTooSmall => "Error: AccountDataTooSmall",
Self::InsufficientFunds => "Error: InsufficientFunds",
Self::IncorrectProgramId => "Error: IncorrectProgramId",
Self::MissingRequiredSignature => "Error: MissingRequiredSignature",
Self::AccountAlreadyInitialized => "Error: AccountAlreadyInitialized",
Self::UninitializedAccount => "Error: UninitializedAccount",
Self::NotEnoughAccountKeys => "Error: NotEnoughAccountKeys",
Self::AccountBorrowFailed => "Error: AccountBorrowFailed",
Self::MaxSeedLengthExceeded => "Error: MaxSeedLengthExceeded",
Self::InvalidSeeds => "Error: InvalidSeeds",
Self::BorshIoError => "Error: BorshIoError",
Self::AccountNotRentExempt => "Error: AccountNotRentExempt",
Self::UnsupportedSysvar => "Error: UnsupportedSysvar",
Self::IllegalOwner => "Error: IllegalOwner",
Self::MaxAccountsDataAllocationsExceeded => "Error: MaxAccountsDataAllocationsExceeded",
Self::InvalidRealloc => "Error: InvalidRealloc",
Self::MaxInstructionTraceLengthExceeded => "Error: MaxInstructionTraceLengthExceeded",
Self::BuiltinProgramsMustConsumeComputeUnits => {
"Error: BuiltinProgramsMustConsumeComputeUnits"
}
Self::InvalidAccountOwner => "Error: InvalidAccountOwner",
Self::ArithmeticOverflow => "Error: ArithmeticOverflow",
Self::Immutable => "Error: Immutable",
Self::IncorrectAuthority => "Error: IncorrectAuthority",
}
}
}
impl From<ProgramError> for u64 {
fn from(error: ProgramError) -> Self {
match error {
ProgramError::InvalidArgument => INVALID_ARGUMENT,
ProgramError::InvalidInstructionData => INVALID_INSTRUCTION_DATA,
ProgramError::InvalidAccountData => INVALID_ACCOUNT_DATA,
ProgramError::AccountDataTooSmall => ACCOUNT_DATA_TOO_SMALL,
ProgramError::InsufficientFunds => INSUFFICIENT_FUNDS,
ProgramError::IncorrectProgramId => INCORRECT_PROGRAM_ID,
ProgramError::MissingRequiredSignature => MISSING_REQUIRED_SIGNATURES,
ProgramError::AccountAlreadyInitialized => ACCOUNT_ALREADY_INITIALIZED,
ProgramError::UninitializedAccount => UNINITIALIZED_ACCOUNT,
ProgramError::NotEnoughAccountKeys => NOT_ENOUGH_ACCOUNT_KEYS,
ProgramError::AccountBorrowFailed => ACCOUNT_BORROW_FAILED,
ProgramError::MaxSeedLengthExceeded => MAX_SEED_LENGTH_EXCEEDED,
ProgramError::InvalidSeeds => INVALID_SEEDS,
ProgramError::BorshIoError => BORSH_IO_ERROR,
ProgramError::AccountNotRentExempt => ACCOUNT_NOT_RENT_EXEMPT,
ProgramError::UnsupportedSysvar => UNSUPPORTED_SYSVAR,
ProgramError::IllegalOwner => ILLEGAL_OWNER,
ProgramError::MaxAccountsDataAllocationsExceeded => {
MAX_ACCOUNTS_DATA_ALLOCATIONS_EXCEEDED
}
ProgramError::InvalidRealloc => INVALID_ACCOUNT_DATA_REALLOC,
ProgramError::MaxInstructionTraceLengthExceeded => {
MAX_INSTRUCTION_TRACE_LENGTH_EXCEEDED
}
ProgramError::BuiltinProgramsMustConsumeComputeUnits => {
BUILTIN_PROGRAMS_MUST_CONSUME_COMPUTE_UNITS
}
ProgramError::InvalidAccountOwner => INVALID_ACCOUNT_OWNER,
ProgramError::ArithmeticOverflow => ARITHMETIC_OVERFLOW,
ProgramError::Immutable => IMMUTABLE,
ProgramError::IncorrectAuthority => INCORRECT_AUTHORITY,
ProgramError::Custom(error) => {
if error == 0 {
CUSTOM_ZERO
} else {
error as u64
}
}
}
}
}
impl From<u64> for ProgramError {
fn from(error: u64) -> Self {
match error {
CUSTOM_ZERO => Self::Custom(0),
INVALID_ARGUMENT => Self::InvalidArgument,
INVALID_INSTRUCTION_DATA => Self::InvalidInstructionData,
INVALID_ACCOUNT_DATA => Self::InvalidAccountData,
ACCOUNT_DATA_TOO_SMALL => Self::AccountDataTooSmall,
INSUFFICIENT_FUNDS => Self::InsufficientFunds,
INCORRECT_PROGRAM_ID => Self::IncorrectProgramId,
MISSING_REQUIRED_SIGNATURES => Self::MissingRequiredSignature,
ACCOUNT_ALREADY_INITIALIZED => Self::AccountAlreadyInitialized,
UNINITIALIZED_ACCOUNT => Self::UninitializedAccount,
NOT_ENOUGH_ACCOUNT_KEYS => Self::NotEnoughAccountKeys,
ACCOUNT_BORROW_FAILED => Self::AccountBorrowFailed,
MAX_SEED_LENGTH_EXCEEDED => Self::MaxSeedLengthExceeded,
INVALID_SEEDS => Self::InvalidSeeds,
BORSH_IO_ERROR => Self::BorshIoError,
ACCOUNT_NOT_RENT_EXEMPT => Self::AccountNotRentExempt,
UNSUPPORTED_SYSVAR => Self::UnsupportedSysvar,
ILLEGAL_OWNER => Self::IllegalOwner,
MAX_ACCOUNTS_DATA_ALLOCATIONS_EXCEEDED => Self::MaxAccountsDataAllocationsExceeded,
INVALID_ACCOUNT_DATA_REALLOC => Self::InvalidRealloc,
MAX_INSTRUCTION_TRACE_LENGTH_EXCEEDED => Self::MaxInstructionTraceLengthExceeded,
BUILTIN_PROGRAMS_MUST_CONSUME_COMPUTE_UNITS => {
Self::BuiltinProgramsMustConsumeComputeUnits
}
INVALID_ACCOUNT_OWNER => Self::InvalidAccountOwner,
ARITHMETIC_OVERFLOW => Self::ArithmeticOverflow,
IMMUTABLE => Self::Immutable,
INCORRECT_AUTHORITY => Self::IncorrectAuthority,
_ => Self::Custom(error as u32),
}
}
}
#[cfg(feature = "borsh")]
impl From<BorshIoError> for ProgramError {
fn from(_error: BorshIoError) -> Self {
Self::BorshIoError
}
}