Skip to content

Commit

Permalink
Reimplement subclasses with a weak linked list
Browse files Browse the repository at this point in the history
This is similar to the CRuby subclasses list except for two
key features:

* We use weak references instead of pointer values; the GC vacates
  references for us.
* The linked list structure is immutable and concurrency-safe.

This impl is thread-safe and lock-free due to the immutable linked
list structure.

Adding a new class creates a new head node and atomically
reassigns it into the class. Removing a class finds that element
and vacates its reference. Replacing a class first removes the old
and then adds the new. Traversing is a matter of walking the chain
and omitting vacated references.

Periodically, the list must be rebuilt without dead references.
This is hardcoded currently to be when the list contains more than
25% vacated references.

Adding a class is an O(1) operation. Removal, replacement, and
traversal are amortized O(n).

This structure is also lighter-weight than either the original
ConcurrentWeakHashMap or any implementation of WeakHashMap
provided by the JDK, plus it has no lock overhead and very little
synchronization overhead.
  • Loading branch information
headius committed Nov 27, 2024
1 parent 09bcd0d commit 680d2e9
Showing 1 changed file with 125 additions and 48 deletions.
173 changes: 125 additions & 48 deletions core/src/main/java/org/jruby/RubyClass.java
Original file line number Diff line number Diff line change
Expand Up @@ -48,11 +48,13 @@
import static org.objectweb.asm.Opcodes.ACC_VARARGS;

import java.io.IOException;
import java.lang.ref.WeakReference;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.*;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import java.util.stream.Collectors;

import org.jruby.anno.JRubyClass;
Expand Down Expand Up @@ -93,7 +95,6 @@
import org.jruby.util.Loader;
import org.jruby.util.OneShotClassLoader;
import org.jruby.util.StringSupport;
import org.jruby.util.WeakIdentityHashMap;
import org.jruby.util.log.Logger;
import org.jruby.util.log.LoggerFactory;
import org.objectweb.asm.AnnotationVisitor;
Expand Down Expand Up @@ -1060,8 +1061,8 @@ public final Collection<RubyClass> subclasses() {
}

public Collection<RubyClass> subclasses(boolean includeDescendants) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses != null) {
SubclassNode subclassNode = this.subclassNode;
if (subclassNode != null) {
Collection<RubyClass> mine = new ArrayList<>();
subclassesInner(mine, includeDescendants);

Expand All @@ -1071,30 +1072,92 @@ public Collection<RubyClass> subclasses(boolean includeDescendants) {
}

private void subclassesInner(Collection<RubyClass> mine, boolean includeDescendants) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses != null) {
Set<RubyClass> keys = subclasses.keySet();
mine.addAll(keys);
if (includeDescendants) {
for (RubyClass klass: keys) {
klass.subclassesInner(mine, includeDescendants);
SubclassNode subclassNode = this.subclassNode;
if (subclassNode != null) {
int clearedCount = 0;
while (subclassNode != null) {
RubyClass klass = subclassNode.ref.get();
subclassNode = subclassNode.next;

if (klass == null) {
clearedCount++;
continue;
}

if (includeDescendants) klass.subclassesInner(mine, includeDescendants);
}
int newSize = mine.size();

// tidy up if more than 25% cleared references
if ((double) clearedCount / newSize > 0.25) {
cleanSubclasses();
}
}
}

private void concreteSubclasses(RubyArray<RubyClass> subs) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses != null) {
subclasses.forEach((klass, $) -> {
SubclassNode subclassNode = this.subclassNode;
if (subclassNode != null) {
int clearedCount = 0;
while (subclassNode != null) {
RubyClass klass = subclassNode.ref.get();
subclassNode = subclassNode.next;

if (klass == null) {
clearedCount++;
continue;
}

if (!klass.isSingleton()) {
if (klass.isIncluded() || klass.isPrepended()) {
klass.concreteSubclasses(subs);
} else {
subs.append(klass);
}
}
});
}
int newSize = subs.size();
subclassEstimate = newSize + 4;

// tidy up if more than 25% cleared references
if ((double) clearedCount / newSize > 0.25) {
cleanSubclasses();
}
}
}

private void cleanSubclasses() {
SubclassNode subclassNode = this.subclassNode;
SubclassNode newTop = rebuildSubclasses(subclassNode);
while (!SUBCLASS_UPDATER.compareAndSet(this, subclassNode, newTop)) {
subclassNode = this.subclassNode;
newTop = rebuildSubclasses(subclassNode);
}
}

private static SubclassNode rebuildSubclasses(SubclassNode subclassNode) {
SubclassNode newTop = null;
while (subclassNode != null) {
WeakReference<RubyClass> ref = subclassNode.ref;
RubyClass klass = ref.get();
subclassNode = subclassNode.next;
if (klass == null) continue;
newTop = new SubclassNode(ref, newTop);
}
return newTop;
}

// TODO: make into a Record
static class SubclassNode {
final SubclassNode next;
final WeakReference<RubyClass> ref;
SubclassNode(RubyClass klass, SubclassNode next) {
ref = new WeakReference<>(klass);
this.next = next;
}
SubclassNode(WeakReference<RubyClass> ref, SubclassNode next) {
this.ref = ref;
this.next = next;
}
}

Expand All @@ -1108,18 +1171,12 @@ private void concreteSubclasses(RubyArray<RubyClass> subs) {
* @param subclass The subclass to add
*/
public void addSubclass(RubyClass subclass) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses == null) {
// check again
synchronized (this) {
subclasses = this.subclasses;
if (subclasses == null) {
this.subclasses = subclasses = Collections.synchronizedMap(new WeakHashMap<>(4));
}
}
SubclassNode subclassNode = this.subclassNode;
SubclassNode newNode = new SubclassNode(subclass, subclassNode);
while (!SUBCLASS_UPDATER.compareAndSet(this, subclassNode, newNode)) {
subclassNode = this.subclassNode;
newNode = new SubclassNode(subclass, subclassNode);
}

subclasses.put(subclass, NEVER);
}

/**
Expand All @@ -1128,10 +1185,16 @@ public void addSubclass(RubyClass subclass) {
* @param subclass The subclass to remove
*/
public void removeSubclass(RubyClass subclass) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses == null) return;

subclasses.remove(subclass);
SubclassNode subclassNode = this.subclassNode;
while (subclassNode != null) {
WeakReference<RubyClass> ref = subclassNode.ref;
RubyClass klass = ref.get();
if (klass == subclass) {
ref.clear();
return;
}
subclassNode = subclassNode.next;
}
}

