use core::fmt::Debug;
use wiring::{
result::{HaltReason, OutOfGasError, SuccessReason},
HaltReasonTrait,
};
#[repr(u8)]
#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, Hash)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub enum InstructionResult {
#[default]
Continue = 0x00,
Stop,
Return,
SelfDestruct,
ReturnContract,
Revert = 0x10,
CallTooDeep,
OutOfFunds,
CreateInitCodeStartingEF00,
InvalidEOFInitCode,
InvalidExtDelegateCallTarget,
CallOrCreate = 0x20,
OutOfGas = 0x50,
MemoryOOG,
MemoryLimitOOG,
PrecompileOOG,
InvalidOperandOOG,
ReentrancySentryOOG,
OpcodeNotFound,
CallNotAllowedInsideStatic,
StateChangeDuringStaticCall,
InvalidFEOpcode,
InvalidJump,
NotActivated,
StackUnderflow,
StackOverflow,
OutOfOffset,
CreateCollision,
OverflowPayment,
PrecompileError,
NonceOverflow,
CreateContractSizeLimit,
CreateContractStartingWithEF,
CreateInitCodeSizeLimit,
FatalExternalError,
ReturnContractInNotInitEOF,
EOFOpcodeDisabledInLegacy,
EOFFunctionStackOverflow,
EofAuxDataOverflow,
EofAuxDataTooSmall,
InvalidEXTCALLTarget,
}
impl From<SuccessReason> for InstructionResult {
fn from(value: SuccessReason) -> Self {
match value {
SuccessReason::Return => InstructionResult::Return,
SuccessReason::Stop => InstructionResult::Stop,
SuccessReason::SelfDestruct => InstructionResult::SelfDestruct,
SuccessReason::EofReturnContract => InstructionResult::ReturnContract,
}
}
}
impl From<HaltReason> for InstructionResult {
fn from(value: HaltReason) -> Self {
match value {
HaltReason::OutOfGas(error) => match error {
OutOfGasError::Basic => Self::OutOfGas,
OutOfGasError::InvalidOperand => Self::InvalidOperandOOG,
OutOfGasError::Memory => Self::MemoryOOG,
OutOfGasError::MemoryLimit => Self::MemoryLimitOOG,
OutOfGasError::Precompile => Self::PrecompileOOG,
OutOfGasError::ReentrancySentry => Self::ReentrancySentryOOG,
},
HaltReason::OpcodeNotFound => Self::OpcodeNotFound,
HaltReason::InvalidFEOpcode => Self::InvalidFEOpcode,
HaltReason::InvalidJump => Self::InvalidJump,
HaltReason::NotActivated => Self::NotActivated,
HaltReason::StackOverflow => Self::StackOverflow,
HaltReason::StackUnderflow => Self::StackUnderflow,
HaltReason::OutOfOffset => Self::OutOfOffset,
HaltReason::CreateCollision => Self::CreateCollision,
HaltReason::PrecompileError => Self::PrecompileError,
HaltReason::NonceOverflow => Self::NonceOverflow,
HaltReason::CreateContractSizeLimit => Self::CreateContractSizeLimit,
HaltReason::CreateContractStartingWithEF => Self::CreateContractStartingWithEF,
HaltReason::CreateInitCodeSizeLimit => Self::CreateInitCodeSizeLimit,
HaltReason::OverflowPayment => Self::OverflowPayment,
HaltReason::StateChangeDuringStaticCall => Self::StateChangeDuringStaticCall,
HaltReason::CallNotAllowedInsideStatic => Self::CallNotAllowedInsideStatic,
HaltReason::OutOfFunds => Self::OutOfFunds,
HaltReason::CallTooDeep => Self::CallTooDeep,
HaltReason::EofAuxDataOverflow => Self::EofAuxDataOverflow,
HaltReason::EofAuxDataTooSmall => Self::EofAuxDataTooSmall,
HaltReason::EOFFunctionStackOverflow => Self::EOFFunctionStackOverflow,
HaltReason::InvalidEXTCALLTarget => Self::InvalidEXTCALLTarget,
}
}
}
#[macro_export]
macro_rules! return_ok {
() => {
$crate::InstructionResult::Continue
| $crate::InstructionResult::Stop
| $crate::InstructionResult::Return
| $crate::InstructionResult::SelfDestruct
| $crate::InstructionResult::ReturnContract
};
}
#[macro_export]
macro_rules! return_revert {
() => {
$crate::InstructionResult::Revert
| $crate::InstructionResult::CallTooDeep
| $crate::InstructionResult::OutOfFunds
| $crate::InstructionResult::InvalidEOFInitCode
| $crate::InstructionResult::CreateInitCodeStartingEF00
| $crate::InstructionResult::InvalidExtDelegateCallTarget
};
}
#[macro_export]
macro_rules! return_error {
() => {
$crate::InstructionResult::OutOfGas
| $crate::InstructionResult::MemoryOOG
| $crate::InstructionResult::MemoryLimitOOG
| $crate::InstructionResult::PrecompileOOG
| $crate::InstructionResult::InvalidOperandOOG
| $crate::InstructionResult::ReentrancySentryOOG
| $crate::InstructionResult::OpcodeNotFound
| $crate::InstructionResult::CallNotAllowedInsideStatic
| $crate::InstructionResult::StateChangeDuringStaticCall
| $crate::InstructionResult::InvalidFEOpcode
| $crate::InstructionResult::InvalidJump
| $crate::InstructionResult::NotActivated
| $crate::InstructionResult::StackUnderflow
| $crate::InstructionResult::StackOverflow
| $crate::InstructionResult::OutOfOffset
| $crate::InstructionResult::CreateCollision
| $crate::InstructionResult::OverflowPayment
| $crate::InstructionResult::PrecompileError
| $crate::InstructionResult::NonceOverflow
| $crate::InstructionResult::CreateContractSizeLimit
| $crate::InstructionResult::CreateContractStartingWithEF
| $crate::InstructionResult::CreateInitCodeSizeLimit
| $crate::InstructionResult::FatalExternalError
| $crate::InstructionResult::ReturnContractInNotInitEOF
| $crate::InstructionResult::EOFOpcodeDisabledInLegacy
| $crate::InstructionResult::EOFFunctionStackOverflow
| $crate::InstructionResult::EofAuxDataTooSmall
| $crate::InstructionResult::EofAuxDataOverflow
| $crate::InstructionResult::InvalidEXTCALLTarget
};
}
impl InstructionResult {
#[inline]
pub const fn is_ok(self) -> bool {
matches!(self, crate::return_ok!())
}
#[inline]
pub const fn is_revert(self) -> bool {
matches!(self, crate::return_revert!())
}
#[inline]
pub const fn is_error(self) -> bool {
matches!(self, return_error!())
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub enum InternalResult {
InternalContinue,
InternalCallOrCreate,
CreateInitCodeStartingEF00,
InvalidExtDelegateCallTarget,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum SuccessOrHalt<HaltReasonT: HaltReasonTrait> {
Success(SuccessReason),
Revert,
Halt(HaltReasonT),
FatalExternalError,
Internal(InternalResult),
}
impl<HaltReasonT: HaltReasonTrait> SuccessOrHalt<HaltReasonT> {
#[inline]
pub fn is_success(self) -> bool {
matches!(self, SuccessOrHalt::Success(_))
}
#[inline]
pub fn to_success(self) -> Option<SuccessReason> {
match self {
SuccessOrHalt::Success(reason) => Some(reason),
_ => None,
}
}
#[inline]
pub fn is_revert(self) -> bool {
matches!(self, SuccessOrHalt::Revert)
}
#[inline]
pub fn is_halt(self) -> bool {
matches!(self, SuccessOrHalt::Halt(_))
}
#[inline]
pub fn to_halt(self) -> Option<HaltReasonT> {
match self {
SuccessOrHalt::Halt(reason) => Some(reason),
_ => None,
}
}
}
impl<HaltReasonT: HaltReasonTrait> From<InstructionResult> for SuccessOrHalt<HaltReasonT> {
fn from(result: InstructionResult) -> Self {
match result {
InstructionResult::Continue => Self::Internal(InternalResult::InternalContinue), InstructionResult::Stop => Self::Success(SuccessReason::Stop),
InstructionResult::Return => Self::Success(SuccessReason::Return),
InstructionResult::SelfDestruct => Self::Success(SuccessReason::SelfDestruct),
InstructionResult::Revert => Self::Revert,
InstructionResult::CreateInitCodeStartingEF00 => Self::Revert,
InstructionResult::CallOrCreate => Self::Internal(InternalResult::InternalCallOrCreate), InstructionResult::CallTooDeep => Self::Halt(HaltReason::CallTooDeep.into()), InstructionResult::OutOfFunds => Self::Halt(HaltReason::OutOfFunds.into()), InstructionResult::OutOfGas => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::Basic).into())
}
InstructionResult::MemoryLimitOOG => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::MemoryLimit).into())
}
InstructionResult::MemoryOOG => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::Memory).into())
}
InstructionResult::PrecompileOOG => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::Precompile).into())
}
InstructionResult::InvalidOperandOOG => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::InvalidOperand).into())
}
InstructionResult::ReentrancySentryOOG => {
Self::Halt(HaltReason::OutOfGas(OutOfGasError::ReentrancySentry).into())
}
InstructionResult::OpcodeNotFound | InstructionResult::ReturnContractInNotInitEOF => {
Self::Halt(HaltReason::OpcodeNotFound.into())
}
InstructionResult::CallNotAllowedInsideStatic => {
Self::Halt(HaltReason::CallNotAllowedInsideStatic.into())
} InstructionResult::StateChangeDuringStaticCall => {
Self::Halt(HaltReason::StateChangeDuringStaticCall.into())
}
InstructionResult::InvalidFEOpcode => Self::Halt(HaltReason::InvalidFEOpcode.into()),
InstructionResult::InvalidJump => Self::Halt(HaltReason::InvalidJump.into()),
InstructionResult::NotActivated => Self::Halt(HaltReason::NotActivated.into()),
InstructionResult::StackUnderflow => Self::Halt(HaltReason::StackUnderflow.into()),
InstructionResult::StackOverflow => Self::Halt(HaltReason::StackOverflow.into()),
InstructionResult::OutOfOffset => Self::Halt(HaltReason::OutOfOffset.into()),
InstructionResult::CreateCollision => Self::Halt(HaltReason::CreateCollision.into()),
InstructionResult::OverflowPayment => Self::Halt(HaltReason::OverflowPayment.into()), InstructionResult::PrecompileError => Self::Halt(HaltReason::PrecompileError.into()),
InstructionResult::NonceOverflow => Self::Halt(HaltReason::NonceOverflow.into()),
InstructionResult::CreateContractSizeLimit
| InstructionResult::CreateContractStartingWithEF => {
Self::Halt(HaltReason::CreateContractSizeLimit.into())
}
InstructionResult::CreateInitCodeSizeLimit => {
Self::Halt(HaltReason::CreateInitCodeSizeLimit.into())
}
InstructionResult::InvalidEOFInitCode => Self::Revert,
InstructionResult::FatalExternalError => Self::FatalExternalError,
InstructionResult::EOFOpcodeDisabledInLegacy => {
Self::Halt(HaltReason::OpcodeNotFound.into())
}
InstructionResult::EOFFunctionStackOverflow => {
Self::Halt(HaltReason::EOFFunctionStackOverflow.into())
}
InstructionResult::ReturnContract => Self::Success(SuccessReason::EofReturnContract),
InstructionResult::EofAuxDataOverflow => {
Self::Halt(HaltReason::EofAuxDataOverflow.into())
}
InstructionResult::EofAuxDataTooSmall => {
Self::Halt(HaltReason::EofAuxDataTooSmall.into())
}
InstructionResult::InvalidEXTCALLTarget => {
Self::Halt(HaltReason::InvalidEXTCALLTarget.into())
}
InstructionResult::InvalidExtDelegateCallTarget => {
Self::Internal(InternalResult::InvalidExtDelegateCallTarget)
}
}
}
}
#[cfg(test)]
mod tests {
use crate::InstructionResult;
#[test]
fn all_results_are_covered() {
match InstructionResult::Continue {
return_error!() => {}
return_revert!() => {}
return_ok!() => {}
InstructionResult::CallOrCreate => {}
}
}
#[test]
fn test_results() {
let ok_results = vec![
InstructionResult::Continue,
InstructionResult::Stop,
InstructionResult::Return,
InstructionResult::SelfDestruct,
];
for result in ok_results {
assert!(result.is_ok());
assert!(!result.is_revert());
assert!(!result.is_error());
}
let revert_results = vec![
InstructionResult::Revert,
InstructionResult::CallTooDeep,
InstructionResult::OutOfFunds,
];
for result in revert_results {
assert!(!result.is_ok());
assert!(result.is_revert());
assert!(!result.is_error());
}
let error_results = vec![
InstructionResult::OutOfGas,
InstructionResult::MemoryOOG,
InstructionResult::MemoryLimitOOG,
InstructionResult::PrecompileOOG,
InstructionResult::InvalidOperandOOG,
InstructionResult::OpcodeNotFound,
InstructionResult::CallNotAllowedInsideStatic,
InstructionResult::StateChangeDuringStaticCall,
InstructionResult::InvalidFEOpcode,
InstructionResult::InvalidJump,
InstructionResult::NotActivated,
InstructionResult::StackUnderflow,
InstructionResult::StackOverflow,
InstructionResult::OutOfOffset,
InstructionResult::CreateCollision,
InstructionResult::OverflowPayment,
InstructionResult::PrecompileError,
InstructionResult::NonceOverflow,
InstructionResult::CreateContractSizeLimit,
InstructionResult::CreateContractStartingWithEF,
InstructionResult::CreateInitCodeSizeLimit,
InstructionResult::FatalExternalError,
];
for result in error_results {
assert!(!result.is_ok());
assert!(!result.is_revert());
assert!(result.is_error());
}
}
}