diff --git a/.hgtags b/.hgtags index 3631a8c5dad..6c4f0a4fd76 100644 --- a/.hgtags +++ b/.hgtags @@ -634,3 +634,4 @@ dd5198db2e5b1ebcafe065d987c03ba9fcb50fc3 jdk-15+17 12b55fad80f30d24b1f8fdb3b947ea6465ef9518 jdk-15+21 7223c6d610343fd8323af9d07d501e01fa1a7696 jdk-15+22 f143729ca00ec14a98ea5c7f73acba88da97746e jdk-15+23 +497fd9f9129c4928fd5a876dd55e0daf6298b511 jdk-15+24 diff --git a/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.cpp b/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.cpp index 25721b7f6ce..970a2cdbb2d 100644 --- a/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.cpp +++ b/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.cpp @@ -37,26 +37,8 @@ #include "oops/oop.inline.hpp" #include "utilities/align.hpp" -// max dfs depth should not exceed size of stack -static const size_t max_dfs_depth = 5000; - -EdgeStore* DFSClosure::_edge_store = NULL; -BitSet* DFSClosure::_mark_bits = NULL; -const Edge* DFSClosure::_start_edge = NULL; -size_t DFSClosure::_max_depth = max_dfs_depth; -bool DFSClosure::_ignore_root_set = false; - -DFSClosure::DFSClosure() : - _parent(NULL), - _reference(UnifiedOopRef::encode_null()), - _depth(0) { -} - -DFSClosure::DFSClosure(DFSClosure* parent, size_t depth) : - _parent(parent), - _reference(UnifiedOopRef::encode_null()), - _depth(depth) { -} + // max dfs depth should not exceed size of stack +static const size_t max_dfs_depth = 4000; void DFSClosure::find_leaks_from_edge(EdgeStore* edge_store, BitSet* mark_bits, @@ -65,14 +47,8 @@ void DFSClosure::find_leaks_from_edge(EdgeStore* edge_store, assert(mark_bits != NULL," invariant"); assert(start_edge != NULL, "invariant"); - _edge_store = edge_store; - _mark_bits = mark_bits; - _start_edge = start_edge; - _ignore_root_set = false; - assert(_max_depth == max_dfs_depth, "invariant"); - // Depth-first search, starting from a BFS egde - DFSClosure dfs; + DFSClosure dfs(edge_store, mark_bits, start_edge); start_edge->pointee()->oop_iterate(&dfs); } @@ -81,42 +57,45 @@ void DFSClosure::find_leaks_from_root_set(EdgeStore* edge_store, assert(edge_store != NULL, "invariant"); assert(mark_bits != NULL, "invariant"); - _edge_store = edge_store; - _mark_bits = mark_bits; - _start_edge = NULL; - // Mark root set, to avoid going sideways - _max_depth = 1; - _ignore_root_set = false; - DFSClosure dfs; + DFSClosure dfs(edge_store, mark_bits, NULL); + dfs._max_depth = 1; RootSetClosure rs(&dfs); rs.process(); // Depth-first search - _max_depth = max_dfs_depth; - _ignore_root_set = true; - assert(_start_edge == NULL, "invariant"); + dfs._max_depth = max_dfs_depth; + dfs._ignore_root_set = true; rs.process(); } +DFSClosure::DFSClosure(EdgeStore* edge_store, BitSet* mark_bits, const Edge* start_edge) + :_edge_store(edge_store), _mark_bits(mark_bits), _start_edge(start_edge), + _max_depth(max_dfs_depth), _depth(0), _ignore_root_set(false) { + _reference_stack = NEW_C_HEAP_ARRAY(UnifiedOopRef, max_dfs_depth, mtTracing); +} + +DFSClosure::~DFSClosure() { + FREE_C_HEAP_ARRAY(UnifiedOopRef, _reference_stack); +} + void DFSClosure::closure_impl(UnifiedOopRef reference, const oop pointee) { assert(pointee != NULL, "invariant"); assert(!reference.is_null(), "invariant"); if (GranularTimer::is_finished()) { - return; + return; } if (_depth == 0 && _ignore_root_set) { // Root set is already marked, but we want // to continue, so skip is_marked check. assert(_mark_bits->is_marked(pointee), "invariant"); - } else { + } else { if (_mark_bits->is_marked(pointee)) { return; } } - - _reference = reference; + _reference_stack[_depth] = reference; _mark_bits->mark_obj(pointee); assert(_mark_bits->is_marked(pointee), "invariant"); @@ -127,8 +106,10 @@ void DFSClosure::closure_impl(UnifiedOopRef reference, const oop pointee) { assert(_max_depth >= 1, "invariant"); if (_depth < _max_depth - 1) { - DFSClosure next_level(this, _depth + 1); - pointee->oop_iterate(&next_level); + _depth++; + pointee->oop_iterate(this); + assert(_depth > 0, "invariant"); + _depth--; } } @@ -140,11 +121,10 @@ void DFSClosure::add_chain() { size_t idx = 0; // aggregate from depth-first search - const DFSClosure* c = this; - while (c != NULL) { + for (size_t i = 0; i <= _depth; i++) { const size_t next = idx + 1; - chain[idx++] = Edge(&chain[next], c->reference()); - c = c->parent(); + const size_t depth = _depth - i; + chain[idx++] = Edge(&chain[next], _reference_stack[depth]); } assert(_depth + 1 == idx, "invariant"); assert(array_length == idx + 1, "invariant"); diff --git a/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.hpp b/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.hpp index bf7c2d2418a..4c2be1095f7 100644 --- a/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.hpp +++ b/src/hotspot/share/jfr/leakprofiler/chains/dfsClosure.hpp @@ -36,24 +36,20 @@ class EdgeQueue; // Class responsible for iterating the heap depth-first class DFSClosure : public BasicOopIterateClosure { private: - static EdgeStore* _edge_store; - static BitSet* _mark_bits; - static const Edge*_start_edge; - static size_t _max_depth; - static bool _ignore_root_set; - DFSClosure* _parent; - UnifiedOopRef _reference; + EdgeStore* _edge_store; + BitSet* _mark_bits; + const Edge*_start_edge; + size_t _max_depth; size_t _depth; + bool _ignore_root_set; + UnifiedOopRef* _reference_stack; + + DFSClosure(EdgeStore* edge_store, BitSet* mark_bits, const Edge* start_edge); + ~DFSClosure(); void add_chain(); void closure_impl(UnifiedOopRef reference, const oop pointee); - DFSClosure* parent() const { return _parent; } - UnifiedOopRef reference() const { return _reference; } - - DFSClosure(DFSClosure* parent, size_t depth); - DFSClosure(); - public: virtual ReferenceIterationMode reference_iteration_mode() { return DO_FIELDS_EXCEPT_REFERENT; } virtual bool should_verify_oops() { return false; } diff --git a/src/hotspot/share/jfr/leakprofiler/utilities/unifiedOopRef.inline.hpp b/src/hotspot/share/jfr/leakprofiler/utilities/unifiedOopRef.inline.hpp index a615b595a5d..3aa347d6dcf 100644 --- a/src/hotspot/share/jfr/leakprofiler/utilities/unifiedOopRef.inline.hpp +++ b/src/hotspot/share/jfr/leakprofiler/utilities/unifiedOopRef.inline.hpp @@ -85,7 +85,8 @@ inline UnifiedOopRef UnifiedOopRef::encode_in_heap(const oop* ref) { } inline UnifiedOopRef UnifiedOopRef::encode_null() { - return UnifiedOopRef(); + UnifiedOopRef result = { 0 }; + return result; } inline oop UnifiedOopRef::dereference() const { diff --git a/test/jdk/ProblemList.txt b/test/jdk/ProblemList.txt index 32f9d48a455..4af8a8d1e54 100644 --- a/test/jdk/ProblemList.txt +++ b/test/jdk/ProblemList.txt @@ -628,6 +628,7 @@ java/net/MulticastSocket/SetGetNetworkInterfaceTest.java 8219083 windows- java/net/ServerSocket/AcceptInheritHandle.java 8211854 aix-ppc64 +java/net/SocketOption/AfterClose.java 8245517 linux-all ############################################################################