8179387: Factor out CMS specific code from GenCollectedHeap into its own subclass

Reviewed-by: ehelin, coleenp
This commit is contained in:
Roman Kennke 2017-10-12 15:08:19 +02:00
parent 3f2081a618
commit 4516caf125
18 changed files with 538 additions and 407 deletions

View File

@ -0,0 +1,170 @@
/*
* Copyright (c) 2017, 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/cms/concurrentMarkSweepThread.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/vmCMSOperations.hpp"
#include "gc/shared/genOopClosures.inline.hpp"
#include "gc/shared/strongRootsScope.hpp"
#include "gc/shared/workgroup.hpp"
#include "oops/oop.inline.hpp"
#include "runtime/vmThread.hpp"
#include "utilities/stack.inline.hpp"
CMSHeap::CMSHeap(GenCollectorPolicy *policy) : GenCollectedHeap(policy) {
_workers = new WorkGang("GC Thread", ParallelGCThreads,
/* are_GC_task_threads */true,
/* are_ConcurrentGC_threads */false);
_workers->initialize_workers();
}
jint CMSHeap::initialize() {
jint status = GenCollectedHeap::initialize();
if (status != JNI_OK) return status;
// If we are running CMS, create the collector responsible
// for collecting the CMS generations.
assert(collector_policy()->is_concurrent_mark_sweep_policy(), "must be CMS policy");
create_cms_collector();
return JNI_OK;
}
void CMSHeap::check_gen_kinds() {
assert(young_gen()->kind() == Generation::ParNew,
"Wrong youngest generation type");
assert(old_gen()->kind() == Generation::ConcurrentMarkSweep,
"Wrong generation kind");
}
CMSHeap* CMSHeap::heap() {
CollectedHeap* heap = Universe::heap();
assert(heap != NULL, "Uninitialized access to CMSHeap::heap()");
assert(heap->kind() == CollectedHeap::CMSHeap, "Not a CMSHeap");
return (CMSHeap*) heap;
}
void CMSHeap::gc_threads_do(ThreadClosure* tc) const {
assert(workers() != NULL, "should have workers here");
workers()->threads_do(tc);
ConcurrentMarkSweepThread::threads_do(tc);
}
void CMSHeap::print_gc_threads_on(outputStream* st) const {
assert(workers() != NULL, "should have workers here");
workers()->print_worker_threads_on(st);
ConcurrentMarkSweepThread::print_all_on(st);
}
void CMSHeap::print_on_error(outputStream* st) const {
GenCollectedHeap::print_on_error(st);
st->cr();
CMSCollector::print_on_error(st);
}
void CMSHeap::create_cms_collector() {
assert(old_gen()->kind() == Generation::ConcurrentMarkSweep,
"Unexpected generation kinds");
assert(gen_policy()->is_concurrent_mark_sweep_policy(), "Unexpected policy type");
CMSCollector* collector =
new CMSCollector((ConcurrentMarkSweepGeneration*) old_gen(),
rem_set(),
gen_policy()->as_concurrent_mark_sweep_policy());
if (!collector->completed_initialization()) {
vm_shutdown_during_initialization("Could not create CMS collector");
}
}
void CMSHeap::collect(GCCause::Cause cause) {
if (should_do_concurrent_full_gc(cause)) {
// Mostly concurrent full collection.
collect_mostly_concurrent(cause);
} else {
GenCollectedHeap::collect(cause);
}
}
bool CMSHeap::should_do_concurrent_full_gc(GCCause::Cause cause) {
switch (cause) {
case GCCause::_gc_locker: return GCLockerInvokesConcurrent;
case GCCause::_java_lang_system_gc:
case GCCause::_dcmd_gc_run: return ExplicitGCInvokesConcurrent;
default: return false;
}
}
void CMSHeap::collect_mostly_concurrent(GCCause::Cause cause) {
assert(!Heap_lock->owned_by_self(), "Should not own Heap_lock");
MutexLocker ml(Heap_lock);
// Read the GC counts while holding the Heap_lock
unsigned int full_gc_count_before = total_full_collections();
unsigned int gc_count_before = total_collections();
{
MutexUnlocker mu(Heap_lock);
VM_GenCollectFullConcurrent op(gc_count_before, full_gc_count_before, cause);
VMThread::execute(&op);
}
}
void CMSHeap::stop() {
ConcurrentMarkSweepThread::cmst()->stop();
}
void CMSHeap::cms_process_roots(StrongRootsScope* scope,
bool young_gen_as_roots,
ScanningOption so,
bool only_strong_roots,
OopsInGenClosure* root_closure,
CLDClosure* cld_closure) {
MarkingCodeBlobClosure mark_code_closure(root_closure, !CodeBlobToOopClosure::FixRelocations);
OopsInGenClosure* weak_roots = only_strong_roots ? NULL : root_closure;
CLDClosure* weak_cld_closure = only_strong_roots ? NULL : cld_closure;
process_roots(scope, so, root_closure, weak_roots, cld_closure, weak_cld_closure, &mark_code_closure);
if (!only_strong_roots) {
process_string_table_roots(scope, root_closure);
}
if (young_gen_as_roots &&
!_process_strong_tasks->is_task_claimed(GCH_PS_younger_gens)) {
root_closure->set_generation(young_gen());
young_gen()->oop_iterate(root_closure);
root_closure->reset_generation();
}
_process_strong_tasks->all_tasks_completed(scope->n_threads());
}
void CMSHeap::gc_prologue(bool full) {
always_do_update_barrier = false;
GenCollectedHeap::gc_prologue(full);
};
void CMSHeap::gc_epilogue(bool full) {
GenCollectedHeap::gc_epilogue(full);
always_do_update_barrier = true;
};

View File

@ -0,0 +1,115 @@
/*
* Copyright (c) 2017, 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.
*
*/
#ifndef SHARE_VM_GC_CMS_CMSHEAP_HPP
#define SHARE_VM_GC_CMS_CMSHEAP_HPP
#include "gc/cms/concurrentMarkSweepGeneration.hpp"
#include "gc/shared/collectedHeap.hpp"
#include "gc/shared/gcCause.hpp"
#include "gc/shared/genCollectedHeap.hpp"
class CLDClosure;
class GenCollectorPolicy;
class OopsInGenClosure;
class outputStream;
class StrongRootsScope;
class ThreadClosure;
class WorkGang;
class CMSHeap : public GenCollectedHeap {
public:
CMSHeap(GenCollectorPolicy *policy);
// Returns JNI_OK on success
virtual jint initialize();
virtual void check_gen_kinds();
// Convenience function to be used in situations where the heap type can be
// asserted to be this type.
static CMSHeap* heap();
virtual Name kind() const {
return CollectedHeap::CMSHeap;
}
virtual const char* name() const {
return "Concurrent Mark Sweep";
}
WorkGang* workers() const { return _workers; }
virtual void print_gc_threads_on(outputStream* st) const;
virtual void gc_threads_do(ThreadClosure* tc) const;
virtual void print_on_error(outputStream* st) const;
// Perform a full collection of the heap; intended for use in implementing
// "System.gc". This implies as full a collection as the CollectedHeap
// supports. Caller does not hold the Heap_lock on entry.
void collect(GCCause::Cause cause);
bool is_in_closed_subset(const void* p) const {
return is_in_reserved(p);
}
bool card_mark_must_follow_store() const {
return true;
}
void stop();
// If "young_gen_as_roots" is false, younger generations are
// not scanned as roots; in this case, the caller must be arranging to
// scan the younger generations itself. (For example, a generation might
// explicitly mark reachable objects in younger generations, to avoid
// excess storage retention.)
void cms_process_roots(StrongRootsScope* scope,
bool young_gen_as_roots,
ScanningOption so,
bool only_strong_roots,
OopsInGenClosure* root_closure,
CLDClosure* cld_closure);
private:
WorkGang* _workers;
virtual void gc_prologue(bool full);
virtual void gc_epilogue(bool full);
// Accessor for memory state verification support
NOT_PRODUCT(
virtual size_t skip_header_HeapWords() { return CMSCollector::skip_header_HeapWords(); }
)
// Returns success or failure.
void create_cms_collector();
// In support of ExplicitGCInvokesConcurrent functionality
bool should_do_concurrent_full_gc(GCCause::Cause cause);
void collect_mostly_concurrent(GCCause::Cause cause);
};
#endif // SHARE_VM_GC_CMS_CMSHEAP_HPP

View File

