6b911624f2
Reviewed-by: lkorinth, kbarrett
174 lines
6.9 KiB
C++
174 lines
6.9 KiB
C++
/*
|
|
* Copyright (c) 2001, 2020, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*
|
|
*/
|
|
|
|
#include "precompiled.hpp"
|
|
#include "gc/g1/g1CollectedHeap.inline.hpp"
|
|
#include "gc/g1/g1ConcurrentMarkThread.inline.hpp"
|
|
#include "gc/g1/g1Policy.hpp"
|
|
#include "gc/g1/g1VMOperations.hpp"
|
|
#include "gc/shared/concurrentGCBreakpoints.hpp"
|
|
#include "gc/shared/gcCause.hpp"
|
|
#include "gc/shared/gcId.hpp"
|
|
#include "gc/shared/gcTimer.hpp"
|
|
#include "gc/shared/gcTraceTime.inline.hpp"
|
|
#include "gc/shared/isGCActiveMark.hpp"
|
|
#include "memory/universe.hpp"
|
|
#include "runtime/interfaceSupport.inline.hpp"
|
|
|
|
void VM_G1CollectFull::doit() {
|
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
|
GCCauseSetter x(g1h, _gc_cause);
|
|
_gc_succeeded = g1h->do_full_collection(true /* explicit_gc */, false /* clear_all_soft_refs */);
|
|
}
|
|
|
|
VM_G1TryInitiateConcMark::VM_G1TryInitiateConcMark(uint gc_count_before,
|
|
GCCause::Cause gc_cause,
|
|
double target_pause_time_ms) :
|
|
VM_GC_Operation(gc_count_before, gc_cause),
|
|
_target_pause_time_ms(target_pause_time_ms),
|
|
_transient_failure(false),
|
|
_cycle_already_in_progress(false),
|
|
_whitebox_attached(false),
|
|
_terminating(false),
|
|
_gc_succeeded(false)
|
|
{}
|
|
|
|
bool VM_G1TryInitiateConcMark::doit_prologue() {
|
|
bool result = VM_GC_Operation::doit_prologue();
|
|
// The prologue can fail for a couple of reasons. The first is that another GC
|
|
// got scheduled and prevented the scheduling of the concurrent start GC. The
|
|
// second is that the GC locker may be active and the heap can't be expanded.
|
|
// In both cases we want to retry the GC so that the concurrent start pause is
|
|
// actually scheduled. In the second case, however, we should stall until
|
|
// until the GC locker is no longer active and then retry the concurrent start GC.
|
|
if (!result) _transient_failure = true;
|
|
return result;
|
|
}
|
|
|
|
void VM_G1TryInitiateConcMark::doit() {
|
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
|
|
|
GCCauseSetter x(g1h, _gc_cause);
|
|
|
|
// Record for handling by caller.
|
|
_terminating = g1h->_cm_thread->should_terminate();
|
|
|
|
if (_terminating && GCCause::is_user_requested_gc(_gc_cause)) {
|
|
// When terminating, the request to initiate a concurrent cycle will be
|
|
// ignored by do_collection_pause_at_safepoint; instead it will just do
|
|
// a young-only or mixed GC (depending on phase). For a user request
|
|
// there's no point in even doing that much, so done. For some non-user
|
|
// requests the alternative GC might still be needed.
|
|
} else if (!g1h->policy()->force_concurrent_start_if_outside_cycle(_gc_cause)) {
|
|
// Failure to force the next GC pause to be a concurrent start indicates
|
|
// there is already a concurrent marking cycle in progress. Set flag
|
|
// to notify the caller and return immediately.
|
|
_cycle_already_in_progress = true;
|
|
} else if ((_gc_cause != GCCause::_wb_breakpoint) &&
|
|
ConcurrentGCBreakpoints::is_controlled()) {
|
|
// WhiteBox wants to be in control of concurrent cycles, so don't try to
|
|
// start one. This check is after the force_concurrent_start_xxx so that a
|
|
// request will be remembered for a later partial collection, even though
|
|
// we've rejected this request.
|
|
_whitebox_attached = true;
|
|
} else if (g1h->do_collection_pause_at_safepoint(_target_pause_time_ms)) {
|
|
_gc_succeeded = true;
|
|
} else {
|
|
// Failure to perform the collection at all occurs because GCLocker is
|
|
// active, and we have the bad luck to be the collection request that
|
|
// makes a later _gc_locker collection needed. (Else we would have hit
|
|
// the GCLocker check in the prologue.)
|
|
_transient_failure = true;
|
|
}
|
|
}
|
|
|
|
VM_G1CollectForAllocation::VM_G1CollectForAllocation(size_t word_size,
|
|
uint gc_count_before,
|
|
GCCause::Cause gc_cause,
|
|
double target_pause_time_ms) :
|
|
VM_CollectForAllocation(word_size, gc_count_before, gc_cause),
|
|
_gc_succeeded(false),
|
|
_target_pause_time_ms(target_pause_time_ms) {
|
|
|
|
guarantee(target_pause_time_ms > 0.0,
|
|
"target_pause_time_ms = %1.6lf should be positive",
|
|
target_pause_time_ms);
|
|
_gc_cause = gc_cause;
|
|
}
|
|
|
|
void VM_G1CollectForAllocation::doit() {
|
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
|
|
|
if (_word_size > 0) {
|
|
// An allocation has been requested. So, try to do that first.
|
|
_result = g1h->attempt_allocation_at_safepoint(_word_size,
|
|
false /* expect_null_cur_alloc_region */);
|
|
if (_result != NULL) {
|
|
// If we can successfully allocate before we actually do the
|
|
// pause then we will consider this pause successful.
|
|
_gc_succeeded = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
GCCauseSetter x(g1h, _gc_cause);
|
|
// Try a partial collection of some kind.
|
|
_gc_succeeded = g1h->do_collection_pause_at_safepoint(_target_pause_time_ms);
|
|
|
|
if (_gc_succeeded && (_word_size > 0)) {
|
|
// An allocation had been requested. Do it, eventually trying a stronger
|
|
// kind of GC.
|
|
_result = g1h->satisfy_failed_allocation(_word_size, &_gc_succeeded);
|
|
}
|
|
}
|
|
|
|
void VM_G1Concurrent::doit() {
|
|
GCIdMark gc_id_mark(_gc_id);
|
|
GCTraceCPUTime tcpu;
|
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
|
|
|
// GCTraceTime(...) only supports sub-phases, so a more verbose version
|
|
// is needed when we report the top-level pause phase.
|
|
GCTraceTimeLogger(Info, gc) logger(_message, GCCause::_no_gc, true);
|
|
GCTraceTimePauseTimer timer(_message, g1h->concurrent_mark()->gc_timer_cm());
|
|
GCTraceTimeDriver t(&logger, &timer);
|
|
|
|
TraceCollectorStats tcs(g1h->g1mm()->conc_collection_counters());
|
|
SvcGCMarker sgcm(SvcGCMarker::CONCURRENT);
|
|
IsGCActiveMark x;
|
|
_cl->do_void();
|
|
}
|
|
|
|
bool VM_G1Concurrent::doit_prologue() {
|
|
Heap_lock->lock();
|
|
return true;
|
|
}
|
|
|
|
void VM_G1Concurrent::doit_epilogue() {
|
|
if (Universe::has_reference_pending_list()) {
|
|
Heap_lock->notify_all();
|
|
}
|
|
Heap_lock->unlock();
|
|
}
|