Skip to content

Commit 958ea3b

Browse files
author
William Kemper
committedMar 16, 2022
Allow old collections to span degenerated young collections
Reviewed-by: kdnilsen
1 parent a2b5be2 commit 958ea3b

16 files changed

+139
-99
lines changed
 

‎src/hotspot/share/gc/shenandoah/shenandoahConcurrentGC.cpp

+19-15
Original file line numberDiff line numberDiff line change
@@ -309,14 +309,6 @@ void ShenandoahConcurrentGC::entry_init_mark() {
309309
ShenandoahWorkerPolicy::calc_workers_for_init_marking(),
310310
"init marking");
311311

312-
if (ShenandoahHeap::heap()->mode()->is_generational()
313-
&& (_generation->generation_mode() == YOUNG || (_generation->generation_mode() == GLOBAL && ShenandoahVerify))) {
314-
// The current implementation of swap_remembered_set() copies the write-card-table
315-
// to the read-card-table. The remembered sets are also swapped for GLOBAL collections
316-
// so that the verifier works with the correct copy of the card table when verifying.
317-
_generation->swap_remembered_set();
318-
}
319-
320312
op_init_mark();
321313
}
322314

@@ -607,6 +599,25 @@ void ShenandoahConcurrentGC::op_init_mark() {
607599
assert(!_generation->is_mark_complete(), "should not be complete");
608600
assert(!heap->has_forwarded_objects(), "No forwarded objects on this path");
609601

602+
603+
if (heap->mode()->is_generational()) {
604+
if (_generation->generation_mode() == YOUNG || (_generation->generation_mode() == GLOBAL && ShenandoahVerify)) {
605+
// The current implementation of swap_remembered_set() copies the write-card-table
606+
// to the read-card-table. The remembered sets are also swapped for GLOBAL collections
607+
// so that the verifier works with the correct copy of the card table when verifying.
608+
_generation->swap_remembered_set();
609+
}
610+
611+
if (_generation->generation_mode() == GLOBAL) {
612+
heap->cancel_old_gc();
613+
} else if (heap->is_concurrent_old_mark_in_progress()) {
614+
// Purge the SATB buffers, transferring any valid, old pointers to the
615+
// old generation mark queue. Any pointers in a young region will be
616+
// abandoned.
617+
heap->transfer_old_pointers_from_satb();
618+
}
619+
}
620+
610621
if (ShenandoahVerify) {
611622
heap->verifier()->verify_before_concmark();
612623
}
@@ -1131,13 +1142,6 @@ void ShenandoahConcurrentGC::op_final_updaterefs() {
11311142

11321143
heap->update_heap_region_states(true /*concurrent*/);
11331144

1134-
if (heap->is_concurrent_old_mark_in_progress()) {
1135-
// Purge the SATB buffers, transferring any valid, old pointers to the
1136-
// old generation mark queue. From here on, no mutator will have access
1137-
// to anything that will be trashed and recycled.
1138-
heap->purge_old_satb_buffers(false /* abandon */);
1139-
}
1140-
11411145
heap->set_update_refs_in_progress(false);
11421146
heap->set_has_forwarded_objects(false);
11431147

‎src/hotspot/share/gc/shenandoah/shenandoahControlThread.cpp

+9-7
Original file line numberDiff line numberDiff line change
@@ -467,7 +467,11 @@ void ShenandoahControlThread::service_concurrent_old_cycle(const ShenandoahHeap*
467467
young_generation->set_mark_incomplete();
468468
old_generation->set_mark_incomplete();
469469
service_concurrent_cycle(young_generation, cause, true);
470-
if (!heap->cancelled_gc()) {
470+
if (heap->cancelled_gc()) {
471+
// Young generation bootstrap cycle has failed. Concurrent mark for old generation
472+
// is not going to resume after degenerated young cycle completes.
473+
log_info(gc)("Bootstrap cycle for old generation was cancelled.");
474+
} else {
471475
// Reset the degenerated point. Normally this would happen at the top
472476
// of the control loop, but here we have just completed a young cycle
473477
// which has bootstrapped the old concurrent marking.
@@ -663,15 +667,13 @@ bool ShenandoahControlThread::service_stw_degenerated_cycle(GCCause::Cause cause
663667
ShenandoahGCSession session(cause, _degen_generation);
664668

665669
ShenandoahDegenGC gc(point, _degen_generation);
666-
667-
// Just in case degenerated cycle preempted old-gen marking, clear the old-gen task queues.
668-
heap->young_generation()->set_old_gen_task_queues(NULL);
669-
670670
gc.collect(cause);
671671

672672
assert(heap->young_generation()->task_queues()->is_empty(), "Unexpected young generation marking tasks");
673-
assert(heap->old_generation()->task_queues()->is_empty(), "Unexpected old generation marking tasks");
674-
assert(heap->global_generation()->task_queues()->is_empty(), "Unexpected global generation marking tasks");
673+
if (_degen_generation->generation_mode() == GLOBAL) {
674+
assert(heap->old_generation()->task_queues()->is_empty(), "Unexpected old generation marking tasks");
675+
assert(heap->global_generation()->task_queues()->is_empty(), "Unexpected global generation marking tasks");
676+
}
675677

676678
_degen_generation->heuristics()->record_success_degenerated();
677679
heap->shenandoah_policy()->record_success_degenerated();

‎src/hotspot/share/gc/shenandoah/shenandoahDegeneratedGC.cpp

+34-9
Original file line numberDiff line numberDiff line change
@@ -90,13 +90,21 @@ void ShenandoahDegenGC::op_degenerated() {
9090
// some phase, we have to upgrade the Degenerate GC to Full GC.
9191
heap->clear_cancelled_gc(true /* clear oom handler */);
9292

93-
// We can't easily clear the old mark in progress flag because it must be done
94-
// on a safepoint (not sure if that is a hard requirement). At any rate, once
95-
// we are in a degenerated cycle, there should be no more old marking.
96-
if (heap->is_concurrent_old_mark_in_progress()) {
97-
heap->old_generation()->cancel_marking();
93+
#ifdef ASSERT
94+
if (heap->mode()->is_generational()) {
95+
if (_generation->generation_mode() == GenerationMode::GLOBAL) {
96+
// We can only get to a degenerated global cycle _after_ a concurrent global cycle
97+
// has been cancelled. In which case, we expect the concurrent global cycle to have
98+
// cancelled the old gc already.
99+
assert(!heap->is_old_gc_active(), "Old GC should not be active during global cycle.");
100+
}
101+
102+
if (!heap->is_concurrent_old_mark_in_progress()) {
103+
// If we are not marking the old generation, there should be nothing in the old mark queues
104+
assert(heap->old_generation()->task_queues()->is_empty(), "Old gen task queues should be empty.");
105+
}
98106
}
99-
assert(heap->old_generation()->task_queues()->is_empty(), "Old gen task queues should be empty.");
107+
#endif
100108

101109
ShenandoahMetricsSnapshot metrics;
102110
metrics.snap_before();
@@ -113,6 +121,14 @@ void ShenandoahDegenGC::op_degenerated() {
113121
// we can do the most aggressive degen cycle, which includes processing references and
114122
// class unloading, unless those features are explicitly disabled.
115123

124+
if (heap->is_concurrent_old_mark_in_progress()) {
125+
// We have come straight into a degenerated cycle without running a concurrent cycle
126+
// first and the SATB barrier is enabled to support concurrent old marking. The SATB buffer
127+
// may hold a mix of old and young pointers. The old pointers need to be transferred
128+
// to the old generation mark queues and the young pointers are _not_ part of this
129+
// snapshot, so they must be dropped here.
130+
heap->transfer_old_pointers_from_satb();
131+
}
116132

117133
// Note that we can only do this for "outside-cycle" degens, otherwise we would risk
118134
// changing the cycle parameters mid-cycle during concurrent -> degenerated handover.
@@ -125,8 +141,17 @@ void ShenandoahDegenGC::op_degenerated() {
125141

126142
case _degenerated_roots:
127143
// Degenerated from concurrent root mark, reset the flag for STW mark
128-
if (heap->is_concurrent_mark_in_progress()) {
129-
heap->cancel_concurrent_mark();
144+
if (!heap->mode()->is_generational()) {
145+
if (heap->is_concurrent_mark_in_progress()) {
146+
heap->cancel_concurrent_mark();
147+
}
148+
} else {
149+
if (_generation->is_concurrent_mark_in_progress()) {
150+
// We want to allow old generation marking to be punctuated by young collections
151+
// (even if they have degenerated). If this is a global cycle, we'd have cancelled
152+
// the entire old gc before coming into this switch.
153+
_generation->cancel_marking();
154+
}
130155
}
131156

132157
if (_degen_point == ShenandoahDegenPoint::_degenerated_roots) {
@@ -293,7 +318,7 @@ void ShenandoahDegenGC::op_reset() {
293318
}
294319

295320
void ShenandoahDegenGC::op_mark() {
296-
assert(!ShenandoahHeap::heap()->is_concurrent_mark_in_progress(), "Should be reset");
321+
assert(!_generation->is_concurrent_mark_in_progress(), "Should be reset");
297322
ShenandoahGCPhase phase(ShenandoahPhaseTimings::degen_gc_stw_mark);
298323
ShenandoahSTWMark mark(_generation, false /*full gc*/);
299324
mark.mark();

‎src/hotspot/share/gc/shenandoah/shenandoahFullGC.cpp

+5
Original file line numberDiff line numberDiff line change
@@ -194,6 +194,11 @@ void ShenandoahFullGC::do_it(GCCause::Cause gc_cause) {
194194
heap->reset_old_evac_expended();
195195
heap->set_promotion_reserve(0);
196196

197+
if (heap->mode()->is_generational()) {
198+
// Full GC supersedes any marking or coalescing in old generation.
199+
heap->cancel_old_gc();
200+
}
201+
197202
if (ShenandoahVerify) {
198203
heap->verifier()->verify_before_fullgc();
199204
}

‎src/hotspot/share/gc/shenandoah/shenandoahGeneration.cpp

+2-3
Original file line numberDiff line numberDiff line change
@@ -397,12 +397,11 @@ ShenandoahMarkingContext* ShenandoahGeneration::complete_marking_context() {
397397

398398
void ShenandoahGeneration::cancel_marking() {
399399
if (is_concurrent_mark_in_progress()) {
400-
set_concurrent_mark_in_progress(false);
400+
set_mark_incomplete();
401401
}
402-
set_mark_incomplete();
403402
_task_queues->clear();
404-
405403
ref_processor()->abandon_partial_discovery();
404+
set_concurrent_mark_in_progress(false);
406405
}
407406

408407
ShenandoahGeneration::ShenandoahGeneration(GenerationMode generation_mode,

‎src/hotspot/share/gc/shenandoah/shenandoahGeneration.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -109,13 +109,13 @@ class ShenandoahGeneration : public CHeapObj<mtGC> {
109109
void merge_write_table();
110110

111111
// Used by concurrent and degenerated GC to reset regions.
112-
virtual void prepare_gc(bool do_old_gc_bootstrap);
112+
void prepare_gc(bool do_old_gc_bootstrap);
113113

114114
// Return true iff prepared collection set includes at least one old-gen HeapRegion.
115115
virtual bool prepare_regions_and_collection_set(bool concurrent);
116116

117117
// Cancel marking (used by Full collect and when cancelling cycle).
118-
void cancel_marking();
118+
virtual void cancel_marking();
119119

120120
// Return true if this region is affiliated with this generation.
121121
virtual bool contains(ShenandoahHeapRegion* region) const = 0;

‎src/hotspot/share/gc/shenandoah/shenandoahGlobalGeneration.cpp

+2-14
Original file line numberDiff line numberDiff line change
@@ -58,14 +58,12 @@ size_t ShenandoahGlobalGeneration::available() const {
5858

5959
void ShenandoahGlobalGeneration::set_concurrent_mark_in_progress(bool in_progress) {
6060
ShenandoahHeap* heap = ShenandoahHeap::heap();
61-
if (in_progress && heap->is_concurrent_old_mark_in_progress()) {
61+
if (in_progress && heap->mode()->is_generational()) {
6262
// Global collection has preempted an old generation mark. This is fine
6363
// because the global generation includes the old generation, but we
6464
// want the global collect to start from a clean slate and we don't want
6565
// any stale state in the old generation.
66-
heap->purge_old_satb_buffers(true /* abandon */);
67-
heap->old_generation()->cancel_marking();
68-
heap->young_generation()->set_old_gen_task_queues(nullptr);
66+
heap->cancel_old_gc();
6967
}
7068

7169
heap->set_concurrent_young_mark_in_progress(in_progress);
@@ -88,13 +86,3 @@ bool ShenandoahGlobalGeneration::is_concurrent_mark_in_progress() {
8886
ShenandoahHeap* heap = ShenandoahHeap::heap();
8987
return heap->is_concurrent_mark_in_progress();
9088
}
91-
92-
void ShenandoahGlobalGeneration::prepare_gc(bool do_old_gc_bootstrap) {
93-
ShenandoahGeneration::prepare_gc(do_old_gc_bootstrap);
94-
95-
ShenandoahHeap* heap = ShenandoahHeap::heap();
96-
if (heap->mode()->is_generational()) {
97-
heap->cancel_mixed_collections();
98-
}
99-
}
100-

‎src/hotspot/share/gc/shenandoah/shenandoahGlobalGeneration.hpp

-2
Original file line numberDiff line numberDiff line change
@@ -42,8 +42,6 @@ class ShenandoahGlobalGeneration : public ShenandoahGeneration {
4242
virtual size_t used() const override;
4343
virtual size_t available() const override;
4444

45-
virtual void prepare_gc(bool do_old_gc_bootstrap) override;
46-
4745
virtual void set_concurrent_mark_in_progress(bool in_progress) override;
4846

4947
bool contains(ShenandoahHeapRegion* region) const override;

‎src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp

+24-15
Original file line numberDiff line numberDiff line change
@@ -981,9 +981,26 @@ void ShenandoahHeap::retire_plab(PLAB* plab) {
981981
}
982982
}
983983

984-
void ShenandoahHeap::cancel_mixed_collections() {
984+
void ShenandoahHeap::cancel_old_gc() {
985+
shenandoah_assert_safepoint();
985986
assert(_old_generation != NULL, "Should only have mixed collections in generation mode.");
987+
log_info(gc)("Terminating old gc cycle.");
988+
989+
// Stop marking
990+
old_generation()->cancel_marking();
991+
// Stop coalescing undead objects
992+
set_concurrent_prep_for_mixed_evacuation_in_progress(false);
993+
// Stop tracking old regions
986994
old_heuristics()->abandon_collection_candidates();
995+
// Remove old generation access to young generation mark queues
996+
young_generation()->set_old_gen_task_queues(nullptr);
997+
}
998+
999+
bool ShenandoahHeap::is_old_gc_active() {
1000+
return is_concurrent_old_mark_in_progress()
1001+
|| is_concurrent_prep_for_mixed_evacuation_in_progress()
1002+
|| old_heuristics()->unprocessed_old_or_hidden_collection_candidates() > 0
1003+
|| young_generation()->old_gen_task_queues() != nullptr;
9871004
}
9881005

9891006
void ShenandoahHeap::coalesce_and_fill_old_regions() {
@@ -2111,18 +2128,10 @@ bool ShenandoahHeap::try_cancel_gc() {
21112128
assert(prev == NOT_CANCELLED, "must be NOT_CANCELLED");
21122129
Thread* thread = Thread::current();
21132130
if (thread->is_Java_thread()) {
2114-
JavaThread* java_thread = JavaThread::cast(thread);
2115-
if (java_thread->thread_state() == _thread_in_Java) {
2116-
// ThreadBlockInVM requires thread state to be _thread_in_vm. If we are in Java, safely transition thread state.
2117-
ThreadInVMfromJava transition(java_thread);
2118-
// We need to provide a safepoint here. Otherwise we might spin forever if a SP is pending.
2119-
ThreadBlockInVM sp(JavaThread::cast(thread));
2120-
SpinPause();
2121-
} else {
2122-
// We need to provide a safepoint here. Otherwise we might spin forever if a SP is pending.
2123-
ThreadBlockInVM sp(JavaThread::cast(thread));
2124-
SpinPause();
2125-
}
2131+
// We need to provide a safepoint here, otherwise we might
2132+
// spin forever if a SP is pending.
2133+
ThreadBlockInVM sp(JavaThread::cast(thread));
2134+
SpinPause();
21262135
}
21272136
}
21282137
}
@@ -2810,8 +2819,8 @@ void ShenandoahHeap::flush_liveness_cache(uint worker_id) {
28102819
}
28112820
}
28122821

2813-
void ShenandoahHeap::purge_old_satb_buffers(bool abandon) {
2814-
((ShenandoahOldGeneration*)_old_generation)->purge_satb_buffers(abandon);
2822+
void ShenandoahHeap::transfer_old_pointers_from_satb() {
2823+
((ShenandoahOldGeneration*) _old_generation)->transfer_pointers_from_satb();
28152824
}
28162825

28172826
template<>

‎src/hotspot/share/gc/shenandoah/shenandoahHeap.hpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -810,7 +810,8 @@ class ShenandoahHeap : public CollectedHeap {
810810
void clear_cards(HeapWord* start, HeapWord* end);
811811
void mark_card_as_dirty(void* location);
812812
void retire_plab(PLAB* plab);
813-
void cancel_mixed_collections();
813+
void cancel_old_gc();
814+
bool is_old_gc_active();
814815
void coalesce_and_fill_old_regions();
815816

816817
// ---------- Helper functions
@@ -838,7 +839,7 @@ class ShenandoahHeap : public CollectedHeap {
838839

839840
static inline void increase_object_age(oop obj, uint additional_age);
840841

841-
void purge_old_satb_buffers(bool abandon);
842+
void transfer_old_pointers_from_satb();
842843
private:
843844
void trash_cset_regions();
844845

‎src/hotspot/share/gc/shenandoah/shenandoahHeapRegion.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -825,7 +825,7 @@ void ShenandoahHeapRegion::set_affiliation(ShenandoahRegionAffiliation new_affil
825825
{
826826
ShenandoahMarkingContext* const ctx = heap->complete_marking_context();
827827
log_debug(gc)("Setting affiliation of Region " SIZE_FORMAT " from %s to %s, top: " PTR_FORMAT ", TAMS: " PTR_FORMAT
828-
", watermark: " PTR_FORMAT ", top_bitmap: " PTR_FORMAT "\n",
828+
", watermark: " PTR_FORMAT ", top_bitmap: " PTR_FORMAT,
829829
index(), affiliation_name(_affiliation), affiliation_name(new_affiliation),
830830
p2i(top()), p2i(ctx->top_at_mark_start(this)), p2i(_update_watermark), p2i(ctx->top_bitmap(this)));
831831
}

‎src/hotspot/share/gc/shenandoah/shenandoahMark.inline.hpp

+4-2
Original file line numberDiff line numberDiff line change
@@ -300,8 +300,10 @@ inline void ShenandoahMark::mark_through_ref(T *p, ShenandoahObjToScanQueue* q,
300300
// Old mark, found a young pointer.
301301
// TODO: Rethink this: may be redundant with dirtying of cards identified during young-gen remembered set scanning
302302
// and by mutator write barriers. Assert
303-
assert(heap->is_in_young(obj), "Expected young object.");
304-
heap->mark_card_as_dirty(p);
303+
if (heap->is_in(p)) {
304+
assert(heap->is_in_young(obj), "Expected young object.");
305+
heap->mark_card_as_dirty(p);
306+
}
305307
}
306308
}
307309
}

‎src/hotspot/share/gc/shenandoah/shenandoahMarkingContext.inline.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -89,7 +89,7 @@ inline void ShenandoahMarkingContext::capture_top_at_mark_start(ShenandoahHeapRe
8989
"Region " SIZE_FORMAT ", bitmap should be clear while adjusting TAMS: " PTR_FORMAT " -> " PTR_FORMAT,
9090
idx, p2i(old_tams), p2i(new_tams));
9191

92-
log_debug(gc)("Capturing TAMS for %s Region " SIZE_FORMAT ", was: %llx, now: %llx\n",
92+
log_debug(gc)("Capturing TAMS for %s Region " SIZE_FORMAT ", was: %llx, now: %llx",
9393
affiliation_name(r->affiliation()), idx, (unsigned long long) old_tams, (unsigned long long) new_tams);
9494

9595
if ((old_tams == r->bottom()) && (new_tams > old_tams)) {

0 commit comments

Comments
 (0)
Please sign in to comment.