From 4c3e072611346b9f1ff9c8a8ce0f14e48a2b9343 Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Tue, 6 Mar 2018 17:15:16 -0500 Subject: [PATCH] 8198926: Move ClassLoaderData::_dependencies to ClassLoaderData::_handles Move dependency creation and cleaned up logging Reviewed-by: hseigel, jiangli --- .../share/classfile/classFileParser.cpp | 17 +- .../share/classfile/classLoaderData.cpp | 314 ++++++------------ .../share/classfile/classLoaderData.hpp | 58 +--- .../classfile/classLoaderData.inline.hpp | 4 +- src/hotspot/share/classfile/dictionary.cpp | 15 +- src/hotspot/share/classfile/javaClasses.cpp | 6 +- src/hotspot/share/classfile/javaClasses.hpp | 2 +- src/hotspot/share/classfile/modules.cpp | 22 +- .../share/classfile/systemDictionary.cpp | 15 +- .../share/classfile/systemDictionary.hpp | 2 +- .../classfile/systemDictionaryShared.cpp | 2 +- src/hotspot/share/memory/metaspace.cpp | 2 +- src/hotspot/share/memory/universe.cpp | 4 - src/hotspot/share/oops/constantPool.cpp | 2 +- src/hotspot/share/oops/instanceKlass.cpp | 7 - src/hotspot/share/oops/method.cpp | 4 +- src/hotspot/share/oops/method.hpp | 2 +- src/hotspot/share/runtime/java.cpp | 6 +- .../DumpSharedDictionary.java | 2 +- .../DumpSymbolAndStringTable.java | 4 +- .../runtime/logging/ClassLoadUnloadTest.java | 4 +- 21 files changed, 181 insertions(+), 313 deletions(-) diff --git a/src/hotspot/share/classfile/classFileParser.cpp b/src/hotspot/share/classfile/classFileParser.cpp index 51a63239adc..e31c7e48963 100644 --- a/src/hotspot/share/classfile/classFileParser.cpp +++ b/src/hotspot/share/classfile/classFileParser.cpp @@ -4382,7 +4382,7 @@ static void record_defined_class_dependencies(const InstanceKlass* defined_klass // add super class dependency Klass* const super = defined_klass->super(); if (super != NULL) { - defining_loader_data->record_dependency(super, CHECK); + defining_loader_data->record_dependency(super); } // add super interface dependencies @@ -4390,7 +4390,7 @@ static void record_defined_class_dependencies(const InstanceKlass* defined_klass if (local_interfaces != NULL) { const int length = local_interfaces->length(); for (int i = 0; i < length; i++) { - defining_loader_data->record_dependency(local_interfaces->at(i), CHECK); + defining_loader_data->record_dependency(local_interfaces->at(i)); } } } @@ -5363,6 +5363,16 @@ InstanceKlass* ClassFileParser::create_instance_klass(bool changed_by_loadhook, void ClassFileParser::fill_instance_klass(InstanceKlass* ik, bool changed_by_loadhook, TRAPS) { assert(ik != NULL, "invariant"); + // Set name and CLD before adding to CLD + ik->set_class_loader_data(_loader_data); + ik->set_name(_class_name); + + // Add all classes to our internal class loader list here, + // including classes in the bootstrap (NULL) class loader. + const bool publicize = !is_internal(); + + _loader_data->add_class(ik, publicize); + set_klass_to_deallocate(ik); assert(_field_info != NULL, "invariant"); @@ -5377,7 +5387,6 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, bool changed_by_loa ik->set_should_verify_class(_need_verify); // Not yet: supers are done below to support the new subtype-checking fields - ik->set_class_loader_data(_loader_data); ik->set_nonstatic_field_size(_field_info->nonstatic_field_size); ik->set_has_nonstatic_fields(_field_info->has_nonstatic_fields); assert(_fac != NULL, "invariant"); @@ -5408,8 +5417,6 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, bool changed_by_loa // has to be changed accordingly. ik->set_initial_method_idnum(ik->methods()->length()); - ik->set_name(_class_name); - if (is_anonymous()) { // _this_class_index is a CONSTANT_Class entry that refers to this // anonymous class itself. If this class needs to refer to its own methods or diff --git a/src/hotspot/share/classfile/classLoaderData.cpp b/src/hotspot/share/classfile/classLoaderData.cpp index fb7eb43c245..987f79172b7 100644 --- a/src/hotspot/share/classfile/classLoaderData.cpp +++ b/src/hotspot/share/classfile/classLoaderData.cpp @@ -85,7 +85,25 @@ volatile size_t ClassLoaderDataGraph::_num_instance_classes = 0; ClassLoaderData * ClassLoaderData::_the_null_class_loader_data = NULL; -ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous, Dependencies dependencies) : +void ClassLoaderData::init_null_class_loader_data() { + assert(_the_null_class_loader_data == NULL, "cannot initialize twice"); + assert(ClassLoaderDataGraph::_head == NULL, "cannot initialize twice"); + + _the_null_class_loader_data = new ClassLoaderData(Handle(), false); + ClassLoaderDataGraph::_head = _the_null_class_loader_data; + assert(_the_null_class_loader_data->is_the_null_class_loader_data(), "Must be"); + + LogTarget(Debug, class, loader, data) lt; + if (lt.is_enabled()) { + ResourceMark rm; + LogStream ls(lt); + ls.print("create "); + _the_null_class_loader_data->print_value_on(&ls); + ls.cr(); + } +} + +ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous) : _class_loader(h_class_loader()), _is_anonymous(is_anonymous), // An anonymous class loader data doesn't have anything to keep @@ -96,7 +114,7 @@ ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous, Depen _modules(NULL), _packages(NULL), _claimed(0), _modified_oops(true), _accumulated_modified_oops(false), _jmethod_ids(NULL), _handles(), _deallocate_list(NULL), - _next(NULL), _dependencies(dependencies), + _next(NULL), _metaspace_lock(new Mutex(Monitor::leaf+1, "Metaspace allocation lock", true, Monitor::_safepoint_check_never)) { @@ -112,30 +130,18 @@ ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous, Depen // Create unnamed module for all other loaders _unnamed_module = ModuleEntry::create_unnamed_module(this); } - } else { - _unnamed_module = NULL; - } - - if (!is_anonymous) { _dictionary = create_dictionary(); } else { + _packages = NULL; + _unnamed_module = NULL; _dictionary = NULL; } + + NOT_PRODUCT(_dependency_count = 0); // number of class loader dependencies + TRACE_INIT_ID(this); } -void ClassLoaderData::init_dependencies(TRAPS) { - assert(!Universe::is_fully_initialized(), "should only be called when initializing"); - assert(is_the_null_class_loader_data(), "should only call this for the null class loader"); - _dependencies.init(CHECK); -} - -void ClassLoaderData::Dependencies::init(TRAPS) { - // Create empty dependencies array to add to. CMS requires this to be - // an oop so that it can track additions via card marks. We think. - _list_head = oopFactory::new_objectArray(2, CHECK); -} - ClassLoaderData::ChunkedHandleList::~ChunkedHandleList() { Chunk* c = _head; while (c != NULL) { @@ -156,6 +162,16 @@ oop* ClassLoaderData::ChunkedHandleList::add(oop o) { return handle; } +int ClassLoaderData::ChunkedHandleList::count() const { + int count = 0; + Chunk* chunk = _head; + while (chunk != NULL) { + count += chunk->_size; + chunk = chunk->_next; + } + return count; +} + inline void ClassLoaderData::ChunkedHandleList::oops_do_chunk(OopClosure* f, Chunk* c, const juint size) { for (juint i = 0; i < size; i++) { if (c->_data[i] != NULL) { @@ -175,16 +191,15 @@ void ClassLoaderData::ChunkedHandleList::oops_do(OopClosure* f) { } } -#ifdef ASSERT class VerifyContainsOopClosure : public OopClosure { - oop* _target; + oop _target; bool _found; public: - VerifyContainsOopClosure(oop* target) : _target(target), _found(false) {} + VerifyContainsOopClosure(oop target) : _target(target), _found(false) {} void do_oop(oop* p) { - if (p == _target) { + if (p != NULL && *p == _target) { _found = true; } } @@ -199,12 +214,11 @@ class VerifyContainsOopClosure : public OopClosure { } }; -bool ClassLoaderData::ChunkedHandleList::contains(oop* p) { +bool ClassLoaderData::ChunkedHandleList::contains(oop p) { VerifyContainsOopClosure cl(p); oops_do(&cl); return cl.found(); } -#endif // ASSERT bool ClassLoaderData::claim() { if (_claimed == 1) { @@ -244,14 +258,9 @@ void ClassLoaderData::oops_do(OopClosure* f, bool must_claim, bool clear_mod_oop } f->do_oop(&_class_loader); - _dependencies.oops_do(f); _handles.oops_do(f); } -void ClassLoaderData::Dependencies::oops_do(OopClosure* f) { - f->do_oop((oop*)&_list_head); -} - void ClassLoaderData::classes_do(KlassClosure* klass_closure) { // Lock-free access requires load_acquire for (Klass* k = OrderAccess::load_acquire(&_klasses); k != NULL; k = k->next_link()) { @@ -326,7 +335,7 @@ void ClassLoaderData::packages_do(void f(PackageEntry*)) { } } -void ClassLoaderData::record_dependency(const Klass* k, TRAPS) { +void ClassLoaderData::record_dependency(const Klass* k) { assert(k != NULL, "invariant"); ClassLoaderData * const from_cld = this; @@ -361,77 +370,27 @@ void ClassLoaderData::record_dependency(const Klass* k, TRAPS) { } } - // It's a dependency we won't find through GC, add it. This is relatively rare. - // Must handle over GC point. - Handle dependency(THREAD, to); - from_cld->_dependencies.add(dependency, CHECK); - - // Added a potentially young gen oop to the ClassLoaderData - record_modified_oops(); -} - - -void ClassLoaderData::Dependencies::add(Handle dependency, TRAPS) { - // Check first if this dependency is already in the list. - // Save a pointer to the last to add to under the lock. - objArrayOop ok = _list_head; - objArrayOop last = NULL; - while (ok != NULL) { - last = ok; - if (ok->obj_at(0) == dependency()) { - // Don't need to add it - return; + // It's a dependency we won't find through GC, add it. + if (!_handles.contains(to)) { + NOT_PRODUCT(Atomic::inc(&_dependency_count)); + LogTarget(Trace, class, loader, data) lt; + if (lt.is_enabled()) { + ResourceMark rm; + LogStream ls(lt); + ls.print("adding dependency from "); + print_value_on(&ls); + ls.print(" to "); + to_cld->print_value_on(&ls); + ls.cr(); } - ok = (objArrayOop)ok->obj_at(1); - } - - // Must handle over GC points - assert (last != NULL, "dependencies should be initialized"); - objArrayHandle last_handle(THREAD, last); - - // Create a new dependency node with fields for (class_loader or mirror, next) - objArrayOop deps = oopFactory::new_objectArray(2, CHECK); - deps->obj_at_put(0, dependency()); - - // Must handle over GC points - objArrayHandle new_dependency(THREAD, deps); - - // Add the dependency under lock - locked_add(last_handle, new_dependency, THREAD); -} - -void ClassLoaderData::Dependencies::locked_add(objArrayHandle last_handle, - objArrayHandle new_dependency, - Thread* THREAD) { - - // Have to lock and put the new dependency on the end of the dependency - // array so the card mark for CMS sees that this dependency is new. - // Can probably do this lock free with some effort. - ObjectLocker ol(Handle(THREAD, _list_head), THREAD); - - oop loader_or_mirror = new_dependency->obj_at(0); - - // Since the dependencies are only added, add to the end. - objArrayOop end = last_handle(); - objArrayOop last = NULL; - while (end != NULL) { - last = end; - // check again if another thread added it to the end. - if (end->obj_at(0) == loader_or_mirror) { - // Don't need to add it - return; - } - end = (objArrayOop)end->obj_at(1); - } - assert (last != NULL, "dependencies should be initialized"); - // fill in the first element with the oop in new_dependency. - if (last->obj_at(0) == NULL) { - last->obj_at_put(0, new_dependency->obj_at(0)); - } else { - last->obj_at_put(1, new_dependency()); + Handle dependency(Thread::current(), to); + add_handle(dependency); + // Added a potentially young gen oop to the ClassLoaderData + record_modified_oops(); } } + void ClassLoaderDataGraph::clear_claimed_marks() { for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { cld->clear_claimed(); @@ -453,15 +412,15 @@ void ClassLoaderData::add_class(Klass* k, bool publicize /* true */) { } } - if (publicize && k->class_loader_data() != NULL) { - ResourceMark rm; - log_trace(class, loader, data)("Adding k: " PTR_FORMAT " %s to CLD: " - PTR_FORMAT " loader: " PTR_FORMAT " %s", - p2i(k), - k->external_name(), - p2i(k->class_loader_data()), - p2i((void *)k->class_loader()), - loader_name()); + if (publicize) { + LogTarget(Trace, class, loader, data) lt; + if (lt.is_enabled()) { + ResourceMark rm; + LogStream ls(lt); + ls.print("Adding k: " PTR_FORMAT " %s to ", p2i(k), k->external_name()); + print_value_on(&ls); + ls.cr(); + } } } @@ -578,12 +537,8 @@ void ClassLoaderData::unload() { if (lt.is_enabled()) { ResourceMark rm; LogStream ls(lt); - ls.print(": unload loader data " INTPTR_FORMAT, p2i(this)); - ls.print(" for instance " INTPTR_FORMAT " of %s", p2i((void *)class_loader()), - loader_name()); - if (is_anonymous()) { - ls.print(" for anonymous class " INTPTR_FORMAT " ", p2i(_klasses)); - } + ls.print("unload "); + print_value_on(&ls); ls.cr(); } @@ -779,14 +734,8 @@ Metaspace* ClassLoaderData::metaspace_non_null() { assert (class_loader() == NULL, "Must be"); metaspace = new Metaspace(_metaspace_lock, Metaspace::BootMetaspaceType); } else if (is_anonymous()) { - if (class_loader() != NULL) { - log_trace(class, loader, data)("is_anonymous: %s", class_loader()->klass()->internal_name()); - } metaspace = new Metaspace(_metaspace_lock, Metaspace::AnonymousMetaspaceType); } else if (class_loader()->is_a(SystemDictionary::reflect_DelegatingClassLoader_klass())) { - if (class_loader() != NULL) { - log_trace(class, loader, data)("is_reflection: %s", class_loader()->klass()->internal_name()); - } metaspace = new Metaspace(_metaspace_lock, Metaspace::ReflectionMetaspaceType); } else { metaspace = new Metaspace(_metaspace_lock, Metaspace::StandardMetaspaceType); @@ -808,7 +757,7 @@ void ClassLoaderData::remove_handle(OopHandle h) { assert(!is_unloading(), "Do not remove a handle for a CLD that is unloading"); oop* ptr = h.ptr_raw(); if (ptr != NULL) { - assert(_handles.contains(ptr), "Got unexpected handle " PTR_FORMAT, p2i(ptr)); + assert(_handles.contains(*ptr), "Got unexpected handle " PTR_FORMAT, p2i(ptr)); // This root is not walked in safepoints, and hence requires an appropriate // decorator that e.g. maintains the SATB invariant in SATB collectors. RootAccess::oop_store(ptr, oop(NULL)); @@ -902,49 +851,44 @@ void ClassLoaderData::unload_deallocate_list() { } // These anonymous class loaders are to contain classes used for JSR292 -ClassLoaderData* ClassLoaderData::anonymous_class_loader_data(oop loader, TRAPS) { +ClassLoaderData* ClassLoaderData::anonymous_class_loader_data(Handle loader) { // Add a new class loader data to the graph. - Handle lh(THREAD, loader); - return ClassLoaderDataGraph::add(lh, true, THREAD); + return ClassLoaderDataGraph::add(loader, true); } -const char* ClassLoaderData::loader_name() { +const char* ClassLoaderData::loader_name() const { // Handles null class loader return SystemDictionary::loader_name(class_loader()); } -#ifndef PRODUCT -// Define to dump klasses -#undef CLD_DUMP_KLASSES -void ClassLoaderData::dump(outputStream * const out) { - out->print("ClassLoaderData CLD: " PTR_FORMAT ", loader: " PTR_FORMAT ", loader_klass: " PTR_FORMAT " %s {", - p2i(this), p2i((void *)class_loader()), - p2i(class_loader() != NULL ? class_loader()->klass() : NULL), loader_name()); - if (claimed()) out->print(" claimed "); - if (is_unloading()) out->print(" unloading "); - out->cr(); - if (metaspace_or_null() != NULL) { - out->print_cr("metaspace: " INTPTR_FORMAT, p2i(metaspace_or_null())); - metaspace_or_null()->dump(out); +void ClassLoaderData::print_value_on(outputStream* out) const { + if (class_loader() != NULL) { + out->print("loader data: " INTPTR_FORMAT " for instance ", p2i(this)); + class_loader()->print_value_on(out); // includes loader_name() and address of class loader instance } else { - out->print_cr("metaspace: NULL"); + // loader data: 0xsomeaddr of + out->print("loader data: " INTPTR_FORMAT " of %s", p2i(this), loader_name()); } + if (is_anonymous()) { + out->print(" anonymous"); + } +} + +#ifndef PRODUCT +void ClassLoaderData::print_on(outputStream* out) const { + out->print("ClassLoaderData CLD: " PTR_FORMAT ", loader: " PTR_FORMAT ", loader_klass: %s {", + p2i(this), p2i((void *)class_loader()), loader_name()); + if (is_anonymous()) out->print(" anonymous"); + if (claimed()) out->print(" claimed"); + if (is_unloading()) out->print(" unloading"); + out->print(" metaspace: " INTPTR_FORMAT, p2i(metaspace_or_null())); -#ifdef CLD_DUMP_KLASSES - if (Verbose) { - Klass* k = _klasses; - while (k != NULL) { - out->print_cr("klass " PTR_FORMAT ", %s", p2i(k), k->name()->as_C_string()); - assert(k != k->next_link(), "no loops!"); - k = k->next_link(); - } - } -#endif // CLD_DUMP_KLASSES -#undef CLD_DUMP_KLASSES if (_jmethod_ids != NULL) { Method::print_jmethod_ids(this, out); } + out->print(" handles count %d", _handles.count()); + out->print(" dependencies %d", _dependency_count); out->print_cr("}"); } #endif // PRODUCT @@ -988,16 +932,12 @@ bool ClassLoaderDataGraph::_metaspace_oom = false; // Add a new class loader data node to the list. Assign the newly created // ClassLoaderData into the java/lang/ClassLoader object as a hidden field -ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous, TRAPS) { - // We need to allocate all the oops for the ClassLoaderData before allocating the - // actual ClassLoaderData object. - ClassLoaderData::Dependencies dependencies(CHECK_NULL); - +ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous) { NoSafepointVerifier no_safepoints; // we mustn't GC until we've installed the // ClassLoaderData in the graph since the CLD // contains unhandled oops - ClassLoaderData* cld = new ClassLoaderData(loader, is_anonymous, dependencies); + ClassLoaderData* cld = new ClassLoaderData(loader, is_anonymous); if (!is_anonymous) { @@ -1021,9 +961,11 @@ ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous, TRA if (exchanged == next) { LogTarget(Debug, class, loader, data) lt; if (lt.is_enabled()) { - PauseNoSafepointVerifier pnsv(&no_safepoints); // Need safe points for JavaCalls::call_virtual - LogStream ls(lt); - print_creation(&ls, loader, cld, CHECK_NULL); + ResourceMark rm; + LogStream ls(lt); + ls.print("create "); + cld->print_value_on(&ls); + ls.cr(); } return cld; } @@ -1031,36 +973,6 @@ ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous, TRA } while (true); } -void ClassLoaderDataGraph::print_creation(outputStream* out, Handle loader, ClassLoaderData* cld, TRAPS) { - Handle string; - if (loader.not_null()) { - // Include the result of loader.toString() in the output. This allows - // the user of the log to identify the class loader instance. - JavaValue result(T_OBJECT); - Klass* spec_klass = SystemDictionary::ClassLoader_klass(); - JavaCalls::call_virtual(&result, - loader, - spec_klass, - vmSymbols::toString_name(), - vmSymbols::void_string_signature(), - CHECK); - assert(result.get_type() == T_OBJECT, "just checking"); - string = Handle(THREAD, (oop)result.get_jobject()); - } - - ResourceMark rm; - out->print("create class loader data " INTPTR_FORMAT, p2i(cld)); - out->print(" for instance " INTPTR_FORMAT " of %s", p2i((void *)cld->class_loader()), - cld->loader_name()); - - if (string.not_null()) { - out->print(": "); - java_lang_String::print(string(), out); - } - out->cr(); -} - - void ClassLoaderDataGraph::oops_do(OopClosure* f, bool must_claim) { for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { cld->oops_do(f, must_claim); @@ -1477,7 +1389,8 @@ ClassLoaderDataGraphMetaspaceIterator::~ClassLoaderDataGraphMetaspaceIterator() #ifndef PRODUCT // callable from debugger extern "C" int print_loader_data_graph() { - ClassLoaderDataGraph::dump_on(tty); + ResourceMark rm; + ClassLoaderDataGraph::print_on(tty); return 0; } @@ -1487,32 +1400,13 @@ void ClassLoaderDataGraph::verify() { } } -void ClassLoaderDataGraph::dump_on(outputStream * const out) { +void ClassLoaderDataGraph::print_on(outputStream * const out) { for (ClassLoaderData* data = _head; data != NULL; data = data->next()) { - data->dump(out); + data->print_on(out); } - MetaspaceAux::dump(out); } #endif // PRODUCT -void ClassLoaderData::print_value_on(outputStream* out) const { - if (class_loader() == NULL) { - out->print("NULL class loader"); - } else { - out->print("class loader " INTPTR_FORMAT " ", p2i(this)); - class_loader()->print_value_on(out); - } -} - -void ClassLoaderData::print_on(outputStream* out) const { - if (class_loader() == NULL) { - out->print("NULL class loader"); - } else { - out->print("class loader " INTPTR_FORMAT " ", p2i(this)); - class_loader()->print_on(out); - } -} - #if INCLUDE_TRACE Ticks ClassLoaderDataGraph::_class_unload_time; diff --git a/src/hotspot/share/classfile/classLoaderData.hpp b/src/hotspot/share/classfile/classLoaderData.hpp index 8c1fed3bb07..ea1933a1992 100644 --- a/src/hotspot/share/classfile/classLoaderData.hpp +++ b/src/hotspot/share/classfile/classLoaderData.hpp @@ -83,10 +83,10 @@ class ClassLoaderDataGraph : public AllStatic { static volatile size_t _num_instance_classes; static volatile size_t _num_array_classes; - static ClassLoaderData* add(Handle class_loader, bool anonymous, TRAPS); + static ClassLoaderData* add(Handle class_loader, bool anonymous); static void post_class_unload_events(); public: - static ClassLoaderData* find_or_create(Handle class_loader, TRAPS); + static ClassLoaderData* find_or_create(Handle class_loader); static void purge(); static void clear_claimed_marks(); // oops do @@ -151,10 +151,9 @@ class ClassLoaderDataGraph : public AllStatic { static bool has_metaspace_oom() { return _metaspace_oom; } static void set_metaspace_oom(bool value) { _metaspace_oom = value; } - static void dump_on(outputStream * const out) PRODUCT_RETURN; - static void dump() { dump_on(tty); } + static void print_on(outputStream * const out) PRODUCT_RETURN; + static void print() { print_on(tty); } static void verify(); - static void print_creation(outputStream* out, Handle loader, ClassLoaderData* cld, TRAPS); static bool unload_list_contains(const void* x); @@ -181,22 +180,8 @@ class ClassLoaderDataGraph : public AllStatic { class ClassLoaderData : public CHeapObj { friend class VMStructs; - private: - class Dependencies VALUE_OBJ_CLASS_SPEC { - objArrayOop _list_head; - void locked_add(objArrayHandle last, - objArrayHandle new_dependency, - Thread* THREAD); - public: - Dependencies() : _list_head(NULL) {} - Dependencies(TRAPS) : _list_head(NULL) { - init(CHECK); - } - void add(Handle dependency, TRAPS); - void init(TRAPS); - void oops_do(OopClosure* f); - }; + private: class ChunkedHandleList VALUE_OBJ_CLASS_SPEC { struct Chunk : public CHeapObj { static const size_t CAPACITY = 32; @@ -219,10 +204,10 @@ class ClassLoaderData : public CHeapObj { // Only one thread at a time can add, guarded by ClassLoaderData::metaspace_lock(). // However, multiple threads can execute oops_do concurrently with add. oop* add(oop o); -#ifdef ASSERT - bool contains(oop* p); -#endif + bool contains(oop p); void oops_do(OopClosure* f); + + int count() const; }; friend class ClassLoaderDataGraph; @@ -237,8 +222,6 @@ class ClassLoaderData : public CHeapObj { oop _class_loader; // oop used to uniquely identify a class loader // class loader or a canonical class path - Dependencies _dependencies; // holds dependencies from this class loader - // data to others. Metaspace * volatile _metaspace; // Meta-space where meta-data defined by the // classes in the class loader are allocated. @@ -261,6 +244,8 @@ class ClassLoaderData : public CHeapObj { ChunkedHandleList _handles; // Handles to constant pool arrays, Modules, etc, which // have the same life cycle of the corresponding ClassLoader. + NOT_PRODUCT(volatile int _dependency_count;) // number of class loader dependencies + Klass* volatile _klasses; // The classes defined by the class loader. PackageEntryTable* volatile _packages; // The packages defined by the class loader. ModuleEntryTable* volatile _modules; // The modules defined by the class loader. @@ -289,7 +274,7 @@ class ClassLoaderData : public CHeapObj { void set_next(ClassLoaderData* next) { _next = next; } ClassLoaderData* next() const { return _next; } - ClassLoaderData(Handle h_class_loader, bool is_anonymous, Dependencies dependencies); + ClassLoaderData(Handle h_class_loader, bool is_anonymous); ~ClassLoaderData(); // The CLD are not placed in the Heap, so the Card Table or @@ -341,15 +326,7 @@ class ClassLoaderData : public CHeapObj { bool is_anonymous() const { return _is_anonymous; } - static void init_null_class_loader_data() { - assert(_the_null_class_loader_data == NULL, "cannot initialize twice"); - assert(ClassLoaderDataGraph::_head == NULL, "cannot initialize twice"); - - // We explicitly initialize the Dependencies object at a later phase in the initialization - _the_null_class_loader_data = new ClassLoaderData(Handle(), false, Dependencies()); - ClassLoaderDataGraph::_head = _the_null_class_loader_data; - assert(_the_null_class_loader_data->is_the_null_class_loader_data(), "Must be"); - } + static void init_null_class_loader_data(); bool is_the_null_class_loader_data() const { return this == _the_null_class_loader_data; @@ -397,12 +374,11 @@ class ClassLoaderData : public CHeapObj { void set_jmethod_ids(JNIMethodBlock* new_block) { _jmethod_ids = new_block; } void print() { print_on(tty); } - void print_on(outputStream* out) const; + void print_on(outputStream* out) const PRODUCT_RETURN; void print_value() { print_value_on(tty); } void print_value_on(outputStream* out) const; - void dump(outputStream * const out) PRODUCT_RETURN; void verify(); - const char* loader_name(); + const char* loader_name() const; OopHandle add_handle(Handle h); void remove_handle(OopHandle h); @@ -410,8 +386,7 @@ class ClassLoaderData : public CHeapObj { void add_class(Klass* k, bool publicize = true); void remove_class(Klass* k); bool contains_klass(Klass* k); - void record_dependency(const Klass* to, TRAPS); - void init_dependencies(TRAPS); + void record_dependency(const Klass* to); PackageEntryTable* packages() { return _packages; } ModuleEntry* unnamed_module() { return _unnamed_module; } ModuleEntryTable* modules(); @@ -424,8 +399,7 @@ class ClassLoaderData : public CHeapObj { static ClassLoaderData* class_loader_data(oop loader); static ClassLoaderData* class_loader_data_or_null(oop loader); - static ClassLoaderData* anonymous_class_loader_data(oop loader, TRAPS); - static void print_loader(ClassLoaderData *loader_data, outputStream *out); + static ClassLoaderData* anonymous_class_loader_data(Handle loader); TRACE_DEFINE_TRACE_ID_METHODS; }; diff --git a/src/hotspot/share/classfile/classLoaderData.inline.hpp b/src/hotspot/share/classfile/classLoaderData.inline.hpp index 588dcc273c5..d7aa7d15d12 100644 --- a/src/hotspot/share/classfile/classLoaderData.inline.hpp +++ b/src/hotspot/share/classfile/classLoaderData.inline.hpp @@ -43,7 +43,7 @@ inline ClassLoaderData* ClassLoaderData::class_loader_data(oop loader) { } -inline ClassLoaderData *ClassLoaderDataGraph::find_or_create(Handle loader, TRAPS) { +inline ClassLoaderData *ClassLoaderDataGraph::find_or_create(Handle loader) { guarantee(loader() != NULL && oopDesc::is_oop(loader()), "Loader must be oop"); // Gets the class loader data out of the java/lang/ClassLoader object, if non-null // it's already in the loader_data, so no need to add @@ -51,7 +51,7 @@ inline ClassLoaderData *ClassLoaderDataGraph::find_or_create(Handle loader, TRAP if (loader_data) { return loader_data; } - return ClassLoaderDataGraph::add(loader, false, THREAD); + return ClassLoaderDataGraph::add(loader, false); } size_t ClassLoaderDataGraph::num_instance_classes() { diff --git a/src/hotspot/share/classfile/dictionary.cpp b/src/hotspot/share/classfile/dictionary.cpp index 4af2060b0bd..daa6d8b0e9a 100644 --- a/src/hotspot/share/classfile/dictionary.cpp +++ b/src/hotspot/share/classfile/dictionary.cpp @@ -605,13 +605,16 @@ void Dictionary::print_on(outputStream* st) const { Klass* e = probe->instance_klass(); bool is_defining_class = (loader_data() == e->class_loader_data()); - st->print("%4d: %s%s, loader ", index, is_defining_class ? " " : "^", e->external_name()); - ClassLoaderData* loader_data = e->class_loader_data(); - if (loader_data == NULL) { + st->print("%4d: %s%s", index, is_defining_class ? " " : "^", e->external_name()); + ClassLoaderData* cld = e->class_loader_data(); + if (cld == NULL) { // Shared class not restored yet in shared dictionary - st->print(""); - } else { - loader_data->print_value_on(st); + st->print(", loader data "); + } else if (!loader_data()->is_the_null_class_loader_data()) { + // Class loader output for the dictionary for the null class loader data is + // redundant and obvious. + st->print(", "); + cld->print_value_on(st); } st->cr(); } diff --git a/src/hotspot/share/classfile/javaClasses.cpp b/src/hotspot/share/classfile/javaClasses.cpp index a24d5b8d75c..a4b8ec4c5ad 100644 --- a/src/hotspot/share/classfile/javaClasses.cpp +++ b/src/hotspot/share/classfile/javaClasses.cpp @@ -3300,7 +3300,7 @@ void java_lang_Module::set_name(oop module, oop value) { module->obj_field_put(name_offset, value); } -ModuleEntry* java_lang_Module::module_entry(oop module, TRAPS) { +ModuleEntry* java_lang_Module::module_entry(oop module) { assert(_module_entry_offset != -1, "Uninitialized module_entry_offset"); assert(module != NULL, "module can't be null"); assert(oopDesc::is_oop(module), "module must be oop"); @@ -3310,8 +3310,8 @@ ModuleEntry* java_lang_Module::module_entry(oop module, TRAPS) { // If the inject field containing the ModuleEntry* is null then return the // class loader's unnamed module. oop loader = java_lang_Module::loader(module); - Handle h_loader = Handle(THREAD, loader); - ClassLoaderData* loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + Handle h_loader = Handle(Thread::current(), loader); + ClassLoaderData* loader_cld = SystemDictionary::register_loader(h_loader); return loader_cld->unnamed_module(); } return module_entry; diff --git a/src/hotspot/share/classfile/javaClasses.hpp b/src/hotspot/share/classfile/javaClasses.hpp index c884a4a3961..77634ae2e7f 100644 --- a/src/hotspot/share/classfile/javaClasses.hpp +++ b/src/hotspot/share/classfile/javaClasses.hpp @@ -798,7 +798,7 @@ class java_lang_Module { static oop name(oop module); static void set_name(oop module, oop value); - static ModuleEntry* module_entry(oop module, TRAPS); + static ModuleEntry* module_entry(oop module); static void set_module_entry(oop module, ModuleEntry* module_entry); friend class JavaClasses; diff --git a/src/hotspot/share/classfile/modules.cpp b/src/hotspot/share/classfile/modules.cpp index 30250770215..42357c0689d 100644 --- a/src/hotspot/share/classfile/modules.cpp +++ b/src/hotspot/share/classfile/modules.cpp @@ -85,27 +85,27 @@ static const char* get_module_version(jstring version) { return java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(version)); } -static ModuleEntryTable* get_module_entry_table(Handle h_loader, TRAPS) { +static ModuleEntryTable* get_module_entry_table(Handle h_loader) { // This code can be called during start-up, before the classLoader's classLoader data got // created. So, call register_loader() to make sure the classLoader data gets created. - ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader); return loader_cld->modules(); } -static PackageEntryTable* get_package_entry_table(Handle h_loader, TRAPS) { +static PackageEntryTable* get_package_entry_table(Handle h_loader) { // This code can be called during start-up, before the classLoader's classLoader data got // created. So, call register_loader() to make sure the classLoader data gets created. - ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader); return loader_cld->packages(); } static ModuleEntry* get_module_entry(jobject module, TRAPS) { - Handle module_h(THREAD, JNIHandles::resolve(module)); - if (!java_lang_Module::is_instance(module_h())) { + oop m = JNIHandles::resolve(module); + if (!java_lang_Module::is_instance(m)) { THROW_MSG_NULL(vmSymbols::java_lang_IllegalArgumentException(), "module is not an instance of type java.lang.Module"); } - return java_lang_Module::module_entry(module_h(), CHECK_NULL); + return java_lang_Module::module_entry(m); } static PackageEntry* get_package_entry(ModuleEntry* module_entry, const char* package_name, TRAPS) { @@ -124,7 +124,7 @@ static PackageEntry* get_package_entry_by_name(Symbol* package, ResourceMark rm(THREAD); if (Modules::verify_package_name(package->as_C_string())) { PackageEntryTable* const package_entry_table = - get_package_entry_table(h_loader, CHECK_NULL); + get_package_entry_table(h_loader); assert(package_entry_table != NULL, "Unexpected null package entry table"); return package_entry_table->lookup_only(package); } @@ -186,7 +186,7 @@ static void define_javabase_module(jobject module, jstring version, Handle h_loader(THREAD, loader); // Ensure the boot loader's PackageEntryTable has been created - PackageEntryTable* package_table = get_package_entry_table(h_loader, CHECK); + PackageEntryTable* package_table = get_package_entry_table(h_loader); assert(pkg_list->length() == 0 || package_table != NULL, "Bad package_table"); // Ensure java.base's ModuleEntry has been created @@ -346,7 +346,7 @@ void Modules::define_module(jobject module, jboolean is_open, jstring version, pkg_list->append(pkg_symbol); } - ModuleEntryTable* module_table = get_module_entry_table(h_loader, CHECK); + ModuleEntryTable* module_table = get_module_entry_table(h_loader); assert(module_table != NULL, "module entry table shouldn't be null"); // Create symbol* entry for module name. @@ -382,7 +382,7 @@ void Modules::define_module(jobject module, jboolean is_open, jstring version, MutexLocker ml(Module_lock, THREAD); if (num_packages > 0) { - package_table = get_package_entry_table(h_loader, CHECK); + package_table = get_package_entry_table(h_loader); assert(package_table != NULL, "Missing package_table"); // Check that none of the packages exist in the class loader's package table. diff --git a/src/hotspot/share/classfile/systemDictionary.cpp b/src/hotspot/share/classfile/systemDictionary.cpp index 6fe7dd693b0..f004e25ac27 100644 --- a/src/hotspot/share/classfile/systemDictionary.cpp +++ b/src/hotspot/share/classfile/systemDictionary.cpp @@ -150,9 +150,9 @@ void SystemDictionary::compute_java_loaders(TRAPS) { CDS_ONLY(SystemDictionaryShared::initialize(CHECK);) } -ClassLoaderData* SystemDictionary::register_loader(Handle class_loader, TRAPS) { +ClassLoaderData* SystemDictionary::register_loader(Handle class_loader) { if (class_loader() == NULL) return ClassLoaderData::the_null_class_loader_data(); - return ClassLoaderDataGraph::find_or_create(class_loader, THREAD); + return ClassLoaderDataGraph::find_or_create(class_loader); } // ---------------------------------------------------------------------------- @@ -664,7 +664,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, // Fix for 4474172; see evaluation for more details class_loader = Handle(THREAD, java_lang_ClassLoader::non_reflection_class_loader(class_loader())); - ClassLoaderData *loader_data = register_loader(class_loader, CHECK_NULL); + ClassLoaderData* loader_data = register_loader(class_loader); Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(name); @@ -989,7 +989,7 @@ InstanceKlass* SystemDictionary::parse_stream(Symbol* class_name, // Create a new CLD for anonymous class, that uses the same class loader // as the host_klass guarantee(host_klass->class_loader() == class_loader(), "should be the same"); - loader_data = ClassLoaderData::anonymous_class_loader_data(class_loader(), CHECK_NULL); + loader_data = ClassLoaderData::anonymous_class_loader_data(class_loader); } else { loader_data = ClassLoaderData::class_loader_data(class_loader()); } @@ -1067,7 +1067,7 @@ InstanceKlass* SystemDictionary::resolve_from_stream(Symbol* class_name, DoObjectLock = false; } - ClassLoaderData* loader_data = register_loader(class_loader, CHECK_NULL); + ClassLoaderData* loader_data = register_loader(class_loader); // Make sure we are synchronized on the class loader before we proceed Handle lockObject = compute_loader_lock_object(class_loader, THREAD); @@ -2505,11 +2505,10 @@ static methodHandle unpack_method_and_appendix(Handle mname, } (*appendix_result) = Handle(THREAD, appendix); // the target is stored in the cpCache and if a reference to this - // MethodName is dropped we need a way to make sure the + // MemberName is dropped we need a way to make sure the // class_loader containing this method is kept alive. - // FIXME: the appendix might also preserve this dependency. ClassLoaderData* this_key = accessing_klass->class_loader_data(); - this_key->record_dependency(m->method_holder(), CHECK_NULL); // Can throw OOM + this_key->record_dependency(m->method_holder()); return methodHandle(THREAD, m); } } diff --git a/src/hotspot/share/classfile/systemDictionary.hpp b/src/hotspot/share/classfile/systemDictionary.hpp index a86bb27c3de..c83858e92d0 100644 --- a/src/hotspot/share/classfile/systemDictionary.hpp +++ b/src/hotspot/share/classfile/systemDictionary.hpp @@ -493,7 +493,7 @@ public: static void compute_java_loaders(TRAPS); // Register a new class loader - static ClassLoaderData* register_loader(Handle class_loader, TRAPS); + static ClassLoaderData* register_loader(Handle class_loader); protected: // Mirrors for primitive classes (created eagerly) static oop check_mirror(oop m) { diff --git a/src/hotspot/share/classfile/systemDictionaryShared.cpp b/src/hotspot/share/classfile/systemDictionaryShared.cpp index 939e45fcb98..eeaa140f556 100644 --- a/src/hotspot/share/classfile/systemDictionaryShared.cpp +++ b/src/hotspot/share/classfile/systemDictionaryShared.cpp @@ -497,7 +497,7 @@ InstanceKlass* SystemDictionaryShared::find_or_load_shared_class( // Fix for 4474172; see evaluation for more details class_loader = Handle( THREAD, java_lang_ClassLoader::non_reflection_class_loader(class_loader())); - ClassLoaderData *loader_data = register_loader(class_loader, CHECK_NULL); + ClassLoaderData *loader_data = register_loader(class_loader); Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(name); diff --git a/src/hotspot/share/memory/metaspace.cpp b/src/hotspot/share/memory/metaspace.cpp index 3f12399084f..5203f6b1ae4 100644 --- a/src/hotspot/share/memory/metaspace.cpp +++ b/src/hotspot/share/memory/metaspace.cpp @@ -3994,7 +3994,7 @@ void Metaspace::report_metadata_oome(ClassLoaderData* loader_data, size_t word_s if (log.is_debug()) { if (loader_data->metaspace_or_null() != NULL) { LogStream ls(log.debug()); - loader_data->dump(&ls); + loader_data->print_value_on(&ls); } } LogStream ls(log.info()); diff --git a/src/hotspot/share/memory/universe.cpp b/src/hotspot/share/memory/universe.cpp index 15cb34ac0c7..e1ac18773e1 100644 --- a/src/hotspot/share/memory/universe.cpp +++ b/src/hotspot/share/memory/universe.cpp @@ -460,10 +460,6 @@ void Universe::genesis(TRAPS) { assert(i == _fullgc_alot_dummy_array->length(), "just checking"); } #endif - - // Initialize dependency array for null class loader - ClassLoaderData::the_null_class_loader_data()->init_dependencies(CHECK); - } void Universe::initialize_basic_type_mirrors(TRAPS) { diff --git a/src/hotspot/share/oops/constantPool.cpp b/src/hotspot/share/oops/constantPool.cpp index 50a99413f40..130080f9538 100644 --- a/src/hotspot/share/oops/constantPool.cpp +++ b/src/hotspot/share/oops/constantPool.cpp @@ -499,7 +499,7 @@ Klass* ConstantPool::klass_at_impl(const constantPoolHandle& this_cp, int which, // Make this class loader depend upon the class loader owning the class reference ClassLoaderData* this_key = this_cp->pool_holder()->class_loader_data(); - this_key->record_dependency(k, CHECK_NULL); // Can throw OOM + this_key->record_dependency(k); // logging for class+resolve. if (log_is_enabled(Debug, class, resolve)){ diff --git a/src/hotspot/share/oops/instanceKlass.cpp b/src/hotspot/share/oops/instanceKlass.cpp index 50d744d69c1..f58108723b6 100644 --- a/src/hotspot/share/oops/instanceKlass.cpp +++ b/src/hotspot/share/oops/instanceKlass.cpp @@ -184,13 +184,6 @@ InstanceKlass* InstanceKlass::allocate_instance_klass(const ClassFileParser& par return NULL; } - assert(ik != NULL, "invariant"); - - const bool publicize = !parser.is_internal(); - - // Add all classes to our internal class loader list here, - // including classes in the bootstrap (NULL) class loader. - loader_data->add_class(ik, publicize); return ik; } diff --git a/src/hotspot/share/oops/method.cpp b/src/hotspot/share/oops/method.cpp index 1ded76238de..9d6b7cf4cc8 100644 --- a/src/hotspot/share/oops/method.cpp +++ b/src/hotspot/share/oops/method.cpp @@ -2190,8 +2190,8 @@ bool Method::is_valid_method() const { } #ifndef PRODUCT -void Method::print_jmethod_ids(ClassLoaderData* loader_data, outputStream* out) { - out->print_cr("jni_method_id count = %d", loader_data->jmethod_ids()->count_methods()); +void Method::print_jmethod_ids(const ClassLoaderData* loader_data, outputStream* out) { + out->print(" jni_method_id count = %d", loader_data->jmethod_ids()->count_methods()); } #endif // PRODUCT diff --git a/src/hotspot/share/oops/method.hpp b/src/hotspot/share/oops/method.hpp index e7d23f10795..6f341640d73 100644 --- a/src/hotspot/share/oops/method.hpp +++ b/src/hotspot/share/oops/method.hpp @@ -809,7 +809,7 @@ class Method : public Metadata { // Clear methods static void clear_jmethod_ids(ClassLoaderData* loader_data); - static void print_jmethod_ids(ClassLoaderData* loader_data, outputStream* out) PRODUCT_RETURN; + static void print_jmethod_ids(const ClassLoaderData* loader_data, outputStream* out) PRODUCT_RETURN; // Get this method's jmethodID -- allocate if it doesn't exist jmethodID jmethod_id() { return method_holder()->get_jmethod_id(this); } diff --git a/src/hotspot/share/runtime/java.cpp b/src/hotspot/share/runtime/java.cpp index 4f76467a0f0..2bbef67d1ab 100644 --- a/src/hotspot/share/runtime/java.cpp +++ b/src/hotspot/share/runtime/java.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -341,7 +341,9 @@ void print_statistics() { } if (PrintSystemDictionaryAtExit) { + ResourceMark rm; SystemDictionary::print(); + ClassLoaderDataGraph::print(); } if (LogTouchedMethods && PrintTouchedMethodsAtExit) { @@ -483,7 +485,7 @@ void before_exit(JavaThread* thread) { Universe::print_on(&ls_info); if (log.is_trace()) { LogStream ls_trace(log.trace()); - ClassLoaderDataGraph::dump_on(&ls_trace); + ClassLoaderDataGraph::print_on(&ls_trace); } } diff --git a/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSharedDictionary.java b/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSharedDictionary.java index ba0af003bad..4fb681f86d2 100644 --- a/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSharedDictionary.java +++ b/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSharedDictionary.java @@ -81,7 +81,7 @@ public class DumpSharedDictionary { output = CDSTestUtils.executeAndLog(pb, "jcmd-systemdictionary-verbose"); try { output.shouldContain("Shared Dictionary"); - output.shouldContain("Dictionary for class loader 0x"); + output.shouldContain("Dictionary for loader data: 0x"); output.shouldContain("^java.lang.String"); } catch (RuntimeException e) { output.shouldContain("Unknown diagnostic command"); diff --git a/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSymbolAndStringTable.java b/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSymbolAndStringTable.java index bef181f8271..bf092eae08e 100644 --- a/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSymbolAndStringTable.java +++ b/test/hotspot/jtreg/runtime/SharedArchiveFile/DumpSymbolAndStringTable.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2018, 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 @@ -71,7 +71,7 @@ public class DumpSymbolAndStringTable { pb.command(new String[] {JDKToolFinder.getJDKTool("jcmd"), pid, "VM.systemdictionary", "-verbose"}); output = CDSTestUtils.executeAndLog(pb, "jcmd-systemdictionary"); try { - output.shouldContain("Dictionary for class loader 0x"); + output.shouldContain("Dictionary for loader data: 0x"); output.shouldContain("^java.lang.String"); } catch (RuntimeException e) { output.shouldContain("Unknown diagnostic command"); diff --git a/test/hotspot/jtreg/runtime/logging/ClassLoadUnloadTest.java b/test/hotspot/jtreg/runtime/logging/ClassLoadUnloadTest.java index 07c0c9c8be2..66909560cbc 100644 --- a/test/hotspot/jtreg/runtime/logging/ClassLoadUnloadTest.java +++ b/test/hotspot/jtreg/runtime/logging/ClassLoadUnloadTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2018, 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 @@ -124,7 +124,7 @@ public class ClassLoadUnloadTest { // -Xlog:class+loader+data=trace pb = exec("-Xlog:class+loader+data=trace"); - checkFor("[class,loader,data]", "create class loader data"); + checkFor("[class,loader,data]", "create loader data"); } }