8029178: Parallel class loading test anonymous-simple gets SIGSEGV in Metaspace::contains

Metaspace::contains cannot look at purged metaspaces while CMS concurrently deallocates them.

Reviewed-by: mgerdin, sspitsyn, jmasa
This commit is contained in:
Coleen Phillimore 2014-01-07 13:26:56 -05:00
parent 3e537df17c
commit 469f290817
10 changed files with 43 additions and 54 deletions

View File

@ -648,12 +648,12 @@ GrowableArray<ClassLoaderData*>* ClassLoaderDataGraph::new_clds() {
return array; return array;
} }
#ifndef PRODUCT // For profiling and hsfind() only. Otherwise, this is unsafe (and slow). This
// for debugging and hsfind(x) // is done lock free to avoid lock inversion problems. It is safe because
bool ClassLoaderDataGraph::contains(address x) { // new ClassLoaderData are added to the end of the CLDG, and only removed at
// I think we need the _metaspace_lock taken here because the class loader // safepoint. The _unloading list can be deallocated concurrently with CMS so
// data graph could be changing while we are walking it (new entries added, // this doesn't look in metaspace for classes that have been unloaded.
// new entries being unloaded, etc). bool ClassLoaderDataGraph::contains(const void* x) {
if (DumpSharedSpaces) { if (DumpSharedSpaces) {
// There are only two metaspaces to worry about. // There are only two metaspaces to worry about.
ClassLoaderData* ncld = ClassLoaderData::the_null_class_loader_data(); ClassLoaderData* ncld = ClassLoaderData::the_null_class_loader_data();
@ -670,16 +670,11 @@ bool ClassLoaderDataGraph::contains(address x) {
} }
} }
// Could also be on an unloading list which is okay, ie. still allocated // Do not check unloading list because deallocation can be concurrent.
// for a little while.
for (ClassLoaderData* ucld = _unloading; ucld != NULL; ucld = ucld->next()) {
if (ucld->metaspace_or_null() != NULL && ucld->metaspace_or_null()->contains(x)) {
return true;
}
}
return false; return false;
} }
#ifndef PRODUCT
bool ClassLoaderDataGraph::contains_loader_data(ClassLoaderData* loader_data) { bool ClassLoaderDataGraph::contains_loader_data(ClassLoaderData* loader_data) {
for (ClassLoaderData* data = _head; data != NULL; data = data->next()) { for (ClassLoaderData* data = _head; data != NULL; data = data->next()) {
if (loader_data == data) { if (loader_data == data) {

View File

@ -90,9 +90,9 @@ class ClassLoaderDataGraph : public AllStatic {
static void dump() { dump_on(tty); } static void dump() { dump_on(tty); }
static void verify(); static void verify();
#ifndef PRODUCT
// expensive test for pointer in metaspace for debugging // expensive test for pointer in metaspace for debugging
static bool contains(address x); static bool contains(const void* x);
#ifndef PRODUCT
static bool contains_loader_data(ClassLoaderData* loader_data); static bool contains_loader_data(ClassLoaderData* loader_data);
#endif #endif

View File

@ -655,8 +655,6 @@ inline Metadata* Dependencies::DepStream::recorded_metadata_at(int i) {
} else { } else {
o = _deps->oop_recorder()->metadata_at(i); o = _deps->oop_recorder()->metadata_at(i);
} }
assert(o == NULL || o->is_metaspace_object(),
err_msg("Should be metadata " PTR_FORMAT, o));
return o; return o;
} }

View File

@ -71,9 +71,8 @@ bool MetaspaceObj::is_shared() const {
return MetaspaceShared::is_in_shared_space(this); return MetaspaceShared::is_in_shared_space(this);
} }
bool MetaspaceObj::is_metaspace_object() const { bool MetaspaceObj::is_metaspace_object() const {
return Metaspace::contains((void*)this); return ClassLoaderDataGraph::contains((void*)this);
} }
void MetaspaceObj::print_address_on(outputStream* st) const { void MetaspaceObj::print_address_on(outputStream* st) const {

View File

@ -264,7 +264,7 @@ class ClassLoaderData;
class MetaspaceObj { class MetaspaceObj {
public: public:
bool is_metaspace_object() const; // more specific test but slower bool is_metaspace_object() const;
bool is_shared() const; bool is_shared() const;
void print_address_on(outputStream* st) const; // nonvirtual address printing void print_address_on(outputStream* st) const; // nonvirtual address printing

View File

@ -143,6 +143,8 @@ class Metachunk : public Metabase<Metachunk> {
void set_is_tagged_free(bool v) { _is_tagged_free = v; } void set_is_tagged_free(bool v) { _is_tagged_free = v; }
#endif #endif
bool contains(const void* ptr) { return bottom() <= ptr && ptr < _top; }
NOT_PRODUCT(void mangle();) NOT_PRODUCT(void mangle();)
void print_on(outputStream* st) const; void print_on(outputStream* st) const;

View File

@ -513,8 +513,6 @@ class VirtualSpaceList : public CHeapObj<mtClass> {
// Unlink empty VirtualSpaceNodes and free it. // Unlink empty VirtualSpaceNodes and free it.
void purge(ChunkManager* chunk_manager); void purge(ChunkManager* chunk_manager);
bool contains(const void *ptr);
void print_on(outputStream* st) const; void print_on(outputStream* st) const;
class VirtualSpaceListIterator : public StackObj { class VirtualSpaceListIterator : public StackObj {
@ -558,7 +556,7 @@ class SpaceManager : public CHeapObj<mtClass> {
private: private:
// protects allocations and contains. // protects allocations
Mutex* const _lock; Mutex* const _lock;
// Type of metadata allocated. // Type of metadata allocated.
@ -595,7 +593,11 @@ class SpaceManager : public CHeapObj<mtClass> {
private: private:
// Accessors // Accessors
Metachunk* chunks_in_use(ChunkIndex index) const { return _chunks_in_use[index]; } Metachunk* chunks_in_use(ChunkIndex index) const { return _chunks_in_use[index]; }
void set_chunks_in_use(ChunkIndex index, Metachunk* v) { _chunks_in_use[index] = v; } void set_chunks_in_use(ChunkIndex index, Metachunk* v) {
// ensure lock-free iteration sees fully initialized node
OrderAccess::storestore();
_chunks_in_use[index] = v;
}
BlockFreelist* block_freelists() const { BlockFreelist* block_freelists() const {
return (BlockFreelist*) &_block_freelists; return (BlockFreelist*) &_block_freelists;
@ -708,6 +710,8 @@ class SpaceManager : public CHeapObj<mtClass> {
void print_on(outputStream* st) const; void print_on(outputStream* st) const;
void locked_print_chunks_in_use_on(outputStream* st) const; void locked_print_chunks_in_use_on(outputStream* st) const;
bool contains(const void *ptr);
void verify(); void verify();
void verify_chunk_size(Metachunk* chunk); void verify_chunk_size(Metachunk* chunk);
NOT_PRODUCT(void mangle_freed_chunks();) NOT_PRODUCT(void mangle_freed_chunks();)
@ -1159,8 +1163,6 @@ bool VirtualSpaceList::create_new_virtual_space(size_t vs_word_size) {
} else { } else {
assert(new_entry->reserved_words() == vs_word_size, assert(new_entry->reserved_words() == vs_word_size,
"Reserved memory size differs from requested memory size"); "Reserved memory size differs from requested memory size");
// ensure lock-free iteration sees fully initialized node
OrderAccess::storestore();
link_vs(new_entry); link_vs(new_entry);
return true; return true;
} }
@ -1287,19 +1289,6 @@ void VirtualSpaceList::print_on(outputStream* st) const {
} }
} }
bool VirtualSpaceList::contains(const void *ptr) {
VirtualSpaceNode* list = virtual_space_list();
VirtualSpaceListIterator iter(list);
while (iter.repeat()) {
VirtualSpaceNode* node = iter.get_next();
if (node->reserved()->contains(ptr)) {
return true;
}
}
return false;
}
// MetaspaceGC methods // MetaspaceGC methods
// VM_CollectForMetadataAllocation is the vm operation used to GC. // VM_CollectForMetadataAllocation is the vm operation used to GC.
@ -2392,6 +2381,21 @@ MetaWord* SpaceManager::allocate_work(size_t word_size) {
return result; return result;
} }
// This function looks at the chunks in the metaspace without locking.
// The chunks are added with store ordering and not deleted except for at
// unloading time.
bool SpaceManager::contains(const void *ptr) {
for (ChunkIndex i = ZeroIndex; i < NumberOfInUseLists; i = next_chunk_index(i))
{
Metachunk* curr = chunks_in_use(i);
while (curr != NULL) {
if (curr->contains(ptr)) return true;
curr = curr->next();
}
}
return false;
}
void SpaceManager::verify() { void SpaceManager::verify() {
// If there are blocks in the dictionary, then // If there are blocks in the dictionary, then
// verfication of chunks does not work since // verfication of chunks does not work since
@ -3463,17 +3467,12 @@ void Metaspace::print_on(outputStream* out) const {
} }
} }
bool Metaspace::contains(const void * ptr) { bool Metaspace::contains(const void* ptr) {
if (MetaspaceShared::is_in_shared_space(ptr)) { if (vsm()->contains(ptr)) return true;
return true; if (using_class_space()) {
return class_vsm()->contains(ptr);
} }
// This is checked while unlocked. As long as the virtualspaces are added return false;
// at the end, the pointer will be in one of them. The virtual spaces
// aren't deleted presently. When they are, some sort of locking might
// be needed. Note, locking this can cause inversion problems with the
// caller in MetaspaceObj::is_metadata() function.
return space_list()->contains(ptr) ||
(using_class_space() && class_space_list()->contains(ptr));
} }
void Metaspace::verify() { void Metaspace::verify() {

View File

@ -225,7 +225,7 @@ class Metaspace : public CHeapObj<mtClass> {
MetaWord* expand_and_allocate(size_t size, MetaWord* expand_and_allocate(size_t size,
MetadataType mdtype); MetadataType mdtype);
static bool contains(const void *ptr); bool contains(const void* ptr);
void dump(outputStream* const out) const; void dump(outputStream* const out) const;
// Free empty virtualspaces // Free empty virtualspaces

View File

@ -376,8 +376,6 @@ void Klass::append_to_sibling_list() {
} }
bool Klass::is_loader_alive(BoolObjectClosure* is_alive) { bool Klass::is_loader_alive(BoolObjectClosure* is_alive) {
assert(ClassLoaderDataGraph::contains((address)this), "is in the metaspace");
#ifdef ASSERT #ifdef ASSERT
// The class is alive iff the class loader is alive. // The class is alive iff the class loader is alive.
oop loader = class_loader(); oop loader = class_loader();

View File

@ -1081,7 +1081,6 @@ void os::print_location(outputStream* st, intptr_t x, bool verbose) {
} }
#ifndef PRODUCT
// Check if in metaspace. // Check if in metaspace.
if (ClassLoaderDataGraph::contains((address)addr)) { if (ClassLoaderDataGraph::contains((address)addr)) {
// Use addr->print() from the debugger instead (not here) // Use addr->print() from the debugger instead (not here)
@ -1089,7 +1088,6 @@ void os::print_location(outputStream* st, intptr_t x, bool verbose) {
" is pointing into metadata", addr); " is pointing into metadata", addr);
return; return;
} }
#endif
// Try an OS specific find // Try an OS specific find
if (os::find(addr, st)) { if (os::find(addr, st)) {