-
Notifications
You must be signed in to change notification settings - Fork 156
Performance degradation and crashes due to malloc/gc #450
Comments
Unfortunately there's nothing we can do for this - TinyGo is fundamentally flawed with -scheduler=none required environment like Proxy-Wasm. In fact, there was a solution (workaround) for this called nottinygc but not maintained anymore: af31621 |
anyway this is a dup of #349 and in any case, this is not the issue of Go SDK, but more about TinyGo itself. |
Thanks @mathetake for the quick replies! Out of curiosity how do you view the current/future utility of proxy-wasm-go-sdk in this context? Is it for low-traffic plugins? Is it waiting on full Golang wasm support? I'm wondering if there are other, less complex use cases where we should still recommend proxy-wasm-go-sdk. |
IMHO, I wouldn't recommend Proxy-Wasm to anyone regardless of languages, too fragile with little to zero benefit if you don't need sandbox. That's my personal take, not our company's though. |
I happened to find this and thought I'd add some background. Maybe it's a useful brain dump if any motivated individual appears. As you've found, the TinyGo GC implementation is not very performant, which the project honestly describes. nottinygc linked in bdwgc into TinyGo's garbage collector interface. It actually seemed to work quite well, before completely unusable performance became usable, in coraza-proxy-wasm we saw something like 300ms per request shrink to 30ms, with several hundred ms GC pauses vs 5-10s pauses. However, we found that some workloads were ok but many would run with unbounded memory usage. This isn't surprising - a 32-bit, non-randomized address space can cause conservative GC to perform poorly because pointers and normal math values overlap a lot. So I tried using bdwgc precise GC, which again brought reasonable performance to some failing workloads. But there were still many reports of memory leaks, and one thing I noticed is that the TinyGo compiler only populates information for precise GC in some cases, not all, which looked worrisome due to the fundamental issues with conservative GC. At that point, I decided the unpredictable stability is a cure worse than the disease and nottinygc is not viable. Though to go to the point of if there are any use cases where proxy-wasm-go-sdk could be recommended, maybe with nottinygc if it weren't archived, I would say they do seem to exist but it's not really possible to predict what will work until trying it in production, which IMO makes it not possible to recommend at all. I think someone could bring bdwgc directly into TinyGo with a proper native integration and there has been interest from maintainers. It would take quite some effort from a dedicated person though (I moved on from compiling Go to Wasm).- one important point would be reworking TinyGo's LLVM to generate descriptors for precise GC in bdwgc's format (I had to fragily map them in nottinygc) and to do it for all allocations. If this were done, I think runtime performance for individual TinyGo wasm programs can be reasonable. Unfortunately, that leaves another problem, where for example one Envoy plugin would have reasonable memory behavior, but if you had multiple plugins written in Go, they would all have independent GC heaps and be wasting a lot of memory. This can be solved for GC languages using the wasm-gc proposal, but as it doesn't support inner pointers, it cannot be used with Go - the proposal was designed mostly to get kotlin working in wasm on the browser. The proposal can be adjusted, but presumably from starting the conversation to actually having compiler and runtime support for a wasm-gc driven Go, it would take ~2 years which is a long time to invest. So particularly because of the second point, I personally don't believe it's worth investing in compiling Go to Wasm for any real solutions and think currently Rust and C++ are the most suited - I suspect kotlin will support WASI at some point and will become viable as well. Others may have the resources to push above changes through though and could make Go to Wasm also viable. PS: If you do still want to try Regexp's with TinyGo, you probably want to use https://github.com/wasilibs/go-re2. |
Thanks Rag for the good explanation and 100% matches my stance. I think it’s worth reopening this and pin it, and meanwhile I am going to update README that we are no longer recommending this entire Proxy-Wasm project for any use case unless you need to execute untrusted binaries which almost no one cares if they are not doing Envoy based cloud service where it needs to take client binary like you @martijneken |
Hi folks, I'm looking for advice. We are building a product around ProxyWasm and we'd love to support as many languages/SDKs as possible. We've been doing benchmarking and stress testing on various plugins, including those built with TinyGo / ProxyWasmGoSdk. In benchmarking TinyGo we are seeing high malloc performance costs and even crashes.
My core questions are:
This TinyGo documentation is worrisome:
Describe the bug / error
We see significant costs in malloc(), presumably due to GC. Here are CPU profiles of 2 plugin versions:
-print-allocs=.
shows loads of heap allocations in the Regexp libs. If we run this plugin often enough (100 seconds, ~40k iterations), it crashes on malloc (stack below)... Crash stack:What is your Envoy/Istio version?
N/A, benchmarks run with a shim ProxyWasm host implementation and v8 wasm engine.
What is the SDK version?
Go 1.22.5
What is your TinyGo version?
TinyGo 0.32.0
URL or snippet of your code including Envoy configuration
Customer proprietary, can't share exact code. I can create a min repro plugin if needed.
The text was updated successfully, but these errors were encountered: