Stream: git-wasmtime

Topic: wasmtime / PR #2847 Optimize `table.init` instruction and...


view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 20:30):

alexcrichton opened PR #2847 from speed-up-table-init to main:

This commit optimizes table initialization as part of instance
instantiation and also applies the same optimization to the table.init
instruction. One part of this commit is to remove some preexisting
duplication between instance instantiation and the table.init
instruction itself, after this the actual implementation of table.init
is optimized to effectively have fewer bounds checks in fewer places and
have a much tighter loop for instantiation.

A big fallout from this change is that memory/table initializer offsets
are now stored as u32 instead of usize to remove a few casts in a
few places. This ended up requiring moving some overflow checks that
happened in parsing to later in code itself because otherwise the wrong
spec test errors are emitted during testing. I've tried to trace where
these can possibly overflow but I think that I managed to get
everything.

In a local synthetic test where an empty module with a single 80,000
element initializer this improves total instantiation time by 4x (562us
=> 141us)

<!--

Please ensure that the following steps are all taken care of before submitting
the PR.

Please ensure all communication adheres to the code of conduct.
-->

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen created PR Review Comment:

Why this change? The anyfunc_ptr call should be inlined just fine, unless I'm missing something. This just makes things more open coded, which isn't the direction I'd generally move in.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen created PR Review Comment:

This seems very tautological as written; is this supposed to be checking self.element_type?

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen created PR Review Comment:

Does this get boiled down into a memcpy by LLVM? If so, we can close #983. If not, would we be able to do that here?

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 21:25):

fitzgen created PR Review Comment:

It seems like this has just one call site, and I don't think anything is preventing us from doing a memcpy in theory here.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:22):

alexcrichton submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:22):

alexcrichton created PR Review Comment:

This mostly changed because it was previously used in a loop but the pattern it was using was a bit slower than necessary. The main slowdown is that vmctx_anyfunc performed a multiplication instruction with the size of a pointer, where both the size and the index were not constant, resulting in bad codegen. By using the native types here it makes pointer arithmetic much speedier since everything is known at compile time.

Once I removed one main use of the function in the table initialization bits I figured I might as well go ahead and remove the function and speed up other callsites as well.

I do agree it's a bit more open coded, but then again everything about VMContext feels sort of inherently open-coded

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:23):

alexcrichton submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:23):

alexcrichton created PR Review Comment:

Oh this should actually probably close that issue. We don't actually have anything to memcpy though because our element segments are stored as lists of function indices, but the tables themselves are *mut VMCallerCheckedAnyfunc. This means that there's a bit of a mismatch where unless we allocate table segments per-instance to memcpy from (which I dont think we should do) then there's actually nothing to memcpy.

Otherwise, though, I just used set_raw here to be the same as the other bits and pieces of this module

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:24):

alexcrichton submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:24):

alexcrichton created PR Review Comment:

Oops, yes!

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:25):

alexcrichton updated PR #2847 from speed-up-table-init to main.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:25):

alexcrichton submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:25):

alexcrichton created PR Review Comment:

I went ahead though and made it a bit more raw to help streamline this a tiny bit more

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:29):

fitzgen submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:29):

fitzgen created PR Review Comment:

Huh, I would really have expected anyfunc_ptr to be inlined and then for inst combine to see that one of the operands of the multiplication was a constant and go to town from there.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:40):

alexcrichton submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:40):

alexcrichton created PR Review Comment:

Oh it is inlined, but that's not the problem here. The function called has a multiplication with 3 * self.pointer_size, but self.pointer_size is not a constant value. We could probably improve codegen by having something like NativeVMOffsets for when you're specifically not cross-compiling somewhere else but for now LLVM has no way of seeing that the value originally stored there was mem::size_of::<usize>()

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:44):

fitzgen submitted PR Review.

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 22:44):

fitzgen created PR Review Comment:

Ah, I get it now. Thanks for your patience explaining it to me!

view this post on Zulip Wasmtime GitHub notifications bot (Apr 19 2021 at 23:44):

alexcrichton merged PR #2847.


Last updated: Jan 24 2025 at 00:11 UTC