Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Deferred calls #4723

Merged
merged 109 commits into from
Dec 4, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
109 commits
Select commit Hold shift + click to select a range
19e89b4
Add structure async call and async call id
AurelienFT May 9, 2024
4407e86
progress
damip May 9, 2024
b6948d0
progress
damip May 9, 2024
bcfaed3
todos
damip May 9, 2024
eb59c84
Improve general behavior
damip May 10, 2024
5315ea4
typo
damip May 10, 2024
d4300c3
reformat registry
damip May 10, 2024
60cbf28
make things compile
damip May 10, 2024
42c47cd
updates
damip May 21, 2024
37a1961
Merge remote-tracking branch 'aurelien/new_asc' into deferred_calls
modship Jul 8, 2024
94016cb
update new interface
modship Jul 8, 2024
d62dbee
deferred calls first step
modship Jul 9, 2024
d18ed1b
rename massa_asc package to massa_deferred_calls
modship Jul 9, 2024
8b3b2da
TDD for ser_deser validation
modship Jul 10, 2024
7d7e850
Test(DeferredRegistrySlotChanges) : Ser/deser
modship Jul 10, 2024
f945386
Test(DeferredRegistryChanges) : Ser/Deser
modship Jul 10, 2024
9814537
update StateChanges Deserializer
modship Jul 10, 2024
90202d3
create macros, put_entry , delete_entry on DB
modship Jul 12, 2024
698a033
refactor + get_total_gas
modship Jul 12, 2024
cfe824a
get_slot_calls && get_slot_base_fee
modship Jul 15, 2024
1a86834
add mod tests / deferred_call_quote
modship Jul 16, 2024
9d74423
get_deferred_call_quote (interface_impl)
modship Jul 16, 2024
e1f2eca
register call
modship Jul 17, 2024
1bcc026
register set gas
modship Jul 17, 2024
14ca8d5
deferred_call_cancel
modship Jul 17, 2024
4a0c7db
execute deferred call, advance_slot, snapshot
modship Jul 24, 2024
ee901f1
execution trace && execution info
modship Jul 24, 2024
693cf0e
delete call from DB
modship Jul 24, 2024
43923d1
slot remaining gas for async msg
modship Jul 25, 2024
e94c57f
in case of fail, refound sender
modship Jul 25, 2024
6297693
get_slot from DeferredCallId
modship Jul 25, 2024
8bbb687
Fix(ser/deser): error and update test
modship Jul 26, 2024
801e65c
Fix(exec_deferred_call) : Fix lock
modship Jul 26, 2024
d59a556
Test(deferred_call) : first execution test
modship Jul 26, 2024
cd44aa1
fix(deferred_call) : remove double delete_call
modship Jul 26, 2024
cc14cdc
Test(abi) : First test for register deferred_call
modship Jul 29, 2024
1668f4d
Test(register) : more assert
modship Jul 30, 2024
24e09fe
Fix(set_total_gas) : fix set / keep
modship Jul 31, 2024
64f6c24
Test(deferred_call) : Register call fail (total_gas)
modship Jul 31, 2024
9a022ef
Fix(execution) : cancel deferred on check target sc + test refound
modship Jul 31, 2024
08bbef2
update wasm file
modship Jul 31, 2024
9fb3c0e
Fix(deferred_call) : fix compute call fee
modship Aug 1, 2024
82938ec
Fix(deferred_call) : some fix
modship Aug 1, 2024
3ce418f
Test(deferred_call) : Add slot base fee
modship Aug 1, 2024
baa1c3a
rename DEFERRED_CALLS_SLOT_PREFIX to DEFERRED_CALLS_PREFIX
modship Aug 14, 2024
87bb3a2
deferred calls ser/deser error msg
modship Aug 14, 2024
1dcab48
deferred_call_id deserializer limit
modship Aug 14, 2024
ac7ccde
models error
modship Aug 14, 2024
ebe33e1
update max_async_gas for async msg
modship Aug 16, 2024
12971b5
remove useless condition
modship Aug 16, 2024
1d61d14
overbooking_fee return Result
modship Aug 16, 2024
8f82d3e
fix operator
modship Aug 16, 2024
6c1caec
add penalty constant
modship Aug 16, 2024
f3331b5
pr comment # 1
modship Aug 26, 2024
9664ca7
Update massa-execution-worker/src/context.rs
modship Aug 26, 2024
a08664e
pr comment # 2
modship Aug 26, 2024
b53c77f
create DeferredCallsConfig
modship Aug 26, 2024
91ac5e3
use config struct
modship Aug 27, 2024
fe347bb
clippy
modship Aug 27, 2024
07863fc
allow unused_macros
modship Aug 27, 2024
5f28f44
fix : massa-bootstrap import
modship Aug 27, 2024
fd69b2e
wrap deferred call execution into a closure
modship Aug 28, 2024
67ad55f
update sc runtime && wasm test file
modship Aug 28, 2024
6815f44
update interface params from bytes to string
modship Aug 29, 2024
db87465
first metrics
modship Aug 29, 2024
d49dda7
Test: Add check on total_gas after execution
modship Aug 29, 2024
f43fdea
Test : Add 2nd call in scenario and check total_gas
modship Aug 30, 2024
5a58f5f
move type Set Update Delete from Ledger to massa_models
modship Sep 2, 2024
3f5f3cc
remove todo
modship Sep 3, 2024
4df7053
add vm allocation gas cost
modship Sep 4, 2024
c8ca2a8
remove todo
modship Sep 5, 2024
3dcd4ab
Fee: calc relu_occupancy_before
modship Sep 5, 2024
b5dae7c
update sc runtime
modship Sep 5, 2024
89c4d1d
add abi gas cost
modship Sep 5, 2024
039e9ba
Fix : total gas booked
modship Sep 6, 2024
562c16f
BugFix && add nb_call_registered
modship Sep 18, 2024
9583fc7
API : Jsonrpc DeferredCallQuote
modship Sep 20, 2024
341db02
API : JsonRpc get_deferred_call_info && list_deferred_calls_by_slot
modship Sep 23, 2024
4f7524d
API : openrpc spec
modship Sep 24, 2024
5b68c00
grpc api
modship Sep 26, 2024
815ed4c
Fix grpc log error
modship Sep 27, 2024
fd9d945
API : Update list call ids by slot
modship Sep 27, 2024
ed88731
rename openrpc method
modship Sep 27, 2024
7cd95b3
Fix : reset_to_snapshot
modship Oct 1, 2024
08be8a9
Fix: reset_to_snapshot
modship Oct 1, 2024
bc225ec
FIX : Api double alloc vm cst
modship Oct 2, 2024
a4e21a3
Fix : compute_fee get_effective_total_gas()
modship Oct 2, 2024
a4f3a6d
Update runtime
modship Oct 2, 2024
34d9975
Fix abi gas cost
modship Oct 2, 2024
9e57b53
Test : compute_call_fee setup && first basic test
modship Oct 2, 2024
6d25ee6
Test : add test for deferred call exists
modship Oct 3, 2024
2aea5a0
clippy
modship Oct 3, 2024
46fccbe
Fix : fee calculation
modship Oct 7, 2024
8ebf906
add storage cost for deferred call
modship Oct 16, 2024
2af6725
Merge remote-tracking branch 'origin/mainnet_2_3' into deferred_calls
modship Oct 18, 2024
88ecb0d
Storage: Refund sender (storage cost for params)
modship Oct 18, 2024
ff28375
Test : deferred call quote
modship Oct 21, 2024
64f4cdb
update runtime
modship Oct 21, 2024
95e1bc8
Fix all feature && update runtime
modship Oct 21, 2024
c8e45d8
fix doc
modship Oct 21, 2024
2453a80
Update storage cost for deferred calls and set MAX_FUNCTION_NAME_LENG…
modship Oct 30, 2024
1ef9d71
Fix comment
modship Oct 30, 2024
dd31e59
condom limits default with gas_calibration
modship Nov 7, 2024
49180c3
Merge branch 'mainnet_2_3' into deferred_calls
modship Nov 27, 2024
7afc88a
fix import
modship Nov 28, 2024
b20b556
update fee compute
modship Nov 29, 2024
f33057f
Test : fix SC address
modship Dec 3, 2024
de3cba3
Merge remote-tracking branch 'origin/mainnet_2_3' into deferred_calls
modship Dec 3, 2024
df245a8
Merge remote-tracking branch 'origin/mainnet_2_3' into deferred_calls
modship Dec 3, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
227 changes: 118 additions & 109 deletions massa-execution-worker/src/execution.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1248,123 +1248,135 @@ impl ExecutionState {
let mut result = DeferredCallExecutionResult::new(&call);

let snapshot = {
let context = context_guard!(self);
let mut context = context_guard!(self);

// refund the sender for the storage costs
let amount = self
.config
.storage_costs_constants
.ledger_cost_per_byte
.saturating_mul_u64(call.parameters.len() as u64);
context.transfer_coins(None, Some(call.sender_address), amount, false)?;
modship marked this conversation as resolved.
Show resolved Hide resolved

context.get_snapshot()
};

let deferred_call_execution = || {
let bytecode = {
// acquire write access to the context
let mut context = context_guard!(self);
if call.cancelled {
Ok(result)
} else {
let deferred_call_execution = || {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe extract this to another function for better clarity

let bytecode = {
// acquire write access to the context
let mut context = context_guard!(self);

// Set the call stack
// This needs to be defined before anything can fail, so that the emitted event contains the right stack
context.stack = vec![
ExecutionStackElement {
address: call.sender_address,
coins: Default::default(),
owned_addresses: vec![call.sender_address],
operation_datastore: None,
},
ExecutionStackElement {
address: call.target_address,
coins: call.coins,
owned_addresses: vec![call.target_address],
operation_datastore: None,
},
];

// Ensure that the target address is an SC address
// Ensure that the target address exists
context.check_target_sc_address(call.target_address)?;

// credit coins to the target address
if let Err(err) =
context.transfer_coins(None, Some(call.target_address), call.coins, false)
{
// coin crediting failed: reset context to snapshot and reimburse sender
return Err(ExecutionError::DeferredCallsError(format!(
"could not credit coins to target of deferred call execution: {}",
err
)));
}
// Set the call stack
// This needs to be defined before anything can fail, so that the emitted event contains the right stack
context.stack = vec![
ExecutionStackElement {
address: call.sender_address,
coins: Default::default(),
owned_addresses: vec![call.sender_address],
operation_datastore: None,
},
ExecutionStackElement {
address: call.target_address,
coins: call.coins,
owned_addresses: vec![call.target_address],
operation_datastore: None,
},
];

// Ensure that the target address is an SC address
// Ensure that the target address exists
context.check_target_sc_address(call.target_address)?;

// credit coins to the target address
if let Err(err) =
context.transfer_coins(None, Some(call.target_address), call.coins, false)
{
// coin crediting failed: reset context to snapshot and reimburse sender
return Err(ExecutionError::DeferredCallsError(format!(
"could not credit coins to target of deferred call execution: {}",
err
)));
}

// quit if there is no function to be called
if call.target_function.is_empty() {
return Err(ExecutionError::DeferredCallsError(
"no function to call in the deferred call".to_string(),
));
}
// quit if there is no function to be called
if call.target_function.is_empty() {
return Err(ExecutionError::DeferredCallsError(
"no function to call in the deferred call".to_string(),
));
}

// Load bytecode. Assume empty bytecode if not found.
context
.get_bytecode(&call.target_address)
.ok_or(ExecutionError::DeferredCallsError(
"no bytecode found".to_string(),
))?
.0
};
// Load bytecode. Assume empty bytecode if not found.
context
.get_bytecode(&call.target_address)
.ok_or(ExecutionError::DeferredCallsError(
"no bytecode found".to_string(),
))?
.0
};

let module = self.module_cache.write().load_module(
&bytecode,
call.get_effective_gas(self.config.deferred_calls_config.call_cst_gas_cost),
)?;
let response = massa_sc_runtime::run_function(
&*self.execution_interface,
module,
&call.target_function,
&call.parameters,
call.get_effective_gas(self.config.deferred_calls_config.call_cst_gas_cost),
self.config.gas_costs.clone(),
self.config.condom_limits.clone(),
);
let module = self.module_cache.write().load_module(
&bytecode,
call.get_effective_gas(self.config.deferred_calls_config.call_cst_gas_cost),
)?;
let response = massa_sc_runtime::run_function(
&*self.execution_interface,
module,
&call.target_function,
&call.parameters,
call.get_effective_gas(self.config.deferred_calls_config.call_cst_gas_cost),
self.config.gas_costs.clone(),
self.config.condom_limits.clone(),
);

match response {
Ok(res) => {
self.module_cache
.write()
.set_init_cost(&bytecode, res.init_gas_cost);
#[cfg(feature = "execution-trace")]
{
result.traces =
Some((res.trace.into_iter().map(|t| t.into()).collect(), true));
}
// #[cfg(feature = "execution-info")]
// {
// result.success = true;
// }
result.success = true;
Ok(result)
}
Err(error) => {
if let VMError::ExecutionError { init_gas_cost, .. } = error {
match response {
Ok(res) => {
self.module_cache
.write()
.set_init_cost(&bytecode, init_gas_cost);
.set_init_cost(&bytecode, res.init_gas_cost);
#[cfg(feature = "execution-trace")]
{
result.traces =
Some((res.trace.into_iter().map(|t| t.into()).collect(), true));
}
// #[cfg(feature = "execution-info")]
// {
// result.success = true;
// }
result.success = true;
Ok(result)
}
Err(error) => {
if let VMError::ExecutionError { init_gas_cost, .. } = error {
self.module_cache
.write()
.set_init_cost(&bytecode, init_gas_cost);
}
// execution failed: reset context to snapshot and reimburse sender
Err(ExecutionError::VMError {
context: "Deferred Call".to_string(),
error,
})
}
// execution failed: reset context to snapshot and reimburse sender
Err(ExecutionError::VMError {
context: "Deferred Call".to_string(),
error,
})
}
}
};
};

// execute the deferred call
let execution_result = deferred_call_execution();
// execute the deferred call
let execution_result = deferred_call_execution();

// if the execution failed, reset the context to the snapshot
if let Err(err) = &execution_result {
let mut context = context_guard!(self);
context.reset_to_snapshot(snapshot, err.clone());
context.deferred_call_fail_exec(id, &call);
self.massa_metrics.inc_deferred_calls_failed();
} else {
self.massa_metrics.inc_deferred_calls_executed();
// if the execution failed, reset the context to the snapshot
if let Err(err) = &execution_result {
let mut context = context_guard!(self);
context.reset_to_snapshot(snapshot, err.clone());
context.deferred_call_fail_exec(id, &call);
self.massa_metrics.inc_deferred_calls_failed();
} else {
self.massa_metrics.inc_deferred_calls_executed();
}
execution_result
}

execution_result
}
/// Executes a full slot (with or without a block inside) without causing any changes to the state,
/// just yielding the execution output.
Expand Down Expand Up @@ -1409,19 +1421,16 @@ impl ExecutionState {
// Deferred calls
let calls = execution_context.deferred_calls_advance_slot(*slot);

self.massa_metrics
.set_deferred_calls_total_gas(calls.effective_total_gas);

// Apply the created execution context for slot execution
*context_guard!(self) = execution_context;

for (id, call) in calls.slot_calls {
if call.cancelled {
// Skip cancelled calls
continue;
}
let cancelled = call.cancelled;
match self.execute_deferred_call(&id, call) {
Ok(_exec) => {
if cancelled {
continue;
}
info!("executed deferred call: {:?}", id);
cfg_if::cfg_if! {
if #[cfg(feature = "execution-trace")] {
Expand Down
17 changes: 8 additions & 9 deletions massa-execution-worker/src/speculative_deferred_calls.rs
Original file line number Diff line number Diff line change
Expand Up @@ -244,6 +244,8 @@ impl SpeculativeDeferredCallRegistry {

slot_calls.effective_total_gas = total_gas_after;

massa_metrics::set_deferred_calls_total_gas(slot_calls.effective_total_gas);

// delete call in the current slot
let mut nb_call_to_execute = 0;
for (id, call) in &slot_calls.slot_calls {
Expand Down Expand Up @@ -494,15 +496,7 @@ impl SpeculativeDeferredCallRegistry {
)?;

// Storage cost for the parameters
let storage_cost = self
.config
.ledger_cost_per_byte
.checked_mul_u64(params_size)
.ok_or_else(|| {
ExecutionError::DeferredCallsError(
"overflow when calculating storage cost".to_string(),
)
})?;
let storage_cost = self.get_storage_cost_by_size(params_size);

// return the fee
Ok(integral_fee
Expand All @@ -511,6 +505,11 @@ impl SpeculativeDeferredCallRegistry {
.saturating_add(storage_cost))
}

/// Get the storage cost for a given size
pub fn get_storage_cost_by_size(&self, size: u64) -> Amount {
self.config.ledger_cost_per_byte.saturating_mul_u64(size)
}

/// Register a new call
/// Returns the call id
/// # Arguments
Expand Down
Loading
Loading