diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java index a1c7db2f40c..a3bf1596824 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java @@ -383,7 +383,25 @@ assert factories != null : "sanity"; /** * Cache for speeding up {@link #fromClass(Class)}. */ - @NativeImageReinitialize private volatile ClassValue> resolvedJavaType; + @NativeImageReinitialize private volatile ClassValue> resolvedJavaType; + + /** + * To avoid calling ClassValue.remove to refresh the weak reference, which + * under certain circumstances can lead to an infinite loop, we use a + * permanent holder with a mutable field that we refresh. + */ + private static class WeakReferenceHolder { + private volatile WeakReference ref; + WeakReferenceHolder(T value) { + set(value); + } + void set(T value) { + ref = new WeakReference(value); + } + T get() { + return ref.get(); + } + }; @NativeImageReinitialize private HashMap> resolvedJavaTypes; @@ -485,27 +503,25 @@ assert factories != null : "sanity"; if (resolvedJavaType == null) { synchronized (this) { if (resolvedJavaType == null) { - resolvedJavaType = new ClassValue>() { + resolvedJavaType = new ClassValue>() { @Override - protected WeakReference computeValue(Class type) { - return new WeakReference<>(createClass(type)); + protected WeakReferenceHolder computeValue(Class type) { + return new WeakReferenceHolder<>(createClass(type)); } }; } } } - HotSpotResolvedJavaType javaType = null; - while (javaType == null) { - WeakReference type = resolvedJavaType.get(javaClass); - javaType = type.get(); - if (javaType == null) { - /* - * If the referent has become null, clear out the current value and let computeValue - * above create a new value. Reload the value in a loop because in theory the - * WeakReference referent can be reclaimed at any point. - */ - resolvedJavaType.remove(javaClass); - } + + WeakReferenceHolder ref = resolvedJavaType.get(javaClass); + HotSpotResolvedJavaType javaType = ref.get(); + if (javaType == null) { + /* + * If the referent has become null, create a new value and + * update cached weak reference. + */ + javaType = createClass(javaClass); + ref.set(javaType); } return javaType; } diff --git a/test/hotspot/jtreg/ProblemList-graal.txt b/test/hotspot/jtreg/ProblemList-graal.txt index 09cca07e255..a05d6eb14d7 100644 --- a/test/hotspot/jtreg/ProblemList-graal.txt +++ b/test/hotspot/jtreg/ProblemList-graal.txt @@ -210,8 +210,6 @@ serviceability/tmtools/jstat/GcTest02.java 81 serviceability/tmtools/jstat/GcCapacityTest.java 8196611 generic-all serviceability/jvmti/HeapMonitor/MyPackage/HeapMonitorMultiArrayTest.java 8196611 generic-all -vmTestbase/gc/lock/jvmti/alloc/jvmtialloclock02/TestDescription.java 8218700 generic-all - vmTestbase/nsk/jdb/unmonitor/unmonitor001/unmonitor001.java 8218701 generic-all vmTestbase/nsk/jdb/clear/clear003/clear003.java 8218701 generic-all