sui_adapter_v1/
adapter.rs1pub use checked::*;
5
6#[sui_macros::with_checked_arithmetic]
7mod checked {
8 use std::{collections::BTreeMap, sync::Arc};
9
10 use anyhow::Result;
11 use move_binary_format::file_format::CompiledModule;
12 use move_bytecode_verifier::verify_module_with_config_metered;
13 use move_bytecode_verifier_meter::Meter;
14 use move_core_types::account_address::AccountAddress;
15 use move_vm_config::{
16 runtime::{VMConfig, VMRuntimeLimitsConfig},
17 verifier::VerifierConfig,
18 };
19 use move_vm_runtime::{
20 move_vm::MoveVM, native_extensions::NativeContextExtensions,
21 native_functions::NativeFunctionTable,
22 };
23 use sui_move_natives::object_runtime;
24 use sui_types::{error::SuiErrorKind, metrics::BytecodeVerifierMetrics};
25 use sui_verifier::check_for_verifier_timeout;
26 use tracing::instrument;
27
28 use sui_move_natives::{object_runtime::ObjectRuntime, NativesCostTable};
29 use sui_protocol_config::ProtocolConfig;
30 use sui_types::{
31 base_types::*,
32 error::{ExecutionError, SuiError},
33 execution_status::ExecutionErrorKind,
34 metrics::LimitsMetrics,
35 storage::ChildObjectResolver,
36 };
37 use sui_verifier::verifier::sui_verify_module_metered_check_timeout_only;
38
39 pub fn new_move_vm(
40 natives: NativeFunctionTable,
41 protocol_config: &ProtocolConfig,
42 ) -> Result<MoveVM, SuiError> {
43 MoveVM::new_with_config(
44 natives,
45 VMConfig {
46 verifier: protocol_config.verifier_config(None),
47 max_binary_format_version: protocol_config.move_binary_format_version(),
48 runtime_limits_config: VMRuntimeLimitsConfig {
49 vector_len_max: protocol_config.max_move_vector_len(),
50 max_value_nest_depth: protocol_config.max_move_value_depth_as_option(),
51 hardened_otw_check: protocol_config.hardened_otw_check(),
52 },
53 enable_invariant_violation_check_in_swap_loc: !protocol_config
54 .disable_invariant_violation_check_in_swap_loc(),
55 check_no_extraneous_bytes_during_deserialization: protocol_config
56 .no_extraneous_module_bytes(),
57
58 error_execution_state: false,
60 binary_config: protocol_config.binary_config(None),
61 rethrow_serialization_type_layout_errors: protocol_config
62 .rethrow_serialization_type_layout_errors(),
63 max_type_to_layout_nodes: protocol_config.max_type_to_layout_nodes_as_option(),
64 variant_nodes: protocol_config.variant_nodes(),
65 deprecate_global_storage_ops_during_deserialization: protocol_config
66 .deprecate_global_storage_ops_during_deserialization(),
67 normalize_depth_formula: protocol_config.normalize_depth_formula(),
68 },
69 )
70 .map_err(|_| SuiErrorKind::ExecutionInvariantViolation.into())
71 }
72
73 pub fn new_native_extensions<'r>(
74 child_resolver: &'r dyn ChildObjectResolver,
75 input_objects: BTreeMap<ObjectID, object_runtime::InputObject>,
76 is_metered: bool,
77 protocol_config: &ProtocolConfig,
78 metrics: Arc<LimitsMetrics>,
79 current_epoch_id: EpochId,
80 ) -> NativeContextExtensions<'r> {
81 let mut extensions = NativeContextExtensions::default();
82 extensions.add(ObjectRuntime::new(
83 child_resolver,
84 input_objects,
85 is_metered,
86 protocol_config,
87 metrics,
88 current_epoch_id,
89 ));
90 extensions.add(NativesCostTable::from_protocol_config(protocol_config));
91 extensions
92 }
93
94 pub fn substitute_package_id(
97 modules: &mut [CompiledModule],
98 object_id: ObjectID,
99 ) -> Result<(), ExecutionError> {
100 let new_address = AccountAddress::from(object_id);
101
102 for module in modules.iter_mut() {
103 let self_handle = module.self_handle().clone();
104 let self_address_idx = self_handle.address;
105
106 let addrs = &mut module.address_identifiers;
107 let Some(address_mut) = addrs.get_mut(self_address_idx.0 as usize) else {
108 let name = module.identifier_at(self_handle.name);
109 return Err(ExecutionError::new_with_source(
110 ExecutionErrorKind::PublishErrorNonZeroAddress,
111 format!("Publishing module {name} with invalid address index"),
112 ));
113 };
114
115 if *address_mut != AccountAddress::ZERO {
116 let name = module.identifier_at(self_handle.name);
117 return Err(ExecutionError::new_with_source(
118 ExecutionErrorKind::PublishErrorNonZeroAddress,
119 format!("Publishing module {name} with non-zero address is not allowed"),
120 ));
121 };
122
123 *address_mut = new_address;
124 }
125
126 Ok(())
127 }
128
129 pub fn missing_unwrapped_msg(id: &ObjectID) -> String {
130 format!(
131 "Unable to unwrap object {}. Was unable to retrieve last known version in the parent sync",
132 id
133 )
134 }
135
136 #[instrument(level = "trace", skip_all)]
142 pub fn run_metered_move_bytecode_verifier(
143 modules: &[CompiledModule],
144 verifier_config: &VerifierConfig,
145 meter: &mut (impl Meter + ?Sized),
146 metrics: &Arc<BytecodeVerifierMetrics>,
147 ) -> Result<(), SuiError> {
148 for module in modules.iter() {
150 let per_module_meter_verifier_timer = metrics
151 .verifier_runtime_per_module_success_latency
152 .start_timer();
153
154 if let Err(e) = verify_module_with_config_metered(verifier_config, module, meter) {
155 if check_for_verifier_timeout(&e.major_status()) {
157 metrics
159 .verifier_runtime_per_module_timeout_latency
160 .observe(per_module_meter_verifier_timer.stop_and_discard());
161 metrics
162 .verifier_timeout_metrics
163 .with_label_values(&[
164 BytecodeVerifierMetrics::MOVE_VERIFIER_TAG,
165 BytecodeVerifierMetrics::TIMEOUT_TAG,
166 ])
167 .inc();
168 return Err(SuiErrorKind::ModuleVerificationFailure {
169 error: format!("Verification timedout: {}", e),
170 }
171 .into());
172 };
173 } else if let Err(err) =
174 sui_verify_module_metered_check_timeout_only(module, &BTreeMap::new(), meter)
175 {
176 metrics
179 .verifier_runtime_per_module_timeout_latency
180 .observe(per_module_meter_verifier_timer.stop_and_discard());
181 metrics
182 .verifier_timeout_metrics
183 .with_label_values(&[
184 BytecodeVerifierMetrics::SUI_VERIFIER_TAG,
185 BytecodeVerifierMetrics::TIMEOUT_TAG,
186 ])
187 .inc();
188 return Err(err.into());
189 }
190 per_module_meter_verifier_timer.stop_and_record();
192 metrics
193 .verifier_timeout_metrics
194 .with_label_values(&[
195 BytecodeVerifierMetrics::OVERALL_TAG,
196 BytecodeVerifierMetrics::SUCCESS_TAG,
197 ])
198 .inc();
199 }
200 Ok(())
201 }
202}