@@ -49,7 +49,8 @@ ShenandoahDegenGC::ShenandoahDegenGC(ShenandoahDegenPoint degen_point, Shenandoa
49
49
ShenandoahGC(),
50
50
_degen_point(degen_point),
51
51
_generation(generation),
52
- _abbreviated(false ) {
52
+ _abbreviated(false ),
53
+ _consecutive_degen_with_bad_progress(0 ) {
53
54
}
54
55
55
56
bool ShenandoahDegenGC::collect (GCCause::Cause cause) {
@@ -305,9 +306,24 @@ void ShenandoahDegenGC::op_degenerated() {
305
306
306
307
metrics.snap_after ();
307
308
308
- // Check for futility and fail. There is no reason to do several back-to-back Degenerated cycles,
309
- // because that probably means the heap is overloaded and/or fragmented.
309
+ // The most common scenario for lack of good progress following a degenerated GC is an accumulation of floating
310
+ // garbage during the most recently aborted concurrent GC effort. With generational GC, it is far more effective to
311
+ // reclaim this floating garbage with another degenerated cycle (which focuses on young generation and might require
312
+ // a pause of 200 ms) rather than a full GC cycle (which may require over 2 seconds with a 10 GB old generation).
313
+ //
314
+ // In generational mode, we'll only upgrade to full GC if we've done two degen cycles in a row and both indicated
315
+ // bad progress. In non-generational mode, we'll preserve the original behavior, which is to upgrade to full
316
+ // immediately following a degenerated cycle with bad progress. This preserves original behavior of non-generational
317
+ // Shenandoah so as to avoid introducing "surprising new behavior." It also makes less sense with non-generational
318
+ // Shenandoah to replace a full GC with a degenerated GC, because both have similar pause times in non-generational
319
+ // mode.
310
320
if (!metrics.is_good_progress (_generation)) {
321
+ _consecutive_degen_with_bad_progress++;
322
+ } else {
323
+ _consecutive_degen_with_bad_progress = 0 ;
324
+ }
325
+ if (!heap->mode ()->is_generational () ||
326
+ ((heap->shenandoah_policy ()->consecutive_degenerated_gc_count () > 1 ) && (_consecutive_degen_with_bad_progress >= 2 ))) {
311
327
heap->cancel_gc (GCCause::_shenandoah_upgrade_to_full_gc);
312
328
op_degenerated_futile ();
313
329
} else {
0 commit comments