@ -23,13 +23,13 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/cmsLockVerifier.hpp" #include "gc/cms/cmsLockVerifier.hpp"
#include "gc/cms/compactibleFreeListSpace.hpp" #include "gc/cms/compactibleFreeListSpace.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.inline.hpp" #include "gc/cms/concurrentMarkSweepGeneration.inline.hpp"
#include "gc/cms/concurrentMarkSweepThread.hpp" #include "gc/cms/concurrentMarkSweepThread.hpp"
#include "gc/shared/blockOffsetTable.inline.hpp" #include "gc/shared/blockOffsetTable.inline.hpp"
#include "gc/shared/collectedHeap.inline.hpp" #include "gc/shared/collectedHeap.inline.hpp"
#include "gc/shared/genCollectedHeap.hpp"
#include "gc/shared/space.inline.hpp" #include "gc/shared/space.inline.hpp"
#include "gc/shared/spaceDecorator.hpp" #include "gc/shared/spaceDecorator.hpp"
#include "logging/log.hpp" #include "logging/log.hpp"
@ -154,7 +154,7 @@ HeapWord* CompactibleFreeListSpace::forward(oop q, size_t size,
cp->space->set_compaction_top(compact_top); cp->space->set_compaction_top(compact_top);
cp->space = cp->space->next_compaction_space(); cp->space = cp->space->next_compaction_space();
if (cp->space == NULL) { if (cp->space == NULL) {
cp->gen = GenCollectedHeap::heap()->young_gen(); cp->gen = CMSHeap::heap()->young_gen();
assert(cp->gen != NULL, "compaction must succeed"); assert(cp->gen != NULL, "compaction must succeed");
cp->space = cp->gen->first_compaction_space(); cp->space = cp->gen->first_compaction_space();
assert(cp->space != NULL, "generation must have a first compaction space"); assert(cp->space != NULL, "generation must have a first compaction space");
@ -2298,7 +2298,7 @@ void CompactibleFreeListSpace::verify() const {
// Iterate over all oops in the heap. Uses the _no_header version // Iterate over all oops in the heap. Uses the _no_header version
// since we are not interested in following the klass pointers. // since we are not interested in following the klass pointers.
GenCollectedHeap::heap()->oop_iterate_no_header(&cl); CMSHeap::heap()->oop_iterate_no_header(&cl);
} }
if (VerifyObjectStartArray) { if (VerifyObjectStartArray) {

View File

@ -29,6 +29,7 @@
#include "classfile/systemDictionary.hpp" #include "classfile/systemDictionary.hpp"
#include "code/codeCache.hpp" #include "code/codeCache.hpp"
#include "gc/cms/cmsCollectorPolicy.hpp" #include "gc/cms/cmsCollectorPolicy.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/cmsOopClosures.inline.hpp" #include "gc/cms/cmsOopClosures.inline.hpp"
#include "gc/cms/compactibleFreeListSpace.hpp" #include "gc/cms/compactibleFreeListSpace.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.inline.hpp" #include "gc/cms/concurrentMarkSweepGeneration.inline.hpp"
@ -298,14 +299,14 @@ void CMSCollector::ref_processor_init() {
} }
AdaptiveSizePolicy* CMSCollector::size_policy() { AdaptiveSizePolicy* CMSCollector::size_policy() {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
return gch->gen_policy()->size_policy(); return heap->gen_policy()->size_policy();
} }
void ConcurrentMarkSweepGeneration::initialize_performance_counters() { void ConcurrentMarkSweepGeneration::initialize_performance_counters() {
const char* gen_name = "old"; const char* gen_name = "old";
GenCollectorPolicy* gcp = GenCollectedHeap::heap()->gen_policy(); GenCollectorPolicy* gcp = CMSHeap::heap()->gen_policy();
// Generation Counters - generation 1, 1 subspace // Generation Counters - generation 1, 1 subspace
_gen_counters = new GenerationCounters(gen_name, 1, 1, _gen_counters = new GenerationCounters(gen_name, 1, 1,
gcp->min_old_size(), gcp->max_old_size(), &_virtual_space); gcp->min_old_size(), gcp->max_old_size(), &_virtual_space);
@ -354,8 +355,8 @@ void CMSStats::adjust_cms_free_adjustment_factor(bool fail, size_t free) {
// young generation collection. // young generation collection.
double CMSStats::time_until_cms_gen_full() const { double CMSStats::time_until_cms_gen_full() const {
size_t cms_free = _cms_gen->cmsSpace()->free(); size_t cms_free = _cms_gen->cmsSpace()->free();
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
size_t expected_promotion = MIN2(gch->young_gen()->capacity(), size_t expected_promotion = MIN2(heap->young_gen()->capacity(),
(size_t) _cms_gen->gc_stats()->avg_promoted()->padded_average()); (size_t) _cms_gen->gc_stats()->avg_promoted()->padded_average());
if (cms_free > expected_promotion) { if (cms_free > expected_promotion) {
// Start a cms collection if there isn't enough space to promote // Start a cms collection if there isn't enough space to promote
@ -595,12 +596,12 @@ CMSCollector::CMSCollector(ConcurrentMarkSweepGeneration* cmsGen,
assert(CGC_lock != NULL, "Where's the CGC_lock?"); assert(CGC_lock != NULL, "Where's the CGC_lock?");
// Support for parallelizing young gen rescan // Support for parallelizing young gen rescan
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
assert(gch->young_gen()->kind() == Generation::ParNew, "CMS can only be used with ParNew"); assert(heap->young_gen()->kind() == Generation::ParNew, "CMS can only be used with ParNew");
_young_gen = (ParNewGeneration*)gch->young_gen(); _young_gen = (ParNewGeneration*)heap->young_gen();
if (gch->supports_inline_contig_alloc()) { if (heap->supports_inline_contig_alloc()) {
_top_addr = gch->top_addr(); _top_addr = heap->top_addr();
_end_addr = gch->end_addr(); _end_addr = heap->end_addr();
assert(_young_gen != NULL, "no _young_gen"); assert(_young_gen != NULL, "no _young_gen");
_eden_chunk_index = 0; _eden_chunk_index = 0;
_eden_chunk_capacity = (_young_gen->max_capacity() + CMSSamplingGrain) / CMSSamplingGrain; _eden_chunk_capacity = (_young_gen->max_capacity() + CMSSamplingGrain) / CMSSamplingGrain;
@ -762,9 +763,9 @@ void ConcurrentMarkSweepGeneration::compute_new_size_free_list() {
log.trace(" Maximum free fraction %f", maximum_free_percentage); log.trace(" Maximum free fraction %f", maximum_free_percentage);
log.trace(" Capacity " SIZE_FORMAT, capacity() / 1000); log.trace(" Capacity " SIZE_FORMAT, capacity() / 1000);
log.trace(" Desired capacity " SIZE_FORMAT, desired_capacity / 1000); log.trace(" Desired capacity " SIZE_FORMAT, desired_capacity / 1000);
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
assert(gch->is_old_gen(this), "The CMS generation should always be the old generation"); assert(heap->is_old_gen(this), "The CMS generation should always be the old generation");
size_t young_size = gch->young_gen()->capacity(); size_t young_size = heap->young_gen()->capacity();
log.trace(" Young gen size " SIZE_FORMAT, young_size / 1000); log.trace(" Young gen size " SIZE_FORMAT, young_size / 1000);
log.trace(" unsafe_max_alloc_nogc " SIZE_FORMAT, unsafe_max_alloc_nogc() / 1000); log.trace(" unsafe_max_alloc_nogc " SIZE_FORMAT, unsafe_max_alloc_nogc() / 1000);
log.trace(" contiguous available " SIZE_FORMAT, contiguous_available() / 1000); log.trace(" contiguous available " SIZE_FORMAT, contiguous_available() / 1000);
@ -923,7 +924,7 @@ oop ConcurrentMarkSweepGeneration::promote(oop obj, size_t obj_size) {
assert_lock_strong(freelistLock()); assert_lock_strong(freelistLock());
#ifndef PRODUCT #ifndef PRODUCT
if (GenCollectedHeap::heap()->promotion_should_fail()) { if (CMSHeap::heap()->promotion_should_fail()) {
return NULL; return NULL;
} }
#endif // #ifndef PRODUCT #endif // #ifndef PRODUCT
@ -1000,7 +1001,7 @@ ConcurrentMarkSweepGeneration::par_promote(int thread_num,
oop old, markOop m, oop old, markOop m,
size_t word_sz) { size_t word_sz) {
#ifndef PRODUCT #ifndef PRODUCT
if (GenCollectedHeap::heap()->promotion_should_fail()) { if (CMSHeap::heap()->promotion_should_fail()) {
return NULL; return NULL;
} }
#endif // #ifndef PRODUCT #endif // #ifndef PRODUCT
@ -1179,10 +1180,10 @@ bool CMSCollector::shouldConcurrentCollect() {
// We start a collection if we believe an incremental collection may fail; // We start a collection if we believe an incremental collection may fail;
// this is not likely to be productive in practice because it's probably too // this is not likely to be productive in practice because it's probably too
// late anyway. // late anyway.
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
assert(gch->collector_policy()->is_generation_policy(), assert(heap->collector_policy()->is_generation_policy(),
"You may want to check the correctness of the following"); "You may want to check the correctness of the following");
if (gch->incremental_collection_will_fail(true /* consult_young */)) { if (heap->incremental_collection_will_fail(true /* consult_young */)) {
log.print("CMSCollector: collect because incremental collection will fail "); log.print("CMSCollector: collect because incremental collection will fail ");
return true; return true;
} }
@ -1294,8 +1295,8 @@ void CMSCollector::collect(bool full,
} }
void CMSCollector::request_full_gc(unsigned int full_gc_count, GCCause::Cause cause) { void CMSCollector::request_full_gc(unsigned int full_gc_count, GCCause::Cause cause) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
unsigned int gc_count = gch->total_full_collections(); unsigned int gc_count = heap->total_full_collections();
if (gc_count == full_gc_count) { if (gc_count == full_gc_count) {
MutexLockerEx y(CGC_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx y(CGC_lock, Mutex::_no_safepoint_check_flag);
_full_gc_requested = true; _full_gc_requested = true;
@ -1307,7 +1308,7 @@ void CMSCollector::request_full_gc(unsigned int full_gc_count, GCCause::Cause ca
} }
bool CMSCollector::is_external_interruption() { bool CMSCollector::is_external_interruption() {
GCCause::Cause cause = GenCollectedHeap::heap()->gc_cause(); GCCause::Cause cause = CMSHeap::heap()->gc_cause();
return GCCause::is_user_requested_gc(cause) || return GCCause::is_user_requested_gc(cause) ||
GCCause::is_serviceability_requested_gc(cause); GCCause::is_serviceability_requested_gc(cause);
} }
@ -1456,8 +1457,8 @@ void CMSCollector::acquire_control_and_collect(bool full,
// Inform cms gen if this was due to partial collection failing. // Inform cms gen if this was due to partial collection failing.
// The CMS gen may use this fact to determine its expansion policy. // The CMS gen may use this fact to determine its expansion policy.
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
if (gch->incremental_collection_will_fail(false /* don't consult_young */)) { if (heap->incremental_collection_will_fail(false /* don't consult_young */)) {
assert(!_cmsGen->incremental_collection_failed(), assert(!_cmsGen->incremental_collection_failed(),
"Should have been noticed, reacted to and cleared"); "Should have been noticed, reacted to and cleared");
_cmsGen->set_incremental_collection_failed(); _cmsGen->set_incremental_collection_failed();
@ -1489,14 +1490,14 @@ void CMSCollector::acquire_control_and_collect(bool full,
// Has the GC time limit been exceeded? // Has the GC time limit been exceeded?
size_t max_eden_size = _young_gen->max_eden_size(); size_t max_eden_size = _young_gen->max_eden_size();
GCCause::Cause gc_cause = gch->gc_cause(); GCCause::Cause gc_cause = heap->gc_cause();
size_policy()->check_gc_overhead_limit(_young_gen->used(), size_policy()->check_gc_overhead_limit(_young_gen->used(),
_young_gen->eden()->used(), _young_gen->eden()->used(),
_cmsGen->max_capacity(), _cmsGen->max_capacity(),
max_eden_size, max_eden_size,
full, full,
gc_cause, gc_cause,
gch->collector_policy()); heap->collector_policy());
// Reset the expansion cause, now that we just completed // Reset the expansion cause, now that we just completed
// a collection cycle. // a collection cycle.
@ -1518,21 +1519,21 @@ void CMSCollector::compute_new_size() {
// A work method used by the foreground collector to do // A work method used by the foreground collector to do
// a mark-sweep-compact. // a mark-sweep-compact.
void CMSCollector::do_compaction_work(bool clear_all_soft_refs) { void CMSCollector::do_compaction_work(bool clear_all_soft_refs) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
STWGCTimer* gc_timer = GenMarkSweep::gc_timer(); STWGCTimer* gc_timer = GenMarkSweep::gc_timer();
gc_timer->register_gc_start(); gc_timer->register_gc_start();
SerialOldTracer* gc_tracer = GenMarkSweep::gc_tracer(); SerialOldTracer* gc_tracer = GenMarkSweep::gc_tracer();
gc_tracer->report_gc_start(gch->gc_cause(), gc_timer->gc_start()); gc_tracer->report_gc_start(heap->gc_cause(), gc_timer->gc_start());
gch->pre_full_gc_dump(gc_timer); heap->pre_full_gc_dump(gc_timer);
GCTraceTime(Trace, gc, phases) t("CMS:MSC"); GCTraceTime(Trace, gc, phases) t("CMS:MSC");
// Temporarily widen the span of the weak reference processing to // Temporarily widen the span of the weak reference processing to
// the entire heap. // the entire heap.
MemRegion new_span(GenCollectedHeap::heap()->reserved_region()); MemRegion new_span(CMSHeap::heap()->reserved_region());
ReferenceProcessorSpanMutator rp_mut_span(ref_processor(), new_span); ReferenceProcessorSpanMutator rp_mut_span(ref_processor(), new_span);
// Temporarily, clear the "is_alive_non_header" field of the // Temporarily, clear the "is_alive_non_header" field of the
// reference processor. // reference processor.
@ -1608,7 +1609,7 @@ void CMSCollector::do_compaction_work(bool clear_all_soft_refs) {
// No longer a need to do a concurrent collection for Metaspace. // No longer a need to do a concurrent collection for Metaspace.
MetaspaceGC::set_should_concurrent_collect(false); MetaspaceGC::set_should_concurrent_collect(false);
gch->post_full_gc_dump(gc_timer); heap->post_full_gc_dump(gc_timer);
gc_timer->register_gc_end(); gc_timer->register_gc_end();
@ -1702,7 +1703,7 @@ void CMSCollector::collect_in_background(GCCause::Cause cause) {
assert(Thread::current()->is_ConcurrentGC_thread(), assert(Thread::current()->is_ConcurrentGC_thread(),
"A CMS asynchronous collection is only allowed on a CMS thread."); "A CMS asynchronous collection is only allowed on a CMS thread.");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
{ {
bool safepoint_check = Mutex::_no_safepoint_check_flag; bool safepoint_check = Mutex::_no_safepoint_check_flag;
MutexLockerEx hl(Heap_lock, safepoint_check); MutexLockerEx hl(Heap_lock, safepoint_check);
@ -1731,8 +1732,8 @@ void CMSCollector::collect_in_background(GCCause::Cause cause) {
_full_gc_requested = false; // acks all outstanding full gc requests _full_gc_requested = false; // acks all outstanding full gc requests
_full_gc_cause = GCCause::_no_gc; _full_gc_cause = GCCause::_no_gc;
// Signal that we are about to start a collection // Signal that we are about to start a collection
gch->increment_total_full_collections(); // ... starting a collection cycle heap->increment_total_full_collections(); // ... starting a collection cycle
_collection_count_start = gch->total_full_collections(); _collection_count_start = heap->total_full_collections();
} }
size_t prev_used = _cmsGen->used(); size_t prev_used = _cmsGen->used();
@ -1925,9 +1926,9 @@ void CMSCollector::register_gc_end() {
} }
void CMSCollector::save_heap_summary() { void CMSCollector::save_heap_summary() {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
_last_heap_summary = gch->create_heap_summary(); _last_heap_summary = heap->create_heap_summary();
_last_metaspace_summary = gch->create_metaspace_summary(); _last_metaspace_summary = heap->create_metaspace_summary();
} }
void CMSCollector::report_heap_summary(GCWhen::Type when) { void CMSCollector::report_heap_summary(GCWhen::Type when) {
@ -2303,10 +2304,10 @@ bool CMSCollector::verify_after_remark() {
assert(verification_mark_stack()->isEmpty(), "markStack should be empty"); assert(verification_mark_stack()->isEmpty(), "markStack should be empty");
verify_work_stacks_empty(); verify_work_stacks_empty();
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
gch->ensure_parsability(false); // fill TLABs, but no need to retire them heap->ensure_parsability(false); // fill TLABs, but no need to retire them
// Update the saved marks which may affect the root scans. // Update the saved marks which may affect the root scans.
gch->save_marks(); heap->save_marks();
if (CMSRemarkVerifyVariant == 1) { if (CMSRemarkVerifyVariant == 1) {
// In this first variant of verification, we complete // In this first variant of verification, we complete
@ -2329,19 +2330,19 @@ bool CMSCollector::verify_after_remark() {
void CMSCollector::verify_after_remark_work_1() { void CMSCollector::verify_after_remark_work_1() {
ResourceMark rm; ResourceMark rm;
HandleMark hm; HandleMark hm;
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
// Get a clear set of claim bits for the roots processing to work with. // Get a clear set of claim bits for the roots processing to work with.
ClassLoaderDataGraph::clear_claimed_marks(); ClassLoaderDataGraph::clear_claimed_marks();
// Mark from roots one level into CMS // Mark from roots one level into CMS
MarkRefsIntoClosure notOlder(_span, verification_mark_bm()); MarkRefsIntoClosure notOlder(_span, verification_mark_bm());
gch->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel. heap->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel.
{ {
StrongRootsScope srs(1); StrongRootsScope srs(1);
gch->cms_process_roots(&srs, heap->cms_process_roots(&srs,
true, // young gen as roots true, // young gen as roots
GenCollectedHeap::ScanningOption(roots_scanning_options()), GenCollectedHeap::ScanningOption(roots_scanning_options()),
should_unload_classes(), should_unload_classes(),
@ -2376,7 +2377,7 @@ void CMSCollector::verify_after_remark_work_1() {
log.error("Failed marking verification after remark"); log.error("Failed marking verification after remark");
ResourceMark rm; ResourceMark rm;
LogStream ls(log.error()); LogStream ls(log.error());
gch->print_on(&ls); heap->print_on(&ls);
fatal("CMS: failed marking verification after remark"); fatal("CMS: failed marking verification after remark");
} }
} }
@ -2399,7 +2400,7 @@ class VerifyCLDOopsCLDClosure : public CLDClosure {
void CMSCollector::verify_after_remark_work_2() { void CMSCollector::verify_after_remark_work_2() {
ResourceMark rm; ResourceMark rm;
HandleMark hm; HandleMark hm;
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
// Get a clear set of claim bits for the roots processing to work with. // Get a clear set of claim bits for the roots processing to work with.
ClassLoaderDataGraph::clear_claimed_marks(); ClassLoaderDataGraph::clear_claimed_marks();
@ -2409,12 +2410,12 @@ void CMSCollector::verify_after_remark_work_2() {
markBitMap()); markBitMap());
CLDToOopClosure cld_closure(&notOlder, true); CLDToOopClosure cld_closure(&notOlder, true);
gch->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel. heap->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel.
{ {
StrongRootsScope srs(1); StrongRootsScope srs(1);
gch->cms_process_roots(&srs, heap->cms_process_roots(&srs,
true, // young gen as roots true, // young gen as roots
GenCollectedHeap::ScanningOption(roots_scanning_options()), GenCollectedHeap::ScanningOption(roots_scanning_options()),
should_unload_classes(), should_unload_classes(),
@ -2803,7 +2804,7 @@ class CMSParInitialMarkTask: public CMSParMarkTask {
void CMSCollector::checkpointRootsInitial() { void CMSCollector::checkpointRootsInitial() {
assert(_collectorState == InitialMarking, "Wrong collector state"); assert(_collectorState == InitialMarking, "Wrong collector state");
check_correct_thread_executing(); check_correct_thread_executing();
TraceCMSMemoryManagerStats tms(_collectorState,GenCollectedHeap::heap()->gc_cause()); TraceCMSMemoryManagerStats tms(_collectorState, CMSHeap::heap()->gc_cause());
save_heap_summary(); save_heap_summary();
report_heap_summary(GCWhen::BeforeGC); report_heap_summary(GCWhen::BeforeGC);
@ -2844,14 +2845,14 @@ void CMSCollector::checkpointRootsInitialWork() {
HandleMark hm; HandleMark hm;
MarkRefsIntoClosure notOlder(_span, &_markBitMap); MarkRefsIntoClosure notOlder(_span, &_markBitMap);
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
verify_work_stacks_empty(); verify_work_stacks_empty();
verify_overflow_empty(); verify_overflow_empty();
gch->ensure_parsability(false); // fill TLABs, but no need to retire them heap->ensure_parsability(false); // fill TLABs, but no need to retire them
// Update the saved marks which may affect the root scans. // Update the saved marks which may affect the root scans.
gch->save_marks(); heap->save_marks();
// weak reference processing has not started yet. // weak reference processing has not started yet.
ref_processor()->set_enqueuing_is_done(false); ref_processor()->set_enqueuing_is_done(false);
@ -2872,7 +2873,7 @@ void CMSCollector::checkpointRootsInitialWork() {
#endif #endif
if (CMSParallelInitialMarkEnabled) { if (CMSParallelInitialMarkEnabled) {
// The parallel version. // The parallel version.
WorkGang* workers = gch->workers(); WorkGang* workers = heap->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
uint n_workers = workers->active_workers(); uint n_workers = workers->active_workers();
@ -2891,11 +2892,11 @@ void CMSCollector::checkpointRootsInitialWork() {
} else { } else {
// The serial version. // The serial version.
CLDToOopClosure cld_closure(&notOlder, true); CLDToOopClosure cld_closure(&notOlder, true);
gch->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel. heap->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel.
StrongRootsScope srs(1); StrongRootsScope srs(1);
gch->cms_process_roots(&srs, heap->cms_process_roots(&srs,
true, // young gen as roots true, // young gen as roots
GenCollectedHeap::ScanningOption(roots_scanning_options()), GenCollectedHeap::ScanningOption(roots_scanning_options()),
should_unload_classes(), should_unload_classes(),
@ -3800,7 +3801,7 @@ size_t CMSCollector::preclean_work(bool clean_refs, bool clean_survivor) {
bitMapLock()); bitMapLock());
startTimer(); startTimer();
unsigned int before_count = unsigned int before_count =
GenCollectedHeap::heap()->total_collections(); CMSHeap::heap()->total_collections();
SurvivorSpacePrecleanClosure SurvivorSpacePrecleanClosure
sss_cl(this, _span, &_markBitMap, &_markStack, sss_cl(this, _span, &_markBitMap, &_markStack,
&pam_cl, before_count, CMSYield); &pam_cl, before_count, CMSYield);
@ -4103,7 +4104,7 @@ void CMSCollector::checkpointRootsFinal() {
// world is stopped at this checkpoint // world is stopped at this checkpoint
assert(SafepointSynchronize::is_at_safepoint(), assert(SafepointSynchronize::is_at_safepoint(),
"world should be stopped"); "world should be stopped");
TraceCMSMemoryManagerStats tms(_collectorState,GenCollectedHeap::heap()->gc_cause()); TraceCMSMemoryManagerStats tms(_collectorState, CMSHeap::heap()->gc_cause());
verify_work_stacks_empty(); verify_work_stacks_empty();
verify_overflow_empty(); verify_overflow_empty();
@ -4112,16 +4113,16 @@ void CMSCollector::checkpointRootsFinal() {
_young_gen->used() / K, _young_gen->capacity() / K); _young_gen->used() / K, _young_gen->capacity() / K);
{ {
if (CMSScavengeBeforeRemark) { if (CMSScavengeBeforeRemark) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
// Temporarily set flag to false, GCH->do_collection will // Temporarily set flag to false, GCH->do_collection will
// expect it to be false and set to true // expect it to be false and set to true
FlagSetting fl(gch->_is_gc_active, false); FlagSetting fl(heap->_is_gc_active, false);
gch->do_collection(true, // full (i.e. force, see below) heap->do_collection(true, // full (i.e. force, see below)
false, // !clear_all_soft_refs false, // !clear_all_soft_refs
0, // size 0, // size
false, // is_tlab false, // is_tlab
GenCollectedHeap::YoungGen // type GenCollectedHeap::YoungGen // type
); );
} }
FreelistLocker x(this); FreelistLocker x(this);
@ -4142,7 +4143,7 @@ void CMSCollector::checkpointRootsFinalWork() {
ResourceMark rm; ResourceMark rm;
HandleMark hm; HandleMark hm;
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
if (should_unload_classes()) { if (should_unload_classes()) {
CodeCache::gc_prologue(); CodeCache::gc_prologue();
@ -4162,9 +4163,9 @@ void CMSCollector::checkpointRootsFinalWork() {
// or of an indication of whether the scavenge did indeed occur, // or of an indication of whether the scavenge did indeed occur,
// we cannot rely on TLAB's having been filled and must do // we cannot rely on TLAB's having been filled and must do
// so here just in case a scavenge did not happen. // so here just in case a scavenge did not happen.
gch->ensure_parsability(false); // fill TLAB's, but no need to retire them heap->ensure_parsability(false); // fill TLAB's, but no need to retire them
// Update the saved marks which may affect the root scans. // Update the saved marks which may affect the root scans.
gch->save_marks(); heap->save_marks();
print_eden_and_survivor_chunk_arrays(); print_eden_and_survivor_chunk_arrays();
@ -4240,7 +4241,7 @@ void CMSCollector::checkpointRootsFinalWork() {
_markStack._failed_double = 0; _markStack._failed_double = 0;
if ((VerifyAfterGC || VerifyDuringGC) && if ((VerifyAfterGC || VerifyDuringGC) &&
GenCollectedHeap::heap()->total_collections() >= VerifyGCStartAt) { CMSHeap::heap()->total_collections() >= VerifyGCStartAt) {
verify_after_remark(); verify_after_remark();
} }
@ -4262,7 +4263,7 @@ void CMSParInitialMarkTask::work(uint worker_id) {
// ---------- scan from roots -------------- // ---------- scan from roots --------------
_timer.start(); _timer.start();
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
ParMarkRefsIntoClosure par_mri_cl(_collector->_span, &(_collector->_markBitMap)); ParMarkRefsIntoClosure par_mri_cl(_collector->_span, &(_collector->_markBitMap));
// ---------- young gen roots -------------- // ---------- young gen roots --------------
@ -4278,12 +4279,12 @@ void CMSParInitialMarkTask::work(uint worker_id) {
CLDToOopClosure cld_closure(&par_mri_cl, true); CLDToOopClosure cld_closure(&par_mri_cl, true);
gch->cms_process_roots(_strong_roots_scope, heap->cms_process_roots(_strong_roots_scope,
false, // yg was scanned above false, // yg was scanned above
GenCollectedHeap::ScanningOption(_collector->CMSCollector::roots_scanning_options()), GenCollectedHeap::ScanningOption(_collector->CMSCollector::roots_scanning_options()),
_collector->should_unload_classes(), _collector->should_unload_classes(),
&par_mri_cl, &par_mri_cl,
&cld_closure); &cld_closure);
assert(_collector->should_unload_classes() assert(_collector->should_unload_classes()
|| (_collector->CMSCollector::roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache), || (_collector->CMSCollector::roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache),
"if we didn't scan the code cache, we have to be ready to drop nmethods with expired weak oops"); "if we didn't scan the code cache, we have to be ready to drop nmethods with expired weak oops");
@ -4387,7 +4388,7 @@ void CMSParRemarkTask::work(uint worker_id) {
// ---------- rescan from roots -------------- // ---------- rescan from roots --------------
_timer.start(); _timer.start();
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
ParMarkRefsIntoAndScanClosure par_mrias_cl(_collector, ParMarkRefsIntoAndScanClosure par_mrias_cl(_collector,
_collector->_span, _collector->ref_processor(), _collector->_span, _collector->ref_processor(),
&(_collector->_markBitMap), &(_collector->_markBitMap),
@ -4407,12 +4408,12 @@ void CMSParRemarkTask::work(uint worker_id) {
// ---------- remaining roots -------------- // ---------- remaining roots --------------
_timer.reset(); _timer.reset();
_timer.start(); _timer.start();
gch->cms_process_roots(_strong_roots_scope, heap->cms_process_roots(_strong_roots_scope,
false, // yg was scanned above false, // yg was scanned above
GenCollectedHeap::ScanningOption(_collector->CMSCollector::roots_scanning_options()), GenCollectedHeap::ScanningOption(_collector->CMSCollector::roots_scanning_options()),
_collector->should_unload_classes(), _collector->should_unload_classes(),
&par_mrias_cl, &par_mrias_cl,
NULL); // The dirty klasses will be handled below NULL); // The dirty klasses will be handled below
assert(_collector->should_unload_classes() assert(_collector->should_unload_classes()
|| (_collector->CMSCollector::roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache), || (_collector->CMSCollector::roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache),
@ -4839,8 +4840,8 @@ initialize_sequential_subtasks_for_young_gen_rescan(int n_threads) {
// Parallel version of remark // Parallel version of remark
void CMSCollector::do_remark_parallel() { void CMSCollector::do_remark_parallel() {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
WorkGang* workers = gch->workers(); WorkGang* workers = heap->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
// Choose to use the number of GC workers most recently set // Choose to use the number of GC workers most recently set
// into "active_workers". // into "active_workers".
@ -4856,7 +4857,7 @@ void CMSCollector::do_remark_parallel() {
// the younger_gen cards, so we shouldn't call the following else // the younger_gen cards, so we shouldn't call the following else
// the verification code as well as subsequent younger_refs_iterate // the verification code as well as subsequent younger_refs_iterate
// code would get confused. XXX // code would get confused. XXX
// gch->rem_set()->prepare_for_younger_refs_iterate(true); // parallel // heap->rem_set()->prepare_for_younger_refs_iterate(true); // parallel
// The young gen rescan work will not be done as part of // The young gen rescan work will not be done as part of
// process_roots (which currently doesn't know how to // process_roots (which currently doesn't know how to
@ -4898,7 +4899,7 @@ void CMSCollector::do_remark_parallel() {
void CMSCollector::do_remark_non_parallel() { void CMSCollector::do_remark_non_parallel() {
ResourceMark rm; ResourceMark rm;
HandleMark hm; HandleMark hm;
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
ReferenceProcessorMTDiscoveryMutator mt(ref_processor(), false); ReferenceProcessorMTDiscoveryMutator mt(ref_processor(), false);
MarkRefsIntoAndScanClosure MarkRefsIntoAndScanClosure
@ -4939,7 +4940,7 @@ void CMSCollector::do_remark_non_parallel() {
} }
} }
if (VerifyDuringGC && if (VerifyDuringGC &&
GenCollectedHeap::heap()->total_collections() >= VerifyGCStartAt) { CMSHeap::heap()->total_collections() >= VerifyGCStartAt) {
HandleMark hm; // Discard invalid handles created during verification HandleMark hm; // Discard invalid handles created during verification
Universe::verify(); Universe::verify();
} }
@ -4948,15 +4949,15 @@ void CMSCollector::do_remark_non_parallel() {
verify_work_stacks_empty(); verify_work_stacks_empty();
gch->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel. heap->rem_set()->prepare_for_younger_refs_iterate(false); // Not parallel.
StrongRootsScope srs(1); StrongRootsScope srs(1);
gch->cms_process_roots(&srs, heap->cms_process_roots(&srs,
true, // young gen as roots true, // young gen as roots
GenCollectedHeap::ScanningOption(roots_scanning_options()), GenCollectedHeap::ScanningOption(roots_scanning_options()),
should_unload_classes(), should_unload_classes(),
&mrias_cl, &mrias_cl,
NULL); // The dirty klasses will be handled below NULL); // The dirty klasses will be handled below
assert(should_unload_classes() assert(should_unload_classes()
|| (roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache), || (roots_scanning_options() & GenCollectedHeap::SO_AllCodeCache),
@ -5148,8 +5149,8 @@ void CMSRefProcTaskProxy::do_work_steal(int i,
void CMSRefProcTaskExecutor::execute(ProcessTask& task) void CMSRefProcTaskExecutor::execute(ProcessTask& task)
{ {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
WorkGang* workers = gch->workers(); WorkGang* workers = heap->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
CMSRefProcTaskProxy rp_task(task, &_collector, CMSRefProcTaskProxy rp_task(task, &_collector,
_collector.ref_processor()->span(), _collector.ref_processor()->span(),
@ -5161,8 +5162,8 @@ void CMSRefProcTaskExecutor::execute(ProcessTask& task)
void CMSRefProcTaskExecutor::execute(EnqueueTask& task) void CMSRefProcTaskExecutor::execute(EnqueueTask& task)
{ {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
WorkGang* workers = gch->workers(); WorkGang* workers = heap->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
CMSRefEnqueueTaskProxy enq_task(task); CMSRefEnqueueTaskProxy enq_task(task);
workers->run_task(&enq_task); workers->run_task(&enq_task);
@ -5195,9 +5196,9 @@ void CMSCollector::refProcessingWork() {
// and a different number of discovered lists may have Ref objects. // and a different number of discovered lists may have Ref objects.
// That is OK as long as the Reference lists are balanced (see // That is OK as long as the Reference lists are balanced (see
// balance_all_queues() and balance_queues()). // balance_all_queues() and balance_queues()).
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
uint active_workers = ParallelGCThreads; uint active_workers = ParallelGCThreads;
WorkGang* workers = gch->workers(); WorkGang* workers = heap->workers();
if (workers != NULL) { if (workers != NULL) {
active_workers = workers->active_workers(); active_workers = workers->active_workers();
// The expectation is that active_workers will have already // The expectation is that active_workers will have already
@ -5305,7 +5306,7 @@ void CMSCollector::sweep() {
verify_work_stacks_empty(); verify_work_stacks_empty();
verify_overflow_empty(); verify_overflow_empty();
increment_sweep_count(); increment_sweep_count();
TraceCMSMemoryManagerStats tms(_collectorState,GenCollectedHeap::heap()->gc_cause()); TraceCMSMemoryManagerStats tms(_collectorState, CMSHeap::heap()->gc_cause());
_inter_sweep_timer.stop(); _inter_sweep_timer.stop();
_inter_sweep_estimate.sample(_inter_sweep_timer.seconds()); _inter_sweep_estimate.sample(_inter_sweep_timer.seconds());
@ -5378,9 +5379,9 @@ void CMSCollector::sweep() {
// this generation. If such a promotion may still fail, // this generation. If such a promotion may still fail,
// the flag will be set again when a young collection is // the flag will be set again when a young collection is
// attempted. // attempted.
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
gch->clear_incremental_collection_failed(); // Worth retrying as fresh space may have been freed up heap->clear_incremental_collection_failed(); // Worth retrying as fresh space may have been freed up
gch->update_full_collections_completed(_collection_count_start); heap->update_full_collections_completed(_collection_count_start);
} }
// FIX ME!!! Looks like this belongs in CFLSpace, with // FIX ME!!! Looks like this belongs in CFLSpace, with
@ -5415,7 +5416,7 @@ void ConcurrentMarkSweepGeneration::update_gc_stats(Generation* current_generati
bool full) { bool full) {
// If the young generation has been collected, gather any statistics // If the young generation has been collected, gather any statistics
// that are of interest at this point. // that are of interest at this point.
bool current_is_young = GenCollectedHeap::heap()->is_young_gen(current_generation); bool current_is_young = CMSHeap::heap()->is_young_gen(current_generation);
if (!full && current_is_young) { if (!full && current_is_young) {
// Gather statistics on the young generation collection. // Gather statistics on the young generation collection.
collector()->stats().record_gc0_end(used()); collector()->stats().record_gc0_end(used());
@ -6188,7 +6189,7 @@ size_t SurvivorSpacePrecleanClosure::do_object_careful(oop p) {
do_yield_check(); do_yield_check();
} }
unsigned int after_count = unsigned int after_count =
GenCollectedHeap::heap()->total_collections(); CMSHeap::heap()->total_collections();
bool abort = (_before_count != after_count) || bool abort = (_before_count != after_count) ||
_collector->should_abort_preclean(); _collector->should_abort_preclean();
return abort ? 0 : size; return abort ? 0 : size;

View File

@ -25,13 +25,13 @@
#ifndef SHARE_VM_GC_CMS_CONCURRENTMARKSWEEPGENERATION_INLINE_HPP #ifndef SHARE_VM_GC_CMS_CONCURRENTMARKSWEEPGENERATION_INLINE_HPP
#define SHARE_VM_GC_CMS_CONCURRENTMARKSWEEPGENERATION_INLINE_HPP #define SHARE_VM_GC_CMS_CONCURRENTMARKSWEEPGENERATION_INLINE_HPP
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/cmsLockVerifier.hpp" #include "gc/cms/cmsLockVerifier.hpp"
#include "gc/cms/compactibleFreeListSpace.hpp" #include "gc/cms/compactibleFreeListSpace.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.hpp" #include "gc/cms/concurrentMarkSweepGeneration.hpp"
#include "gc/cms/concurrentMarkSweepThread.hpp" #include "gc/cms/concurrentMarkSweepThread.hpp"
#include "gc/cms/parNewGeneration.hpp" #include "gc/cms/parNewGeneration.hpp"
#include "gc/shared/gcUtil.hpp" #include "gc/shared/gcUtil.hpp"
#include "gc/shared/genCollectedHeap.hpp"
#include "utilities/align.hpp" #include "utilities/align.hpp"
#include "utilities/bitMap.inline.hpp" #include "utilities/bitMap.inline.hpp"
@ -256,7 +256,7 @@ inline bool CMSCollector::should_abort_preclean() const {
// scavenge is done or foreground GC wants to take over collection // scavenge is done or foreground GC wants to take over collection
return _collectorState == AbortablePreclean && return _collectorState == AbortablePreclean &&
(_abort_preclean || _foregroundGCIsActive || (_abort_preclean || _foregroundGCIsActive ||
GenCollectedHeap::heap()->incremental_collection_will_fail(true /* consult_young */)); CMSHeap::heap()->incremental_collection_will_fail(true /* consult_young */));
} }
inline size_t CMSCollector::get_eden_used() const { inline size_t CMSCollector::get_eden_used() const {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -24,10 +24,10 @@
#include "precompiled.hpp" #include "precompiled.hpp"
#include "classfile/systemDictionary.hpp" #include "classfile/systemDictionary.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.inline.hpp" #include "gc/cms/concurrentMarkSweepGeneration.inline.hpp"
#include "gc/cms/concurrentMarkSweepThread.hpp" #include "gc/cms/concurrentMarkSweepThread.hpp"
#include "gc/shared/gcId.hpp" #include "gc/shared/gcId.hpp"
#include "gc/shared/genCollectedHeap.hpp"
#include "oops/oop.inline.hpp" #include "oops/oop.inline.hpp"
#include "runtime/init.hpp" #include "runtime/init.hpp"
#include "runtime/interfaceSupport.hpp" #include "runtime/interfaceSupport.hpp"
@ -225,7 +225,7 @@ void ConcurrentMarkSweepThread::wait_on_cms_lock_for_scavenge(long t_millis) {
// Wait time in millis or 0 value representing infinite wait for a scavenge // Wait time in millis or 0 value representing infinite wait for a scavenge
assert(t_millis >= 0, "Wait time for scavenge should be 0 or positive"); assert(t_millis >= 0, "Wait time for scavenge should be 0 or positive");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
double start_time_secs = os::elapsedTime(); double start_time_secs = os::elapsedTime();
double end_time_secs = start_time_secs + (t_millis / ((double) MILLIUNITS)); double end_time_secs = start_time_secs + (t_millis / ((double) MILLIUNITS));
@ -233,7 +233,7 @@ void ConcurrentMarkSweepThread::wait_on_cms_lock_for_scavenge(long t_millis) {
unsigned int before_count; unsigned int before_count;
{ {
MutexLockerEx hl(Heap_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx hl(Heap_lock, Mutex::_no_safepoint_check_flag);
before_count = gch->total_collections(); before_count = heap->total_collections();
} }
unsigned int loop_count = 0; unsigned int loop_count = 0;
@ -279,7 +279,7 @@ void ConcurrentMarkSweepThread::wait_on_cms_lock_for_scavenge(long t_millis) {
unsigned int after_count; unsigned int after_count;
{ {
MutexLockerEx hl(Heap_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx hl(Heap_lock, Mutex::_no_safepoint_check_flag);
after_count = gch->total_collections(); after_count = heap->total_collections();
} }
if(before_count != after_count) { if(before_count != after_count) {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2007, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2007, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,10 +23,10 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/shared/cardTableModRefBS.hpp" #include "gc/shared/cardTableModRefBS.hpp"
#include "gc/shared/cardTableRS.hpp" #include "gc/shared/cardTableRS.hpp"
#include "gc/shared/collectedHeap.hpp" #include "gc/shared/collectedHeap.hpp"
#include "gc/shared/genCollectedHeap.hpp"
#include "gc/shared/space.inline.hpp" #include "gc/shared/space.inline.hpp"
#include "memory/allocation.inline.hpp" #include "memory/allocation.inline.hpp"
#include "memory/virtualspace.hpp" #include "memory/virtualspace.hpp"
@ -394,7 +394,7 @@ get_LNC_array_for_space(Space* sp,
// Do a dirty read here. If we pass the conditional then take the rare // Do a dirty read here. If we pass the conditional then take the rare
// event lock and do the read again in case some other thread had already // event lock and do the read again in case some other thread had already
// succeeded and done the resize. // succeeded and done the resize.
int cur_collection = GenCollectedHeap::heap()->total_collections(); int cur_collection = CMSHeap::heap()->total_collections();
// Updated _last_LNC_resizing_collection[i] must not be visible before // Updated _last_LNC_resizing_collection[i] must not be visible before
// _lowest_non_clean and friends are visible. Therefore use acquire/release // _lowest_non_clean and friends are visible. Therefore use acquire/release
// to guarantee this on non TSO architecures. // to guarantee this on non TSO architecures.

View File

@ -23,6 +23,7 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/compactibleFreeListSpace.hpp" #include "gc/cms/compactibleFreeListSpace.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.hpp" #include "gc/cms/concurrentMarkSweepGeneration.hpp"
#include "gc/cms/parNewGeneration.inline.hpp" #include "gc/cms/parNewGeneration.inline.hpp"
@ -124,7 +125,7 @@ bool ParScanThreadState::should_be_partially_scanned(oop new_obj, oop old_obj) c
void ParScanThreadState::scan_partial_array_and_push_remainder(oop old) { void ParScanThreadState::scan_partial_array_and_push_remainder(oop old) {
assert(old->is_objArray(), "must be obj array"); assert(old->is_objArray(), "must be obj array");
assert(old->is_forwarded(), "must be forwarded"); assert(old->is_forwarded(), "must be forwarded");
assert(GenCollectedHeap::heap()->is_in_reserved(old), "must be in heap."); assert(CMSHeap::heap()->is_in_reserved(old), "must be in heap.");
assert(!old_gen()->is_in(old), "must be in young generation."); assert(!old_gen()->is_in(old), "must be in young generation.");
objArrayOop obj = objArrayOop(old->forwardee()); objArrayOop obj = objArrayOop(old->forwardee());
@ -205,9 +206,9 @@ bool ParScanThreadState::take_from_overflow_stack() {
for (size_t i = 0; i != num_take_elems; i++) { for (size_t i = 0; i != num_take_elems; i++) {
oop cur = of_stack->pop(); oop cur = of_stack->pop();
oop obj_to_push = cur->forwardee(); oop obj_to_push = cur->forwardee();
assert(GenCollectedHeap::heap()->is_in_reserved(cur), "Should be in heap"); assert(CMSHeap::heap()->is_in_reserved(cur), "Should be in heap");
assert(!old_gen()->is_in_reserved(cur), "Should be in young gen"); assert(!old_gen()->is_in_reserved(cur), "Should be in young gen");
assert(GenCollectedHeap::heap()->is_in_reserved(obj_to_push), "Should be in heap"); assert(CMSHeap::heap()->is_in_reserved(obj_to_push), "Should be in heap");
if (should_be_partially_scanned(obj_to_push, cur)) { if (should_be_partially_scanned(obj_to_push, cur)) {
assert(arrayOop(cur)->length() == 0, "entire array remaining to be scanned"); assert(arrayOop(cur)->length() == 0, "entire array remaining to be scanned");
obj_to_push = cur; obj_to_push = cur;
@ -590,7 +591,7 @@ ParNewGenTask::ParNewGenTask(ParNewGeneration* young_gen,
{} {}
void ParNewGenTask::work(uint worker_id) { void ParNewGenTask::work(uint worker_id) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
// Since this is being done in a separate thread, need new resource // Since this is being done in a separate thread, need new resource
// and handle marks. // and handle marks.
ResourceMark rm; ResourceMark rm;
@ -602,10 +603,10 @@ void ParNewGenTask::work(uint worker_id) {
par_scan_state.set_young_old_boundary(_young_old_boundary); par_scan_state.set_young_old_boundary(_young_old_boundary);
CLDScanClosure cld_scan_closure(&par_scan_state.to_space_root_closure(), CLDScanClosure cld_scan_closure(&par_scan_state.to_space_root_closure(),
gch->rem_set()->cld_rem_set()->accumulate_modified_oops()); heap->rem_set()->cld_rem_set()->accumulate_modified_oops());
par_scan_state.start_strong_roots(); par_scan_state.start_strong_roots();
gch->young_process_roots(_strong_roots_scope, heap->young_process_roots(_strong_roots_scope,
&par_scan_state.to_space_root_closure(), &par_scan_state.to_space_root_closure(),
&par_scan_state.older_gen_closure(), &par_scan_state.older_gen_closure(),
&cld_scan_closure); &cld_scan_closure);
@ -687,7 +688,7 @@ void /*ParNewGeneration::*/ParKeepAliveClosure::do_oop_work(T* p) {
_par_cl->do_oop_nv(p); _par_cl->do_oop_nv(p);
if (GenCollectedHeap::heap()->is_in_reserved(p)) { if (CMSHeap::heap()->is_in_reserved(p)) {
oop obj = oopDesc::load_decode_heap_oop_not_null(p); oop obj = oopDesc::load_decode_heap_oop_not_null(p);
_rs->write_ref_field_gc_par(p, obj); _rs->write_ref_field_gc_par(p, obj);
} }
@ -714,7 +715,7 @@ void /*ParNewGeneration::*/KeepAliveClosure::do_oop_work(T* p) {
_cl->do_oop_nv(p); _cl->do_oop_nv(p);
if (GenCollectedHeap::heap()->is_in_reserved(p)) { if (CMSHeap::heap()->is_in_reserved(p)) {
oop obj = oopDesc::load_decode_heap_oop_not_null(p); oop obj = oopDesc::load_decode_heap_oop_not_null(p);
_rs->write_ref_field_gc_par(p, obj); _rs->write_ref_field_gc_par(p, obj);
} }
@ -804,7 +805,7 @@ public:
}; };
void ParNewRefProcTaskExecutor::execute(ProcessTask& task) { void ParNewRefProcTaskExecutor::execute(ProcessTask& task) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* gch = CMSHeap::heap();
WorkGang* workers = gch->workers(); WorkGang* workers = gch->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
_state_set.reset(workers->active_workers(), _young_gen.promotion_failed()); _state_set.reset(workers->active_workers(), _young_gen.promotion_failed());
@ -816,7 +817,7 @@ void ParNewRefProcTaskExecutor::execute(ProcessTask& task) {
} }
void ParNewRefProcTaskExecutor::execute(EnqueueTask& task) { void ParNewRefProcTaskExecutor::execute(EnqueueTask& task) {
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* gch = CMSHeap::heap();
WorkGang* workers = gch->workers(); WorkGang* workers = gch->workers();
assert(workers != NULL, "Need parallel worker threads."); assert(workers != NULL, "Need parallel worker threads.");
ParNewRefEnqueueTaskProxy enq_task(task); ParNewRefEnqueueTaskProxy enq_task(task);
@ -825,8 +826,8 @@ void ParNewRefProcTaskExecutor::execute(EnqueueTask& task) {
void ParNewRefProcTaskExecutor::set_single_threaded_mode() { void ParNewRefProcTaskExecutor::set_single_threaded_mode() {
_state_set.flush(); _state_set.flush();
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
gch->save_marks(); heap->save_marks();
} }
ScanClosureWithParBarrier:: ScanClosureWithParBarrier::
@ -835,10 +836,10 @@ ScanClosureWithParBarrier(ParNewGeneration* g, bool gc_barrier) :
{ } { }
EvacuateFollowersClosureGeneral:: EvacuateFollowersClosureGeneral::
EvacuateFollowersClosureGeneral(GenCollectedHeap* gch, EvacuateFollowersClosureGeneral(CMSHeap* heap,
OopsInGenClosure* cur, OopsInGenClosure* cur,
OopsInGenClosure* older) : OopsInGenClosure* older) :
_gch(gch), _heap(heap),
_scan_cur_or_nonheap(cur), _scan_older(older) _scan_cur_or_nonheap(cur), _scan_older(older)
{ } { }
@ -846,15 +847,15 @@ void EvacuateFollowersClosureGeneral::do_void() {
do { do {
// Beware: this call will lead to closure applications via virtual // Beware: this call will lead to closure applications via virtual
// calls. // calls.
_gch->oop_since_save_marks_iterate(GenCollectedHeap::YoungGen, _heap->oop_since_save_marks_iterate(GenCollectedHeap::YoungGen,
_scan_cur_or_nonheap, _scan_cur_or_nonheap,
_scan_older); _scan_older);
} while (!_gch->no_allocs_since_save_marks()); } while (!_heap->no_allocs_since_save_marks());
} }
// A Generation that does parallel young-gen collection. // A Generation that does parallel young-gen collection.
void ParNewGeneration::handle_promotion_failed(GenCollectedHeap* gch, ParScanThreadStateSet& thread_state_set) { void ParNewGeneration::handle_promotion_failed(CMSHeap* gch, ParScanThreadStateSet& thread_state_set) {
assert(_promo_failure_scan_stack.is_empty(), "post condition"); assert(_promo_failure_scan_stack.is_empty(), "post condition");
_promo_failure_scan_stack.clear(true); // Clear cached segments. _promo_failure_scan_stack.clear(true); // Clear cached segments.
@ -883,7 +884,7 @@ void ParNewGeneration::collect(bool full,
bool is_tlab) { bool is_tlab) {
assert(full || size > 0, "otherwise we don't want to collect"); assert(full || size > 0, "otherwise we don't want to collect");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* gch = CMSHeap::heap();
_gc_timer->register_gc_start(); _gc_timer->register_gc_start();
@ -1064,7 +1065,7 @@ void ParNewGeneration::collect(bool full,
} }
size_t ParNewGeneration::desired_plab_sz() { size_t ParNewGeneration::desired_plab_sz() {
return _plab_stats.desired_plab_sz(GenCollectedHeap::heap()->workers()->active_workers()); return _plab_stats.desired_plab_sz(CMSHeap::heap()->workers()->active_workers());
} }
static int sum; static int sum;
@ -1168,7 +1169,7 @@ oop ParNewGeneration::copy_to_survivor_space(ParScanThreadState* par_scan_state,
} else { } else {
// Is in to-space; do copying ourselves. // Is in to-space; do copying ourselves.
Copy::aligned_disjoint_words((HeapWord*)old, (HeapWord*)new_obj, sz); Copy::aligned_disjoint_words((HeapWord*)old, (HeapWord*)new_obj, sz);
assert(GenCollectedHeap::heap()->is_in_reserved(new_obj), "illegal forwarding pointer value."); assert(CMSHeap::heap()->is_in_reserved(new_obj), "illegal forwarding pointer value.");
forward_ptr = old->forward_to_atomic(new_obj); forward_ptr = old->forward_to_atomic(new_obj);
// Restore the mark word copied above. // Restore the mark word copied above.
new_obj->set_mark(m); new_obj->set_mark(m);
@ -1475,3 +1476,9 @@ void ParNewGeneration::ref_processor_init() {
const char* ParNewGeneration::name() const { const char* ParNewGeneration::name() const {
return "par new generation"; return "par new generation";
} }
void ParNewGeneration::restore_preserved_marks() {
SharedRestorePreservedMarksTaskExecutor task_executor(CMSHeap::heap()->workers());
_preserved_marks_set.restore(&task_executor);
}

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -35,6 +35,7 @@
#include "memory/padded.hpp" #include "memory/padded.hpp"
class ChunkArray; class ChunkArray;
class CMSHeap;
class ParScanWithoutBarrierClosure; class ParScanWithoutBarrierClosure;
class ParScanWithBarrierClosure; class ParScanWithBarrierClosure;
class ParRootScanWithoutBarrierClosure; class ParRootScanWithoutBarrierClosure;
@ -259,11 +260,11 @@ class KeepAliveClosure: public DefNewGeneration::KeepAliveClosure {
class EvacuateFollowersClosureGeneral: public VoidClosure { class EvacuateFollowersClosureGeneral: public VoidClosure {
private: private:
GenCollectedHeap* _gch; CMSHeap* _heap;
OopsInGenClosure* _scan_cur_or_nonheap; OopsInGenClosure* _scan_cur_or_nonheap;
OopsInGenClosure* _scan_older; OopsInGenClosure* _scan_older;
public: public:
EvacuateFollowersClosureGeneral(GenCollectedHeap* gch, EvacuateFollowersClosureGeneral(CMSHeap* heap,
OopsInGenClosure* cur, OopsInGenClosure* cur,
OopsInGenClosure* older); OopsInGenClosure* older);
virtual void do_void(); virtual void do_void();
@ -336,7 +337,7 @@ class ParNewGeneration: public DefNewGeneration {
static oop real_forwardee_slow(oop obj); static oop real_forwardee_slow(oop obj);
static void waste_some_time(); static void waste_some_time();
void handle_promotion_failed(GenCollectedHeap* gch, ParScanThreadStateSet& thread_state_set); void handle_promotion_failed(CMSHeap* gch, ParScanThreadStateSet& thread_state_set);
protected: protected:
@ -345,6 +346,8 @@ class ParNewGeneration: public DefNewGeneration {
bool survivor_overflow() { return _survivor_overflow; } bool survivor_overflow() { return _survivor_overflow; }
void set_survivor_overflow(bool v) { _survivor_overflow = v; } void set_survivor_overflow(bool v) { _survivor_overflow = v; }
void restore_preserved_marks();
public: public:
ParNewGeneration(ReservedSpace rs, size_t initial_byte_size); ParNewGeneration(ReservedSpace rs, size_t initial_byte_size);

View File

@ -25,10 +25,10 @@
#ifndef SHARE_VM_GC_CMS_PAROOPCLOSURES_INLINE_HPP #ifndef SHARE_VM_GC_CMS_PAROOPCLOSURES_INLINE_HPP
#define SHARE_VM_GC_CMS_PAROOPCLOSURES_INLINE_HPP #define SHARE_VM_GC_CMS_PAROOPCLOSURES_INLINE_HPP
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/parNewGeneration.hpp" #include "gc/cms/parNewGeneration.hpp"
#include "gc/cms/parOopClosures.hpp" #include "gc/cms/parOopClosures.hpp"
#include "gc/shared/cardTableRS.hpp" #include "gc/shared/cardTableRS.hpp"
#include "gc/shared/genCollectedHeap.hpp"
#include "gc/shared/genOopClosures.inline.hpp" #include "gc/shared/genOopClosures.inline.hpp"
#include "logging/log.hpp" #include "logging/log.hpp"
#include "logging/logStream.hpp" #include "logging/logStream.hpp"
@ -72,9 +72,9 @@ template <class T>
inline void ParScanClosure::do_oop_work(T* p, inline void ParScanClosure::do_oop_work(T* p,
bool gc_barrier, bool gc_barrier,
bool root_scan) { bool root_scan) {
assert((!GenCollectedHeap::heap()->is_in_reserved(p) || assert((!CMSHeap::heap()->is_in_reserved(p) ||
generation()->is_in_reserved(p)) generation()->is_in_reserved(p))
&& (GenCollectedHeap::heap()->is_young_gen(generation()) || gc_barrier), && (CMSHeap::heap()->is_young_gen(generation()) || gc_barrier),
"The gen must be right, and we must be doing the barrier " "The gen must be right, and we must be doing the barrier "
"in older generations."); "in older generations.");
T heap_oop = oopDesc::load_heap_oop(p); T heap_oop = oopDesc::load_heap_oop(p);
@ -85,8 +85,8 @@ inline void ParScanClosure::do_oop_work(T* p,
if (_g->to()->is_in_reserved(obj)) { if (_g->to()->is_in_reserved(obj)) {
Log(gc) log; Log(gc) log;
log.error("Scanning field (" PTR_FORMAT ") twice?", p2i(p)); log.error("Scanning field (" PTR_FORMAT ") twice?", p2i(p));
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
Space* sp = gch->space_containing(p); Space* sp = heap->space_containing(p);
oop obj = oop(sp->block_start(p)); oop obj = oop(sp->block_start(p));
assert((HeapWord*)obj < (HeapWord*)p, "Error"); assert((HeapWord*)obj < (HeapWord*)p, "Error");
log.error("Object: " PTR_FORMAT, p2i((void *)obj)); log.error("Object: " PTR_FORMAT, p2i((void *)obj));
@ -96,7 +96,7 @@ inline void ParScanClosure::do_oop_work(T* p,
log.error("-----"); log.error("-----");
log.error("Heap:"); log.error("Heap:");
log.error("-----"); log.error("-----");
gch->print_on(&ls); heap->print_on(&ls);
ShouldNotReachHere(); ShouldNotReachHere();
} }
#endif #endif

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,6 +23,7 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/cms/concurrentMarkSweepGeneration.inline.hpp" #include "gc/cms/concurrentMarkSweepGeneration.inline.hpp"
#include "gc/cms/concurrentMarkSweepThread.hpp" #include "gc/cms/concurrentMarkSweepThread.hpp"
#include "gc/cms/vmCMSOperations.hpp" #include "gc/cms/vmCMSOperations.hpp"
@ -39,19 +40,19 @@
////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////
void VM_CMS_Operation::verify_before_gc() { void VM_CMS_Operation::verify_before_gc() {
if (VerifyBeforeGC && if (VerifyBeforeGC &&
GenCollectedHeap::heap()->total_collections() >= VerifyGCStartAt) { CMSHeap::heap()->total_collections() >= VerifyGCStartAt) {
GCTraceTime(Info, gc, phases, verify) tm("Verify Before", _collector->_gc_timer_cm); GCTraceTime(Info, gc, phases, verify) tm("Verify Before", _collector->_gc_timer_cm);
HandleMark hm; HandleMark hm;
FreelistLocker x(_collector); FreelistLocker x(_collector);
MutexLockerEx y(_collector->bitMapLock(), Mutex::_no_safepoint_check_flag); MutexLockerEx y(_collector->bitMapLock(), Mutex::_no_safepoint_check_flag);
GenCollectedHeap::heap()->prepare_for_verify(); CMSHeap::heap()->prepare_for_verify();
Universe::verify(); Universe::verify();
} }
} }
void VM_CMS_Operation::verify_after_gc() { void VM_CMS_Operation::verify_after_gc() {
if (VerifyAfterGC && if (VerifyAfterGC &&
GenCollectedHeap::heap()->total_collections() >= VerifyGCStartAt) { CMSHeap::heap()->total_collections() >= VerifyGCStartAt) {
GCTraceTime(Info, gc, phases, verify) tm("Verify After", _collector->_gc_timer_cm); GCTraceTime(Info, gc, phases, verify) tm("Verify After", _collector->_gc_timer_cm);
HandleMark hm; HandleMark hm;
FreelistLocker x(_collector); FreelistLocker x(_collector);
@ -112,13 +113,13 @@ void VM_CMS_Initial_Mark::doit() {
_collector->_gc_timer_cm->register_gc_pause_start("Initial Mark"); _collector->_gc_timer_cm->register_gc_pause_start("Initial Mark");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
GCCauseSetter gccs(gch, GCCause::_cms_initial_mark); GCCauseSetter gccs(heap, GCCause::_cms_initial_mark);
VM_CMS_Operation::verify_before_gc(); VM_CMS_Operation::verify_before_gc();
IsGCActiveMark x; // stop-world GC active IsGCActiveMark x; // stop-world GC active
_collector->do_CMS_operation(CMSCollector::CMS_op_checkpointRootsInitial, gch->gc_cause()); _collector->do_CMS_operation(CMSCollector::CMS_op_checkpointRootsInitial, heap->gc_cause());
VM_CMS_Operation::verify_after_gc(); VM_CMS_Operation::verify_after_gc();
@ -140,13 +141,13 @@ void VM_CMS_Final_Remark::doit() {
_collector->_gc_timer_cm->register_gc_pause_start("Final Mark"); _collector->_gc_timer_cm->register_gc_pause_start("Final Mark");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
GCCauseSetter gccs(gch, GCCause::_cms_final_remark); GCCauseSetter gccs(heap, GCCause::_cms_final_remark);
VM_CMS_Operation::verify_before_gc(); VM_CMS_Operation::verify_before_gc();
IsGCActiveMark x; // stop-world GC active IsGCActiveMark x; // stop-world GC active
_collector->do_CMS_operation(CMSCollector::CMS_op_checkpointRootsFinal, gch->gc_cause()); _collector->do_CMS_operation(CMSCollector::CMS_op_checkpointRootsFinal, heap->gc_cause());
VM_CMS_Operation::verify_after_gc(); VM_CMS_Operation::verify_after_gc();
@ -162,8 +163,8 @@ void VM_GenCollectFullConcurrent::doit() {
assert(Thread::current()->is_VM_thread(), "Should be VM thread"); assert(Thread::current()->is_VM_thread(), "Should be VM thread");
assert(GCLockerInvokesConcurrent || ExplicitGCInvokesConcurrent, "Unexpected"); assert(GCLockerInvokesConcurrent || ExplicitGCInvokesConcurrent, "Unexpected");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
if (_gc_count_before == gch->total_collections()) { if (_gc_count_before == heap->total_collections()) {
// The "full" of do_full_collection call below "forces" // The "full" of do_full_collection call below "forces"
// a collection; the second arg, 0, below ensures that // a collection; the second arg, 0, below ensures that
// only the young gen is collected. XXX In the future, // only the young gen is collected. XXX In the future,
@ -173,21 +174,21 @@ void VM_GenCollectFullConcurrent::doit() {
// for the future. // for the future.
assert(SafepointSynchronize::is_at_safepoint(), assert(SafepointSynchronize::is_at_safepoint(),
"We can only be executing this arm of if at a safepoint"); "We can only be executing this arm of if at a safepoint");
GCCauseSetter gccs(gch, _gc_cause); GCCauseSetter gccs(heap, _gc_cause);
gch->do_full_collection(gch->must_clear_all_soft_refs(), GenCollectedHeap::YoungGen); heap->do_full_collection(heap->must_clear_all_soft_refs(), GenCollectedHeap::YoungGen);
} // Else no need for a foreground young gc } // Else no need for a foreground young gc
assert((_gc_count_before < gch->total_collections()) || assert((_gc_count_before < heap->total_collections()) ||
(GCLocker::is_active() /* gc may have been skipped */ (GCLocker::is_active() /* gc may have been skipped */
&& (_gc_count_before == gch->total_collections())), && (_gc_count_before == heap->total_collections())),
"total_collections() should be monotonically increasing"); "total_collections() should be monotonically increasing");
MutexLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
assert(_full_gc_count_before <= gch->total_full_collections(), "Error"); assert(_full_gc_count_before <= heap->total_full_collections(), "Error");
if (gch->total_full_collections() == _full_gc_count_before) { if (heap->total_full_collections() == _full_gc_count_before) {
// Nudge the CMS thread to start a concurrent collection. // Nudge the CMS thread to start a concurrent collection.
CMSCollector::request_full_gc(_full_gc_count_before, _gc_cause); CMSCollector::request_full_gc(_full_gc_count_before, _gc_cause);
} else { } else {
assert(_full_gc_count_before < gch->total_full_collections(), "Error"); assert(_full_gc_count_before < heap->total_full_collections(), "Error");
FullGCCount_lock->notify_all(); // Inform the Java thread its work is done FullGCCount_lock->notify_all(); // Inform the Java thread its work is done
} }
} }
@ -197,11 +198,11 @@ bool VM_GenCollectFullConcurrent::evaluate_at_safepoint() const {
assert(thr != NULL, "Unexpected tid"); assert(thr != NULL, "Unexpected tid");
if (!thr->is_Java_thread()) { if (!thr->is_Java_thread()) {
assert(thr->is_VM_thread(), "Expected to be evaluated by VM thread"); assert(thr->is_VM_thread(), "Expected to be evaluated by VM thread");
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
if (_gc_count_before != gch->total_collections()) { if (_gc_count_before != heap->total_collections()) {
// No need to do a young gc, we'll just nudge the CMS thread // No need to do a young gc, we'll just nudge the CMS thread
// in the doit() method above, to be executed soon. // in the doit() method above, to be executed soon.
assert(_gc_count_before < gch->total_collections(), assert(_gc_count_before < heap->total_collections(),
"total_collections() should be monotonically increasing"); "total_collections() should be monotonically increasing");
return false; // no need for foreground young gc return false; // no need for foreground young gc
} }
@ -227,9 +228,9 @@ void VM_GenCollectFullConcurrent::doit_epilogue() {
// count overflows and wraps around. XXX fix me !!! // count overflows and wraps around. XXX fix me !!!
// e.g. at the rate of 1 full gc per ms, this could // e.g. at the rate of 1 full gc per ms, this could
// overflow in about 1000 years. // overflow in about 1000 years.
GenCollectedHeap* gch = GenCollectedHeap::heap(); CMSHeap* heap = CMSHeap::heap();
if (_gc_cause != GCCause::_gc_locker && if (_gc_cause != GCCause::_gc_locker &&
gch->total_full_collections_completed() <= _full_gc_count_before) { heap->total_full_collections_completed() <= _full_gc_count_before) {
// maybe we should change the condition to test _gc_cause == // maybe we should change the condition to test _gc_cause ==
// GCCause::_java_lang_system_gc or GCCause::_dcmd_gc_run, // GCCause::_java_lang_system_gc or GCCause::_dcmd_gc_run,
// instead of _gc_cause != GCCause::_gc_locker // instead of _gc_cause != GCCause::_gc_locker
@ -245,7 +246,7 @@ void VM_GenCollectFullConcurrent::doit_epilogue() {
MutexLockerEx ml(FullGCCount_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx ml(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
// Either a concurrent or a stop-world full gc is sufficient // Either a concurrent or a stop-world full gc is sufficient
// witness to our request. // witness to our request.
while (gch->total_full_collections_completed() <= _full_gc_count_before) { while (heap->total_full_collections_completed() <= _full_gc_count_before) {
FullGCCount_lock->wait(Mutex::_no_safepoint_check_flag); FullGCCount_lock->wait(Mutex::_no_safepoint_check_flag);
} }
} }

View File

@ -734,8 +734,11 @@ void DefNewGeneration::remove_forwarding_pointers() {
RemoveForwardedPointerClosure rspc; RemoveForwardedPointerClosure rspc;
eden()->object_iterate(&rspc); eden()->object_iterate(&rspc);
from()->object_iterate(&rspc); from()->object_iterate(&rspc);
restore_preserved_marks();
}
SharedRestorePreservedMarksTaskExecutor task_executor(GenCollectedHeap::heap()->workers()); void DefNewGeneration::restore_preserved_marks() {
SharedRestorePreservedMarksTaskExecutor task_executor(NULL);
_preserved_marks_set.restore(&task_executor); _preserved_marks_set.restore(&task_executor);
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -89,6 +89,8 @@ protected:
// therefore we must remove their forwarding pointers. // therefore we must remove their forwarding pointers.
void remove_forwarding_pointers(); void remove_forwarding_pointers();
virtual void restore_preserved_marks();
// Preserved marks // Preserved marks
PreservedMarksSet _preserved_marks_set; PreservedMarksSet _preserved_marks_set;

View File

@ -83,6 +83,7 @@ class GCHeapLog : public EventLogBase<GCMessage> {
// GenCollectedHeap // GenCollectedHeap
// G1CollectedHeap // G1CollectedHeap
// ParallelScavengeHeap // ParallelScavengeHeap
// CMSHeap
// //
class CollectedHeap : public CHeapObj<mtInternal> { class CollectedHeap : public CHeapObj<mtInternal> {
friend class VMStructs; friend class VMStructs;
@ -194,7 +195,8 @@ class CollectedHeap : public CHeapObj<mtInternal> {
enum Name { enum Name {
GenCollectedHeap, GenCollectedHeap,
ParallelScavengeHeap, ParallelScavengeHeap,
G1CollectedHeap G1CollectedHeap,
CMSHeap
}; };
static inline size_t filler_array_max_size() { static inline size_t filler_array_max_size() {

View File

@ -58,28 +58,6 @@
#include "utilities/macros.hpp" #include "utilities/macros.hpp"
#include "utilities/stack.inline.hpp" #include "utilities/stack.inline.hpp"
#include "utilities/vmError.hpp" #include "utilities/vmError.hpp"
#if INCLUDE_ALL_GCS
#include "gc/cms/concurrentMarkSweepThread.hpp"
#include "gc/cms/vmCMSOperations.hpp"
#endif // INCLUDE_ALL_GCS
NOT_PRODUCT(size_t GenCollectedHeap::_skip_header_HeapWords = 0;)
// The set of potentially parallel tasks in root scanning.
enum GCH_strong_roots_tasks {
GCH_PS_Universe_oops_do,
GCH_PS_JNIHandles_oops_do,
GCH_PS_ObjectSynchronizer_oops_do,
GCH_PS_Management_oops_do,
GCH_PS_SystemDictionary_oops_do,
GCH_PS_ClassLoaderDataGraph_oops_do,
GCH_PS_jvmti_oops_do,
GCH_PS_CodeCache_oops_do,
GCH_PS_aot_oops_do,
GCH_PS_younger_gens,
// Leave this one last.
GCH_PS_NumElements
};
GenCollectedHeap::GenCollectedHeap(GenCollectorPolicy *policy) : GenCollectedHeap::GenCollectedHeap(GenCollectorPolicy *policy) :
CollectedHeap(), CollectedHeap(),
@ -89,15 +67,6 @@ GenCollectedHeap::GenCollectedHeap(GenCollectorPolicy *policy) :
_full_collections_completed(0) _full_collections_completed(0)
{ {
assert(policy != NULL, "Sanity check"); assert(policy != NULL, "Sanity check");
if (UseConcMarkSweepGC) {
_workers = new WorkGang("GC Thread", ParallelGCThreads,
/* are_GC_task_threads */true,
/* are_ConcurrentGC_threads */false);
_workers->initialize_workers();
} else {
// Serial GC does not use workers.
_workers = NULL;
}
} }
jint GenCollectedHeap::initialize() { jint GenCollectedHeap::initialize() {
@ -138,15 +107,6 @@ jint GenCollectedHeap::initialize() {
_old_gen = gen_policy()->old_gen_spec()->init(old_rs, rem_set()); _old_gen = gen_policy()->old_gen_spec()->init(old_rs, rem_set());
clear_incremental_collection_failed(); clear_incremental_collection_failed();
#if INCLUDE_ALL_GCS
// If we are running CMS, create the collector responsible
// for collecting the CMS generations.
if (collector_policy()->is_concurrent_mark_sweep_policy()) {
bool success = create_cms_collector();
if (!success) return JNI_ENOMEM;
}
#endif // INCLUDE_ALL_GCS
return JNI_OK; return JNI_OK;
} }
@ -183,21 +143,22 @@ char* GenCollectedHeap::allocate(size_t alignment,
void GenCollectedHeap::post_initialize() { void GenCollectedHeap::post_initialize() {
ref_processing_init(); ref_processing_init();
assert((_young_gen->kind() == Generation::DefNew) || check_gen_kinds();
(_young_gen->kind() == Generation::ParNew),
"Wrong youngest generation type");
DefNewGeneration* def_new_gen = (DefNewGeneration*)_young_gen; DefNewGeneration* def_new_gen = (DefNewGeneration*)_young_gen;
assert(_old_gen->kind() == Generation::ConcurrentMarkSweep ||
_old_gen->kind() == Generation::MarkSweepCompact,
"Wrong generation kind");
_gen_policy->initialize_size_policy(def_new_gen->eden()->capacity(), _gen_policy->initialize_size_policy(def_new_gen->eden()->capacity(),
_old_gen->capacity(), _old_gen->capacity(),
def_new_gen->from()->capacity()); def_new_gen->from()->capacity());
_gen_policy->initialize_gc_policy_counters(); _gen_policy->initialize_gc_policy_counters();
} }
void GenCollectedHeap::check_gen_kinds() {
assert(young_gen()->kind() == Generation::DefNew,
"Wrong youngest generation type");
assert(old_gen()->kind() == Generation::MarkSweepCompact,
"Wrong generation kind");
}
void GenCollectedHeap::ref_processing_init() { void GenCollectedHeap::ref_processing_init() {
_young_gen->ref_processor_init(); _young_gen->ref_processor_init();
_old_gen->ref_processor_init(); _old_gen->ref_processor_init();
@ -309,19 +270,6 @@ bool GenCollectedHeap::must_clear_all_soft_refs() {
_gc_cause == GCCause::_wb_full_gc; _gc_cause == GCCause::_wb_full_gc;
} }
bool GenCollectedHeap::should_do_concurrent_full_gc(GCCause::Cause cause) {
if (!UseConcMarkSweepGC) {
return false;
}
switch (cause) {
case GCCause::_gc_locker: return GCLockerInvokesConcurrent;
case GCCause::_java_lang_system_gc:
case GCCause::_dcmd_gc_run: return ExplicitGCInvokesConcurrent;
default: return false;
}
}
void GenCollectedHeap::collect_generation(Generation* gen, bool full, size_t size, void GenCollectedHeap::collect_generation(Generation* gen, bool full, size_t size,
bool is_tlab, bool run_verification, bool clear_soft_refs, bool is_tlab, bool run_verification, bool clear_soft_refs,
bool restore_marks_for_biased_locking) { bool restore_marks_for_biased_locking) {
@ -674,31 +622,6 @@ void GenCollectedHeap::young_process_roots(StrongRootsScope* scope,
_process_strong_tasks->all_tasks_completed(scope->n_threads()); _process_strong_tasks->all_tasks_completed(scope->n_threads());
} }
void GenCollectedHeap::cms_process_roots(StrongRootsScope* scope,
bool young_gen_as_roots,
ScanningOption so,
bool only_strong_roots,
OopsInGenClosure* root_closure,
CLDClosure* cld_closure) {
MarkingCodeBlobClosure mark_code_closure(root_closure, !CodeBlobToOopClosure::FixRelocations);
OopsInGenClosure* weak_roots = only_strong_roots ? NULL : root_closure;
CLDClosure* weak_cld_closure = only_strong_roots ? NULL : cld_closure;
process_roots(scope, so, root_closure, weak_roots, cld_closure, weak_cld_closure, &mark_code_closure);
if (!only_strong_roots) {
process_string_table_roots(scope, root_closure);
}
if (young_gen_as_roots &&
!_process_strong_tasks->is_task_claimed(GCH_PS_younger_gens)) {
root_closure->set_generation(_young_gen);
_young_gen->oop_iterate(root_closure);
root_closure->reset_generation();
}
_process_strong_tasks->all_tasks_completed(scope->n_threads());
}
void GenCollectedHeap::full_process_roots(StrongRootsScope* scope, void GenCollectedHeap::full_process_roots(StrongRootsScope* scope,
bool is_adjust_phase, bool is_adjust_phase,
ScanningOption so, ScanningOption so,
@ -763,14 +686,7 @@ HeapWord** GenCollectedHeap::end_addr() const {
// public collection interfaces // public collection interfaces
void GenCollectedHeap::collect(GCCause::Cause cause) { void GenCollectedHeap::collect(GCCause::Cause cause) {
if (should_do_concurrent_full_gc(cause)) { if (cause == GCCause::_wb_young_gc) {
#if INCLUDE_ALL_GCS
// Mostly concurrent full collection.
collect_mostly_concurrent(cause);
#else // INCLUDE_ALL_GCS
ShouldNotReachHere();
#endif // INCLUDE_ALL_GCS
} else if (cause == GCCause::_wb_young_gc) {
// Young collection for the WhiteBox API. // Young collection for the WhiteBox API.
collect(cause, YoungGen); collect(cause, YoungGen);
} else { } else {
@ -817,44 +733,6 @@ void GenCollectedHeap::collect_locked(GCCause::Cause cause, GenerationType max_g
} }
} }
#if INCLUDE_ALL_GCS
bool GenCollectedHeap::create_cms_collector() {
assert(_old_gen->kind() == Generation::ConcurrentMarkSweep,
"Unexpected generation kinds");
// Skip two header words in the block content verification
NOT_PRODUCT(_skip_header_HeapWords = CMSCollector::skip_header_HeapWords();)
assert(_gen_policy->is_concurrent_mark_sweep_policy(), "Unexpected policy type");
CMSCollector* collector =
new CMSCollector((ConcurrentMarkSweepGeneration*)_old_gen,
_rem_set,
_gen_policy->as_concurrent_mark_sweep_policy());
if (collector == NULL || !collector->completed_initialization()) {
if (collector) {
delete collector; // Be nice in embedded situation
}
vm_shutdown_during_initialization("Could not create CMS collector");
return false;
}
return true; // success
}
void GenCollectedHeap::collect_mostly_concurrent(GCCause::Cause cause) {
assert(!Heap_lock->owned_by_self(), "Should not own Heap_lock");
MutexLocker ml(Heap_lock);
// Read the GC counts while holding the Heap_lock
unsigned int full_gc_count_before = total_full_collections();
unsigned int gc_count_before = total_collections();
{
MutexUnlocker mu(Heap_lock);
VM_GenCollectFullConcurrent op(gc_count_before, full_gc_count_before, cause);
VMThread::execute(&op);
}
}
#endif // INCLUDE_ALL_GCS
void GenCollectedHeap::do_full_collection(bool clear_all_soft_refs) { void GenCollectedHeap::do_full_collection(bool clear_all_soft_refs) {
do_full_collection(clear_all_soft_refs, OldGen); do_full_collection(clear_all_soft_refs, OldGen);
} }
@ -1097,8 +975,9 @@ void GenCollectedHeap::save_marks() {
GenCollectedHeap* GenCollectedHeap::heap() { GenCollectedHeap* GenCollectedHeap::heap() {
CollectedHeap* heap = Universe::heap(); CollectedHeap* heap = Universe::heap();
assert(heap != NULL, "Uninitialized access to GenCollectedHeap::heap()"); assert(heap != NULL, "Uninitialized access to GenCollectedHeap::heap()");
assert(heap->kind() == CollectedHeap::GenCollectedHeap, "Not a GenCollectedHeap"); assert(heap->kind() == CollectedHeap::GenCollectedHeap ||
return (GenCollectedHeap*)heap; heap->kind() == CollectedHeap::CMSHeap, "Not a GenCollectedHeap");
return (GenCollectedHeap*) heap;
} }
void GenCollectedHeap::prepare_for_compaction() { void GenCollectedHeap::prepare_for_compaction() {
@ -1126,34 +1005,9 @@ void GenCollectedHeap::print_on(outputStream* st) const {
} }
void GenCollectedHeap::gc_threads_do(ThreadClosure* tc) const { void GenCollectedHeap::gc_threads_do(ThreadClosure* tc) const {
if (workers() != NULL) {
workers()->threads_do(tc);
}
#if INCLUDE_ALL_GCS
if (UseConcMarkSweepGC) {
ConcurrentMarkSweepThread::threads_do(tc);
}
#endif // INCLUDE_ALL_GCS
} }
void GenCollectedHeap::print_gc_threads_on(outputStream* st) const { void GenCollectedHeap::print_gc_threads_on(outputStream* st) const {
#if INCLUDE_ALL_GCS
if (UseConcMarkSweepGC) {
workers()->print_worker_threads_on(st);
ConcurrentMarkSweepThread::print_all_on(st);
}
#endif // INCLUDE_ALL_GCS
}
void GenCollectedHeap::print_on_error(outputStream* st) const {
this->CollectedHeap::print_on_error(st);
#if INCLUDE_ALL_GCS
if (UseConcMarkSweepGC) {
st->cr();
CMSCollector::print_on_error(st);
}
#endif // INCLUDE_ALL_GCS
} }
void GenCollectedHeap::print_tracing_info() const { void GenCollectedHeap::print_tracing_info() const {
@ -1184,7 +1038,6 @@ class GenGCPrologueClosure: public GenCollectedHeap::GenClosure {
void GenCollectedHeap::gc_prologue(bool full) { void GenCollectedHeap::gc_prologue(bool full) {
assert(InlineCacheBuffer::is_empty(), "should have cleaned up ICBuffer"); assert(InlineCacheBuffer::is_empty(), "should have cleaned up ICBuffer");
always_do_update_barrier = false;
// Fill TLAB's and such // Fill TLAB's and such
CollectedHeap::accumulate_statistics_all_tlabs(); CollectedHeap::accumulate_statistics_all_tlabs();
ensure_parsability(true); // retire TLABs ensure_parsability(true); // retire TLABs
@ -1222,8 +1075,6 @@ void GenCollectedHeap::gc_epilogue(bool full) {
MetaspaceCounters::update_performance_counters(); MetaspaceCounters::update_performance_counters();
CompressedClassSpaceCounters::update_performance_counters(); CompressedClassSpaceCounters::update_performance_counters();
always_do_update_barrier = UseConcMarkSweepGC;
}; };
#ifndef PRODUCT #ifndef PRODUCT
@ -1304,11 +1155,3 @@ jlong GenCollectedHeap::millis_since_last_gc() {
} }
return retVal; return retVal;
} }
void GenCollectedHeap::stop() {
#if INCLUDE_ALL_GCS
if (UseConcMarkSweepGC) {
ConcurrentMarkSweepThread::cmst()->stop();
}
#endif
}

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2000, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -78,21 +78,34 @@ private:
// In support of ExplicitGCInvokesConcurrent functionality // In support of ExplicitGCInvokesConcurrent functionality
unsigned int _full_collections_completed; unsigned int _full_collections_completed;
// Data structure for claiming the (potentially) parallel tasks in
// (gen-specific) roots processing.
SubTasksDone* _process_strong_tasks;
// Collects the given generation. // Collects the given generation.
void collect_generation(Generation* gen, bool full, size_t size, bool is_tlab, void collect_generation(Generation* gen, bool full, size_t size, bool is_tlab,
bool run_verification, bool clear_soft_refs, bool run_verification, bool clear_soft_refs,
bool restore_marks_for_biased_locking); bool restore_marks_for_biased_locking);
// In block contents verification, the number of header words to skip
NOT_PRODUCT(static size_t _skip_header_HeapWords;)
WorkGang* _workers;
protected: protected:
// The set of potentially parallel tasks in root scanning.
enum GCH_strong_roots_tasks {
GCH_PS_Universe_oops_do,
GCH_PS_JNIHandles_oops_do,
GCH_PS_ObjectSynchronizer_oops_do,
GCH_PS_FlatProfiler_oops_do,
GCH_PS_Management_oops_do,
GCH_PS_SystemDictionary_oops_do,
GCH_PS_ClassLoaderDataGraph_oops_do,
GCH_PS_jvmti_oops_do,
GCH_PS_CodeCache_oops_do,
GCH_PS_aot_oops_do,
GCH_PS_younger_gens,
// Leave this one last.
GCH_PS_NumElements
};
// Data structure for claiming the (potentially) parallel tasks in
// (gen-specific) roots processing.
SubTasksDone* _process_strong_tasks;
// Helper functions for allocation // Helper functions for allocation
HeapWord* attempt_allocation(size_t size, HeapWord* attempt_allocation(size_t size,
bool is_tlab, bool is_tlab,
@ -124,8 +137,6 @@ protected:
public: public:
GenCollectedHeap(GenCollectorPolicy *policy); GenCollectedHeap(GenCollectorPolicy *policy);
WorkGang* workers() const { return _workers; }
// Returns JNI_OK on success // Returns JNI_OK on success
virtual jint initialize(); virtual jint initialize();
@ -135,6 +146,8 @@ public:
// Does operations required after initialization has been done. // Does operations required after initialization has been done.
void post_initialize(); void post_initialize();
virtual void check_gen_kinds();
// Initialize ("weak") refs processing support // Initialize ("weak") refs processing support
virtual void ref_processing_init(); virtual void ref_processing_init();
@ -143,11 +156,7 @@ public:
} }
virtual const char* name() const { virtual const char* name() const {
if (UseConcMarkSweepGC) { return "Serial";
return "Concurrent Mark Sweep";
} else {
return "Serial";
}
} }
Generation* young_gen() const { return _young_gen; } Generation* young_gen() const { return _young_gen; }
@ -190,7 +199,7 @@ public:
// Perform a full collection of the heap; intended for use in implementing // Perform a full collection of the heap; intended for use in implementing
// "System.gc". This implies as full a collection as the CollectedHeap // "System.gc". This implies as full a collection as the CollectedHeap
// supports. Caller does not hold the Heap_lock on entry. // supports. Caller does not hold the Heap_lock on entry.
void collect(GCCause::Cause cause); virtual void collect(GCCause::Cause cause);
// The same as above but assume that the caller holds the Heap_lock. // The same as above but assume that the caller holds the Heap_lock.
void collect_locked(GCCause::Cause cause); void collect_locked(GCCause::Cause cause);
@ -207,12 +216,8 @@ public:
bool is_in(const void* p) const; bool is_in(const void* p) const;
// override // override
bool is_in_closed_subset(const void* p) const { virtual bool is_in_closed_subset(const void* p) const {
if (UseConcMarkSweepGC) { return is_in(p);
return is_in_reserved(p);
} else {
return is_in(p);
}
} }
// Returns true if the reference is to an object in the reserved space // Returns true if the reference is to an object in the reserved space
@ -278,7 +283,7 @@ public:
} }
virtual bool card_mark_must_follow_store() const { virtual bool card_mark_must_follow_store() const {
return UseConcMarkSweepGC; return false;
} }
// We don't need barriers for stores to objects in the // We don't need barriers for stores to objects in the
@ -344,7 +349,6 @@ public:
virtual void print_gc_threads_on(outputStream* st) const; virtual void print_gc_threads_on(outputStream* st) const;
virtual void gc_threads_do(ThreadClosure* tc) const; virtual void gc_threads_do(ThreadClosure* tc) const;
virtual void print_tracing_info() const; virtual void print_tracing_info() const;
virtual void print_on_error(outputStream* st) const;
void print_heap_change(size_t young_prev_used, size_t old_prev_used) const; void print_heap_change(size_t young_prev_used, size_t old_prev_used) const;
@ -383,7 +387,7 @@ public:
SO_ScavengeCodeCache = 0x10 SO_ScavengeCodeCache = 0x10
}; };
private: protected:
void process_roots(StrongRootsScope* scope, void process_roots(StrongRootsScope* scope,
ScanningOption so, ScanningOption so,
OopClosure* strong_roots, OopClosure* strong_roots,
@ -395,24 +399,20 @@ public:
void process_string_table_roots(StrongRootsScope* scope, void process_string_table_roots(StrongRootsScope* scope,
OopClosure* root_closure); OopClosure* root_closure);
// Accessor for memory state verification support
NOT_PRODUCT(
virtual size_t skip_header_HeapWords() { return 0; }
)
virtual void gc_prologue(bool full);
virtual void gc_epilogue(bool full);
public: public:
void young_process_roots(StrongRootsScope* scope, void young_process_roots(StrongRootsScope* scope,
OopsInGenClosure* root_closure, OopsInGenClosure* root_closure,
OopsInGenClosure* old_gen_closure, OopsInGenClosure* old_gen_closure,
CLDClosure* cld_closure); CLDClosure* cld_closure);
// If "young_gen_as_roots" is false, younger generations are
// not scanned as roots; in this case, the caller must be arranging to
// scan the younger generations itself. (For example, a generation might
// explicitly mark reachable objects in younger generations, to avoid
// excess storage retention.)
void cms_process_roots(StrongRootsScope* scope,
bool young_gen_as_roots,
ScanningOption so,
bool only_strong_roots,
OopsInGenClosure* root_closure,
CLDClosure* cld_closure);
void full_process_roots(StrongRootsScope* scope, void full_process_roots(StrongRootsScope* scope,
bool is_adjust_phase, bool is_adjust_phase,
ScanningOption so, ScanningOption so,
@ -479,12 +479,8 @@ public:
oop obj, oop obj,
size_t obj_size); size_t obj_size);
private:
// Accessor for memory state verification support
NOT_PRODUCT(
static size_t skip_header_HeapWords() { return _skip_header_HeapWords; }
)
private:
// Override // Override
void check_for_non_bad_heap_word_value(HeapWord* addr, void check_for_non_bad_heap_word_value(HeapWord* addr,
size_t size) PRODUCT_RETURN; size_t size) PRODUCT_RETURN;
@ -499,22 +495,8 @@ private:
// collect() and collect_locked(). Caller holds the Heap_lock on entry. // collect() and collect_locked(). Caller holds the Heap_lock on entry.
void collect_locked(GCCause::Cause cause, GenerationType max_generation); void collect_locked(GCCause::Cause cause, GenerationType max_generation);
// Returns success or failure.
bool create_cms_collector();
// In support of ExplicitGCInvokesConcurrent functionality
bool should_do_concurrent_full_gc(GCCause::Cause cause);
void collect_mostly_concurrent(GCCause::Cause cause);
// Save the tops of the spaces in all generations // Save the tops of the spaces in all generations
void record_gen_tops_before_GC() PRODUCT_RETURN; void record_gen_tops_before_GC() PRODUCT_RETURN;
protected:
void gc_prologue(bool full);
void gc_epilogue(bool full);
public:
void stop();
}; };
#endif // SHARE_VM_GC_SHARED_GENCOLLECTEDHEAP_HPP #endif // SHARE_VM_GC_SHARED_GENCOLLECTEDHEAP_HPP

View File

@ -84,6 +84,7 @@
#include "utilities/preserveException.hpp" #include "utilities/preserveException.hpp"
#if INCLUDE_ALL_GCS #if INCLUDE_ALL_GCS
#include "gc/cms/cmsCollectorPolicy.hpp" #include "gc/cms/cmsCollectorPolicy.hpp"
#include "gc/cms/cmsHeap.hpp"
#include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp"
#include "gc/g1/g1CollectorPolicy.hpp" #include "gc/g1/g1CollectorPolicy.hpp"
#include "gc/parallel/parallelScavengeHeap.hpp" #include "gc/parallel/parallelScavengeHeap.hpp"
@ -758,7 +759,7 @@ CollectedHeap* Universe::create_heap() {
} else if (UseG1GC) { } else if (UseG1GC) {
return Universe::create_heap_with_policy<G1CollectedHeap, G1CollectorPolicy>(); return Universe::create_heap_with_policy<G1CollectedHeap, G1CollectorPolicy>();
} else if (UseConcMarkSweepGC) { } else if (UseConcMarkSweepGC) {
return Universe::create_heap_with_policy<GenCollectedHeap, ConcurrentMarkSweepPolicy>(); return Universe::create_heap_with_policy<CMSHeap, ConcurrentMarkSweepPolicy>();
#endif #endif
} else if (UseSerialGC) { } else if (UseSerialGC) {
return Universe::create_heap_with_policy<GenCollectedHeap, MarkSweepPolicy>(); return Universe::create_heap_with_policy<GenCollectedHeap, MarkSweepPolicy>();

View File

@ -86,7 +86,8 @@ void GcThreadCountClosure::do_thread(Thread* thread) {
void MemoryService::set_universe_heap(CollectedHeap* heap) { void MemoryService::set_universe_heap(CollectedHeap* heap) {
CollectedHeap::Name kind = heap->kind(); CollectedHeap::Name kind = heap->kind();
switch (kind) { switch (kind) {
case CollectedHeap::GenCollectedHeap : { case CollectedHeap::GenCollectedHeap :
case CollectedHeap::CMSHeap : {
add_gen_collected_heap_info(GenCollectedHeap::heap()); add_gen_collected_heap_info(GenCollectedHeap::heap());
break; break;
} }