8232588: G1 concurrent System.gc can return early or late
8233279: G1: GCLocker GC with +GCLockerInvokesConcurrent spins while cycle in progress Refactor G1CH::try_collect and fix bugs with concurrent collections. Reviewed-by: tschatzl, sjohanss
This commit is contained in:
parent
9ad86a68cc
commit
19bed24b1f
@ -2009,7 +2009,7 @@ bool G1CollectedHeap::should_do_concurrent_full_gc(GCCause::Cause cause) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool G1CollectedHeap::should_upgrade_to_full_gc(GCCause::Cause cause) {
|
bool G1CollectedHeap::should_upgrade_to_full_gc(GCCause::Cause cause) {
|
||||||
if(policy()->force_upgrade_to_full()) {
|
if (policy()->force_upgrade_to_full()) {
|
||||||
return true;
|
return true;
|
||||||
} else if (should_do_concurrent_full_gc(_gc_cause)) {
|
} else if (should_do_concurrent_full_gc(_gc_cause)) {
|
||||||
return false;
|
return false;
|
||||||
@ -2056,7 +2056,7 @@ void G1CollectedHeap::increment_old_marking_cycles_started() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void G1CollectedHeap::increment_old_marking_cycles_completed(bool concurrent) {
|
void G1CollectedHeap::increment_old_marking_cycles_completed(bool concurrent) {
|
||||||
MonitorLocker x(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
|
MonitorLocker ml(G1OldGCCount_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
|
||||||
// We assume that if concurrent == true, then the caller is a
|
// We assume that if concurrent == true, then the caller is a
|
||||||
// concurrent thread that was joined the Suspendible Thread
|
// concurrent thread that was joined the Suspendible Thread
|
||||||
@ -2096,91 +2096,211 @@ void G1CollectedHeap::increment_old_marking_cycles_completed(bool concurrent) {
|
|||||||
_cm_thread->set_idle();
|
_cm_thread->set_idle();
|
||||||
}
|
}
|
||||||
|
|
||||||
// This notify_all() will ensure that a thread that called
|
// Notify threads waiting in System.gc() (with ExplicitGCInvokesConcurrent)
|
||||||
// System.gc() with (with ExplicitGCInvokesConcurrent set or not)
|
// for a full GC to finish that their wait is over.
|
||||||
// and it's waiting for a full GC to finish will be woken up. It is
|
ml.notify_all();
|
||||||
// waiting in VM_G1CollectForAllocation::doit_epilogue().
|
|
||||||
FullGCCount_lock->notify_all();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void G1CollectedHeap::collect(GCCause::Cause cause) {
|
void G1CollectedHeap::collect(GCCause::Cause cause) {
|
||||||
try_collect(cause, true);
|
try_collect(cause);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool G1CollectedHeap::try_collect(GCCause::Cause cause, bool retry_on_gc_failure) {
|
// Return true if (x < y) with allowance for wraparound.
|
||||||
|
static bool gc_counter_less_than(uint x, uint y) {
|
||||||
|
return (x - y) > (UINT_MAX/2);
|
||||||
|
}
|
||||||
|
|
||||||
|
// LOG_COLLECT_CONCURRENTLY(cause, msg, args...)
|
||||||
|
// Macro so msg printing is format-checked.
|
||||||
|
#define LOG_COLLECT_CONCURRENTLY(cause, ...) \
|
||||||
|
do { \
|
||||||
|
LogTarget(Trace, gc) LOG_COLLECT_CONCURRENTLY_lt; \
|
||||||
|
if (LOG_COLLECT_CONCURRENTLY_lt.is_enabled()) { \
|
||||||
|
ResourceMark rm; /* For thread name. */ \
|
||||||
|
LogStream LOG_COLLECT_CONCURRENTLY_s(&LOG_COLLECT_CONCURRENTLY_lt); \
|
||||||
|
LOG_COLLECT_CONCURRENTLY_s.print("%s: Try Collect Concurrently (%s): ", \
|
||||||
|
Thread::current()->name(), \
|
||||||
|
GCCause::to_string(cause)); \
|
||||||
|
LOG_COLLECT_CONCURRENTLY_s.print(__VA_ARGS__); \
|
||||||
|
} \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define LOG_COLLECT_CONCURRENTLY_COMPLETE(cause, result) \
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "complete %s", BOOL_TO_STR(result))
|
||||||
|
|
||||||
|
bool G1CollectedHeap::try_collect_concurrently(GCCause::Cause cause,
|
||||||
|
uint gc_counter,
|
||||||
|
uint old_marking_started_before) {
|
||||||
assert_heap_not_locked();
|
assert_heap_not_locked();
|
||||||
|
assert(should_do_concurrent_full_gc(cause),
|
||||||
|
"Non-concurrent cause %s", GCCause::to_string(cause));
|
||||||
|
|
||||||
bool gc_succeeded;
|
for (uint i = 1; true; ++i) {
|
||||||
bool should_retry_gc;
|
// Try to schedule an initial-mark evacuation pause that will
|
||||||
|
// start a concurrent cycle.
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "attempt %u", i);
|
||||||
|
VM_G1TryInitiateConcMark op(gc_counter,
|
||||||
|
cause,
|
||||||
|
policy()->max_pause_time_ms());
|
||||||
|
VMThread::execute(&op);
|
||||||
|
|
||||||
do {
|
// Request is trivially finished.
|
||||||
should_retry_gc = false;
|
if (cause == GCCause::_g1_periodic_collection) {
|
||||||
|
LOG_COLLECT_CONCURRENTLY_COMPLETE(cause, op.gc_succeeded());
|
||||||
uint gc_count_before;
|
return op.gc_succeeded();
|
||||||
uint old_marking_count_before;
|
}
|
||||||
uint full_gc_count_before;
|
|
||||||
|
|
||||||
|
// Lock to get consistent set of values.
|
||||||
|
uint old_marking_started_after;
|
||||||
|
uint old_marking_completed_after;
|
||||||
{
|
{
|
||||||
MutexLocker ml(Heap_lock);
|
MutexLocker ml(Heap_lock);
|
||||||
|
// Update gc_counter for retrying VMOp if needed. Captured here to be
|
||||||
// Read the GC count while holding the Heap_lock
|
// consistent with the values we use below for termination tests. If
|
||||||
gc_count_before = total_collections();
|
// a retry is needed after a possible wait, and another collection
|
||||||
full_gc_count_before = total_full_collections();
|
// occurs in the meantime, it will cause our retry to be skipped and
|
||||||
old_marking_count_before = _old_marking_cycles_started;
|
// we'll recheck for termination with updated conditions from that
|
||||||
|
// more recent collection. That's what we want, rather than having
|
||||||
|
// our retry possibly perform an unnecessary collection.
|
||||||
|
gc_counter = total_collections();
|
||||||
|
old_marking_started_after = _old_marking_cycles_started;
|
||||||
|
old_marking_completed_after = _old_marking_cycles_completed;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (should_do_concurrent_full_gc(cause)) {
|
if (!GCCause::is_user_requested_gc(cause)) {
|
||||||
// Schedule an initial-mark evacuation pause that will start a
|
// For an "automatic" (not user-requested) collection, we just need to
|
||||||
// concurrent cycle. We're setting word_size to 0 which means that
|
// ensure that progress is made.
|
||||||
// we are not requesting a post-GC allocation.
|
//
|
||||||
VM_G1CollectForAllocation op(0, /* word_size */
|
// Request is finished if any of
|
||||||
gc_count_before,
|
// (1) the VMOp successfully performed a GC,
|
||||||
cause,
|
// (2) a concurrent cycle was already in progress,
|
||||||
true, /* should_initiate_conc_mark */
|
// (3) a new cycle was started (by this thread or some other), or
|
||||||
policy()->max_pause_time_ms());
|
// (4) a Full GC was performed.
|
||||||
VMThread::execute(&op);
|
// Cases (3) and (4) are detected together by a change to
|
||||||
gc_succeeded = op.gc_succeeded();
|
// _old_marking_cycles_started.
|
||||||
if (!gc_succeeded && retry_on_gc_failure) {
|
//
|
||||||
if (old_marking_count_before == _old_marking_cycles_started) {
|
// Note that (1) does not imply (3). If we're still in the mixed
|
||||||
should_retry_gc = op.should_retry_gc();
|
// phase of an earlier concurrent collection, the request to make the
|
||||||
} else {
|
// collection an initial-mark won't be honored. If we don't check for
|
||||||
// A Full GC happened while we were trying to schedule the
|
// both conditions we'll spin doing back-to-back collections.
|
||||||
// concurrent cycle. No point in starting a new cycle given
|
if (op.gc_succeeded() ||
|
||||||
// that the whole heap was collected anyway.
|
op.cycle_already_in_progress() ||
|
||||||
}
|
(old_marking_started_before != old_marking_started_after)) {
|
||||||
|
LOG_COLLECT_CONCURRENTLY_COMPLETE(cause, true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} else { // User-requested GC.
|
||||||
|
// For a user-requested collection, we want to ensure that a complete
|
||||||
|
// full collection has been performed before returning, but without
|
||||||
|
// waiting for more than needed.
|
||||||
|
|
||||||
if (should_retry_gc && GCLocker::is_active_and_needs_gc()) {
|
// For user-requested GCs (unlike non-UR), a successful VMOp implies a
|
||||||
GCLocker::stall_until_clear();
|
// new cycle was started. That's good, because it's not clear what we
|
||||||
|
// should do otherwise. Trying again just does back to back GCs.
|
||||||
|
// Can't wait for someone else to start a cycle. And returning fails
|
||||||
|
// to meet the goal of ensuring a full collection was performed.
|
||||||
|
assert(!op.gc_succeeded() ||
|
||||||
|
(old_marking_started_before != old_marking_started_after),
|
||||||
|
"invariant: succeeded %s, started before %u, started after %u",
|
||||||
|
BOOL_TO_STR(op.gc_succeeded()),
|
||||||
|
old_marking_started_before, old_marking_started_after);
|
||||||
|
|
||||||
|
// Request is finished if a full collection (concurrent or stw)
|
||||||
|
// was started after this request and has completed, e.g.
|
||||||
|
// started_before < completed_after.
|
||||||
|
if (gc_counter_less_than(old_marking_started_before,
|
||||||
|
old_marking_completed_after)) {
|
||||||
|
LOG_COLLECT_CONCURRENTLY_COMPLETE(cause, true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (old_marking_started_after != old_marking_completed_after) {
|
||||||
|
// If there is an in-progress cycle (possibly started by us), then
|
||||||
|
// wait for that cycle to complete, e.g.
|
||||||
|
// while completed_now < started_after.
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "wait");
|
||||||
|
MonitorLocker ml(G1OldGCCount_lock);
|
||||||
|
while (gc_counter_less_than(_old_marking_cycles_completed,
|
||||||
|
old_marking_started_after)) {
|
||||||
|
ml.wait();
|
||||||
|
}
|
||||||
|
// Request is finished if the collection we just waited for was
|
||||||
|
// started after this request.
|
||||||
|
if (old_marking_started_before != old_marking_started_after) {
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "complete after wait");
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (GCLocker::should_discard(cause, gc_count_before)) {
|
|
||||||
// Return false to be consistent with VMOp failure due to
|
|
||||||
// another collection slipping in after our gc_count but before
|
|
||||||
// our request is processed. _gc_locker collections upgraded by
|
|
||||||
// GCLockerInvokesConcurrent are handled above and never discarded.
|
|
||||||
return false;
|
|
||||||
} else {
|
|
||||||
if (cause == GCCause::_gc_locker || cause == GCCause::_wb_young_gc
|
|
||||||
DEBUG_ONLY(|| cause == GCCause::_scavenge_alot)) {
|
|
||||||
|
|
||||||
// Schedule a standard evacuation pause. We're setting word_size
|
// If VMOp was successful then it started a new cycle that the above
|
||||||
// to 0 which means that we are not requesting a post-GC allocation.
|
// wait &etc should have recognized as finishing this request. This
|
||||||
VM_G1CollectForAllocation op(0, /* word_size */
|
// differs from a non-user-request, where gc_succeeded does not imply
|
||||||
gc_count_before,
|
// a new cycle was started.
|
||||||
cause,
|
assert(!op.gc_succeeded(), "invariant");
|
||||||
false, /* should_initiate_conc_mark */
|
|
||||||
policy()->max_pause_time_ms());
|
// If VMOp failed because a cycle was already in progress, it is now
|
||||||
VMThread::execute(&op);
|
// complete. But it didn't finish this user-requested GC, so try
|
||||||
gc_succeeded = op.gc_succeeded();
|
// again.
|
||||||
} else {
|
if (op.cycle_already_in_progress()) {
|
||||||
// Schedule a Full GC.
|
LOG_COLLECT_CONCURRENTLY(cause, "retry after in-progress");
|
||||||
VM_G1CollectFull op(gc_count_before, full_gc_count_before, cause);
|
continue;
|
||||||
VMThread::execute(&op);
|
|
||||||
gc_succeeded = op.gc_succeeded();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} while (should_retry_gc);
|
|
||||||
return gc_succeeded;
|
// Collection failed and should be retried.
|
||||||
|
assert(op.transient_failure(), "invariant");
|
||||||
|
|
||||||
|
// If GCLocker is active, wait until clear before retrying.
|
||||||
|
if (GCLocker::is_active_and_needs_gc()) {
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "gc-locker stall");
|
||||||
|
GCLocker::stall_until_clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_COLLECT_CONCURRENTLY(cause, "retry");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool G1CollectedHeap::try_collect(GCCause::Cause cause) {
|
||||||
|
assert_heap_not_locked();
|
||||||
|
|
||||||
|
// Lock to get consistent set of values.
|
||||||
|
uint gc_count_before;
|
||||||
|
uint full_gc_count_before;
|
||||||
|
uint old_marking_started_before;
|
||||||
|
{
|
||||||
|
MutexLocker ml(Heap_lock);
|
||||||
|
gc_count_before = total_collections();
|
||||||
|
full_gc_count_before = total_full_collections();
|
||||||
|
old_marking_started_before = _old_marking_cycles_started;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (should_do_concurrent_full_gc(cause)) {
|
||||||
|
return try_collect_concurrently(cause,
|
||||||
|
gc_count_before,
|
||||||
|
old_marking_started_before);
|
||||||
|
} else if (GCLocker::should_discard(cause, gc_count_before)) {
|
||||||
|
// Indicate failure to be consistent with VMOp failure due to
|
||||||
|
// another collection slipping in after our gc_count but before
|
||||||
|
// our request is processed. _gc_locker collections upgraded by
|
||||||
|
// GCLockerInvokesConcurrent are handled above and never discarded.
|
||||||
|
return false;
|
||||||
|
} else if (cause == GCCause::_gc_locker || cause == GCCause::_wb_young_gc
|
||||||
|
DEBUG_ONLY(|| cause == GCCause::_scavenge_alot)) {
|
||||||
|
|
||||||
|
// Schedule a standard evacuation pause. We're setting word_size
|
||||||
|
// to 0 which means that we are not requesting a post-GC allocation.
|
||||||
|
VM_G1CollectForAllocation op(0, /* word_size */
|
||||||
|
gc_count_before,
|
||||||
|
cause,
|
||||||
|
policy()->max_pause_time_ms());
|
||||||
|
VMThread::execute(&op);
|
||||||
|
return op.gc_succeeded();
|
||||||
|
} else {
|
||||||
|
// Schedule a Full GC.
|
||||||
|
VM_G1CollectFull op(gc_count_before, full_gc_count_before, cause);
|
||||||
|
VMThread::execute(&op);
|
||||||
|
return op.gc_succeeded();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool G1CollectedHeap::is_in(const void* p) const {
|
bool G1CollectedHeap::is_in(const void* p) const {
|
||||||
@ -2611,7 +2731,6 @@ HeapWord* G1CollectedHeap::do_collection_pause(size_t word_size,
|
|||||||
VM_G1CollectForAllocation op(word_size,
|
VM_G1CollectForAllocation op(word_size,
|
||||||
gc_count_before,
|
gc_count_before,
|
||||||
gc_cause,
|
gc_cause,
|
||||||
false, /* should_initiate_conc_mark */
|
|
||||||
policy()->max_pause_time_ms());
|
policy()->max_pause_time_ms());
|
||||||
VMThread::execute(&op);
|
VMThread::execute(&op);
|
||||||
|
|
||||||
|
@ -133,6 +133,7 @@ class G1CollectedHeap : public CollectedHeap {
|
|||||||
friend class VM_CollectForMetadataAllocation;
|
friend class VM_CollectForMetadataAllocation;
|
||||||
friend class VM_G1CollectForAllocation;
|
friend class VM_G1CollectForAllocation;
|
||||||
friend class VM_G1CollectFull;
|
friend class VM_G1CollectFull;
|
||||||
|
friend class VM_G1TryInitiateConcMark;
|
||||||
friend class VMStructs;
|
friend class VMStructs;
|
||||||
friend class MutatorAllocRegion;
|
friend class MutatorAllocRegion;
|
||||||
friend class G1FullCollector;
|
friend class G1FullCollector;
|
||||||
@ -259,16 +260,22 @@ private:
|
|||||||
|
|
||||||
G1HRPrinter _hr_printer;
|
G1HRPrinter _hr_printer;
|
||||||
|
|
||||||
// It decides whether an explicit GC should start a concurrent cycle
|
// Return true if an explicit GC should start a concurrent cycle instead
|
||||||
// instead of doing a STW GC. Currently, a concurrent cycle is
|
// of doing a STW full GC. A concurrent cycle should be started if:
|
||||||
// explicitly started if:
|
// (a) cause == _gc_locker and +GCLockerInvokesConcurrent,
|
||||||
// (a) cause == _gc_locker and +GCLockerInvokesConcurrent, or
|
// (b) cause == _g1_humongous_allocation,
|
||||||
// (b) cause == _g1_humongous_allocation
|
// (c) cause == _java_lang_system_gc and +ExplicitGCInvokesConcurrent,
|
||||||
// (c) cause == _java_lang_system_gc and +ExplicitGCInvokesConcurrent.
|
// (d) cause == _dcmd_gc_run and +ExplicitGCInvokesConcurrent,
|
||||||
// (d) cause == _dcmd_gc_run and +ExplicitGCInvokesConcurrent.
|
// (e) cause == _wb_conc_mark,
|
||||||
// (e) cause == _wb_conc_mark
|
// (f) cause == _g1_periodic_collection and +G1PeriodicGCInvokesConcurrent.
|
||||||
bool should_do_concurrent_full_gc(GCCause::Cause cause);
|
bool should_do_concurrent_full_gc(GCCause::Cause cause);
|
||||||
|
|
||||||
|
// Attempt to start a concurrent cycle with the indicated cause.
|
||||||
|
// precondition: should_do_concurrent_full_gc(cause)
|
||||||
|
bool try_collect_concurrently(GCCause::Cause cause,
|
||||||
|
uint gc_counter,
|
||||||
|
uint old_marking_started_before);
|
||||||
|
|
||||||
// Return true if should upgrade to full gc after an incremental one.
|
// Return true if should upgrade to full gc after an incremental one.
|
||||||
bool should_upgrade_to_full_gc(GCCause::Cause cause);
|
bool should_upgrade_to_full_gc(GCCause::Cause cause);
|
||||||
|
|
||||||
@ -630,7 +637,7 @@ public:
|
|||||||
// Full GC). If concurrent is true, the caller is the outer caller
|
// Full GC). If concurrent is true, the caller is the outer caller
|
||||||
// in this nesting (i.e., the concurrent cycle). Further nesting is
|
// in this nesting (i.e., the concurrent cycle). Further nesting is
|
||||||
// not currently supported. The end of this call also notifies
|
// not currently supported. The end of this call also notifies
|
||||||
// the FullGCCount_lock in case a Java thread is waiting for a full
|
// the G1OldGCCount_lock in case a Java thread is waiting for a full
|
||||||
// GC to happen (e.g., it called System.gc() with
|
// GC to happen (e.g., it called System.gc() with
|
||||||
// +ExplicitGCInvokesConcurrent).
|
// +ExplicitGCInvokesConcurrent).
|
||||||
void increment_old_marking_cycles_completed(bool concurrent);
|
void increment_old_marking_cycles_completed(bool concurrent);
|
||||||
@ -1088,10 +1095,9 @@ public:
|
|||||||
// "CollectedHeap" supports.
|
// "CollectedHeap" supports.
|
||||||
virtual void collect(GCCause::Cause cause);
|
virtual void collect(GCCause::Cause cause);
|
||||||
|
|
||||||
// Perform a collection of the heap with the given cause; if the VM operation
|
// Perform a collection of the heap with the given cause.
|
||||||
// fails to execute for any reason, retry only if retry_on_gc_failure is set.
|
|
||||||
// Returns whether this collection actually executed.
|
// Returns whether this collection actually executed.
|
||||||
bool try_collect(GCCause::Cause cause, bool retry_on_gc_failure);
|
bool try_collect(GCCause::Cause cause);
|
||||||
|
|
||||||
// True iff an evacuation has failed in the most-recent collection.
|
// True iff an evacuation has failed in the most-recent collection.
|
||||||
bool evacuation_failed() { return _evacuation_failed; }
|
bool evacuation_failed() { return _evacuation_failed; }
|
||||||
|
@ -393,7 +393,7 @@ void G1ConcurrentMarkThread::run_service() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update the number of full collections that have been
|
// Update the number of full collections that have been
|
||||||
// completed. This will also notify the FullGCCount_lock in case a
|
// completed. This will also notify the G1OldGCCount_lock in case a
|
||||||
// Java thread is waiting for a full GC to happen (e.g., it
|
// Java thread is waiting for a full GC to happen (e.g., it
|
||||||
// called System.gc() with +ExplicitGCInvokesConcurrent).
|
// called System.gc() with +ExplicitGCInvokesConcurrent).
|
||||||
{
|
{
|
||||||
|
@ -40,17 +40,61 @@ void VM_G1CollectFull::doit() {
|
|||||||
_gc_succeeded = g1h->do_full_collection(true /* explicit_gc */, false /* clear_all_soft_refs */);
|
_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),
|
||||||
|
_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 initial mark 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 initial mark 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 initial mark GC.
|
||||||
|
if (!result) _transient_failure = true;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void VM_G1TryInitiateConcMark::doit() {
|
||||||
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
||||||
|
|
||||||
|
GCCauseSetter x(g1h, _gc_cause);
|
||||||
|
if (!g1h->policy()->force_initial_mark_if_outside_cycle(_gc_cause)) {
|
||||||
|
// Failure to force the next GC pause to be an initial mark 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 (!g1h->do_collection_pause_at_safepoint(_target_pause_time_ms)) {
|
||||||
|
// 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;
|
||||||
|
} else if (g1h->should_upgrade_to_full_gc(_gc_cause)) {
|
||||||
|
// GC ran, but we're still in trouble and need a full GC.
|
||||||
|
log_info(gc, ergo)("Attempting maximally compacting collection");
|
||||||
|
_gc_succeeded = g1h->do_full_collection(false, /* explicit gc */
|
||||||
|
true /* clear_all_soft_refs */);
|
||||||
|
guarantee(_gc_succeeded, "Elevated collections during the safepoint must always succeed");
|
||||||
|
} else {
|
||||||
|
_gc_succeeded = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
VM_G1CollectForAllocation::VM_G1CollectForAllocation(size_t word_size,
|
VM_G1CollectForAllocation::VM_G1CollectForAllocation(size_t word_size,
|
||||||
uint gc_count_before,
|
uint gc_count_before,
|
||||||
GCCause::Cause gc_cause,
|
GCCause::Cause gc_cause,
|
||||||
bool should_initiate_conc_mark,
|
|
||||||
double target_pause_time_ms) :
|
double target_pause_time_ms) :
|
||||||
VM_CollectForAllocation(word_size, gc_count_before, gc_cause),
|
VM_CollectForAllocation(word_size, gc_count_before, gc_cause),
|
||||||
_gc_succeeded(false),
|
_gc_succeeded(false),
|
||||||
_should_initiate_conc_mark(should_initiate_conc_mark),
|
_target_pause_time_ms(target_pause_time_ms) {
|
||||||
_should_retry_gc(false),
|
|
||||||
_target_pause_time_ms(target_pause_time_ms),
|
|
||||||
_old_marking_cycles_completed_before(0) {
|
|
||||||
|
|
||||||
guarantee(target_pause_time_ms > 0.0,
|
guarantee(target_pause_time_ms > 0.0,
|
||||||
"target_pause_time_ms = %1.6lf should be positive",
|
"target_pause_time_ms = %1.6lf should be positive",
|
||||||
@ -58,26 +102,8 @@ VM_G1CollectForAllocation::VM_G1CollectForAllocation(size_t word_size,
|
|||||||
_gc_cause = gc_cause;
|
_gc_cause = gc_cause;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool VM_G1CollectForAllocation::doit_prologue() {
|
|
||||||
bool res = VM_CollectForAllocation::doit_prologue();
|
|
||||||
if (!res) {
|
|
||||||
if (_should_initiate_conc_mark) {
|
|
||||||
// The prologue can fail for a couple of reasons. The first is that another GC
|
|
||||||
// got scheduled and prevented the scheduling of the initial mark 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 initial mark 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 initial mark GC.
|
|
||||||
_should_retry_gc = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
void VM_G1CollectForAllocation::doit() {
|
void VM_G1CollectForAllocation::doit() {
|
||||||
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
||||||
assert(!_should_initiate_conc_mark || g1h->should_do_concurrent_full_gc(_gc_cause),
|
|
||||||
"only a GC locker, a System.gc(), stats update, whitebox, or a hum allocation induced GC should start a cycle");
|
|
||||||
|
|
||||||
if (_word_size > 0) {
|
if (_word_size > 0) {
|
||||||
// An allocation has been requested. So, try to do that first.
|
// An allocation has been requested. So, try to do that first.
|
||||||
@ -92,44 +118,6 @@ void VM_G1CollectForAllocation::doit() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
GCCauseSetter x(g1h, _gc_cause);
|
GCCauseSetter x(g1h, _gc_cause);
|
||||||
if (_should_initiate_conc_mark) {
|
|
||||||
// It's safer to read old_marking_cycles_completed() here, given
|
|
||||||
// that noone else will be updating it concurrently. Since we'll
|
|
||||||
// only need it if we're initiating a marking cycle, no point in
|
|
||||||
// setting it earlier.
|
|
||||||
_old_marking_cycles_completed_before = g1h->old_marking_cycles_completed();
|
|
||||||
|
|
||||||
// At this point we are supposed to start a concurrent cycle. We
|
|
||||||
// will do so if one is not already in progress.
|
|
||||||
bool res = g1h->policy()->force_initial_mark_if_outside_cycle(_gc_cause);
|
|
||||||
|
|
||||||
// The above routine returns true if we were able to force the
|
|
||||||
// next GC pause to be an initial mark; it returns false if a
|
|
||||||
// marking cycle is already in progress.
|
|
||||||
//
|
|
||||||
// If a marking cycle is already in progress just return and skip the
|
|
||||||
// pause below - if the reason for requesting this initial mark pause
|
|
||||||
// was due to a System.gc() then the requesting thread should block in
|
|
||||||
// doit_epilogue() until the marking cycle is complete.
|
|
||||||
//
|
|
||||||
// If this initial mark pause was requested as part of a humongous
|
|
||||||
// allocation then we know that the marking cycle must just have
|
|
||||||
// been started by another thread (possibly also allocating a humongous
|
|
||||||
// object) as there was no active marking cycle when the requesting
|
|
||||||
// thread checked before calling collect() in
|
|
||||||
// attempt_allocation_humongous(). Retrying the GC, in this case,
|
|
||||||
// will cause the requesting thread to spin inside collect() until the
|
|
||||||
// just started marking cycle is complete - which may be a while. So
|
|
||||||
// we do NOT retry the GC.
|
|
||||||
if (!res) {
|
|
||||||
assert(_word_size == 0, "Concurrent Full GC/Humongous Object IM shouldn't be allocating");
|
|
||||||
if (_gc_cause != GCCause::_g1_humongous_allocation) {
|
|
||||||
_should_retry_gc = true;
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try a partial collection of some kind.
|
// Try a partial collection of some kind.
|
||||||
_gc_succeeded = g1h->do_collection_pause_at_safepoint(_target_pause_time_ms);
|
_gc_succeeded = g1h->do_collection_pause_at_safepoint(_target_pause_time_ms);
|
||||||
|
|
||||||
@ -138,66 +126,15 @@ void VM_G1CollectForAllocation::doit() {
|
|||||||
// An allocation had been requested. Do it, eventually trying a stronger
|
// An allocation had been requested. Do it, eventually trying a stronger
|
||||||
// kind of GC.
|
// kind of GC.
|
||||||
_result = g1h->satisfy_failed_allocation(_word_size, &_gc_succeeded);
|
_result = g1h->satisfy_failed_allocation(_word_size, &_gc_succeeded);
|
||||||
} else {
|
} else if (g1h->should_upgrade_to_full_gc(_gc_cause)) {
|
||||||
bool should_upgrade_to_full = g1h->should_upgrade_to_full_gc(_gc_cause);
|
// There has been a request to perform a GC to free some space. We have no
|
||||||
|
// information on how much memory has been asked for. In case there are
|
||||||
if (should_upgrade_to_full) {
|
// absolutely no regions left to allocate into, do a maximally compacting full GC.
|
||||||
// There has been a request to perform a GC to free some space. We have no
|
log_info(gc, ergo)("Attempting maximally compacting collection");
|
||||||
// information on how much memory has been asked for. In case there are
|
_gc_succeeded = g1h->do_full_collection(false, /* explicit gc */
|
||||||
// absolutely no regions left to allocate into, do a maximally compacting full GC.
|
true /* clear_all_soft_refs */);
|
||||||
log_info(gc, ergo)("Attempting maximally compacting collection");
|
|
||||||
_gc_succeeded = g1h->do_full_collection(false, /* explicit gc */
|
|
||||||
true /* clear_all_soft_refs */);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
guarantee(_gc_succeeded, "Elevated collections during the safepoint must always succeed.");
|
guarantee(_gc_succeeded, "Elevated collections during the safepoint must always succeed.");
|
||||||
} else {
|
|
||||||
assert(_result == NULL, "invariant");
|
|
||||||
// The only reason for the pause to not be successful is that, the GC locker is
|
|
||||||
// active (or has become active since the prologue was executed). In this case
|
|
||||||
// we should retry the pause after waiting for the GC locker to become inactive.
|
|
||||||
_should_retry_gc = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void VM_G1CollectForAllocation::doit_epilogue() {
|
|
||||||
VM_CollectForAllocation::doit_epilogue();
|
|
||||||
|
|
||||||
// If the pause was initiated by a System.gc() and
|
|
||||||
// +ExplicitGCInvokesConcurrent, we have to wait here for the cycle
|
|
||||||
// that just started (or maybe one that was already in progress) to
|
|
||||||
// finish.
|
|
||||||
if (GCCause::is_user_requested_gc(_gc_cause) &&
|
|
||||||
_should_initiate_conc_mark) {
|
|
||||||
assert(ExplicitGCInvokesConcurrent,
|
|
||||||
"the only way to be here is if ExplicitGCInvokesConcurrent is set");
|
|
||||||
|
|
||||||
G1CollectedHeap* g1h = G1CollectedHeap::heap();
|
|
||||||
|
|
||||||
// In the doit() method we saved g1h->old_marking_cycles_completed()
|
|
||||||
// in the _old_marking_cycles_completed_before field. We have to
|
|
||||||
// wait until we observe that g1h->old_marking_cycles_completed()
|
|
||||||
// has increased by at least one. This can happen if a) we started
|
|
||||||
// a cycle and it completes, b) a cycle already in progress
|
|
||||||
// completes, or c) a Full GC happens.
|
|
||||||
|
|
||||||
// If the condition has already been reached, there's no point in
|
|
||||||
// actually taking the lock and doing the wait.
|
|
||||||
if (g1h->old_marking_cycles_completed() <=
|
|
||||||
_old_marking_cycles_completed_before) {
|
|
||||||
// The following is largely copied from CMS
|
|
||||||
|
|
||||||
Thread* thr = Thread::current();
|
|
||||||
assert(thr->is_Java_thread(), "invariant");
|
|
||||||
JavaThread* jt = (JavaThread*)thr;
|
|
||||||
ThreadToNativeFromVM native(jt);
|
|
||||||
|
|
||||||
MonitorLocker ml(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
|
|
||||||
while (g1h->old_marking_cycles_completed() <=
|
|
||||||
_old_marking_cycles_completed_before) {
|
|
||||||
ml.wait();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -45,29 +45,39 @@ public:
|
|||||||
_gc_succeeded(false) { }
|
_gc_succeeded(false) { }
|
||||||
virtual VMOp_Type type() const { return VMOp_G1CollectFull; }
|
virtual VMOp_Type type() const { return VMOp_G1CollectFull; }
|
||||||
virtual void doit();
|
virtual void doit();
|
||||||
bool gc_succeeded() { return _gc_succeeded; }
|
bool gc_succeeded() const { return _gc_succeeded; }
|
||||||
|
};
|
||||||
|
|
||||||
|
class VM_G1TryInitiateConcMark : public VM_GC_Operation {
|
||||||
|
double _target_pause_time_ms;
|
||||||
|
bool _transient_failure;
|
||||||
|
bool _cycle_already_in_progress;
|
||||||
|
bool _gc_succeeded;
|
||||||
|
|
||||||
|
public:
|
||||||
|
VM_G1TryInitiateConcMark(uint gc_count_before,
|
||||||
|
GCCause::Cause gc_cause,
|
||||||
|
double target_pause_time_ms);
|
||||||
|
virtual VMOp_Type type() const { return VMOp_G1TryInitiateConcMark; }
|
||||||
|
virtual bool doit_prologue();
|
||||||
|
virtual void doit();
|
||||||
|
bool transient_failure() const { return _transient_failure; }
|
||||||
|
bool cycle_already_in_progress() const { return _cycle_already_in_progress; }
|
||||||
|
bool gc_succeeded() const { return _gc_succeeded; }
|
||||||
};
|
};
|
||||||
|
|
||||||
class VM_G1CollectForAllocation : public VM_CollectForAllocation {
|
class VM_G1CollectForAllocation : public VM_CollectForAllocation {
|
||||||
bool _gc_succeeded;
|
bool _gc_succeeded;
|
||||||
|
|
||||||
bool _should_initiate_conc_mark;
|
|
||||||
bool _should_retry_gc;
|
|
||||||
double _target_pause_time_ms;
|
double _target_pause_time_ms;
|
||||||
uint _old_marking_cycles_completed_before;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
VM_G1CollectForAllocation(size_t word_size,
|
VM_G1CollectForAllocation(size_t word_size,
|
||||||
uint gc_count_before,
|
uint gc_count_before,
|
||||||
GCCause::Cause gc_cause,
|
GCCause::Cause gc_cause,
|
||||||
bool should_initiate_conc_mark,
|
|
||||||
double target_pause_time_ms);
|
double target_pause_time_ms);
|
||||||
virtual VMOp_Type type() const { return VMOp_G1CollectForAllocation; }
|
virtual VMOp_Type type() const { return VMOp_G1CollectForAllocation; }
|
||||||
virtual bool doit_prologue();
|
|
||||||
virtual void doit();
|
virtual void doit();
|
||||||
virtual void doit_epilogue();
|
bool gc_succeeded() const { return _gc_succeeded; }
|
||||||
bool should_retry_gc() const { return _should_retry_gc; }
|
|
||||||
bool gc_succeeded() { return _gc_succeeded; }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Concurrent G1 stop-the-world operations such as remark and cleanup.
|
// Concurrent G1 stop-the-world operations such as remark and cleanup.
|
||||||
|
@ -90,8 +90,7 @@ void G1YoungRemSetSamplingThread::check_for_periodic_gc(){
|
|||||||
if ((os::elapsedTime() - _last_periodic_gc_attempt_s) > (G1PeriodicGCInterval / 1000.0)) {
|
if ((os::elapsedTime() - _last_periodic_gc_attempt_s) > (G1PeriodicGCInterval / 1000.0)) {
|
||||||
log_debug(gc, periodic)("Checking for periodic GC.");
|
log_debug(gc, periodic)("Checking for periodic GC.");
|
||||||
if (should_start_periodic_gc()) {
|
if (should_start_periodic_gc()) {
|
||||||
if (!G1CollectedHeap::heap()->try_collect(GCCause::_g1_periodic_collection,
|
if (!G1CollectedHeap::heap()->try_collect(GCCause::_g1_periodic_collection)) {
|
||||||
false /* retry_on_vmop_failure */)) {
|
|
||||||
log_debug(gc, periodic)("GC request denied. Skipping.");
|
log_debug(gc, periodic)("GC request denied. Skipping.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -72,6 +72,7 @@ Mutex* NonJavaThreadsListSync_lock = NULL;
|
|||||||
Monitor* CGC_lock = NULL;
|
Monitor* CGC_lock = NULL;
|
||||||
Monitor* STS_lock = NULL;
|
Monitor* STS_lock = NULL;
|
||||||
Monitor* FullGCCount_lock = NULL;
|
Monitor* FullGCCount_lock = NULL;
|
||||||
|
Monitor* G1OldGCCount_lock = NULL;
|
||||||
Monitor* DirtyCardQ_CBL_mon = NULL;
|
Monitor* DirtyCardQ_CBL_mon = NULL;
|
||||||
Mutex* Shared_DirtyCardQ_lock = NULL;
|
Mutex* Shared_DirtyCardQ_lock = NULL;
|
||||||
Mutex* MarkStackFreeList_lock = NULL;
|
Mutex* MarkStackFreeList_lock = NULL;
|
||||||
@ -203,6 +204,8 @@ void mutex_init() {
|
|||||||
|
|
||||||
def(FullGCCount_lock , PaddedMonitor, leaf, true, _safepoint_check_never); // in support of ExplicitGCInvokesConcurrent
|
def(FullGCCount_lock , PaddedMonitor, leaf, true, _safepoint_check_never); // in support of ExplicitGCInvokesConcurrent
|
||||||
if (UseG1GC) {
|
if (UseG1GC) {
|
||||||
|
def(G1OldGCCount_lock , PaddedMonitor, leaf, true, _safepoint_check_always);
|
||||||
|
|
||||||
def(DirtyCardQ_CBL_mon , PaddedMonitor, access, true, _safepoint_check_never);
|
def(DirtyCardQ_CBL_mon , PaddedMonitor, access, true, _safepoint_check_never);
|
||||||
def(Shared_DirtyCardQ_lock , PaddedMutex , access + 1, true, _safepoint_check_never);
|
def(Shared_DirtyCardQ_lock , PaddedMutex , access + 1, true, _safepoint_check_never);
|
||||||
|
|
||||||
|
@ -68,6 +68,7 @@ extern Monitor* CGC_lock; // used for coordination betwee
|
|||||||
// fore- & background GC threads.
|
// fore- & background GC threads.
|
||||||
extern Monitor* STS_lock; // used for joining/leaving SuspendibleThreadSet.
|
extern Monitor* STS_lock; // used for joining/leaving SuspendibleThreadSet.
|
||||||
extern Monitor* FullGCCount_lock; // in support of "concurrent" full gc
|
extern Monitor* FullGCCount_lock; // in support of "concurrent" full gc
|
||||||
|
extern Monitor* G1OldGCCount_lock; // in support of "concurrent" full gc
|
||||||
extern Monitor* DirtyCardQ_CBL_mon; // Protects dirty card Q
|
extern Monitor* DirtyCardQ_CBL_mon; // Protects dirty card Q
|
||||||
// completed buffer queue.
|
// completed buffer queue.
|
||||||
extern Mutex* Shared_DirtyCardQ_lock; // Lock protecting dirty card
|
extern Mutex* Shared_DirtyCardQ_lock; // Lock protecting dirty card
|
||||||
|
@ -66,6 +66,7 @@
|
|||||||
template(G1CollectForAllocation) \
|
template(G1CollectForAllocation) \
|
||||||
template(G1CollectFull) \
|
template(G1CollectFull) \
|
||||||
template(G1Concurrent) \
|
template(G1Concurrent) \
|
||||||
|
template(G1TryInitiateConcMark) \
|
||||||
template(ZMarkStart) \
|
template(ZMarkStart) \
|
||||||
template(ZMarkEnd) \
|
template(ZMarkEnd) \
|
||||||
template(ZRelocateStart) \
|
template(ZRelocateStart) \
|
||||||
|
Loading…
x
Reference in New Issue
Block a user