I'm experimenting with Kotlin as a Wasm guest. It requires Wasm GC proposal to be implemented by the runtime. The only version that works currently is the prerelease one so I apologize if this is an issue related to the code that is WIP.
I'm embedding Wasmtime
in Go (I've updated Wasmtime version in wasmtime-go
manually) and running a benchmark that calls a simple guest function multiple times:
@OptIn(UnsafeWasmMemoryApi::class)
@WasmExport("test")
fun memSimple4IntsVoid(callId: ULong, dataSize: Int) {
withScopedMemoryAllocator { allocator ->
val ptr = allocator.allocate(dataSize)
val result = requestHostToStoreArguments(callId, ptr.address)
// ... error handling
val x = ptr.loadInt()
println("Received $x")
}
}
After several successful calls of this function from the host, GC heap out of memory
error happens which I assume is a Wasmtime
error. Kotlin's withScopedMemoryAllocator
is supposed to free all the memory allocated, instance.GetExport(store, "memory").Memory().Size(store)
shows that at the time when the error happens there is still just 1 page allocated so memory doesn't grow during the test.
Memory section of the guest looks like this:
Memory[1]:
- memory[0] pages: initial=1
Max memory is not limited in the module, nor do I use Wasmtime Limitter
.
I'm not sure where exactly the error happens, I've attempted to catch(e: Throwable)
in the guest and print it to stdout but it doesn't catch anything.
Is this a Kotlin issue, Wasmtime issue or this is some sort of bug/misconfiguration on my side? Thank you!
Ok, I see now that Wasm GC memory is supposed to be separate from linear memory according to the proposal. Is its size configurable?
You might have to wait for @fitzgen (he/him) to get back from vacation to get much help on GC; there aren't many of us that understand Wasmtime GC yet as (afaik) it is still a work in progress.
Yes wasm gc memory is separate from linear memory. What'll need to happen here is either:
What you're running into is most likely an instance of the first here where a GC should be performed but it's not being performed automatically. (Nick would know more). The second point here isn't currently possible from Go I believe as the GC operation isn't exposed in the C API which is what wasmtime-go builds on.
Ok, got it, thank you!
GC heap out of memory
Looks like an issue on wasmtime side -- it seems like you are out of heap (GCed) memory.
Kotlin's withScopedMemoryAllocator
uses Wasm linear memory, which is separate from the GC heap in terms of wasm.
I've attempted to catch(e: Throwable) in the guest and print it to stdout but it doesn't catch anything.
It's not an error from Kotlin, so you can't catch it.
Additionally, wasmtime doesn't support Exception Handling proposal yet, which is used for exceptions in Kotlin, so if you try to throw an exception from Kotlin code you will get a trap at that point at runtime, which basically means that execution of wasm module will be stopped.
Thanks, I'll retest when Wasm GC is officially released in Wasmtime.
I'm also seeing these errors in the released wasmtime 27, so this is not resolved yet it seems. I'm running a small function about 120 times per second that creates like 3 small GC objects each time (not retained in any way). After about 2:15 minutes wasmtime runs out of memory. (To clarify, this is not the null GC and there are no cycles)
image.png
So what seems to happen is that the GC heap runs out of memory, but before erroring out, it tries to collect all the garbage and tries allocating again. So it seems like the problem is that the GC (DRC) does not find all garbage properly (there's no cycles, so this should not happen).
hey -- just got back from vacation, still working on catching up with my inbox and everything that's happened while I was off
CryZe said:
So what seems to happen is that the GC heap runs out of memory, but before erroring out, it tries to collect all the garbage and tries allocating again. So it seems like the problem is that the GC (DRC) does not find all garbage properly (there's no cycles, so this should not happen).
FYI: the DRC collector still only does ref-counting operations shallowly (i.e. it will not transitively decrement reference counts) which means that even most acyclic garbage will lead to leaks (currently; this is not fundamental and not intended long term, this is all still just a very WIP implementation)
I'm not sure that is necessarily it, because afair these objects were all shallow to begin with. But I guess we can check again after that's addressed.
CryZe said:
I'm not sure that is necessarily it, because afair these objects were all shallow to begin with. But I guess we can check again after that's addressed.
if you can construct a small .wat
that shows memory exhaustion under the DRC collector despite not needing transitive decrementing, that would be super useful
I've reduced it to a minimal WAT file now... and indeed shallow objects are all you need. In fact, I'm not even sure it's collecting anything at all considering I can't even really make the reproduction much smaller than that: https://gist.github.com/CryZe/951bc9fa0dc265adbac3d883484e43a0
After around ~16500 calls to update
it runs out of memory, which weirdly enough has consistently been exactly this number throughout the last 1.5 weeks of me using wasmtime with GC, almost regardless of how much I allocate on the GC heap. Could it be that there's some general per call garbage left behind that is independent of the actual struct.new that I do?
And yes it's the DRC GC as I'm also getting panics like these occasionally:
image.png
Actually if the GC heap is 512 KiB then the ~16500 is probably actually 16384, which would mean each call to update
leaks exactly 32 bytes of memory.
I've just now tried different sizes and it seems like it always allocates in multiples of 32 bytes... and leaks them entirely. (The 32 bytes rounding also would explain why different types always all errored out after the same amount of update calls)
Thanks for the reduction! these things help a ton
fwiw, it is probably 16383 since the first slot is always empty so that all GC refs are "non-null"
fyi, I probably won't get to this for a couple weeks, maybe not until the new year, since I need to write up Wasmtime's end-of-year recap blog post and all that kind of stuff
and yes, we have minimum alignment and block size for all GC objects in the DRC: https://github.com/bytecodealliance/wasmtime/blob/main/crates/wasmtime/src/runtime/vm/gc/enabled/free_list.rs#L15-L21
Last updated: Jan 24 2025 at 00:11 UTC