diff --git a/src/hotspot/share/memory/iterator.hpp b/src/hotspot/share/memory/iterator.hpp index 149b1d7a87f..b5828eb1799 100644 --- a/src/hotspot/share/memory/iterator.hpp +++ b/src/hotspot/share/memory/iterator.hpp @@ -77,7 +77,8 @@ class OopIterateClosure : public OopClosure { enum ReferenceIterationMode { DO_DISCOVERY, // Apply closure and discover references DO_DISCOVERED_AND_DISCOVERY, // Apply closure to discovered field and do discovery - DO_FIELDS // Apply closure to all fields + DO_FIELDS, // Apply closure to all fields + DO_FIELDS_EXCEPT_REFERENT // Apply closure to all fields except the referent field }; // The default iteration mode is to do discovery. diff --git a/src/hotspot/share/oops/instanceRefKlass.hpp b/src/hotspot/share/oops/instanceRefKlass.hpp index f0aa82096a8..8c331139b60 100644 --- a/src/hotspot/share/oops/instanceRefKlass.hpp +++ b/src/hotspot/share/oops/instanceRefKlass.hpp @@ -131,6 +131,10 @@ class InstanceRefKlass: public InstanceKlass { template static void oop_oop_iterate_fields(oop obj, OopClosureType* closure, Contains& contains); + // Apply the closure to all fields, except the referent field. No reference discovery is done. + template + static void oop_oop_iterate_fields_except_referent(oop obj, OopClosureType* closure, Contains& contains); + template static void trace_reference_gc(const char *s, oop obj) NOT_DEBUG_RETURN; diff --git a/src/hotspot/share/oops/instanceRefKlass.inline.hpp b/src/hotspot/share/oops/instanceRefKlass.inline.hpp index d8c18e8ea93..84790a312de 100644 --- a/src/hotspot/share/oops/instanceRefKlass.inline.hpp +++ b/src/hotspot/share/oops/instanceRefKlass.inline.hpp @@ -88,6 +88,14 @@ void InstanceRefKlass::oop_oop_iterate_discovery(oop obj, ReferenceType type, Oo do_discovered(obj, closure, contains); } +template +void InstanceRefKlass::oop_oop_iterate_discovered_and_discovery(oop obj, ReferenceType type, OopClosureType* closure, Contains& contains) { + // Explicitly apply closure to the discovered field. + do_discovered(obj, closure, contains); + // Then do normal reference processing with discovery. + oop_oop_iterate_discovery(obj, type, closure, contains); +} + template void InstanceRefKlass::oop_oop_iterate_fields(oop obj, OopClosureType* closure, Contains& contains) { do_referent(obj, closure, contains); @@ -95,11 +103,8 @@ void InstanceRefKlass::oop_oop_iterate_fields(oop obj, OopClosureType* closure, } template -void InstanceRefKlass::oop_oop_iterate_discovered_and_discovery(oop obj, ReferenceType type, OopClosureType* closure, Contains& contains) { - // Explicitly apply closure to the discovered field. +void InstanceRefKlass::oop_oop_iterate_fields_except_referent(oop obj, OopClosureType* closure, Contains& contains) { do_discovered(obj, closure, contains); - // Then do normal reference processing with discovery. - oop_oop_iterate_discovery(obj, type, closure, contains); } template @@ -117,6 +122,10 @@ void InstanceRefKlass::oop_oop_iterate_ref_processing(oop obj, OopClosureType* c trace_reference_gc("do_fields", obj); oop_oop_iterate_fields(obj, closure, contains); break; + case OopIterateClosure::DO_FIELDS_EXCEPT_REFERENT: + trace_reference_gc("do_fields_except_referent", obj); + oop_oop_iterate_fields_except_referent(obj, closure, contains); + break; default: ShouldNotReachHere(); }