8173335: Improve logging for j.l.ref.reference processing

Add logs for each phases of references and stats for each worker threads

Reviewed-by: tschatzl, shade
This commit is contained in:
Sangheon Kim 2017-08-10 18:09:19 -07:00
parent ebb8c32e7f
commit ebcca449a0
31 changed files with 1077 additions and 186 deletions

View File

@ -5185,6 +5185,7 @@ void CMSCollector::refProcessingWork() {
CMSDrainMarkingStackClosure cmsDrainMarkingStackClosure(this, CMSDrainMarkingStackClosure cmsDrainMarkingStackClosure(this,
_span, &_markBitMap, &_markStack, _span, &_markBitMap, &_markStack,
&cmsKeepAliveClosure, false /* !preclean */); &cmsKeepAliveClosure, false /* !preclean */);
ReferenceProcessorPhaseTimes pt(_gc_timer_cm, rp->num_q());
{ {
GCTraceTime(Debug, gc, phases) t("Reference Processing", _gc_timer_cm); GCTraceTime(Debug, gc, phases) t("Reference Processing", _gc_timer_cm);
@ -5211,16 +5212,16 @@ void CMSCollector::refProcessingWork() {
&cmsKeepAliveClosure, &cmsKeepAliveClosure,
&cmsDrainMarkingStackClosure, &cmsDrainMarkingStackClosure,
&task_executor, &task_executor,
_gc_timer_cm); &pt);
} else { } else {
stats = rp->process_discovered_references(&_is_alive_closure, stats = rp->process_discovered_references(&_is_alive_closure,
&cmsKeepAliveClosure, &cmsKeepAliveClosure,
&cmsDrainMarkingStackClosure, &cmsDrainMarkingStackClosure,
NULL, NULL,
_gc_timer_cm); &pt);
} }
_gc_tracer_cm->report_gc_reference_stats(stats); _gc_tracer_cm->report_gc_reference_stats(stats);
pt.print_all_references();
} }
// This is the point where the entire marking should have completed. // This is the point where the entire marking should have completed.
@ -5261,11 +5262,12 @@ void CMSCollector::refProcessingWork() {
if (rp->processing_is_mt()) { if (rp->processing_is_mt()) {
rp->balance_all_queues(); rp->balance_all_queues();
CMSRefProcTaskExecutor task_executor(*this); CMSRefProcTaskExecutor task_executor(*this);
rp->enqueue_discovered_references(&task_executor); rp->enqueue_discovered_references(&task_executor, &pt);
} else { } else {
rp->enqueue_discovered_references(NULL); rp->enqueue_discovered_references(NULL, &pt);
} }
rp->verify_no_references_recorded(); rp->verify_no_references_recorded();
pt.print_enqueue_phase();
assert(!rp->discovery_enabled(), "should have been disabled"); assert(!rp->discovery_enabled(), "should have been disabled");
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -984,20 +984,22 @@ void ParNewGeneration::collect(bool full,
// Can the mt_degree be set later (at run_task() time would be best)? // Can the mt_degree be set later (at run_task() time would be best)?
rp->set_active_mt_degree(active_workers); rp->set_active_mt_degree(active_workers);
ReferenceProcessorStats stats; ReferenceProcessorStats stats;
ReferenceProcessorPhaseTimes pt(_gc_timer, rp->num_q());
if (rp->processing_is_mt()) { if (rp->processing_is_mt()) {
ParNewRefProcTaskExecutor task_executor(*this, *_old_gen, thread_state_set); ParNewRefProcTaskExecutor task_executor(*this, *_old_gen, thread_state_set);
stats = rp->process_discovered_references(&is_alive, &keep_alive, stats = rp->process_discovered_references(&is_alive, &keep_alive,
&evacuate_followers, &task_executor, &evacuate_followers, &task_executor,
_gc_timer); &pt);
} else { } else {
thread_state_set.flush(); thread_state_set.flush();
gch->save_marks(); gch->save_marks();
stats = rp->process_discovered_references(&is_alive, &keep_alive, stats = rp->process_discovered_references(&is_alive, &keep_alive,
&evacuate_followers, NULL, &evacuate_followers, NULL,
_gc_timer); &pt);
} }
_gc_tracer.report_gc_reference_stats(stats); _gc_tracer.report_gc_reference_stats(stats);
_gc_tracer.report_tenuring_threshold(tenuring_threshold()); _gc_tracer.report_tenuring_threshold(tenuring_threshold());
pt.print_all_references();
if (!promotion_failed()) { if (!promotion_failed()) {
// Swap the survivor spaces. // Swap the survivor spaces.
@ -1049,14 +1051,16 @@ void ParNewGeneration::collect(bool full,
rp->set_enqueuing_is_done(true); rp->set_enqueuing_is_done(true);
if (rp->processing_is_mt()) { if (rp->processing_is_mt()) {
ParNewRefProcTaskExecutor task_executor(*this, *_old_gen, thread_state_set); ParNewRefProcTaskExecutor task_executor(*this, *_old_gen, thread_state_set);
rp->enqueue_discovered_references(&task_executor); rp->enqueue_discovered_references(&task_executor, &pt);
} else { } else {
rp->enqueue_discovered_references(NULL); rp->enqueue_discovered_references(NULL, &pt);
} }
rp->verify_no_references_recorded(); rp->verify_no_references_recorded();
gch->trace_heap_after_gc(gc_tracer()); gch->trace_heap_after_gc(gc_tracer());
pt.print_enqueue_phase();
_gc_timer->register_gc_end(); _gc_timer->register_gc_end();
_gc_tracer.report_gc_end(_gc_timer->gc_end(), _gc_timer->time_partitions()); _gc_tracer.report_gc_end(_gc_timer->gc_end(), _gc_timer->time_partitions());

View File

@ -1260,9 +1260,13 @@ bool G1CollectedHeap::do_full_collection(bool explicit_gc,
assert(num_free_regions() == 0, "we should not have added any free regions"); assert(num_free_regions() == 0, "we should not have added any free regions");
rebuild_region_sets(false /* free_list_only */); rebuild_region_sets(false /* free_list_only */);
ReferenceProcessorPhaseTimes pt(NULL, ref_processor_stw()->num_q());
// Enqueue any discovered reference objects that have // Enqueue any discovered reference objects that have
// not been removed from the discovered lists. // not been removed from the discovered lists.
ref_processor_stw()->enqueue_discovered_references(); ref_processor_stw()->enqueue_discovered_references(NULL, &pt);
pt.print_enqueue_phase();
#if defined(COMPILER2) || INCLUDE_JVMCI #if defined(COMPILER2) || INCLUDE_JVMCI
DerivedPointerTable::update_pointers(); DerivedPointerTable::update_pointers();
@ -1667,7 +1671,9 @@ void G1CollectedHeap::shrink(size_t shrink_bytes) {
G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) : G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) :
CollectedHeap(), CollectedHeap(),
_collector_policy(collector_policy), _collector_policy(collector_policy),
_g1_policy(create_g1_policy()), _gc_timer_stw(new (ResourceObj::C_HEAP, mtGC) STWGCTimer()),
_gc_tracer_stw(new (ResourceObj::C_HEAP, mtGC) G1NewTracer()),
_g1_policy(create_g1_policy(_gc_timer_stw)),
_collection_set(this, _g1_policy), _collection_set(this, _g1_policy),
_dirty_card_queue_set(false), _dirty_card_queue_set(false),
_is_alive_closure_cm(this), _is_alive_closure_cm(this),
@ -1694,9 +1700,7 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) :
_expand_heap_after_alloc_failure(true), _expand_heap_after_alloc_failure(true),
_old_marking_cycles_started(0), _old_marking_cycles_started(0),
_old_marking_cycles_completed(0), _old_marking_cycles_completed(0),
_in_cset_fast_test(), _in_cset_fast_test() {
_gc_timer_stw(new (ResourceObj::C_HEAP, mtGC) STWGCTimer()),
_gc_tracer_stw(new (ResourceObj::C_HEAP, mtGC) G1NewTracer()) {
_workers = new WorkGang("GC Thread", ParallelGCThreads, _workers = new WorkGang("GC Thread", ParallelGCThreads,
/* are_GC_task_threads */true, /* are_GC_task_threads */true,
@ -2015,10 +2019,12 @@ void G1CollectedHeap::ref_processing_init() {
MemRegion mr = reserved_region(); MemRegion mr = reserved_region();
bool mt_processing = ParallelRefProcEnabled && (ParallelGCThreads > 1);
// Concurrent Mark ref processor // Concurrent Mark ref processor
_ref_processor_cm = _ref_processor_cm =
new ReferenceProcessor(mr, // span new ReferenceProcessor(mr, // span
ParallelRefProcEnabled && (ParallelGCThreads > 1), mt_processing,
// mt processing // mt processing
ParallelGCThreads, ParallelGCThreads,
// degree of mt processing // degree of mt processing
@ -2035,7 +2041,7 @@ void G1CollectedHeap::ref_processing_init() {
// STW ref processor // STW ref processor
_ref_processor_stw = _ref_processor_stw =
new ReferenceProcessor(mr, // span new ReferenceProcessor(mr, // span
ParallelRefProcEnabled && (ParallelGCThreads > 1), mt_processing,
// mt processing // mt processing
ParallelGCThreads, ParallelGCThreads,
// degree of mt processing // degree of mt processing
@ -4313,6 +4319,8 @@ void G1CollectedHeap::process_discovered_references(G1ParScanThreadStateSet* per
// Setup the soft refs policy... // Setup the soft refs policy...
rp->setup_policy(false); rp->setup_policy(false);
ReferenceProcessorPhaseTimes* pt = g1_policy()->phase_times()->ref_phase_times();
ReferenceProcessorStats stats; ReferenceProcessorStats stats;
if (!rp->processing_is_mt()) { if (!rp->processing_is_mt()) {
// Serial reference processing... // Serial reference processing...
@ -4320,7 +4328,7 @@ void G1CollectedHeap::process_discovered_references(G1ParScanThreadStateSet* per
&keep_alive, &keep_alive,
&drain_queue, &drain_queue,
NULL, NULL,
_gc_timer_stw); pt);
} else { } else {
uint no_of_gc_workers = workers()->active_workers(); uint no_of_gc_workers = workers()->active_workers();
@ -4334,7 +4342,7 @@ void G1CollectedHeap::process_discovered_references(G1ParScanThreadStateSet* per
&keep_alive, &keep_alive,
&drain_queue, &drain_queue,
&par_task_executor, &par_task_executor,
_gc_timer_stw); pt);
} }
_gc_tracer_stw->report_gc_reference_stats(stats); _gc_tracer_stw->report_gc_reference_stats(stats);
@ -4353,11 +4361,13 @@ void G1CollectedHeap::enqueue_discovered_references(G1ParScanThreadStateSet* per
ReferenceProcessor* rp = _ref_processor_stw; ReferenceProcessor* rp = _ref_processor_stw;
assert(!rp->discovery_enabled(), "should have been disabled as part of processing"); assert(!rp->discovery_enabled(), "should have been disabled as part of processing");
ReferenceProcessorPhaseTimes* pt = g1_policy()->phase_times()->ref_phase_times();
// Now enqueue any remaining on the discovered lists on to // Now enqueue any remaining on the discovered lists on to
// the pending list. // the pending list.
if (!rp->processing_is_mt()) { if (!rp->processing_is_mt()) {
// Serial reference processing... // Serial reference processing...
rp->enqueue_discovered_references(); rp->enqueue_discovered_references(NULL, pt);
} else { } else {
// Parallel reference enqueueing // Parallel reference enqueueing
@ -4368,7 +4378,7 @@ void G1CollectedHeap::enqueue_discovered_references(G1ParScanThreadStateSet* per
n_workers, rp->max_num_q()); n_workers, rp->max_num_q());
G1STWRefProcTaskExecutor par_task_executor(this, per_thread_states, workers(), _task_queues, n_workers); G1STWRefProcTaskExecutor par_task_executor(this, per_thread_states, workers(), _task_queues, n_workers);
rp->enqueue_discovered_references(&par_task_executor); rp->enqueue_discovered_references(&par_task_executor, pt);
} }
rp->verify_no_references_recorded(); rp->verify_no_references_recorded();

View File

@ -298,7 +298,7 @@ private:
size_t size, size_t size,
size_t translation_factor); size_t translation_factor);
static G1Policy* create_g1_policy(); static G1Policy* create_g1_policy(STWGCTimer* gc_timer);
void trace_heap(GCWhen::Type when, const GCTracer* tracer); void trace_heap(GCWhen::Type when, const GCTracer* tracer);
@ -370,6 +370,10 @@ protected:
G1EdenRegions _eden; G1EdenRegions _eden;
G1SurvivorRegions _survivor; G1SurvivorRegions _survivor;
STWGCTimer* _gc_timer_stw;
G1NewTracer* _gc_tracer_stw;
// The current policy object for the collector. // The current policy object for the collector.
G1Policy* _g1_policy; G1Policy* _g1_policy;
G1HeapSizingPolicy* _heap_sizing_policy; G1HeapSizingPolicy* _heap_sizing_policy;
@ -901,10 +905,6 @@ protected:
// The (stw) reference processor... // The (stw) reference processor...
ReferenceProcessor* _ref_processor_stw; ReferenceProcessor* _ref_processor_stw;
STWGCTimer* _gc_timer_stw;
G1NewTracer* _gc_tracer_stw;
// During reference object discovery, the _is_alive_non_header // During reference object discovery, the _is_alive_non_header
// closure (if non-null) is applied to the referent object to // closure (if non-null) is applied to the referent object to
// determine whether the referent is live. If so then the // determine whether the referent is live. If so then the

View File

@ -28,6 +28,8 @@
#include "gc/g1/g1ParScanThreadState.hpp" #include "gc/g1/g1ParScanThreadState.hpp"
#include "gc/g1/heapRegion.inline.hpp" #include "gc/g1/heapRegion.inline.hpp"
class STWGCTimer;
bool G1CollectedHeap::copy_allocation_context_stats(const jint* contexts, bool G1CollectedHeap::copy_allocation_context_stats(const jint* contexts,
jlong* totals, jlong* totals,
jbyte* accuracy, jbyte* accuracy,
@ -40,6 +42,6 @@ HeapRegion* G1CollectedHeap::new_heap_region(uint hrs_index,
return new HeapRegion(hrs_index, bot(), mr); return new HeapRegion(hrs_index, bot(), mr);
} }
G1Policy* G1CollectedHeap::create_g1_policy() { G1Policy* G1CollectedHeap::create_g1_policy(STWGCTimer* gc_timer) {
return new G1DefaultPolicy(); return new G1DefaultPolicy(gc_timer);
} }

View File

@ -1660,14 +1660,17 @@ void G1ConcurrentMark::weakRefsWork(bool clear_all_soft_refs) {
// Reference lists are balanced (see balance_all_queues() and balance_queues()). // Reference lists are balanced (see balance_all_queues() and balance_queues()).
rp->set_active_mt_degree(active_workers); rp->set_active_mt_degree(active_workers);
ReferenceProcessorPhaseTimes pt(_gc_timer_cm, rp->num_q());
// Process the weak references. // Process the weak references.
const ReferenceProcessorStats& stats = const ReferenceProcessorStats& stats =
rp->process_discovered_references(&g1_is_alive, rp->process_discovered_references(&g1_is_alive,
&g1_keep_alive, &g1_keep_alive,
&g1_drain_mark_stack, &g1_drain_mark_stack,
executor, executor,
_gc_timer_cm); &pt);
_gc_tracer_cm->report_gc_reference_stats(stats); _gc_tracer_cm->report_gc_reference_stats(stats);
pt.print_all_references();
// The do_oop work routines of the keep_alive and drain_marking_stack // The do_oop work routines of the keep_alive and drain_marking_stack
// oop closures will set the has_overflown flag if we overflow the // oop closures will set the has_overflown flag if we overflow the
@ -1678,9 +1681,12 @@ void G1ConcurrentMark::weakRefsWork(bool clear_all_soft_refs) {
assert(rp->num_q() == active_workers, "why not"); assert(rp->num_q() == active_workers, "why not");
rp->enqueue_discovered_references(executor); rp->enqueue_discovered_references(executor, &pt);
rp->verify_no_references_recorded(); rp->verify_no_references_recorded();
pt.print_enqueue_phase();
assert(!rp->discovery_enabled(), "Post condition"); assert(!rp->discovery_enabled(), "Post condition");
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -47,7 +47,7 @@
#include "utilities/growableArray.hpp" #include "utilities/growableArray.hpp"
#include "utilities/pair.hpp" #include "utilities/pair.hpp"
G1DefaultPolicy::G1DefaultPolicy() : G1DefaultPolicy::G1DefaultPolicy(STWGCTimer* gc_timer) :
_predictor(G1ConfidencePercent / 100.0), _predictor(G1ConfidencePercent / 100.0),
_analytics(new G1Analytics(&_predictor)), _analytics(new G1Analytics(&_predictor)),
_mmu_tracker(new G1MMUTrackerQueue(GCPauseIntervalMillis / 1000.0, MaxGCPauseMillis / 1000.0)), _mmu_tracker(new G1MMUTrackerQueue(GCPauseIntervalMillis / 1000.0, MaxGCPauseMillis / 1000.0)),
@ -63,7 +63,7 @@ G1DefaultPolicy::G1DefaultPolicy() :
_initial_mark_to_mixed(), _initial_mark_to_mixed(),
_collection_set(NULL), _collection_set(NULL),
_g1(NULL), _g1(NULL),
_phase_times(new G1GCPhaseTimes(ParallelGCThreads)), _phase_times(new G1GCPhaseTimes(gc_timer, ParallelGCThreads)),
_tenuring_threshold(MaxTenuringThreshold), _tenuring_threshold(MaxTenuringThreshold),
_max_survivor_regions(0), _max_survivor_regions(0),
_survivors_age_table(true), _survivors_age_table(true),

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -44,6 +44,7 @@ class G1Analytics;
class G1SurvivorRegions; class G1SurvivorRegions;
class G1YoungGenSizer; class G1YoungGenSizer;
class GCPolicyCounters; class GCPolicyCounters;
class STWGCTimer;
class G1DefaultPolicy: public G1Policy { class G1DefaultPolicy: public G1Policy {
private: private:
@ -264,7 +265,7 @@ private:
void abort_time_to_mixed_tracking(); void abort_time_to_mixed_tracking();
public: public:
G1DefaultPolicy(); G1DefaultPolicy(STWGCTimer* gc_timer);
virtual ~G1DefaultPolicy(); virtual ~G1DefaultPolicy();

View File

@ -27,7 +27,7 @@
#include "gc/g1/g1GCPhaseTimes.hpp" #include "gc/g1/g1GCPhaseTimes.hpp"
#include "gc/g1/g1HotCardCache.hpp" #include "gc/g1/g1HotCardCache.hpp"
#include "gc/g1/g1StringDedup.hpp" #include "gc/g1/g1StringDedup.hpp"
#include "gc/g1/workerDataArray.inline.hpp" #include "gc/shared/workerDataArray.inline.hpp"
#include "memory/resourceArea.hpp" #include "memory/resourceArea.hpp"
#include "logging/log.hpp" #include "logging/log.hpp"
#include "logging/logStream.hpp" #include "logging/logStream.hpp"
@ -37,10 +37,11 @@
static const char* Indents[5] = {"", " ", " ", " ", " "}; static const char* Indents[5] = {"", " ", " ", " ", " "};
G1GCPhaseTimes::G1GCPhaseTimes(uint max_gc_threads) : G1GCPhaseTimes::G1GCPhaseTimes(STWGCTimer* gc_timer, uint max_gc_threads) :
_max_gc_threads(max_gc_threads), _max_gc_threads(max_gc_threads),
_gc_start_counter(0), _gc_start_counter(0),
_gc_pause_time_ms(0.0) _gc_pause_time_ms(0.0),
_ref_phase_times((GCTimer*)gc_timer, max_gc_threads)
{ {
assert(max_gc_threads > 0, "Must have some GC threads"); assert(max_gc_threads > 0, "Must have some GC threads");
@ -156,6 +157,8 @@ void G1GCPhaseTimes::reset() {
_gc_par_phases[i]->reset(); _gc_par_phases[i]->reset();
} }
} }
_ref_phase_times.reset();
} }
void G1GCPhaseTimes::note_gc_start() { void G1GCPhaseTimes::note_gc_start() {
@ -288,6 +291,19 @@ void G1GCPhaseTimes::debug_time(const char* name, double value) const {
log_debug(gc, phases)("%s%s: " TIME_FORMAT, Indents[2], name, value); log_debug(gc, phases)("%s%s: " TIME_FORMAT, Indents[2], name, value);
} }
void G1GCPhaseTimes::debug_time_for_reference(const char* name, double value) const {
LogTarget(Debug, gc, phases) lt;
LogTarget(Debug, gc, phases, ref) lt2;
if (lt.is_enabled()) {
LogStream ls(lt);
ls.print_cr("%s%s: " TIME_FORMAT, Indents[2], name, value);
} else if (lt2.is_enabled()) {
LogStream ls(lt2);
ls.print_cr("%s%s: " TIME_FORMAT, Indents[2], name, value);
}
}
void G1GCPhaseTimes::trace_time(const char* name, double value) const { void G1GCPhaseTimes::trace_time(const char* name, double value) const {
log_trace(gc, phases)("%s%s: " TIME_FORMAT, Indents[3], name, value); log_trace(gc, phases)("%s%s: " TIME_FORMAT, Indents[3], name, value);
} }
@ -374,7 +390,8 @@ double G1GCPhaseTimes::print_post_evacuate_collection_set() const {
debug_time("Preserve CM Refs", _recorded_preserve_cm_referents_time_ms); debug_time("Preserve CM Refs", _recorded_preserve_cm_referents_time_ms);
trace_phase(_gc_par_phases[PreserveCMReferents]); trace_phase(_gc_par_phases[PreserveCMReferents]);
debug_time("Reference Processing", _cur_ref_proc_time_ms); debug_time_for_reference("Reference Processing", _cur_ref_proc_time_ms);
_ref_phase_times.print_all_references(2, false);
if (G1StringDedup::is_enabled()) { if (G1StringDedup::is_enabled()) {
debug_time("String Dedup Fixup", _cur_string_dedup_fixup_time_ms); debug_time("String Dedup Fixup", _cur_string_dedup_fixup_time_ms);
@ -390,7 +407,8 @@ double G1GCPhaseTimes::print_post_evacuate_collection_set() const {
trace_time("Remove Self Forwards",_cur_evac_fail_remove_self_forwards); trace_time("Remove Self Forwards",_cur_evac_fail_remove_self_forwards);
} }
debug_time("Reference Enqueuing", _cur_ref_enq_time_ms); debug_time_for_reference("Reference Enqueuing", _cur_ref_enq_time_ms);
_ref_phase_times.print_enqueue_phase(2, false);
debug_time("Merge Per-Thread State", _recorded_merge_pss_time_ms); debug_time("Merge Per-Thread State", _recorded_merge_pss_time_ms);
debug_time("Code Roots Purge", _cur_strong_code_root_purge_time_ms); debug_time("Code Roots Purge", _cur_strong_code_root_purge_time_ms);

View File

@ -25,11 +25,13 @@
#ifndef SHARE_VM_GC_G1_G1GCPHASETIMES_HPP #ifndef SHARE_VM_GC_G1_G1GCPHASETIMES_HPP
#define SHARE_VM_GC_G1_G1GCPHASETIMES_HPP #define SHARE_VM_GC_G1_G1GCPHASETIMES_HPP
#include "gc/shared/referenceProcessorPhaseTimes.hpp"
#include "logging/logLevel.hpp" #include "logging/logLevel.hpp"
#include "memory/allocation.hpp" #include "memory/allocation.hpp"
#include "utilities/macros.hpp" #include "utilities/macros.hpp"
class LineBuffer; class LineBuffer;
class STWGCTimer;
template <class T> class WorkerDataArray; template <class T> class WorkerDataArray;
@ -159,6 +161,8 @@ class G1GCPhaseTimes : public CHeapObj<mtGC> {
double _cur_verify_before_time_ms; double _cur_verify_before_time_ms;
double _cur_verify_after_time_ms; double _cur_verify_after_time_ms;
ReferenceProcessorPhaseTimes _ref_phase_times;
double worker_time(GCParPhases phase, uint worker); double worker_time(GCParPhases phase, uint worker);
void note_gc_end(); void note_gc_end();
void reset(); void reset();
@ -172,6 +176,8 @@ class G1GCPhaseTimes : public CHeapObj<mtGC> {
void info_time(const char* name, double value) const; void info_time(const char* name, double value) const;
void debug_time(const char* name, double value) const; void debug_time(const char* name, double value) const;
// This will print logs for both 'gc+phases' and 'gc+phases+ref'.
void debug_time_for_reference(const char* name, double value) const;
void trace_time(const char* name, double value) const; void trace_time(const char* name, double value) const;
void trace_count(const char* name, size_t value) const; void trace_count(const char* name, size_t value) const;
@ -181,7 +187,7 @@ class G1GCPhaseTimes : public CHeapObj<mtGC> {
void print_other(double accounted_ms) const; void print_other(double accounted_ms) const;
public: public:
G1GCPhaseTimes(uint max_gc_threads); G1GCPhaseTimes(STWGCTimer* gc_timer, uint max_gc_threads);
void note_gc_start(); void note_gc_start();
void print(); void print();
@ -354,6 +360,8 @@ class G1GCPhaseTimes : public CHeapObj<mtGC> {
double fast_reclaim_humongous_time_ms() { double fast_reclaim_humongous_time_ms() {
return _cur_fast_reclaim_humongous_time_ms; return _cur_fast_reclaim_humongous_time_ms;
} }
ReferenceProcessorPhaseTimes* ref_phase_times() { return &_ref_phase_times; }
}; };
class G1GCParPhaseTimesTracker : public StackObj { class G1GCParPhaseTimesTracker : public StackObj {

View File

@ -149,13 +149,16 @@ void G1MarkSweep::mark_sweep_phase1(bool& marked_for_unloading,
assert(rp == g1h->ref_processor_stw(), "Sanity"); assert(rp == g1h->ref_processor_stw(), "Sanity");
rp->setup_policy(clear_all_softrefs); rp->setup_policy(clear_all_softrefs);
ReferenceProcessorPhaseTimes pt(gc_timer(), rp->num_q());
const ReferenceProcessorStats& stats = const ReferenceProcessorStats& stats =
rp->process_discovered_references(&GenMarkSweep::is_alive, rp->process_discovered_references(&GenMarkSweep::is_alive,
&GenMarkSweep::keep_alive, &GenMarkSweep::keep_alive,
&GenMarkSweep::follow_stack_closure, &GenMarkSweep::follow_stack_closure,
NULL, NULL,
gc_timer()); &pt);
gc_tracer()->report_gc_reference_stats(stats); gc_tracer()->report_gc_reference_stats(stats);
pt.print_all_references();
} }
// This is the point where the entire marking should have completed. // This is the point where the entire marking should have completed.

View File

@ -254,7 +254,11 @@ bool PSMarkSweep::invoke_no_policy(bool clear_all_softrefs) {
DerivedPointerTable::update_pointers(); DerivedPointerTable::update_pointers();
#endif #endif
ref_processor()->enqueue_discovered_references(NULL); ReferenceProcessorPhaseTimes pt(_gc_timer, ref_processor()->num_q());
ref_processor()->enqueue_discovered_references(NULL, &pt);
pt.print_enqueue_phase();
// Update time of last GC // Update time of last GC
reset_millis_since_last_gc(); reset_millis_since_last_gc();
@ -528,10 +532,12 @@ void PSMarkSweep::mark_sweep_phase1(bool clear_all_softrefs) {
GCTraceTime(Debug, gc, phases) t("Reference Processing", _gc_timer); GCTraceTime(Debug, gc, phases) t("Reference Processing", _gc_timer);
ref_processor()->setup_policy(clear_all_softrefs); ref_processor()->setup_policy(clear_all_softrefs);
ReferenceProcessorPhaseTimes pt(_gc_timer, ref_processor()->num_q());
const ReferenceProcessorStats& stats = const ReferenceProcessorStats& stats =
ref_processor()->process_discovered_references( ref_processor()->process_discovered_references(
is_alive_closure(), mark_and_push_closure(), follow_stack_closure(), NULL, _gc_timer); is_alive_closure(), mark_and_push_closure(), follow_stack_closure(), NULL, &pt);
gc_tracer()->report_gc_reference_stats(stats); gc_tracer()->report_gc_reference_stats(stats);
pt.print_all_references();
} }
// This is the point where the entire marking should have completed. // This is the point where the entire marking should have completed.

View File

@ -1041,7 +1041,11 @@ void PSParallelCompact::post_compact()
DerivedPointerTable::update_pointers(); DerivedPointerTable::update_pointers();
#endif #endif
ref_processor()->enqueue_discovered_references(NULL); ReferenceProcessorPhaseTimes pt(&_gc_timer, ref_processor()->num_q());
ref_processor()->enqueue_discovered_references(NULL, &pt);
pt.print_enqueue_phase();
if (ZapUnusedHeapArea) { if (ZapUnusedHeapArea) {
heap->gen_mangle_unused_area(); heap->gen_mangle_unused_area();
@ -2103,18 +2107,20 @@ void PSParallelCompact::marking_phase(ParCompactionManager* cm,
GCTraceTime(Debug, gc, phases) tm("Reference Processing", &_gc_timer); GCTraceTime(Debug, gc, phases) tm("Reference Processing", &_gc_timer);
ReferenceProcessorStats stats; ReferenceProcessorStats stats;
ReferenceProcessorPhaseTimes pt(&_gc_timer, ref_processor()->num_q());
if (ref_processor()->processing_is_mt()) { if (ref_processor()->processing_is_mt()) {
RefProcTaskExecutor task_executor; RefProcTaskExecutor task_executor;
stats = ref_processor()->process_discovered_references( stats = ref_processor()->process_discovered_references(
is_alive_closure(), &mark_and_push_closure, &follow_stack_closure, is_alive_closure(), &mark_and_push_closure, &follow_stack_closure,
&task_executor, &_gc_timer); &task_executor, &pt);
} else { } else {
stats = ref_processor()->process_discovered_references( stats = ref_processor()->process_discovered_references(
is_alive_closure(), &mark_and_push_closure, &follow_stack_closure, NULL, is_alive_closure(), &mark_and_push_closure, &follow_stack_closure, NULL,
&_gc_timer); &pt);
} }
gc_tracer->report_gc_reference_stats(stats); gc_tracer->report_gc_reference_stats(stats);
pt.print_all_references();
} }
// This is the point where the entire marking should have completed. // This is the point where the entire marking should have completed.

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2002, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2002, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -415,25 +415,29 @@ bool PSScavenge::invoke_no_policy() {
PSKeepAliveClosure keep_alive(promotion_manager); PSKeepAliveClosure keep_alive(promotion_manager);
PSEvacuateFollowersClosure evac_followers(promotion_manager); PSEvacuateFollowersClosure evac_followers(promotion_manager);
ReferenceProcessorStats stats; ReferenceProcessorStats stats;
ReferenceProcessorPhaseTimes pt(&_gc_timer, reference_processor()->num_q());
if (reference_processor()->processing_is_mt()) { if (reference_processor()->processing_is_mt()) {
PSRefProcTaskExecutor task_executor; PSRefProcTaskExecutor task_executor;
stats = reference_processor()->process_discovered_references( stats = reference_processor()->process_discovered_references(
&_is_alive_closure, &keep_alive, &evac_followers, &task_executor, &_is_alive_closure, &keep_alive, &evac_followers, &task_executor,
&_gc_timer); &pt);
} else { } else {
stats = reference_processor()->process_discovered_references( stats = reference_processor()->process_discovered_references(
&_is_alive_closure, &keep_alive, &evac_followers, NULL, &_gc_timer); &_is_alive_closure, &keep_alive, &evac_followers, NULL, &pt);
} }
_gc_tracer.report_gc_reference_stats(stats); _gc_tracer.report_gc_reference_stats(stats);
pt.print_all_references();
// Enqueue reference objects discovered during scavenge. // Enqueue reference objects discovered during scavenge.
if (reference_processor()->processing_is_mt()) { if (reference_processor()->processing_is_mt()) {
PSRefProcTaskExecutor task_executor; PSRefProcTaskExecutor task_executor;
reference_processor()->enqueue_discovered_references(&task_executor); reference_processor()->enqueue_discovered_references(&task_executor, &pt);
} else { } else {
reference_processor()->enqueue_discovered_references(NULL); reference_processor()->enqueue_discovered_references(NULL, &pt);
} }
pt.print_enqueue_phase();
} }
{ {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -661,11 +661,13 @@ void DefNewGeneration::collect(bool full,
FastKeepAliveClosure keep_alive(this, &scan_weak_ref); FastKeepAliveClosure keep_alive(this, &scan_weak_ref);
ReferenceProcessor* rp = ref_processor(); ReferenceProcessor* rp = ref_processor();
rp->setup_policy(clear_all_soft_refs); rp->setup_policy(clear_all_soft_refs);
ReferenceProcessorPhaseTimes pt(_gc_timer, rp->num_q());
const ReferenceProcessorStats& stats = const ReferenceProcessorStats& stats =
rp->process_discovered_references(&is_alive, &keep_alive, &evacuate_followers, rp->process_discovered_references(&is_alive, &keep_alive, &evacuate_followers,
NULL, _gc_timer); NULL, &pt);
gc_tracer.report_gc_reference_stats(stats); gc_tracer.report_gc_reference_stats(stats);
gc_tracer.report_tenuring_threshold(tenuring_threshold()); gc_tracer.report_tenuring_threshold(tenuring_threshold());
pt.print_all_references();
if (!_promotion_failed) { if (!_promotion_failed) {
// Swap the survivor spaces. // Swap the survivor spaces.

View File

@ -210,9 +210,11 @@ void GenMarkSweep::mark_sweep_phase1(bool clear_all_softrefs) {
GCTraceTime(Debug, gc, phases) tm_m("Reference Processing", gc_timer()); GCTraceTime(Debug, gc, phases) tm_m("Reference Processing", gc_timer());
ref_processor()->setup_policy(clear_all_softrefs); ref_processor()->setup_policy(clear_all_softrefs);
ReferenceProcessorPhaseTimes pt(_gc_timer, ref_processor()->num_q());
const ReferenceProcessorStats& stats = const ReferenceProcessorStats& stats =
ref_processor()->process_discovered_references( ref_processor()->process_discovered_references(
&is_alive, &keep_alive, &follow_stack_closure, NULL, _gc_timer); &is_alive, &keep_alive, &follow_stack_closure, NULL, &pt);
pt.print_all_references();
gc_tracer()->report_gc_reference_stats(stats); gc_tracer()->report_gc_reference_stats(stats);
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2012, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2012, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -78,8 +78,8 @@ class GCPhase {
class PhasesStack { class PhasesStack {
public: public:
// Set to 5, since Reference processing needs it. // Set to 6, since Reference processing needs it.
static const int PHASE_LEVELS = 5; static const int PHASE_LEVELS = 6;
private: private:
int _phase_indices[PHASE_LEVELS]; int _phase_indices[PHASE_LEVELS];

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2012, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2012, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -428,6 +428,7 @@ class PhaseSender : public PhaseVisitor {
case 1: send_phase<EventGCPhasePauseLevel1>(phase); break; case 1: send_phase<EventGCPhasePauseLevel1>(phase); break;
case 2: send_phase<EventGCPhasePauseLevel2>(phase); break; case 2: send_phase<EventGCPhasePauseLevel2>(phase); break;
case 3: send_phase<EventGCPhasePauseLevel3>(phase); break; case 3: send_phase<EventGCPhasePauseLevel3>(phase); break;
case 4: send_phase<EventGCPhasePauseLevel4>(phase); break;
default: /* Ignore sending this phase */ break; default: /* Ignore sending this phase */ break;
} }
} }

View File

@ -387,7 +387,9 @@ void GenCollectedHeap::collect_generation(Generation* gen, bool full, size_t siz
} }
gen->collect(full, clear_soft_refs, size, is_tlab); gen->collect(full, clear_soft_refs, size, is_tlab);
if (!rp->enqueuing_is_done()) { if (!rp->enqueuing_is_done()) {
rp->enqueue_discovered_references(); ReferenceProcessorPhaseTimes pt(NULL, rp->num_q());
rp->enqueue_discovered_references(NULL, &pt);
pt.print_enqueue_phase();
} else { } else {
rp->set_enqueuing_is_done(false); rp->set_enqueuing_is_done(false);
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -179,7 +179,7 @@ void ReferenceProcessor::update_soft_ref_master_clock() {
// past clock value. // past clock value.
} }
size_t ReferenceProcessor::total_count(DiscoveredList lists[]) { size_t ReferenceProcessor::total_count(DiscoveredList lists[]) const {
size_t total = 0; size_t total = 0;
for (uint i = 0; i < _max_num_q; ++i) { for (uint i = 0; i < _max_num_q; ++i) {
total += lists[i].length(); total += lists[i].length();
@ -192,7 +192,9 @@ ReferenceProcessorStats ReferenceProcessor::process_discovered_references(
OopClosure* keep_alive, OopClosure* keep_alive,
VoidClosure* complete_gc, VoidClosure* complete_gc,
AbstractRefProcTaskExecutor* task_executor, AbstractRefProcTaskExecutor* task_executor,
GCTimer* gc_timer) { ReferenceProcessorPhaseTimes* phase_times) {
double start_time = os::elapsedTime();
assert(!enqueuing_is_done(), "If here enqueuing should not be complete"); assert(!enqueuing_is_done(), "If here enqueuing should not be complete");
// Stop treating discovered references specially. // Stop treating discovered references specially.
@ -208,40 +210,39 @@ ReferenceProcessorStats ReferenceProcessor::process_discovered_references(
_soft_ref_timestamp_clock = java_lang_ref_SoftReference::clock(); _soft_ref_timestamp_clock = java_lang_ref_SoftReference::clock();
ReferenceProcessorStats stats( ReferenceProcessorStats stats(total_count(_discoveredSoftRefs),
total_count(_discoveredSoftRefs),
total_count(_discoveredWeakRefs), total_count(_discoveredWeakRefs),
total_count(_discoveredFinalRefs), total_count(_discoveredFinalRefs),
total_count(_discoveredPhantomRefs)); total_count(_discoveredPhantomRefs));
// Soft references // Soft references
{ {
GCTraceTime(Debug, gc, ref) tt("SoftReference", gc_timer); RefProcPhaseTimesTracker tt(REF_SOFT, phase_times, this);
process_discovered_reflist(_discoveredSoftRefs, _current_soft_ref_policy, true, process_discovered_reflist(_discoveredSoftRefs, _current_soft_ref_policy, true,
is_alive, keep_alive, complete_gc, task_executor); is_alive, keep_alive, complete_gc, task_executor, phase_times);
} }
update_soft_ref_master_clock(); update_soft_ref_master_clock();
// Weak references // Weak references
{ {
GCTraceTime(Debug, gc, ref) tt("WeakReference", gc_timer); RefProcPhaseTimesTracker tt(REF_WEAK, phase_times, this);
process_discovered_reflist(_discoveredWeakRefs, NULL, true, process_discovered_reflist(_discoveredWeakRefs, NULL, true,
is_alive, keep_alive, complete_gc, task_executor); is_alive, keep_alive, complete_gc, task_executor, phase_times);
} }
// Final references // Final references
{ {
GCTraceTime(Debug, gc, ref) tt("FinalReference", gc_timer); RefProcPhaseTimesTracker tt(REF_FINAL, phase_times, this);
process_discovered_reflist(_discoveredFinalRefs, NULL, false, process_discovered_reflist(_discoveredFinalRefs, NULL, false,
is_alive, keep_alive, complete_gc, task_executor); is_alive, keep_alive, complete_gc, task_executor, phase_times);
} }
// Phantom references // Phantom references
{ {
GCTraceTime(Debug, gc, ref) tt("PhantomReference", gc_timer); RefProcPhaseTimesTracker tt(REF_PHANTOM, phase_times, this);
process_discovered_reflist(_discoveredPhantomRefs, NULL, true, process_discovered_reflist(_discoveredPhantomRefs, NULL, true,
is_alive, keep_alive, complete_gc, task_executor); is_alive, keep_alive, complete_gc, task_executor, phase_times);
} }
// Weak global JNI references. It would make more sense (semantically) to // Weak global JNI references. It would make more sense (semantically) to
@ -250,15 +251,15 @@ ReferenceProcessorStats ReferenceProcessor::process_discovered_references(
// thus use JNI weak references to circumvent the phantom references and // thus use JNI weak references to circumvent the phantom references and
// resurrect a "post-mortem" object. // resurrect a "post-mortem" object.
{ {
GCTraceTime(Debug, gc, ref) tt("JNI Weak Reference", gc_timer); GCTraceTime(Debug, gc, ref) tt("JNI Weak Reference", phase_times->gc_timer());
if (task_executor != NULL) { if (task_executor != NULL) {
task_executor->set_single_threaded_mode(); task_executor->set_single_threaded_mode();
} }
process_phaseJNI(is_alive, keep_alive, complete_gc); process_phaseJNI(is_alive, keep_alive, complete_gc);
} }
log_debug(gc, ref)("Ref Counts: Soft: " SIZE_FORMAT " Weak: " SIZE_FORMAT " Final: " SIZE_FORMAT " Phantom: " SIZE_FORMAT, phase_times->set_total_time_ms((os::elapsedTime() - start_time) * 1000);
stats.soft_count(), stats.weak_count(), stats.final_count(), stats.phantom_count());
log_develop_trace(gc, ref)("JNI Weak Reference count: " SIZE_FORMAT, count_jni_refs()); log_develop_trace(gc, ref)("JNI Weak Reference count: " SIZE_FORMAT, count_jni_refs());
return stats; return stats;
@ -289,10 +290,11 @@ void ReferenceProcessor::process_phaseJNI(BoolObjectClosure* is_alive,
complete_gc->do_void(); complete_gc->do_void();
} }
void ReferenceProcessor::enqueue_discovered_references(AbstractRefProcTaskExecutor* task_executor) { void ReferenceProcessor::enqueue_discovered_references(AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times) {
// Enqueue references that are not made active again, and // Enqueue references that are not made active again, and
// clear the decks for the next collection (cycle). // clear the decks for the next collection (cycle).
enqueue_discovered_reflists(task_executor); enqueue_discovered_reflists(task_executor, phase_times);
// Stop treating discovered references specially. // Stop treating discovered references specially.
disable_discovery(); disable_discovery();
@ -345,11 +347,14 @@ class RefProcEnqueueTask: public AbstractRefProcTaskExecutor::EnqueueTask {
public: public:
RefProcEnqueueTask(ReferenceProcessor& ref_processor, RefProcEnqueueTask(ReferenceProcessor& ref_processor,
DiscoveredList discovered_refs[], DiscoveredList discovered_refs[],
int n_queues) int n_queues,
: EnqueueTask(ref_processor, discovered_refs, n_queues) ReferenceProcessorPhaseTimes* phase_times)
: EnqueueTask(ref_processor, discovered_refs, n_queues, phase_times)
{ } { }
virtual void work(unsigned int work_id) { virtual void work(unsigned int work_id) {
RefProcWorkerTimeTracker tt(ReferenceProcessorPhaseTimes::RefEnqueue, _phase_times, work_id);
assert(work_id < (unsigned int)_ref_processor.max_num_q(), "Index out-of-bounds"); assert(work_id < (unsigned int)_ref_processor.max_num_q(), "Index out-of-bounds");
// Simplest first cut: static partitioning. // Simplest first cut: static partitioning.
int index = work_id; int index = work_id;
@ -369,10 +374,19 @@ public:
}; };
// Enqueue references that are not made active again // Enqueue references that are not made active again
void ReferenceProcessor::enqueue_discovered_reflists(AbstractRefProcTaskExecutor* task_executor) { void ReferenceProcessor::enqueue_discovered_reflists(AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times) {
ReferenceProcessorStats stats(total_count(_discoveredSoftRefs),
total_count(_discoveredWeakRefs),
total_count(_discoveredFinalRefs),
total_count(_discoveredPhantomRefs));
RefProcEnqueueTimeTracker tt(phase_times, stats);
if (_processing_is_mt && task_executor != NULL) { if (_processing_is_mt && task_executor != NULL) {
// Parallel code // Parallel code
RefProcEnqueueTask tsk(*this, _discovered_refs, _max_num_q); RefProcEnqueueTask tsk(*this, _discovered_refs, _max_num_q, phase_times);
task_executor->execute(tsk); task_executor->execute(tsk);
} else { } else {
// Serial code: call the parent class's implementation // Serial code: call the parent class's implementation
@ -598,19 +612,46 @@ void ReferenceProcessor::abandon_partial_discovery() {
} }
} }
size_t ReferenceProcessor::total_reference_count(ReferenceType type) const {
DiscoveredList* list = NULL;
switch (type) {
case REF_SOFT:
list = _discoveredSoftRefs;
break;
case REF_WEAK:
list = _discoveredWeakRefs;
break;
case REF_FINAL:
list = _discoveredFinalRefs;
break;
case REF_PHANTOM:
list = _discoveredPhantomRefs;
break;
case REF_OTHER:
case REF_NONE:
default:
ShouldNotReachHere();
}
return total_count(list);
}
class RefProcPhase1Task: public AbstractRefProcTaskExecutor::ProcessTask { class RefProcPhase1Task: public AbstractRefProcTaskExecutor::ProcessTask {
public: public:
RefProcPhase1Task(ReferenceProcessor& ref_processor, RefProcPhase1Task(ReferenceProcessor& ref_processor,
DiscoveredList refs_lists[], DiscoveredList refs_lists[],
ReferencePolicy* policy, ReferencePolicy* policy,
bool marks_oops_alive) bool marks_oops_alive,
: ProcessTask(ref_processor, refs_lists, marks_oops_alive), ReferenceProcessorPhaseTimes* phase_times)
: ProcessTask(ref_processor, refs_lists, marks_oops_alive, phase_times),
_policy(policy) _policy(policy)
{ } { }
virtual void work(unsigned int i, BoolObjectClosure& is_alive, virtual void work(unsigned int i, BoolObjectClosure& is_alive,
OopClosure& keep_alive, OopClosure& keep_alive,
VoidClosure& complete_gc) VoidClosure& complete_gc)
{ {
RefProcWorkerTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase1, _phase_times, i);
_ref_processor.process_phase1(_refs_lists[i], _policy, _ref_processor.process_phase1(_refs_lists[i], _policy,
&is_alive, &keep_alive, &complete_gc); &is_alive, &keep_alive, &complete_gc);
} }
@ -622,13 +663,16 @@ class RefProcPhase2Task: public AbstractRefProcTaskExecutor::ProcessTask {
public: public:
RefProcPhase2Task(ReferenceProcessor& ref_processor, RefProcPhase2Task(ReferenceProcessor& ref_processor,
DiscoveredList refs_lists[], DiscoveredList refs_lists[],
bool marks_oops_alive) bool marks_oops_alive,
: ProcessTask(ref_processor, refs_lists, marks_oops_alive) ReferenceProcessorPhaseTimes* phase_times)
: ProcessTask(ref_processor, refs_lists, marks_oops_alive, phase_times)
{ } { }
virtual void work(unsigned int i, BoolObjectClosure& is_alive, virtual void work(unsigned int i, BoolObjectClosure& is_alive,
OopClosure& keep_alive, OopClosure& keep_alive,
VoidClosure& complete_gc) VoidClosure& complete_gc)
{ {
RefProcWorkerTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase2, _phase_times, i);
_ref_processor.process_phase2(_refs_lists[i], _ref_processor.process_phase2(_refs_lists[i],
&is_alive, &keep_alive, &complete_gc); &is_alive, &keep_alive, &complete_gc);
} }
@ -639,14 +683,17 @@ public:
RefProcPhase3Task(ReferenceProcessor& ref_processor, RefProcPhase3Task(ReferenceProcessor& ref_processor,
DiscoveredList refs_lists[], DiscoveredList refs_lists[],
bool clear_referent, bool clear_referent,
bool marks_oops_alive) bool marks_oops_alive,
: ProcessTask(ref_processor, refs_lists, marks_oops_alive), ReferenceProcessorPhaseTimes* phase_times)
: ProcessTask(ref_processor, refs_lists, marks_oops_alive, phase_times),
_clear_referent(clear_referent) _clear_referent(clear_referent)
{ } { }
virtual void work(unsigned int i, BoolObjectClosure& is_alive, virtual void work(unsigned int i, BoolObjectClosure& is_alive,
OopClosure& keep_alive, OopClosure& keep_alive,
VoidClosure& complete_gc) VoidClosure& complete_gc)
{ {
RefProcWorkerTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase3, _phase_times, i);
_ref_processor.process_phase3(_refs_lists[i], _clear_referent, _ref_processor.process_phase3(_refs_lists[i], _clear_referent,
&is_alive, &keep_alive, &complete_gc); &is_alive, &keep_alive, &complete_gc);
} }
@ -776,9 +823,13 @@ void ReferenceProcessor::process_discovered_reflist(
BoolObjectClosure* is_alive, BoolObjectClosure* is_alive,
OopClosure* keep_alive, OopClosure* keep_alive,
VoidClosure* complete_gc, VoidClosure* complete_gc,
AbstractRefProcTaskExecutor* task_executor) AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times)
{ {
bool mt_processing = task_executor != NULL && _processing_is_mt; bool mt_processing = task_executor != NULL && _processing_is_mt;
phase_times->set_processing_is_mt(mt_processing);
// If discovery used MT and a dynamic number of GC threads, then // If discovery used MT and a dynamic number of GC threads, then
// the queues must be balanced for correctness if fewer than the // the queues must be balanced for correctness if fewer than the
// maximum number of queues were used. The number of queue used // maximum number of queues were used. The number of queue used
@ -789,6 +840,7 @@ void ReferenceProcessor::process_discovered_reflist(
if ((mt_processing && ParallelRefProcBalancingEnabled) || if ((mt_processing && ParallelRefProcBalancingEnabled) ||
must_balance) { must_balance) {
RefProcBalanceQueuesTimeTracker tt(phase_times);
balance_queues(refs_lists); balance_queues(refs_lists);
} }
@ -798,8 +850,10 @@ void ReferenceProcessor::process_discovered_reflist(
// policy reasons. Keep alive the transitive closure of all // policy reasons. Keep alive the transitive closure of all
// such referents. // such referents.
if (policy != NULL) { if (policy != NULL) {
RefProcParPhaseTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase1, phase_times);
if (mt_processing) { if (mt_processing) {
RefProcPhase1Task phase1(*this, refs_lists, policy, true /*marks_oops_alive*/); RefProcPhase1Task phase1(*this, refs_lists, policy, true /*marks_oops_alive*/, phase_times);
task_executor->execute(phase1); task_executor->execute(phase1);
} else { } else {
for (uint i = 0; i < _max_num_q; i++) { for (uint i = 0; i < _max_num_q; i++) {
@ -814,19 +868,26 @@ void ReferenceProcessor::process_discovered_reflist(
// Phase 2: // Phase 2:
// . Traverse the list and remove any refs whose referents are alive. // . Traverse the list and remove any refs whose referents are alive.
{
RefProcParPhaseTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase2, phase_times);
if (mt_processing) { if (mt_processing) {
RefProcPhase2Task phase2(*this, refs_lists, !discovery_is_atomic() /*marks_oops_alive*/); RefProcPhase2Task phase2(*this, refs_lists, !discovery_is_atomic() /*marks_oops_alive*/, phase_times);
task_executor->execute(phase2); task_executor->execute(phase2);
} else { } else {
for (uint i = 0; i < _max_num_q; i++) { for (uint i = 0; i < _max_num_q; i++) {
process_phase2(refs_lists[i], is_alive, keep_alive, complete_gc); process_phase2(refs_lists[i], is_alive, keep_alive, complete_gc);
} }
} }
}
// Phase 3: // Phase 3:
// . Traverse the list and process referents as appropriate. // . Traverse the list and process referents as appropriate.
{
RefProcParPhaseTimeTracker tt(ReferenceProcessorPhaseTimes::RefPhase3, phase_times);
if (mt_processing) { if (mt_processing) {
RefProcPhase3Task phase3(*this, refs_lists, clear_referent, true /*marks_oops_alive*/); RefProcPhase3Task phase3(*this, refs_lists, clear_referent, true /*marks_oops_alive*/, phase_times);
task_executor->execute(phase3); task_executor->execute(phase3);
} else { } else {
for (uint i = 0; i < _max_num_q; i++) { for (uint i = 0; i < _max_num_q; i++) {
@ -835,6 +896,7 @@ void ReferenceProcessor::process_discovered_reflist(
} }
} }
} }
}
inline DiscoveredList* ReferenceProcessor::get_discovered_list(ReferenceType rt) { inline DiscoveredList* ReferenceProcessor::get_discovered_list(ReferenceType rt) {
uint id = 0; uint id = 0;
@ -1196,4 +1258,3 @@ const char* ReferenceProcessor::list_name(uint i) {
ShouldNotReachHere(); ShouldNotReachHere();
return NULL; return NULL;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -26,6 +26,7 @@
#define SHARE_VM_GC_SHARED_REFERENCEPROCESSOR_HPP #define SHARE_VM_GC_SHARED_REFERENCEPROCESSOR_HPP
#include "gc/shared/referencePolicy.hpp" #include "gc/shared/referencePolicy.hpp"
#include "gc/shared/referenceProcessorPhaseTimes.hpp"
#include "gc/shared/referenceProcessorStats.hpp" #include "gc/shared/referenceProcessorStats.hpp"
#include "memory/referenceType.hpp" #include "memory/referenceType.hpp"
#include "oops/instanceRefKlass.hpp" #include "oops/instanceRefKlass.hpp"
@ -168,7 +169,7 @@ public:
class ReferenceProcessor : public CHeapObj<mtGC> { class ReferenceProcessor : public CHeapObj<mtGC> {
private: private:
size_t total_count(DiscoveredList lists[]); size_t total_count(DiscoveredList lists[]) const;
protected: protected:
// The SoftReference master timestamp clock // The SoftReference master timestamp clock
@ -242,7 +243,8 @@ class ReferenceProcessor : public CHeapObj<mtGC> {
BoolObjectClosure* is_alive, BoolObjectClosure* is_alive,
OopClosure* keep_alive, OopClosure* keep_alive,
VoidClosure* complete_gc, VoidClosure* complete_gc,
AbstractRefProcTaskExecutor* task_executor); AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times);
void process_phaseJNI(BoolObjectClosure* is_alive, void process_phaseJNI(BoolObjectClosure* is_alive,
OopClosure* keep_alive, OopClosure* keep_alive,
@ -310,7 +312,8 @@ class ReferenceProcessor : public CHeapObj<mtGC> {
// occupying the i / _num_q slot. // occupying the i / _num_q slot.
const char* list_name(uint i); const char* list_name(uint i);
void enqueue_discovered_reflists(AbstractRefProcTaskExecutor* task_executor); void enqueue_discovered_reflists(AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times);
protected: protected:
// "Preclean" the given discovered reference list // "Preclean" the given discovered reference list
@ -420,16 +423,19 @@ class ReferenceProcessor : public CHeapObj<mtGC> {
OopClosure* keep_alive, OopClosure* keep_alive,
VoidClosure* complete_gc, VoidClosure* complete_gc,
AbstractRefProcTaskExecutor* task_executor, AbstractRefProcTaskExecutor* task_executor,
GCTimer *gc_timer); ReferenceProcessorPhaseTimes* phase_times);
// Enqueue references at end of GC (called by the garbage collector) // Enqueue references at end of GC (called by the garbage collector)
void enqueue_discovered_references(AbstractRefProcTaskExecutor* task_executor = NULL); void enqueue_discovered_references(AbstractRefProcTaskExecutor* task_executor,
ReferenceProcessorPhaseTimes* phase_times);
// If a discovery is in process that is being superceded, abandon it: all // If a discovery is in process that is being superceded, abandon it: all
// the discovered lists will be empty, and all the objects on them will // the discovered lists will be empty, and all the objects on them will
// have NULL discovered fields. Must be called only at a safepoint. // have NULL discovered fields. Must be called only at a safepoint.
void abandon_partial_discovery(); void abandon_partial_discovery();
size_t total_reference_count(ReferenceType rt) const;
// debugging // debugging
void verify_no_references_recorded() PRODUCT_RETURN; void verify_no_references_recorded() PRODUCT_RETURN;
void verify_referent(oop obj) PRODUCT_RETURN; void verify_referent(oop obj) PRODUCT_RETURN;
@ -586,9 +592,11 @@ class AbstractRefProcTaskExecutor::ProcessTask {
protected: protected:
ProcessTask(ReferenceProcessor& ref_processor, ProcessTask(ReferenceProcessor& ref_processor,
DiscoveredList refs_lists[], DiscoveredList refs_lists[],
bool marks_oops_alive) bool marks_oops_alive,
ReferenceProcessorPhaseTimes* phase_times)
: _ref_processor(ref_processor), : _ref_processor(ref_processor),
_refs_lists(refs_lists), _refs_lists(refs_lists),
_phase_times(phase_times),
_marks_oops_alive(marks_oops_alive) _marks_oops_alive(marks_oops_alive)
{ } { }
@ -604,6 +612,7 @@ public:
protected: protected:
ReferenceProcessor& _ref_processor; ReferenceProcessor& _ref_processor;
DiscoveredList* _refs_lists; DiscoveredList* _refs_lists;
ReferenceProcessorPhaseTimes* _phase_times;
const bool _marks_oops_alive; const bool _marks_oops_alive;
}; };
@ -612,10 +621,12 @@ class AbstractRefProcTaskExecutor::EnqueueTask {
protected: protected:
EnqueueTask(ReferenceProcessor& ref_processor, EnqueueTask(ReferenceProcessor& ref_processor,
DiscoveredList refs_lists[], DiscoveredList refs_lists[],
int n_queues) int n_queues,
ReferenceProcessorPhaseTimes* phase_times)
: _ref_processor(ref_processor), : _ref_processor(ref_processor),
_refs_lists(refs_lists), _refs_lists(refs_lists),
_n_queues(n_queues) _n_queues(n_queues),
_phase_times(phase_times)
{ } { }
public: public:
@ -624,6 +635,7 @@ public:
protected: protected:
ReferenceProcessor& _ref_processor; ReferenceProcessor& _ref_processor;
DiscoveredList* _refs_lists; DiscoveredList* _refs_lists;
ReferenceProcessorPhaseTimes* _phase_times;
int _n_queues; int _n_queues;
}; };

View File

@ -0,0 +1,475 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/
#include "precompiled.hpp"
#include "gc/shared/gcTimer.hpp"
#include "gc/shared/referenceProcessorPhaseTimes.hpp"
#include "gc/shared/referenceProcessor.inline.hpp"
#include "logging/log.hpp"
#include "logging/logStream.hpp"
RefProcWorkerTimeTracker::RefProcWorkerTimeTracker(ReferenceProcessorPhaseTimes::RefProcPhaseNumbers number,
ReferenceProcessorPhaseTimes* phase_times,
uint worker_id) :
_worker_time(NULL), _start_time(os::elapsedTime()), _worker_id(worker_id) {
assert (phase_times != NULL, "Invariant");
_worker_time = phase_times->worker_time_sec(phase_times->par_phase(number));
}
RefProcWorkerTimeTracker::RefProcWorkerTimeTracker(ReferenceProcessorPhaseTimes::RefProcParPhases phase,
ReferenceProcessorPhaseTimes* phase_times,
uint worker_id) :
_worker_time(NULL), _start_time(os::elapsedTime()), _worker_id(worker_id) {
assert (phase_times != NULL, "Invariant");
_worker_time = phase_times->worker_time_sec(phase);
}
RefProcWorkerTimeTracker::~RefProcWorkerTimeTracker() {
_worker_time->set(_worker_id, os::elapsedTime() - _start_time);
}
RefProcPhaseTimeBaseTracker::RefProcPhaseTimeBaseTracker(const char* title,
ReferenceProcessorPhaseTimes* phase_times) :
_title(title), _phase_times(phase_times), _start_ticks(), _end_ticks() {
assert(_phase_times != NULL, "Invariant");
_start_ticks.stamp();
if (_phase_times->gc_timer() != NULL) {
_phase_times->gc_timer()->register_gc_phase_start(_title, _start_ticks);
}
}
static const char* phase_enum_2_phase_string(ReferenceProcessorPhaseTimes::RefProcParPhases phase) {
switch(phase) {
case ReferenceProcessorPhaseTimes::SoftRefPhase1:
return "Phase1";
case ReferenceProcessorPhaseTimes::SoftRefPhase2:
case ReferenceProcessorPhaseTimes::WeakRefPhase2:
case ReferenceProcessorPhaseTimes::FinalRefPhase2:
case ReferenceProcessorPhaseTimes::PhantomRefPhase2:
return "Phase2";
case ReferenceProcessorPhaseTimes::SoftRefPhase3:
case ReferenceProcessorPhaseTimes::WeakRefPhase3:
case ReferenceProcessorPhaseTimes::FinalRefPhase3:
case ReferenceProcessorPhaseTimes::PhantomRefPhase3:
return "Phase3";
case ReferenceProcessorPhaseTimes::RefEnqueue:
return "Reference Enqueuing";
default:
ShouldNotReachHere();
return NULL;
}
}
static const char* Indents[6] = {"", " ", " ", " ", " ", " "};
Ticks RefProcPhaseTimeBaseTracker::end_ticks() {
// If ASSERT is defined, the default value of Ticks will be -2.
if (_end_ticks.value() <= 0) {
_end_ticks.stamp();
}
return _end_ticks;
}
double RefProcPhaseTimeBaseTracker::elapsed_time() {
jlong end_value = end_ticks().value();
return TimeHelper::counter_to_millis(end_value - _start_ticks.value());
}
RefProcPhaseTimeBaseTracker::~RefProcPhaseTimeBaseTracker() {
if (_phase_times->gc_timer() != NULL) {
Ticks ticks = end_ticks();
_phase_times->gc_timer()->register_gc_phase_end(ticks);
}
}
RefProcBalanceQueuesTimeTracker::RefProcBalanceQueuesTimeTracker(ReferenceProcessorPhaseTimes* phase_times) :
RefProcPhaseTimeBaseTracker("Balance queues", phase_times) {}
RefProcBalanceQueuesTimeTracker::~RefProcBalanceQueuesTimeTracker() {
double elapsed = elapsed_time();
phase_times()->set_balance_queues_time_ms(phase_times()->processing_ref_type(), elapsed);
}
#define ASSERT_REF_TYPE(ref_type) assert(ref_type >= REF_SOFT && ref_type <= REF_PHANTOM, \
"Invariant (%d)", (int)ref_type)
#define ASSERT_PHASE_NUMBER(phase_number) assert(phase_number >= ReferenceProcessorPhaseTimes::RefPhase1 && \
phase_number <= ReferenceProcessorPhaseTimes::RefPhaseMax, \
"Invariant (%d)", phase_number);
static const char* phase_number_2_string(ReferenceProcessorPhaseTimes::RefProcPhaseNumbers phase_number) {
ASSERT_PHASE_NUMBER(phase_number);
switch(phase_number) {
case ReferenceProcessorPhaseTimes::RefPhase1:
return "Phase1";
case ReferenceProcessorPhaseTimes::RefPhase2:
return "Phase2";
case ReferenceProcessorPhaseTimes::RefPhase3:
return "Phase3";
default:
ShouldNotReachHere();
return NULL;
}
}
RefProcParPhaseTimeTracker::RefProcParPhaseTimeTracker(ReferenceProcessorPhaseTimes::RefProcPhaseNumbers phase_number,
ReferenceProcessorPhaseTimes* phase_times) :
_phase_number(phase_number),
RefProcPhaseTimeBaseTracker(phase_number_2_string(phase_number), phase_times) {}
RefProcParPhaseTimeTracker::~RefProcParPhaseTimeTracker() {
double elapsed = elapsed_time();
ReferenceProcessorPhaseTimes::RefProcParPhases phase = phase_times()->par_phase(_phase_number);
phase_times()->set_par_phase_time_ms(phase, elapsed);
}
static const char* ref_type_2_string(ReferenceType ref_type) {
ASSERT_REF_TYPE(ref_type);
switch(ref_type) {
case REF_SOFT:
return "SoftReference";
case REF_WEAK:
return "WeakReference";
case REF_FINAL:
return "FinalReference";
case REF_PHANTOM:
return "PhantomReference";
default:
ShouldNotReachHere();
return NULL;
}
}
RefProcPhaseTimesTracker::RefProcPhaseTimesTracker(ReferenceType ref_type,
ReferenceProcessorPhaseTimes* phase_times,
ReferenceProcessor* rp) :
_rp(rp), RefProcPhaseTimeBaseTracker(ref_type_2_string(ref_type), phase_times) {
phase_times->set_processing_ref_type(ref_type);
size_t discovered = rp->total_reference_count(ref_type);
phase_times->set_ref_discovered(ref_type, discovered);
}
RefProcPhaseTimesTracker::~RefProcPhaseTimesTracker() {
double elapsed = elapsed_time();
ReferenceProcessorPhaseTimes* times = phase_times();
ReferenceType ref_type = times->processing_ref_type();
times->set_ref_proc_time_ms(ref_type, elapsed);
size_t after_count = _rp->total_reference_count(ref_type);
size_t discovered = times->ref_discovered(ref_type);
times->set_ref_cleared(ref_type, discovered - after_count);
}
RefProcEnqueueTimeTracker::RefProcEnqueueTimeTracker(ReferenceProcessorPhaseTimes* phase_times,
ReferenceProcessorStats& stats) :
RefProcPhaseTimeBaseTracker("Reference Enqueuing", phase_times) {
phase_times->set_ref_enqueued(REF_SOFT, stats.soft_count());
phase_times->set_ref_enqueued(REF_WEAK, stats.weak_count());
phase_times->set_ref_enqueued(REF_FINAL, stats.final_count());
phase_times->set_ref_enqueued(REF_PHANTOM, stats.phantom_count());
}
RefProcEnqueueTimeTracker::~RefProcEnqueueTimeTracker() {
double elapsed = elapsed_time();
phase_times()->set_par_phase_time_ms(ReferenceProcessorPhaseTimes::RefEnqueue, elapsed);
}
ReferenceProcessorPhaseTimes::ReferenceProcessorPhaseTimes(GCTimer* gc_timer, uint max_gc_threads) :
_gc_timer(gc_timer), _processing_is_mt(false) {
for (int i = 0; i < RefParPhaseMax; i++) {
_worker_time_sec[i] = new WorkerDataArray<double>(max_gc_threads, "Process lists (ms)");
_par_phase_time_ms[i] = uninitialized();
}
for (int i = 0; i < number_of_subclasses_of_ref; i++) {
_ref_proc_time_ms[i] = uninitialized();
_balance_queues_time_ms[i] = uninitialized();
_ref_cleared[i] = 0;
_ref_discovered[i] = 0;
_ref_enqueued[i] = 0;
}
}
inline int ref_type_2_index(ReferenceType ref_type) {
return ref_type - REF_SOFT;
}
#define ASSERT_PAR_PHASE(phase) assert(phase >= ReferenceProcessorPhaseTimes::SoftRefPhase1 && \
phase < ReferenceProcessorPhaseTimes::RefParPhaseMax, \
"Invariant (%d)", (int)phase);
WorkerDataArray<double>* ReferenceProcessorPhaseTimes::worker_time_sec(RefProcParPhases par_phase) const {
ASSERT_PAR_PHASE(par_phase);
return _worker_time_sec[par_phase];
}
double ReferenceProcessorPhaseTimes::par_phase_time_ms(RefProcParPhases par_phase) const {
ASSERT_PAR_PHASE(par_phase);
return _par_phase_time_ms[par_phase];
}
void ReferenceProcessorPhaseTimes::set_par_phase_time_ms(RefProcParPhases par_phase,
double par_phase_time_ms) {
ASSERT_PAR_PHASE(par_phase);
_par_phase_time_ms[par_phase] = par_phase_time_ms;
}
void ReferenceProcessorPhaseTimes::reset() {
for (int i = 0; i < RefParPhaseMax; i++) {
_worker_time_sec[i]->reset();
_par_phase_time_ms[i] = uninitialized();
}
for (int i = 0; i < number_of_subclasses_of_ref; i++) {
_ref_proc_time_ms[i] = uninitialized();
_balance_queues_time_ms[i] = uninitialized();
_ref_cleared[i] = 0;
_ref_discovered[i] = 0;
_ref_enqueued[i] = 0;
}
_total_time_ms = uninitialized();
_processing_is_mt = false;
}
ReferenceProcessorPhaseTimes::~ReferenceProcessorPhaseTimes() {
for (int i = 0; i < RefParPhaseMax; i++) {
delete _worker_time_sec[i];
}
}
double ReferenceProcessorPhaseTimes::ref_proc_time_ms(ReferenceType ref_type) const {
ASSERT_REF_TYPE(ref_type);
return _par_phase_time_ms[ref_type_2_index(ref_type)];
}
void ReferenceProcessorPhaseTimes::set_ref_proc_time_ms(ReferenceType ref_type,
double ref_proc_time_ms) {
ASSERT_REF_TYPE(ref_type);
_ref_proc_time_ms[ref_type_2_index(ref_type)] = ref_proc_time_ms;
}
size_t ReferenceProcessorPhaseTimes::ref_cleared(ReferenceType ref_type) const {
ASSERT_REF_TYPE(ref_type);
return _ref_cleared[ref_type_2_index(ref_type)];
}
void ReferenceProcessorPhaseTimes::set_ref_cleared(ReferenceType ref_type, size_t count) {
ASSERT_REF_TYPE(ref_type);
_ref_cleared[ref_type_2_index(ref_type)] = count;
}
size_t ReferenceProcessorPhaseTimes::ref_discovered(ReferenceType ref_type) const {
ASSERT_REF_TYPE(ref_type);
return _ref_discovered[ref_type_2_index(ref_type)];
}
void ReferenceProcessorPhaseTimes::set_ref_discovered(ReferenceType ref_type, size_t count) {
ASSERT_REF_TYPE(ref_type);
_ref_discovered[ref_type_2_index(ref_type)] = count;
}
size_t ReferenceProcessorPhaseTimes::ref_enqueued(ReferenceType ref_type) const {
ASSERT_REF_TYPE(ref_type);
return _ref_enqueued[ref_type_2_index(ref_type)];
}
void ReferenceProcessorPhaseTimes::set_ref_enqueued(ReferenceType ref_type, size_t count) {
ASSERT_REF_TYPE(ref_type);
_ref_enqueued[ref_type_2_index(ref_type)] = count;
}
double ReferenceProcessorPhaseTimes::balance_queues_time_ms(ReferenceType ref_type) const {
ASSERT_REF_TYPE(ref_type);
return _balance_queues_time_ms[ref_type_2_index(ref_type)];
}
void ReferenceProcessorPhaseTimes::set_balance_queues_time_ms(ReferenceType ref_type, double time_ms) {
ASSERT_REF_TYPE(ref_type);
_balance_queues_time_ms[ref_type_2_index(ref_type)] = time_ms;
}
ReferenceProcessorPhaseTimes::RefProcParPhases
ReferenceProcessorPhaseTimes::par_phase(RefProcPhaseNumbers phase_number) const {
ASSERT_PHASE_NUMBER(phase_number);
ASSERT_REF_TYPE(_processing_ref_type);
int result = SoftRefPhase1;
switch(_processing_ref_type) {
case REF_SOFT:
result = (int)SoftRefPhase1;
result += phase_number;
assert((RefProcParPhases)result >= SoftRefPhase1 &&
(RefProcParPhases)result <= SoftRefPhase3,
"Invariant (%d)", result);
break;
case REF_WEAK:
result = (int)WeakRefPhase2;
result += (phase_number - 1);
assert((RefProcParPhases)result >= WeakRefPhase2 &&
(RefProcParPhases)result <= WeakRefPhase3,
"Invariant (%d)", result);
break;
case REF_FINAL:
result = (int)FinalRefPhase2;
result += (phase_number - 1);
assert((RefProcParPhases)result >= FinalRefPhase2 &&
(RefProcParPhases)result <= FinalRefPhase3,
"Invariant (%d)", result);
break;
case REF_PHANTOM:
result = (int)PhantomRefPhase2;
result += (phase_number - 1);
assert((RefProcParPhases)result >= PhantomRefPhase2 &&
(RefProcParPhases)result <= PhantomRefPhase3,
"Invariant (%d)", result);
break;
default:
ShouldNotReachHere();
}
ASSERT_PAR_PHASE(result);
return (RefProcParPhases)result;
}
void ReferenceProcessorPhaseTimes::print_enqueue_phase(uint base_indent, bool print_total) const {
if (print_total) {
print_phase(RefEnqueue, base_indent);
}
log_debug(gc, phases, ref)("%sReference Counts: Soft: " SIZE_FORMAT " Weak: " SIZE_FORMAT
" Final: " SIZE_FORMAT " Phantom: " SIZE_FORMAT ,
Indents[base_indent + 1], ref_enqueued(REF_SOFT), ref_enqueued(REF_WEAK),
ref_enqueued(REF_FINAL), ref_enqueued(REF_PHANTOM));
}
#define TIME_FORMAT "%.1lfms"
void ReferenceProcessorPhaseTimes::print_all_references(uint base_indent, bool print_total) const {
if (print_total) {
LogTarget(Debug, gc, phases, ref) lt;
if (lt.is_enabled()) {
LogStream ls(lt);
ls.print_cr("%s%s: " TIME_FORMAT,
Indents[base_indent], "Reference Processing", total_time_ms());
}
}
uint next_indent = base_indent + 1;
print_reference(REF_SOFT, next_indent);
print_reference(REF_WEAK, next_indent);
print_reference(REF_FINAL, next_indent);
print_reference(REF_PHANTOM, next_indent);
}
void ReferenceProcessorPhaseTimes::print_reference(ReferenceType ref_type, uint base_indent) const {
LogTarget(Debug, gc, phases, ref) lt;
if (lt.is_enabled()) {
LogStream ls(lt);
uint next_indent = base_indent + 1;
ResourceMark rm;
ls.print_cr("%s%s: " TIME_FORMAT,
Indents[base_indent], ref_type_2_string(ref_type), ref_proc_time_ms(ref_type));
double balance_time = balance_queues_time_ms(ref_type);
if (balance_time != uninitialized()) {
ls.print_cr("%s%s " TIME_FORMAT, Indents[next_indent], "Balance queues:", balance_time);
}
switch(ref_type) {
case REF_SOFT:
print_phase(SoftRefPhase1, next_indent);
print_phase(SoftRefPhase2, next_indent);
print_phase(SoftRefPhase3, next_indent);
break;
case REF_WEAK:
print_phase(WeakRefPhase2, next_indent);
print_phase(WeakRefPhase3, next_indent);
break;
case REF_FINAL:
print_phase(FinalRefPhase2, next_indent);
print_phase(FinalRefPhase3, next_indent);
break;
case REF_PHANTOM:
print_phase(PhantomRefPhase2, next_indent);
print_phase(PhantomRefPhase3, next_indent);
break;
default:
ShouldNotReachHere();
}
ls.print_cr("%s%s " SIZE_FORMAT, Indents[next_indent], "Discovered:", ref_discovered(ref_type));
ls.print_cr("%s%s " SIZE_FORMAT, Indents[next_indent], "Cleared:", ref_cleared(ref_type));
}
}
void ReferenceProcessorPhaseTimes::print_phase(RefProcParPhases phase, uint indent) const {
double phase_time = par_phase_time_ms(phase);
if (phase_time != uninitialized()) {
LogTarget(Debug, gc, phases, ref) lt;
LogStream ls(lt);
ls.print_cr("%s%s%s " TIME_FORMAT,
Indents[indent],
phase_enum_2_phase_string(phase),
indent == 0 ? "" : ":", /* 0 indent logs don't need colon. */
phase_time);
LogTarget(Trace, gc, phases, ref) lt2;
if (_processing_is_mt && lt2.is_enabled()) {
LogStream ls(lt2);
ls.print("%s", Indents[indent + 1]);
// worker_time_sec is recorded in seconds but it will be printed in milliseconds.
worker_time_sec(phase)->print_summary_on(&ls, true);
}
}
}
#undef ASSERT_REF_TYPE
#undef ASSERT_PHASE_NUMBER
#undef ASSERT_PAR_PHASE
#undef TIME_FORMAT

View File

@ -0,0 +1,215 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/
#ifndef SHARE_VM_GC_SHARED_REFERENCEPROCESSORPHASETIMES_HPP
#define SHARE_VM_GC_SHARED_REFERENCEPROCESSORPHASETIMES_HPP
#include "gc/shared/referenceProcessorStats.hpp"
#include "gc/shared/workerDataArray.inline.hpp"
#include "memory/referenceType.hpp"
#include "utilities/ticks.hpp"
class DiscoveredList;
class GCTimer;
class ReferenceProcessorPhaseTimes : public CHeapObj<mtGC> {
public:
// Detailed phases that has parallel work.
enum RefProcParPhases {
SoftRefPhase1,
SoftRefPhase2,
SoftRefPhase3,
WeakRefPhase2,
WeakRefPhase3,
FinalRefPhase2,
FinalRefPhase3,
PhantomRefPhase2,
PhantomRefPhase3,
RefEnqueue,
RefParPhaseMax
};
// Sub-phases that are used when processing each j.l.Reference types.
// Only SoftReference has RefPhase1.
enum RefProcPhaseNumbers {
RefPhase1,
RefPhase2,
RefPhase3,
RefPhaseMax
};
private:
static const int number_of_subclasses_of_ref = REF_PHANTOM - REF_OTHER; // 5 - 1 = 4
// Records per thread information of each phase.
WorkerDataArray<double>* _worker_time_sec[RefParPhaseMax];
// Records elapsed time of each phase.
double _par_phase_time_ms[RefParPhaseMax];
// Total spent time for references.
// e.g. _ref_proc_time_ms[0] = _par_phase_time_ms[SoftRefPhase1] +
// _par_phase_time_ms[SoftRefPhase2] +
// _par_phase_time_ms[SoftRefPhase3] + extra time.
double _ref_proc_time_ms[number_of_subclasses_of_ref];
double _total_time_ms;
size_t _ref_cleared[number_of_subclasses_of_ref];
size_t _ref_discovered[number_of_subclasses_of_ref];
size_t _ref_enqueued[number_of_subclasses_of_ref];
double _balance_queues_time_ms[number_of_subclasses_of_ref];
bool _processing_is_mt;
// Currently processing reference type.
ReferenceType _processing_ref_type;
GCTimer* _gc_timer;
double par_phase_time_ms(RefProcParPhases phase) const;
double ref_proc_time_ms(ReferenceType ref_type) const;
double total_time_ms() const { return _total_time_ms; }
size_t ref_cleared(ReferenceType ref_type) const;
size_t ref_enqueued(ReferenceType ref_type) const;
double balance_queues_time_ms(ReferenceType ref_type) const;
void print_reference(ReferenceType ref_type, uint base_indent) const;
void print_phase(RefProcParPhases phase, uint indent) const;
public:
ReferenceProcessorPhaseTimes(GCTimer* gc_timer, uint max_gc_threads);
~ReferenceProcessorPhaseTimes();
static double uninitialized() { return -1.0; }
WorkerDataArray<double>* worker_time_sec(RefProcParPhases phase) const;
void set_par_phase_time_ms(RefProcParPhases phase, double par_phase_time_ms);
void set_ref_proc_time_ms(ReferenceType ref_type, double ref_proc_time_ms);
void set_total_time_ms(double total_time_ms) { _total_time_ms = total_time_ms; }
void set_ref_cleared(ReferenceType ref_type, size_t count);
size_t ref_discovered(ReferenceType ref_type) const;
void set_ref_discovered(ReferenceType ref_type, size_t count);
void set_ref_enqueued(ReferenceType ref_type, size_t count);
void set_balance_queues_time_ms(ReferenceType ref_type, double time_ms);
void set_processing_is_mt(bool processing_is_mt) { _processing_is_mt = processing_is_mt; }
ReferenceType processing_ref_type() const { return _processing_ref_type; }
void set_processing_ref_type(ReferenceType processing_ref_type) { _processing_ref_type = processing_ref_type; }
// Returns RefProcParPhases calculated from phase_number and _processing_ref_type.
RefProcParPhases par_phase(RefProcPhaseNumbers phase_number) const;
GCTimer* gc_timer() const { return _gc_timer; }
// Reset all fields. If not reset at next cycle, an assertion will fail.
void reset();
void print_enqueue_phase(uint base_indent = 0, bool print_total = true) const;
void print_all_references(uint base_indent = 0, bool print_total = true) const;
};
// Updates working time of each worker thread.
class RefProcWorkerTimeTracker : public StackObj {
protected:
WorkerDataArray<double>* _worker_time;
double _start_time;
uint _worker_id;
public:
RefProcWorkerTimeTracker(ReferenceProcessorPhaseTimes::RefProcPhaseNumbers number,
ReferenceProcessorPhaseTimes* phase_times,
uint worker_id);
RefProcWorkerTimeTracker(ReferenceProcessorPhaseTimes::RefProcParPhases phase,
ReferenceProcessorPhaseTimes* phase_times,
uint worker_id);
~RefProcWorkerTimeTracker();
};
class RefProcPhaseTimeBaseTracker : public StackObj {
protected:
const char* _title;
ReferenceProcessorPhaseTimes* _phase_times;
Ticks _start_ticks;
Ticks _end_ticks;
Ticks end_ticks();
double elapsed_time();
ReferenceProcessorPhaseTimes* phase_times() const { return _phase_times; }
// Print phase elapsed time with each worker information if MT processed.
void print_phase(ReferenceProcessorPhaseTimes::RefProcParPhases phase, uint indent);
public:
RefProcPhaseTimeBaseTracker(const char* title,
ReferenceProcessorPhaseTimes* phase_times);
~RefProcPhaseTimeBaseTracker();
};
// Updates queue balance time at ReferenceProcessorPhaseTimes and
// save it into GCTimer.
class RefProcBalanceQueuesTimeTracker : public RefProcPhaseTimeBaseTracker {
public:
RefProcBalanceQueuesTimeTracker(ReferenceProcessorPhaseTimes* phase_times);
~RefProcBalanceQueuesTimeTracker();
};
// Updates phase time at ReferenceProcessorPhaseTimes and save it into GCTimer.
class RefProcParPhaseTimeTracker : public RefProcPhaseTimeBaseTracker {
ReferenceProcessorPhaseTimes::RefProcPhaseNumbers _phase_number;
public:
RefProcParPhaseTimeTracker(ReferenceProcessorPhaseTimes::RefProcPhaseNumbers phase_number,
ReferenceProcessorPhaseTimes* phase_times);
~RefProcParPhaseTimeTracker();
};
// Updates phase time related information.
// - Each phase processing time, cleared/discovered reference counts and stats for each working threads if MT processed.
class RefProcPhaseTimesTracker : public RefProcPhaseTimeBaseTracker {
ReferenceProcessor* _rp;
public:
RefProcPhaseTimesTracker(ReferenceType ref_type,
ReferenceProcessorPhaseTimes* phase_times,
ReferenceProcessor* rp);
~RefProcPhaseTimesTracker();
};
// Updates enqueue time related information.
// - Enqueueing time, enqueued reference count and stats for each working thread if MT processed.
class RefProcEnqueueTimeTracker : public RefProcPhaseTimeBaseTracker {
public:
RefProcEnqueueTimeTracker(ReferenceProcessorPhaseTimes* phase_times,
ReferenceProcessorStats& stats);
~RefProcEnqueueTimeTracker();
};
#endif // SHARE_VM_GC_SHARED_REFERENCEPROCESSORPHASETIMES_HPP

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,7 +23,7 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/g1/workerDataArray.inline.hpp" #include "gc/shared/workerDataArray.inline.hpp"
#include "utilities/ostream.hpp" #include "utilities/ostream.hpp"
template <> template <>

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -22,8 +22,8 @@
* *
*/ */
#ifndef SHARE_VM_GC_G1_WORKERDATAARRAY_HPP #ifndef SHARE_VM_GC_SHARED_WORKERDATAARRAY_HPP
#define SHARE_VM_GC_G1_WORKERDATAARRAY_HPP #define SHARE_VM_GC_SHARED_WORKERDATAARRAY_HPP
#include "memory/allocation.hpp" #include "memory/allocation.hpp"
#include "utilities/debug.hpp" #include "utilities/debug.hpp"
@ -88,4 +88,4 @@ private:
void print_details_on(outputStream* out) const; void print_details_on(outputStream* out) const;
}; };
#endif // SHARE_VM_GC_G1_WORKERDATAARRAY_HPP #endif // SHARE_VM_GC_SHARED_WORKERDATAARRAY_HPP

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -22,10 +22,10 @@
* *
*/ */
#ifndef SHARE_VM_GC_G1_WORKERDATAARRAY_INLINE_HPP #ifndef SHARE_VM_GC_SHARED_WORKERDATAARRAY_INLINE_HPP
#define SHARE_VM_GC_G1_WORKERDATAARRAY_INLINE_HPP #define SHARE_VM_GC_SHARED_WORKERDATAARRAY_INLINE_HPP
#include "gc/g1/workerDataArray.hpp" #include "gc/shared/workerDataArray.hpp"
#include "memory/allocation.inline.hpp" #include "memory/allocation.inline.hpp"
#include "utilities/ostream.hpp" #include "utilities/ostream.hpp"
@ -166,4 +166,4 @@ void WorkerDataArray<T>::reset() {
} }
} }
#endif // SHARE_VM_GC_G1_WORKERDATAARRAY_INLINE_HPP #endif // SHARE_VM_GC_SHARED_WORKERDATAARRAY_INLINE_HPP

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -66,6 +66,7 @@ DEBUG_ONLY(size_t Test_log_prefix_prefixer(char* buf, size_t len);)
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, metaspace)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, metaspace)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, mmu)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, mmu)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases, ref)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases, start)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases, start)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases, task)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, phases, task)) \
LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, plab)) \ LOG_PREFIX(GCId::print_prefix, LOG_TAGS(gc, plab)) \

View File

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<!-- <!--
Copyright (c) 2012, 2016, Oracle and/or its affiliates. All rights reserved. Copyright (c) 2012, 2017, Oracle and/or its affiliates. All rights reserved.
DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
This code is free software; you can redistribute it and/or modify it This code is free software; you can redistribute it and/or modify it
@ -463,6 +463,11 @@ Declares a structure type that can be used in other events.
<value type="STRING" field="name" label="Name" /> <value type="STRING" field="name" label="Name" />
</event> </event>
<event id="GCPhasePauseLevel4" path="vm/gc/phases/pause_level_4" label="GC Phase Pause Level 4" has_thread="true">
<value type="UINT" field="gcId" label="GC Identifier" relation="GcId"/>
<value type="STRING" field="name" label="Name" />
</event>
<event id="GCPhaseConcurrent" path="vm/gc/phases/concurrent" label="GC Phase Concurrent" has_thread="true"> <event id="GCPhaseConcurrent" path="vm/gc/phases/concurrent" label="GC Phase Concurrent" has_thread="true">
<value type="UINT" field="gcId" label="GC Identifier" relation="GcId"/> <value type="UINT" field="gcId" label="GC Identifier" relation="GcId"/>
<value type="STRING" field="name" label="Name" /> <value type="STRING" field="name" label="Name" />

View File

@ -142,6 +142,9 @@ public class TestGCLogMessages {
// TLAB handling // TLAB handling
new LogMessageWithLevel("Prepare TLABs", Level.DEBUG), new LogMessageWithLevel("Prepare TLABs", Level.DEBUG),
new LogMessageWithLevel("Resize TLABs", Level.DEBUG), new LogMessageWithLevel("Resize TLABs", Level.DEBUG),
// Reference Processing
new LogMessageWithLevel("Reference Processing", Level.DEBUG),
new LogMessageWithLevel("Reference Enqueuing", Level.DEBUG),
new LogMessageWithLevelC2OrJVMCIOnly("DerivedPointerTable Update", Level.DEBUG), new LogMessageWithLevelC2OrJVMCIOnly("DerivedPointerTable Update", Level.DEBUG),
new LogMessageWithLevel("Start New Collection Set", Level.DEBUG), new LogMessageWithLevel("Start New Collection Set", Level.DEBUG),

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -31,31 +31,73 @@
* java.management * java.management
*/ */
import java.lang.ref.SoftReference;
import java.util.ArrayList;
import jdk.test.lib.process.OutputAnalyzer; import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools; import jdk.test.lib.process.ProcessTools;
public class TestPrintReferences { public class TestPrintReferences {
public static void main(String[] args) throws Exception { public static void main(String[] args) throws Exception {
ProcessBuilder pb_enabled = ProcessBuilder pb_enabled = ProcessTools.createJavaProcessBuilder("-Xlog:gc+phases+ref=debug",
ProcessTools.createJavaProcessBuilder("-Xlog:gc+ref=debug", "-Xmx10M", GCTest.class.getName()); "-XX:+UseG1GC",
"-Xmx10M",
GCTest.class.getName());
OutputAnalyzer output = new OutputAnalyzer(pb_enabled.start()); OutputAnalyzer output = new OutputAnalyzer(pb_enabled.start());
String countRegex = "[0-9]+ refs"; String indent_4 = " ";
String indent_6 = " ";
String indent_8 = " ";
String gcLogTimeRegex = ".* GC\\([0-9]+\\) ";
String countRegex = "[0-9]+";
String timeRegex = "[0-9]+[.,][0-9]+ms"; String timeRegex = "[0-9]+[.,][0-9]+ms";
String totalRegex = gcLogTimeRegex + indent_4 + "Reference Processing: " + timeRegex + "\n";
String balanceRegex = gcLogTimeRegex + indent_8 + "Balance queues: " + timeRegex + "\n";
String softRefRegex = gcLogTimeRegex + indent_6 + "SoftReference: " + timeRegex + "\n";
String weakRefRegex = gcLogTimeRegex + indent_6 + "WeakReference: " + timeRegex + "\n";
String finalRefRegex = gcLogTimeRegex + indent_6 + "FinalReference: " + timeRegex + "\n";
String phantomRefRegex = gcLogTimeRegex + indent_6 + "PhantomReference: " + timeRegex + "\n";
String refDetailRegex = gcLogTimeRegex + indent_8 + "Phase2: " + timeRegex + "\n" +
gcLogTimeRegex + indent_8 + "Phase3: " + timeRegex + "\n" +
gcLogTimeRegex + indent_8 + "Discovered: " + countRegex + "\n" +
gcLogTimeRegex + indent_8 + "Cleared: " + countRegex + "\n";
String softRefDetailRegex = gcLogTimeRegex + indent_8 + "Phase1: " + timeRegex + "\n" + refDetailRegex;
String enqueueRegex = gcLogTimeRegex + indent_4 + "Reference Enqueuing: " + timeRegex + "\n";
String enqueueDetailRegex = gcLogTimeRegex + indent_6 + "Reference Counts: Soft: " + countRegex +
" Weak: " + countRegex + " Final: " + countRegex + " Phantom: " + countRegex + "\n";
output.shouldMatch(".* GC\\([0-9]+\\) SoftReference " + timeRegex + "\n" + output.shouldMatch(/* Total Reference processing time */
".* GC\\([0-9]+\\) WeakReference " + timeRegex + "\n" + totalRegex +
".* GC\\([0-9]+\\) FinalReference " + timeRegex + "\n" + /* SoftReference processing */
".* GC\\([0-9]+\\) PhantomReference " + timeRegex + "\n" + softRefRegex + balanceRegex + softRefDetailRegex +
".* GC\\([0-9]+\\) JNI Weak Reference " + timeRegex + "\n" + /* WeakReference processing */
".* GC\\([0-9]+\\) Ref Counts: Soft: [0-9]+ Weak: [0-9]+ Final: [0-9]+ Phantom: [0-9]+\n"); weakRefRegex + balanceRegex + refDetailRegex +
/* FinalReference processing */
finalRefRegex + balanceRegex + refDetailRegex +
/* PhantomReference processing */
phantomRefRegex + balanceRegex + refDetailRegex +
/* Total Enqueuing time */
enqueueRegex +
/* Enqueued Stats */
enqueueDetailRegex
);
output.shouldHaveExitValue(0); output.shouldHaveExitValue(0);
} }
static class GCTest { static class GCTest {
static final int M = 1024 * 1024;
public static void main(String [] args) { public static void main(String [] args) {
System.gc();
ArrayList arrSoftRefs = new ArrayList();
// Populate to triger GC and then Reference related logs will be printed.
for (int i = 0; i < 10; i++) {
byte[] tmp = new byte[M];
arrSoftRefs.add(new SoftReference(tmp));
}
} }
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -22,7 +22,7 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/g1/workerDataArray.inline.hpp" #include "gc/shared/workerDataArray.inline.hpp"
#include "memory/resourceArea.hpp" #include "memory/resourceArea.hpp"
#include "unittest.hpp" #include "unittest.hpp"
#include "utilities/ostream.hpp" #include "utilities/ostream.hpp"