/**
Expand All @@ -1141,20 +1204,25 @@ public void removeSubclass(RubyClass subclass) {
* @param newSubclass The subclass to replace it with
*/
public void replaceSubclass(RubyClass subclass, RubyClass newSubclass) {
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses == null) return;

subclasses.remove(subclass);
subclasses.put(newSubclass, NEVER);
removeSubclass(subclass);
addSubclass(newSubclass);
}

/**
* make this class and all subclasses sync
*/
@Override
public void becomeSynchronized() {
// make this class and all subclasses sync
super.becomeSynchronized();
Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses != null) {
for (RubyClass subclass : subclasses.keySet()) subclass.becomeSynchronized();

SubclassNode subclassNode = this.subclassNode;
while (subclassNode != null) {
WeakReference<RubyClass> ref = subclassNode.ref;
RubyClass klass = ref.get();
if (klass != null) {
klass.becomeSynchronized();
}
subclassNode = subclassNode.next;
}
}

Expand All @@ -1174,9 +1242,14 @@ public void becomeSynchronized() {
public void invalidateCacheDescendants() {
super.invalidateCacheDescendants();

Map<RubyClass, Object> subclasses = this.subclasses;
if (subclasses != null) {
for (RubyClass subclass : subclasses.keySet()) subclass.invalidateCacheDescendants();
SubclassNode subclassNode = this.subclassNode;
while (subclassNode != null) {
WeakReference<RubyClass> ref = subclassNode.ref;
RubyClass klass = ref.get();
if (klass != null) {
klass.invalidateCacheDescendants();
}
subclassNode = subclassNode.next;
}
}

Expand All @@ -1187,12 +1260,15 @@ public void addInvalidatorsAndFlush(List<Invalidator> invalidators) {
// if we're not at boot time, don't bother fully clearing caches
if (!runtime.isBootingCore()) cachedMethods.clear();

Map<RubyClass, Object> subclasses = this.subclasses;
// no subclasses, don't bother with lock and iteration
if (subclasses == null || subclasses.isEmpty()) return;

// cascade into subclasses
for (RubyClass subclass : subclasses.keySet()) subclass.addInvalidatorsAndFlush(invalidators);
SubclassNode subclassNode = this.subclassNode;
while (subclassNode != null) {
WeakReference<RubyClass> ref = subclassNode.ref;
RubyClass klass = ref.get();
if (klass != null) {
klass.addInvalidatorsAndFlush(invalidators);
}
subclassNode = subclassNode.next;
}
}

public final Ruby getClassRuntime() {
Expand Down Expand Up @@ -3087,7 +3163,8 @@ public IRubyObject invokeFrom(ThreadContext context, CallType callType, IRubyObj
protected final Ruby runtime;
private ObjectAllocator allocator; // the default allocator
protected ObjectMarshal marshal;
private volatile Map<RubyClass, Object> subclasses;
private volatile SubclassNode subclassNode;
private static final AtomicReferenceFieldUpdater SUBCLASS_UPDATER = AtomicReferenceFieldUpdater.newUpdater(RubyClass.class, SubclassNode.class, "subclassNode");
private int subclassEstimate = -1;
public static final int CS_IDX_INITIALIZE = 0;
public enum CS_NAMES {
Expand Down

0 comments on commit 680d2e9

Please sign in to comment.