mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 23:57:56 +00:00
wasm-executor: Support growing the memory (#12520)
* As always, start with something :P * Add support for max_heap_pages * Add support for wasmtime * Make it compile * Fix compilation * Copy wrongly merged code * Fix compilation * Some fixes * Fix * Get stuff working * More work * More fixes * ... * More * FIXEs * Switch wasmi to use `RuntimeBlob` like wasmtime * Removed unused stuff * Cleanup * More cleanups * Introduce `CallContext` * Fixes * More fixes * Add builder for creating the `WasmExecutor` * Adds some docs * FMT * First round of feedback. * Review feedback round 2 * More fixes * Fix try-runtime * Update client/executor/wasmtime/src/instance_wrapper.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Update client/executor/common/src/wasm_runtime.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Update client/executor/common/src/runtime_blob/runtime_blob.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Update client/executor/common/src/wasm_runtime.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Update client/allocator/src/freeing_bump.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Update client/allocator/src/freeing_bump.rs Co-authored-by: Koute <koute@users.noreply.github.com> * Feedback round 3 * FMT * Review comments --------- Co-authored-by: Koute <koute@users.noreply.github.com>
This commit is contained in:
@@ -17,7 +17,11 @@
|
||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
use codec::{Decode as _, Encode as _};
|
||||
use sc_executor_common::{error::Error, runtime_blob::RuntimeBlob, wasm_runtime::WasmModule};
|
||||
use sc_executor_common::{
|
||||
error::Error,
|
||||
runtime_blob::RuntimeBlob,
|
||||
wasm_runtime::{HeapAllocStrategy, WasmModule},
|
||||
};
|
||||
use sc_runtime_test::wasm_binary_unwrap;
|
||||
|
||||
use crate::InstantiationStrategy;
|
||||
@@ -77,8 +81,7 @@ struct RuntimeBuilder {
|
||||
instantiation_strategy: InstantiationStrategy,
|
||||
canonicalize_nans: bool,
|
||||
deterministic_stack: bool,
|
||||
extra_heap_pages: u64,
|
||||
max_memory_size: Option<usize>,
|
||||
heap_pages: HeapAllocStrategy,
|
||||
precompile_runtime: bool,
|
||||
tmpdir: Option<tempfile::TempDir>,
|
||||
}
|
||||
@@ -90,8 +93,7 @@ impl RuntimeBuilder {
|
||||
instantiation_strategy,
|
||||
canonicalize_nans: false,
|
||||
deterministic_stack: false,
|
||||
extra_heap_pages: 1024,
|
||||
max_memory_size: None,
|
||||
heap_pages: HeapAllocStrategy::Static { extra_pages: 1024 },
|
||||
precompile_runtime: false,
|
||||
tmpdir: None,
|
||||
}
|
||||
@@ -117,8 +119,8 @@ impl RuntimeBuilder {
|
||||
self
|
||||
}
|
||||
|
||||
fn max_memory_size(mut self, max_memory_size: Option<usize>) -> Self {
|
||||
self.max_memory_size = max_memory_size;
|
||||
fn heap_alloc_strategy(mut self, heap_pages: HeapAllocStrategy) -> Self {
|
||||
self.heap_pages = heap_pages;
|
||||
self
|
||||
}
|
||||
|
||||
@@ -152,8 +154,7 @@ impl RuntimeBuilder {
|
||||
},
|
||||
canonicalize_nans: self.canonicalize_nans,
|
||||
parallel_compilation: true,
|
||||
extra_heap_pages: self.extra_heap_pages,
|
||||
max_memory_size: self.max_memory_size,
|
||||
heap_alloc_strategy: self.heap_pages,
|
||||
},
|
||||
};
|
||||
|
||||
@@ -227,7 +228,7 @@ fn deep_call_stack_wat(depth: usize) -> String {
|
||||
// We need two limits here since depending on whether the code is compiled in debug
|
||||
// or in release mode the maximum call depth is slightly different.
|
||||
const CALL_DEPTH_LOWER_LIMIT: usize = 65455;
|
||||
const CALL_DEPTH_UPPER_LIMIT: usize = 65503;
|
||||
const CALL_DEPTH_UPPER_LIMIT: usize = 65509;
|
||||
|
||||
test_wasm_execution!(test_consume_under_1mb_of_stack_does_not_trap);
|
||||
fn test_consume_under_1mb_of_stack_does_not_trap(instantiation_strategy: InstantiationStrategy) {
|
||||
@@ -344,29 +345,25 @@ fn test_max_memory_pages(
|
||||
import_memory: bool,
|
||||
precompile_runtime: bool,
|
||||
) {
|
||||
fn try_instantiate(
|
||||
max_memory_size: Option<usize>,
|
||||
fn call(
|
||||
heap_alloc_strategy: HeapAllocStrategy,
|
||||
wat: String,
|
||||
instantiation_strategy: InstantiationStrategy,
|
||||
precompile_runtime: bool,
|
||||
) -> Result<(), Box<dyn std::error::Error>> {
|
||||
let mut builder = RuntimeBuilder::new(instantiation_strategy)
|
||||
.use_wat(wat)
|
||||
.max_memory_size(max_memory_size)
|
||||
.heap_alloc_strategy(heap_alloc_strategy)
|
||||
.precompile_runtime(precompile_runtime);
|
||||
|
||||
let runtime = builder.build();
|
||||
let mut instance = runtime.new_instance()?;
|
||||
let mut instance = runtime.new_instance().unwrap();
|
||||
let _ = instance.call_export("main", &[])?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn memory(initial: u32, maximum: Option<u32>, import: bool) -> String {
|
||||
let memory = if let Some(maximum) = maximum {
|
||||
format!("(memory $0 {} {})", initial, maximum)
|
||||
} else {
|
||||
format!("(memory $0 {})", initial)
|
||||
};
|
||||
fn memory(initial: u32, maximum: u32, import: bool) -> String {
|
||||
let memory = format!("(memory $0 {} {})", initial, maximum);
|
||||
|
||||
if import {
|
||||
format!("(import \"env\" \"memory\" {})", memory)
|
||||
@@ -375,152 +372,90 @@ fn test_max_memory_pages(
|
||||
}
|
||||
}
|
||||
|
||||
const WASM_PAGE_SIZE: usize = 65536;
|
||||
let assert_grow_ok = |alloc_strategy: HeapAllocStrategy, initial_pages: u32, max_pages: u32| {
|
||||
eprintln!("assert_grow_ok({alloc_strategy:?}, {initial_pages}, {max_pages})");
|
||||
|
||||
// check the old behavior if preserved. That is, if no limit is set we allow 4 GiB of memory.
|
||||
try_instantiate(
|
||||
None,
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
(i64.const 0)
|
||||
)
|
||||
)
|
||||
"#,
|
||||
/*
|
||||
We want to allocate the maximum number of pages supported in wasm for this test.
|
||||
However, due to a bug in wasmtime (I think wasmi is also affected) it is only possible
|
||||
to allocate 65536 - 1 pages.
|
||||
call(
|
||||
alloc_strategy,
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
|
||||
Then, during creation of the Substrate Runtime instance, 1024 (heap_pages) pages are
|
||||
mounted.
|
||||
|
||||
Thus 65535 = 64511 + 1024
|
||||
*/
|
||||
memory(64511, None, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
// max is not specified, therefore it's implied to be 65536 pages (4 GiB).
|
||||
//
|
||||
// max_memory_size = (1 (initial) + 1024 (heap_pages)) * WASM_PAGE_SIZE
|
||||
try_instantiate(
|
||||
Some((1 + 1024) * WASM_PAGE_SIZE),
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
(i64.const 0)
|
||||
)
|
||||
)
|
||||
"#,
|
||||
// 1 initial, max is not specified.
|
||||
memory(1, None, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
// max is specified explicitly to 2048 pages.
|
||||
try_instantiate(
|
||||
Some((1 + 1024) * WASM_PAGE_SIZE),
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
(i64.const 0)
|
||||
)
|
||||
)
|
||||
"#,
|
||||
// Max is 2048.
|
||||
memory(1, Some(2048), import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
// memory grow should work as long as it doesn't exceed 1025 pages in total.
|
||||
try_instantiate(
|
||||
Some((0 + 1024 + 25) * WASM_PAGE_SIZE),
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
|
||||
;; assert(memory.grow returns != -1)
|
||||
(if
|
||||
(i32.eq
|
||||
(memory.grow
|
||||
(i32.const 25)
|
||||
;; assert(memory.grow returns != -1)
|
||||
(if
|
||||
(i32.eq
|
||||
(memory.grow
|
||||
(i32.const 1)
|
||||
)
|
||||
(i32.const -1)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
(i32.const -1)
|
||||
|
||||
(i64.const 0)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
|
||||
(i64.const 0)
|
||||
)
|
||||
)
|
||||
"#,
|
||||
// Zero starting pages.
|
||||
memory(0, None, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap();
|
||||
memory(initial_pages, max_pages, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
// We start with 1025 pages and try to grow at least one.
|
||||
try_instantiate(
|
||||
Some((1 + 1024) * WASM_PAGE_SIZE),
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
let assert_grow_fail =
|
||||
|alloc_strategy: HeapAllocStrategy, initial_pages: u32, max_pages: u32| {
|
||||
eprintln!("assert_grow_fail({alloc_strategy:?}, {initial_pages}, {max_pages})");
|
||||
|
||||
;; assert(memory.grow returns == -1)
|
||||
(if
|
||||
(i32.ne
|
||||
(memory.grow
|
||||
(i32.const 1)
|
||||
call(
|
||||
alloc_strategy,
|
||||
format!(
|
||||
r#"
|
||||
(module
|
||||
{}
|
||||
(global (export "__heap_base") i32 (i32.const 0))
|
||||
(func (export "main")
|
||||
(param i32 i32) (result i64)
|
||||
|
||||
;; assert(memory.grow returns == -1)
|
||||
(if
|
||||
(i32.ne
|
||||
(memory.grow
|
||||
(i32.const 1)
|
||||
)
|
||||
(i32.const -1)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
|
||||
(i64.const 0)
|
||||
)
|
||||
(i32.const -1)
|
||||
)
|
||||
(unreachable)
|
||||
)
|
||||
|
||||
(i64.const 0)
|
||||
)
|
||||
"#,
|
||||
memory(initial_pages, max_pages, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
"#,
|
||||
// Initial=1, meaning after heap pages mount the total will be already 1025.
|
||||
memory(1, None, import_memory)
|
||||
),
|
||||
instantiation_strategy,
|
||||
precompile_runtime,
|
||||
)
|
||||
.unwrap();
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
assert_grow_ok(HeapAllocStrategy::Dynamic { maximum_pages: Some(10) }, 1, 10);
|
||||
assert_grow_ok(HeapAllocStrategy::Dynamic { maximum_pages: Some(10) }, 9, 10);
|
||||
assert_grow_fail(HeapAllocStrategy::Dynamic { maximum_pages: Some(10) }, 10, 10);
|
||||
|
||||
assert_grow_ok(HeapAllocStrategy::Dynamic { maximum_pages: None }, 1, 10);
|
||||
assert_grow_ok(HeapAllocStrategy::Dynamic { maximum_pages: None }, 9, 10);
|
||||
assert_grow_ok(HeapAllocStrategy::Dynamic { maximum_pages: None }, 10, 10);
|
||||
|
||||
assert_grow_fail(HeapAllocStrategy::Static { extra_pages: 10 }, 1, 10);
|
||||
assert_grow_fail(HeapAllocStrategy::Static { extra_pages: 10 }, 9, 10);
|
||||
assert_grow_fail(HeapAllocStrategy::Static { extra_pages: 10 }, 10, 10);
|
||||
}
|
||||
|
||||
// This test takes quite a while to execute in a debug build (over 6 minutes on a TR 3970x)
|
||||
@@ -538,8 +473,7 @@ fn test_instances_without_reuse_are_not_leaked() {
|
||||
deterministic_stack_limit: None,
|
||||
canonicalize_nans: false,
|
||||
parallel_compilation: true,
|
||||
extra_heap_pages: 2048,
|
||||
max_memory_size: None,
|
||||
heap_alloc_strategy: HeapAllocStrategy::Static { extra_pages: 2048 },
|
||||
},
|
||||
},
|
||||
)
|
||||
@@ -583,6 +517,10 @@ fn test_rustix_version_matches_with_wasmtime() {
|
||||
.unwrap();
|
||||
|
||||
if wasmtime_rustix.req != our_rustix.req {
|
||||
panic!("our version of rustix ({0}) doesn't match wasmtime's ({1}); bump the version in `sc-executor-wasmtime`'s `Cargo.toml' to '{1}' and try again", our_rustix.req, wasmtime_rustix.req);
|
||||
panic!(
|
||||
"our version of rustix ({0}) doesn't match wasmtime's ({1}); \
|
||||
bump the version in `sc-executor-wasmtime`'s `Cargo.toml' to '{1}' and try again",
|
||||
our_rustix.req, wasmtime_rustix.req,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user