8134626: Misc cleanups after generation array removal
Reviewed-by: david, dholmes, tschatzl
This commit is contained in:
parent
c166f75b22
commit
49fb91407d
@ -2989,7 +2989,7 @@ initialize_sequential_subtasks_for_marking(int n_threads,
|
|||||||
assert(task_size > CardTableModRefBS::card_size_in_words &&
|
assert(task_size > CardTableModRefBS::card_size_in_words &&
|
||||||
(task_size % CardTableModRefBS::card_size_in_words == 0),
|
(task_size % CardTableModRefBS::card_size_in_words == 0),
|
||||||
"Otherwise arithmetic below would be incorrect");
|
"Otherwise arithmetic below would be incorrect");
|
||||||
MemRegion span = _gen->reserved();
|
MemRegion span = _old_gen->reserved();
|
||||||
if (low != NULL) {
|
if (low != NULL) {
|
||||||
if (span.contains(low)) {
|
if (span.contains(low)) {
|
||||||
// Align low down to a card boundary so that
|
// Align low down to a card boundary so that
|
||||||
|
@ -99,7 +99,7 @@ class CompactibleFreeListSpace: public CompactibleSpace {
|
|||||||
BlockOffsetArrayNonContigSpace _bt;
|
BlockOffsetArrayNonContigSpace _bt;
|
||||||
|
|
||||||
CMSCollector* _collector;
|
CMSCollector* _collector;
|
||||||
ConcurrentMarkSweepGeneration* _gen;
|
ConcurrentMarkSweepGeneration* _old_gen;
|
||||||
|
|
||||||
// Data structures for free blocks (used during allocation/sweeping)
|
// Data structures for free blocks (used during allocation/sweeping)
|
||||||
|
|
||||||
|
@ -212,7 +212,7 @@ ConcurrentMarkSweepGeneration::ConcurrentMarkSweepGeneration(
|
|||||||
use_adaptive_freelists,
|
use_adaptive_freelists,
|
||||||
dictionaryChoice);
|
dictionaryChoice);
|
||||||
NOT_PRODUCT(debug_cms_space = _cmsSpace;)
|
NOT_PRODUCT(debug_cms_space = _cmsSpace;)
|
||||||
_cmsSpace->_gen = this;
|
_cmsSpace->_old_gen = this;
|
||||||
|
|
||||||
_gc_stats = new CMSGCStats();
|
_gc_stats = new CMSGCStats();
|
||||||
|
|
||||||
@ -359,13 +359,13 @@ double CMSStats::time_until_cms_gen_full() const {
|
|||||||
(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
|
||||||
// for the next minor collection. Use the padded average as
|
// for the next young collection. Use the padded average as
|
||||||
// a safety factor.
|
// a safety factor.
|
||||||
cms_free -= expected_promotion;
|
cms_free -= expected_promotion;
|
||||||
|
|
||||||
// Adjust by the safety factor.
|
// Adjust by the safety factor.
|
||||||
double cms_free_dbl = (double)cms_free;
|
double cms_free_dbl = (double)cms_free;
|
||||||
double cms_adjustment = (100.0 - CMSIncrementalSafetyFactor)/100.0;
|
double cms_adjustment = (100.0 - CMSIncrementalSafetyFactor) / 100.0;
|
||||||
// Apply a further correction factor which tries to adjust
|
// Apply a further correction factor which tries to adjust
|
||||||
// for recent occurance of concurrent mode failures.
|
// for recent occurance of concurrent mode failures.
|
||||||
cms_adjustment = cms_adjustment * cms_free_adjustment_factor(cms_free);
|
cms_adjustment = cms_adjustment * cms_free_adjustment_factor(cms_free);
|
||||||
@ -531,7 +531,7 @@ CMSCollector::CMSCollector(ConcurrentMarkSweepGeneration* cmsGen,
|
|||||||
if (CMSConcurrentMTEnabled) {
|
if (CMSConcurrentMTEnabled) {
|
||||||
if (FLAG_IS_DEFAULT(ConcGCThreads)) {
|
if (FLAG_IS_DEFAULT(ConcGCThreads)) {
|
||||||
// just for now
|
// just for now
|
||||||
FLAG_SET_DEFAULT(ConcGCThreads, (ParallelGCThreads + 3)/4);
|
FLAG_SET_DEFAULT(ConcGCThreads, (ParallelGCThreads + 3) / 4);
|
||||||
}
|
}
|
||||||
if (ConcGCThreads > 1) {
|
if (ConcGCThreads > 1) {
|
||||||
_conc_workers = new YieldingFlexibleWorkGang("CMS Thread",
|
_conc_workers = new YieldingFlexibleWorkGang("CMS Thread",
|
||||||
@ -592,7 +592,7 @@ CMSCollector::CMSCollector(ConcurrentMarkSweepGeneration* cmsGen,
|
|||||||
_cmsGen ->init_initiating_occupancy(CMSInitiatingOccupancyFraction, CMSTriggerRatio);
|
_cmsGen ->init_initiating_occupancy(CMSInitiatingOccupancyFraction, CMSTriggerRatio);
|
||||||
|
|
||||||
// Clip CMSBootstrapOccupancy between 0 and 100.
|
// Clip CMSBootstrapOccupancy between 0 and 100.
|
||||||
_bootstrap_occupancy = ((double)CMSBootstrapOccupancy)/(double)100;
|
_bootstrap_occupancy = CMSBootstrapOccupancy / 100.0;
|
||||||
|
|
||||||
// Now tell CMS generations the identity of their collector
|
// Now tell CMS generations the identity of their collector
|
||||||
ConcurrentMarkSweepGeneration::set_collector(this);
|
ConcurrentMarkSweepGeneration::set_collector(this);
|
||||||
@ -613,7 +613,7 @@ CMSCollector::CMSCollector(ConcurrentMarkSweepGeneration* cmsGen,
|
|||||||
_end_addr = gch->end_addr();
|
_end_addr = gch->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;
|
||||||
_eden_chunk_array = NEW_C_HEAP_ARRAY(HeapWord*, _eden_chunk_capacity, mtGC);
|
_eden_chunk_array = NEW_C_HEAP_ARRAY(HeapWord*, _eden_chunk_capacity, mtGC);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -795,29 +795,22 @@ void ConcurrentMarkSweepGeneration::compute_new_size_free_list() {
|
|||||||
size_t desired_capacity = (size_t)(used() / ((double) 1 - desired_free_percentage));
|
size_t desired_capacity = (size_t)(used() / ((double) 1 - desired_free_percentage));
|
||||||
gclog_or_tty->print_cr("\nFrom compute_new_size: ");
|
gclog_or_tty->print_cr("\nFrom compute_new_size: ");
|
||||||
gclog_or_tty->print_cr(" Free fraction %f", free_percentage);
|
gclog_or_tty->print_cr(" Free fraction %f", free_percentage);
|
||||||
gclog_or_tty->print_cr(" Desired free fraction %f",
|
gclog_or_tty->print_cr(" Desired free fraction %f", desired_free_percentage);
|
||||||
desired_free_percentage);
|
gclog_or_tty->print_cr(" Maximum free fraction %f", maximum_free_percentage);
|
||||||
gclog_or_tty->print_cr(" Maximum free fraction %f",
|
gclog_or_tty->print_cr(" Capacity " SIZE_FORMAT, capacity() / 1000);
|
||||||
maximum_free_percentage);
|
gclog_or_tty->print_cr(" Desired capacity " SIZE_FORMAT, desired_capacity / 1000);
|
||||||
gclog_or_tty->print_cr(" Capacity " SIZE_FORMAT, capacity()/1000);
|
|
||||||
gclog_or_tty->print_cr(" Desired capacity " SIZE_FORMAT,
|
|
||||||
desired_capacity/1000);
|
|
||||||
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
||||||
assert(gch->is_old_gen(this), "The CMS generation should always be the old generation");
|
assert(gch->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 = gch->young_gen()->capacity();
|
||||||
gclog_or_tty->print_cr(" Young gen size " SIZE_FORMAT, young_size / 1000);
|
gclog_or_tty->print_cr(" Young gen size " SIZE_FORMAT, young_size / 1000);
|
||||||
gclog_or_tty->print_cr(" unsafe_max_alloc_nogc " SIZE_FORMAT,
|
gclog_or_tty->print_cr(" unsafe_max_alloc_nogc " SIZE_FORMAT, unsafe_max_alloc_nogc() / 1000);
|
||||||
unsafe_max_alloc_nogc()/1000);
|
gclog_or_tty->print_cr(" contiguous available " SIZE_FORMAT, contiguous_available() / 1000);
|
||||||
gclog_or_tty->print_cr(" contiguous available " SIZE_FORMAT,
|
gclog_or_tty->print_cr(" Expand by " SIZE_FORMAT " (bytes)", expand_bytes);
|
||||||
contiguous_available()/1000);
|
|
||||||
gclog_or_tty->print_cr(" Expand by " SIZE_FORMAT " (bytes)",
|
|
||||||
expand_bytes);
|
|
||||||
}
|
}
|
||||||
// safe if expansion fails
|
// safe if expansion fails
|
||||||
expand_for_gc_cause(expand_bytes, 0, CMSExpansionCause::_satisfy_free_ratio);
|
expand_for_gc_cause(expand_bytes, 0, CMSExpansionCause::_satisfy_free_ratio);
|
||||||
if (PrintGCDetails && Verbose) {
|
if (PrintGCDetails && Verbose) {
|
||||||
gclog_or_tty->print_cr(" Expanded free fraction %f",
|
gclog_or_tty->print_cr(" Expanded free fraction %f", ((double) free()) / capacity());
|
||||||
((double) free()) / capacity());
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
size_t desired_capacity = (size_t)(used() / ((double) 1 - desired_free_percentage));
|
size_t desired_capacity = (size_t)(used() / ((double) 1 - desired_free_percentage));
|
||||||
@ -834,11 +827,9 @@ Mutex* ConcurrentMarkSweepGeneration::freelistLock() const {
|
|||||||
return cmsSpace()->freelistLock();
|
return cmsSpace()->freelistLock();
|
||||||
}
|
}
|
||||||
|
|
||||||
HeapWord* ConcurrentMarkSweepGeneration::allocate(size_t size,
|
HeapWord* ConcurrentMarkSweepGeneration::allocate(size_t size, bool tlab) {
|
||||||
bool tlab) {
|
|
||||||
CMSSynchronousYieldRequest yr;
|
CMSSynchronousYieldRequest yr;
|
||||||
MutexLockerEx x(freelistLock(),
|
MutexLockerEx x(freelistLock(), Mutex::_no_safepoint_check_flag);
|
||||||
Mutex::_no_safepoint_check_flag);
|
|
||||||
return have_lock_and_allocate(size, tlab);
|
return have_lock_and_allocate(size, tlab);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2426,7 +2417,7 @@ void CMSCollector::verify_after_remark_work_1() {
|
|||||||
|
|
||||||
gch->gen_process_roots(&srs,
|
gch->gen_process_roots(&srs,
|
||||||
GenCollectedHeap::OldGen,
|
GenCollectedHeap::OldGen,
|
||||||
true, // younger gens are roots
|
true, // young gen as roots
|
||||||
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
||||||
should_unload_classes(),
|
should_unload_classes(),
|
||||||
¬Older,
|
¬Older,
|
||||||
@ -2498,7 +2489,7 @@ void CMSCollector::verify_after_remark_work_2() {
|
|||||||
|
|
||||||
gch->gen_process_roots(&srs,
|
gch->gen_process_roots(&srs,
|
||||||
GenCollectedHeap::OldGen,
|
GenCollectedHeap::OldGen,
|
||||||
true, // younger gens are roots
|
true, // young gen as roots
|
||||||
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
||||||
should_unload_classes(),
|
should_unload_classes(),
|
||||||
¬Older,
|
¬Older,
|
||||||
@ -2952,12 +2943,7 @@ void CMSCollector::checkpointRootsInitialWork() {
|
|||||||
assert(SafepointSynchronize::is_at_safepoint(), "world should be stopped");
|
assert(SafepointSynchronize::is_at_safepoint(), "world should be stopped");
|
||||||
assert(_collectorState == InitialMarking, "just checking");
|
assert(_collectorState == InitialMarking, "just checking");
|
||||||
|
|
||||||
// If there has not been a GC[n-1] since last GC[n] cycle completed,
|
// Already have locks.
|
||||||
// precede our marking with a collection of all
|
|
||||||
// younger generations to keep floating garbage to a minimum.
|
|
||||||
// XXX: we won't do this for now -- it's an optimization to be done later.
|
|
||||||
|
|
||||||
// already have locks
|
|
||||||
assert_lock_strong(bitMapLock());
|
assert_lock_strong(bitMapLock());
|
||||||
assert(_markBitMap.isAllClear(), "was reset at end of previous cycle");
|
assert(_markBitMap.isAllClear(), "was reset at end of previous cycle");
|
||||||
|
|
||||||
@ -3027,7 +3013,7 @@ void CMSCollector::checkpointRootsInitialWork() {
|
|||||||
|
|
||||||
gch->gen_process_roots(&srs,
|
gch->gen_process_roots(&srs,
|
||||||
GenCollectedHeap::OldGen,
|
GenCollectedHeap::OldGen,
|
||||||
true, // younger gens are roots
|
true, // young gen as roots
|
||||||
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
GenCollectedHeap::ScanningOption(roots_scanning_options()),
|
||||||
should_unload_classes(),
|
should_unload_classes(),
|
||||||
¬Older,
|
¬Older,
|
||||||
@ -3037,7 +3023,7 @@ void CMSCollector::checkpointRootsInitialWork() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Clear mod-union table; it will be dirtied in the prologue of
|
// Clear mod-union table; it will be dirtied in the prologue of
|
||||||
// CMS generation per each younger generation collection.
|
// CMS generation per each young generation collection.
|
||||||
|
|
||||||
assert(_modUnionTable.isAllClear(),
|
assert(_modUnionTable.isAllClear(),
|
||||||
"Was cleared in most recent final checkpoint phase"
|
"Was cleared in most recent final checkpoint phase"
|
||||||
@ -3057,7 +3043,7 @@ bool CMSCollector::markFromRoots() {
|
|||||||
// assert(!SafepointSynchronize::is_at_safepoint(),
|
// assert(!SafepointSynchronize::is_at_safepoint(),
|
||||||
// "inconsistent argument?");
|
// "inconsistent argument?");
|
||||||
// However that wouldn't be right, because it's possible that
|
// However that wouldn't be right, because it's possible that
|
||||||
// a safepoint is indeed in progress as a younger generation
|
// a safepoint is indeed in progress as a young generation
|
||||||
// stop-the-world GC happens even as we mark in this generation.
|
// stop-the-world GC happens even as we mark in this generation.
|
||||||
assert(_collectorState == Marking, "inconsistent state?");
|
assert(_collectorState == Marking, "inconsistent state?");
|
||||||
check_correct_thread_executing();
|
check_correct_thread_executing();
|
||||||
@ -3065,7 +3051,7 @@ bool CMSCollector::markFromRoots() {
|
|||||||
|
|
||||||
// Weak ref discovery note: We may be discovering weak
|
// Weak ref discovery note: We may be discovering weak
|
||||||
// refs in this generation concurrent (but interleaved) with
|
// refs in this generation concurrent (but interleaved) with
|
||||||
// weak ref discovery by a younger generation collector.
|
// weak ref discovery by the young generation collector.
|
||||||
|
|
||||||
CMSTokenSyncWithLocks ts(true, bitMapLock());
|
CMSTokenSyncWithLocks ts(true, bitMapLock());
|
||||||
TraceCPUTime tcpu(PrintGCDetails, true, gclog_or_tty);
|
TraceCPUTime tcpu(PrintGCDetails, true, gclog_or_tty);
|
||||||
@ -3095,7 +3081,7 @@ bool CMSCollector::markFromRootsWork() {
|
|||||||
|
|
||||||
// Note that when we do a marking step we need to hold the
|
// Note that when we do a marking step we need to hold the
|
||||||
// bit map lock -- recall that direct allocation (by mutators)
|
// bit map lock -- recall that direct allocation (by mutators)
|
||||||
// and promotion (by younger generation collectors) is also
|
// and promotion (by the young generation collector) is also
|
||||||
// marking the bit map. [the so-called allocate live policy.]
|
// marking the bit map. [the so-called allocate live policy.]
|
||||||
// Because the implementation of bit map marking is not
|
// Because the implementation of bit map marking is not
|
||||||
// robust wrt simultaneous marking of bits in the same word,
|
// robust wrt simultaneous marking of bits in the same word,
|
||||||
@ -4049,7 +4035,7 @@ size_t CMSCollector::preclean_work(bool clean_refs, bool clean_survivor) {
|
|||||||
// one of these methods, please check the other method too.
|
// one of these methods, please check the other method too.
|
||||||
|
|
||||||
size_t CMSCollector::preclean_mod_union_table(
|
size_t CMSCollector::preclean_mod_union_table(
|
||||||
ConcurrentMarkSweepGeneration* gen,
|
ConcurrentMarkSweepGeneration* old_gen,
|
||||||
ScanMarkedObjectsAgainCarefullyClosure* cl) {
|
ScanMarkedObjectsAgainCarefullyClosure* cl) {
|
||||||
verify_work_stacks_empty();
|
verify_work_stacks_empty();
|
||||||
verify_overflow_empty();
|
verify_overflow_empty();
|
||||||
@ -4064,10 +4050,10 @@ size_t CMSCollector::preclean_mod_union_table(
|
|||||||
// generation, but we might potentially miss cards when the
|
// generation, but we might potentially miss cards when the
|
||||||
// generation is rapidly expanding while we are in the midst
|
// generation is rapidly expanding while we are in the midst
|
||||||
// of precleaning.
|
// of precleaning.
|
||||||
HeapWord* startAddr = gen->reserved().start();
|
HeapWord* startAddr = old_gen->reserved().start();
|
||||||
HeapWord* endAddr = gen->reserved().end();
|
HeapWord* endAddr = old_gen->reserved().end();
|
||||||
|
|
||||||
cl->setFreelistLock(gen->freelistLock()); // needed for yielding
|
cl->setFreelistLock(old_gen->freelistLock()); // needed for yielding
|
||||||
|
|
||||||
size_t numDirtyCards, cumNumDirtyCards;
|
size_t numDirtyCards, cumNumDirtyCards;
|
||||||
HeapWord *nextAddr, *lastAddr;
|
HeapWord *nextAddr, *lastAddr;
|
||||||
@ -4109,7 +4095,7 @@ size_t CMSCollector::preclean_mod_union_table(
|
|||||||
HeapWord* stop_point = NULL;
|
HeapWord* stop_point = NULL;
|
||||||
stopTimer();
|
stopTimer();
|
||||||
// Potential yield point
|
// Potential yield point
|
||||||
CMSTokenSyncWithLocks ts(true, gen->freelistLock(),
|
CMSTokenSyncWithLocks ts(true, old_gen->freelistLock(),
|
||||||
bitMapLock());
|
bitMapLock());
|
||||||
startTimer();
|
startTimer();
|
||||||
{
|
{
|
||||||
@ -4117,7 +4103,7 @@ size_t CMSCollector::preclean_mod_union_table(
|
|||||||
verify_overflow_empty();
|
verify_overflow_empty();
|
||||||
sample_eden();
|
sample_eden();
|
||||||
stop_point =
|
stop_point =
|
||||||
gen->cmsSpace()->object_iterate_careful_m(dirtyRegion, cl);
|
old_gen->cmsSpace()->object_iterate_careful_m(dirtyRegion, cl);
|
||||||
}
|
}
|
||||||
if (stop_point != NULL) {
|
if (stop_point != NULL) {
|
||||||
// The careful iteration stopped early either because it found an
|
// The careful iteration stopped early either because it found an
|
||||||
@ -4152,15 +4138,15 @@ size_t CMSCollector::preclean_mod_union_table(
|
|||||||
// below are largely identical; if you need to modify
|
// below are largely identical; if you need to modify
|
||||||
// one of these methods, please check the other method too.
|
// one of these methods, please check the other method too.
|
||||||
|
|
||||||
size_t CMSCollector::preclean_card_table(ConcurrentMarkSweepGeneration* gen,
|
size_t CMSCollector::preclean_card_table(ConcurrentMarkSweepGeneration* old_gen,
|
||||||
ScanMarkedObjectsAgainCarefullyClosure* cl) {
|
ScanMarkedObjectsAgainCarefullyClosure* cl) {
|
||||||
// strategy: it's similar to precleamModUnionTable above, in that
|
// strategy: it's similar to precleamModUnionTable above, in that
|
||||||
// we accumulate contiguous ranges of dirty cards, mark these cards
|
// we accumulate contiguous ranges of dirty cards, mark these cards
|
||||||
// precleaned, then scan the region covered by these cards.
|
// precleaned, then scan the region covered by these cards.
|
||||||
HeapWord* endAddr = (HeapWord*)(gen->_virtual_space.high());
|
HeapWord* endAddr = (HeapWord*)(old_gen->_virtual_space.high());
|
||||||
HeapWord* startAddr = (HeapWord*)(gen->_virtual_space.low());
|
HeapWord* startAddr = (HeapWord*)(old_gen->_virtual_space.low());
|
||||||
|
|
||||||
cl->setFreelistLock(gen->freelistLock()); // needed for yielding
|
cl->setFreelistLock(old_gen->freelistLock()); // needed for yielding
|
||||||
|
|
||||||
size_t numDirtyCards, cumNumDirtyCards;
|
size_t numDirtyCards, cumNumDirtyCards;
|
||||||
HeapWord *lastAddr, *nextAddr;
|
HeapWord *lastAddr, *nextAddr;
|
||||||
@ -4197,13 +4183,13 @@ size_t CMSCollector::preclean_card_table(ConcurrentMarkSweepGeneration* gen,
|
|||||||
|
|
||||||
if (!dirtyRegion.is_empty()) {
|
if (!dirtyRegion.is_empty()) {
|
||||||
stopTimer();
|
stopTimer();
|
||||||
CMSTokenSyncWithLocks ts(true, gen->freelistLock(), bitMapLock());
|
CMSTokenSyncWithLocks ts(true, old_gen->freelistLock(), bitMapLock());
|
||||||
startTimer();
|
startTimer();
|
||||||
sample_eden();
|
sample_eden();
|
||||||
verify_work_stacks_empty();
|
verify_work_stacks_empty();
|
||||||
verify_overflow_empty();
|
verify_overflow_empty();
|
||||||
HeapWord* stop_point =
|
HeapWord* stop_point =
|
||||||
gen->cmsSpace()->object_iterate_careful_m(dirtyRegion, cl);
|
old_gen->cmsSpace()->object_iterate_careful_m(dirtyRegion, cl);
|
||||||
if (stop_point != NULL) {
|
if (stop_point != NULL) {
|
||||||
assert((_collectorState == AbortablePreclean && should_abort_preclean()),
|
assert((_collectorState == AbortablePreclean && should_abort_preclean()),
|
||||||
"Should only be AbortablePreclean.");
|
"Should only be AbortablePreclean.");
|
||||||
@ -5086,7 +5072,7 @@ void CMSCollector::do_remark_parallel() {
|
|||||||
// preclean phase did of eden, plus the [two] tasks of
|
// preclean phase did of eden, plus the [two] tasks of
|
||||||
// scanning the [two] survivor spaces. Further fine-grain
|
// scanning the [two] survivor spaces. Further fine-grain
|
||||||
// parallelization of the scanning of the survivor spaces
|
// parallelization of the scanning of the survivor spaces
|
||||||
// themselves, and of precleaning of the younger gen itself
|
// themselves, and of precleaning of the young gen itself
|
||||||
// is deferred to the future.
|
// is deferred to the future.
|
||||||
initialize_sequential_subtasks_for_young_gen_rescan(n_workers);
|
initialize_sequential_subtasks_for_young_gen_rescan(n_workers);
|
||||||
|
|
||||||
@ -5177,7 +5163,7 @@ void CMSCollector::do_remark_non_parallel() {
|
|||||||
|
|
||||||
gch->gen_process_roots(&srs,
|
gch->gen_process_roots(&srs,
|
||||||
GenCollectedHeap::OldGen,
|
GenCollectedHeap::OldGen,
|
||||||
true, // younger gens 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,
|
||||||
@ -5661,7 +5647,7 @@ void ConcurrentMarkSweepGeneration::update_gc_stats(Generation* current_generati
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void CMSCollector::sweepWork(ConcurrentMarkSweepGeneration* gen) {
|
void CMSCollector::sweepWork(ConcurrentMarkSweepGeneration* old_gen) {
|
||||||
// We iterate over the space(s) underlying this generation,
|
// We iterate over the space(s) underlying this generation,
|
||||||
// checking the mark bit map to see if the bits corresponding
|
// checking the mark bit map to see if the bits corresponding
|
||||||
// to specific blocks are marked or not. Blocks that are
|
// to specific blocks are marked or not. Blocks that are
|
||||||
@ -5690,26 +5676,26 @@ void CMSCollector::sweepWork(ConcurrentMarkSweepGeneration* gen) {
|
|||||||
// check that we hold the requisite locks
|
// check that we hold the requisite locks
|
||||||
assert(have_cms_token(), "Should hold cms token");
|
assert(have_cms_token(), "Should hold cms token");
|
||||||
assert(ConcurrentMarkSweepThread::cms_thread_has_cms_token(), "Should possess CMS token to sweep");
|
assert(ConcurrentMarkSweepThread::cms_thread_has_cms_token(), "Should possess CMS token to sweep");
|
||||||
assert_lock_strong(gen->freelistLock());
|
assert_lock_strong(old_gen->freelistLock());
|
||||||
assert_lock_strong(bitMapLock());
|
assert_lock_strong(bitMapLock());
|
||||||
|
|
||||||
assert(!_inter_sweep_timer.is_active(), "Was switched off in an outer context");
|
assert(!_inter_sweep_timer.is_active(), "Was switched off in an outer context");
|
||||||
assert(_intra_sweep_timer.is_active(), "Was switched on in an outer context");
|
assert(_intra_sweep_timer.is_active(), "Was switched on in an outer context");
|
||||||
gen->cmsSpace()->beginSweepFLCensus((float)(_inter_sweep_timer.seconds()),
|
old_gen->cmsSpace()->beginSweepFLCensus((float)(_inter_sweep_timer.seconds()),
|
||||||
_inter_sweep_estimate.padded_average(),
|
_inter_sweep_estimate.padded_average(),
|
||||||
_intra_sweep_estimate.padded_average());
|
_intra_sweep_estimate.padded_average());
|
||||||
gen->setNearLargestChunk();
|
old_gen->setNearLargestChunk();
|
||||||
|
|
||||||
{
|
{
|
||||||
SweepClosure sweepClosure(this, gen, &_markBitMap, CMSYield);
|
SweepClosure sweepClosure(this, old_gen, &_markBitMap, CMSYield);
|
||||||
gen->cmsSpace()->blk_iterate_careful(&sweepClosure);
|
old_gen->cmsSpace()->blk_iterate_careful(&sweepClosure);
|
||||||
// We need to free-up/coalesce garbage/blocks from a
|
// We need to free-up/coalesce garbage/blocks from a
|
||||||
// co-terminal free run. This is done in the SweepClosure
|
// co-terminal free run. This is done in the SweepClosure
|
||||||
// destructor; so, do not remove this scope, else the
|
// destructor; so, do not remove this scope, else the
|
||||||
// end-of-sweep-census below will be off by a little bit.
|
// end-of-sweep-census below will be off by a little bit.
|
||||||
}
|
}
|
||||||
gen->cmsSpace()->sweep_completed();
|
old_gen->cmsSpace()->sweep_completed();
|
||||||
gen->cmsSpace()->endSweepFLCensus(sweep_count());
|
old_gen->cmsSpace()->endSweepFLCensus(sweep_count());
|
||||||
if (should_unload_classes()) { // unloaded classes this cycle,
|
if (should_unload_classes()) { // unloaded classes this cycle,
|
||||||
_concurrent_cycles_since_last_unload = 0; // ... reset count
|
_concurrent_cycles_since_last_unload = 0; // ... reset count
|
||||||
} else { // did not unload classes,
|
} else { // did not unload classes,
|
||||||
|
@ -723,7 +723,7 @@ class CMSCollector: public CHeapObj<mtGC> {
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
// Support for parallelizing young gen rescan in CMS remark phase
|
// Support for parallelizing young gen rescan in CMS remark phase
|
||||||
ParNewGeneration* _young_gen; // the younger gen
|
ParNewGeneration* _young_gen;
|
||||||
|
|
||||||
HeapWord** _top_addr; // ... Top of Eden
|
HeapWord** _top_addr; // ... Top of Eden
|
||||||
HeapWord** _end_addr; // ... End of Eden
|
HeapWord** _end_addr; // ... End of Eden
|
||||||
@ -772,9 +772,9 @@ class CMSCollector: public CHeapObj<mtGC> {
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
// Concurrent precleaning work
|
// Concurrent precleaning work
|
||||||
size_t preclean_mod_union_table(ConcurrentMarkSweepGeneration* gen,
|
size_t preclean_mod_union_table(ConcurrentMarkSweepGeneration* old_gen,
|
||||||
ScanMarkedObjectsAgainCarefullyClosure* cl);
|
ScanMarkedObjectsAgainCarefullyClosure* cl);
|
||||||
size_t preclean_card_table(ConcurrentMarkSweepGeneration* gen,
|
size_t preclean_card_table(ConcurrentMarkSweepGeneration* old_gen,
|
||||||
ScanMarkedObjectsAgainCarefullyClosure* cl);
|
ScanMarkedObjectsAgainCarefullyClosure* cl);
|
||||||
// Does precleaning work, returning a quantity indicative of
|
// Does precleaning work, returning a quantity indicative of
|
||||||
// the amount of "useful work" done.
|
// the amount of "useful work" done.
|
||||||
@ -797,7 +797,7 @@ class CMSCollector: public CHeapObj<mtGC> {
|
|||||||
void refProcessingWork();
|
void refProcessingWork();
|
||||||
|
|
||||||
// Concurrent sweeping work
|
// Concurrent sweeping work
|
||||||
void sweepWork(ConcurrentMarkSweepGeneration* gen);
|
void sweepWork(ConcurrentMarkSweepGeneration* old_gen);
|
||||||
|
|
||||||
// (Concurrent) resetting of support data structures
|
// (Concurrent) resetting of support data structures
|
||||||
void reset(bool concurrent);
|
void reset(bool concurrent);
|
||||||
@ -1120,10 +1120,8 @@ class ConcurrentMarkSweepGeneration: public CardGeneration {
|
|||||||
MemRegion used_region_at_save_marks() const;
|
MemRegion used_region_at_save_marks() const;
|
||||||
|
|
||||||
// Does a "full" (forced) collection invoked on this generation collect
|
// Does a "full" (forced) collection invoked on this generation collect
|
||||||
// all younger generations as well? Note that the second conjunct is a
|
// the young generation as well?
|
||||||
// hack to allow the collection of the younger gen first if the flag is
|
virtual bool full_collects_young_generation() const {
|
||||||
// set.
|
|
||||||
virtual bool full_collects_younger_generations() const {
|
|
||||||
return !ScavengeBeforeFullGC;
|
return !ScavengeBeforeFullGC;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1153,9 +1151,8 @@ class ConcurrentMarkSweepGeneration: public CardGeneration {
|
|||||||
|
|
||||||
virtual bool promotion_attempt_is_safe(size_t promotion_in_bytes) const;
|
virtual bool promotion_attempt_is_safe(size_t promotion_in_bytes) const;
|
||||||
|
|
||||||
// Inform this (non-young) generation that a promotion failure was
|
// Inform this (old) generation that a promotion failure was
|
||||||
// encountered during a collection of a younger generation that
|
// encountered during a collection of the young generation.
|
||||||
// promotes into this generation.
|
|
||||||
virtual void promotion_failure_occurred();
|
virtual void promotion_failure_occurred();
|
||||||
|
|
||||||
bool should_collect(bool full, size_t size, bool tlab);
|
bool should_collect(bool full, size_t size, bool tlab);
|
||||||
|
@ -295,7 +295,7 @@ inline void CMSStats::record_gc0_end(size_t cms_gen_bytes_used) {
|
|||||||
promoted_bytes = _cms_used_at_gc0_end - _cms_used_at_gc0_begin;
|
promoted_bytes = _cms_used_at_gc0_end - _cms_used_at_gc0_begin;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the younger gen collections were skipped, then the
|
// If the young gen collection was skipped, then the
|
||||||
// number of promoted bytes will be 0 and adding it to the
|
// number of promoted bytes will be 0 and adding it to the
|
||||||
// average will incorrectly lessen the average. It is, however,
|
// average will incorrectly lessen the average. It is, however,
|
||||||
// also possible that no promotion was needed.
|
// also possible that no promotion was needed.
|
||||||
|
@ -39,23 +39,17 @@
|
|||||||
|
|
||||||
// ======= Concurrent Mark Sweep Thread ========
|
// ======= Concurrent Mark Sweep Thread ========
|
||||||
|
|
||||||
// The CMS thread is created when Concurrent Mark Sweep is used in the
|
ConcurrentMarkSweepThread* ConcurrentMarkSweepThread::_cmst = NULL;
|
||||||
// older of two generations in a generational memory system.
|
|
||||||
|
|
||||||
ConcurrentMarkSweepThread*
|
|
||||||
ConcurrentMarkSweepThread::_cmst = NULL;
|
|
||||||
CMSCollector* ConcurrentMarkSweepThread::_collector = NULL;
|
CMSCollector* ConcurrentMarkSweepThread::_collector = NULL;
|
||||||
bool ConcurrentMarkSweepThread::_should_terminate = false;
|
bool ConcurrentMarkSweepThread::_should_terminate = false;
|
||||||
int ConcurrentMarkSweepThread::_CMS_flag = CMS_nil;
|
int ConcurrentMarkSweepThread::_CMS_flag = CMS_nil;
|
||||||
|
|
||||||
volatile jint ConcurrentMarkSweepThread::_pending_yields = 0;
|
volatile jint ConcurrentMarkSweepThread::_pending_yields = 0;
|
||||||
|
|
||||||
SurrogateLockerThread*
|
SurrogateLockerThread* ConcurrentMarkSweepThread::_slt = NULL;
|
||||||
ConcurrentMarkSweepThread::_slt = NULL;
|
|
||||||
SurrogateLockerThread::SLT_msg_type
|
SurrogateLockerThread::SLT_msg_type
|
||||||
ConcurrentMarkSweepThread::_sltBuffer = SurrogateLockerThread::empty;
|
ConcurrentMarkSweepThread::_sltBuffer = SurrogateLockerThread::empty;
|
||||||
Monitor*
|
Monitor* ConcurrentMarkSweepThread::_sltMonitor = NULL;
|
||||||
ConcurrentMarkSweepThread::_sltMonitor = NULL;
|
|
||||||
|
|
||||||
ConcurrentMarkSweepThread::ConcurrentMarkSweepThread(CMSCollector* collector)
|
ConcurrentMarkSweepThread::ConcurrentMarkSweepThread(CMSCollector* collector)
|
||||||
: ConcurrentGCThread() {
|
: ConcurrentGCThread() {
|
||||||
|
@ -69,20 +69,28 @@ ParScanThreadState::ParScanThreadState(Space* to_space_,
|
|||||||
Stack<oop, mtGC>* overflow_stacks_,
|
Stack<oop, mtGC>* overflow_stacks_,
|
||||||
size_t desired_plab_sz_,
|
size_t desired_plab_sz_,
|
||||||
ParallelTaskTerminator& term_) :
|
ParallelTaskTerminator& term_) :
|
||||||
_to_space(to_space_), _old_gen(old_gen_), _young_gen(young_gen_), _thread_num(thread_num_),
|
_to_space(to_space_),
|
||||||
_work_queue(work_queue_set_->queue(thread_num_)), _to_space_full(false),
|
_old_gen(old_gen_),
|
||||||
|
_young_gen(young_gen_),
|
||||||
|
_thread_num(thread_num_),
|
||||||
|
_work_queue(work_queue_set_->queue(thread_num_)),
|
||||||
|
_to_space_full(false),
|
||||||
_overflow_stack(overflow_stacks_ ? overflow_stacks_ + thread_num_ : NULL),
|
_overflow_stack(overflow_stacks_ ? overflow_stacks_ + thread_num_ : NULL),
|
||||||
_ageTable(false), // false ==> not the global age table, no perf data.
|
_ageTable(false), // false ==> not the global age table, no perf data.
|
||||||
_to_space_alloc_buffer(desired_plab_sz_),
|
_to_space_alloc_buffer(desired_plab_sz_),
|
||||||
_to_space_closure(young_gen_, this), _old_gen_closure(young_gen_, this),
|
_to_space_closure(young_gen_, this),
|
||||||
_to_space_root_closure(young_gen_, this), _old_gen_root_closure(young_gen_, this),
|
_old_gen_closure(young_gen_, this),
|
||||||
|
_to_space_root_closure(young_gen_, this),
|
||||||
|
_old_gen_root_closure(young_gen_, this),
|
||||||
_older_gen_closure(young_gen_, this),
|
_older_gen_closure(young_gen_, this),
|
||||||
_evacuate_followers(this, &_to_space_closure, &_old_gen_closure,
|
_evacuate_followers(this, &_to_space_closure, &_old_gen_closure,
|
||||||
&_to_space_root_closure, young_gen_, &_old_gen_root_closure,
|
&_to_space_root_closure, young_gen_, &_old_gen_root_closure,
|
||||||
work_queue_set_, &term_),
|
work_queue_set_, &term_),
|
||||||
_is_alive_closure(young_gen_), _scan_weak_ref_closure(young_gen_, this),
|
_is_alive_closure(young_gen_),
|
||||||
|
_scan_weak_ref_closure(young_gen_, this),
|
||||||
_keep_alive_closure(&_scan_weak_ref_closure),
|
_keep_alive_closure(&_scan_weak_ref_closure),
|
||||||
_strong_roots_time(0.0), _term_time(0.0)
|
_strong_roots_time(0.0),
|
||||||
|
_term_time(0.0)
|
||||||
{
|
{
|
||||||
#if TASKQUEUE_STATS
|
#if TASKQUEUE_STATS
|
||||||
_term_attempts = 0;
|
_term_attempts = 0;
|
||||||
@ -90,8 +98,7 @@ ParScanThreadState::ParScanThreadState(Space* to_space_,
|
|||||||
_overflow_refill_objs = 0;
|
_overflow_refill_objs = 0;
|
||||||
#endif // TASKQUEUE_STATS
|
#endif // TASKQUEUE_STATS
|
||||||
|
|
||||||
_survivor_chunk_array =
|
_survivor_chunk_array = (ChunkArray*) old_gen()->get_data_recorder(thread_num());
|
||||||
(ChunkArray*) old_gen()->get_data_recorder(thread_num());
|
|
||||||
_hash_seed = 17; // Might want to take time-based random value.
|
_hash_seed = 17; // Might want to take time-based random value.
|
||||||
_start = os::elapsedTime();
|
_start = os::elapsedTime();
|
||||||
_old_gen_closure.set_generation(old_gen_);
|
_old_gen_closure.set_generation(old_gen_);
|
||||||
@ -154,7 +161,6 @@ void ParScanThreadState::scan_partial_array_and_push_remainder(oop old) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ParScanThreadState::trim_queues(int max_size) {
|
void ParScanThreadState::trim_queues(int max_size) {
|
||||||
ObjToScanQueue* queue = work_queue();
|
ObjToScanQueue* queue = work_queue();
|
||||||
do {
|
do {
|
||||||
@ -222,15 +228,12 @@ void ParScanThreadState::push_on_overflow_stack(oop p) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
HeapWord* ParScanThreadState::alloc_in_to_space_slow(size_t word_sz) {
|
HeapWord* ParScanThreadState::alloc_in_to_space_slow(size_t word_sz) {
|
||||||
|
// If the object is small enough, try to reallocate the buffer.
|
||||||
// Otherwise, if the object is small enough, try to reallocate the
|
|
||||||
// buffer.
|
|
||||||
HeapWord* obj = NULL;
|
HeapWord* obj = NULL;
|
||||||
if (!_to_space_full) {
|
if (!_to_space_full) {
|
||||||
PLAB* const plab = to_space_alloc_buffer();
|
PLAB* const plab = to_space_alloc_buffer();
|
||||||
Space* const sp = to_space();
|
Space* const sp = to_space();
|
||||||
if (word_sz * 100 <
|
if (word_sz * 100 < ParallelGCBufferWastePct * plab->word_sz()) {
|
||||||
ParallelGCBufferWastePct * plab->word_sz()) {
|
|
||||||
// Is small enough; abandon this buffer and start a new one.
|
// Is small enough; abandon this buffer and start a new one.
|
||||||
plab->retire();
|
plab->retire();
|
||||||
size_t buf_size = plab->word_sz();
|
size_t buf_size = plab->word_sz();
|
||||||
@ -241,8 +244,7 @@ HeapWord* ParScanThreadState::alloc_in_to_space_slow(size_t word_sz) {
|
|||||||
size_t free_bytes = sp->free();
|
size_t free_bytes = sp->free();
|
||||||
while(buf_space == NULL && free_bytes >= min_bytes) {
|
while(buf_space == NULL && free_bytes >= min_bytes) {
|
||||||
buf_size = free_bytes >> LogHeapWordSize;
|
buf_size = free_bytes >> LogHeapWordSize;
|
||||||
assert(buf_size == (size_t)align_object_size(buf_size),
|
assert(buf_size == (size_t)align_object_size(buf_size), "Invariant");
|
||||||
"Invariant");
|
|
||||||
buf_space = sp->par_allocate(buf_size);
|
buf_space = sp->par_allocate(buf_size);
|
||||||
free_bytes = sp->free();
|
free_bytes = sp->free();
|
||||||
}
|
}
|
||||||
@ -262,7 +264,6 @@ HeapWord* ParScanThreadState::alloc_in_to_space_slow(size_t word_sz) {
|
|||||||
// We're used up.
|
// We're used up.
|
||||||
_to_space_full = true;
|
_to_space_full = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
// Too large; allocate the object individually.
|
// Too large; allocate the object individually.
|
||||||
obj = sp->par_allocate(word_sz);
|
obj = sp->par_allocate(word_sz);
|
||||||
@ -271,7 +272,6 @@ HeapWord* ParScanThreadState::alloc_in_to_space_slow(size_t word_sz) {
|
|||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ParScanThreadState::undo_alloc_in_to_space(HeapWord* obj, size_t word_sz) {
|
void ParScanThreadState::undo_alloc_in_to_space(HeapWord* obj, size_t word_sz) {
|
||||||
to_space_alloc_buffer()->undo_allocation(obj, word_sz);
|
to_space_alloc_buffer()->undo_allocation(obj, word_sz);
|
||||||
}
|
}
|
||||||
@ -288,7 +288,7 @@ public:
|
|||||||
// Initializes states for the specified number of threads;
|
// Initializes states for the specified number of threads;
|
||||||
ParScanThreadStateSet(int num_threads,
|
ParScanThreadStateSet(int num_threads,
|
||||||
Space& to_space,
|
Space& to_space,
|
||||||
ParNewGeneration& gen,
|
ParNewGeneration& young_gen,
|
||||||
Generation& old_gen,
|
Generation& old_gen,
|
||||||
ObjToScanQueueSet& queue_set,
|
ObjToScanQueueSet& queue_set,
|
||||||
Stack<oop, mtGC>* overflow_stacks_,
|
Stack<oop, mtGC>* overflow_stacks_,
|
||||||
@ -315,21 +315,25 @@ public:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
ParallelTaskTerminator& _term;
|
ParallelTaskTerminator& _term;
|
||||||
ParNewGeneration& _gen;
|
ParNewGeneration& _young_gen;
|
||||||
Generation& _old_gen;
|
Generation& _old_gen;
|
||||||
public:
|
public:
|
||||||
bool is_valid(int id) const { return id < length(); }
|
bool is_valid(int id) const { return id < length(); }
|
||||||
ParallelTaskTerminator* terminator() { return &_term; }
|
ParallelTaskTerminator* terminator() { return &_term; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ParScanThreadStateSet::ParScanThreadStateSet(int num_threads,
|
||||||
ParScanThreadStateSet::ParScanThreadStateSet(
|
Space& to_space,
|
||||||
int num_threads, Space& to_space, ParNewGeneration& gen,
|
ParNewGeneration& young_gen,
|
||||||
Generation& old_gen, ObjToScanQueueSet& queue_set,
|
Generation& old_gen,
|
||||||
|
ObjToScanQueueSet& queue_set,
|
||||||
Stack<oop, mtGC>* overflow_stacks,
|
Stack<oop, mtGC>* overflow_stacks,
|
||||||
size_t desired_plab_sz, ParallelTaskTerminator& term)
|
size_t desired_plab_sz,
|
||||||
|
ParallelTaskTerminator& term)
|
||||||
: ResourceArray(sizeof(ParScanThreadState), num_threads),
|
: ResourceArray(sizeof(ParScanThreadState), num_threads),
|
||||||
_gen(gen), _old_gen(old_gen), _term(term)
|
_young_gen(young_gen),
|
||||||
|
_old_gen(old_gen),
|
||||||
|
_term(term)
|
||||||
{
|
{
|
||||||
assert(num_threads > 0, "sanity check!");
|
assert(num_threads > 0, "sanity check!");
|
||||||
assert(ParGCUseLocalOverflow == (overflow_stacks != NULL),
|
assert(ParGCUseLocalOverflow == (overflow_stacks != NULL),
|
||||||
@ -337,13 +341,12 @@ ParScanThreadStateSet::ParScanThreadStateSet(
|
|||||||
// Initialize states.
|
// Initialize states.
|
||||||
for (int i = 0; i < num_threads; ++i) {
|
for (int i = 0; i < num_threads; ++i) {
|
||||||
new ((ParScanThreadState*)_data + i)
|
new ((ParScanThreadState*)_data + i)
|
||||||
ParScanThreadState(&to_space, &gen, &old_gen, i, &queue_set,
|
ParScanThreadState(&to_space, &young_gen, &old_gen, i, &queue_set,
|
||||||
overflow_stacks, desired_plab_sz, term);
|
overflow_stacks, desired_plab_sz, term);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inline ParScanThreadState& ParScanThreadStateSet::thread_state(int i)
|
inline ParScanThreadState& ParScanThreadStateSet::thread_state(int i) {
|
||||||
{
|
|
||||||
assert(i >= 0 && i < length(), "sanity check!");
|
assert(i >= 0 && i < length(), "sanity check!");
|
||||||
return ((ParScanThreadState*)_data)[i];
|
return ((ParScanThreadState*)_data)[i];
|
||||||
}
|
}
|
||||||
@ -357,8 +360,7 @@ void ParScanThreadStateSet::trace_promotion_failed(const YoungGCTracer* gc_trace
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParScanThreadStateSet::reset(uint active_threads, bool promotion_failed)
|
void ParScanThreadStateSet::reset(uint active_threads, bool promotion_failed) {
|
||||||
{
|
|
||||||
_term.reset_for_reuse(active_threads);
|
_term.reset_for_reuse(active_threads);
|
||||||
if (promotion_failed) {
|
if (promotion_failed) {
|
||||||
for (int i = 0; i < length(); ++i) {
|
for (int i = 0; i < length(); ++i) {
|
||||||
@ -368,36 +370,27 @@ void ParScanThreadStateSet::reset(uint active_threads, bool promotion_failed)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if TASKQUEUE_STATS
|
#if TASKQUEUE_STATS
|
||||||
void
|
void ParScanThreadState::reset_stats() {
|
||||||
ParScanThreadState::reset_stats()
|
|
||||||
{
|
|
||||||
taskqueue_stats().reset();
|
taskqueue_stats().reset();
|
||||||
_term_attempts = 0;
|
_term_attempts = 0;
|
||||||
_overflow_refills = 0;
|
_overflow_refills = 0;
|
||||||
_overflow_refill_objs = 0;
|
_overflow_refill_objs = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParScanThreadStateSet::reset_stats()
|
void ParScanThreadStateSet::reset_stats() {
|
||||||
{
|
|
||||||
for (int i = 0; i < length(); ++i) {
|
for (int i = 0; i < length(); ++i) {
|
||||||
thread_state(i).reset_stats();
|
thread_state(i).reset_stats();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void ParScanThreadStateSet::print_termination_stats_hdr(outputStream* const st) {
|
||||||
ParScanThreadStateSet::print_termination_stats_hdr(outputStream* const st)
|
|
||||||
{
|
|
||||||
st->print_raw_cr("GC Termination Stats");
|
st->print_raw_cr("GC Termination Stats");
|
||||||
st->print_raw_cr(" elapsed --strong roots-- "
|
st->print_raw_cr(" elapsed --strong roots-- -------termination-------");
|
||||||
"-------termination-------");
|
st->print_raw_cr("thr ms ms % ms % attempts");
|
||||||
st->print_raw_cr("thr ms ms % "
|
st->print_raw_cr("--- --------- --------- ------ --------- ------ --------");
|
||||||
" ms % attempts");
|
|
||||||
st->print_raw_cr("--- --------- --------- ------ "
|
|
||||||
"--------- ------ --------");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParScanThreadStateSet::print_termination_stats(outputStream* const st)
|
void ParScanThreadStateSet::print_termination_stats(outputStream* const st) {
|
||||||
{
|
|
||||||
print_termination_stats_hdr(st);
|
print_termination_stats_hdr(st);
|
||||||
|
|
||||||
for (int i = 0; i < length(); ++i) {
|
for (int i = 0; i < length(); ++i) {
|
||||||
@ -405,23 +398,20 @@ void ParScanThreadStateSet::print_termination_stats(outputStream* const st)
|
|||||||
const double elapsed_ms = pss.elapsed_time() * 1000.0;
|
const double elapsed_ms = pss.elapsed_time() * 1000.0;
|
||||||
const double s_roots_ms = pss.strong_roots_time() * 1000.0;
|
const double s_roots_ms = pss.strong_roots_time() * 1000.0;
|
||||||
const double term_ms = pss.term_time() * 1000.0;
|
const double term_ms = pss.term_time() * 1000.0;
|
||||||
st->print_cr("%3d %9.2f %9.2f %6.2f "
|
st->print_cr("%3d %9.2f %9.2f %6.2f %9.2f %6.2f " SIZE_FORMAT_W(8),
|
||||||
"%9.2f %6.2f " SIZE_FORMAT_W(8),
|
|
||||||
i, elapsed_ms, s_roots_ms, s_roots_ms * 100 / elapsed_ms,
|
i, elapsed_ms, s_roots_ms, s_roots_ms * 100 / elapsed_ms,
|
||||||
term_ms, term_ms * 100 / elapsed_ms, pss.term_attempts());
|
term_ms, term_ms * 100 / elapsed_ms, pss.term_attempts());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print stats related to work queue activity.
|
// Print stats related to work queue activity.
|
||||||
void ParScanThreadStateSet::print_taskqueue_stats_hdr(outputStream* const st)
|
void ParScanThreadStateSet::print_taskqueue_stats_hdr(outputStream* const st) {
|
||||||
{
|
|
||||||
st->print_raw_cr("GC Task Stats");
|
st->print_raw_cr("GC Task Stats");
|
||||||
st->print_raw("thr "); TaskQueueStats::print_header(1, st); st->cr();
|
st->print_raw("thr "); TaskQueueStats::print_header(1, st); st->cr();
|
||||||
st->print_raw("--- "); TaskQueueStats::print_header(2, st); st->cr();
|
st->print_raw("--- "); TaskQueueStats::print_header(2, st); st->cr();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParScanThreadStateSet::print_taskqueue_stats(outputStream* const st)
|
void ParScanThreadStateSet::print_taskqueue_stats(outputStream* const st) {
|
||||||
{
|
|
||||||
print_taskqueue_stats_hdr(st);
|
print_taskqueue_stats_hdr(st);
|
||||||
|
|
||||||
TaskQueueStats totals;
|
TaskQueueStats totals;
|
||||||
@ -443,8 +433,7 @@ void ParScanThreadStateSet::print_taskqueue_stats(outputStream* const st)
|
|||||||
}
|
}
|
||||||
#endif // TASKQUEUE_STATS
|
#endif // TASKQUEUE_STATS
|
||||||
|
|
||||||
void ParScanThreadStateSet::flush()
|
void ParScanThreadStateSet::flush() {
|
||||||
{
|
|
||||||
// Work in this loop should be kept as lightweight as
|
// Work in this loop should be kept as lightweight as
|
||||||
// possible since this might otherwise become a bottleneck
|
// possible since this might otherwise become a bottleneck
|
||||||
// to scaling. Should we add heavy-weight work into this
|
// to scaling. Should we add heavy-weight work into this
|
||||||
@ -454,12 +443,12 @@ void ParScanThreadStateSet::flush()
|
|||||||
|
|
||||||
// Flush stats related to To-space PLAB activity and
|
// Flush stats related to To-space PLAB activity and
|
||||||
// retire the last buffer.
|
// retire the last buffer.
|
||||||
par_scan_state.to_space_alloc_buffer()->flush_and_retire_stats(_gen.plab_stats());
|
par_scan_state.to_space_alloc_buffer()->flush_and_retire_stats(_young_gen.plab_stats());
|
||||||
|
|
||||||
// Every thread has its own age table. We need to merge
|
// Every thread has its own age table. We need to merge
|
||||||
// them all into one.
|
// them all into one.
|
||||||
ageTable *local_table = par_scan_state.age_table();
|
ageTable *local_table = par_scan_state.age_table();
|
||||||
_gen.age_table()->merge(local_table);
|
_young_gen.age_table()->merge(local_table);
|
||||||
|
|
||||||
// Inform old gen that we're done.
|
// Inform old gen that we're done.
|
||||||
_old_gen.par_promote_alloc_done(i);
|
_old_gen.par_promote_alloc_done(i);
|
||||||
@ -478,8 +467,7 @@ void ParScanThreadStateSet::flush()
|
|||||||
|
|
||||||
ParScanClosure::ParScanClosure(ParNewGeneration* g,
|
ParScanClosure::ParScanClosure(ParNewGeneration* g,
|
||||||
ParScanThreadState* par_scan_state) :
|
ParScanThreadState* par_scan_state) :
|
||||||
OopsInKlassOrGenClosure(g), _par_scan_state(par_scan_state), _g(g)
|
OopsInKlassOrGenClosure(g), _par_scan_state(par_scan_state), _g(g) {
|
||||||
{
|
|
||||||
_boundary = _g->reserved().end();
|
_boundary = _g->reserved().end();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -531,24 +519,23 @@ void ParEvacuateFollowersClosure::do_void() {
|
|||||||
ObjToScanQueue* work_q = par_scan_state()->work_queue();
|
ObjToScanQueue* work_q = par_scan_state()->work_queue();
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
|
||||||
// Scan to-space and old-gen objs until we run out of both.
|
// Scan to-space and old-gen objs until we run out of both.
|
||||||
oop obj_to_scan;
|
oop obj_to_scan;
|
||||||
par_scan_state()->trim_queues(0);
|
par_scan_state()->trim_queues(0);
|
||||||
|
|
||||||
// We have no local work, attempt to steal from other threads.
|
// We have no local work, attempt to steal from other threads.
|
||||||
|
|
||||||
// attempt to steal work from promoted.
|
// Attempt to steal work from promoted.
|
||||||
if (task_queues()->steal(par_scan_state()->thread_num(),
|
if (task_queues()->steal(par_scan_state()->thread_num(),
|
||||||
par_scan_state()->hash_seed(),
|
par_scan_state()->hash_seed(),
|
||||||
obj_to_scan)) {
|
obj_to_scan)) {
|
||||||
bool res = work_q->push(obj_to_scan);
|
bool res = work_q->push(obj_to_scan);
|
||||||
assert(res, "Empty queue should have room for a push.");
|
assert(res, "Empty queue should have room for a push.");
|
||||||
|
|
||||||
// if successful, goto Start.
|
// If successful, goto Start.
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
// try global overflow list.
|
// Try global overflow list.
|
||||||
} else if (par_gen()->take_from_overflow_list(par_scan_state())) {
|
} else if (par_gen()->take_from_overflow_list(par_scan_state())) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -564,15 +551,17 @@ void ParEvacuateFollowersClosure::do_void() {
|
|||||||
par_scan_state()->end_term_time();
|
par_scan_state()->end_term_time();
|
||||||
}
|
}
|
||||||
|
|
||||||
ParNewGenTask::ParNewGenTask(ParNewGeneration* young_gen, Generation* old_gen,
|
ParNewGenTask::ParNewGenTask(ParNewGeneration* young_gen,
|
||||||
HeapWord* young_old_boundary, ParScanThreadStateSet* state_set,
|
Generation* old_gen,
|
||||||
|
HeapWord* young_old_boundary,
|
||||||
|
ParScanThreadStateSet* state_set,
|
||||||
StrongRootsScope* strong_roots_scope) :
|
StrongRootsScope* strong_roots_scope) :
|
||||||
AbstractGangTask("ParNewGeneration collection"),
|
AbstractGangTask("ParNewGeneration collection"),
|
||||||
_young_gen(young_gen), _old_gen(old_gen),
|
_young_gen(young_gen), _old_gen(old_gen),
|
||||||
_young_old_boundary(young_old_boundary),
|
_young_old_boundary(young_old_boundary),
|
||||||
_state_set(state_set),
|
_state_set(state_set),
|
||||||
_strong_roots_scope(strong_roots_scope)
|
_strong_roots_scope(strong_roots_scope)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
void ParNewGenTask::work(uint worker_id) {
|
void ParNewGenTask::work(uint worker_id) {
|
||||||
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
||||||
@ -595,8 +584,7 @@ void ParNewGenTask::work(uint worker_id) {
|
|||||||
par_scan_state.start_strong_roots();
|
par_scan_state.start_strong_roots();
|
||||||
gch->gen_process_roots(_strong_roots_scope,
|
gch->gen_process_roots(_strong_roots_scope,
|
||||||
GenCollectedHeap::YoungGen,
|
GenCollectedHeap::YoungGen,
|
||||||
true, // Process younger gens, if any,
|
true, // Process younger gens, if any, as strong roots.
|
||||||
// as strong roots.
|
|
||||||
GenCollectedHeap::SO_ScavengeCodeCache,
|
GenCollectedHeap::SO_ScavengeCodeCache,
|
||||||
GenCollectedHeap::StrongAndWeakRoots,
|
GenCollectedHeap::StrongAndWeakRoots,
|
||||||
&par_scan_state.to_space_root_closure(),
|
&par_scan_state.to_space_root_closure(),
|
||||||
@ -613,8 +601,7 @@ void ParNewGenTask::work(uint worker_id) {
|
|||||||
#pragma warning( push )
|
#pragma warning( push )
|
||||||
#pragma warning( disable:4355 ) // 'this' : used in base member initializer list
|
#pragma warning( disable:4355 ) // 'this' : used in base member initializer list
|
||||||
#endif
|
#endif
|
||||||
ParNewGeneration::
|
ParNewGeneration::ParNewGeneration(ReservedSpace rs, size_t initial_byte_size)
|
||||||
ParNewGeneration(ReservedSpace rs, size_t initial_byte_size)
|
|
||||||
: DefNewGeneration(rs, initial_byte_size, "PCopy"),
|
: DefNewGeneration(rs, initial_byte_size, "PCopy"),
|
||||||
_overflow_list(NULL),
|
_overflow_list(NULL),
|
||||||
_is_alive_closure(this),
|
_is_alive_closure(this),
|
||||||
@ -625,20 +612,19 @@ ParNewGeneration(ReservedSpace rs, size_t initial_byte_size)
|
|||||||
_task_queues = new ObjToScanQueueSet(ParallelGCThreads);
|
_task_queues = new ObjToScanQueueSet(ParallelGCThreads);
|
||||||
guarantee(_task_queues != NULL, "task_queues allocation failure.");
|
guarantee(_task_queues != NULL, "task_queues allocation failure.");
|
||||||
|
|
||||||
for (uint i1 = 0; i1 < ParallelGCThreads; i1++) {
|
for (uint i = 0; i < ParallelGCThreads; i++) {
|
||||||
ObjToScanQueue *q = new ObjToScanQueue();
|
ObjToScanQueue *q = new ObjToScanQueue();
|
||||||
guarantee(q != NULL, "work_queue Allocation failure.");
|
guarantee(q != NULL, "work_queue Allocation failure.");
|
||||||
_task_queues->register_queue(i1, q);
|
_task_queues->register_queue(i, q);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (uint i2 = 0; i2 < ParallelGCThreads; i2++)
|
for (uint i = 0; i < ParallelGCThreads; i++) {
|
||||||
_task_queues->queue(i2)->initialize();
|
_task_queues->queue(i)->initialize();
|
||||||
|
}
|
||||||
|
|
||||||
_overflow_stacks = NULL;
|
_overflow_stacks = NULL;
|
||||||
if (ParGCUseLocalOverflow) {
|
if (ParGCUseLocalOverflow) {
|
||||||
|
// typedef to workaround NEW_C_HEAP_ARRAY macro, which can not deal with ','
|
||||||
// typedef to workaround NEW_C_HEAP_ARRAY macro, which can not deal
|
|
||||||
// with ','
|
|
||||||
typedef Stack<oop, mtGC> GCOopStack;
|
typedef Stack<oop, mtGC> GCOopStack;
|
||||||
|
|
||||||
_overflow_stacks = NEW_C_HEAP_ARRAY(GCOopStack, ParallelGCThreads, mtGC);
|
_overflow_stacks = NEW_C_HEAP_ARRAY(GCOopStack, ParallelGCThreads, mtGC);
|
||||||
@ -742,7 +728,7 @@ class ParNewRefProcTaskProxy: public AbstractGangTask {
|
|||||||
typedef AbstractRefProcTaskExecutor::ProcessTask ProcessTask;
|
typedef AbstractRefProcTaskExecutor::ProcessTask ProcessTask;
|
||||||
public:
|
public:
|
||||||
ParNewRefProcTaskProxy(ProcessTask& task,
|
ParNewRefProcTaskProxy(ProcessTask& task,
|
||||||
ParNewGeneration& gen,
|
ParNewGeneration& young_gen,
|
||||||
Generation& old_gen,
|
Generation& old_gen,
|
||||||
HeapWord* young_old_boundary,
|
HeapWord* young_old_boundary,
|
||||||
ParScanThreadStateSet& state_set);
|
ParScanThreadStateSet& state_set);
|
||||||
@ -768,11 +754,9 @@ ParNewRefProcTaskProxy::ParNewRefProcTaskProxy(ProcessTask& task,
|
|||||||
_old_gen(old_gen),
|
_old_gen(old_gen),
|
||||||
_young_old_boundary(young_old_boundary),
|
_young_old_boundary(young_old_boundary),
|
||||||
_state_set(state_set)
|
_state_set(state_set)
|
||||||
{
|
{ }
|
||||||
}
|
|
||||||
|
|
||||||
void ParNewRefProcTaskProxy::work(uint worker_id)
|
void ParNewRefProcTaskProxy::work(uint worker_id) {
|
||||||
{
|
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
HandleMark hm;
|
HandleMark hm;
|
||||||
ParScanThreadState& par_scan_state = _state_set.thread_state(worker_id);
|
ParScanThreadState& par_scan_state = _state_set.thread_state(worker_id);
|
||||||
@ -792,15 +776,12 @@ public:
|
|||||||
_task(task)
|
_task(task)
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
virtual void work(uint worker_id)
|
virtual void work(uint worker_id) {
|
||||||
{
|
|
||||||
_task.work(worker_id);
|
_task.work(worker_id);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void ParNewRefProcTaskExecutor::execute(ProcessTask& task) {
|
||||||
void ParNewRefProcTaskExecutor::execute(ProcessTask& task)
|
|
||||||
{
|
|
||||||
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
||||||
WorkGang* workers = gch->workers();
|
WorkGang* workers = gch->workers();
|
||||||
assert(workers != NULL, "Need parallel worker threads.");
|
assert(workers != NULL, "Need parallel worker threads.");
|
||||||
@ -812,8 +793,7 @@ void ParNewRefProcTaskExecutor::execute(ProcessTask& task)
|
|||||||
_young_gen.promotion_failed());
|
_young_gen.promotion_failed());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParNewRefProcTaskExecutor::execute(EnqueueTask& task)
|
void ParNewRefProcTaskExecutor::execute(EnqueueTask& task) {
|
||||||
{
|
|
||||||
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
||||||
WorkGang* workers = gch->workers();
|
WorkGang* workers = gch->workers();
|
||||||
assert(workers != NULL, "Need parallel worker threads.");
|
assert(workers != NULL, "Need parallel worker threads.");
|
||||||
@ -821,8 +801,7 @@ void ParNewRefProcTaskExecutor::execute(EnqueueTask& task)
|
|||||||
workers->run_task(&enq_task);
|
workers->run_task(&enq_task);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParNewRefProcTaskExecutor::set_single_threaded_mode()
|
void ParNewRefProcTaskExecutor::set_single_threaded_mode() {
|
||||||
{
|
|
||||||
_state_set.flush();
|
_state_set.flush();
|
||||||
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
GenCollectedHeap* gch = GenCollectedHeap::heap();
|
||||||
gch->save_marks();
|
gch->save_marks();
|
||||||
@ -830,7 +809,8 @@ void ParNewRefProcTaskExecutor::set_single_threaded_mode()
|
|||||||
|
|
||||||
ScanClosureWithParBarrier::
|
ScanClosureWithParBarrier::
|
||||||
ScanClosureWithParBarrier(ParNewGeneration* g, bool gc_barrier) :
|
ScanClosureWithParBarrier(ParNewGeneration* g, bool gc_barrier) :
|
||||||
ScanClosure(g, gc_barrier) {}
|
ScanClosure(g, gc_barrier)
|
||||||
|
{ }
|
||||||
|
|
||||||
EvacuateFollowersClosureGeneral::
|
EvacuateFollowersClosureGeneral::
|
||||||
EvacuateFollowersClosureGeneral(GenCollectedHeap* gch,
|
EvacuateFollowersClosureGeneral(GenCollectedHeap* gch,
|
||||||
@ -838,7 +818,7 @@ EvacuateFollowersClosureGeneral(GenCollectedHeap* gch,
|
|||||||
OopsInGenClosure* older) :
|
OopsInGenClosure* older) :
|
||||||
_gch(gch),
|
_gch(gch),
|
||||||
_scan_cur_or_nonheap(cur), _scan_older(older)
|
_scan_cur_or_nonheap(cur), _scan_older(older)
|
||||||
{}
|
{ }
|
||||||
|
|
||||||
void EvacuateFollowersClosureGeneral::do_void() {
|
void EvacuateFollowersClosureGeneral::do_void() {
|
||||||
do {
|
do {
|
||||||
@ -850,7 +830,6 @@ void EvacuateFollowersClosureGeneral::do_void() {
|
|||||||
} while (!_gch->no_allocs_since_save_marks());
|
} while (!_gch->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(GenCollectedHeap* gch, ParScanThreadStateSet& thread_state_set) {
|
||||||
@ -996,9 +975,9 @@ void ParNewGeneration::collect(bool full,
|
|||||||
if (ZapUnusedHeapArea) {
|
if (ZapUnusedHeapArea) {
|
||||||
// This is now done here because of the piece-meal mangling which
|
// This is now done here because of the piece-meal mangling which
|
||||||
// can check for valid mangling at intermediate points in the
|
// can check for valid mangling at intermediate points in the
|
||||||
// collection(s). When a minor collection fails to collect
|
// collection(s). When a young collection fails to collect
|
||||||
// sufficient space resizing of the young generation can occur
|
// sufficient space resizing of the young generation can occur
|
||||||
// an redistribute the spaces in the young generation. Mangle
|
// and redistribute the spaces in the young generation. Mangle
|
||||||
// here so that unzapped regions don't get distributed to
|
// here so that unzapped regions don't get distributed to
|
||||||
// other spaces.
|
// other spaces.
|
||||||
to()->mangle_unused_area();
|
to()->mangle_unused_area();
|
||||||
@ -1113,8 +1092,10 @@ void ParNewGeneration::preserve_mark_if_necessary(oop obj, markOop m) {
|
|||||||
// thus avoiding the need to undo the copy as in
|
// thus avoiding the need to undo the copy as in
|
||||||
// copy_to_survivor_space_avoiding_with_undo.
|
// copy_to_survivor_space_avoiding_with_undo.
|
||||||
|
|
||||||
oop ParNewGeneration::copy_to_survivor_space(
|
oop ParNewGeneration::copy_to_survivor_space(ParScanThreadState* par_scan_state,
|
||||||
ParScanThreadState* par_scan_state, oop old, size_t sz, markOop m) {
|
oop old,
|
||||||
|
size_t sz,
|
||||||
|
markOop m) {
|
||||||
// In the sequential version, this assert also says that the object is
|
// In the sequential version, this assert also says that the object is
|
||||||
// not forwarded. That might not be the case here. It is the case that
|
// not forwarded. That might not be the case here. It is the case that
|
||||||
// the caller observed it to be not forwarded at some time in the past.
|
// the caller observed it to be not forwarded at some time in the past.
|
||||||
@ -1141,8 +1122,7 @@ oop ParNewGeneration::copy_to_survivor_space(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (new_obj == NULL) {
|
if (new_obj == NULL) {
|
||||||
// Either to-space is full or we decided to promote
|
// Either to-space is full or we decided to promote try allocating obj tenured
|
||||||
// try allocating obj tenured
|
|
||||||
|
|
||||||
// Attempt to install a null forwarding pointer (atomically),
|
// Attempt to install a null forwarding pointer (atomically),
|
||||||
// to claim the right to install the real forwarding pointer.
|
// to claim the right to install the real forwarding pointer.
|
||||||
|
@ -71,11 +71,7 @@ class ParScanThreadState {
|
|||||||
ParScanWithoutBarrierClosure _to_space_closure; // scan_without_gc_barrier
|
ParScanWithoutBarrierClosure _to_space_closure; // scan_without_gc_barrier
|
||||||
ParScanWithBarrierClosure _old_gen_closure; // scan_with_gc_barrier
|
ParScanWithBarrierClosure _old_gen_closure; // scan_with_gc_barrier
|
||||||
ParRootScanWithoutBarrierClosure _to_space_root_closure; // scan_root_without_gc_barrier
|
ParRootScanWithoutBarrierClosure _to_space_root_closure; // scan_root_without_gc_barrier
|
||||||
// One of these two will be passed to process_roots, which will
|
// Will be passed to process_roots to set its generation.
|
||||||
// set its generation. The first is for two-gen configs where the
|
|
||||||
// old gen collects the perm gen; the second is for arbitrary configs.
|
|
||||||
// The second isn't used right now (it used to be used for the train, an
|
|
||||||
// incremental collector) but the declaration has been left as a reminder.
|
|
||||||
ParRootScanWithBarrierTwoGensClosure _older_gen_closure;
|
ParRootScanWithBarrierTwoGensClosure _older_gen_closure;
|
||||||
// This closure will always be bound to the old gen; it will be used
|
// This closure will always be bound to the old gen; it will be used
|
||||||
// in evacuate_followers.
|
// in evacuate_followers.
|
||||||
@ -85,7 +81,6 @@ class ParScanThreadState {
|
|||||||
ParScanWeakRefClosure _scan_weak_ref_closure;
|
ParScanWeakRefClosure _scan_weak_ref_closure;
|
||||||
ParKeepAliveClosure _keep_alive_closure;
|
ParKeepAliveClosure _keep_alive_closure;
|
||||||
|
|
||||||
|
|
||||||
Space* _to_space;
|
Space* _to_space;
|
||||||
Space* to_space() { return _to_space; }
|
Space* to_space() { return _to_space; }
|
||||||
|
|
||||||
|
@ -35,7 +35,7 @@ private:
|
|||||||
// We encode the value of the heap region type so the generation can be
|
// We encode the value of the heap region type so the generation can be
|
||||||
// determined quickly. The tag is split into two parts:
|
// determined quickly. The tag is split into two parts:
|
||||||
//
|
//
|
||||||
// major type (young, humongous) : top N-1 bits
|
// major type (young, old, humongous, archive) : top N-1 bits
|
||||||
// minor type (eden / survivor, starts / cont hum, etc.) : bottom 1 bit
|
// minor type (eden / survivor, starts / cont hum, etc.) : bottom 1 bit
|
||||||
//
|
//
|
||||||
// If there's need to increase the number of minor types in the
|
// If there's need to increase the number of minor types in the
|
||||||
|
@ -30,26 +30,22 @@
|
|||||||
#include "gc/parallel/psParallelCompact.hpp"
|
#include "gc/parallel/psParallelCompact.hpp"
|
||||||
#include "gc/parallel/psScavenge.hpp"
|
#include "gc/parallel/psScavenge.hpp"
|
||||||
|
|
||||||
inline size_t ParallelScavengeHeap::total_invocations()
|
inline size_t ParallelScavengeHeap::total_invocations() {
|
||||||
{
|
|
||||||
return UseParallelOldGC ? PSParallelCompact::total_invocations() :
|
return UseParallelOldGC ? PSParallelCompact::total_invocations() :
|
||||||
PSMarkSweep::total_invocations();
|
PSMarkSweep::total_invocations();
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool ParallelScavengeHeap::should_alloc_in_eden(const size_t size) const
|
inline bool ParallelScavengeHeap::should_alloc_in_eden(const size_t size) const {
|
||||||
{
|
|
||||||
const size_t eden_size = young_gen()->eden_space()->capacity_in_words();
|
const size_t eden_size = young_gen()->eden_space()->capacity_in_words();
|
||||||
return size < eden_size / 2;
|
return size < eden_size / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void ParallelScavengeHeap::invoke_scavenge()
|
inline void ParallelScavengeHeap::invoke_scavenge() {
|
||||||
{
|
|
||||||
PSScavenge::invoke();
|
PSScavenge::invoke();
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool ParallelScavengeHeap::is_in_young(oop p) {
|
inline bool ParallelScavengeHeap::is_in_young(oop p) {
|
||||||
// Assumes the the old gen address range is lower than that of the young gen.
|
// Assumes the the old gen address range is lower than that of the young gen.
|
||||||
const void* loc = (void*) p;
|
|
||||||
bool result = ((HeapWord*)p) >= young_gen()->reserved().start();
|
bool result = ((HeapWord*)p) >= young_gen()->reserved().start();
|
||||||
assert(result == young_gen()->is_in_reserved(p),
|
assert(result == young_gen()->is_in_reserved(p),
|
||||||
err_msg("incorrect test - result=%d, p=" PTR_FORMAT, result, p2i((void*)p)));
|
err_msg("incorrect test - result=%d, p=" PTR_FORMAT, result, p2i((void*)p)));
|
||||||
|
@ -299,7 +299,7 @@ void PSAdaptiveSizePolicy::compute_eden_space_size(
|
|||||||
// subtracted out.
|
// subtracted out.
|
||||||
size_t eden_limit = max_eden_size;
|
size_t eden_limit = max_eden_size;
|
||||||
|
|
||||||
const double gc_cost_limit = GCTimeLimit/100.0;
|
const double gc_cost_limit = GCTimeLimit / 100.0;
|
||||||
|
|
||||||
// Which way should we go?
|
// Which way should we go?
|
||||||
// if pause requirement is not met
|
// if pause requirement is not met
|
||||||
|
@ -486,12 +486,12 @@ void PSOldGen::verify() {
|
|||||||
object_space()->verify();
|
object_space()->verify();
|
||||||
}
|
}
|
||||||
class VerifyObjectStartArrayClosure : public ObjectClosure {
|
class VerifyObjectStartArrayClosure : public ObjectClosure {
|
||||||
PSOldGen* _gen;
|
PSOldGen* _old_gen;
|
||||||
ObjectStartArray* _start_array;
|
ObjectStartArray* _start_array;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
VerifyObjectStartArrayClosure(PSOldGen* gen, ObjectStartArray* start_array) :
|
VerifyObjectStartArrayClosure(PSOldGen* old_gen, ObjectStartArray* start_array) :
|
||||||
_gen(gen), _start_array(start_array) { }
|
_old_gen(old_gen), _start_array(start_array) { }
|
||||||
|
|
||||||
virtual void do_object(oop obj) {
|
virtual void do_object(oop obj) {
|
||||||
HeapWord* test_addr = (HeapWord*)obj + 1;
|
HeapWord* test_addr = (HeapWord*)obj + 1;
|
||||||
|
@ -958,7 +958,7 @@ void PSParallelCompact::pre_compact(PreGCValues* pre_gc_values)
|
|||||||
{
|
{
|
||||||
// Update the from & to space pointers in space_info, since they are swapped
|
// Update the from & to space pointers in space_info, since they are swapped
|
||||||
// at each young gen gc. Do the update unconditionally (even though a
|
// at each young gen gc. Do the update unconditionally (even though a
|
||||||
// promotion failure does not swap spaces) because an unknown number of minor
|
// promotion failure does not swap spaces) because an unknown number of young
|
||||||
// collections will have swapped the spaces an unknown number of times.
|
// collections will have swapped the spaces an unknown number of times.
|
||||||
GCTraceTime tm("pre compact", print_phases(), true, &_gc_timer, _gc_tracer.gc_id());
|
GCTraceTime tm("pre compact", print_phases(), true, &_gc_timer, _gc_tracer.gc_id());
|
||||||
ParallelScavengeHeap* heap = ParallelScavengeHeap::heap();
|
ParallelScavengeHeap* heap = ParallelScavengeHeap::heap();
|
||||||
|
@ -979,7 +979,6 @@ class PSParallelCompact : AllStatic {
|
|||||||
static bool _dwl_initialized;
|
static bool _dwl_initialized;
|
||||||
#endif // #ifdef ASSERT
|
#endif // #ifdef ASSERT
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static ParallelOldTracer* gc_tracer() { return &_gc_tracer; }
|
static ParallelOldTracer* gc_tracer() { return &_gc_tracer; }
|
||||||
|
|
||||||
|
@ -597,9 +597,9 @@ bool PSScavenge::invoke_no_policy() {
|
|||||||
// to allow resizes that may have been inhibited by the
|
// to allow resizes that may have been inhibited by the
|
||||||
// relative location of the "to" and "from" spaces.
|
// relative location of the "to" and "from" spaces.
|
||||||
|
|
||||||
// Resizing the old gen at minor collects can cause increases
|
// Resizing the old gen at young collections can cause increases
|
||||||
// that don't feed back to the generation sizing policy until
|
// that don't feed back to the generation sizing policy until
|
||||||
// a major collection. Don't resize the old gen here.
|
// a full collection. Don't resize the old gen here.
|
||||||
|
|
||||||
heap->resize_young_gen(size_policy->calculated_eden_size_in_bytes(),
|
heap->resize_young_gen(size_policy->calculated_eden_size_in_bytes(),
|
||||||
size_policy->calculated_survivor_size_in_bytes());
|
size_policy->calculated_survivor_size_in_bytes());
|
||||||
|
@ -172,10 +172,10 @@ void StealTask::do_it(GCTaskManager* manager, uint which) {
|
|||||||
|
|
||||||
void OldToYoungRootsTask::do_it(GCTaskManager* manager, uint which) {
|
void OldToYoungRootsTask::do_it(GCTaskManager* manager, uint which) {
|
||||||
// There are not old-to-young pointers if the old gen is empty.
|
// There are not old-to-young pointers if the old gen is empty.
|
||||||
assert(!_gen->object_space()->is_empty(),
|
assert(!_old_gen->object_space()->is_empty(),
|
||||||
"Should not be called is there is no work");
|
"Should not be called is there is no work");
|
||||||
assert(_gen != NULL, "Sanity");
|
assert(_old_gen != NULL, "Sanity");
|
||||||
assert(_gen->object_space()->contains(_gen_top) || _gen_top == _gen->object_space()->top(), "Sanity");
|
assert(_old_gen->object_space()->contains(_gen_top) || _gen_top == _old_gen->object_space()->top(), "Sanity");
|
||||||
assert(_stripe_number < ParallelGCThreads, "Sanity");
|
assert(_stripe_number < ParallelGCThreads, "Sanity");
|
||||||
|
|
||||||
{
|
{
|
||||||
@ -183,8 +183,8 @@ void OldToYoungRootsTask::do_it(GCTaskManager* manager, uint which) {
|
|||||||
CardTableExtension* card_table =
|
CardTableExtension* card_table =
|
||||||
barrier_set_cast<CardTableExtension>(ParallelScavengeHeap::heap()->barrier_set());
|
barrier_set_cast<CardTableExtension>(ParallelScavengeHeap::heap()->barrier_set());
|
||||||
|
|
||||||
card_table->scavenge_contents_parallel(_gen->start_array(),
|
card_table->scavenge_contents_parallel(_old_gen->start_array(),
|
||||||
_gen->object_space(),
|
_old_gen->object_space(),
|
||||||
_gen_top,
|
_gen_top,
|
||||||
pm,
|
pm,
|
||||||
_stripe_number,
|
_stripe_number,
|
||||||
|
@ -160,17 +160,17 @@ class StealTask : public GCTask {
|
|||||||
|
|
||||||
class OldToYoungRootsTask : public GCTask {
|
class OldToYoungRootsTask : public GCTask {
|
||||||
private:
|
private:
|
||||||
PSOldGen* _gen;
|
PSOldGen* _old_gen;
|
||||||
HeapWord* _gen_top;
|
HeapWord* _gen_top;
|
||||||
uint _stripe_number;
|
uint _stripe_number;
|
||||||
uint _stripe_total;
|
uint _stripe_total;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
OldToYoungRootsTask(PSOldGen *gen,
|
OldToYoungRootsTask(PSOldGen *old_gen,
|
||||||
HeapWord* gen_top,
|
HeapWord* gen_top,
|
||||||
uint stripe_number,
|
uint stripe_number,
|
||||||
uint stripe_total) :
|
uint stripe_total) :
|
||||||
_gen(gen),
|
_old_gen(old_gen),
|
||||||
_gen_top(gen_top),
|
_gen_top(gen_top),
|
||||||
_stripe_number(stripe_number),
|
_stripe_number(stripe_number),
|
||||||
_stripe_total(stripe_total) { }
|
_stripe_total(stripe_total) { }
|
||||||
|
@ -106,14 +106,14 @@ FastEvacuateFollowersClosure(GenCollectedHeap* gch,
|
|||||||
_gch(gch), _scan_cur_or_nonheap(cur), _scan_older(older)
|
_gch(gch), _scan_cur_or_nonheap(cur), _scan_older(older)
|
||||||
{
|
{
|
||||||
assert(_gch->young_gen()->kind() == Generation::DefNew, "Generation should be DefNew");
|
assert(_gch->young_gen()->kind() == Generation::DefNew, "Generation should be DefNew");
|
||||||
_gen = (DefNewGeneration*)_gch->young_gen();
|
_young_gen = (DefNewGeneration*)_gch->young_gen();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DefNewGeneration::FastEvacuateFollowersClosure::do_void() {
|
void DefNewGeneration::FastEvacuateFollowersClosure::do_void() {
|
||||||
do {
|
do {
|
||||||
_gch->oop_since_save_marks_iterate(GenCollectedHeap::YoungGen, _scan_cur_or_nonheap, _scan_older);
|
_gch->oop_since_save_marks_iterate(GenCollectedHeap::YoungGen, _scan_cur_or_nonheap, _scan_older);
|
||||||
} while (!_gch->no_allocs_since_save_marks());
|
} while (!_gch->no_allocs_since_save_marks());
|
||||||
guarantee(_gen->promo_failure_scan_is_complete(), "Failed to finish scan");
|
guarantee(_young_gen->promo_failure_scan_is_complete(), "Failed to finish scan");
|
||||||
}
|
}
|
||||||
|
|
||||||
ScanClosure::ScanClosure(DefNewGeneration* g, bool gc_barrier) :
|
ScanClosure::ScanClosure(DefNewGeneration* g, bool gc_barrier) :
|
||||||
@ -200,8 +200,9 @@ DefNewGeneration::DefNewGeneration(ReservedSpace rs,
|
|||||||
_from_space = new ContiguousSpace();
|
_from_space = new ContiguousSpace();
|
||||||
_to_space = new ContiguousSpace();
|
_to_space = new ContiguousSpace();
|
||||||
|
|
||||||
if (_eden_space == NULL || _from_space == NULL || _to_space == NULL)
|
if (_eden_space == NULL || _from_space == NULL || _to_space == NULL) {
|
||||||
vm_exit_during_initialization("Could not allocate a new gen space");
|
vm_exit_during_initialization("Could not allocate a new gen space");
|
||||||
|
}
|
||||||
|
|
||||||
// Compute the maximum eden and survivor space sizes. These sizes
|
// Compute the maximum eden and survivor space sizes. These sizes
|
||||||
// are computed assuming the entire reserved space is committed.
|
// are computed assuming the entire reserved space is committed.
|
||||||
@ -655,7 +656,7 @@ void DefNewGeneration::collect(bool full,
|
|||||||
if (ZapUnusedHeapArea) {
|
if (ZapUnusedHeapArea) {
|
||||||
// This is now done here because of the piece-meal mangling which
|
// This is now done here because of the piece-meal mangling which
|
||||||
// can check for valid mangling at intermediate points in the
|
// can check for valid mangling at intermediate points in the
|
||||||
// collection(s). When a minor collection fails to collect
|
// collection(s). When a young collection fails to collect
|
||||||
// sufficient space resizing of the young generation can occur
|
// sufficient space resizing of the young generation can occur
|
||||||
// an redistribute the spaces in the young generation. Mangle
|
// an redistribute the spaces in the young generation. Mangle
|
||||||
// here so that unzapped regions don't get distributed to
|
// here so that unzapped regions don't get distributed to
|
||||||
|
@ -193,7 +193,7 @@ protected:
|
|||||||
|
|
||||||
class FastEvacuateFollowersClosure: public VoidClosure {
|
class FastEvacuateFollowersClosure: public VoidClosure {
|
||||||
GenCollectedHeap* _gch;
|
GenCollectedHeap* _gch;
|
||||||
DefNewGeneration* _gen;
|
DefNewGeneration* _young_gen;
|
||||||
FastScanClosure* _scan_cur_or_nonheap;
|
FastScanClosure* _scan_cur_or_nonheap;
|
||||||
FastScanClosure* _scan_older;
|
FastScanClosure* _scan_older;
|
||||||
public:
|
public:
|
||||||
|
@ -57,8 +57,8 @@ inline void DefNewGeneration::KeepAliveClosure::do_oop_work(T* p) {
|
|||||||
// each generation, allowing them in turn to examine the modified
|
// each generation, allowing them in turn to examine the modified
|
||||||
// field.
|
// field.
|
||||||
//
|
//
|
||||||
// We could check that p is also in an older generation, but
|
// We could check that p is also in the old generation, but
|
||||||
// dirty cards in the youngest gen are never scanned, so the
|
// dirty cards in the young gen are never scanned, so the
|
||||||
// extra check probably isn't worthwhile.
|
// extra check probably isn't worthwhile.
|
||||||
if (GenCollectedHeap::heap()->is_in_reserved(p)) {
|
if (GenCollectedHeap::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);
|
||||||
|
@ -108,7 +108,7 @@ bool TenuredGeneration::should_collect(bool full,
|
|||||||
free());
|
free());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If we had to expand to accommodate promotions from younger generations
|
// If we had to expand to accommodate promotions from the young generation
|
||||||
if (!result && _capacity_at_prologue < capacity()) {
|
if (!result && _capacity_at_prologue < capacity()) {
|
||||||
result = true;
|
result = true;
|
||||||
if (PrintGC && Verbose) {
|
if (PrintGC && Verbose) {
|
||||||
@ -140,11 +140,11 @@ void TenuredGeneration::update_gc_stats(Generation* current_generation,
|
|||||||
// 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 = GenCollectedHeap::heap()->is_young_gen(current_generation);
|
||||||
if (!full && current_is_young) {
|
if (!full && current_is_young) {
|
||||||
// Calculate size of data promoted from the younger generations
|
// Calculate size of data promoted from the young generation
|
||||||
// before doing the collection.
|
// before doing the collection.
|
||||||
size_t used_before_gc = used();
|
size_t used_before_gc = used();
|
||||||
|
|
||||||
// If the younger gen collections were skipped, then the
|
// If the young gen collection was skipped, then the
|
||||||
// number of promoted bytes will be 0 and adding it to the
|
// number of promoted bytes will be 0 and adding it to the
|
||||||
// average will incorrectly lessen the average. It is, however,
|
// average will incorrectly lessen the average. It is, however,
|
||||||
// also possible that no promotion was needed.
|
// also possible that no promotion was needed.
|
||||||
|
@ -54,6 +54,7 @@ class TenuredGeneration: public CardGeneration {
|
|||||||
ContiguousSpace* space() const { return _the_space; }
|
ContiguousSpace* space() const { return _the_space; }
|
||||||
|
|
||||||
void assert_correct_size_change_locking();
|
void assert_correct_size_change_locking();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TenuredGeneration(ReservedSpace rs,
|
TenuredGeneration(ReservedSpace rs,
|
||||||
size_t initial_byte_size,
|
size_t initial_byte_size,
|
||||||
@ -66,10 +67,9 @@ class TenuredGeneration: public CardGeneration {
|
|||||||
const char* short_name() const { return "Tenured"; }
|
const char* short_name() const { return "Tenured"; }
|
||||||
|
|
||||||
// Does a "full" (forced) collection invoked on this generation collect
|
// Does a "full" (forced) collection invoked on this generation collect
|
||||||
// all younger generations as well? Note that this is a
|
// the young generation as well? Note that this is a hack to allow the
|
||||||
// hack to allow the collection of the younger gen first if the flag is
|
// collection of the young gen first if the flag is set.
|
||||||
// set.
|
virtual bool full_collects_young_generation() const {
|
||||||
virtual bool full_collects_younger_generations() const {
|
|
||||||
return !ScavengeBeforeFullGC;
|
return !ScavengeBeforeFullGC;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,15 +99,16 @@ class TenuredGeneration: public CardGeneration {
|
|||||||
bool clear_all_soft_refs,
|
bool clear_all_soft_refs,
|
||||||
size_t size,
|
size_t size,
|
||||||
bool is_tlab);
|
bool is_tlab);
|
||||||
|
|
||||||
HeapWord* expand_and_allocate(size_t size,
|
HeapWord* expand_and_allocate(size_t size,
|
||||||
bool is_tlab,
|
bool is_tlab,
|
||||||
bool parallel = false);
|
bool parallel = false);
|
||||||
|
|
||||||
virtual void prepare_for_verify();
|
virtual void prepare_for_verify();
|
||||||
|
|
||||||
|
|
||||||
virtual void gc_prologue(bool full);
|
virtual void gc_prologue(bool full);
|
||||||
virtual void gc_epilogue(bool full);
|
virtual void gc_epilogue(bool full);
|
||||||
|
|
||||||
bool should_collect(bool full,
|
bool should_collect(bool full,
|
||||||
size_t word_size,
|
size_t word_size,
|
||||||
bool is_tlab);
|
bool is_tlab);
|
||||||
|
@ -266,12 +266,12 @@ void AdaptiveSizePolicy::minor_collection_end(GCCause::Cause gc_cause) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// The policy does not have enough data until at least some
|
// The policy does not have enough data until at least some
|
||||||
// minor collections have been done.
|
// young collections have been done.
|
||||||
_young_gen_policy_is_ready =
|
_young_gen_policy_is_ready =
|
||||||
(_avg_minor_gc_cost->count() >= AdaptiveSizePolicyReadyThreshold);
|
(_avg_minor_gc_cost->count() >= AdaptiveSizePolicyReadyThreshold);
|
||||||
|
|
||||||
// Calculate variables used to estimate pause time vs. gen sizes
|
// Calculate variables used to estimate pause time vs. gen sizes
|
||||||
double eden_size_in_mbytes = ((double)_eden_size)/((double)M);
|
double eden_size_in_mbytes = ((double)_eden_size) / ((double)M);
|
||||||
update_minor_pause_young_estimator(minor_pause_in_ms);
|
update_minor_pause_young_estimator(minor_pause_in_ms);
|
||||||
update_minor_pause_old_estimator(minor_pause_in_ms);
|
update_minor_pause_old_estimator(minor_pause_in_ms);
|
||||||
|
|
||||||
@ -295,8 +295,7 @@ void AdaptiveSizePolicy::minor_collection_end(GCCause::Cause gc_cause) {
|
|||||||
_minor_timer.start();
|
_minor_timer.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t AdaptiveSizePolicy::eden_increment(size_t cur_eden,
|
size_t AdaptiveSizePolicy::eden_increment(size_t cur_eden, uint percent_change) {
|
||||||
uint percent_change) {
|
|
||||||
size_t eden_heap_delta;
|
size_t eden_heap_delta;
|
||||||
eden_heap_delta = cur_eden / 100 * percent_change;
|
eden_heap_delta = cur_eden / 100 * percent_change;
|
||||||
return eden_heap_delta;
|
return eden_heap_delta;
|
||||||
@ -312,8 +311,7 @@ size_t AdaptiveSizePolicy::eden_decrement(size_t cur_eden) {
|
|||||||
return eden_heap_delta;
|
return eden_heap_delta;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t AdaptiveSizePolicy::promo_increment(size_t cur_promo,
|
size_t AdaptiveSizePolicy::promo_increment(size_t cur_promo, uint percent_change) {
|
||||||
uint percent_change) {
|
|
||||||
size_t promo_heap_delta;
|
size_t promo_heap_delta;
|
||||||
promo_heap_delta = cur_promo / 100 * percent_change;
|
promo_heap_delta = cur_promo / 100 * percent_change;
|
||||||
return promo_heap_delta;
|
return promo_heap_delta;
|
||||||
|
@ -80,7 +80,9 @@ jbyte CardTableRS::find_unused_youngergenP_card_value() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!seen) return v;
|
if (!seen) {
|
||||||
|
return v;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ShouldNotReachHere();
|
ShouldNotReachHere();
|
||||||
return 0;
|
return 0;
|
||||||
@ -502,7 +504,7 @@ void CardTableRS::verify_space(Space* s, HeapWord* gen_boundary) {
|
|||||||
//
|
//
|
||||||
// The main point below is that the parallel card scanning code
|
// The main point below is that the parallel card scanning code
|
||||||
// deals correctly with these stale card values. There are two main
|
// deals correctly with these stale card values. There are two main
|
||||||
// cases to consider where we have a stale "younger gen" value and a
|
// cases to consider where we have a stale "young gen" value and a
|
||||||
// "derivative" case to consider, where we have a stale
|
// "derivative" case to consider, where we have a stale
|
||||||
// "cur_younger_gen_and_prev_non_clean" value, as will become
|
// "cur_younger_gen_and_prev_non_clean" value, as will become
|
||||||
// apparent in the case analysis below.
|
// apparent in the case analysis below.
|
||||||
|
@ -160,16 +160,20 @@ void CollectedHeap::trace_heap_after_gc(const GCTracer* gc_tracer) {
|
|||||||
// Memory state functions.
|
// Memory state functions.
|
||||||
|
|
||||||
|
|
||||||
CollectedHeap::CollectedHeap() {
|
CollectedHeap::CollectedHeap() :
|
||||||
|
_barrier_set(NULL),
|
||||||
|
_is_gc_active(false),
|
||||||
|
_total_collections(0),
|
||||||
|
_total_full_collections(0),
|
||||||
|
_gc_cause(GCCause::_no_gc),
|
||||||
|
_gc_lastcause(GCCause::_no_gc),
|
||||||
|
_defer_initial_card_mark(false) // strengthened by subclass in pre_initialize() below.
|
||||||
|
{
|
||||||
const size_t max_len = size_t(arrayOopDesc::max_array_length(T_INT));
|
const size_t max_len = size_t(arrayOopDesc::max_array_length(T_INT));
|
||||||
const size_t elements_per_word = HeapWordSize / sizeof(jint);
|
const size_t elements_per_word = HeapWordSize / sizeof(jint);
|
||||||
_filler_array_max_size = align_object_size(filler_array_hdr_size() +
|
_filler_array_max_size = align_object_size(filler_array_hdr_size() +
|
||||||
max_len / elements_per_word);
|
max_len / elements_per_word);
|
||||||
|
|
||||||
_barrier_set = NULL;
|
|
||||||
_is_gc_active = false;
|
|
||||||
_total_collections = _total_full_collections = 0;
|
|
||||||
_gc_cause = _gc_lastcause = GCCause::_no_gc;
|
|
||||||
NOT_PRODUCT(_promotion_failure_alot_count = 0;)
|
NOT_PRODUCT(_promotion_failure_alot_count = 0;)
|
||||||
NOT_PRODUCT(_promotion_failure_alot_gc_number = 0;)
|
NOT_PRODUCT(_promotion_failure_alot_gc_number = 0;)
|
||||||
|
|
||||||
@ -184,7 +188,7 @@ CollectedHeap::CollectedHeap() {
|
|||||||
PerfDataManager::create_string_variable(SUN_GC, "lastCause",
|
PerfDataManager::create_string_variable(SUN_GC, "lastCause",
|
||||||
80, GCCause::to_string(_gc_lastcause), CHECK);
|
80, GCCause::to_string(_gc_lastcause), CHECK);
|
||||||
}
|
}
|
||||||
_defer_initial_card_mark = false; // strengthened by subclass in pre_initialize() below.
|
|
||||||
// Create the ring log
|
// Create the ring log
|
||||||
if (LogEvents) {
|
if (LogEvents) {
|
||||||
_gc_heap_log = new GCHeapLog();
|
_gc_heap_log = new GCHeapLog();
|
||||||
@ -570,8 +574,8 @@ void CollectedHeap::resize_all_tlabs() {
|
|||||||
void CollectedHeap::pre_full_gc_dump(GCTimer* timer) {
|
void CollectedHeap::pre_full_gc_dump(GCTimer* timer) {
|
||||||
if (HeapDumpBeforeFullGC) {
|
if (HeapDumpBeforeFullGC) {
|
||||||
GCTraceTime tt("Heap Dump (before full gc): ", PrintGCDetails, false, timer, GCId::create());
|
GCTraceTime tt("Heap Dump (before full gc): ", PrintGCDetails, false, timer, GCId::create());
|
||||||
// We are doing a "major" collection and a heap dump before
|
// We are doing a full collection and a heap dump before
|
||||||
// major collection has been requested.
|
// full collection has been requested.
|
||||||
HeapDumper::dump_heap();
|
HeapDumper::dump_heap();
|
||||||
}
|
}
|
||||||
if (PrintClassHistogramBeforeFullGC) {
|
if (PrintClassHistogramBeforeFullGC) {
|
||||||
|
@ -464,7 +464,7 @@ void GenCollectedHeap::do_collection(bool full,
|
|||||||
bool prepared_for_verification = false;
|
bool prepared_for_verification = false;
|
||||||
bool collected_old = false;
|
bool collected_old = false;
|
||||||
bool old_collects_young = complete &&
|
bool old_collects_young = complete &&
|
||||||
_old_gen->full_collects_younger_generations();
|
_old_gen->full_collects_young_generation();
|
||||||
if (!old_collects_young &&
|
if (!old_collects_young &&
|
||||||
_young_gen->should_collect(full, size, is_tlab)) {
|
_young_gen->should_collect(full, size, is_tlab)) {
|
||||||
if (run_verification && VerifyGCLevel <= 0 && VerifyBeforeGC) {
|
if (run_verification && VerifyGCLevel <= 0 && VerifyBeforeGC) {
|
||||||
@ -521,7 +521,7 @@ void GenCollectedHeap::do_collection(bool full,
|
|||||||
// a whole heap collection.
|
// a whole heap collection.
|
||||||
complete = complete || collected_old;
|
complete = complete || collected_old;
|
||||||
|
|
||||||
if (complete) { // We did a "major" collection
|
if (complete) { // We did a full collection
|
||||||
// FIXME: See comment at pre_full_gc_dump call
|
// FIXME: See comment at pre_full_gc_dump call
|
||||||
post_full_gc_dump(NULL); // do any post full gc dumps
|
post_full_gc_dump(NULL); // do any post full gc dumps
|
||||||
}
|
}
|
||||||
@ -668,13 +668,13 @@ void GenCollectedHeap::process_roots(StrongRootsScope* scope,
|
|||||||
|
|
||||||
void GenCollectedHeap::gen_process_roots(StrongRootsScope* scope,
|
void GenCollectedHeap::gen_process_roots(StrongRootsScope* scope,
|
||||||
GenerationType type,
|
GenerationType type,
|
||||||
bool younger_gens_as_roots,
|
bool young_gen_as_roots,
|
||||||
ScanningOption so,
|
ScanningOption so,
|
||||||
bool only_strong_roots,
|
bool only_strong_roots,
|
||||||
OopsInGenClosure* not_older_gens,
|
OopsInGenClosure* not_older_gens,
|
||||||
OopsInGenClosure* older_gens,
|
OopsInGenClosure* older_gens,
|
||||||
CLDClosure* cld_closure) {
|
CLDClosure* cld_closure) {
|
||||||
const bool is_adjust_phase = !only_strong_roots && !younger_gens_as_roots;
|
const bool is_adjust_phase = !only_strong_roots && !young_gen_as_roots;
|
||||||
|
|
||||||
bool is_moving_collection = false;
|
bool is_moving_collection = false;
|
||||||
if (type == YoungGen || is_adjust_phase) {
|
if (type == YoungGen || is_adjust_phase) {
|
||||||
@ -691,7 +691,7 @@ void GenCollectedHeap::gen_process_roots(StrongRootsScope* scope,
|
|||||||
cld_closure, weak_cld_closure,
|
cld_closure, weak_cld_closure,
|
||||||
&mark_code_closure);
|
&mark_code_closure);
|
||||||
|
|
||||||
if (younger_gens_as_roots) {
|
if (young_gen_as_roots) {
|
||||||
if (!_process_strong_tasks->is_task_claimed(GCH_PS_younger_gens)) {
|
if (!_process_strong_tasks->is_task_claimed(GCH_PS_younger_gens)) {
|
||||||
if (type == OldGen) {
|
if (type == OldGen) {
|
||||||
not_older_gens->set_generation(_young_gen);
|
not_older_gens->set_generation(_young_gen);
|
||||||
@ -763,25 +763,25 @@ HeapWord** GenCollectedHeap::end_addr() const {
|
|||||||
void GenCollectedHeap::collect(GCCause::Cause cause) {
|
void GenCollectedHeap::collect(GCCause::Cause cause) {
|
||||||
if (should_do_concurrent_full_gc(cause)) {
|
if (should_do_concurrent_full_gc(cause)) {
|
||||||
#if INCLUDE_ALL_GCS
|
#if INCLUDE_ALL_GCS
|
||||||
// mostly concurrent full collection
|
// Mostly concurrent full collection.
|
||||||
collect_mostly_concurrent(cause);
|
collect_mostly_concurrent(cause);
|
||||||
#else // INCLUDE_ALL_GCS
|
#else // INCLUDE_ALL_GCS
|
||||||
ShouldNotReachHere();
|
ShouldNotReachHere();
|
||||||
#endif // INCLUDE_ALL_GCS
|
#endif // INCLUDE_ALL_GCS
|
||||||
} else if (cause == GCCause::_wb_young_gc) {
|
} else if (cause == GCCause::_wb_young_gc) {
|
||||||
// minor collection for WhiteBox API
|
// Young collection for the WhiteBox API.
|
||||||
collect(cause, YoungGen);
|
collect(cause, YoungGen);
|
||||||
} else {
|
} else {
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
if (cause == GCCause::_scavenge_alot) {
|
if (cause == GCCause::_scavenge_alot) {
|
||||||
// minor collection only
|
// Young collection only.
|
||||||
collect(cause, YoungGen);
|
collect(cause, YoungGen);
|
||||||
} else {
|
} else {
|
||||||
// Stop-the-world full collection
|
// Stop-the-world full collection.
|
||||||
collect(cause, OldGen);
|
collect(cause, OldGen);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
// Stop-the-world full collection
|
// Stop-the-world full collection.
|
||||||
collect(cause, OldGen);
|
collect(cause, OldGen);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -173,8 +173,7 @@ public:
|
|||||||
|
|
||||||
size_t max_capacity() const;
|
size_t max_capacity() const;
|
||||||
|
|
||||||
HeapWord* mem_allocate(size_t size,
|
HeapWord* mem_allocate(size_t size, bool* gc_overhead_limit_was_exceeded);
|
||||||
bool* gc_overhead_limit_was_exceeded);
|
|
||||||
|
|
||||||
// We may support a shared contiguous allocation area, if the youngest
|
// We may support a shared contiguous allocation area, if the youngest
|
||||||
// generation does.
|
// generation does.
|
||||||
@ -403,7 +402,7 @@ public:
|
|||||||
|
|
||||||
void gen_process_roots(StrongRootsScope* scope,
|
void gen_process_roots(StrongRootsScope* scope,
|
||||||
GenerationType type,
|
GenerationType type,
|
||||||
bool younger_gens_as_roots,
|
bool young_gen_as_roots,
|
||||||
ScanningOption so,
|
ScanningOption so,
|
||||||
bool only_strong_roots,
|
bool only_strong_roots,
|
||||||
OopsInGenClosure* not_older_gens,
|
OopsInGenClosure* not_older_gens,
|
||||||
|
@ -110,13 +110,11 @@ public:
|
|||||||
virtual void print() {}
|
virtual void print() {}
|
||||||
|
|
||||||
// Informs the RS that the given memregion contains no references to
|
// Informs the RS that the given memregion contains no references to
|
||||||
// younger generations.
|
// the young generation.
|
||||||
virtual void clear(MemRegion mr) = 0;
|
virtual void clear(MemRegion mr) = 0;
|
||||||
|
|
||||||
// Informs the RS that there are no references to generations
|
// Informs the RS that there are no references to the young generation
|
||||||
// younger than gen from generations gen and older.
|
// from old_gen.
|
||||||
// The parameter clear_perm indicates if the perm_gen's
|
|
||||||
// remembered set should also be processed/cleared.
|
|
||||||
virtual void clear_into_younger(Generation* old_gen) = 0;
|
virtual void clear_into_younger(Generation* old_gen) = 0;
|
||||||
|
|
||||||
// Informs the RS that refs in the given "mr" may have changed
|
// Informs the RS that refs in the given "mr" may have changed
|
||||||
|
@ -80,7 +80,6 @@ struct ScratchBlock {
|
|||||||
// first two fields are word-sized.)
|
// first two fields are word-sized.)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
class Generation: public CHeapObj<mtGC> {
|
class Generation: public CHeapObj<mtGC> {
|
||||||
friend class VMStructs;
|
friend class VMStructs;
|
||||||
private:
|
private:
|
||||||
@ -299,8 +298,7 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
// word of "obj" may have been overwritten with a forwarding pointer, and
|
// word of "obj" may have been overwritten with a forwarding pointer, and
|
||||||
// also taking care to copy the klass pointer *last*. Returns the new
|
// also taking care to copy the klass pointer *last*. Returns the new
|
||||||
// object if successful, or else NULL.
|
// object if successful, or else NULL.
|
||||||
virtual oop par_promote(int thread_num,
|
virtual oop par_promote(int thread_num, oop obj, markOop m, size_t word_sz);
|
||||||
oop obj, markOop m, size_t word_sz);
|
|
||||||
|
|
||||||
// Informs the current generation that all par_promote_alloc's in the
|
// Informs the current generation that all par_promote_alloc's in the
|
||||||
// collection have been completed; any supporting data structures can be
|
// collection have been completed; any supporting data structures can be
|
||||||
@ -315,7 +313,7 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
|
|
||||||
// This generation will collect all younger generations
|
// This generation will collect all younger generations
|
||||||
// during a full collection.
|
// during a full collection.
|
||||||
virtual bool full_collects_younger_generations() const { return false; }
|
virtual bool full_collects_young_generation() const { return false; }
|
||||||
|
|
||||||
// This generation does in-place marking, meaning that mark words
|
// This generation does in-place marking, meaning that mark words
|
||||||
// are mutated during the marking phase and presumably reinitialized
|
// are mutated during the marking phase and presumably reinitialized
|
||||||
@ -370,18 +368,18 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
|
|
||||||
// Some generations may require some cleanup or preparation actions before
|
// Some generations may require some cleanup or preparation actions before
|
||||||
// allowing a collection. The default is to do nothing.
|
// allowing a collection. The default is to do nothing.
|
||||||
virtual void gc_prologue(bool full) {};
|
virtual void gc_prologue(bool full) {}
|
||||||
|
|
||||||
// Some generations may require some cleanup actions after a collection.
|
// Some generations may require some cleanup actions after a collection.
|
||||||
// The default is to do nothing.
|
// The default is to do nothing.
|
||||||
virtual void gc_epilogue(bool full) {};
|
virtual void gc_epilogue(bool full) {}
|
||||||
|
|
||||||
// Save the high water marks for the used space in a generation.
|
// Save the high water marks for the used space in a generation.
|
||||||
virtual void record_spaces_top() {};
|
virtual void record_spaces_top() {}
|
||||||
|
|
||||||
// Some generations may need to be "fixed-up" after some allocation
|
// Some generations may need to be "fixed-up" after some allocation
|
||||||
// activity to make them parsable again. The default is to do nothing.
|
// activity to make them parsable again. The default is to do nothing.
|
||||||
virtual void ensure_parsability() {};
|
virtual void ensure_parsability() {}
|
||||||
|
|
||||||
// Time (in ms) when we were last collected or now if a collection is
|
// Time (in ms) when we were last collected or now if a collection is
|
||||||
// in progress.
|
// in progress.
|
||||||
@ -417,7 +415,7 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
virtual void adjust_pointers();
|
virtual void adjust_pointers();
|
||||||
// Mark sweep support phase4
|
// Mark sweep support phase4
|
||||||
virtual void compact();
|
virtual void compact();
|
||||||
virtual void post_compact() {ShouldNotReachHere();}
|
virtual void post_compact() { ShouldNotReachHere(); }
|
||||||
|
|
||||||
// Support for CMS's rescan. In this general form we return a pointer
|
// Support for CMS's rescan. In this general form we return a pointer
|
||||||
// to an abstract object that can be used, based on specific previously
|
// to an abstract object that can be used, based on specific previously
|
||||||
@ -432,7 +430,7 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
|
|
||||||
// Some generations may require some cleanup actions before allowing
|
// Some generations may require some cleanup actions before allowing
|
||||||
// a verification.
|
// a verification.
|
||||||
virtual void prepare_for_verify() {};
|
virtual void prepare_for_verify() {}
|
||||||
|
|
||||||
// Accessing "marks".
|
// Accessing "marks".
|
||||||
|
|
||||||
@ -483,7 +481,7 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
|
|
||||||
// Give each generation an opportunity to do clean up for any
|
// Give each generation an opportunity to do clean up for any
|
||||||
// contributed scratch.
|
// contributed scratch.
|
||||||
virtual void reset_scratch() {};
|
virtual void reset_scratch() {}
|
||||||
|
|
||||||
// When an older generation has been collected, and perhaps resized,
|
// When an older generation has been collected, and perhaps resized,
|
||||||
// this method will be invoked on all younger generations (from older to
|
// this method will be invoked on all younger generations (from older to
|
||||||
|
@ -1065,7 +1065,7 @@ bool ReferenceProcessor::discover_reference(oop obj, ReferenceType rt) {
|
|||||||
// can mark through them now, rather than delaying that
|
// can mark through them now, rather than delaying that
|
||||||
// to the reference-processing phase. Since all current
|
// to the reference-processing phase. Since all current
|
||||||
// time-stamp policies advance the soft-ref clock only
|
// time-stamp policies advance the soft-ref clock only
|
||||||
// at a major collection cycle, this is always currently
|
// at a full collection cycle, this is always currently
|
||||||
// accurate.
|
// accurate.
|
||||||
if (!_current_soft_ref_policy->should_clear_reference(obj, _soft_ref_timestamp_clock)) {
|
if (!_current_soft_ref_policy->should_clear_reference(obj, _soft_ref_timestamp_clock)) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -213,15 +213,18 @@ class VM_CollectForMetadataAllocation: public VM_GC_Operation {
|
|||||||
size_t _size; // size of object to be allocated
|
size_t _size; // size of object to be allocated
|
||||||
Metaspace::MetadataType _mdtype;
|
Metaspace::MetadataType _mdtype;
|
||||||
ClassLoaderData* _loader_data;
|
ClassLoaderData* _loader_data;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
VM_CollectForMetadataAllocation(ClassLoaderData* loader_data,
|
VM_CollectForMetadataAllocation(ClassLoaderData* loader_data,
|
||||||
size_t size, Metaspace::MetadataType mdtype,
|
size_t size,
|
||||||
|
Metaspace::MetadataType mdtype,
|
||||||
uint gc_count_before,
|
uint gc_count_before,
|
||||||
uint full_gc_count_before,
|
uint full_gc_count_before,
|
||||||
GCCause::Cause gc_cause)
|
GCCause::Cause gc_cause)
|
||||||
: VM_GC_Operation(gc_count_before, gc_cause, full_gc_count_before, true),
|
: VM_GC_Operation(gc_count_before, gc_cause, full_gc_count_before, true),
|
||||||
_loader_data(loader_data), _size(size), _mdtype(mdtype), _result(NULL) {
|
_loader_data(loader_data), _size(size), _mdtype(mdtype), _result(NULL) {
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual VMOp_Type type() const { return VMOp_CollectForMetadataAllocation; }
|
virtual VMOp_Type type() const { return VMOp_CollectForMetadataAllocation; }
|
||||||
virtual void doit();
|
virtual void doit();
|
||||||
MetaWord* result() const { return _result; }
|
MetaWord* result() const { return _result; }
|
||||||
|
@ -1596,7 +1596,7 @@ public:
|
|||||||
"(ParallelGC only)") \
|
"(ParallelGC only)") \
|
||||||
\
|
\
|
||||||
product(bool, ScavengeBeforeFullGC, true, \
|
product(bool, ScavengeBeforeFullGC, true, \
|
||||||
"Scavenge youngest generation before each full GC.") \
|
"Scavenge young generation before each full GC.") \
|
||||||
\
|
\
|
||||||
develop(bool, ScavengeWithObjectsInToSpace, false, \
|
develop(bool, ScavengeWithObjectsInToSpace, false, \
|
||||||
"Allow scavenges to occur when to-space contains objects") \
|
"Allow scavenges to occur when to-space contains objects") \
|
||||||
@ -2094,7 +2094,7 @@ public:
|
|||||||
"promotion failure") \
|
"promotion failure") \
|
||||||
\
|
\
|
||||||
notproduct(bool, PromotionFailureALot, false, \
|
notproduct(bool, PromotionFailureALot, false, \
|
||||||
"Use promotion failure handling on every youngest generation " \
|
"Use promotion failure handling on every young generation " \
|
||||||
"collection") \
|
"collection") \
|
||||||
\
|
\
|
||||||
develop(uintx, PromotionFailureALotCount, 1000, \
|
develop(uintx, PromotionFailureALotCount, 1000, \
|
||||||
|
@ -1360,7 +1360,7 @@ ObjectMonitor * NOINLINE ObjectSynchronizer::inflate(Thread * Self,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// We've successfully installed INFLATING (0) into the mark-word.
|
// We've successfully installed INFLATING (0) into the mark-word.
|
||||||
// This is the only case where 0 will appear in a mark-work.
|
// This is the only case where 0 will appear in a mark-word.
|
||||||
// Only the singular thread that successfully swings the mark-word
|
// Only the singular thread that successfully swings the mark-word
|
||||||
// to 0 can perform (or more precisely, complete) inflation.
|
// to 0 can perform (or more precisely, complete) inflation.
|
||||||
//
|
//
|
||||||
|
@ -204,21 +204,21 @@ MemoryUsage ContiguousSpacePool::get_memory_usage() {
|
|||||||
return MemoryUsage(initial_size(), used, committed, maxSize);
|
return MemoryUsage(initial_size(), used, committed, maxSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
SurvivorContiguousSpacePool::SurvivorContiguousSpacePool(DefNewGeneration* gen,
|
SurvivorContiguousSpacePool::SurvivorContiguousSpacePool(DefNewGeneration* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
size_t max_size,
|
size_t max_size,
|
||||||
bool support_usage_threshold) :
|
bool support_usage_threshold) :
|
||||||
CollectedMemoryPool(name, type, gen->from()->capacity(), max_size,
|
CollectedMemoryPool(name, type, young_gen->from()->capacity(), max_size,
|
||||||
support_usage_threshold), _gen(gen) {
|
support_usage_threshold), _young_gen(young_gen) {
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t SurvivorContiguousSpacePool::used_in_bytes() {
|
size_t SurvivorContiguousSpacePool::used_in_bytes() {
|
||||||
return _gen->from()->used();
|
return _young_gen->from()->used();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t SurvivorContiguousSpacePool::committed_in_bytes() {
|
size_t SurvivorContiguousSpacePool::committed_in_bytes() {
|
||||||
return _gen->from()->capacity();
|
return _young_gen->from()->capacity();
|
||||||
}
|
}
|
||||||
|
|
||||||
MemoryUsage SurvivorContiguousSpacePool::get_memory_usage() {
|
MemoryUsage SurvivorContiguousSpacePool::get_memory_usage() {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2003, 2015, 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
|
||||||
@ -163,10 +163,10 @@ public:
|
|||||||
|
|
||||||
class SurvivorContiguousSpacePool : public CollectedMemoryPool {
|
class SurvivorContiguousSpacePool : public CollectedMemoryPool {
|
||||||
private:
|
private:
|
||||||
DefNewGeneration* _gen;
|
DefNewGeneration* _young_gen;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SurvivorContiguousSpacePool(DefNewGeneration* gen,
|
SurvivorContiguousSpacePool(DefNewGeneration* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
size_t max_size,
|
size_t max_size,
|
||||||
|
@ -212,13 +212,13 @@ MemoryPool* MemoryService::add_space(ContiguousSpace* space,
|
|||||||
return (MemoryPool*) pool;
|
return (MemoryPool*) pool;
|
||||||
}
|
}
|
||||||
|
|
||||||
MemoryPool* MemoryService::add_survivor_spaces(DefNewGeneration* gen,
|
MemoryPool* MemoryService::add_survivor_spaces(DefNewGeneration* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
bool is_heap,
|
bool is_heap,
|
||||||
size_t max_size,
|
size_t max_size,
|
||||||
bool support_usage_threshold) {
|
bool support_usage_threshold) {
|
||||||
MemoryPool::PoolType type = (is_heap ? MemoryPool::Heap : MemoryPool::NonHeap);
|
MemoryPool::PoolType type = (is_heap ? MemoryPool::Heap : MemoryPool::NonHeap);
|
||||||
SurvivorContiguousSpacePool* pool = new SurvivorContiguousSpacePool(gen, name, type, max_size, support_usage_threshold);
|
SurvivorContiguousSpacePool* pool = new SurvivorContiguousSpacePool(young_gen, name, type, max_size, support_usage_threshold);
|
||||||
|
|
||||||
_pools_list->append(pool);
|
_pools_list->append(pool);
|
||||||
return (MemoryPool*) pool;
|
return (MemoryPool*) pool;
|
||||||
@ -328,18 +328,18 @@ void MemoryService::add_generation_memory_pool(Generation* gen,
|
|||||||
|
|
||||||
|
|
||||||
#if INCLUDE_ALL_GCS
|
#if INCLUDE_ALL_GCS
|
||||||
void MemoryService::add_psYoung_memory_pool(PSYoungGen* gen, MemoryManager* major_mgr, MemoryManager* minor_mgr) {
|
void MemoryService::add_psYoung_memory_pool(PSYoungGen* young_gen, MemoryManager* major_mgr, MemoryManager* minor_mgr) {
|
||||||
assert(major_mgr != NULL && minor_mgr != NULL, "Should have two managers");
|
assert(major_mgr != NULL && minor_mgr != NULL, "Should have two managers");
|
||||||
|
|
||||||
// Add a memory pool for each space and young gen doesn't
|
// Add a memory pool for each space and young gen doesn't
|
||||||
// support low memory detection as it is expected to get filled up.
|
// support low memory detection as it is expected to get filled up.
|
||||||
EdenMutableSpacePool* eden = new EdenMutableSpacePool(gen,
|
EdenMutableSpacePool* eden = new EdenMutableSpacePool(young_gen,
|
||||||
gen->eden_space(),
|
young_gen->eden_space(),
|
||||||
"PS Eden Space",
|
"PS Eden Space",
|
||||||
MemoryPool::Heap,
|
MemoryPool::Heap,
|
||||||
false /* support_usage_threshold */);
|
false /* support_usage_threshold */);
|
||||||
|
|
||||||
SurvivorMutableSpacePool* survivor = new SurvivorMutableSpacePool(gen,
|
SurvivorMutableSpacePool* survivor = new SurvivorMutableSpacePool(young_gen,
|
||||||
"PS Survivor Space",
|
"PS Survivor Space",
|
||||||
MemoryPool::Heap,
|
MemoryPool::Heap,
|
||||||
false /* support_usage_threshold */);
|
false /* support_usage_threshold */);
|
||||||
@ -352,13 +352,13 @@ void MemoryService::add_psYoung_memory_pool(PSYoungGen* gen, MemoryManager* majo
|
|||||||
_pools_list->append(survivor);
|
_pools_list->append(survivor);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MemoryService::add_psOld_memory_pool(PSOldGen* gen, MemoryManager* mgr) {
|
void MemoryService::add_psOld_memory_pool(PSOldGen* old_gen, MemoryManager* mgr) {
|
||||||
PSGenerationPool* old_gen = new PSGenerationPool(gen,
|
PSGenerationPool* old_gen_pool = new PSGenerationPool(old_gen,
|
||||||
"PS Old Gen",
|
"PS Old Gen",
|
||||||
MemoryPool::Heap,
|
MemoryPool::Heap,
|
||||||
true /* support_usage_threshold */);
|
true /* support_usage_threshold */);
|
||||||
mgr->add_pool(old_gen);
|
mgr->add_pool(old_gen_pool);
|
||||||
_pools_list->append(old_gen);
|
_pools_list->append(old_gen_pool);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MemoryService::add_g1YoungGen_memory_pool(G1CollectedHeap* g1h,
|
void MemoryService::add_g1YoungGen_memory_pool(G1CollectedHeap* g1h,
|
||||||
@ -548,7 +548,7 @@ Handle MemoryService::create_MemoryUsage_obj(MemoryUsage usage, TRAPS) {
|
|||||||
}
|
}
|
||||||
//
|
//
|
||||||
// GC manager type depends on the type of Generation. Depending on the space
|
// GC manager type depends on the type of Generation. Depending on the space
|
||||||
// availablity and vm options the gc uses major gc manager or minor gc
|
// availability and vm options the gc uses major gc manager or minor gc
|
||||||
// manager or both. The type of gc manager depends on the generation kind.
|
// manager or both. The type of gc manager depends on the generation kind.
|
||||||
// For DefNew and ParNew generation doing scavenge gc uses minor gc manager (so
|
// For DefNew and ParNew generation doing scavenge gc uses minor gc manager (so
|
||||||
// _fullGC is set to false ) and for other generation kinds doing
|
// _fullGC is set to false ) and for other generation kinds doing
|
||||||
|
@ -80,10 +80,10 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void add_psYoung_memory_pool(PSYoungGen* gen,
|
static void add_psYoung_memory_pool(PSYoungGen* young_gen,
|
||||||
MemoryManager* major_mgr,
|
MemoryManager* major_mgr,
|
||||||
MemoryManager* minor_mgr);
|
MemoryManager* minor_mgr);
|
||||||
static void add_psOld_memory_pool(PSOldGen* gen,
|
static void add_psOld_memory_pool(PSOldGen* old_gen,
|
||||||
MemoryManager* mgr);
|
MemoryManager* mgr);
|
||||||
|
|
||||||
static void add_g1YoungGen_memory_pool(G1CollectedHeap* g1h,
|
static void add_g1YoungGen_memory_pool(G1CollectedHeap* g1h,
|
||||||
@ -97,7 +97,7 @@ private:
|
|||||||
bool is_heap,
|
bool is_heap,
|
||||||
size_t max_size,
|
size_t max_size,
|
||||||
bool support_usage_threshold);
|
bool support_usage_threshold);
|
||||||
static MemoryPool* add_survivor_spaces(DefNewGeneration* gen,
|
static MemoryPool* add_survivor_spaces(DefNewGeneration* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
bool is_heap,
|
bool is_heap,
|
||||||
size_t max_size,
|
size_t max_size,
|
||||||
@ -162,7 +162,6 @@ public:
|
|||||||
bool recordGCEndTime, bool countCollection,
|
bool recordGCEndTime, bool countCollection,
|
||||||
GCCause::Cause cause);
|
GCCause::Cause cause);
|
||||||
|
|
||||||
|
|
||||||
static void oops_do(OopClosure* f);
|
static void oops_do(OopClosure* f);
|
||||||
|
|
||||||
static bool get_verbose() { return PrintGC; }
|
static bool get_verbose() { return PrintGC; }
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2007, 2012, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2007, 2015, 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
|
||||||
@ -33,18 +33,18 @@
|
|||||||
#include "services/memoryManager.hpp"
|
#include "services/memoryManager.hpp"
|
||||||
#include "services/psMemoryPool.hpp"
|
#include "services/psMemoryPool.hpp"
|
||||||
|
|
||||||
PSGenerationPool::PSGenerationPool(PSOldGen* gen,
|
PSGenerationPool::PSGenerationPool(PSOldGen* old_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
bool support_usage_threshold) :
|
bool support_usage_threshold) :
|
||||||
CollectedMemoryPool(name, type, gen->capacity_in_bytes(),
|
CollectedMemoryPool(name, type, old_gen->capacity_in_bytes(),
|
||||||
gen->reserved().byte_size(), support_usage_threshold), _gen(gen) {
|
old_gen->reserved().byte_size(), support_usage_threshold), _old_gen(old_gen) {
|
||||||
}
|
}
|
||||||
|
|
||||||
MemoryUsage PSGenerationPool::get_memory_usage() {
|
MemoryUsage PSGenerationPool::get_memory_usage() {
|
||||||
size_t maxSize = (available_for_allocation() ? max_size() : 0);
|
size_t maxSize = (available_for_allocation() ? max_size() : 0);
|
||||||
size_t used = used_in_bytes();
|
size_t used = used_in_bytes();
|
||||||
size_t committed = _gen->capacity_in_bytes();
|
size_t committed = _old_gen->capacity_in_bytes();
|
||||||
|
|
||||||
return MemoryUsage(initial_size(), used, committed, maxSize);
|
return MemoryUsage(initial_size(), used, committed, maxSize);
|
||||||
}
|
}
|
||||||
@ -55,15 +55,16 @@ MemoryUsage PSGenerationPool::get_memory_usage() {
|
|||||||
// Max size of PS eden space is changing due to ergonomic.
|
// Max size of PS eden space is changing due to ergonomic.
|
||||||
// PSYoungGen, PSOldGen, Eden, Survivor spaces are all resizable.
|
// PSYoungGen, PSOldGen, Eden, Survivor spaces are all resizable.
|
||||||
//
|
//
|
||||||
EdenMutableSpacePool::EdenMutableSpacePool(PSYoungGen* gen,
|
EdenMutableSpacePool::EdenMutableSpacePool(PSYoungGen* young_gen,
|
||||||
MutableSpace* space,
|
MutableSpace* space,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
bool support_usage_threshold) :
|
bool support_usage_threshold) :
|
||||||
CollectedMemoryPool(name, type, space->capacity_in_bytes(),
|
CollectedMemoryPool(name, type, space->capacity_in_bytes(),
|
||||||
(gen->max_size() - gen->from_space()->capacity_in_bytes() - gen->to_space()->capacity_in_bytes()),
|
(young_gen->max_size() - young_gen->from_space()->capacity_in_bytes() - young_gen->to_space()->capacity_in_bytes()),
|
||||||
support_usage_threshold),
|
support_usage_threshold),
|
||||||
_gen(gen), _space(space) {
|
_young_gen(young_gen),
|
||||||
|
_space(space) {
|
||||||
}
|
}
|
||||||
|
|
||||||
MemoryUsage EdenMutableSpacePool::get_memory_usage() {
|
MemoryUsage EdenMutableSpacePool::get_memory_usage() {
|
||||||
@ -79,13 +80,13 @@ MemoryUsage EdenMutableSpacePool::get_memory_usage() {
|
|||||||
//
|
//
|
||||||
// PS from and to survivor spaces could have different sizes.
|
// PS from and to survivor spaces could have different sizes.
|
||||||
//
|
//
|
||||||
SurvivorMutableSpacePool::SurvivorMutableSpacePool(PSYoungGen* gen,
|
SurvivorMutableSpacePool::SurvivorMutableSpacePool(PSYoungGen* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
bool support_usage_threshold) :
|
bool support_usage_threshold) :
|
||||||
CollectedMemoryPool(name, type, gen->from_space()->capacity_in_bytes(),
|
CollectedMemoryPool(name, type, young_gen->from_space()->capacity_in_bytes(),
|
||||||
gen->from_space()->capacity_in_bytes(),
|
young_gen->from_space()->capacity_in_bytes(),
|
||||||
support_usage_threshold), _gen(gen) {
|
support_usage_threshold), _young_gen(young_gen) {
|
||||||
}
|
}
|
||||||
|
|
||||||
MemoryUsage SurvivorMutableSpacePool::get_memory_usage() {
|
MemoryUsage SurvivorMutableSpacePool::get_memory_usage() {
|
||||||
|
@ -39,23 +39,23 @@
|
|||||||
|
|
||||||
class PSGenerationPool : public CollectedMemoryPool {
|
class PSGenerationPool : public CollectedMemoryPool {
|
||||||
private:
|
private:
|
||||||
PSOldGen* _gen;
|
PSOldGen* _old_gen;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
PSGenerationPool(PSOldGen* pool, const char* name, PoolType type, bool support_usage_threshold);
|
PSGenerationPool(PSOldGen* pool, const char* name, PoolType type, bool support_usage_threshold);
|
||||||
|
|
||||||
MemoryUsage get_memory_usage();
|
MemoryUsage get_memory_usage();
|
||||||
size_t used_in_bytes() { return _gen->used_in_bytes(); }
|
size_t used_in_bytes() { return _old_gen->used_in_bytes(); }
|
||||||
size_t max_size() const { return _gen->reserved().byte_size(); }
|
size_t max_size() const { return _old_gen->reserved().byte_size(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
class EdenMutableSpacePool : public CollectedMemoryPool {
|
class EdenMutableSpacePool : public CollectedMemoryPool {
|
||||||
private:
|
private:
|
||||||
PSYoungGen* _gen;
|
PSYoungGen* _young_gen;
|
||||||
MutableSpace* _space;
|
MutableSpace* _space;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
EdenMutableSpacePool(PSYoungGen* gen,
|
EdenMutableSpacePool(PSYoungGen* young_gen,
|
||||||
MutableSpace* space,
|
MutableSpace* space,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
@ -66,16 +66,16 @@ public:
|
|||||||
size_t used_in_bytes() { return space()->used_in_bytes(); }
|
size_t used_in_bytes() { return space()->used_in_bytes(); }
|
||||||
size_t max_size() const {
|
size_t max_size() const {
|
||||||
// Eden's max_size = max_size of Young Gen - the current committed size of survivor spaces
|
// Eden's max_size = max_size of Young Gen - the current committed size of survivor spaces
|
||||||
return _gen->max_size() - _gen->from_space()->capacity_in_bytes() - _gen->to_space()->capacity_in_bytes();
|
return _young_gen->max_size() - _young_gen->from_space()->capacity_in_bytes() - _young_gen->to_space()->capacity_in_bytes();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
class SurvivorMutableSpacePool : public CollectedMemoryPool {
|
class SurvivorMutableSpacePool : public CollectedMemoryPool {
|
||||||
private:
|
private:
|
||||||
PSYoungGen* _gen;
|
PSYoungGen* _young_gen;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SurvivorMutableSpacePool(PSYoungGen* gen,
|
SurvivorMutableSpacePool(PSYoungGen* young_gen,
|
||||||
const char* name,
|
const char* name,
|
||||||
PoolType type,
|
PoolType type,
|
||||||
bool support_usage_threshold);
|
bool support_usage_threshold);
|
||||||
@ -83,14 +83,14 @@ public:
|
|||||||
MemoryUsage get_memory_usage();
|
MemoryUsage get_memory_usage();
|
||||||
|
|
||||||
size_t used_in_bytes() {
|
size_t used_in_bytes() {
|
||||||
return _gen->from_space()->used_in_bytes();
|
return _young_gen->from_space()->used_in_bytes();
|
||||||
}
|
}
|
||||||
size_t committed_in_bytes() {
|
size_t committed_in_bytes() {
|
||||||
return _gen->from_space()->capacity_in_bytes();
|
return _young_gen->from_space()->capacity_in_bytes();
|
||||||
}
|
}
|
||||||
size_t max_size() const {
|
size_t max_size() const {
|
||||||
// Return current committed size of the from-space
|
// Return current committed size of the from-space
|
||||||
return _gen->from_space()->capacity_in_bytes();
|
return _young_gen->from_space()->capacity_in_bytes();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user