sui_adapter_v0/gas_charger.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355
// Copyright (c) 2021, Facebook, Inc. and its affiliates
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0
pub use checked::*;
#[sui_macros::with_checked_arithmetic]
pub mod checked {
use crate::sui_types::gas::SuiGasStatusAPI;
use crate::temporary_store::TemporaryStore;
use sui_protocol_config::ProtocolConfig;
use sui_types::gas::{deduct_gas, GasCostSummary, SuiGasStatus};
use sui_types::gas_model::gas_predicates::dont_charge_budget_on_storage_oog;
use sui_types::{
base_types::{ObjectID, ObjectRef},
digests::TransactionDigest,
error::ExecutionError,
gas_model::tables::GasStatus,
is_system_package,
object::Data,
storage::{DeleteKindWithOldVersion, WriteKind},
};
use tracing::trace;
/// Tracks all gas operations for a single transaction.
/// This is the main entry point for gas accounting.
/// All the information about gas is stored in this object.
/// The objective here is two-fold:
/// 1- Isolate al version info into a single entry point. This file and the other gas
/// related files are the only one that check for gas version.
/// 2- Isolate all gas accounting into a single implementation. Gas objects are not
/// passed around, and they are retrieved from this instance.
#[derive(Debug)]
pub struct GasCharger {
tx_digest: TransactionDigest,
gas_model_version: u64,
gas_coins: Vec<ObjectRef>,
// this is the first gas coin in `gas_coins` and the one that all others will
// be smashed into. It can be None for system transactions when `gas_coins` is empty.
smashed_gas_coin: Option<ObjectID>,
gas_status: SuiGasStatus,
}
impl GasCharger {
pub fn new(
tx_digest: TransactionDigest,
gas_coins: Vec<ObjectRef>,
gas_status: SuiGasStatus,
protocol_config: &ProtocolConfig,
) -> Self {
let gas_model_version = protocol_config.gas_model_version();
Self {
tx_digest,
gas_model_version,
gas_coins,
smashed_gas_coin: None,
gas_status,
}
}
pub fn new_unmetered(tx_digest: TransactionDigest) -> Self {
Self {
tx_digest,
gas_model_version: 6, // pick any of the latest, it should not matter
gas_coins: vec![],
smashed_gas_coin: None,
gas_status: SuiGasStatus::new_unmetered(),
}
}
// TODO: there is only one caller to this function that should not exist otherwise.
// Explore way to remove it.
pub(crate) fn gas_coins(&self) -> &[ObjectRef] {
&self.gas_coins
}
// Return the logical gas coin for this transactions or None if no gas coin was present
// (system transactions).
pub fn gas_coin(&self) -> Option<ObjectID> {
self.smashed_gas_coin
}
pub fn gas_budget(&self) -> u64 {
self.gas_status.gas_budget()
}
pub fn unmetered_storage_rebate(&self) -> u64 {
self.gas_status.unmetered_storage_rebate()
}
pub fn no_charges(&self) -> bool {
self.gas_status.gas_used() == 0
&& self.gas_status.storage_rebate() == 0
&& self.gas_status.storage_gas_units() == 0
}
pub fn is_unmetered(&self) -> bool {
self.gas_status.is_unmetered()
}
pub fn move_gas_status(&self) -> &GasStatus {
self.gas_status.move_gas_status()
}
pub fn move_gas_status_mut(&mut self) -> &mut GasStatus {
self.gas_status.move_gas_status_mut()
}
pub fn into_gas_status(self) -> SuiGasStatus {
self.gas_status
}
pub fn summary(&self) -> GasCostSummary {
self.gas_status.summary()
}
// This function is called when the transaction is about to be executed.
// It will smash all gas coins into a single one and set the logical gas coin
// to be the first one in the list.
// After this call, `gas_coin` will return it id of the gas coin.
// This function panics if errors are found while operation on the gas coins.
// Transaction and certificate input checks must have insured that all gas coins
// are correct.
pub fn smash_gas(&mut self, temporary_store: &mut TemporaryStore<'_>) {
let gas_coin_count = self.gas_coins.len();
if gas_coin_count == 0 || (gas_coin_count == 1 && self.gas_coins[0].0 == ObjectID::ZERO)
{
return; // self.smashed_gas_coin is None
}
// set the first coin to be the transaction only gas coin.
// All others will be smashed into this one.
let gas_coin_id = self.gas_coins[0].0;
self.smashed_gas_coin = Some(gas_coin_id);
if gas_coin_count == 1 {
return;
}
// sum the value of all gas coins
let new_balance = self
.gas_coins
.iter()
.map(|obj_ref| {
let obj = temporary_store.objects().get(&obj_ref.0).unwrap();
let Data::Move(move_obj) = &obj.data else {
return Err(ExecutionError::invariant_violation(
"Provided non-gas coin object as input for gas!",
));
};
if !move_obj.type_().is_gas_coin() {
return Err(ExecutionError::invariant_violation(
"Provided non-gas coin object as input for gas!",
));
}
Ok(move_obj.get_coin_value_unsafe())
})
.collect::<Result<Vec<u64>, ExecutionError>>()
// transaction and certificate input checks must have insured that all gas coins
// are valid
.unwrap_or_else(|_| {
panic!(
"Invariant violation: non-gas coin object as input for gas in txn {}",
self.tx_digest
)
})
.iter()
.sum();
let mut primary_gas_object = temporary_store
.objects()
.get(&gas_coin_id)
// unwrap should be safe because we checked that this exists in `self.objects()` above
.unwrap_or_else(|| {
panic!(
"Invariant violation: gas coin not found in store in txn {}",
self.tx_digest
)
})
.clone();
// delete all gas objects except the primary_gas_object
for (id, version, _digest) in &self.gas_coins[1..] {
debug_assert_ne!(*id, primary_gas_object.id());
temporary_store.delete_object(id, DeleteKindWithOldVersion::Normal(*version));
}
primary_gas_object
.data
.try_as_move_mut()
// unwrap should be safe because we checked that the primary gas object was a coin object above.
.unwrap_or_else(|| {
panic!(
"Invariant violation: invalid coin object in txn {}",
self.tx_digest
)
})
.set_coin_value_unsafe(new_balance);
temporary_store.write_object(primary_gas_object, WriteKind::Mutate);
}
//
// Gas charging operations
//
pub fn track_storage_mutation(
&mut self,
object_id: ObjectID,
new_size: usize,
storage_rebate: u64,
) -> u64 {
self.gas_status
.track_storage_mutation(object_id, new_size, storage_rebate)
}
pub fn reset_storage_cost_and_rebate(&mut self) {
self.gas_status.reset_storage_cost_and_rebate();
}
pub fn charge_publish_package(&mut self, size: usize) -> Result<(), ExecutionError> {
self.gas_status.charge_publish_package(size)
}
pub fn charge_input_objects(
&mut self,
temporary_store: &TemporaryStore<'_>,
) -> Result<(), ExecutionError> {
let objects = temporary_store.objects();
// TODO: Charge input object count.
let _object_count = objects.len();
// Charge bytes read
let total_size = temporary_store
.objects()
.iter()
// don't charge for loading Sui Framework or Move stdlib
.filter(|(id, _)| !is_system_package(**id))
.map(|(_, obj)| obj.object_size_for_gas_metering())
.sum();
self.gas_status.charge_storage_read(total_size)
}
/// Resets any mutations, deletions, and events recorded in the store, as well as any storage costs and
/// rebates, then Re-runs gas smashing. Effects on store are now as if we were about to begin execution
pub fn reset(&mut self, temporary_store: &mut TemporaryStore<'_>) {
temporary_store.drop_writes();
self.gas_status.reset_storage_cost_and_rebate();
self.smash_gas(temporary_store);
}
/// Entry point for gas charging.
/// 1. Compute tx storage gas costs and tx storage rebates, update storage_rebate field of
/// mutated objects
/// 2. Deduct computation gas costs and storage costs, credit storage rebates.
/// The happy path of this function follows (1) + (2) and is fairly simple.
/// Most of the complexity is in the unhappy paths:
/// - if execution aborted before calling this function, we have to dump all writes +
/// re-smash gas, then charge for storage
/// - if we run out of gas while charging for storage, we have to dump all writes +
/// re-smash gas, then charge for storage again
pub fn charge_gas<T>(
&mut self,
temporary_store: &mut TemporaryStore<'_>,
execution_result: &mut Result<T, ExecutionError>,
) -> GasCostSummary {
// at this point, we have done *all* charging for computation,
// but have not yet set the storage rebate or storage gas units
debug_assert!(self.gas_status.storage_rebate() == 0);
debug_assert!(self.gas_status.storage_gas_units() == 0);
if self.smashed_gas_coin.is_some() {
// bucketize computation cost
if let Err(err) = self.gas_status.bucketize_computation() {
if execution_result.is_ok() {
*execution_result = Err(err);
}
}
// On error we need to dump writes, deletes, etc before charging storage gas
if execution_result.is_err() {
self.reset(temporary_store);
}
}
// compute and collect storage charges
temporary_store.ensure_gas_and_input_mutated(self);
temporary_store.collect_storage_and_rebate(self);
if self.smashed_gas_coin.is_some() {
#[skip_checked_arithmetic]
trace!(target: "replay_gas_info", "Gas smashing has occurred for this transaction");
}
// system transactions (None smashed_gas_coin) do not have gas and so do not charge
// for storage, however they track storage values to check for conservation rules
if let Some(gas_object_id) = self.smashed_gas_coin {
if dont_charge_budget_on_storage_oog(self.gas_model_version) {
self.handle_storage_and_rebate_v2(temporary_store, execution_result)
} else {
self.handle_storage_and_rebate_v1(temporary_store, execution_result)
}
let cost_summary = self.gas_status.summary();
let gas_used = cost_summary.net_gas_usage();
let mut gas_object = temporary_store.read_object(&gas_object_id).unwrap().clone();
deduct_gas(&mut gas_object, gas_used);
#[skip_checked_arithmetic]
trace!(gas_used, gas_obj_id =? gas_object.id(), gas_obj_ver =? gas_object.version(), "Updated gas object");
temporary_store.write_object(gas_object, WriteKind::Mutate);
cost_summary
} else {
GasCostSummary::default()
}
}
fn handle_storage_and_rebate_v1<T>(
&mut self,
temporary_store: &mut TemporaryStore<'_>,
execution_result: &mut Result<T, ExecutionError>,
) {
if let Err(err) = self.gas_status.charge_storage_and_rebate() {
self.reset(temporary_store);
self.gas_status.adjust_computation_on_out_of_gas();
temporary_store.ensure_gas_and_input_mutated(self);
temporary_store.collect_rebate(self);
if execution_result.is_ok() {
*execution_result = Err(err);
}
}
}
fn handle_storage_and_rebate_v2<T>(
&mut self,
temporary_store: &mut TemporaryStore<'_>,
execution_result: &mut Result<T, ExecutionError>,
) {
if let Err(err) = self.gas_status.charge_storage_and_rebate() {
// we run out of gas charging storage, reset and try charging for storage again.
// Input objects are touched and so they have a storage cost
self.reset(temporary_store);
temporary_store.ensure_gas_and_input_mutated(self);
temporary_store.collect_storage_and_rebate(self);
if let Err(err) = self.gas_status.charge_storage_and_rebate() {
// we run out of gas attempting to charge for the input objects exclusively,
// deal with this edge case by not charging for storage
self.reset(temporary_store);
self.gas_status.adjust_computation_on_out_of_gas();
temporary_store.ensure_gas_and_input_mutated(self);
temporary_store.collect_rebate(self);
if execution_result.is_ok() {
*execution_result = Err(err);
}
} else if execution_result.is_ok() {
*execution_result = Err(err);
}
}
}
}
}