Stream: wasmtime

Topic: Custom hashers


view this post on Zulip Josh Groves (Nov 28 2023 at 04:03):

Hi! Some of the hashmaps used internally seem to show up in flamegraphs for my program that creates a lot of smaller wasm modules. e.g., https://github.com/bytecodealliance/wasmtime/blob/cd97c9f14713d81de5453fb9861319e78a761432/crates/wasmtime/src/linker.rs#L88 is one of them. Is there any interest in being able to pass custom hashers (like rustc-hash) to help with hashing performance?

view this post on Zulip Till Schneidereit (Nov 28 2023 at 13:39):

Would another option be to change the HashMap implementation used here? It doesn't seem to me like this would be a DoS attack vector, so perhaps a fast HashMap that'd be less robust against those would work here? @Alex Crichton, thoughts?

view this post on Zulip Alex Crichton (Nov 28 2023 at 14:50):

Ah yeah we likely don't need dos protection for most of our usage of hash maps, so rather than specifying custom hashers as an embedder it should be fine to hardcode many hash maps to a faster hasher.

That being said though I'd probably first go the route of trying to reduce the usage of APIs that trigger hashing. That's not always possible, though, for example if you have a single Linker and instantiate lots of small moudles through that there's not much we can do to reduce the hashing any more.

view this post on Zulip Josh Groves (Nov 28 2023 at 15:22):

Yeah I currently use separate Linkers per module but still spend a significant amount of time hashing during the initial inserts. I can try to submit a PR for some of the spots I notice, would you prefer any specific fast hasher (rustc-hash, ahash, something else)?

view this post on Zulip Till Schneidereit (Nov 28 2023 at 16:48):

we have ahash in our audits, so that'd probably be good to use. Though I think rustc-hash should also just work in that regard

view this post on Zulip Lann Martin (Nov 28 2023 at 16:50):

Isn't ahash the default rust hasher now (via being hashbrown's default)? Ah no I see libstd opts out of ahash.

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:08):

Is it not possible to use one Linker for all modules?

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:09):

(that'd be best to reduce all the insertions and clones involved there too)

view this post on Zulip Josh Groves (Nov 28 2023 at 17:10):

Yeah I also want to try that soon, although I might have the opposite problem of having a large hashmap to search afterwards. I think a faster hasher would be beneficial either way

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:12):

indeed!

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:13):

Also, are you doing repeated instantiations of the same module? Or instantiations of new modules primarily?

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:13):

I ask b/c InstancePre can help with the former, repeated instantaitions of the same module, and skips hash maps entirely

view this post on Zulip Josh Groves (Nov 28 2023 at 17:19):

That's also a good idea, it's mostly new modules right now but I think I could probably reuse many of them. I need to spend some more understanding the implications of reusing modules and linkers, especially trade-offs (e.g., inlining different constants into a module from the host at compilation time with separate modules, vs. reusing one module but using different environments or something)

view this post on Zulip Josh Groves (Nov 28 2023 at 17:21):

FWIW if it helps with context, my use case is analogous to Excel/Google Sheets spreadsheets where each formula becomes its own tiny wasm module (using its own linker) in my current design

view this post on Zulip Josh Groves (Nov 28 2023 at 17:22):

So it's easy to end up with tens of thousands-ish modules if they're all completely unique, but a lot of the time the formulas are the exact same but just need slightly different environments passed to them

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:23):

ah ok makes sense, well in any case switching hashers definitely makes sense, and any other optimizations you'd like to apply to Linker or the compilation process in general I think would make sense too

view this post on Zulip Alex Crichton (Nov 28 2023 at 17:23):

most of Wasmtime's optimizations have been around runtime and InstancePre-to-Instance so this part isn't quite so well optimized

view this post on Zulip scottmcm (Nov 28 2023 at 18:41):

Josh Groves said:

e.g., https://github.com/bytecodealliance/wasmtime/blob/cd97c9f14713d81de5453fb9861319e78a761432/crates/wasmtime/src/linker.rs#L88 is one of them.

How much is it specifically that one? It looks like ImportKey is two indexes into an interned strings list? If so, that's a great candidate for some type-specific hashing logic. Or maybe even making a struct InternKey(u32); instead of using raw usize to shrink ImportKey from 16 bytes down to 8 on 64-bit, which save a bunch of hashing work.

(This is totally a peanut-gallery comment because zulip put this thread in a digest and I've been looking recently at hashtables where the key is two integers. Feel free to ignore if I'm off-base.)

(This is my first PR here, so I've probably missed some things. Please let me know what else I should do to help you as a reviewer!) Objective Due to rust-lang/rust#117800, the derive'd PartialEq:...
Objective Keep essentially the same structure of EntityHasher from #9903, but rephrase the multiplication slightly to save an instruction. cc @superdump Discord thread: https://discord.com/channels...

view this post on Zulip Josh Groves (Nov 28 2023 at 18:48):

Unfortunately I don’t have great stats right now, the sampling profiler in vtune makes it difficult to get detailed statistics for sub-millisecond code without setting up artificial microbenchmarks. I just noticed SipHash show up in a few hot paths for wasmtime so I figured it might be worth raising here. That partialeq optimization looks neat!

view this post on Zulip scottmcm (Nov 28 2023 at 18:54):

It's a fun one, though I'm hopeful that between Rust and LLVM we can make it unnecessary in future (<https://github.com/rust-lang/rust/issues/117800>).

Entity in bevy shows up everywhere, so it's worth some uglification for even tiny speedups. I don't know nearly enough about wasmtime to know if ImportKey is anywhere close to as critical.

Given a basic struct like this, #[derive(Copy, Clone, PartialEq, Eq)] pub struct Entity { g: u32, i: u32 } The generated == is suboptimal: #[no_mangle] pub fn derived_eq(x: &Entity, y: &Entity) -> ...

view this post on Zulip Josh Groves (Jan 26 2024 at 16:10):

Opened https://github.com/bytecodealliance/wasmtime/pull/7828 as a follow-up to this conversation

From a Zulip discussion a few months ago (https://bytecodealliance.zulipchat.com/#narrow/stream/217126-wasmtime/topic/Custom.20hashers) it was mentioned that custom hashers like rustc-hash should b...

Last updated: Jan 24 2025 at 00:11 UTC