Stream: wasmtime

Topic: Preallocation of memory across instance life


view this post on Zulip spino17 (Nov 27 2024 at 20:58):

I am caching (instance, store) tuple. I want to minimize my allocation as much as possible. So for every wasm exported function call, I preallocate a large chunk of memory and any time host wants to send the data to module, this chunk is used. I am thinking of extending this for the lifetime of the instance, is it possible to do so ? My question is if while caching I allocate memory and store it's pointer, will the allocated memory be valid across subsequent wasm function calls if I use the same instance and store ?

view this post on Zulip Chris Fallin (Nov 27 2024 at 21:09):

That sounds like a possible ABI. It's completely up to you to decide how your host code and your guest code cooperate -- if they both agree to use a single long-lived buffer to exchange data, that's fine.

Note that if the Wasm memory grows, the host-side address (the actual native address in the process) may change. So you would want to keep around the Wasm/guest-side address and access guest memory via the safe Rust APIs when needed.

view this post on Zulip spino17 (Nov 27 2024 at 21:29):

Thanks @Chris Fallin for the response. My only doubt after reading the above message is how will new allocators inside WASM module know about the already allocated memory from the previous runs ? each run of exported wasm function I assume would setup up new allocator right ?

view this post on Zulip Chris Fallin (Nov 27 2024 at 21:36):

It's hard to really say more -- this design is up to you. You've specified that you'll pre-allocate guest memory -- presumably by calling a function in the guest? Maybe you want to pass a pointer to that same buffer each time you call the guest, then, I'm not sure. Or perhaps you could use global state in the guest.

The distinction I'm trying to draw is that this is purely a contract issue between your code (host side) and your code (guest side); Wasmtime doesn't care how you allocate guest memory; so you can invent whatever convention you want. E.g. "each run ... would setup new allocator right?" is a question we can't answer, because you're asking us what your code is doing. You can make it do whatever you want!

If you have questions about specific existing guest language runtimes and how they might interact with all this, we can try to help. With wasi-libc for example, you probably want to call the guest's malloc, then keep that pointer around and reuse it by passing it into each call.

view this post on Zulip spino17 (Nov 27 2024 at 21:44):

ohh okay, I understand what you saying @Chris Fallin . My bad, to clarify the guest language is Rust and the target is wasm32-unknown-unknown I am compiling it to and I am exporting a function called __new which allocates a manuallydrop Vec<u8> and I pass ptr to it around while making allocations to pass data from host to guest

view this post on Zulip spino17 (Nov 27 2024 at 21:45):

so specifically my question would be then, whether this ptr would be valid and pointing to the same allocated buffer when I execute subsequent WASM functions using the same instance and store ?

view this post on Zulip Chris Fallin (Nov 27 2024 at 21:58):

Gotcha -- yes, the Wasm pointer will continue to be valid as long as the instance is alive. Note this is an offset into the Wasm heap. The host-side pointer that corresponds to where that data is stored in the host process may change, because the storage location of the Wasm heap may move when we resize it. But you can always use the Wasm pointer together with the Store and Memory to get at the current contents.

So to summarize: store the Instance, Store, and a u32 corresponding to the Wasm address. Hopefully that makes sense?

view this post on Zulip Joel Dice (Nov 27 2024 at 22:06):

If you don't need this buffer to be resizeable (and note that if you do resize it, the old pointer may become invalid), consider using Box<[u8]>, plus Box::into_raw to turn it into a raw pointer you can pass to the host. That pointer should remain valid until the guest uses Box::from_raw to reclaim (and possibly drop) it.

Or even just use alloc directly if you only ever plan to access the buffer via the raw pointer.

view this post on Zulip spino17 (Nov 28 2024 at 05:28):

ohh much thanks @Chris Fallin for clearing this up! I guess the allocator and globals of the guest binary would belong to the Instance I presume that's why it remembers the allocated memory in subsequent exported function calls ?

view this post on Zulip spino17 (Nov 28 2024 at 05:30):

thanks @Joel Dice for this detail. I should definitely put a check and evict the stored offset when memory resizes! If I understand it correct the function is

memory.grow(caller.as_context_mut(), num_pages as u64)

right ? This is the only way memory might resize correct ?

view this post on Zulip Chris Fallin (Nov 28 2024 at 19:15):

spino17 said:

ohh much thanks @Chris Fallin for clearing this up! I guess the allocator and globals of the guest binary would belong to the Instance I presume that's why it remembers the allocated memory in subsequent exported function calls ?

Basically yeah -- to make it more precise, a malloc-like allocator typically keeps its state in global variables (in the C/Rust sense) which for a Wasm target are compiled to accesses to specific addresses in the Wasm heap, and in data structures pointed to from those variables. So malloc's "next block to allocate from" data might be a Block* at some arbitrary Wasm address 0x1020 or something like that. As long as the instance exists, the data in the heap remains there; so when you next call into the guest, it has the same malloc state.

memory.grow(caller.as_context_mut(), num_pages as u64)

right ? This is the only way memory might resize correct ?

The main way memory will grow is usually by action of the guest -- the memory.grow Wasm opcode grows the heap. A malloc implementation will use this under the hood to grow memory as more is needed.

view this post on Zulip spino17 (Nov 28 2024 at 19:21):

so @Chris Fallin if guest resize the memory then my stored offset might get invalid and I have no way to know it ?

view this post on Zulip Chris Fallin (Nov 28 2024 at 19:28):

The "offset" (offset into heap, Wasm pointer) is still valid; the host-side pointer may not be. This bit from my earlier message above is hopefully useful:

the Wasm pointer will continue to be valid as long as the instance is alive. Note this is an offset into the Wasm heap. The host-side pointer that corresponds to where that data is stored in the host process may change, because the storage location of the Wasm heap may move when we resize it. But you can always use the Wasm pointer together with the Store and Memory to get at the current contents.

view this post on Zulip spino17 (Nov 28 2024 at 19:29):

ahh ohh okay, got it! yeah I get this offset by calling a wasm exported function called __new which allocates a manually drop Vec<u8> and returns it's pointer and this is the pointer I am storing along with instance and store so I believe this will still be valid

view this post on Zulip Chris Fallin (Nov 28 2024 at 19:39):

Yes. The key bit to realize is that the two sides (host and guest) have different notions of "pointer". To the code inside the Wasm module, the "offset into the heap" is the pointer: this u32 is what Rust or C pass around as actual memory addresses. It's only outside the Wasm module, in the host code, that you see it as an "offset". Wasm pointers (offsets) are as stable as the guest's memory allocation code defines them to be.

view this post on Zulip spino17 (Nov 28 2024 at 19:40):

so this means that if I use Wasm pointers (offsets) to index into MemoryView at any point of time before or after resize, it will be pointing to the same expected memory buffer ?

view this post on Zulip spino17 (Nov 28 2024 at 19:41):

given I use only save APIs available

view this post on Zulip Chris Fallin (Nov 28 2024 at 19:41):

Yes, exactly

view this post on Zulip spino17 (Nov 28 2024 at 19:43):

ahh this makes the whole picture clear about this and Wasmtime in general ! much thanks @Chris Fallin for keeping up the patience to answer these queries, really you have helped me avoiding weeks of head-scratching. Much appreciated and thanks again :)

view this post on Zulip Chris Fallin (Nov 28 2024 at 19:47):

No problem, best of luck!


Last updated: Jan 24 2025 at 00:11 UTC