Skip to content

Commit 3ae80bf

Browse files
committedFeb 27, 2025
8349766: GenShen: Bad progress after degen does not always need full gc
Reviewed-by: wkemper
1 parent 2fd7156 commit 3ae80bf

File tree

2 files changed

+20
-3
lines changed

2 files changed

+20
-3
lines changed
 

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

+19-3
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,8 @@ ShenandoahDegenGC::ShenandoahDegenGC(ShenandoahDegenPoint degen_point, Shenandoa
4949
ShenandoahGC(),
5050
_degen_point(degen_point),
5151
_generation(generation),
52-
_abbreviated(false) {
52+
_abbreviated(false),
53+
_consecutive_degen_with_bad_progress(0) {
5354
}
5455

5556
bool ShenandoahDegenGC::collect(GCCause::Cause cause) {
@@ -305,9 +306,24 @@ void ShenandoahDegenGC::op_degenerated() {
305306

306307
metrics.snap_after();
307308

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.
310320
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))) {
311327
heap->cancel_gc(GCCause::_shenandoah_upgrade_to_full_gc);
312328
op_degenerated_futile();
313329
} else {

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

+1
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ class ShenandoahDegenGC : public ShenandoahGC {
3636
const ShenandoahDegenPoint _degen_point;
3737
ShenandoahGeneration* _generation;
3838
bool _abbreviated;
39+
size_t _consecutive_degen_with_bad_progress;
3940

4041
public:
4142
ShenandoahDegenGC(ShenandoahDegenPoint degen_point, ShenandoahGeneration* generation);

0 commit comments

Comments
 (0)
Please sign in to comment.