From 32016662c930eb3203b9ad00c0bddc070ae62918 Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Fri, 8 Feb 2019 12:55:20 +0100 Subject: [PATCH] 8217778: StringTable cleanup miscalculates amount of dead objects Reviewed-by: kbarrett --- src/hotspot/share/gc/shared/weakProcessor.cpp | 20 +++++---- .../share/gc/shared/weakProcessor.inline.hpp | 44 +++++++++++++++---- 2 files changed, 48 insertions(+), 16 deletions(-) diff --git a/src/hotspot/share/gc/shared/weakProcessor.cpp b/src/hotspot/share/gc/shared/weakProcessor.cpp index 1eed6db0f5b..97cd65b1d71 100644 --- a/src/hotspot/share/gc/shared/weakProcessor.cpp +++ b/src/hotspot/share/gc/shared/weakProcessor.cpp @@ -35,19 +35,23 @@ #include "utilities/macros.hpp" void WeakProcessor::weak_oops_do(BoolObjectClosure* is_alive, OopClosure* keep_alive) { - StringTable::reset_dead_counter(); - CountingIsAliveClosure cl(is_alive); FOR_EACH_WEAK_PROCESSOR_PHASE(phase) { if (WeakProcessorPhases::is_serial(phase)) { - WeakProcessorPhases::processor(phase)(&cl, keep_alive); + WeakProcessorPhases::processor(phase)(is_alive, keep_alive); } else { - WeakProcessorPhases::oop_storage(phase)->weak_oops_do(&cl, keep_alive); - } - if (WeakProcessorPhases::is_stringtable(phase)) { - StringTable::inc_dead_counter(cl.num_dead()); + if (WeakProcessorPhases::is_stringtable(phase)) { + StringTable::reset_dead_counter(); + + CountingSkippedIsAliveClosure cl(is_alive, keep_alive); + WeakProcessorPhases::oop_storage(phase)->oops_do(&cl); + + StringTable::inc_dead_counter(cl.num_dead() + cl.num_skipped()); + StringTable::finish_dead_counter(); + } else { + WeakProcessorPhases::oop_storage(phase)->weak_oops_do(is_alive, keep_alive); + } } } - StringTable::finish_dead_counter(); } void WeakProcessor::oops_do(OopClosure* closure) { diff --git a/src/hotspot/share/gc/shared/weakProcessor.inline.hpp b/src/hotspot/share/gc/shared/weakProcessor.inline.hpp index 3f5b7ceee79..57766424662 100644 --- a/src/hotspot/share/gc/shared/weakProcessor.inline.hpp +++ b/src/hotspot/share/gc/shared/weakProcessor.inline.hpp @@ -37,15 +37,15 @@ class BoolObjectClosure; class OopClosure; -template +template class CountingIsAliveClosure : public BoolObjectClosure { - T* _inner; + IsAlive* _inner; size_t _num_dead; size_t _num_total; public: - CountingIsAliveClosure(T* cl) : _inner(cl), _num_dead(0), _num_total(0) { } + CountingIsAliveClosure(IsAlive* cl) : _inner(cl), _num_dead(0), _num_total(0) { } virtual bool do_object_b(oop obj) { bool result = _inner->do_object_b(obj); @@ -58,6 +58,33 @@ public: size_t num_total() const { return _num_total; } }; +template +class CountingSkippedIsAliveClosure : public Closure { + CountingIsAliveClosure _counting_is_alive; + KeepAlive* _keep_alive; + + size_t _num_skipped; + +public: + CountingSkippedIsAliveClosure(IsAlive* is_alive, KeepAlive* keep_alive) : + _counting_is_alive(is_alive), _keep_alive(keep_alive), _num_skipped(0) { } + + void do_oop(oop* p) { + oop obj = *p; + if (obj == NULL) { + _num_skipped++; + } else if (_counting_is_alive.do_object_b(obj)) { + _keep_alive->do_oop(p); + } else { + *p = NULL; + } + } + + size_t num_dead() const { return _counting_is_alive.num_dead(); } + size_t num_skipped() const { return _num_skipped; } + size_t num_total() const { return _counting_is_alive.num_total() + num_skipped(); } +}; + template void WeakProcessor::Task::work(uint worker_id, IsAlive* is_alive, @@ -67,8 +94,8 @@ void WeakProcessor::Task::work(uint worker_id, worker_id, _nworkers); FOR_EACH_WEAK_PROCESSOR_PHASE(phase) { - CountingIsAliveClosure cl(is_alive); if (WeakProcessorPhases::is_serial(phase)) { + CountingIsAliveClosure cl(is_alive); uint serial_index = WeakProcessorPhases::serial_index(phase); if (_serial_phases_done.try_claim_task(serial_index)) { WeakProcessorPhaseTimeTracker pt(_phase_times, phase); @@ -78,15 +105,16 @@ void WeakProcessor::Task::work(uint worker_id, } } } else { + CountingSkippedIsAliveClosure cl(is_alive, keep_alive); WeakProcessorPhaseTimeTracker pt(_phase_times, phase, worker_id); uint storage_index = WeakProcessorPhases::oop_storage_index(phase); - _storage_states[storage_index].weak_oops_do(&cl, keep_alive); + _storage_states[storage_index].oops_do(&cl); if (_phase_times != NULL) { _phase_times->record_worker_items(worker_id, phase, cl.num_dead(), cl.num_total()); } - } - if (WeakProcessorPhases::is_stringtable(phase)) { - StringTable::inc_dead_counter(cl.num_dead()); + if (WeakProcessorPhases::is_stringtable(phase)) { + StringTable::inc_dead_counter(cl.num_dead() + cl.num_skipped()); + } } }