提交 9bd2f522 编写于 作者: J jcoomes

Merge

...@@ -293,7 +293,7 @@ void ConcurrentMarkThread::run() { ...@@ -293,7 +293,7 @@ void ConcurrentMarkThread::run() {
// Java thread is waiting for a full GC to happen (e.g., it // Java thread is waiting for a full GC to happen (e.g., it
// called System.gc() with +ExplicitGCInvokesConcurrent). // called System.gc() with +ExplicitGCInvokesConcurrent).
_sts.join(); _sts.join();
g1h->increment_full_collections_completed(true /* concurrent */); g1h->increment_old_marking_cycles_completed(true /* concurrent */);
_sts.leave(); _sts.leave();
} }
assert(_should_terminate, "just checking"); assert(_should_terminate, "just checking");
......
...@@ -1299,6 +1299,7 @@ bool G1CollectedHeap::do_collection(bool explicit_gc, ...@@ -1299,6 +1299,7 @@ bool G1CollectedHeap::do_collection(bool explicit_gc,
gc_prologue(true); gc_prologue(true);
increment_total_collections(true /* full gc */); increment_total_collections(true /* full gc */);
increment_old_marking_cycles_started();
size_t g1h_prev_used = used(); size_t g1h_prev_used = used();
assert(used() == recalculate_used(), "Should be equal"); assert(used() == recalculate_used(), "Should be equal");
...@@ -1492,22 +1493,28 @@ bool G1CollectedHeap::do_collection(bool explicit_gc, ...@@ -1492,22 +1493,28 @@ bool G1CollectedHeap::do_collection(bool explicit_gc,
JavaThread::dirty_card_queue_set().abandon_logs(); JavaThread::dirty_card_queue_set().abandon_logs();
assert(!G1DeferredRSUpdate assert(!G1DeferredRSUpdate
|| (G1DeferredRSUpdate && (dirty_card_queue_set().completed_buffers_num() == 0)), "Should not be any"); || (G1DeferredRSUpdate && (dirty_card_queue_set().completed_buffers_num() == 0)), "Should not be any");
}
_young_list->reset_sampled_info(); _young_list->reset_sampled_info();
// At this point there should be no regions in the // At this point there should be no regions in the
// entire heap tagged as young. // entire heap tagged as young.
assert( check_young_list_empty(true /* check_heap */), assert( check_young_list_empty(true /* check_heap */),
"young list should be empty at this point"); "young list should be empty at this point");
// Update the number of full collections that have been completed. // Update the number of full collections that have been completed.
increment_full_collections_completed(false /* concurrent */); increment_old_marking_cycles_completed(false /* concurrent */);
_hrs.verify_optional(); _hrs.verify_optional();
verify_region_sets_optional(); verify_region_sets_optional();
print_heap_after_gc();
// We must call G1MonitoringSupport::update_sizes() in the same scoping level
// as an active TraceMemoryManagerStats object (i.e. before the destructor for the
// TraceMemoryManagerStats is called) so that the G1 memory pools are updated
// before any GC notifications are raised.
g1mm()->update_sizes();
}
print_heap_after_gc();
g1mm()->update_sizes();
post_full_gc_dump(); post_full_gc_dump();
return true; return true;
...@@ -1888,7 +1895,8 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* policy_) : ...@@ -1888,7 +1895,8 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* policy_) :
_retained_old_gc_alloc_region(NULL), _retained_old_gc_alloc_region(NULL),
_expand_heap_after_alloc_failure(true), _expand_heap_after_alloc_failure(true),
_surviving_young_words(NULL), _surviving_young_words(NULL),
_full_collections_completed(0), _old_marking_cycles_started(0),
_old_marking_cycles_completed(0),
_in_cset_fast_test(NULL), _in_cset_fast_test(NULL),
_in_cset_fast_test_base(NULL), _in_cset_fast_test_base(NULL),
_dirty_cards_region_list(NULL), _dirty_cards_region_list(NULL),
...@@ -2360,7 +2368,16 @@ void G1CollectedHeap::allocate_dummy_regions() { ...@@ -2360,7 +2368,16 @@ void G1CollectedHeap::allocate_dummy_regions() {
} }
#endif // !PRODUCT #endif // !PRODUCT
void G1CollectedHeap::increment_full_collections_completed(bool concurrent) { void G1CollectedHeap::increment_old_marking_cycles_started() {
assert(_old_marking_cycles_started == _old_marking_cycles_completed ||
_old_marking_cycles_started == _old_marking_cycles_completed + 1,
err_msg("Wrong marking cycle count (started: %d, completed: %d)",
_old_marking_cycles_started, _old_marking_cycles_completed));
_old_marking_cycles_started++;
}
void G1CollectedHeap::increment_old_marking_cycles_completed(bool concurrent) {
MonitorLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag); MonitorLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
// We assume that if concurrent == true, then the caller is a // We assume that if concurrent == true, then the caller is a
...@@ -2368,11 +2385,6 @@ void G1CollectedHeap::increment_full_collections_completed(bool concurrent) { ...@@ -2368,11 +2385,6 @@ void G1CollectedHeap::increment_full_collections_completed(bool concurrent) {
// Set. If there's ever a cheap way to check this, we should add an // Set. If there's ever a cheap way to check this, we should add an
// assert here. // assert here.
// We have already incremented _total_full_collections at the start
// of the GC, so total_full_collections() represents how many full
// collections have been started.
unsigned int full_collections_started = total_full_collections();
// Given that this method is called at the end of a Full GC or of a // Given that this method is called at the end of a Full GC or of a
// concurrent cycle, and those can be nested (i.e., a Full GC can // concurrent cycle, and those can be nested (i.e., a Full GC can
// interrupt a concurrent cycle), the number of full collections // interrupt a concurrent cycle), the number of full collections
...@@ -2382,21 +2394,21 @@ void G1CollectedHeap::increment_full_collections_completed(bool concurrent) { ...@@ -2382,21 +2394,21 @@ void G1CollectedHeap::increment_full_collections_completed(bool concurrent) {
// This is the case for the inner caller, i.e. a Full GC. // This is the case for the inner caller, i.e. a Full GC.
assert(concurrent || assert(concurrent ||
(full_collections_started == _full_collections_completed + 1) || (_old_marking_cycles_started == _old_marking_cycles_completed + 1) ||
(full_collections_started == _full_collections_completed + 2), (_old_marking_cycles_started == _old_marking_cycles_completed + 2),
err_msg("for inner caller (Full GC): full_collections_started = %u " err_msg("for inner caller (Full GC): _old_marking_cycles_started = %u "
"is inconsistent with _full_collections_completed = %u", "is inconsistent with _old_marking_cycles_completed = %u",
full_collections_started, _full_collections_completed)); _old_marking_cycles_started, _old_marking_cycles_completed));
// This is the case for the outer caller, i.e. the concurrent cycle. // This is the case for the outer caller, i.e. the concurrent cycle.
assert(!concurrent || assert(!concurrent ||
(full_collections_started == _full_collections_completed + 1), (_old_marking_cycles_started == _old_marking_cycles_completed + 1),
err_msg("for outer caller (concurrent cycle): " err_msg("for outer caller (concurrent cycle): "
"full_collections_started = %u " "_old_marking_cycles_started = %u "
"is inconsistent with _full_collections_completed = %u", "is inconsistent with _old_marking_cycles_completed = %u",
full_collections_started, _full_collections_completed)); _old_marking_cycles_started, _old_marking_cycles_completed));
_full_collections_completed += 1; _old_marking_cycles_completed += 1;
// We need to clear the "in_progress" flag in the CM thread before // We need to clear the "in_progress" flag in the CM thread before
// we wake up any waiters (especially when ExplicitInvokesConcurrent // we wake up any waiters (especially when ExplicitInvokesConcurrent
...@@ -2432,7 +2444,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) { ...@@ -2432,7 +2444,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) {
assert_heap_not_locked(); assert_heap_not_locked();
unsigned int gc_count_before; unsigned int gc_count_before;
unsigned int full_gc_count_before; unsigned int old_marking_count_before;
bool retry_gc; bool retry_gc;
do { do {
...@@ -2443,7 +2455,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) { ...@@ -2443,7 +2455,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) {
// Read the GC count while holding the Heap_lock // Read the GC count while holding the Heap_lock
gc_count_before = total_collections(); gc_count_before = total_collections();
full_gc_count_before = total_full_collections(); old_marking_count_before = _old_marking_cycles_started;
} }
if (should_do_concurrent_full_gc(cause)) { if (should_do_concurrent_full_gc(cause)) {
...@@ -2458,7 +2470,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) { ...@@ -2458,7 +2470,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) {
VMThread::execute(&op); VMThread::execute(&op);
if (!op.pause_succeeded()) { if (!op.pause_succeeded()) {
if (full_gc_count_before == total_full_collections()) { if (old_marking_count_before == _old_marking_cycles_started) {
retry_gc = op.should_retry_gc(); retry_gc = op.should_retry_gc();
} else { } else {
// A Full GC happened while we were trying to schedule the // A Full GC happened while we were trying to schedule the
...@@ -2486,7 +2498,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) { ...@@ -2486,7 +2498,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) {
VMThread::execute(&op); VMThread::execute(&op);
} else { } else {
// Schedule a Full GC. // Schedule a Full GC.
VM_G1CollectFull op(gc_count_before, full_gc_count_before, cause); VM_G1CollectFull op(gc_count_before, old_marking_count_before, cause);
VMThread::execute(&op); VMThread::execute(&op);
} }
} }
...@@ -3613,7 +3625,7 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { ...@@ -3613,7 +3625,7 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) {
if (g1_policy()->during_initial_mark_pause()) { if (g1_policy()->during_initial_mark_pause()) {
// We are about to start a marking cycle, so we increment the // We are about to start a marking cycle, so we increment the
// full collection counter. // full collection counter.
increment_total_full_collections(); increment_old_marking_cycles_started();
} }
// if the log level is "finer" is on, we'll print long statistics information // if the log level is "finer" is on, we'll print long statistics information
// in the collector policy code, so let's not print this as the output // in the collector policy code, so let's not print this as the output
...@@ -3930,25 +3942,30 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { ...@@ -3930,25 +3942,30 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) {
gc_epilogue(false); gc_epilogue(false);
} }
}
// The closing of the inner scope, immediately above, will complete // The closing of the inner scope, immediately above, will complete
// logging at the "fine" level. The record_collection_pause_end() call // logging at the "fine" level. The record_collection_pause_end() call
// above will complete logging at the "finer" level. // above will complete logging at the "finer" level.
// //
// It is not yet to safe, however, to tell the concurrent mark to // It is not yet to safe, however, to tell the concurrent mark to
// start as we have some optional output below. We don't want the // start as we have some optional output below. We don't want the
// output from the concurrent mark thread interfering with this // output from the concurrent mark thread interfering with this
// logging output either. // logging output either.
_hrs.verify_optional(); _hrs.verify_optional();
verify_region_sets_optional(); verify_region_sets_optional();
TASKQUEUE_STATS_ONLY(if (ParallelGCVerbose) print_taskqueue_stats());
TASKQUEUE_STATS_ONLY(reset_taskqueue_stats());
TASKQUEUE_STATS_ONLY(if (ParallelGCVerbose) print_taskqueue_stats()); print_heap_after_gc();
TASKQUEUE_STATS_ONLY(reset_taskqueue_stats());
print_heap_after_gc(); // We must call G1MonitoringSupport::update_sizes() in the same scoping level
g1mm()->update_sizes(); // as an active TraceMemoryManagerStats object (i.e. before the destructor for the
// TraceMemoryManagerStats is called) so that the G1 memory pools are updated
// before any GC notifications are raised.
g1mm()->update_sizes();
}
if (G1SummarizeRSetStats && if (G1SummarizeRSetStats &&
(G1SummarizeRSetStatsPeriod > 0) && (G1SummarizeRSetStatsPeriod > 0) &&
......
...@@ -359,10 +359,13 @@ private: ...@@ -359,10 +359,13 @@ private:
// (c) cause == _g1_humongous_allocation // (c) cause == _g1_humongous_allocation
bool should_do_concurrent_full_gc(GCCause::Cause cause); bool should_do_concurrent_full_gc(GCCause::Cause cause);
// Keeps track of how many "full collections" (i.e., Full GCs or // Keeps track of how many "old marking cycles" (i.e., Full GCs or
// concurrent cycles) we have completed. The number of them we have // concurrent cycles) we have started.
// started is maintained in _total_full_collections in CollectedHeap. volatile unsigned int _old_marking_cycles_started;
volatile unsigned int _full_collections_completed;
// Keeps track of how many "old marking cycles" (i.e., Full GCs or
// concurrent cycles) we have completed.
volatile unsigned int _old_marking_cycles_completed;
// This is a non-product method that is helpful for testing. It is // This is a non-product method that is helpful for testing. It is
// called at the end of a GC and artificially expands the heap by // called at the end of a GC and artificially expands the heap by
...@@ -673,8 +676,12 @@ public: ...@@ -673,8 +676,12 @@ public:
(size_t) _in_cset_fast_test_length * sizeof(bool)); (size_t) _in_cset_fast_test_length * sizeof(bool));
} }
// This is called at the start of either a concurrent cycle or a Full
// GC to update the number of old marking cycles started.
void increment_old_marking_cycles_started();
// This is called at the end of either a concurrent cycle or a Full // This is called at the end of either a concurrent cycle or a Full
// GC to update the number of full collections completed. Those two // GC to update the number of old marking cycles completed. Those two
// can happen in a nested fashion, i.e., we start a concurrent // can happen in a nested fashion, i.e., we start a concurrent
// cycle, a Full GC happens half-way through it which ends first, // cycle, a Full GC happens half-way through it which ends first,
// and then the cycle notices that a Full GC happened and ends // and then the cycle notices that a Full GC happened and ends
...@@ -683,14 +690,14 @@ public: ...@@ -683,14 +690,14 @@ public:
// false, the caller is the inner caller in the nesting (i.e., the // false, the caller is the inner caller in the nesting (i.e., the
// Full GC). If concurrent is true, the caller is the outer caller // Full GC). If concurrent is true, the caller is the outer caller
// in this nesting (i.e., the concurrent cycle). Further nesting is // in this nesting (i.e., the concurrent cycle). Further nesting is
// not currently supported. The end of the this call also notifies // not currently supported. The end of this call also notifies
// the FullGCCount_lock in case a Java thread is waiting for a full // the FullGCCount_lock in case a Java thread is waiting for a full
// GC to happen (e.g., it called System.gc() with // GC to happen (e.g., it called System.gc() with
// +ExplicitGCInvokesConcurrent). // +ExplicitGCInvokesConcurrent).
void increment_full_collections_completed(bool concurrent); void increment_old_marking_cycles_completed(bool concurrent);
unsigned int full_collections_completed() { unsigned int old_marking_cycles_completed() {
return _full_collections_completed; return _old_marking_cycles_completed;
} }
G1HRPrinter* hr_printer() { return &_hr_printer; } G1HRPrinter* hr_printer() { return &_hr_printer; }
......
...@@ -64,7 +64,7 @@ VM_G1IncCollectionPause::VM_G1IncCollectionPause( ...@@ -64,7 +64,7 @@ VM_G1IncCollectionPause::VM_G1IncCollectionPause(
_should_initiate_conc_mark(should_initiate_conc_mark), _should_initiate_conc_mark(should_initiate_conc_mark),
_target_pause_time_ms(target_pause_time_ms), _target_pause_time_ms(target_pause_time_ms),
_should_retry_gc(false), _should_retry_gc(false),
_full_collections_completed_before(0) { _old_marking_cycles_completed_before(0) {
guarantee(target_pause_time_ms > 0.0, guarantee(target_pause_time_ms > 0.0,
err_msg("target_pause_time_ms = %1.6lf should be positive", err_msg("target_pause_time_ms = %1.6lf should be positive",
target_pause_time_ms)); target_pause_time_ms));
...@@ -112,11 +112,11 @@ void VM_G1IncCollectionPause::doit() { ...@@ -112,11 +112,11 @@ void VM_G1IncCollectionPause::doit() {
GCCauseSetter x(g1h, _gc_cause); GCCauseSetter x(g1h, _gc_cause);
if (_should_initiate_conc_mark) { if (_should_initiate_conc_mark) {
// It's safer to read full_collections_completed() here, given // It's safer to read old_marking_cycles_completed() here, given
// that noone else will be updating it concurrently. Since we'll // that noone else will be updating it concurrently. Since we'll
// only need it if we're initiating a marking cycle, no point in // only need it if we're initiating a marking cycle, no point in
// setting it earlier. // setting it earlier.
_full_collections_completed_before = g1h->full_collections_completed(); _old_marking_cycles_completed_before = g1h->old_marking_cycles_completed();
// At this point we are supposed to start a concurrent cycle. We // At this point we are supposed to start a concurrent cycle. We
// will do so if one is not already in progress. // will do so if one is not already in progress.
...@@ -181,17 +181,17 @@ void VM_G1IncCollectionPause::doit_epilogue() { ...@@ -181,17 +181,17 @@ void VM_G1IncCollectionPause::doit_epilogue() {
G1CollectedHeap* g1h = G1CollectedHeap::heap(); G1CollectedHeap* g1h = G1CollectedHeap::heap();
// In the doit() method we saved g1h->full_collections_completed() // In the doit() method we saved g1h->old_marking_cycles_completed()
// in the _full_collections_completed_before field. We have to // in the _old_marking_cycles_completed_before field. We have to
// wait until we observe that g1h->full_collections_completed() // wait until we observe that g1h->old_marking_cycles_completed()
// has increased by at least one. This can happen if a) we started // has increased by at least one. This can happen if a) we started
// a cycle and it completes, b) a cycle already in progress // a cycle and it completes, b) a cycle already in progress
// completes, or c) a Full GC happens. // completes, or c) a Full GC happens.
// If the condition has already been reached, there's no point in // If the condition has already been reached, there's no point in
// actually taking the lock and doing the wait. // actually taking the lock and doing the wait.
if (g1h->full_collections_completed() <= if (g1h->old_marking_cycles_completed() <=
_full_collections_completed_before) { _old_marking_cycles_completed_before) {
// The following is largely copied from CMS // The following is largely copied from CMS
Thread* thr = Thread::current(); Thread* thr = Thread::current();
...@@ -200,8 +200,8 @@ void VM_G1IncCollectionPause::doit_epilogue() { ...@@ -200,8 +200,8 @@ void VM_G1IncCollectionPause::doit_epilogue() {
ThreadToNativeFromVM native(jt); ThreadToNativeFromVM native(jt);
MutexLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag); MutexLockerEx x(FullGCCount_lock, Mutex::_no_safepoint_check_flag);
while (g1h->full_collections_completed() <= while (g1h->old_marking_cycles_completed() <=
_full_collections_completed_before) { _old_marking_cycles_completed_before) {
FullGCCount_lock->wait(Mutex::_no_safepoint_check_flag); FullGCCount_lock->wait(Mutex::_no_safepoint_check_flag);
} }
} }
......
...@@ -80,7 +80,7 @@ private: ...@@ -80,7 +80,7 @@ private:
bool _should_initiate_conc_mark; bool _should_initiate_conc_mark;
bool _should_retry_gc; bool _should_retry_gc;
double _target_pause_time_ms; double _target_pause_time_ms;
unsigned int _full_collections_completed_before; unsigned int _old_marking_cycles_completed_before;
public: public:
VM_G1IncCollectionPause(unsigned int gc_count_before, VM_G1IncCollectionPause(unsigned int gc_count_before,
size_t word_size, size_t word_size,
......
...@@ -230,7 +230,7 @@ void TreeList<Chunk>::return_chunk_at_tail(TreeChunk<Chunk>* chunk) { ...@@ -230,7 +230,7 @@ void TreeList<Chunk>::return_chunk_at_tail(TreeChunk<Chunk>* chunk) {
link_tail(chunk); link_tail(chunk);
assert(!tail() || size() == tail()->size(), "Wrong sized chunk in list"); assert(!tail() || size() == tail()->size(), "Wrong sized chunk in list");
FreeList<Chunk>::increment_count(); increment_count();
debug_only(increment_returned_bytes_by(chunk->size()*sizeof(HeapWord));) debug_only(increment_returned_bytes_by(chunk->size()*sizeof(HeapWord));)
assert(head() == NULL || head()->prev() == NULL, "list invariant"); assert(head() == NULL || head()->prev() == NULL, "list invariant");
assert(tail() == NULL || tail()->next() == NULL, "list invariant"); assert(tail() == NULL || tail()->next() == NULL, "list invariant");
...@@ -258,7 +258,7 @@ void TreeList<Chunk>::return_chunk_at_head(TreeChunk<Chunk>* chunk) { ...@@ -258,7 +258,7 @@ void TreeList<Chunk>::return_chunk_at_head(TreeChunk<Chunk>* chunk) {
} }
head()->link_after(chunk); head()->link_after(chunk);
assert(!head() || size() == head()->size(), "Wrong sized chunk in list"); assert(!head() || size() == head()->size(), "Wrong sized chunk in list");
FreeList<Chunk>::increment_count(); increment_count();
debug_only(increment_returned_bytes_by(chunk->size()*sizeof(HeapWord));) debug_only(increment_returned_bytes_by(chunk->size()*sizeof(HeapWord));)
assert(head() == NULL || head()->prev() == NULL, "list invariant"); assert(head() == NULL || head()->prev() == NULL, "list invariant");
assert(tail() == NULL || tail()->next() == NULL, "list invariant"); assert(tail() == NULL || tail()->next() == NULL, "list invariant");
...@@ -909,6 +909,7 @@ class TreeCensusClosure : public StackObj { ...@@ -909,6 +909,7 @@ class TreeCensusClosure : public StackObj {
template <class Chunk> template <class Chunk>
class AscendTreeCensusClosure : public TreeCensusClosure<Chunk> { class AscendTreeCensusClosure : public TreeCensusClosure<Chunk> {
using TreeCensusClosure<Chunk>::do_list;
public: public:
void do_tree(TreeList<Chunk>* tl) { void do_tree(TreeList<Chunk>* tl) {
if (tl != NULL) { if (tl != NULL) {
...@@ -921,6 +922,7 @@ class AscendTreeCensusClosure : public TreeCensusClosure<Chunk> { ...@@ -921,6 +922,7 @@ class AscendTreeCensusClosure : public TreeCensusClosure<Chunk> {
template <class Chunk> template <class Chunk>
class DescendTreeCensusClosure : public TreeCensusClosure<Chunk> { class DescendTreeCensusClosure : public TreeCensusClosure<Chunk> {
using TreeCensusClosure<Chunk>::do_list;
public: public:
void do_tree(TreeList<Chunk>* tl) { void do_tree(TreeList<Chunk>* tl) {
if (tl != NULL) { if (tl != NULL) {
...@@ -987,6 +989,7 @@ class AscendTreeSearchClosure : public TreeSearchClosure { ...@@ -987,6 +989,7 @@ class AscendTreeSearchClosure : public TreeSearchClosure {
template <class Chunk> template <class Chunk>
class DescendTreeSearchClosure : public TreeSearchClosure<Chunk> { class DescendTreeSearchClosure : public TreeSearchClosure<Chunk> {
using TreeSearchClosure<Chunk>::do_list;
public: public:
bool do_tree(TreeList<Chunk>* tl) { bool do_tree(TreeList<Chunk>* tl) {
if (tl != NULL) { if (tl != NULL) {
......
...@@ -60,13 +60,18 @@ class TreeList: public FreeList<Chunk> { ...@@ -60,13 +60,18 @@ class TreeList: public FreeList<Chunk> {
TreeList<Chunk>* left() const { return _left; } TreeList<Chunk>* left() const { return _left; }
TreeList<Chunk>* right() const { return _right; } TreeList<Chunk>* right() const { return _right; }
// Wrapper on call to base class, to get the template to compile. // Explicitly import these names into our namespace to fix name lookup with templates
Chunk* head() const { return FreeList<Chunk>::head(); } using FreeList<Chunk>::head;
Chunk* tail() const { return FreeList<Chunk>::tail(); } using FreeList<Chunk>::set_head;
void set_head(Chunk* head) { FreeList<Chunk>::set_head(head); }
void set_tail(Chunk* tail) { FreeList<Chunk>::set_tail(tail); } using FreeList<Chunk>::tail;
using FreeList<Chunk>::set_tail;
size_t size() const { return FreeList<Chunk>::size(); } using FreeList<Chunk>::link_tail;
using FreeList<Chunk>::increment_count;
NOT_PRODUCT(using FreeList<Chunk>::increment_returned_bytes_by;)
using FreeList<Chunk>::verify_chunk_in_free_list;
using FreeList<Chunk>::size;
// Accessors for links in tree. // Accessors for links in tree.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册