1use std::collections::HashMap;
3
4use cairo_lang_casm::hints::Hint;
5use cairo_lang_runnable_utils::builder::{BuildError, EntryCodeConfig, RunnableBuilder};
6use cairo_lang_sierra::extensions::NamedType;
7use cairo_lang_sierra::extensions::enm::EnumType;
8use cairo_lang_sierra::extensions::gas::{CostTokenType, GasBuiltinType};
9use cairo_lang_sierra::ids::{ConcreteTypeId, GenericTypeId};
10use cairo_lang_sierra::program::{Function, GenericArg};
11use cairo_lang_sierra_to_casm::metadata::MetadataComputationConfig;
12use cairo_lang_starknet::contract::ContractInfo;
13use cairo_lang_utils::casts::IntoOrPanic;
14use cairo_lang_utils::ordered_hash_map::OrderedHashMap;
15use cairo_lang_utils::{extract_matches, require};
16use cairo_vm::hint_processor::hint_processor_definition::HintProcessor;
17use cairo_vm::serde::deserialize_program::HintParams;
18use cairo_vm::types::builtin_name::BuiltinName;
19use cairo_vm::vm::errors::cairo_run_errors::CairoRunError;
20use cairo_vm::vm::runners::cairo_runner::{ExecutionResources, RunResources};
21use cairo_vm::vm::vm_core::VirtualMachine;
22use casm_run::hint_to_hint_params;
23pub use casm_run::{CairoHintProcessor, StarknetState};
24use num_bigint::BigInt;
25use num_traits::ToPrimitive;
26use profiling::ProfilingInfo;
27use starknet_types_core::felt::Felt as Felt252;
28use thiserror::Error;
29
30use crate::casm_run::{RunFunctionResult, StarknetHintProcessor};
31use crate::profiling::ProfilerConfig;
32
33pub mod casm_run;
34pub mod clap;
35pub mod profiling;
36pub mod short_string;
37
38const MAX_STACK_TRACE_DEPTH_DEFAULT: usize = 100;
39
40#[derive(Debug, Error)]
41pub enum RunnerError {
42 #[error(transparent)]
43 BuildError(#[from] BuildError),
44 #[error("Not enough gas to call function.")]
45 NotEnoughGasToCall,
46 #[error("Function param {param_index} only partially contains argument {arg_index}.")]
47 ArgumentUnaligned { param_index: usize, arg_index: usize },
48 #[error("Function expects arguments of size {expected} and received {actual} instead.")]
49 ArgumentsSizeMismatch { expected: usize, actual: usize },
50 #[error(transparent)]
51 CairoRunError(#[from] Box<CairoRunError>),
52}
53
54pub struct RunResultStarknet {
56 pub gas_counter: Option<Felt252>,
57 pub memory: Vec<Option<Felt252>>,
58 pub value: RunResultValue,
59 pub starknet_state: StarknetState,
60 pub used_resources: StarknetExecutionResources,
61 pub profiling_info: Option<ProfilingInfo>,
63}
64
65#[derive(Debug, Eq, PartialEq, Clone)]
67pub struct RunResult {
68 pub gas_counter: Option<Felt252>,
69 pub memory: Vec<Option<Felt252>>,
70 pub value: RunResultValue,
71 pub used_resources: ExecutionResources,
72 pub profiling_info: Option<ProfilingInfo>,
74}
75
76#[derive(Debug, Eq, PartialEq, Clone, Default)]
79pub struct StarknetExecutionResources {
80 pub basic_resources: ExecutionResources,
82 pub syscalls: HashMap<String, usize>,
84}
85
86impl std::ops::AddAssign<StarknetExecutionResources> for StarknetExecutionResources {
87 fn add_assign(&mut self, other: Self) {
89 self.basic_resources += &other.basic_resources;
90 for (k, v) in other.syscalls {
91 *self.syscalls.entry(k).or_default() += v;
92 }
93 }
94}
95
96#[derive(Debug, Eq, PartialEq, Clone)]
98pub enum RunResultValue {
99 Success(Vec<Felt252>),
101 Panic(Vec<Felt252>),
103}
104
105pub fn token_gas_cost(token_type: CostTokenType) -> usize {
107 match token_type {
108 CostTokenType::Const => 1,
109 CostTokenType::Step
110 | CostTokenType::Hole
111 | CostTokenType::RangeCheck
112 | CostTokenType::RangeCheck96 => {
113 panic!("Token type {token_type:?} has no gas cost.")
114 }
115 CostTokenType::Pedersen => 4050,
116 CostTokenType::Poseidon => 491,
117 CostTokenType::Bitwise => 583,
118 CostTokenType::EcOp => 4085,
119 CostTokenType::AddMod => 230,
120 CostTokenType::MulMod => 604,
121 }
122}
123
124#[derive(Debug, Clone)]
126pub enum Arg {
127 Value(Felt252),
128 Array(Vec<Arg>),
129}
130impl Arg {
131 pub fn size(&self) -> usize {
133 match self {
134 Self::Value(_) => 1,
135 Self::Array(_) => 2,
136 }
137 }
138}
139impl From<Felt252> for Arg {
140 fn from(value: Felt252) -> Self {
141 Self::Value(value)
142 }
143}
144
145pub fn build_hints_dict(
147 hints: &[(usize, Vec<Hint>)],
148) -> (HashMap<usize, Vec<HintParams>>, HashMap<String, Hint>) {
149 let mut hints_dict: HashMap<usize, Vec<HintParams>> = HashMap::new();
150 let mut string_to_hint: HashMap<String, Hint> = HashMap::new();
151
152 for (offset, offset_hints) in hints {
153 for hint in offset_hints {
155 string_to_hint.insert(hint.representing_string(), hint.clone());
156 }
157 hints_dict.insert(*offset, offset_hints.iter().map(hint_to_hint_params).collect());
159 }
160 (hints_dict, string_to_hint)
161}
162
163pub struct PreparedStarknetContext {
171 pub hints_dict: HashMap<usize, Vec<HintParams>>,
172 pub bytecode: Vec<BigInt>,
173 pub builtins: Vec<BuiltinName>,
174}
175
176pub struct SierraCasmRunner {
178 builder: RunnableBuilder,
180 starknet_contracts_info: OrderedHashMap<Felt252, ContractInfo>,
182 run_profiler: Option<ProfilingInfoCollectionConfig>,
184}
185impl SierraCasmRunner {
186 pub fn new(
187 sierra_program: cairo_lang_sierra::program::Program,
188 metadata_config: Option<MetadataComputationConfig>,
189 starknet_contracts_info: OrderedHashMap<Felt252, ContractInfo>,
190 run_profiler: Option<ProfilingInfoCollectionConfig>,
191 ) -> Result<Self, RunnerError> {
192 Ok(Self {
194 builder: RunnableBuilder::new(sierra_program, metadata_config)?,
195 starknet_contracts_info,
196 run_profiler,
197 })
198 }
199
200 pub fn run_function_with_starknet_context(
202 &self,
203 func: &Function,
204 args: Vec<Arg>,
205 available_gas: Option<usize>,
206 starknet_state: StarknetState,
207 ) -> Result<RunResultStarknet, RunnerError> {
208 let (mut hint_processor, ctx) =
209 self.prepare_starknet_context(func, args, available_gas, starknet_state)?;
210 self.run_function_with_prepared_starknet_context(func, &mut hint_processor, ctx)
211 }
212
213 pub fn run_function_with_prepared_starknet_context(
216 &self,
217 func: &Function,
218 hint_processor: &mut dyn StarknetHintProcessor,
219 PreparedStarknetContext { hints_dict, bytecode, builtins }: PreparedStarknetContext,
220 ) -> Result<RunResultStarknet, RunnerError> {
221 let RunResult { gas_counter, memory, value, used_resources, profiling_info } =
222 self.run_function(func, hint_processor, hints_dict, bytecode.iter(), builtins)?;
223 let mut all_used_resources = hint_processor.take_syscalls_used_resources();
224 all_used_resources.basic_resources += &used_resources;
225 Ok(RunResultStarknet {
226 gas_counter,
227 memory,
228 value,
229 starknet_state: hint_processor.take_starknet_state(),
230 used_resources: all_used_resources,
231 profiling_info,
232 })
233 }
234
235 fn inner_type_from_panic_wrapper(
237 &self,
238 ty: &GenericTypeId,
239 func: &Function,
240 ) -> Option<ConcreteTypeId> {
241 let generic_args = &func
242 .signature
243 .ret_types
244 .iter()
245 .find_map(|rt| {
246 let long_id = self.builder.type_long_id(rt);
247 (long_id.generic_id == *ty).then_some(long_id)
248 })
249 .unwrap()
250 .generic_args;
251
252 if *ty == EnumType::ID
253 && matches!(&generic_args[0], GenericArg::UserType(ut)
254 if ut.debug_name.as_ref().unwrap().starts_with("core::panics::PanicResult::"))
255 {
256 return Some(extract_matches!(&generic_args[1], GenericArg::Type).clone());
257 }
258 None
259 }
260
261 pub fn run_function<'a, Bytecode>(
265 &self,
266 func: &Function,
267 hint_processor: &mut dyn HintProcessor,
268 hints_dict: HashMap<usize, Vec<HintParams>>,
269 bytecode: Bytecode,
270 builtins: Vec<BuiltinName>,
271 ) -> Result<RunResult, RunnerError>
272 where
273 Bytecode: ExactSizeIterator<Item = &'a BigInt> + Clone,
274 {
275 let return_types =
276 self.builder.generic_id_and_size_from_concrete(&func.signature.ret_types);
277 let data_len = bytecode.len();
278 let RunFunctionResult { ap, mut used_resources, memory, relocated_trace } =
279 casm_run::run_function(
280 bytecode,
281 builtins,
282 |vm| initialize_vm(vm, data_len),
283 hint_processor,
284 hints_dict,
285 )?;
286
287 let header_end = relocated_trace.last().unwrap().pc;
290 used_resources.n_steps -=
291 relocated_trace.iter().position(|e| e.pc > header_end).unwrap() - 1;
292 used_resources.n_steps -=
293 relocated_trace.iter().rev().position(|e| e.pc > header_end).unwrap() - 1;
294
295 let (results_data, gas_counter) = self.get_results_data(&return_types, &memory, ap);
296 assert!(results_data.len() <= 1);
297
298 let value = if results_data.is_empty() {
299 RunResultValue::Success(vec![])
301 } else {
302 let (ty, values) = results_data[0].clone();
303 let inner_ty =
304 self.inner_type_from_panic_wrapper(&ty, func).map(|it| self.builder.type_size(&it));
305 Self::handle_main_return_value(inner_ty, values, &memory)
306 };
307
308 let Self { builder, starknet_contracts_info: _, run_profiler } = self;
309
310 let load_offset = header_end + 1;
312
313 let profiling_info = run_profiler.as_ref().map(|config| {
314 ProfilingInfo::from_trace(builder, load_offset, config, &relocated_trace)
315 });
316
317 Ok(RunResult { gas_counter, memory, value, used_resources, profiling_info })
318 }
319
320 pub fn prepare_starknet_context(
326 &self,
327 func: &Function,
328 args: Vec<Arg>,
329 available_gas: Option<usize>,
330 starknet_state: StarknetState,
331 ) -> Result<(CairoHintProcessor<'_>, PreparedStarknetContext), RunnerError> {
332 let (assembled_program, builtins) =
333 self.builder.assemble_function_program(func, EntryCodeConfig::testing())?;
334 let (hints_dict, string_to_hint) = build_hints_dict(&assembled_program.hints);
335 let user_args = self.prepare_args(func, available_gas, args)?;
336 let hint_processor = CairoHintProcessor {
337 runner: Some(self),
338 user_args,
339 starknet_state,
340 string_to_hint,
341 run_resources: RunResources::default(),
342 syscalls_used_resources: Default::default(),
343 no_temporary_segments: true,
344 markers: Default::default(),
345 panic_traceback: Default::default(),
346 };
347 Ok((
348 hint_processor,
349 PreparedStarknetContext { hints_dict, bytecode: assembled_program.bytecode, builtins },
350 ))
351 }
352
353 fn prepare_args(
355 &self,
356 func: &Function,
357 available_gas: Option<usize>,
358 args: Vec<Arg>,
359 ) -> Result<Vec<Vec<Arg>>, RunnerError> {
360 let mut user_args = vec![];
361 if let Some(gas) = self
362 .requires_gas_builtin(func)
363 .then_some(self.get_initial_available_gas(func, available_gas)?)
364 {
365 user_args.push(vec![Arg::Value(Felt252::from(gas))]);
366 }
367 let mut expected_arguments_size = 0;
368 let actual_args_size = args_size(&args);
369 let mut arg_iter = args.into_iter().enumerate();
370 for (param_index, (_, param_size)) in self
371 .builder
372 .generic_id_and_size_from_concrete(&func.signature.param_types)
373 .into_iter()
374 .filter(|(ty, _)| self.builder.is_user_arg_type(ty))
375 .enumerate()
376 {
377 let mut curr_arg = vec![];
378 let param_size: usize = param_size.into_or_panic();
379 expected_arguments_size += param_size;
380 let mut taken_size = 0;
381 while taken_size < param_size {
382 let Some((arg_index, arg)) = arg_iter.next() else {
383 break;
384 };
385 taken_size += arg.size();
386 if taken_size > param_size {
387 return Err(RunnerError::ArgumentUnaligned { param_index, arg_index });
388 }
389 curr_arg.push(arg);
390 }
391 user_args.push(curr_arg);
392 }
393 if expected_arguments_size != actual_args_size {
394 return Err(RunnerError::ArgumentsSizeMismatch {
395 expected: expected_arguments_size,
396 actual: actual_args_size,
397 });
398 }
399 Ok(user_args)
400 }
401
402 pub fn handle_main_return_value(
404 inner_type_size: Option<i16>,
405 values: Vec<Felt252>,
406 cells: &[Option<Felt252>],
407 ) -> RunResultValue {
408 if let Some(inner_type_size) = inner_type_size {
409 if values[0] == Felt252::from(0) {
411 let inner_ty_size = inner_type_size as usize;
413 let skip_size = values.len() - inner_ty_size;
414 RunResultValue::Success(values.into_iter().skip(skip_size).collect())
415 } else {
416 let err_data_start = values[values.len() - 2].to_usize().unwrap();
418 let err_data_end = values[values.len() - 1].to_usize().unwrap();
419 RunResultValue::Panic(
420 cells[err_data_start..err_data_end]
421 .iter()
422 .cloned()
423 .map(Option::unwrap)
424 .collect(),
425 )
426 }
427 } else {
428 RunResultValue::Success(values)
430 }
431 }
432
433 pub fn get_results_data(
435 &self,
436 return_types: &[(GenericTypeId, i16)],
437 cells: &[Option<Felt252>],
438 mut ap: usize,
439 ) -> (Vec<(GenericTypeId, Vec<Felt252>)>, Option<Felt252>) {
440 let mut results_data = vec![];
441 for (ty, ty_size) in return_types.iter().rev() {
442 let size = *ty_size as usize;
443 let values: Vec<Felt252> =
444 ((ap - size)..ap).map(|index| cells[index].unwrap()).collect();
445 ap -= size;
446 results_data.push((ty.clone(), values));
447 }
448
449 let mut gas_counter = None;
451 results_data.retain_mut(|(ty, values)| {
452 let generic_ty = ty;
453 if *generic_ty == GasBuiltinType::ID {
454 gas_counter = Some(values.remove(0));
455 assert!(values.is_empty());
456 false
457 } else {
458 self.builder.is_user_arg_type(generic_ty)
459 }
460 });
461
462 (results_data, gas_counter)
463 }
464
465 pub fn find_function(&self, name_suffix: &str) -> Result<&Function, RunnerError> {
467 Ok(self.builder.find_function(name_suffix)?)
468 }
469
470 fn requires_gas_builtin(&self, func: &Function) -> bool {
472 func.signature
473 .param_types
474 .iter()
475 .any(|ty| self.builder.type_long_id(ty).generic_id == GasBuiltinType::ID)
476 }
477
478 pub fn get_initial_available_gas(
481 &self,
482 func: &Function,
483 available_gas: Option<usize>,
484 ) -> Result<usize, RunnerError> {
485 let Some(available_gas) = available_gas else {
486 return Ok(0);
487 };
488
489 let Some(required_gas) = self.initial_required_gas(func) else {
493 return Ok(0);
494 };
495
496 available_gas.checked_sub(required_gas).ok_or(RunnerError::NotEnoughGasToCall)
497 }
498
499 pub fn initial_required_gas(&self, func: &Function) -> Option<usize> {
500 let gas_info = &self.builder.metadata().gas_info;
501 require(!gas_info.function_costs.is_empty())?;
502 Some(
503 gas_info.function_costs[&func.id]
504 .iter()
505 .map(|(token_type, val)| val.into_or_panic::<usize>() * token_gas_cost(*token_type))
506 .sum(),
507 )
508 }
509}
510
511#[derive(Debug, Eq, PartialEq, Clone)]
513pub struct ProfilingInfoCollectionConfig {
514 pub max_stack_trace_depth: usize,
516 pub collect_scoped_sierra_statement_weights: bool,
522}
523
524impl ProfilingInfoCollectionConfig {
525 pub fn set_max_stack_trace_depth(&mut self, max_stack_depth: usize) -> &mut Self {
526 self.max_stack_trace_depth = max_stack_depth;
527 self
528 }
529
530 pub fn from_profiler_config(profiler_config: &ProfilerConfig) -> Self {
531 match profiler_config {
532 ProfilerConfig::Cairo | ProfilerConfig::Sierra => Self::default(),
533 ProfilerConfig::Scoped => ProfilingInfoCollectionConfig {
534 collect_scoped_sierra_statement_weights: true,
535 ..Self::default()
536 },
537 }
538 }
539}
540
541impl Default for ProfilingInfoCollectionConfig {
542 fn default() -> Self {
546 Self {
547 max_stack_trace_depth: if let Ok(max) = std::env::var("MAX_STACK_TRACE_DEPTH") {
548 if max.is_empty() {
549 MAX_STACK_TRACE_DEPTH_DEFAULT
550 } else {
551 max.parse::<usize>().expect("MAX_STACK_TRACE_DEPTH env var is not numeric")
552 }
553 } else {
554 MAX_STACK_TRACE_DEPTH_DEFAULT
555 },
556 collect_scoped_sierra_statement_weights: false,
557 }
558 }
559}
560
561pub fn initialize_vm(vm: &mut VirtualMachine, data_len: usize) -> Result<(), Box<CairoRunError>> {
564 let builtin_cost_segment = vm.add_memory_segment();
566 for token_type in CostTokenType::iter_precost() {
567 vm.insert_value(
568 (builtin_cost_segment + (token_type.offset_in_builtin_costs() as usize)).unwrap(),
569 Felt252::from(token_gas_cost(*token_type)),
570 )
571 .map_err(|e| Box::new(e.into()))?;
572 }
573 vm.insert_value((vm.get_pc() + data_len).unwrap(), builtin_cost_segment)
576 .map_err(|e| Box::new(e.into()))?;
577 Ok(())
578}
579
580fn args_size(args: &[Arg]) -> usize {
582 args.iter().map(Arg::size).sum()
583}