提交 4754509f 编写于 作者: K kbarrett

8075466: SATB queue pre-filter verify found reclaimed humongous object

Summary: Removed pre-filter verify, and made filtering more careful.
Reviewed-by: brutisso, tschatzl
上级 0ab377be
/* /*
* Copyright (c) 2001, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -35,30 +35,67 @@ ...@@ -35,30 +35,67 @@
PRAGMA_FORMAT_MUTE_WARNINGS_FOR_GCC PRAGMA_FORMAT_MUTE_WARNINGS_FOR_GCC
void ObjPtrQueue::flush() { void ObjPtrQueue::flush() {
// The buffer might contain refs into the CSet. We have to filter it // Filter now to possibly save work later. If filtering empties the
// first before we flush it, otherwise we might end up with an // buffer then flush_impl can deallocate the buffer.
// enqueued buffer with refs into the CSet which breaks our invariants.
filter(); filter();
flush_impl(); flush_impl();
} }
// This method removes entries from an SATB buffer that will not be // Return true if a SATB buffer entry refers to an object that
// useful to the concurrent marking threads. An entry is removed if it // requires marking.
// satisfies one of the following conditions:
// //
// * it points to an object outside the G1 heap (G1's concurrent // The entry must point into the G1 heap. In particular, it must not
// marking only visits objects inside the G1 heap), // be a NULL pointer. NULL pointers are pre-filtered and never
// * it points to an object that has been allocated since marking // inserted into a SATB buffer.
// started (according to SATB those objects do not need to be
// visited during marking), or
// * it points to an object that has already been marked (no need to
// process it again).
// //
// The rest of the entries will be retained and are compacted towards // An entry that is below the NTAMS pointer for the containing heap
// the top of the buffer. Note that, because we do not allow old // region requires marking. Such an entry must point to a valid object.
// regions in the CSet during marking, all objects on the CSet regions //
// are young (eden or survivors) and therefore implicitly live. So any // An entry that is at least the NTAMS pointer for the containing heap
// references into the CSet will be removed during filtering. // region might be any of the following, none of which should be marked.
//
// * A reference to an object allocated since marking started.
// According to SATB, such objects are implicitly kept live and do
// not need to be dealt with via SATB buffer processing.
//
// * A reference to a young generation object. Young objects are
// handled separately and are not marked by concurrent marking.
//
// * A stale reference to a young generation object. If a young
// generation object reference is recorded and not filtered out
// before being moved by a young collection, the reference becomes
// stale.
//
// * A stale reference to an eagerly reclaimed humongous object. If a
// humongous object is recorded and then reclaimed, the reference
// becomes stale.
//
// The stale reference cases are implicitly handled by the NTAMS
// comparison. Because of the possibility of stale references, buffer
// processing must be somewhat circumspect and not assume entries
// in an unfiltered buffer refer to valid objects.
inline bool requires_marking(const void* entry, G1CollectedHeap* heap) {
// Includes rejection of NULL pointers.
assert(heap->is_in_reserved(entry),
err_msg("Non-heap pointer in SATB buffer: " PTR_FORMAT, p2i(entry)));
HeapRegion* region = heap->heap_region_containing_raw(entry);
assert(region != NULL, err_msg("No region for " PTR_FORMAT, p2i(entry)));
if (entry >= region->next_top_at_mark_start()) {
return false;
}
assert(((oop)entry)->is_oop(true /* ignore mark word */),
err_msg("Invalid oop in SATB buffer: " PTR_FORMAT, p2i(entry)));
return true;
}
// This method removes entries from a SATB buffer that will not be
// useful to the concurrent marking threads. Entries are retained if
// they require marking and are not already marked. Retained entries
// are compacted toward the top of the buffer.
void ObjPtrQueue::filter() { void ObjPtrQueue::filter() {
G1CollectedHeap* g1h = G1CollectedHeap::heap(); G1CollectedHeap* g1h = G1CollectedHeap::heap();
...@@ -80,26 +117,25 @@ void ObjPtrQueue::filter() { ...@@ -80,26 +117,25 @@ void ObjPtrQueue::filter() {
assert(i > 0, "we should have at least one more entry to process"); assert(i > 0, "we should have at least one more entry to process");
i -= oopSize; i -= oopSize;
debug_only(entries += 1;) debug_only(entries += 1;)
oop* p = (oop*) &buf[byte_index_to_index((int) i)]; void** p = &buf[byte_index_to_index((int) i)];
oop obj = *p; void* entry = *p;
// NULL the entry so that unused parts of the buffer contain NULLs // NULL the entry so that unused parts of the buffer contain NULLs
// at the end. If we are going to retain it we will copy it to its // at the end. If we are going to retain it we will copy it to its
// final place. If we have retained all entries we have visited so // final place. If we have retained all entries we have visited so
// far, we'll just end up copying it to the same place. // far, we'll just end up copying it to the same place.
*p = NULL; *p = NULL;
bool retain = g1h->is_obj_ill(obj); if (requires_marking(entry, g1h) && !g1h->isMarkedNext((oop)entry)) {
if (retain) {
assert(new_index > 0, "we should not have already filled up the buffer"); assert(new_index > 0, "we should not have already filled up the buffer");
new_index -= oopSize; new_index -= oopSize;
assert(new_index >= i, assert(new_index >= i,
"new_index should never be below i, as we alwaysr compact 'up'"); "new_index should never be below i, as we alwaysr compact 'up'");
oop* new_p = (oop*) &buf[byte_index_to_index((int) new_index)]; void** new_p = &buf[byte_index_to_index((int) new_index)];
assert(new_p >= p, "the destination location should never be below " assert(new_p >= p, "the destination location should never be below "
"the source as we always compact 'up'"); "the source as we always compact 'up'");
assert(*new_p == NULL, assert(*new_p == NULL,
"we should have already cleared the destination location"); "we should have already cleared the destination location");
*new_p = obj; *new_p = entry;
debug_only(retained += 1;) debug_only(retained += 1;)
} }
} }
...@@ -186,17 +222,6 @@ void ObjPtrQueue::print(const char* name, ...@@ -186,17 +222,6 @@ void ObjPtrQueue::print(const char* name,
} }
#endif // PRODUCT #endif // PRODUCT
#ifdef ASSERT
void ObjPtrQueue::verify_oops_in_buffer() {
if (_buf == NULL) return;
for (size_t i = _index; i < _sz; i += oopSize) {
oop obj = (oop)_buf[byte_index_to_index((int)i)];
assert(obj != NULL && obj->is_oop(true /* ignore mark word */),
"Not an oop");
}
}
#endif
#ifdef _MSC_VER // the use of 'this' below gets a warning, make it go away #ifdef _MSC_VER // the use of 'this' below gets a warning, make it go away
#pragma warning( disable:4355 ) // 'this' : used in base member initializer list #pragma warning( disable:4355 ) // 'this' : used in base member initializer list
#endif // _MSC_VER #endif // _MSC_VER
...@@ -216,7 +241,6 @@ void SATBMarkQueueSet::initialize(Monitor* cbl_mon, Mutex* fl_lock, ...@@ -216,7 +241,6 @@ void SATBMarkQueueSet::initialize(Monitor* cbl_mon, Mutex* fl_lock,
} }
void SATBMarkQueueSet::handle_zero_index_for_thread(JavaThread* t) { void SATBMarkQueueSet::handle_zero_index_for_thread(JavaThread* t) {
DEBUG_ONLY(t->satb_mark_queue().verify_oops_in_buffer();)
t->satb_mark_queue().handle_zero_index(); t->satb_mark_queue().handle_zero_index();
} }
......
/* /*
* Copyright (c) 2001, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -72,8 +72,6 @@ public: ...@@ -72,8 +72,6 @@ public:
void print(const char* name); void print(const char* name);
static void print(const char* name, void** buf, size_t index, size_t sz); static void print(const char* name, void** buf, size_t index, size_t sz);
#endif // PRODUCT #endif // PRODUCT
void verify_oops_in_buffer() NOT_DEBUG_RETURN;
}; };
class SATBMarkQueueSet: public PtrQueueSet { class SATBMarkQueueSet: public PtrQueueSet {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册