From 4270b948a0ee50fb2f60593bc8c8859ebc4e9a83 Mon Sep 17 00:00:00 2001 From: Martin Raszyk Date: Thu, 17 Oct 2024 12:32:05 +0200 Subject: [PATCH] fix: ic0.call_perform spec --- spec/index.md | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/spec/index.md b/spec/index.md index ee37e789..c195c77b 100644 --- a/spec/index.md +++ b/spec/index.md @@ -1718,7 +1718,7 @@ There must be at most one call to `ic0.call_on_cleanup` between `ic0.call_new` a This deducts `MAX_CYCLES_PER_RESPONSE` cycles from the canister balance and sets them aside for response processing. - If the function returns `0` as the `err_code`, the IC was able to enqueue the call. In this case, the call will either be delivered, returned because the destination canister does not exist or returned because of an out of cycles condition. This also means that exactly one of the reply or reject callbacks will be executed. + If the function returns `0` as the `err_code`, the IC was able to enqueue the call. In this case, the call will either be delivered, returned because the destination canister does not exist, returned due to a lack of resources within the IC, or returned because of an out of cycles condition. This also means that exactly one of the reply or reject callbacks will be executed. If the function returns a non-zero value, the call cannot (and will not be) performed. This can happen due to a lack of resources within the IC, but also if it would reduce the current cycle balance to a level below where the canister would be frozen. @@ -7197,15 +7197,13 @@ ic0.call_peform() : ( err_code : i32 ) = if es.context ∉ {U, CQ, Ry, Rt, CRy, CRt, T} then Trap {cycles_used = es.cycles_used;} if es.pending_call = NoPendingCall then Trap {cycles_used = es.cycles_used;} - es.balance := es.balance - MAX_CYCLES_PER_RESPONSE - - // are we below the freezing threshold? - // Or maybe the system has other reasons to not perform this - if liquid_balance(es) < 0 or system_cannot_do_this_call_now() + // `system_cannot_do_this_call_now` abstracts over resource issues preventing the call from being made + if liquid_balance(es) < MAX_CYCLES_PER_RESPONSE or system_cannot_do_this_call_now() then discard_pending_call() return or + es.balance := es.balance - MAX_CYCLES_PER_RESPONSE es.calls := es.calls · es.pending_call es.pending_call := NoPendingCall return 0 @@ -7213,7 +7211,7 @@ ic0.call_peform() : ( err_code : i32 ) = // helper function discard_pending_call() = if es.pending_call ≠ NoPendingCall then - es.balance := es.balance + MAX_CYCLES_PER_RESPONSE + es.pending_call.transferred_cycles + es.balance := es.balance + es.pending_call.transferred_cycles es.pending_call := NoPendingCall ic0.stable_size() : (page_count : i32) =