11 releases
0.2.0-alpha.7 | Feb 14, 2022 |
---|---|
0.2.0-alpha.6 | Dec 20, 2021 |
0.2.0-alpha.5 | Aug 18, 2021 |
0.2.0-alpha.4 | Jul 10, 2021 |
0.1.0 | Jun 22, 2020 |
#395 in Memory management
Used in 4 crates
270KB
4.5K
SLoC
ZeroGc
[WIP] Zero overhead tracing garbage collection for rust.
Features
- Easy to use, since
Gc<T>
isCopy
and coerces to a reference. - Absolutely zero overhead when modifying pointers, since
Gc<T>
isCopy
. - Implementation agnostic API
- Unsafe code has complete freedom to manipulate garbage collected pointers, and it doesn't need to understand the distinction
- Uses rust's lifetime system to ensure all roots are known at explicit safepoints, without any runtime overhead.
- Collection can only happen with an explicit
safepoint
call and has no overhead between these calls, - API supports moving objects (allowing copying/generational GCs)
Instead of requiring compiler support to track GC roots (like Java/Go), it uses a shadow stack to keep track of GC roots. Collections can only happen at explicit safepoint.
It uses Rust's lifetime system to ensure that freed garbage isn't accessed after a collection. Allocated objects are tied to the lifetime of the garbage collector. A safepoint (potential collection) is treated as a mutation by the borrow checker. Without zerogc (using a Vec or typed-arena), this would invalidate all previously allocated pointers. However, the any 'roots' given to the safepoint are rebound to the new lifetime (via dark magic).
This API aims to be implementation-agnostic,
simply defining the Trace
trait and the interface for safepoints.
Right now the only implementation is zerogc-simple
,
which is a basic mark-sweep collector.
It's relatively fast and lightweight, making it a good default.
It uses fast arena allocation for small objects (optional; on by default) and
falls back to the system allocator for everything else.
In spite of the mark/sweep collector's simplicity, it's reasonably fast and is able to compete with production-quality collectors like Go/Java.
The library is mostly undocumented, since I expect it to change significantly in the future. See the binary-trees example for a basic sample.
Status
This is extremely experimental. It's really uncharted territory in the way it uses the rust borrow checker. It seems to be sound, but I have no way of really knowing for sure.
The simple mark/sweep collector passes basic tests, but it still relies on a lot of unsafe code internally.
Eventually I plan to use this in a language VM, so it needs to be very flexible! I want to support both simple and complex collectors with the same API.
There was previously a copying collector (which worked) 511be539228e7142, but I removed it due to high memory usage.
Motivation
I was originally inspired to create a safe abstraction for garbage collection by rust gc
but I wanted it to have zero runtime overhead and pointers that are Copy
.
The main problem that the rust-gc solves is finding the 'roots' of garbage collection. His collector uses runtime tracking to maintain a reference to every GC object.
I'm familiar with some JIT implementations, and I know they tend to use safepoints to explicitly control where garbage collections can happen. Normally this is an unsafe operation. All live references on the stack must be given on the stack or use after. Eventually I realized I could use the borrow checker to restrict the usage of roots around safepoints. I was inspired in part by the way indexing uses lifetimes to enforce the validity of indexes.
Our collector only has runtime overhead at specific safepoints, when the shadow-stack is being updated.
Not only is this faster than runtime tracking of every single pointer or conservative collection, but it is more flexible. It paves the way for any combination of generational, incremental, and copying garbage collection.
Prior Art
Since the original rust gc others have attempted to make zero-overhead collectors. I was not aware of this when I started this project.
- This is extremely impressive. It appears to be the only other collector
where
Gc: Copy
and coerces to a reference. - However, the collectors have to support pinning of roots (which we do not)
- See this blog post series for more
- Unfortunately this has a rather clunky interface to Rust code, so it wouldn't be suitable .
- They implement a basic List VM as a proof of concept.
- Seems like a similar idea, implemented a little later.
- Works around the awkwardness of lifetimes by using a future-like API.
- Like our collector, safepoints are explicitly requested by the user
- However instead of attempting to rebind lifetimes, they attempt to use futures to build state machines
Disadvantages
- The garbage collector can only run in response to an explicit
safepoint!
, not memory pressure,- This is a fundamental design limitation.
- One can think of this as a feature, since garbage collection can be restricted to specific times and places.
- The user must be liberal about inserting safepoints
- Generally high-level languages which use GCs automatically insert calls to safe points
- Their safepoints tend to be lower-overhead than ours, so you need to balance the cost/benefit
- Implementing
GcSafe
for a type prevents it from having a explicitDrop
implementation.- This is needed to ensure that the destructors don't do bad things, since we don't want to deal with finalizers.
- Of course unsafe code isn't bound by this restriction, since it's assumed to behave properly (and there is an opt-out if you know you're safe).
Implementation Flaws
None of these are fundemental design flaws. They can all be fixed (and should).
- Currently, unable to return a garbage collected type from a method that uses a safepoint
- This is not a fundamental limitation of the design. I have plans to fix the API to support this.
- Unfortunately, this is almost crippling for many use cases, but can be worked around with an 'unchecked_safepoint'
- Currently, unable to use short lifetimes in garbage collected code.
- This is because
Gc<'gc, T>
currently requiresT: 'gc
- It is possible to relax this restriction, but may make things more awkward to use...
- This is because
- Restriction on borrowing possibly aliased vectors of GC memory
- It is difficult to enforce Rust's mutability rules with possibly shared gc memory (See issue #30)
- Implementation isn't generational (yet)
- Awkward to use from a generic context, because the API hasn't yet taken full advantage of generic associated typesd.
Dependencies
~2.3–3.5MB
~62K SLoC