@@ -90,13 +90,21 @@ void ShenandoahDegenGC::op_degenerated() {
90
90
// some phase, we have to upgrade the Degenerate GC to Full GC.
91
91
heap->clear_cancelled_gc (true /* clear oom handler */ );
92
92
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
+ }
98
106
}
99
- assert (heap-> old_generation ()-> task_queues ()-> is_empty (), " Old gen task queues should be empty. " );
107
+ # endif
100
108
101
109
ShenandoahMetricsSnapshot metrics;
102
110
metrics.snap_before ();
@@ -113,6 +121,14 @@ void ShenandoahDegenGC::op_degenerated() {
113
121
// we can do the most aggressive degen cycle, which includes processing references and
114
122
// class unloading, unless those features are explicitly disabled.
115
123
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
+ }
116
132
117
133
// Note that we can only do this for "outside-cycle" degens, otherwise we would risk
118
134
// changing the cycle parameters mid-cycle during concurrent -> degenerated handover.
@@ -125,8 +141,17 @@ void ShenandoahDegenGC::op_degenerated() {
125
141
126
142
case _degenerated_roots:
127
143
// 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
+ }
130
155
}
131
156
132
157
if (_degen_point == ShenandoahDegenPoint::_degenerated_roots) {
@@ -293,7 +318,7 @@ void ShenandoahDegenGC::op_reset() {
293
318
}
294
319
295
320
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" );
297
322
ShenandoahGCPhase phase (ShenandoahPhaseTimings::degen_gc_stw_mark);
298
323
ShenandoahSTWMark mark (_generation, false /* full gc*/ );
299
324
mark.mark ();
0 commit comments