diff --git a/bin/node/runtime/src/lib.rs b/bin/node/runtime/src/lib.rs index d72cd145829317951272741c6e59e9422a6415d3..8d57e6fbf8c27b286a9c76fa02e9c84c20fcec9a 100644 --- a/bin/node/runtime/src/lib.rs +++ b/bin/node/runtime/src/lib.rs @@ -82,8 +82,8 @@ pub const VERSION: RuntimeVersion = RuntimeVersion { // and set impl_version to 0. If only runtime // implementation changes and behavior does not, then leave spec_version as // is and increment impl_version. - spec_version: 235, - impl_version: 1, + spec_version: 236, + impl_version: 0, apis: RUNTIME_API_VERSIONS, }; diff --git a/frame/contracts/COMPLEXITY.md b/frame/contracts/COMPLEXITY.md index e44d3006c8ed363e63909f9c8504545653f71d71..ac0b2535f556e978a5a4190f50af870a30c03b30 100644 --- a/frame/contracts/COMPLEXITY.md +++ b/frame/contracts/COMPLEXITY.md @@ -261,10 +261,20 @@ Each external function invoked from a contract can involve some overhead. This function receives a `key` and `value` as arguments. It consists of the following steps: 1. Reading the sandbox memory for `key` and `value` (see sandboxing memory get). -2. Setting the storage by the given `key` with the given `value` (see `set_storage`). +2. Setting the storage at the given `key` to the given `value` (see `set_storage`). **complexity**: Complexity is proportional to the size of the `value`. This function induces a DB write of size proportional to the `value` size (if flushed to the storage), so should be priced accordingly. +## ext_clear_storage + +This function receives a `key` as argument. It consists of the following steps: + +1. Reading the sandbox memory for `key` (see sandboxing memory get). +2. Clearing the storage at the given `key` (see `set_storage`). + +**complexity**: Complexity is constant. This function induces a DB write to clear the storage entry +(upon being flushed to the storage) and should be priced accordingly. + ## ext_get_storage This function receives a `key` as an argument. It consists of the following steps: diff --git a/frame/contracts/src/tests.rs b/frame/contracts/src/tests.rs index 51da3e8d1ff97ee0048290d073ac53e6e10f291f..1b85e32e7ed2cfb06381a5cf908704f568ce9f92 100644 --- a/frame/contracts/src/tests.rs +++ b/frame/contracts/src/tests.rs @@ -770,7 +770,8 @@ fn run_out_of_gas() { const CODE_SET_RENT: &str = r#" (module (import "env" "ext_dispatch_call" (func $ext_dispatch_call (param i32 i32))) - (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32 i32))) + (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32))) + (import "env" "ext_clear_storage" (func $ext_clear_storage (param i32))) (import "env" "ext_set_rent_allowance" (func $ext_set_rent_allowance (param i32 i32))) (import "env" "ext_scratch_size" (func $ext_scratch_size (result i32))) (import "env" "ext_scratch_read" (func $ext_scratch_read (param i32 i32 i32))) @@ -779,7 +780,6 @@ const CODE_SET_RENT: &str = r#" ;; insert a value of 4 bytes into storage (func $call_0 (call $ext_set_storage - (i32.const 1) (i32.const 1) (i32.const 0) (i32.const 4) @@ -788,11 +788,8 @@ const CODE_SET_RENT: &str = r#" ;; remove the value inserted by call_1 (func $call_1 - (call $ext_set_storage + (call $ext_clear_storage (i32.const 1) - (i32.const 0) - (i32.const 0) - (i32.const 0) ) ) @@ -852,7 +849,6 @@ const CODE_SET_RENT: &str = r#" ) (call $ext_set_storage (i32.const 0) - (i32.const 1) (i32.const 0) (i32.const 4) ) @@ -1327,7 +1323,7 @@ fn default_rent_allowance_on_instantiate() { const CODE_RESTORATION: &str = r#" (module - (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32 i32))) + (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32))) (import "env" "ext_restore_to" (func $ext_restore_to (param i32 i32 i32 i32 i32 i32 i32 i32))) (import "env" "memory" (memory 1 1)) @@ -1352,7 +1348,6 @@ const CODE_RESTORATION: &str = r#" ;; Data to restore (call $ext_set_storage (i32.const 0) - (i32.const 1) (i32.const 0) (i32.const 4) ) @@ -1360,7 +1355,6 @@ const CODE_RESTORATION: &str = r#" ;; ACL (call $ext_set_storage (i32.const 100) - (i32.const 1) (i32.const 0) (i32.const 4) ) @@ -1377,8 +1371,8 @@ const CODE_RESTORATION: &str = r#" ;; Code hash of SET_RENT (data (i32.const 264) - "\14\eb\65\3c\86\98\d6\b2\3d\8d\3c\4a\54\c6\c4\71" - "\b9\fc\19\36\df\ca\a0\a1\f2\dc\ad\9d\e5\36\0b\25" + "\c2\1c\41\10\a5\22\d8\59\1c\4c\77\35\dd\2d\bf\a1" + "\13\0b\50\93\76\9b\92\31\97\b7\c5\74\26\aa\38\2a" ) ;; Rent allowance @@ -1624,7 +1618,7 @@ fn restoration(test_different_storage: bool, test_restore_to_with_dirty_storage: const CODE_STORAGE_SIZE: &str = r#" (module (import "env" "ext_get_storage" (func $ext_get_storage (param i32) (result i32))) - (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32 i32))) + (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32))) (import "env" "ext_scratch_size" (func $ext_scratch_size (result i32))) (import "env" "ext_scratch_read" (func $ext_scratch_read (param i32 i32 i32))) (import "env" "memory" (memory 16 16)) @@ -1657,7 +1651,6 @@ const CODE_STORAGE_SIZE: &str = r#" ;; place a garbage value in storage, the size of which is specified by the call input. (call $ext_set_storage (i32.const 0) ;; Pointer to storage key - (i32.const 1) ;; Value is not null (i32.const 0) ;; Pointer to value (i32.load (i32.const 32)) ;; Size of value ) @@ -2278,7 +2271,7 @@ const CODE_DESTROY_AND_TRANSFER: &str = r#" (import "env" "ext_scratch_size" (func $ext_scratch_size (result i32))) (import "env" "ext_scratch_read" (func $ext_scratch_read (param i32 i32 i32))) (import "env" "ext_get_storage" (func $ext_get_storage (param i32) (result i32))) - (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32 i32))) + (import "env" "ext_set_storage" (func $ext_set_storage (param i32 i32 i32))) (import "env" "ext_call" (func $ext_call (param i32 i32 i64 i32 i32 i32 i32) (result i32))) (import "env" "ext_instantiate" (func $ext_instantiate (param i32 i32 i64 i32 i32 i32 i32) (result i32))) (import "env" "memory" (memory 1 1)) @@ -2340,7 +2333,6 @@ const CODE_DESTROY_AND_TRANSFER: &str = r#" ;; Store the return address. (call $ext_set_storage (i32.const 16) ;; Pointer to the key - (i32.const 1) ;; Value is not null (i32.const 80) ;; Pointer to the value (i32.const 8) ;; Length of the value ) diff --git a/frame/contracts/src/wasm/runtime.rs b/frame/contracts/src/wasm/runtime.rs index a84556d884ce97a3a2c16fec8b1051a6d72cfe35..82232dde39a1b884df6bf43b9408036b10e2a72c 100644 --- a/frame/contracts/src/wasm/runtime.rs +++ b/frame/contracts/src/wasm/runtime.rs @@ -348,30 +348,42 @@ define_env!(Env, <E: Ext>, Ok(()) }, - // Change the value at the given key in the storage or remove the entry. - // The value length must not exceed the maximum defined by the Contracts module parameters. - // - // - key_ptr: pointer into the linear - // memory where the location of the requested value is placed. - // - value_non_null: if set to 0, then the entry - // at the given location will be removed. - // - value_ptr: pointer into the linear memory - // where the value to set is placed. If `value_non_null` is set to 0, then this parameter is ignored. - // - value_len: the length of the value. If `value_non_null` is set to 0, then this parameter is ignored. - ext_set_storage(ctx, key_ptr: u32, value_non_null: u32, value_ptr: u32, value_len: u32) => { - if value_non_null != 0 && ctx.ext.max_value_size() < value_len { + // Set the value at the given key in the contract storage. + // + // The value length must not exceed the maximum defined by the contracts module parameters. + // Storing an empty value is disallowed. + // + // # Parameters + // + // - `key_ptr`: pointer into the linear memory where the location to store the value is placed. + // - `value_ptr`: pointer into the linear memory where the value to set is placed. + // - `value_len`: the length of the value in bytes. + // + // # Errors + // + // - If value length exceeds the configured maximum value length of a storage entry. + // - Upon trying to set an empty storage entry (value length is 0). + ext_set_storage(ctx, key_ptr: u32, value_ptr: u32, value_len: u32) => { + if value_len > ctx.ext.max_value_size() { + // Bail out if value length exceeds the set maximum value size. return Err(sp_sandbox::HostError); } let mut key: StorageKey = [0; 32]; read_sandbox_memory_into_buf(ctx, key_ptr, &mut key)?; - let value = - if value_non_null != 0 { - Some(read_sandbox_memory(ctx, value_ptr, value_len)?) - } else { - None - }; + let value = Some(read_sandbox_memory(ctx, value_ptr, value_len)?); ctx.ext.set_storage(key, value).map_err(|_| sp_sandbox::HostError)?; + Ok(()) + }, + // Clear the value at the given key in the contract storage. + // + // # Parameters + // + // - `key_ptr`: pointer into the linear memory where the location to clear the value is placed. + ext_clear_storage(ctx, key_ptr: u32) => { + let mut key: StorageKey = [0; 32]; + read_sandbox_memory_into_buf(ctx, key_ptr, &mut key)?; + ctx.ext.set_storage(key, None).map_err(|_| sp_sandbox::HostError)?; Ok(()) },