use move_binary_format::errors::PartialVMResult;
use move_core_types::{
gas_algebra::{AbstractMemorySize, InternalGas, NumArgs, NumBytes},
language_storage::ModuleId,
};
use move_vm_profiler::GasProfiler;
use move_vm_types::{
gas::{GasMeter, SimpleInstruction},
loaded_data::runtime_types::Type,
views::{TypeView, ValueView},
};
use sui_types::gas_model::{
gas_predicates::{native_function_threshold_exceeded, use_legacy_abstract_size},
tables::{GasStatus, REFERENCE_SIZE, STRUCT_SIZE, VEC_SIZE},
};
pub struct SuiGasMeter<'g>(pub &'g mut GasStatus);
fn get_simple_instruction_stack_change(
instr: SimpleInstruction,
) -> (u64, u64, AbstractMemorySize, AbstractMemorySize) {
use SimpleInstruction::*;
match instr {
Nop | Ret => (0, 0, 0.into(), 0.into()),
BrTrue | BrFalse => (1, 0, Type::Bool.size(), 0.into()),
Branch => (0, 0, 0.into(), 0.into()),
LdU8 => (0, 1, 0.into(), Type::U8.size()),
LdU16 => (0, 1, 0.into(), Type::U16.size()),
LdU32 => (0, 1, 0.into(), Type::U32.size()),
LdU64 => (0, 1, 0.into(), Type::U64.size()),
LdU128 => (0, 1, 0.into(), Type::U128.size()),
LdU256 => (0, 1, 0.into(), Type::U256.size()),
LdTrue | LdFalse => (0, 1, 0.into(), Type::Bool.size()),
FreezeRef => (1, 1, REFERENCE_SIZE, REFERENCE_SIZE),
ImmBorrowLoc | MutBorrowLoc => (0, 1, 0.into(), REFERENCE_SIZE),
ImmBorrowField | MutBorrowField | ImmBorrowFieldGeneric | MutBorrowFieldGeneric => {
(1, 1, REFERENCE_SIZE, REFERENCE_SIZE)
}
CastU8 => (1, 1, Type::U8.size(), Type::U8.size()),
CastU16 => (1, 1, Type::U8.size(), Type::U16.size()),
CastU32 => (1, 1, Type::U8.size(), Type::U32.size()),
CastU64 => (1, 1, Type::U8.size(), Type::U64.size()),
CastU128 => (1, 1, Type::U8.size(), Type::U128.size()),
CastU256 => (1, 1, Type::U8.size(), Type::U256.size()),
Add | Sub | Mul | Mod | Div => (2, 1, Type::U8.size() + Type::U8.size(), Type::U256.size()),
BitOr | BitAnd | Xor => (2, 1, Type::U8.size() + Type::U8.size(), Type::U256.size()),
Shl | Shr => (2, 1, Type::U8.size() + Type::U8.size(), Type::U256.size()),
Or | And => (
2,
1,
Type::Bool.size() + Type::Bool.size(),
Type::Bool.size(),
),
Lt | Gt | Le | Ge => (2, 1, Type::U8.size() + Type::U8.size(), Type::Bool.size()),
Not => (1, 1, Type::Bool.size(), Type::Bool.size()),
Abort => (1, 0, Type::U64.size(), 0.into()),
}
}
impl GasMeter for SuiGasMeter<'_> {
fn charge_simple_instr(&mut self, instr: SimpleInstruction) -> PartialVMResult<()> {
let (pops, pushes, pop_size, push_size) = get_simple_instruction_stack_change(instr);
self.0
.charge(1, pushes, pops, push_size.into(), pop_size.into())
}
fn charge_pop(&mut self, popped_val: impl ValueView) -> PartialVMResult<()> {
self.0
.charge(1, 0, 1, 0, abstract_memory_size(self.0, popped_val).into())
}
fn charge_native_function(
&mut self,
amount: InternalGas,
ret_vals: Option<impl ExactSizeIterator<Item = impl ValueView>>,
) -> PartialVMResult<()> {
let pushes = ret_vals
.as_ref()
.map(|ret_vals| ret_vals.len())
.unwrap_or(0) as u64;
let size_increase = ret_vals
.map(|ret_vals| {
ret_vals.fold(AbstractMemorySize::zero(), |acc, elem| {
acc + abstract_memory_size(self.0, elem)
})
})
.unwrap_or_else(AbstractMemorySize::zero);
self.0.record_native_call();
if native_function_threshold_exceeded(self.0.gas_model_version, self.0.num_native_calls) {
self.0
.charge(amount.into(), pushes, 0, size_increase.into(), 0)
} else {
self.0.charge(0, pushes, 0, size_increase.into(), 0)?;
self.0.deduct_gas(amount)
}
}
fn charge_native_function_before_execution(
&mut self,
_ty_args: impl ExactSizeIterator<Item = impl TypeView>,
args: impl ExactSizeIterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
let pops = args.len() as u64;
let stack_reduction_size = args.fold(AbstractMemorySize::new(pops), |acc, elem| {
acc + abstract_memory_size(self.0, elem)
});
self.0.charge(1, 0, pops, 0, stack_reduction_size.into())
}
fn charge_call(
&mut self,
_module_id: &ModuleId,
_func_name: &str,
args: impl ExactSizeIterator<Item = impl ValueView>,
_num_locals: NumArgs,
) -> PartialVMResult<()> {
let pops = args.len() as u64;
let stack_reduction_size = args.fold(AbstractMemorySize::new(0), |acc, elem| {
acc + abstract_memory_size(self.0, elem)
});
self.0.charge(1, 0, pops, 0, stack_reduction_size.into())
}
fn charge_call_generic(
&mut self,
_module_id: &ModuleId,
_func_name: &str,
_ty_args: impl ExactSizeIterator<Item = impl TypeView>,
args: impl ExactSizeIterator<Item = impl ValueView>,
_num_locals: NumArgs,
) -> PartialVMResult<()> {
let pops = args.len() as u64;
let stack_reduction_size = args.fold(AbstractMemorySize::new(0), |acc, elem| {
acc + abstract_memory_size(self.0, elem)
});
self.0.charge(1, 0, pops, 0, stack_reduction_size.into())
}
fn charge_ld_const(&mut self, size: NumBytes) -> PartialVMResult<()> {
self.0.charge(1, 1, 0, u64::from(size), 0)
}
fn charge_ld_const_after_deserialization(
&mut self,
_val: impl ValueView,
) -> PartialVMResult<()> {
Ok(())
}
fn charge_copy_loc(&mut self, val: impl ValueView) -> PartialVMResult<()> {
self.0
.charge(1, 1, 0, abstract_memory_size(self.0, val).into(), 0)
}
fn charge_move_loc(&mut self, val: impl ValueView) -> PartialVMResult<()> {
self.0
.charge(1, 1, 0, abstract_memory_size(self.0, val).into(), 0)
}
fn charge_store_loc(&mut self, val: impl ValueView) -> PartialVMResult<()> {
self.0
.charge(1, 0, 1, 0, abstract_memory_size(self.0, val).into())
}
fn charge_pack(
&mut self,
_is_generic: bool,
args: impl ExactSizeIterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
let num_fields = args.len() as u64;
self.0.charge(1, 1, num_fields, STRUCT_SIZE.into(), 0)
}
fn charge_unpack(
&mut self,
_is_generic: bool,
args: impl ExactSizeIterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
let num_fields = args.len() as u64;
self.0.charge(1, num_fields, 1, 0, STRUCT_SIZE.into())
}
fn charge_variant_switch(&mut self, val: impl ValueView) -> PartialVMResult<()> {
self.0
.charge(1, 0, 1, 0, abstract_memory_size(self.0, val).into())
}
fn charge_read_ref(&mut self, ref_val: impl ValueView) -> PartialVMResult<()> {
self.0.charge(
1,
1,
1,
abstract_memory_size(self.0, ref_val).into(),
REFERENCE_SIZE.into(),
)
}
fn charge_write_ref(
&mut self,
new_val: impl ValueView,
old_val: impl ValueView,
) -> PartialVMResult<()> {
self.0.charge(
1,
1,
2,
abstract_memory_size(self.0, new_val).into(),
abstract_memory_size(self.0, old_val).into(),
)
}
fn charge_eq(&mut self, lhs: impl ValueView, rhs: impl ValueView) -> PartialVMResult<()> {
let size_reduction = abstract_memory_size_with_traversal(self.0, lhs)
+ abstract_memory_size_with_traversal(self.0, rhs);
self.0.charge(
1,
1,
2,
(Type::Bool.size() + size_reduction).into(),
size_reduction.into(),
)
}
fn charge_neq(&mut self, lhs: impl ValueView, rhs: impl ValueView) -> PartialVMResult<()> {
let size_reduction = abstract_memory_size_with_traversal(self.0, lhs)
+ abstract_memory_size_with_traversal(self.0, rhs);
let size_increase = if traverse_refs(self.0.gas_model_version) {
Type::Bool.size() + size_reduction
} else {
Type::Bool.size()
};
self.0
.charge(1, 1, 2, size_increase.into(), size_reduction.into())
}
fn charge_vec_pack<'a>(
&mut self,
_ty: impl TypeView + 'a,
args: impl ExactSizeIterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
let num_args = args.len() as u64;
self.0.charge(1, 1, num_args, VEC_SIZE.into(), 0)
}
fn charge_vec_len(&mut self, _ty: impl TypeView) -> PartialVMResult<()> {
self.0
.charge(1, 1, 1, Type::U64.size().into(), REFERENCE_SIZE.into())
}
fn charge_vec_borrow(
&mut self,
_is_mut: bool,
_ty: impl TypeView,
_is_success: bool,
) -> PartialVMResult<()> {
self.0.charge(
1,
1,
2,
REFERENCE_SIZE.into(),
(REFERENCE_SIZE + Type::U64.size()).into(),
)
}
fn charge_vec_push_back(
&mut self,
_ty: impl TypeView,
_val: impl ValueView,
) -> PartialVMResult<()> {
self.0.charge(1, 0, 2, 0, REFERENCE_SIZE.into())
}
fn charge_vec_pop_back(
&mut self,
_ty: impl TypeView,
_val: Option<impl ValueView>,
) -> PartialVMResult<()> {
self.0.charge(1, 1, 1, 0, REFERENCE_SIZE.into())
}
fn charge_vec_unpack(
&mut self,
_ty: impl TypeView,
expect_num_elements: NumArgs,
_elems: impl ExactSizeIterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
let pushes = u64::from(expect_num_elements);
self.0.charge(1, pushes, 1, 0, VEC_SIZE.into())
}
fn charge_vec_swap(&mut self, _ty: impl TypeView) -> PartialVMResult<()> {
let size_decrease = REFERENCE_SIZE + Type::U64.size() + Type::U64.size();
self.0.charge(1, 1, 1, 0, size_decrease.into())
}
fn charge_drop_frame(
&mut self,
_locals: impl Iterator<Item = impl ValueView>,
) -> PartialVMResult<()> {
Ok(())
}
fn remaining_gas(&self) -> InternalGas {
if !self.0.charge {
return InternalGas::new(u64::MAX);
}
self.0.gas_left
}
fn get_profiler_mut(&mut self) -> Option<&mut GasProfiler> {
self.0.profiler.as_mut()
}
fn set_profiler(&mut self, profiler: GasProfiler) {
self.0.profiler = Some(profiler);
}
}
fn abstract_memory_size(status: &GasStatus, val: impl ValueView) -> AbstractMemorySize {
if use_legacy_abstract_size(status.gas_model_version) {
val.legacy_abstract_memory_size()
} else {
val.abstract_memory_size(false)
}
}
fn abstract_memory_size_with_traversal(
status: &GasStatus,
val: impl ValueView,
) -> AbstractMemorySize {
if use_legacy_abstract_size(status.gas_model_version) {
val.legacy_abstract_memory_size()
} else {
val.abstract_memory_size(traverse_refs(status.gas_model_version))
}
}
fn traverse_refs(gas_model_version: u64) -> bool {
gas_model_version > 9
}