# HG changeset patch # User tschatzl # Date 1553519469 -3600 # Node ID 1f9ad92e337b9b0c8b6ba1da0fd9c0b0f89c08ea # Parent b1605a40e3362ff5ffb7103347ac12830d17a6ba 8219100: Improve do_collection_pause_at_safepoint Reviewed-by: kbarrett, sangheki diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1CollectedHeap.cpp --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp Mon Mar 25 14:11:09 2019 +0100 @@ -2535,7 +2535,7 @@ // Update common counters. increment_total_collections(full /* full gc */); - if (full) { + if (full || collector_state()->in_initial_mark_gc()) { increment_old_marking_cycles_started(); } @@ -2861,16 +2861,92 @@ }; void G1CollectedHeap::start_new_collection_set() { + double start = os::elapsedTime(); + collection_set()->start_incremental_building(); clear_cset_fast_test(); guarantee(_eden.length() == 0, "eden should have been cleared"); policy()->transfer_survivors_to_cset(survivor()); + + // We redo the verification but now wrt to the new CSet which + // has just got initialized after the previous CSet was freed. + _cm->verify_no_collection_set_oops(); + + phase_times()->record_start_new_cset_time_ms((os::elapsedTime() - start) * 1000.0); } -bool -G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { +void G1CollectedHeap::calculate_collection_set(G1EvacuationInfo& evacuation_info, double target_pause_time_ms){ + policy()->finalize_collection_set(target_pause_time_ms, &_survivor); + evacuation_info.set_collectionset_regions(collection_set()->region_length()); + + _cm->verify_no_collection_set_oops(); + + if (_hr_printer.is_active()) { + G1PrintCollectionSetClosure cl(&_hr_printer); + _collection_set.iterate(&cl); + } +} + +G1HeapVerifier::G1VerifyType G1CollectedHeap::young_collection_verify_type() const { + if (collector_state()->in_initial_mark_gc()) { + return G1HeapVerifier::G1VerifyConcurrentStart; + } else if (collector_state()->in_young_only_phase()) { + return G1HeapVerifier::G1VerifyYoungNormal; + } else { + return G1HeapVerifier::G1VerifyMixed; + } +} + +void G1CollectedHeap::verify_before_young_collection(G1HeapVerifier::G1VerifyType type) { + if (VerifyRememberedSets) { + log_info(gc, verify)("[Verifying RemSets before GC]"); + VerifyRegionRemSetClosure v_cl; + heap_region_iterate(&v_cl); + } + _verifier->verify_before_gc(type); + _verifier->check_bitmaps("GC Start"); +} + +void G1CollectedHeap::verify_after_young_collection(G1HeapVerifier::G1VerifyType type) { + if (VerifyRememberedSets) { + log_info(gc, verify)("[Verifying RemSets after GC]"); + VerifyRegionRemSetClosure v_cl; + heap_region_iterate(&v_cl); + } + _verifier->verify_after_gc(type); + _verifier->check_bitmaps("GC End"); +} + +void G1CollectedHeap::expand_heap_after_young_collection(){ + size_t expand_bytes = _heap_sizing_policy->expansion_amount(); + if (expand_bytes > 0) { + // No need for an ergo logging here, + // expansion_amount() does this when it returns a value > 0. + double expand_ms; + if (!expand(expand_bytes, _workers, &expand_ms)) { + // We failed to expand the heap. Cannot do anything about it. + } + phase_times()->record_expand_heap_time(expand_ms); + } +} + +const char* G1CollectedHeap::young_gc_name() const { + if (collector_state()->in_initial_mark_gc()) { + return "Pause Young (Concurrent Start)"; + } else if (collector_state()->in_young_only_phase()) { + if (collector_state()->in_young_gc_before_mixed()) { + return "Pause Young (Prepare Mixed)"; + } else { + return "Pause Young (Normal)"; + } + } else { + return "Pause Young (Mixed)"; + } +} + +bool G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { assert_at_safepoint_on_vm_thread(); guarantee(!is_gc_active(), "collection is not reentrant"); @@ -2878,16 +2954,16 @@ return false; } - _gc_timer_stw->register_gc_start(); - GCIdMark gc_id_mark; - _gc_tracer_stw->report_gc_start(gc_cause(), _gc_timer_stw->gc_start()); SvcGCMarker sgcm(SvcGCMarker::MINOR); ResourceMark rm; policy()->note_gc_start(); + _gc_timer_stw->register_gc_start(); + _gc_tracer_stw->report_gc_start(gc_cause(), _gc_timer_stw->gc_start()); + wait_for_root_region_scanning(); print_heap_before_gc(); @@ -2907,8 +2983,7 @@ // We do not allow initial-mark to be piggy-backed on a mixed GC. assert(!collector_state()->in_initial_mark_gc() || - collector_state()->in_young_only_phase(), "sanity"); - + collector_state()->in_young_only_phase(), "sanity"); // We also do not allow mixed GCs during marking. assert(!collector_state()->mark_or_rebuild_in_progress() || collector_state()->in_young_only_phase(), "sanity"); @@ -2916,39 +2991,19 @@ // thread has completed its logging output and it's safe to signal // the CM thread, the flag's value in the policy has been reset. bool should_start_conc_mark = collector_state()->in_initial_mark_gc(); + if (should_start_conc_mark) { + _cm->gc_tracer_cm()->set_gc_cause(gc_cause()); + } // Inner scope for scope based logging, timers, and stats collection { G1EvacuationInfo evacuation_info; - if (collector_state()->in_initial_mark_gc()) { - // We are about to start a marking cycle, so we increment the - // full collection counter. - increment_old_marking_cycles_started(); - _cm->gc_tracer_cm()->set_gc_cause(gc_cause()); - } - _gc_tracer_stw->report_yc_type(collector_state()->yc_type()); GCTraceCPUTime tcpu; - G1HeapVerifier::G1VerifyType verify_type; - FormatBuffer<> gc_string("Pause Young "); - if (collector_state()->in_initial_mark_gc()) { - gc_string.append("(Concurrent Start)"); - verify_type = G1HeapVerifier::G1VerifyConcurrentStart; - } else if (collector_state()->in_young_only_phase()) { - if (collector_state()->in_young_gc_before_mixed()) { - gc_string.append("(Prepare Mixed)"); - } else { - gc_string.append("(Normal)"); - } - verify_type = G1HeapVerifier::G1VerifyYoungNormal; - } else { - gc_string.append("(Mixed)"); - verify_type = G1HeapVerifier::G1VerifyMixed; - } - GCTraceTime(Info, gc) tm(gc_string, NULL, gc_cause(), true); + GCTraceTime(Info, gc) tm(young_gc_name(), NULL, gc_cause(), true); uint active_workers = WorkerPolicy::calc_active_workers(workers()->total_workers(), workers()->active_workers(), @@ -2963,88 +3018,43 @@ G1HeapTransition heap_transition(this); size_t heap_used_bytes_before_gc = used(); - // Don't dynamically change the number of GC threads this early. A value of - // 0 is used to indicate serial work. When parallel work is done, - // it will be set. - - { // Call to jvmpi::post_class_unload_events must occur outside of active GC + { IsGCActiveMark x; gc_prologue(false); - if (VerifyRememberedSets) { - log_info(gc, verify)("[Verifying RemSets before GC]"); - VerifyRegionRemSetClosure v_cl; - heap_region_iterate(&v_cl); - } - - _verifier->verify_before_gc(verify_type); - - _verifier->check_bitmaps("GC Start"); - -#if COMPILER2_OR_JVMCI - DerivedPointerTable::clear(); -#endif - - // Please see comment in g1CollectedHeap.hpp and - // G1CollectedHeap::ref_processing_init() to see how - // reference processing currently works in G1. - - // Enable discovery in the STW reference processor - _ref_processor_stw->enable_discovery(); + G1HeapVerifier::G1VerifyType verify_type = young_collection_verify_type(); + verify_before_young_collection(verify_type); { + // The elapsed time induced by the start time below deliberately elides + // the possible verification above. + double sample_start_time_sec = os::elapsedTime(); + + // Please see comment in g1CollectedHeap.hpp and + // G1CollectedHeap::ref_processing_init() to see how + // reference processing currently works in G1. + _ref_processor_stw->enable_discovery(); + // We want to temporarily turn off discovery by the // CM ref processor, if necessary, and turn it back on // on again later if we do. Using a scoped // NoRefDiscovery object will do this. NoRefDiscovery no_cm_discovery(_ref_processor_cm); - // Forget the current alloc region (we might even choose it to be part + policy()->record_collection_pause_start(sample_start_time_sec); + + // Forget the current allocation region (we might even choose it to be part // of the collection set!). _allocator->release_mutator_alloc_region(); - // This timing is only used by the ergonomics to handle our pause target. - // It is unclear why this should not include the full pause. We will - // investigate this in CR 7178365. - // - // Preserving the old comment here if that helps the investigation: - // - // The elapsed time induced by the start time below deliberately elides - // the possible verification above. - double sample_start_time_sec = os::elapsedTime(); - - policy()->record_collection_pause_start(sample_start_time_sec); - - if (collector_state()->in_initial_mark_gc()) { - concurrent_mark()->pre_initial_mark(); - } - - policy()->finalize_collection_set(target_pause_time_ms, &_survivor); - - evacuation_info.set_collectionset_regions(collection_set()->region_length()); - - register_humongous_regions_with_cset(); - - assert(_verifier->check_cset_fast_test(), "Inconsistency in the InCSetState table."); - - // We call this after finalize_cset() to - // ensure that the CSet has been finalized. - _cm->verify_no_cset_oops(); - - if (_hr_printer.is_active()) { - G1PrintCollectionSetClosure cl(&_hr_printer); - _collection_set.iterate(&cl); - } - - // Initialize the GC alloc regions. - _allocator->init_gc_alloc_regions(evacuation_info); + calculate_collection_set(evacuation_info, target_pause_time_ms); G1ParScanThreadStateSet per_thread_states(this, workers()->active_workers(), collection_set()->young_region_length(), collection_set()->optional_region_length()); - pre_evacuate_collection_set(); + pre_evacuate_collection_set(evacuation_info); // Actually do the work... evacuate_collection_set(&per_thread_states); @@ -3052,39 +3062,12 @@ post_evacuate_collection_set(evacuation_info, &per_thread_states); - const size_t* surviving_young_words = per_thread_states.surviving_young_words(); - free_collection_set(&_collection_set, evacuation_info, surviving_young_words); - - eagerly_reclaim_humongous_regions(); - - record_obj_copy_mem_stats(); + start_new_collection_set(); + _survivor_evac_stats.adjust_desired_plab_sz(); _old_evac_stats.adjust_desired_plab_sz(); - double start = os::elapsedTime(); - start_new_collection_set(); - phase_times()->record_start_new_cset_time_ms((os::elapsedTime() - start) * 1000.0); - - if (evacuation_failed()) { - double recalculate_used_start = os::elapsedTime(); - set_used(recalculate_used()); - phase_times()->record_evac_fail_recalc_used_time((os::elapsedTime() - recalculate_used_start) * 1000.0); - - if (_archive_allocator != NULL) { - _archive_allocator->clear_used(); - } - for (uint i = 0; i < ParallelGCThreads; i++) { - if (_evacuation_failed_info_array[i].has_failed()) { - _gc_tracer_stw->report_evacuation_failed(_evacuation_failed_info_array[i]); - } - } - } else { - // The "used" of the the collection set have already been subtracted - // when they were freed. Add in the bytes evacuated. - increase_used(policy()->bytes_copied_during_gc()); - } - - if (collector_state()->in_initial_mark_gc()) { + if (should_start_conc_mark) { // We have to do this before we notify the CM threads that // they can start working to make sure that all the // appropriate initialization is done on the CM object. @@ -3098,50 +3081,16 @@ _allocator->init_mutator_alloc_region(); - { - size_t expand_bytes = _heap_sizing_policy->expansion_amount(); - if (expand_bytes > 0) { - size_t bytes_before = capacity(); - // No need for an ergo logging here, - // expansion_amount() does this when it returns a value > 0. - double expand_ms; - if (!expand(expand_bytes, _workers, &expand_ms)) { - // We failed to expand the heap. Cannot do anything about it. - } - phase_times()->record_expand_heap_time(expand_ms); - } - } - - // We redo the verification but now wrt to the new CSet which - // has just got initialized after the previous CSet was freed. - _cm->verify_no_cset_oops(); - - // This timing is only used by the ergonomics to handle our pause target. - // It is unclear why this should not include the full pause. We will - // investigate this in CR 7178365. + expand_heap_after_young_collection(); + double sample_end_time_sec = os::elapsedTime(); double pause_time_ms = (sample_end_time_sec - sample_start_time_sec) * MILLIUNITS; size_t total_cards_scanned = phase_times()->sum_thread_work_items(G1GCPhaseTimes::ScanRS, G1GCPhaseTimes::ScanRSScannedCards); policy()->record_collection_pause_end(pause_time_ms, total_cards_scanned, heap_used_bytes_before_gc); - - evacuation_info.set_collectionset_used_before(collection_set()->bytes_used_before()); - evacuation_info.set_bytes_copied(policy()->bytes_copied_during_gc()); - - if (VerifyRememberedSets) { - log_info(gc, verify)("[Verifying RemSets after GC]"); - VerifyRegionRemSetClosure v_cl; - heap_region_iterate(&v_cl); - } - - _verifier->verify_after_gc(verify_type); - _verifier->check_bitmaps("GC End"); - - assert(!_ref_processor_stw->discovery_enabled(), "Postcondition"); - _ref_processor_stw->verify_no_references_recorded(); - - // CM reference discovery will be re-enabled if necessary. } + verify_after_young_collection(verify_type); + #ifdef TRACESPINNING ParallelTaskTerminator::print_termination_counts(); #endif @@ -3157,11 +3106,6 @@ policy()->print_phases(); heap_transition.print(); - // It is not yet to safe to tell the concurrent mark to - // start as we have some optional output below. We don't want the - // output from the concurrent mark thread interfering with this - // logging output either. - _hrm->verify_optional(); _verifier->verify_region_sets_optional(); @@ -3188,13 +3132,11 @@ // that came from the pause. if (should_start_conc_mark) { - // CAUTION: after the doConcurrentMark() call below, - // the concurrent marking thread(s) could be running - // concurrently with us. Make sure that anything after - // this point does not assume that we are the only GC thread - // running. Note: of course, the actual marking work will - // not start until the safepoint itself is released in - // SuspendibleThreadSet::desynchronize(). + // CAUTION: after the doConcurrentMark() call below, the concurrent marking + // thread(s) could be running concurrently with us. Make sure that anything + // after this point does not assume that we are the only GC thread running. + // Note: of course, the actual marking work will not start until the safepoint + // itself is released in SuspendibleThreadSet::desynchronize(). do_concurrent_mark(); } @@ -3675,6 +3617,7 @@ make_pending_list_reachable(); + assert(!rp->discovery_enabled(), "Postcondition"); rp->verify_no_references_recorded(); double ref_proc_time = os::elapsedTime() - ref_proc_start; @@ -3697,7 +3640,7 @@ phase_times()->record_merge_pss_time_ms((os::elapsedTime() - merge_pss_time_start) * 1000.0); } -void G1CollectedHeap::pre_evacuate_collection_set() { +void G1CollectedHeap::pre_evacuate_collection_set(G1EvacuationInfo& evacuation_info) { _expand_heap_after_alloc_failure = true; _evacuation_failed = false; @@ -3705,11 +3648,23 @@ _hot_card_cache->reset_hot_cache_claimed_index(); _hot_card_cache->set_use_cache(false); + // Initialize the GC alloc regions. + _allocator->init_gc_alloc_regions(evacuation_info); + + register_humongous_regions_with_cset(); + assert(_verifier->check_cset_fast_test(), "Inconsistency in the InCSetState table."); + rem_set()->prepare_for_oops_into_collection_set_do(); _preserved_marks_set.assert_empty(); +#if COMPILER2_OR_JVMCI + DerivedPointerTable::clear(); +#endif + // InitialMark needs claim bits to keep track of the marked-through CLDs. if (collector_state()->in_initial_mark_gc()) { + concurrent_mark()->pre_initial_mark(); + double start_clear_claimed_marks = os::elapsedTime(); ClassLoaderDataGraph::clear_claimed_marks(); @@ -3916,19 +3871,34 @@ phase_times()->record_string_deduplication_time(string_cleanup_time_ms); } + _allocator->release_gc_alloc_regions(evacuation_info); + if (evacuation_failed()) { restore_after_evac_failure(); // Reset the G1EvacuationFailureALot counters and flags - // Note: the values are reset only when an actual - // evacuation failure occurs. NOT_PRODUCT(reset_evacuation_should_fail();) + + double recalculate_used_start = os::elapsedTime(); + set_used(recalculate_used()); + phase_times()->record_evac_fail_recalc_used_time((os::elapsedTime() - recalculate_used_start) * 1000.0); + + if (_archive_allocator != NULL) { + _archive_allocator->clear_used(); + } + for (uint i = 0; i < ParallelGCThreads; i++) { + if (_evacuation_failed_info_array[i].has_failed()) { + _gc_tracer_stw->report_evacuation_failed(_evacuation_failed_info_array[i]); + } + } + } else { + // The "used" of the the collection set have already been subtracted + // when they were freed. Add in the bytes evacuated. + increase_used(policy()->bytes_copied_during_gc()); } _preserved_marks_set.assert_empty(); - _allocator->release_gc_alloc_regions(evacuation_info); - merge_per_thread_state_info(per_thread_states); // Reset and re-enable the hot card cache. @@ -3940,6 +3910,16 @@ purge_code_root_memory(); redirty_logged_cards(); + + free_collection_set(&_collection_set, evacuation_info, per_thread_states->surviving_young_words()); + + eagerly_reclaim_humongous_regions(); + + record_obj_copy_mem_stats(); + + evacuation_info.set_collectionset_used_before(collection_set()->bytes_used_before()); + evacuation_info.set_bytes_copied(policy()->bytes_copied_during_gc()); + #if COMPILER2_OR_JVMCI double start = os::elapsedTime(); DerivedPointerTable::update_pointers(); diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1CollectedHeap.hpp --- a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp Mon Mar 25 14:11:09 2019 +0100 @@ -357,6 +357,8 @@ assert(Thread::current()->is_VM_thread(), "current thread is not VM thread"); \ } while (0) + const char* young_gc_name() const; + // The young region list. G1EdenRegions _eden; G1SurvivorRegions _survivor; @@ -730,14 +732,21 @@ // to the GC locker being active, true otherwise bool do_collection_pause_at_safepoint(double target_pause_time_ms); + G1HeapVerifier::G1VerifyType young_collection_verify_type() const; + void verify_before_young_collection(G1HeapVerifier::G1VerifyType type); + void verify_after_young_collection(G1HeapVerifier::G1VerifyType type); + + void calculate_collection_set(G1EvacuationInfo& evacuation_info, double target_pause_time_ms); + // Actually do the work of evacuating the collection set. void evacuate_collection_set(G1ParScanThreadStateSet* per_thread_states); void evacuate_optional_collection_set(G1ParScanThreadStateSet* per_thread_states); void evacuate_optional_regions(G1ParScanThreadStateSet* per_thread_states, G1OptionalCSet* ocset); - void pre_evacuate_collection_set(); + void pre_evacuate_collection_set(G1EvacuationInfo& evacuation_info); void post_evacuate_collection_set(G1EvacuationInfo& evacuation_info, G1ParScanThreadStateSet* pss); + void expand_heap_after_young_collection(); // Update object copying statistics. void record_obj_copy_mem_stats(); diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1ConcurrentMark.cpp --- a/src/hotspot/share/gc/g1/g1ConcurrentMark.cpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1ConcurrentMark.cpp Mon Mar 25 14:11:09 2019 +0100 @@ -734,7 +734,9 @@ }; void G1ConcurrentMark::pre_initial_mark() { - // Initialize marking structures. This has to be done in a STW phase. + assert_at_safepoint_on_vm_thread(); + + // Reset marking state. reset(); // For each region note start of marking. @@ -1944,7 +1946,7 @@ } }; -void G1ConcurrentMark::verify_no_cset_oops() { +void G1ConcurrentMark::verify_no_collection_set_oops() { assert(SafepointSynchronize::is_at_safepoint(), "should be at a safepoint"); if (!_g1h->collector_state()->mark_or_rebuild_in_progress()) { return; diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1ConcurrentMark.hpp --- a/src/hotspot/share/gc/g1/g1ConcurrentMark.hpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1ConcurrentMark.hpp Mon Mar 25 14:11:09 2019 +0100 @@ -579,7 +579,7 @@ // Verify that there are no collection set oops on the stacks (taskqueues / // global mark stack) and fingers (global / per-task). // If marking is not in progress, it's a no-op. - void verify_no_cset_oops() PRODUCT_RETURN; + void verify_no_collection_set_oops() PRODUCT_RETURN; inline bool do_yield_check(); diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1Policy.cpp --- a/src/hotspot/share/gc/g1/g1Policy.cpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1Policy.cpp Mon Mar 25 14:11:09 2019 +0100 @@ -1188,9 +1188,11 @@ return (uint) result; } -void G1Policy::finalize_collection_set(double target_pause_time_ms, G1SurvivorRegions* survivor) { +uint G1Policy::finalize_collection_set(double target_pause_time_ms, G1SurvivorRegions* survivor) { double time_remaining_ms = _collection_set->finalize_young_part(target_pause_time_ms, survivor); _collection_set->finalize_old_part(time_remaining_ms); + + return _collection_set->region_length(); } void G1Policy::transfer_survivors_to_cset(const G1SurvivorRegions* survivors) { diff -r b1605a40e336 -r 1f9ad92e337b src/hotspot/share/gc/g1/g1Policy.hpp --- a/src/hotspot/share/gc/g1/g1Policy.hpp Mon Mar 25 08:37:32 2019 -0400 +++ b/src/hotspot/share/gc/g1/g1Policy.hpp Mon Mar 25 14:11:09 2019 +0100 @@ -344,7 +344,7 @@ bool next_gc_should_be_mixed(const char* true_action_str, const char* false_action_str) const; - void finalize_collection_set(double target_pause_time_ms, G1SurvivorRegions* survivor); + uint finalize_collection_set(double target_pause_time_ms, G1SurvivorRegions* survivor); private: // Set the state to start a concurrent marking cycle and clear // _initiate_conc_mark_if_possible because it has now been diff -r b1605a40e336 -r 1f9ad92e337b test/hotspot/jtreg/gc/g1/TestVerifyGCType.java --- a/test/hotspot/jtreg/gc/g1/TestVerifyGCType.java Mon Mar 25 08:37:32 2019 -0400 +++ b/test/hotspot/jtreg/gc/g1/TestVerifyGCType.java Mon Mar 25 14:11:09 2019 +0100 @@ -174,9 +174,9 @@ private static void verifyType(CollectionInfo ci, boolean shouldExist, String pattern) { if (shouldExist) { - Asserts.assertTrue(ci.containsVerification(pattern), "Missing expected verification for: " + ci.getName()); + Asserts.assertTrue(ci.containsVerification(pattern), "Missing expected verification pattern " + pattern + " for: " + ci.getName()); } else { - Asserts.assertFalse(ci.containsVerification(pattern), "Found unexpected verification for: " + ci.getName()); + Asserts.assertFalse(ci.containsVerification(pattern), "Found unexpected verification pattern " + pattern + " for: " + ci.getName()); } }