Recyclone is a C++ implementation of a modified version of the Recycler described in the paper A Pure Reference Counting Garbage Collector by David F. Bacon, et al.
It requires C++20.
- Concurrent garbage collector
- Threads
- Finalizer
- Monitors (mutexes and condition variables for objects)
- Weak pointers
- Types as objects
- No stop the world
- The collector only stops one thread at a time for a short period of time.
- Each thread does not need to wait unless increment/decrement queues are full or the object free list is empty.
- Header only
- Small - about 2K lines of code.
- High memory overhead - each object has at least 64 bytes of overhead.
- No compaction
- No drop-in replacement for new & delete
This mode uses POSIX signals or SuspendThread & ResumeThread for Windows.
For POSIX signals, SIGRTMAX - 1 & SIGRTMAX are used by default.
This mode uses epoch points and epoch regions which does not require POSIX signals.
See test/type.h and test/pair.h if types are not objects for garbage collection.
See test/test_type_object.cc if types are objects for garbage collection.
See test/thread.h and test/test_threads.cc.
See test/thread.h and test/test_finalizer.cc.
See test/test_monitor.cc.
See test/test_weak_pointer.cc.
Define preprocessor symbol RECYCLONE__COOPERATIVE
.
Each mutator thread has its own increment/decrement queues.
They are single producer single consumer circular queues.
This makes memory synchronization between mutators and collector simple.
Stacks are scanned conservatively.
Only the pointers that differ from the last epoch are checked as candidates.
Other references are tracked by smart pointers.
The cycle collection runs only when an increased number of live objects has exceeded a certain threshold.
The key point is that the list of root candidates is represented as an intrusive doubly linked list.
This avoids the overflow of the list. And this also allows the decrement phase to remove objects from the list and free them immediately when their reference counts have reached to zero.
Therefore, the cycle collection can be skipped unless the number of live objects increases.
Scanning object graphs is done non-recursively by using an intrusive linked list.
Increment/decrement operations do not recursively scan blacks.
Scanning blacks on every increment/decrement operation turned out to be expensive in the implementation.
The increment/decrement operations are changed to mark just only their target objects as black/purple respectively.
There is a case in which the reference count of an orange object is decremented to zero by the cyclic decrement, where a red object in a garbage cycle has the last reference to an orange object in the other candidate cycle.
Given the following candidate cycles:
-> C1 -> C0 -> *
|___|
Where
- They were found in the order of C0 and C1 by the find cycles.
- C1 has the last references to the acyclic portion of C0.
They are tested in the order of C1 and C0 by the free cycles.
If C1 passed tests and decremented reference counts in C0, then C0 has objects with the reference count zero which must be released even if C0 failed tests.
The refurbish is changed to buffer them as garbage.
After the free cycles is completed, the buffered garbage objects are released.
The MIT License (MIT)
Copyright (c) Shin-ichi MORITA
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.