mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-24 17:11:05 +00:00
[contracts] Port host functions to Weight V2 and storage deposit limit (#13565)
* added [unstable][seal2] call() * updated test to cover new seal_call proof_limit * docs updated * add [seal2][unstable] instantiate() and test * add [seal2][unstable] weight_to_fee() + docs and test * add [seal2][unstable] gas_left() + docs and test * update benchmarks * add DefaultDepositLimit to pallet Config * specify deposit limit for nested call add test for nested call deposit limit save: separate deposit limit for nested calls * specify deposit limit for nested instantiate save: works with test cleaned up debugging outputs * update benchmarks * added missing fixtures * fix benches * pass explicit deposit limit to storage bench * explicit deposit limit for another set_storage bench * add more deposit limit for storage benches * moving to simplified benchmarks * moved to simplified benchmarks * fix seal_weight_to_fee bench * fix seal_instantiate benchmark * doc typo fix * default dl for benchmarking more dl for tests dl for tests to max deposit_limit fix in instantiate bench fix instantiate bench fix instantiate benchmark fix instantiate bench again remove dbg fix seal bench again fixing it still seal_instantiate zero deposit less runs to check if deposit enough try try 2 try 3 try 4 * max_runtime_mem to Schedule limits * add default deposit limit fallback check to test * weight params renaming * fmt * Update frame/contracts/src/benchmarking/mod.rs Co-authored-by: PG Herveou <pgherveou@gmail.com> * prettify inputs in tests * typestate param refactored --------- Co-authored-by: PG Herveou <pgherveou@gmail.com>
This commit is contained in:
@@ -1,8 +1,8 @@
|
||||
;; This expects [account_id, gas_limit] as input and calls the account_id with the supplied gas_limit.
|
||||
;; This expects [account_id, ref_time, proof_size] as input and calls the account_id with the supplied 2D Weight limit.
|
||||
;; It returns the result of the call as output data.
|
||||
(module
|
||||
(import "seal0" "seal_input" (func $seal_input (param i32 i32)))
|
||||
(import "seal0" "seal_call" (func $seal_call (param i32 i32 i64 i32 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "seal2" "call" (func $seal_call (param i32 i32 i64 i64 i32 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "seal0" "seal_return" (func $seal_return (param i32 i32 i32)))
|
||||
(import "env" "memory" (memory 1 1))
|
||||
|
||||
@@ -13,24 +13,25 @@
|
||||
(func (export "deploy"))
|
||||
|
||||
(func (export "call")
|
||||
;; Receive the encoded call + gas_limit
|
||||
;; Receive the encoded account_id, ref_time, proof_size
|
||||
(call $seal_input
|
||||
(i32.const 4) ;; Pointer to the input buffer
|
||||
(i32.const 0) ;; Size of the length buffer
|
||||
(i32.const 0) ;; Pointer to the length of the input buffer
|
||||
)
|
||||
(i32.store
|
||||
(i32.const 0)
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; Set no flag.
|
||||
(i32.const 4) ;; Pointer to "callee" address.
|
||||
(i32.const 32) ;; Length of "callee" address.
|
||||
(i64.load (i32.const 36)) ;; How much gas to devote for the execution.
|
||||
(i64.load (i32.const 36)) ;; How much ref_time to devote for the execution.
|
||||
(i64.load (i32.const 44)) ;; How much proof_size to devote for the execution.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 0) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 0) ;; Pointer to input data buffer address
|
||||
(i32.const 0) ;; Length of input data buffer
|
||||
(i32.const 0) ;; Length of input data buffer
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Ptr to output buffer len
|
||||
)
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
)
|
||||
)
|
||||
(call $seal_return (i32.const 0) (i32.const 0) (i32.const 4))
|
||||
)
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
(module
|
||||
(import "seal0" "seal_input" (func $seal_input (param i32 i32)))
|
||||
(import "seal0" "seal_balance" (func $seal_balance (param i32 i32)))
|
||||
(import "seal0" "seal_call" (func $seal_call (param i32 i32 i64 i32 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "seal0" "seal_instantiate" (func $seal_instantiate
|
||||
(param i32 i32 i64 i32 i32 i32 i32 i32 i32 i32 i32 i32 i32) (result i32)
|
||||
(import "seal2" "call" (func $seal_call (param i32 i32 i64 i64 i32 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "seal2" "instantiate" (func $seal_instantiate
|
||||
(param i32 i64 i64 i32 i32 i32 i32 i32 i32 i32 i32 i32 i32) (result i32)
|
||||
))
|
||||
(import "env" "memory" (memory 1 1))
|
||||
|
||||
@@ -43,18 +43,18 @@
|
||||
(set_local $exit_code
|
||||
(call $seal_instantiate
|
||||
(i32.const 24) ;; Pointer to the code hash.
|
||||
(i32.const 32) ;; Length of the code hash.
|
||||
(i64.const 0) ;; How much gas to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 9) ;; Pointer to input data buffer address
|
||||
(i32.const 7) ;; Length of input data buffer
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy address
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy address
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
)
|
||||
)
|
||||
|
||||
@@ -63,22 +63,47 @@
|
||||
(i32.eq (get_local $exit_code) (i32.const 2)) ;; ReturnCode::CalleeReverted
|
||||
)
|
||||
|
||||
;; Fail to deploy the contract due to insufficient gas.
|
||||
;; Fail to deploy the contract due to insufficient ref_time weight.
|
||||
(set_local $exit_code
|
||||
(call $seal_instantiate
|
||||
(i32.const 24) ;; Pointer to the code hash.
|
||||
(i32.const 32) ;; Length of the code hash.
|
||||
(i64.const 1) ;; Supply too little gas
|
||||
(i64.const 1) ;; Supply too little ref_time weight
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy address
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
|
||||
)
|
||||
)
|
||||
|
||||
;; Check for special trap exit status.
|
||||
(call $assert
|
||||
(i32.eq (get_local $exit_code) (i32.const 1)) ;; ReturnCode::CalleeTrapped
|
||||
)
|
||||
|
||||
;; Fail to deploy the contract due to insufficient ref_time weight.
|
||||
(set_local $exit_code
|
||||
(call $seal_instantiate
|
||||
(i32.const 24) ;; Pointer to the code hash.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 1) ;; Supply too little proof_size weight
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy address
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
|
||||
)
|
||||
)
|
||||
@@ -98,18 +123,18 @@
|
||||
(set_local $exit_code
|
||||
(call $seal_instantiate
|
||||
(i32.const 24) ;; Pointer to the code hash.
|
||||
(i32.const 32) ;; Length of the code hash.
|
||||
(i64.const 0) ;; How much gas to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.const 16) ;; Pointer to the address output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Pointer to the address buffer length
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
(i32.const 16) ;; Pointer to the address output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Pointer to the address buffer length
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_le
|
||||
|
||||
)
|
||||
)
|
||||
@@ -139,15 +164,16 @@
|
||||
;; Call the new contract and expect it to return failing exit code.
|
||||
(set_local $exit_code
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; Set no flag
|
||||
(i32.const 16) ;; Pointer to "callee" address.
|
||||
(i32.const 32) ;; Length of "callee" address.
|
||||
(i64.const 0) ;; How much gas to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 9) ;; Pointer to input data buffer address
|
||||
(i32.const 7) ;; Length of input data buffer
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Ptr to output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 8)) ;; Ptr to output buffer len
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Ptr to output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 8)) ;; Ptr to output buffer len
|
||||
)
|
||||
)
|
||||
|
||||
@@ -167,18 +193,40 @@
|
||||
)
|
||||
)
|
||||
|
||||
;; Fail to call the contract due to insufficient gas.
|
||||
;; Fail to call the contract due to insufficient ref_time weight.
|
||||
(set_local $exit_code
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; Set no flag
|
||||
(i32.const 16) ;; Pointer to "callee" address.
|
||||
(i32.const 32) ;; Length of "callee" address.
|
||||
(i64.const 1) ;; Supply too little gas
|
||||
(i64.const 1) ;; Supply too little ref_time weight
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this cas
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this cas
|
||||
)
|
||||
)
|
||||
|
||||
;; Check for special trap exit status.
|
||||
(call $assert
|
||||
(i32.eq (get_local $exit_code) (i32.const 1)) ;; ReturnCode::CalleeTrapped
|
||||
)
|
||||
|
||||
;; Fail to call the contract due to insufficient proof_size weight.
|
||||
(set_local $exit_code
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; Set no flag
|
||||
(i32.const 16) ;; Pointer to "callee" address.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 1) ;; Supply too little proof_size weight
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.const 4294967295) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this cas
|
||||
)
|
||||
)
|
||||
|
||||
@@ -202,15 +250,16 @@
|
||||
;; Call the contract successfully.
|
||||
(set_local $exit_code
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; Set no flag
|
||||
(i32.const 16) ;; Pointer to "callee" address.
|
||||
(i32.const 32) ;; Length of "callee" address.
|
||||
(i64.const 0) ;; How much gas to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: pass no deposit limit.
|
||||
(i32.const 0) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 8) ;; Length of the buffer with value to transfer.
|
||||
(i32.const 8) ;; Pointer to input data buffer address
|
||||
(i32.const 8) ;; Length of input data buffer
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Ptr to output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 8)) ;; Ptr to output buffer len
|
||||
(i32.sub (get_local $sp) (i32.const 4)) ;; Ptr to output buffer
|
||||
(i32.sub (get_local $sp) (i32.const 8)) ;; Ptr to output buffer len
|
||||
)
|
||||
)
|
||||
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
(module
|
||||
(import "seal0" "seal_input" (func $seal_input (param i32 i32)))
|
||||
(import "seal0" "seal_set_storage" (func $seal_set_storage (param i32 i32 i32)))
|
||||
(import "seal1" "seal_call" (func $seal_call (param i32 i32 i64 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "seal2" "call" (func $seal_call (param i32 i32 i64 i64 i32 i32 i32 i32 i32 i32) (result i32)))
|
||||
(import "env" "memory" (memory 1 1))
|
||||
|
||||
(func $assert (param i32)
|
||||
@@ -20,7 +20,10 @@
|
||||
;; store length of input buffer
|
||||
(i32.store (i32.const 0) (i32.const 512))
|
||||
|
||||
;; copy input at address 4
|
||||
;; copy input at address 4:
|
||||
;; first 4 bytes for the size of the storage to be created in callee
|
||||
;; next 32 bytes are for the callee address
|
||||
;; next bytes for the encoded deposit limit
|
||||
(call $seal_input (i32.const 4) (i32.const 0))
|
||||
|
||||
;; create 4 byte of storage before calling
|
||||
@@ -34,8 +37,10 @@
|
||||
(call $assert (i32.eqz
|
||||
(call $seal_call
|
||||
(i32.const 0) ;; No flags
|
||||
(i32.const 8) ;; Pointer to "callee" address.
|
||||
(i64.const 0) ;; How much gas to devote for the execution. 0 = all.
|
||||
(i32.const 8) ;; Pointer to "callee" address
|
||||
(i64.const 0) ;; How much ref_time to devote for the execution. 0 = all
|
||||
(i64.const 0) ;; How much proof_limit to devote for the execution. 0 = all
|
||||
(i32.const 40) ;; Pointer to the storage deposit limit
|
||||
(i32.const 512) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 4) ;; Pointer to input data buffer address
|
||||
(i32.const 4) ;; Length of input data buffer
|
||||
|
||||
@@ -0,0 +1,66 @@
|
||||
;; This instantiates another contract and passes some input to its constructor.
|
||||
(module
|
||||
(import "seal0" "seal_input" (func $seal_input (param i32 i32)))
|
||||
(import "seal0" "seal_set_storage" (func $seal_set_storage (param i32 i32 i32)))
|
||||
(import "seal2" "instantiate" (func $seal_instantiate
|
||||
(param i32 i64 i64 i32 i32 i32 i32 i32 i32 i32 i32 i32 i32) (result i32)
|
||||
))
|
||||
(import "seal0" "seal_return" (func $seal_return (param i32 i32 i32)))
|
||||
(import "env" "memory" (memory 1 1))
|
||||
|
||||
;; [0, 8) send 10_000 balance
|
||||
(data (i32.const 48) "\10\27\00\00\00\00\00\00")
|
||||
|
||||
(func $assert (param i32)
|
||||
(block $ok
|
||||
(br_if $ok
|
||||
(get_local 0)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
)
|
||||
|
||||
(func (export "deploy"))
|
||||
|
||||
(func (export "call")
|
||||
;; store length of input buffer
|
||||
(i32.store (i32.const 0) (i32.const 512))
|
||||
;; store length of contract address
|
||||
(i32.store (i32.const 84) (i32.const 32))
|
||||
|
||||
;; copy input at address 4
|
||||
(call $seal_input (i32.const 4) (i32.const 0))
|
||||
|
||||
;; memory layout is:
|
||||
;; [0,4): size of input buffer
|
||||
;; [4,8): size of the storage to be created in callee
|
||||
;; [8,40): the code hash of the contract to instantiate
|
||||
;; [40,48): for the encoded deposit limit
|
||||
;; [48,52): value to transfer
|
||||
;; [52,84): address of the deployed contract
|
||||
;; [84,88): len of the address
|
||||
|
||||
;; instantiate a contract
|
||||
(call $assert (i32.eqz
|
||||
;; (i32.store
|
||||
;; (i32.const 64)
|
||||
(call $seal_instantiate
|
||||
(i32.const 8) ;; Pointer to the code hash.
|
||||
(i64.const 0) ;; How much ref_time weight to devote for the execution. 0 = all.
|
||||
(i64.const 0) ;; How much proof_size weight to devote for the execution. 0 = all.
|
||||
(i32.const 40) ;; Pointer to the storage deposit limit
|
||||
(i32.const 48) ;; Pointer to the buffer with value to transfer
|
||||
(i32.const 4) ;; Pointer to input data buffer address
|
||||
(i32.const 4) ;; Length of input data buffer
|
||||
(i32.const 52) ;; Pointer to where to copy address
|
||||
(i32.const 84) ;; Pointer to address len ptr
|
||||
(i32.const 0xffffffff) ;; u32 max sentinel value: do not copy output
|
||||
(i32.const 0) ;; Length is ignored in this case
|
||||
(i32.const 0) ;; salt_ptr
|
||||
(i32.const 0) ;; salt_len
|
||||
)
|
||||
))
|
||||
;; return the deployed contract address
|
||||
(call $seal_return (i32.const 0) (i32.const 52) (i32.const 32))
|
||||
)
|
||||
)
|
||||
@@ -0,0 +1,45 @@
|
||||
;; Stores a value of the passed size in constructor.
|
||||
(module
|
||||
(import "seal0" "seal_set_storage" (func $seal_set_storage (param i32 i32 i32)))
|
||||
(import "seal0" "seal_input" (func $seal_input (param i32 i32)))
|
||||
(import "env" "memory" (memory 16 16))
|
||||
|
||||
;; [0, 32) storage key
|
||||
(data (i32.const 0) "\01")
|
||||
|
||||
;; [32, 36) buffer where input is copied (expected size of storage item)
|
||||
|
||||
;; [36, 40) size of the input buffer
|
||||
(data (i32.const 36) "\04")
|
||||
|
||||
(func $assert (param i32)
|
||||
(block $ok
|
||||
(br_if $ok
|
||||
(get_local 0)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
)
|
||||
|
||||
(func (export "deploy")
|
||||
(call $seal_input (i32.const 32) (i32.const 36))
|
||||
|
||||
;; assert input size == 4
|
||||
(call $assert
|
||||
(i32.eq
|
||||
(i32.load (i32.const 36))
|
||||
(i32.const 4)
|
||||
)
|
||||
)
|
||||
|
||||
;; place a value in storage, the size of which is specified by the call input.
|
||||
;; we don't care about the contents of the storage item
|
||||
(call $seal_set_storage
|
||||
(i32.const 0) ;; Pointer to storage key
|
||||
(i32.const 0) ;; Pointer to value
|
||||
(i32.load (i32.const 32)) ;; Size of value
|
||||
)
|
||||
)
|
||||
|
||||
(func (export "call"))
|
||||
)
|
||||
Reference in New Issue
Block a user