diff --git a/Cargo.lock b/Cargo.lock index 481e0e01881..2c70215ac6a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1518,6 +1518,7 @@ dependencies = [ "sha2", "starknet-types-core", "starknet_api", + "starknet_sierra_compile", "strum 0.25.0", "strum_macros 0.25.3", "tempfile", diff --git a/crates/blockifier/Cargo.toml b/crates/blockifier/Cargo.toml index 0df3a1419b7..7f05e3127db 100644 --- a/crates/blockifier/Cargo.toml +++ b/crates/blockifier/Cargo.toml @@ -10,7 +10,7 @@ description = "The transaction-executing component in the Starknet sequencer." workspace = true [features] -cairo_native = ["dep:cairo-native"] +cairo_native = ["dep:cairo-native", "starknet_sierra_compile/cairo_native"] jemalloc = ["dep:tikv-jemallocator"] reexecution = ["transaction_serde"] testing = ["rand", "rstest", "starknet_api/testing"] @@ -50,6 +50,7 @@ serde_json = { workspace = true, features = ["arbitrary_precision"] } sha2.workspace = true starknet-types-core.workspace = true starknet_api.workspace = true +starknet_sierra_compile = { workspace = true, optional = true } strum.workspace = true strum_macros.workspace = true tempfile.workspace = true diff --git a/crates/blockifier/src/state.rs b/crates/blockifier/src/state.rs index e027d2b301c..8aa857c963a 100644 --- a/crates/blockifier/src/state.rs +++ b/crates/blockifier/src/state.rs @@ -1,4 +1,6 @@ pub mod cached_state; +#[cfg(feature = "cairo_native")] +pub mod contract_class_manager; #[cfg(test)] pub mod error_format_test; pub mod errors; diff --git a/crates/blockifier/src/state/contract_class_manager.rs b/crates/blockifier/src/state/contract_class_manager.rs new file mode 100644 index 00000000000..ee7db1ebcd5 --- /dev/null +++ b/crates/blockifier/src/state/contract_class_manager.rs @@ -0,0 +1,200 @@ +use std::fmt::{Display, Formatter}; +use std::sync::atomic::{AtomicBool, Ordering}; +use std::sync::mpsc::{sync_channel, Receiver, SyncSender, TrySendError}; +use std::sync::{Arc, Mutex}; +use std::thread::JoinHandle; + +use log::{error, info}; +use starknet_api::core::ClassHash; +use starknet_api::state::ContractClass as SierraContractClass; +use starknet_sierra_compile::command_line_compiler::CommandLineCompiler; +use starknet_sierra_compile::config::SierraToCasmCompilationConfig; +use starknet_sierra_compile::SierraToNativeCompiler; + +use crate::execution::contract_class::{ContractClassV1, RunnableContractClass}; +use crate::execution::native::contract_class::NativeContractClassV1; +use crate::state::global_cache::{CachedCairoNative, ContractClassCaches}; + +const CHANNEL_SIZE: usize = 1000; + +/// A compilation request is a tuple of the class hash, the sierra contract class, and the casm +/// contract class. +/// * `class_hash` - used to identify the contract class in the cache. +/// * `sierra_contract_class` - the code to be compiled. +/// * `casm_contract_class` - stored in [`NativeContractClassV1`] to allow fallback to cairo_vm +/// execution in case of unxecpected failure during native execution. +// type CompilationRequest = (ClassHash, Arc, ContractClassV1); +pub enum CompilationRequest { + Terminate, + Request(ClassHash, Arc, ContractClassV1), +} + +impl Display for CompilationRequest { + fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { + match self { + CompilationRequest::Terminate => write!(f, "Terminate"), + CompilationRequest::Request(class_hash, _, _) => { + write!(f, "Request(class_hash: {:?})", class_hash) + } + } + } +} + +/// Manages the global cache of contract classes and handles sierra-to-native compilation requests. +struct ContractClassManager { + // The global cache of contract classes: casm, sierra, and native. + contract_class_caches: ContractClassCaches, + // The sending half of the channel for sending compilation requests. + sender: SyncSender, + // A flag that signals the termination of the compilation requests thread. + halt_marker: AtomicBool, + // The handle to the compilation requests thread. + join_handle: Mutex>>, +} + +impl ContractClassManager { + /// Creates a new contract class manager and spawns a thread that listens for compilation + /// requests and processes them. Returns an Arc to the created manager. + pub fn initialize(contract_class_caches: ContractClassCaches) -> Arc { + // TODO(Avi, 15/12/2024): Add the size of the channel to the config. + let (sender, receiver) = sync_channel(CHANNEL_SIZE); + let compiler_config = SierraToCasmCompilationConfig::default(); + let compiler = CommandLineCompiler::new(compiler_config); + let halt_marker = AtomicBool::new(false); + // Store `None` in a mutex-guarded Option to allow setting the join handle after spawning + // the thread. + let join_handle = Mutex::new(None); + + // Create the manager. + let contract_class_manager = Arc::new(ContractClassManager { + contract_class_caches, + sender, + halt_marker, + join_handle, + }); + + // Spawn the compilation requests handler thread. + let join_handle = std::thread::spawn({ + let contract_class_manager = Arc::clone(&contract_class_manager); + move || contract_class_manager.compilation_requests_handler(receiver, compiler) + }); + + // Store the join handle in a mutex-guarded Option to allow waiting for the thread to + // finish. + let mut mutex_guard = contract_class_manager + .join_handle + .lock() + .expect("No other thread should access the join handle."); + *mutex_guard = Some(join_handle); + drop(mutex_guard); + + contract_class_manager + } + + /// Tries to send a compilation request to the manager. Does not block the sender thread. + /// If the channel is full, logs an error. + pub fn try_send_compilation_request(&self, compilation_request: CompilationRequest) { + self.cache_compilation_request_contracts(&compilation_request); + + self.sender.try_send(compilation_request).map_err(|err| match err { + TrySendError::Full(request) => { + error!( + "Compilation request channel is full (size: {}). Compilation request {} was \ + not sent.", + CHANNEL_SIZE, request + ) + } + TrySendError::Disconnected(_) => panic!("Compilation request channel is closed."), + }); + } + + /// Halts the compilation requests thread. + pub fn halt(&self) { + self.halt_marker.store(true, Ordering::Relaxed); + // In case the channel is empty, send a termination request to unblock the receiver. + self.sender.try_send(CompilationRequest::Terminate).unwrap(); + } + + pub fn get_native(&self, class_hash: &ClassHash) -> Option { + self.contract_class_caches.get_native(class_hash) + } + + pub fn get_sierra(&self, class_hash: &ClassHash) -> Option> { + self.contract_class_caches.get_sierra(class_hash) + } + + pub fn get_casm(&self, class_hash: &ClassHash) -> Option { + self.contract_class_caches.get_casm(class_hash) + } + + /// Handles compilation requests from the receiver. For each request, compiles the sierra + /// contract class to a native contract class and stores the result in the cache. + /// If no request is available, non-busy-waits until a request is available. + fn compilation_requests_handler( + &self, + receiver: Receiver, + compiler: CommandLineCompiler, + ) { + info!("Compilation requests handler started."); + for compilation_request in receiver.iter() { + if self.halted() { + info!("Compilation requests handler terminated."); + return; + } + match compilation_request { + CompilationRequest::Terminate => { + info!( + "Compilation requests handler terminated without setting the halt marker." + ); + return; + } + CompilationRequest::Request(class_hash, sierra, casm) => { + if self.contract_class_caches.get_native(&class_hash).is_some() { + // The contract class is already compiled to native - skip the compilation. + continue; + } + // TODO(Avi): Convert `sierra_contract_class` to + // `cairo_lang_starknet_classes::contract_class::ContractClass` + let compilation_result = compiler.compile_to_native(sierra.into()); + match compilation_result { + Ok(executor) => { + let native_contract_class = NativeContractClassV1::new(executor, casm); + self.contract_class_caches.set_native( + class_hash, + CachedCairoNative::Compiled(native_contract_class), + ); + } + Err(err) => { + error!("Error compiling contract class: {}", err); + self.contract_class_caches + .set_native(class_hash, CachedCairoNative::CompilationFailed); + } + } + } + } + } + } + + fn cache_compilation_request_contracts(&self, compilation_request: &CompilationRequest) { + match compilation_request { + CompilationRequest::Terminate => {} + CompilationRequest::Request(class_hash, sierra, casm) => { + self.contract_class_caches.set_sierra(class_hash.clone(), sierra.clone()); + let cached_casm = RunnableContractClass::from(casm.clone()); + self.contract_class_caches.set_casm(class_hash.clone(), cached_casm); + } + } + } + + fn halted(&self) -> bool { + self.halt_marker.load(Ordering::Relaxed) + } +} + +impl Drop for ContractClassManager { + fn drop(&mut self) { + self.halt(); + let join_handle = self.join_handle.lock().unwrap().take().unwrap(); + join_handle.join().unwrap(); + } +} diff --git a/crates/blockifier/src/state/global_cache.rs b/crates/blockifier/src/state/global_cache.rs index 0a21c3759d8..97742970aa6 100644 --- a/crates/blockifier/src/state/global_cache.rs +++ b/crates/blockifier/src/state/global_cache.rs @@ -1,14 +1,14 @@ use std::sync::{Arc, Mutex, MutexGuard}; use cached::{Cached, SizedCache}; -#[cfg(feature = "cairo_native")] -use cairo_native::executor::AotContractExecutor; use starknet_api::core::ClassHash; #[cfg(feature = "cairo_native")] use starknet_api::state::ContractClass as SierraContractClass; #[cfg(feature = "cairo_native")] use crate::execution::contract_class::RunnableContractClass; +#[cfg(feature = "cairo_native")] +use crate::execution::native::contract_class::NativeContractClassV1; type ContractClassLRUCache = SizedCache; pub type LockedContractClassCache<'a, T> = MutexGuard<'a, ContractClassLRUCache>; @@ -21,7 +21,7 @@ pub struct GlobalContractCache(pub Arc> #[cfg(feature = "cairo_native")] #[derive(Debug, Clone)] pub enum CachedCairoNative { - Compiled(AotContractExecutor), + Compiled(NativeContractClassV1), CompilationFailed, } @@ -52,14 +52,14 @@ impl GlobalContractCache { } #[cfg(feature = "cairo_native")] -pub struct GlobalContractCacheManager { +pub struct ContractClassCaches { pub casm_cache: GlobalContractCache, pub native_cache: GlobalContractCache, pub sierra_cache: GlobalContractCache>, } #[cfg(feature = "cairo_native")] -impl GlobalContractCacheManager { +impl ContractClassCaches { pub fn get_casm(&self, class_hash: &ClassHash) -> Option { self.casm_cache.get(class_hash) }