1use crate::{
5 execution_mode::ExecutionMode,
6 gas_charger::GasCharger,
7 object_runtime, sp,
8 static_programmable_transactions::{
9 env::Env,
10 execution::{
11 context::{Context, CtxValue, GasCoinTransfer},
12 trace_utils,
13 },
14 loading::ast::DeserializedPackage,
15 typing::{ast as T, verify::input_arguments::is_coin_send_funds},
16 },
17};
18use move_core_types::account_address::AccountAddress;
19use move_trace_format::format::MoveTraceBuilder;
20use mysten_common::ZipDebugEqIteratorExt;
21use std::{
22 cell::RefCell,
23 collections::BTreeMap,
24 rc::Rc,
25 sync::Arc,
26 time::{Duration, Instant},
27};
28use sui_types::{
29 base_types::TxContext,
30 error::ExecutionErrorTrait,
31 execution::{ExecutionTiming, ResultWithTimings},
32 execution_status::{ExecutionErrorKind, PackageUpgradeError},
33 metrics::ExecutionMetrics,
34 object::Owner,
35};
36use tracing::instrument;
37
38pub fn execute<'env, 'pc, 'vm, 'state, 'linkage, 'extension, Mode: ExecutionMode>(
39 env: &'env mut Env<'pc, 'vm, 'state, 'linkage, 'extension, Mode>,
40 metrics: Arc<ExecutionMetrics>,
41 tx_context: Rc<RefCell<TxContext>>,
42 gas_charger: &mut GasCharger,
43 ast: T::Transaction,
44 trace_builder_opt: &mut Option<MoveTraceBuilder>,
45) -> ResultWithTimings<Mode::ExecutionResults, Mode::Error>
46where
47 'pc: 'state,
48 'env: 'state,
49{
50 let original_command_len = ast.original_command_len;
51 let mut indexed_timings = IndexedExecutionTimings::new(original_command_len);
52 let result = execute_inner::<Mode>(
53 &mut indexed_timings,
54 env,
55 metrics,
56 tx_context,
57 gas_charger,
58 ast,
59 trace_builder_opt,
60 );
61 let timings = indexed_timings.into_coalesced();
62 debug_assert!(
63 timings.len() <= original_command_len,
64 "coalesced timings length {} exceeds original command length {}",
65 timings.len(),
66 original_command_len
67 );
68
69 match result {
70 Ok(result) => Ok((result, timings)),
71 Err(e) => {
72 trace_utils::trace_execution_error(trace_builder_opt, e.to_string());
73 Err((e, timings))
74 }
75 }
76}
77
78fn execute_inner<'env, 'pc, 'vm, 'state, 'linkage, 'extension, Mode: ExecutionMode>(
79 timings: &mut IndexedExecutionTimings,
80 env: &'env mut Env<'pc, 'vm, 'state, 'linkage, 'extension, Mode>,
81 metrics: Arc<ExecutionMetrics>,
82 tx_context: Rc<RefCell<TxContext>>,
83 gas_charger: &mut GasCharger,
84 ast: T::Transaction,
85 trace_builder_opt: &mut Option<MoveTraceBuilder>,
86) -> Result<Mode::ExecutionResults, Mode::Error>
87where
88 'pc: 'state,
89{
90 debug_assert_eq!(gas_charger.move_gas_status().stack_height_current(), 0);
91 let T::Transaction {
92 gas_payment,
93 bytes,
94 objects,
95 withdrawals,
96 pure,
97 receiving,
98 withdrawal_compatibility_conversions: _,
99 original_command_len: _,
100 commands,
101 } = ast;
102 let mut context = Context::new(
103 env,
104 metrics,
105 tx_context,
106 gas_charger,
107 gas_payment,
108 bytes,
109 objects,
110 withdrawals,
111 pure,
112 receiving,
113 )?;
114
115 trace_utils::trace_ptb_summary(&mut context, trace_builder_opt, &commands)?;
116
117 let mut mode_results = Mode::empty_results();
118 for sp!(annotated_index, c) in commands {
119 let annotated_index = annotated_index as usize;
120 let start = Instant::now();
121 if let Err(err) =
122 execute_command::<Mode>(&mut context, &mut mode_results, c, trace_builder_opt)
123 {
124 let loaded_runtime_objects = object_runtime!(context)?.loaded_runtime_objects();
126 drop(context);
128 env.state_view
131 .save_loaded_runtime_objects(loaded_runtime_objects);
132 timings.error(annotated_index, start.elapsed());
133 return Err(err.with_command_index(annotated_index));
134 };
135 timings.executed(annotated_index, start.elapsed());
136 }
137 let loaded_runtime_objects = object_runtime!(context)?.loaded_runtime_objects();
143 let wrapped_object_containers = object_runtime!(context)?.wrapped_object_containers();
146 let generated_object_ids = object_runtime!(context)?.generated_object_ids();
148
149 let finished = context.finish();
151 env.state_view
153 .save_loaded_runtime_objects(loaded_runtime_objects);
154 env.state_view
155 .save_wrapped_object_containers(wrapped_object_containers);
156 env.state_view.record_execution_results(finished?)?;
157 env.state_view
158 .record_generated_object_ids(generated_object_ids);
159 Ok(mode_results)
160}
161
162#[instrument(level = "trace", skip_all)]
164fn execute_command<Mode: ExecutionMode>(
165 context: &mut Context<Mode>,
166 mode_results: &mut Mode::ExecutionResults,
167 c: T::Command_,
168 trace_builder_opt: &mut Option<MoveTraceBuilder>,
169) -> Result<(), Mode::Error> {
170 let T::Command_ {
171 command,
172 result_type,
173 drop_values,
174 incurs_post_execution_checks: _,
175 } = c;
176 assert_invariant!(
177 context.gas_charger.move_gas_status().stack_height_current() == 0,
178 "stack height did not start at 0"
179 );
180 let is_move_call = matches!(command, T::Command__::MoveCall(_));
181 let num_args = command.arguments_len();
182 let mut args_to_update = vec![];
183 let result = match command {
184 T::Command__::MoveCall(move_call) => {
185 trace_utils::trace_move_call_start(trace_builder_opt);
186 let T::MoveCall {
187 function,
188 arguments,
189 } = *move_call;
190 let is_gas_coin_send_funds = is_coin_send_funds(&function)
192 && arguments.first().is_some_and(|arg| {
193 matches!(
194 &arg.value.0,
195 T::Argument__::Use(T::Usage::Move(T::Location::GasCoin))
196 )
197 });
198 if Mode::TRACK_EXECUTION {
199 args_to_update.extend(
200 arguments
201 .iter()
202 .filter(|arg| matches!(&arg.value.1, T::Type::Reference(true, _)))
203 .cloned(),
204 )
205 }
206 let arguments: Vec<CtxValue> = context.arguments(arguments)?;
207 if is_gas_coin_send_funds {
208 assert_invariant!(arguments.len() == 2, "coin::send_funds should have 2 args");
209 let recipient = arguments.last().unwrap().to_address()?;
210 context.record_gas_coin_transfer(GasCoinTransfer::SendFunds { recipient })?;
211 }
212 let res = context.vm_move_call(function, arguments, trace_builder_opt);
213 trace_utils::trace_move_call_end(trace_builder_opt);
214 res?
215 }
216 T::Command__::TransferObjects(objects, recipient) => {
217 let has_gas_coin_move = objects.iter().any(|arg| {
219 matches!(
220 &arg.value.0,
221 T::Argument__::Use(T::Usage::Move(T::Location::GasCoin))
222 )
223 });
224 if has_gas_coin_move {
225 context.record_gas_coin_transfer(GasCoinTransfer::TransferObjects)?;
226 }
227 let object_tys = objects
228 .iter()
229 .map(|sp!(_, (_, ty))| ty.clone())
230 .collect::<Vec<_>>();
231 let object_values: Vec<CtxValue> = context.arguments(objects)?;
232 let recipient: AccountAddress = context.argument(recipient)?;
233 assert_invariant!(
234 object_values.len() == object_tys.len(),
235 "object values and types mismatch"
236 );
237 trace_utils::trace_transfer(context, trace_builder_opt, &object_values, &object_tys)?;
238 for (object_value, ty) in object_values.into_iter().zip_debug_eq(object_tys) {
239 let recipient = Owner::AddressOwner(recipient.into());
241 context.transfer_object(recipient, ty, object_value)?;
242 }
243 vec![]
244 }
245 T::Command__::SplitCoins(ty, coin, amounts) => {
246 let mut trace_values = vec![];
247 if Mode::TRACK_EXECUTION {
249 args_to_update.push(coin.clone());
250 }
251 let coin_ref: CtxValue = context.argument(coin)?;
252 let amount_values: Vec<u64> = context.arguments(amounts)?;
253 let mut total: u64 = 0;
254 for amount in &amount_values {
255 let Some(new_total) = total.checked_add(*amount) else {
256 return Err(Mode::Error::from_kind(
257 ExecutionErrorKind::CoinBalanceOverflow,
258 ));
259 };
260 total = new_total;
261 }
262 trace_utils::add_move_value_info_from_ctx_value(
263 context,
264 trace_builder_opt,
265 &mut trace_values,
266 &ty,
267 &coin_ref,
268 )?;
269 let coin_value = context.copy_value(&coin_ref)?.coin_ref_value()?;
270 fp_ensure!(
271 coin_value >= total,
272 Mode::Error::new_with_source(
273 ExecutionErrorKind::InsufficientCoinBalance,
274 format!("balance: {coin_value} required: {total}")
275 )
276 );
277 coin_ref.coin_ref_subtract_balance(total)?;
278 let amounts = amount_values
279 .into_iter()
280 .map(|a| context.new_coin(a))
281 .collect::<Result<Vec<_>, _>>()?;
282 trace_utils::trace_split_coins(
283 context,
284 trace_builder_opt,
285 &ty,
286 trace_values,
287 &amounts,
288 total,
289 )?;
290
291 amounts
292 }
293 T::Command__::MergeCoins(ty, target, coins) => {
294 let mut trace_values = vec![];
295 if Mode::TRACK_EXECUTION {
297 args_to_update.push(target.clone());
298 }
299 let target_ref: CtxValue = context.argument(target)?;
300 trace_utils::add_move_value_info_from_ctx_value(
301 context,
302 trace_builder_opt,
303 &mut trace_values,
304 &ty,
305 &target_ref,
306 )?;
307 let coins = context.arguments(coins)?;
308 let amounts = coins
309 .into_iter()
310 .map(|coin| {
311 trace_utils::add_move_value_info_from_ctx_value(
312 context,
313 trace_builder_opt,
314 &mut trace_values,
315 &ty,
316 &coin,
317 )?;
318 context.destroy_coin(coin)
319 })
320 .collect::<Result<Vec<_>, _>>()?;
321 let mut additional: u64 = 0;
322 for amount in amounts {
323 let Some(new_additional) = additional.checked_add(amount) else {
324 return Err(Mode::Error::from_kind(
325 ExecutionErrorKind::CoinBalanceOverflow,
326 ));
327 };
328 additional = new_additional;
329 }
330 let target_value = context.copy_value(&target_ref)?.coin_ref_value()?;
331 fp_ensure!(
332 target_value.checked_add(additional).is_some(),
333 Mode::Error::from_kind(ExecutionErrorKind::CoinBalanceOverflow,)
334 );
335 target_ref.coin_ref_add_balance(additional)?;
336 trace_utils::trace_merge_coins(
337 context,
338 trace_builder_opt,
339 &ty,
340 trace_values,
341 additional,
342 )?;
343 vec![]
344 }
345 T::Command__::MakeMoveVec(ty, items) => {
346 let items: Vec<CtxValue> = context.arguments(items)?;
347 trace_utils::trace_make_move_vec(context, trace_builder_opt, &items, &ty)?;
348 vec![CtxValue::vec_pack(ty, items)?]
349 }
350 T::Command__::Publish(payload, dep_ids, linkage) => {
351 trace_utils::trace_publish_event(trace_builder_opt)?;
352 let DeserializedPackage {
353 deserialized_modules,
354 ..
355 } = context.deserialize_package(payload, &dep_ids)?;
356
357 let original_id = context.publish_and_init_package(
358 deserialized_modules,
359 &dep_ids,
360 linkage,
361 trace_builder_opt,
362 )?;
363
364 if <Mode>::packages_are_predefined() {
365 std::vec![]
367 } else {
368 std::vec![context.new_upgrade_cap(original_id)?]
369 }
370 }
371 T::Command__::Upgrade(payload, dep_ids, current_package_id, upgrade_ticket, linkage) => {
372 trace_utils::trace_upgrade_event(trace_builder_opt)?;
373 let upgrade_ticket = context
374 .argument::<CtxValue>(upgrade_ticket)?
375 .into_upgrade_ticket()?;
376 if current_package_id != upgrade_ticket.package.bytes {
378 return Err(Mode::Error::from_kind(
379 ExecutionErrorKind::PackageUpgradeError {
380 upgrade_error: PackageUpgradeError::PackageIDDoesNotMatch {
381 package_id: current_package_id,
382 ticket_id: upgrade_ticket.package.bytes,
383 },
384 },
385 ));
386 }
387 let DeserializedPackage {
389 deserialized_modules,
390 computed_digest,
391 ..
392 } = context.deserialize_package(payload, &dep_ids)?;
393 let computed_digest = computed_digest.to_vec();
394
395 if computed_digest != upgrade_ticket.digest {
396 return Err(Mode::Error::from_kind(
397 ExecutionErrorKind::PackageUpgradeError {
398 upgrade_error: PackageUpgradeError::DigestDoesNotMatch {
399 digest: computed_digest,
400 },
401 },
402 ));
403 }
404
405 let upgraded_package_id = context.upgrade(
406 deserialized_modules,
407 &dep_ids,
408 current_package_id,
409 upgrade_ticket.policy,
410 linkage,
411 )?;
412
413 vec![context.upgrade_receipt(upgrade_ticket, upgraded_package_id)]
414 }
415 };
416 if Mode::TRACK_EXECUTION {
417 let argument_updates = context.argument_updates(args_to_update)?;
418 let command_result = context.tracked_results(&result, &result_type)?;
419 Mode::finish_command(mode_results, argument_updates, command_result)?;
420 }
421 assert_invariant!(
422 result.len() == drop_values.len(),
423 "result values and drop values mismatch"
424 );
425 context.charge_command(is_move_call, num_args, result.len())?;
426 let result = result
427 .into_iter()
428 .zip_debug_eq(drop_values)
429 .map(|(value, drop)| if !drop { Some(value) } else { None })
430 .collect::<Vec<_>>();
431 context.result(result)?;
432 assert_invariant!(
433 context.gas_charger.move_gas_status().stack_height_current() == 0,
434 "stack height did not end at 0"
435 );
436 Ok(())
437}
438
439struct IndexedExecutionTimings {
441 max_allowed_index: usize,
444 executed_commands: BTreeMap<usize, Duration>,
447 error_command: Option<(usize, Duration)>,
450}
451
452impl IndexedExecutionTimings {
453 fn new(original_command_len: usize) -> Self {
454 let max_allowed_index = original_command_len.saturating_sub(1);
455 Self {
456 max_allowed_index,
457 executed_commands: BTreeMap::new(),
458 error_command: None,
459 }
460 }
461
462 fn executed(&mut self, annotated_index: usize, duration: Duration) {
464 debug_assert!(
465 self.error_command.is_none(),
466 "command executed after an error occurred"
467 );
468 let index = annotated_index.min(self.max_allowed_index);
469 let existing = self
470 .executed_commands
471 .entry(index)
472 .or_insert(Duration::ZERO);
473 *existing = existing.saturating_add(duration);
474 }
475
476 fn error(&mut self, annotated_index: usize, duration: Duration) {
478 debug_assert!(self.error_command.is_none(), "multiple errors recorded");
479 let index = annotated_index.min(self.max_allowed_index);
480 debug_assert!(
481 self.executed_commands
482 .last_key_value()
483 .is_none_or(|(last, _)| *last <= index),
484 "execution timings recorded for command index {:?} after error at index {}",
485 self.executed_commands
486 .last_key_value()
487 .map(|(last, _)| *last),
488 index,
489 );
490
491 let existing_opt = self.executed_commands.remove(&index);
492 let total_duration = existing_opt
493 .unwrap_or(Duration::ZERO)
494 .saturating_add(duration);
495 self.error_command = Some((index, total_duration));
496 }
497
498 fn into_coalesced(self) -> Vec<ExecutionTiming> {
503 let Self {
504 max_allowed_index,
505 executed_commands,
506 error_command,
507 } = self;
508
509 let max_executed_index = executed_commands.keys().last().copied();
510 let error_index = error_command.as_ref().map(|(idx, _)| *idx);
511 let max_used_index = match (max_executed_index, error_index) {
512 (Some(exec), Some(err)) => exec.max(err),
513 (Some(idx), None) | (None, Some(idx)) => idx,
514 (None, None) => return vec![],
515 };
516 debug_assert!(
517 max_used_index <= max_allowed_index,
518 "max used index {} exceeds max allowed index {}",
519 max_used_index,
520 max_allowed_index
521 );
522 let size = max_used_index.saturating_add(1);
523
524 let mut coalesced = vec![ExecutionTiming::Success(Duration::ZERO); size];
529 for (index, duration) in executed_commands {
530 let Some(entry) = coalesced.get_mut(index) else {
531 debug_assert!(
532 false,
533 "failed to initialize coalesced timings at index {}",
534 index
535 );
536 continue;
537 };
538 debug_assert!(matches!(entry, ExecutionTiming::Success(d) if d.is_zero()));
539 *entry = ExecutionTiming::Success(duration);
540 }
541
542 if let Some((index, error_duration)) = error_command {
543 debug_assert!(
544 index == coalesced.len().saturating_sub(1),
545 "error index should be last"
546 );
547 if let Some(entry) = coalesced.get_mut(index) {
548 debug_assert!(matches!(entry, ExecutionTiming::Success(d) if d.is_zero()));
549 *entry = ExecutionTiming::Abort(error_duration);
550 } else {
551 debug_assert!(
552 false,
553 "failed to initialize coalesced timings at index {}",
554 index
555 );
556 };
557 }
558
559 coalesced
560 }
561}