Skip to content

Commit 6a1eebb

Browse files
author
William Kemper
committedOct 6, 2021
Improve handling of promotion and old generation evacuation failures
Reviewed-by: kdnilsen
1 parent 4d8ce22 commit 6a1eebb

6 files changed

+65
-22
lines changed
 

‎src/hotspot/share/gc/shenandoah/heuristics/shenandoahOldHeuristics.cpp

+20-13
Original file line numberDiff line numberDiff line change
@@ -294,18 +294,6 @@ void ShenandoahOldHeuristics::get_coalesce_and_fill_candidates(ShenandoahHeapReg
294294
}
295295
}
296296

297-
bool ShenandoahOldHeuristics::should_defer_gc() {
298-
if (unprocessed_old_collection_candidates() > 0) {
299-
// Cannot start a new old-gen GC until previous one has finished.
300-
//
301-
// Future refinement: under certain circumstances, we might be more sophisticated about this choice.
302-
// For example, we could choose to abandon the previous old collection before it has completed evacuations,
303-
// but this would require that we coalesce and fill all garbage within unevacuated collection-set regions.
304-
return true;
305-
}
306-
return false;
307-
}
308-
309297
void ShenandoahOldHeuristics::abandon_collection_candidates() {
310298
_old_collection_candidates = 0;
311299
_next_old_collection_candidate = 0;
@@ -315,7 +303,12 @@ void ShenandoahOldHeuristics::abandon_collection_candidates() {
315303
_first_coalesce_and_fill_candidate = 0;
316304
}
317305

306+
void ShenandoahOldHeuristics::handle_promotion_failure() {
307+
_promotion_failed = true;
308+
}
309+
318310
void ShenandoahOldHeuristics::record_cycle_start() {
311+
_promotion_failed = false;
319312
_trigger_heuristic->record_cycle_start();
320313
}
321314

@@ -324,9 +317,23 @@ void ShenandoahOldHeuristics::record_cycle_end() {
324317
}
325318

326319
bool ShenandoahOldHeuristics::should_start_gc() {
327-
if (should_defer_gc()) {
320+
// Cannot start a new old-gen GC until previous one has finished.
321+
//
322+
// Future refinement: under certain circumstances, we might be more sophisticated about this choice.
323+
// For example, we could choose to abandon the previous old collection before it has completed evacuations,
324+
// but this would require that we coalesce and fill all garbage within unevacuated collection-set regions.
325+
if (unprocessed_old_collection_candidates() > 0) {
328326
return false;
329327
}
328+
329+
// If there's been a promotion failure (and we don't have regions already scheduled for evacuation),
330+
// start a new old generation collection.
331+
if (_promotion_failed) {
332+
log_info(gc)("Trigger: Promotion Failure");
333+
return true;
334+
}
335+
336+
// Otherwise, defer to configured heuristic for gc trigger.
330337
return _trigger_heuristic->should_start_gc();
331338
}
332339

‎src/hotspot/share/gc/shenandoah/heuristics/shenandoahOldHeuristics.hpp

+8-3
Original file line numberDiff line numberDiff line change
@@ -66,10 +66,12 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {
6666
// This can be the 'static' or 'adaptive' heuristic.
6767
ShenandoahHeuristics* _trigger_heuristic;
6868

69+
// Flag is set when promotion failure is detected (by gc thread), cleared when old generation collection begins (by control thread)
70+
volatile bool _promotion_failed;
71+
6972
// Prepare for evacuation of old-gen regions by capturing the mark results of a recently completed concurrent mark pass.
7073
void prepare_for_old_collections();
7174

72-
7375
protected:
7476
virtual void choose_collection_set_from_regiondata(ShenandoahCollectionSet* set, RegionData* data, size_t data_size,
7577
size_t free) override;
@@ -108,12 +110,15 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {
108110
// end of the array.
109111
void get_coalesce_and_fill_candidates(ShenandoahHeapRegion** buffer);
110112

111-
bool should_defer_gc();
112-
113113
// If a GLOBAL gc occurs, it will collect the entire heap which invalidates any collection candidates being
114114
// held by this heuristic for supplying mixed collections.
115115
void abandon_collection_candidates();
116116

117+
// Notify the heuristic of promotion failures. The promotion attempt will be skipped and the object will
118+
// be evacuated into the young generation. The collection should complete normally, but we want to schedule
119+
// an old collection as soon as possible.
120+
void handle_promotion_failure();
121+
117122
virtual void record_cycle_start() override;
118123

119124
virtual void record_cycle_end() override;

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

+5-2
Original file line numberDiff line numberDiff line change
@@ -142,8 +142,10 @@ void ShenandoahControlThread::run_service() {
142142

143143
ShenandoahHeuristics* heuristics = _degen_generation->heuristics();
144144
generation = _degen_generation->generation_mode();
145+
bool old_gen_evacuation_failed = heap->clear_old_evacuation_failure();
145146

146-
if (ShenandoahDegeneratedGC && heuristics->should_degenerate_cycle()) {
147+
// Do not bother with degenerated cycle if old generation evacuation failed.
148+
if (ShenandoahDegeneratedGC && heuristics->should_degenerate_cycle() && !old_gen_evacuation_failed) {
147149
heuristics->record_allocation_failure_gc();
148150
policy->record_alloc_failure_to_degenerated(degen_point);
149151
set_gc_mode(stw_degenerated);
@@ -370,7 +372,8 @@ void ShenandoahControlThread::run_service() {
370372
last_shrink_time = current;
371373
}
372374

373-
{
375+
// Don't wait around if there was an allocation failure - start the next cycle immediately.
376+
if (!is_alloc_failure_gc()) {
374377
// The timed wait is necessary because this thread has a responsibility to send
375378
// 'alloc_words' to the pacer when it does not perform a GC.
376379
MonitorLocker lock(&_control_lock, Mutex::_no_safepoint_check_flag);

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

+10
Original file line numberDiff line numberDiff line change
@@ -824,6 +824,16 @@ void ShenandoahHeap::handle_old_evacuation(HeapWord* obj, size_t words, bool pro
824824
}
825825
}
826826

827+
void ShenandoahHeap::handle_old_evacuation_failure() {
828+
if (_old_gen_oom_evac.try_set()) {
829+
log_info(gc)("Old gen evac failure.");
830+
}
831+
}
832+
833+
void ShenandoahHeap::handle_promotion_failure() {
834+
old_heuristics()->handle_promotion_failure();
835+
}
836+
827837
HeapWord* ShenandoahHeap::allocate_from_gclab_slow(Thread* thread, size_t size) {
828838
// New object should fit the GCLAB size
829839
size_t min_size = MAX2(size, PLAB::min_size());

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

+5
Original file line numberDiff line numberDiff line change
@@ -668,9 +668,12 @@ class ShenandoahHeap : public CollectedHeap {
668668
private:
669669
ShenandoahCollectionSet* _collection_set;
670670
ShenandoahEvacOOMHandler _oom_evac_handler;
671+
ShenandoahSharedFlag _old_gen_oom_evac;
671672

672673
inline oop try_evacuate_object(oop src, Thread* thread, ShenandoahHeapRegion* from_region, ShenandoahRegionAffiliation target_gen);
673674
void handle_old_evacuation(HeapWord* obj, size_t words, bool promotion);
675+
void handle_old_evacuation_failure();
676+
void handle_promotion_failure();
674677

675678
public:
676679
static address in_cset_fast_test_addr();
@@ -691,6 +694,8 @@ class ShenandoahHeap : public CollectedHeap {
691694
inline void enter_evacuation(Thread* t);
692695
inline void leave_evacuation(Thread* t);
693696

697+
inline bool clear_old_evacuation_failure();
698+
694699
// ---------- Generational support
695700
//
696701
private:

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

+17-4
Original file line numberDiff line numberDiff line change
@@ -315,7 +315,8 @@ inline HeapWord* ShenandoahHeap::allocate_from_plab(Thread* thread, size_t size)
315315
}
316316

317317
inline oop ShenandoahHeap::evacuate_object(oop p, Thread* thread) {
318-
if (ShenandoahThreadLocalData::is_oom_during_evac(Thread::current())) {
318+
assert(thread == Thread::current(), "Expected thread parameter to be current thread.");
319+
if (ShenandoahThreadLocalData::is_oom_during_evac(thread)) {
319320
// This thread went through the OOM during evac protocol and it is safe to return
320321
// the forward pointer. It must not attempt to evacuate any more.
321322
return ShenandoahBarrierSet::resolve_forwarded(p);
@@ -388,9 +389,17 @@ inline oop ShenandoahHeap::try_evacuate_object(oop p, Thread* thread, Shenandoah
388389
#endif
389390

390391
if (copy == NULL) {
391-
if (target_gen == OLD_GENERATION && from_region->affiliation() == YOUNG_GENERATION) {
392-
// TODO: Inform old generation heuristic of promotion failure
393-
return NULL;
392+
if (target_gen == OLD_GENERATION) {
393+
assert(mode()->is_generational(), "Should only be here in generational mode.");
394+
if (from_region->is_young()) {
395+
// Signal that promotion failed. Will evacuate this old object somewhere in young gen.
396+
handle_promotion_failure();
397+
return NULL;
398+
} else {
399+
// Remember that evacuation to old gen failed. We'll want to trigger a full gc to recover from this
400+
// after the evacuation threads have finished.
401+
handle_old_evacuation_failure();
402+
}
394403
}
395404

396405
control_thread()->handle_alloc_failure_evac(size);
@@ -466,6 +475,10 @@ void ShenandoahHeap::increase_object_age(oop obj, uint additional_age) {
466475
}
467476
}
468477

478+
inline bool ShenandoahHeap::clear_old_evacuation_failure() {
479+
return _old_gen_oom_evac.try_unset();
480+
}
481+
469482
inline bool ShenandoahHeap::is_old(oop obj) const {
470483
return is_gc_generation_young() && is_in_old(obj);
471484
}

0 commit comments

Comments
 (0)