diff --git a/crates/gateway/src/compilation.rs b/crates/gateway/src/compilation.rs index d52bbd91c..6fabc96a1 100644 --- a/crates/gateway/src/compilation.rs +++ b/crates/gateway/src/compilation.rs @@ -8,12 +8,13 @@ use cairo_lang_starknet_classes::casm_contract_class::{ }; use starknet_api::core::CompiledClassHash; use starknet_api::rpc_transaction::RPCDeclareTransaction; +use starknet_api::transaction::Builtin; use starknet_sierra_compile::compile::compile_sierra_to_casm; use starknet_sierra_compile::errors::CompilationUtilError; use starknet_sierra_compile::utils::into_contract_class_for_compilation; use crate::errors::{GatewayError, GatewayResult}; -use crate::utils::is_subsequence; +use crate::utils::{is_subsequence, IntoEnumIteratorExt}; #[cfg(test)] #[path = "compilation_test.rs"] @@ -60,16 +61,33 @@ pub fn compile_contract_class(declare_tx: &RPCDeclareTransaction) -> GatewayResu Ok(class_info) } +// TODO(Arni): Consider moving into Starknet-api. +// List of supported builtins. +// This is an explicit function so that it is explicitly desiced which builtins are supported. +// If new builtins are added, they should be added here. +fn is_supported_builtin(builtin: &Builtin) -> bool { + match builtin { + Builtin::RangeCheck + | Builtin::Pedersen + | Builtin::Poseidon + | Builtin::EcOp + | Builtin::Ecdsa + | Builtin::Bitwise + | Builtin::SegmentArena => true, + Builtin::Keccak => false, + } +} + // TODO(Arni): Add to a config. // TODO(Arni): Use the Builtin enum from Starknet-api, and explicitly tag each builtin as supported // or unsupported so that the compiler would alert us on new builtins. fn supported_builtins() -> &'static Vec { static SUPPORTED_BUILTINS: OnceLock> = OnceLock::new(); SUPPORTED_BUILTINS.get_or_init(|| { - // The OS expects this order for the builtins. - const SUPPORTED_BUILTIN_NAMES: [&str; 7] = - ["pedersen", "range_check", "ecdsa", "bitwise", "ec_op", "poseidon", "segment_arena"]; - SUPPORTED_BUILTIN_NAMES.iter().map(|builtin| builtin.to_string()).collect::>() + Builtin::iter() + .filter(is_supported_builtin) + .map(|builtin| builtin.name().to_string()) + .collect::>() }) } diff --git a/crates/gateway/src/utils.rs b/crates/gateway/src/utils.rs index 6cebd08d4..4c50e1dba 100644 --- a/crates/gateway/src/utils.rs +++ b/crates/gateway/src/utils.rs @@ -10,8 +10,9 @@ use starknet_api::rpc_transaction::{ RPCDeclareTransaction, RPCDeployAccountTransaction, RPCInvokeTransaction, RPCTransaction, }; use starknet_api::transaction::{ - DeclareTransaction, DeclareTransactionV3, DeployAccountTransaction, DeployAccountTransactionV3, - InvokeTransaction, InvokeTransactionV3, Tip, TransactionHash, TransactionHasher, + Builtin, DeclareTransaction, DeclareTransactionV3, DeployAccountTransaction, + DeployAccountTransactionV3, InvokeTransaction, InvokeTransactionV3, Tip, TransactionHash, + TransactionHasher, }; use starknet_mempool_types::mempool_types::ThinTransaction; @@ -176,3 +177,28 @@ pub fn is_subsequence(subsequence: &[T], sequence: &[T]) -> bool { offset == subsequence.len() } + +// TODO(Arni): Remove the trait IntoEnumIteratorExt once EnumIter is implemented in starknet API. +pub trait IntoEnumIteratorExt { + /// Returns an iterator over all the builtins in the order the Starknet OS expects. + fn iter() -> impl Iterator; +} + +// TODO(Arni): When the trait IntoEnumIteratorExt is removed, Make sure the order is maintained or +// address the different order. +impl IntoEnumIteratorExt for Builtin { + fn iter() -> impl Iterator { + // The OS expects this order for the builtins. + vec![ + Builtin::Pedersen, + Builtin::RangeCheck, + Builtin::Ecdsa, + Builtin::Bitwise, + Builtin::EcOp, + Builtin::Poseidon, + Builtin::SegmentArena, + Builtin::Keccak, + ] + .into_iter() + } +} diff --git a/crates/gateway/src/utils_test.rs b/crates/gateway/src/utils_test.rs index 7b9d2fdb9..3e2e096c3 100644 --- a/crates/gateway/src/utils_test.rs +++ b/crates/gateway/src/utils_test.rs @@ -1,7 +1,8 @@ use pretty_assertions::assert_eq; use rstest::rstest; +use starknet_api::transaction::Builtin; -use crate::utils::is_subsequence; +use crate::utils::{is_subsequence, IntoEnumIteratorExt}; #[rstest] #[case::empty( @@ -76,3 +77,20 @@ fn test_is_subsequence( ) { assert_eq!(is_subsequence(subsequence, sequence), expected_result); } + +#[test] +fn test_os_order_iter() { + let expected_order = vec![ + Builtin::Pedersen, + Builtin::RangeCheck, + Builtin::Ecdsa, + Builtin::Bitwise, + Builtin::EcOp, + Builtin::Poseidon, + Builtin::SegmentArena, + Builtin::Keccak, + ]; + + let actual_order = Builtin::iter().collect::>(); + assert_eq!(actual_order, expected_order); +}