cairo_lang_sierra_to_casm/
metadata.rsuse cairo_lang_sierra::extensions::gas::CostTokenType;
use cairo_lang_sierra::ids::FunctionId;
use cairo_lang_sierra::program::Program;
use cairo_lang_sierra_ap_change::ap_change_info::ApChangeInfo;
use cairo_lang_sierra_ap_change::compute::calc_ap_changes as linear_calc_ap_changes;
use cairo_lang_sierra_ap_change::{ApChangeError, calc_ap_changes};
use cairo_lang_sierra_gas::gas_info::GasInfo;
use cairo_lang_sierra_gas::objects::ConstCost;
use cairo_lang_sierra_gas::{
CostError, calc_gas_postcost_info, calc_gas_precost_info, compute_postcost_info,
compute_precost_info,
};
use cairo_lang_utils::ordered_hash_map::OrderedHashMap;
use thiserror::Error;
#[derive(Default)]
pub struct Metadata {
pub ap_change_info: ApChangeInfo,
pub gas_info: GasInfo,
}
#[derive(Debug, Error, Eq, PartialEq)]
pub enum MetadataError {
#[error(transparent)]
ApChangeError(#[from] ApChangeError),
#[error(transparent)]
CostError(#[from] CostError),
}
#[derive(Clone)]
pub struct MetadataComputationConfig {
pub function_set_costs: OrderedHashMap<FunctionId, OrderedHashMap<CostTokenType, i32>>,
pub linear_gas_solver: bool,
pub linear_ap_change_solver: bool,
pub skip_non_linear_solver_comparisons: bool,
pub compute_runtime_costs: bool,
}
impl Default for MetadataComputationConfig {
fn default() -> Self {
Self {
function_set_costs: Default::default(),
linear_gas_solver: true,
linear_ap_change_solver: true,
skip_non_linear_solver_comparisons: false,
compute_runtime_costs: false,
}
}
}
pub fn calc_metadata_ap_change_only(program: &Program) -> Result<Metadata, MetadataError> {
Ok(Metadata {
ap_change_info: calc_ap_changes(program, |_, _| 0)?,
gas_info: GasInfo {
variable_values: Default::default(),
function_costs: Default::default(),
},
})
}
pub fn calc_metadata(
program: &Program,
config: MetadataComputationConfig,
) -> Result<Metadata, MetadataError> {
let pre_function_set_costs = config
.function_set_costs
.iter()
.map(|(func, costs)| {
(
func.clone(),
CostTokenType::iter_precost()
.filter_map(|token| costs.get(token).map(|v| (*token, *v)))
.collect(),
)
})
.collect();
let pre_gas_info_new = compute_precost_info(program)?;
let pre_gas_info = if config.linear_gas_solver {
pre_gas_info_new
} else {
let pre_gas_info_old = calc_gas_precost_info(program, pre_function_set_costs)?;
if !config.skip_non_linear_solver_comparisons {
pre_gas_info_old.assert_eq_variables(&pre_gas_info_new, program);
pre_gas_info_old.assert_eq_functions(&pre_gas_info_new);
}
pre_gas_info_old
};
let ap_change_info =
if config.linear_ap_change_solver { linear_calc_ap_changes } else { calc_ap_changes }(
program,
|idx, token_type| pre_gas_info.variable_values[&(idx, token_type)] as usize,
)?;
let mut post_gas_info = if config.linear_gas_solver {
let enforced_function_costs: OrderedHashMap<FunctionId, i32> = config
.function_set_costs
.iter()
.map(|(func, costs)| (func.clone(), costs[&CostTokenType::Const]))
.collect();
compute_postcost_info(
program,
&|idx| ap_change_info.variable_values.get(idx).copied().unwrap_or_default(),
&pre_gas_info,
&enforced_function_costs,
)
} else {
let post_function_set_costs = config
.function_set_costs
.iter()
.map(|(func, costs)| {
(
func.clone(),
[CostTokenType::Const]
.iter()
.filter_map(|token| costs.get(token).map(|v| (*token, *v)))
.collect(),
)
})
.collect();
calc_gas_postcost_info(program, post_function_set_costs, &pre_gas_info, |idx| {
ap_change_info.variable_values.get(&idx).copied().unwrap_or_default()
})
}?;
if config.compute_runtime_costs {
let post_gas_info_runtime = compute_postcost_info::<ConstCost>(
program,
&|idx| ap_change_info.variable_values.get(idx).copied().unwrap_or_default(),
&pre_gas_info,
&Default::default(),
)?;
post_gas_info = post_gas_info.combine(post_gas_info_runtime);
}
Ok(Metadata { ap_change_info, gas_info: pre_gas_info.combine(post_gas_info) })
}