From ddfd4c33c6a715bbf35ebf66bea3d4bdbf0853ec Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Wed, 18 Apr 2018 18:43:04 -0400 Subject: [PATCH] 8201556: Disallow reading oops in ClassLoaderData if unloading Move class_loader oop to an OopHandle, and assert that holder is alive when getting class_loader. Reviewed-by: kbarrett, adinn --- .../share/classfile/classLoaderData.cpp | 62 ++++++++++++++++--- .../share/classfile/classLoaderData.hpp | 27 ++++---- .../classfile/classLoaderData.inline.hpp | 12 ++++ .../share/classfile/classLoaderStats.cpp | 3 +- src/hotspot/share/classfile/moduleEntry.cpp | 2 +- src/hotspot/share/classfile/placeholders.cpp | 3 +- .../classfile/systemDictionaryShared.cpp | 8 +++ .../classfile/systemDictionaryShared.hpp | 10 +-- .../share/gc/g1/g1MonitoringSupport.cpp | 1 + .../gc/parallel/parallelScavengeHeap.cpp | 1 + .../share/gc/shared/genCollectedHeap.cpp | 1 + src/hotspot/share/memory/heapInspection.cpp | 4 +- src/hotspot/share/memory/universe.cpp | 1 + src/hotspot/share/oops/instanceKlass.cpp | 1 + src/hotspot/share/oops/klass.cpp | 1 + src/hotspot/share/oops/oopHandle.hpp | 2 +- src/hotspot/share/oops/weakHandle.cpp | 4 +- .../share/prims/jvmtiGetLoadedClasses.cpp | 1 + src/hotspot/share/runtime/vmStructs.cpp | 2 +- .../share/services/classLoadingService.cpp | 4 +- src/hotspot/share/trace/traceStream.cpp | 4 +- .../hotspot/classfile/ClassLoaderData.java | 14 +++-- 22 files changed, 121 insertions(+), 47 deletions(-) diff --git a/src/hotspot/share/classfile/classLoaderData.cpp b/src/hotspot/share/classfile/classLoaderData.cpp index 85d3716861b..0120c9f10e5 100644 --- a/src/hotspot/share/classfile/classLoaderData.cpp +++ b/src/hotspot/share/classfile/classLoaderData.cpp @@ -54,6 +54,7 @@ #include "classfile/metadataOnStackMark.hpp" #include "classfile/moduleEntry.hpp" #include "classfile/packageEntry.hpp" +#include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "logging/log.hpp" #include "logging/logStream.hpp" @@ -102,8 +103,25 @@ void ClassLoaderData::init_null_class_loader_data() { } } +// JFR and logging support so that the name and klass are available after the +// class_loader oop is no longer alive, during unloading. +void ClassLoaderData::initialize_name_and_klass(Handle class_loader) { + _class_loader_klass = class_loader->klass(); + oop class_loader_name = java_lang_ClassLoader::name(class_loader()); + if (class_loader_name != NULL) { + Thread* THREAD = Thread::current(); + ResourceMark rm(THREAD); + const char* class_loader_instance_name = + java_lang_String::as_utf8_string(class_loader_name); + + if (class_loader_instance_name != NULL && class_loader_instance_name[0] != '\0') { + // Can't throw InternalError and SymbolTable doesn't throw OOM anymore. + _class_loader_name = SymbolTable::new_symbol(class_loader_instance_name, CATCH); + } + } +} + 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 // it from being unloaded during parsing of the anonymous class. @@ -114,9 +132,14 @@ ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous) : _claimed(0), _modified_oops(true), _accumulated_modified_oops(false), _jmethod_ids(NULL), _handles(), _deallocate_list(NULL), _next(NULL), + _class_loader_klass(NULL), _class_loader_name(NULL), _metaspace_lock(new Mutex(Monitor::leaf+1, "Metaspace allocation lock", true, Monitor::_safepoint_check_never)) { + if (!h_class_loader.is_null()) { + _class_loader = _handles.add(h_class_loader()); + } + if (!is_anonymous) { // The holder is initialized later for anonymous classes, and before calling anything // that call class_loader(). @@ -269,7 +292,6 @@ void ClassLoaderData::oops_do(OopClosure* f, bool must_claim, bool clear_mod_oop clear_modified_oops(); } - f->do_oop(&_class_loader); _handles.oops_do(f); } @@ -556,7 +578,7 @@ void ClassLoaderData::unload() { if (lt.is_enabled()) { ResourceMark rm; LogStream ls(lt); - ls.print("unload "); + ls.print("unload"); print_value_on(&ls); ls.cr(); } @@ -631,7 +653,7 @@ oop ClassLoaderData::holder_phantom() const { // Unloading support bool ClassLoaderData::is_alive() const { bool alive = keep_alive() // null class loader and incomplete anonymous klasses. - || (_holder.peek() != NULL); // not cleaned by weak reference processing + || (_holder.peek() != NULL); // and not cleaned by the GC weak handle processing. return alive; } @@ -887,13 +909,23 @@ ClassLoaderData* ClassLoaderData::anonymous_class_loader_data(Handle loader) { } const char* ClassLoaderData::loader_name() const { - // Handles null class loader - return SystemDictionary::loader_name(class_loader()); + if (is_unloading()) { + if (_class_loader_klass == NULL) { + return ""; + } else if (_class_loader_name != NULL) { + return _class_loader_name->as_C_string(); + } else { + return _class_loader_klass->name()->as_C_string(); + } + } else { + // Handles null class loader + return SystemDictionary::loader_name(class_loader()); + } } void ClassLoaderData::print_value_on(outputStream* out) const { - if (class_loader() != NULL) { + if (!is_unloading() && 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 { @@ -908,7 +940,7 @@ void ClassLoaderData::print_value_on(outputStream* out) const { #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()); + p2i(this), p2i(_class_loader.ptr_raw()), loader_name()); if (is_anonymous()) out->print(" anonymous"); if (claimed()) out->print(" claimed"); if (is_unloading()) out->print(" unloading"); @@ -962,10 +994,10 @@ 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) { +ClassLoaderData* ClassLoaderDataGraph::add_to_graph(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 + // contains oops in _handles that must be walked. ClassLoaderData* cld = new ClassLoaderData(loader, is_anonymous); @@ -1002,6 +1034,16 @@ ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous) { } while (true); } +ClassLoaderData* ClassLoaderDataGraph::add(Handle loader, bool is_anonymous) { + ClassLoaderData* loader_data = add_to_graph(loader, is_anonymous); + // Initialize name and class after the loader data is added to the CLDG + // because adding the Symbol for the name might safepoint. + if (loader.not_null()) { + loader_data->initialize_name_and_klass(loader); + } + return loader_data; +} + void ClassLoaderDataGraph::oops_do(OopClosure* f, bool must_claim) { for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { cld->oops_do(f, must_claim); diff --git a/src/hotspot/share/classfile/classLoaderData.hpp b/src/hotspot/share/classfile/classLoaderData.hpp index a585a2bea15..92745fc46b5 100644 --- a/src/hotspot/share/classfile/classLoaderData.hpp +++ b/src/hotspot/share/classfile/classLoaderData.hpp @@ -28,7 +28,6 @@ #include "memory/allocation.hpp" #include "memory/memRegion.hpp" #include "memory/metaspace.hpp" -#include "memory/metaspaceCounters.hpp" #include "oops/oopHandle.hpp" #include "oops/weakHandle.hpp" #include "runtime/mutex.hpp" @@ -84,6 +83,7 @@ class ClassLoaderDataGraph : public AllStatic { static volatile size_t _num_instance_classes; static volatile size_t _num_array_classes; + static ClassLoaderData* add_to_graph(Handle class_loader, bool anonymous); static ClassLoaderData* add(Handle class_loader, bool anonymous); static void post_class_unload_events(); public: @@ -220,8 +220,8 @@ class ClassLoaderData : public CHeapObj { static ClassLoaderData * _the_null_class_loader_data; - ClassLoaderWeakHandle _holder; // The oop that determines lifetime of this class loader - oop _class_loader; // The instance of java/lang/ClassLoader associated with + WeakHandle _holder; // The oop that determines lifetime of this class loader + OopHandle _class_loader; // The instance of java/lang/ClassLoader associated with // this ClassLoaderData ClassLoaderMetaspace * volatile _metaspace; // Meta-space where meta-data defined by the @@ -234,7 +234,7 @@ class ClassLoaderData : public CHeapObj { bool _modified_oops; // Card Table Equivalent (YC/CMS support) bool _accumulated_modified_oops; // Mod Union Equivalent (CMS support) - s2 _keep_alive; // if this CLD is kept alive without a keep_alive_object(). + s2 _keep_alive; // if this CLD is kept alive. // Used for anonymous classes and the boot class // loader. _keep_alive does not need to be volatile or // atomic since there is one unique CLD per anonymous class. @@ -265,6 +265,9 @@ class ClassLoaderData : public CHeapObj { // Support for walking class loader data objects ClassLoaderData* _next; /// Next loader_datas created + // JFR support + Klass* _class_loader_klass; + Symbol* _class_loader_name; TRACE_DEFINE_TRACE_ID_FIELD; void set_next(ClassLoaderData* next) { _next = next; } @@ -305,6 +308,8 @@ class ClassLoaderData : public CHeapObj { MetaWord* allocate(size_t size); Dictionary* create_dictionary(); + + void initialize_name_and_klass(Handle class_loader); public: // GC interface. void clear_claimed() { _claimed = 0; } @@ -340,9 +345,7 @@ class ClassLoaderData : public CHeapObj { // Returns true if this class loader data is for the boot class loader. // (Note that the class loader data may be anonymous.) - bool is_boot_class_loader_data() const { - return class_loader() == NULL; - } + inline bool is_boot_class_loader_data() const; bool is_builtin_class_loader_data() const; bool is_permanent_class_loader_data() const; @@ -351,10 +354,7 @@ class ClassLoaderData : public CHeapObj { // method will allocate a Metaspace if needed. ClassLoaderMetaspace* metaspace_non_null(); - oop class_loader() const { return _class_loader; } - - // The object the GC is using to keep this ClassLoaderData alive. - oop keep_alive_object() const; + inline oop class_loader() const; // Returns true if this class loader data is for a loader going away. bool is_unloading() const { @@ -363,7 +363,7 @@ class ClassLoaderData : public CHeapObj { } // Used to refcount an anonymous class's CLD in order to - // indicate their aliveness without a keep_alive_object(). + // indicate their aliveness. void inc_keep_alive(); void dec_keep_alive(); @@ -407,6 +407,9 @@ class ClassLoaderData : public CHeapObj { static ClassLoaderData* class_loader_data_or_null(oop loader); static ClassLoaderData* anonymous_class_loader_data(Handle loader); + + Klass* class_loader_klass() const { return _class_loader_klass; } + Symbol* class_loader_name() const { return _class_loader_name; } TRACE_DEFINE_TRACE_ID_METHODS; }; diff --git a/src/hotspot/share/classfile/classLoaderData.inline.hpp b/src/hotspot/share/classfile/classLoaderData.inline.hpp index d7aa7d15d12..a6e84e92612 100644 --- a/src/hotspot/share/classfile/classLoaderData.inline.hpp +++ b/src/hotspot/share/classfile/classLoaderData.inline.hpp @@ -28,6 +28,18 @@ #include "classfile/classLoaderData.hpp" #include "classfile/javaClasses.hpp" #include "oops/oop.inline.hpp" +#include "oops/oopHandle.inline.hpp" +#include "oops/weakHandle.inline.hpp" + +inline oop ClassLoaderData::class_loader() const { + assert(!_unloading, "This oop is not available to unloading class loader data"); + assert(_holder.is_null() || _holder.peek() != NULL , "This class loader data holder must be alive"); + return _class_loader.resolve(); +} + +inline bool ClassLoaderData::is_boot_class_loader_data() const { + return class_loader() == NULL; + } inline ClassLoaderData* ClassLoaderData::class_loader_data_or_null(oop loader) { if (loader == NULL) { diff --git a/src/hotspot/share/classfile/classLoaderStats.cpp b/src/hotspot/share/classfile/classLoaderStats.cpp index a62e7bada0b..7d1ea4ba28c 100644 --- a/src/hotspot/share/classfile/classLoaderStats.cpp +++ b/src/hotspot/share/classfile/classLoaderStats.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2015, 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 @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/classLoaderStats.hpp" #include "oops/oop.inline.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/src/hotspot/share/classfile/moduleEntry.cpp b/src/hotspot/share/classfile/moduleEntry.cpp index c6f1f2a589b..ce2c960b714 100644 --- a/src/hotspot/share/classfile/moduleEntry.cpp +++ b/src/hotspot/share/classfile/moduleEntry.cpp @@ -24,7 +24,7 @@ #include "precompiled.hpp" #include "jni.h" -#include "classfile/classLoaderData.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/javaClasses.hpp" #include "classfile/moduleEntry.hpp" #include "logging/log.hpp" diff --git a/src/hotspot/share/classfile/placeholders.cpp b/src/hotspot/share/classfile/placeholders.cpp index 7fd3c205608..5dc6c74ed53 100644 --- a/src/hotspot/share/classfile/placeholders.cpp +++ b/src/hotspot/share/classfile/placeholders.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 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 @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/placeholders.hpp" #include "classfile/systemDictionary.hpp" #include "oops/oop.inline.hpp" diff --git a/src/hotspot/share/classfile/systemDictionaryShared.cpp b/src/hotspot/share/classfile/systemDictionaryShared.cpp index fed4ae381f7..d9a6fb8555f 100644 --- a/src/hotspot/share/classfile/systemDictionaryShared.cpp +++ b/src/hotspot/share/classfile/systemDictionaryShared.cpp @@ -357,6 +357,14 @@ Handle SystemDictionaryShared::init_security_info(Handle class_loader, InstanceK return pd; } +bool SystemDictionaryShared::is_sharing_possible(ClassLoaderData* loader_data) { + oop class_loader = loader_data->class_loader(); + return (class_loader == NULL || + (UseAppCDS && (SystemDictionary::is_system_class_loader(class_loader) || + SystemDictionary::is_platform_class_loader(class_loader))) + ); +} + // Currently AppCDS only archives classes from the run-time image, the // -Xbootclasspath/a path, the class path, and the module path. // diff --git a/src/hotspot/share/classfile/systemDictionaryShared.hpp b/src/hotspot/share/classfile/systemDictionaryShared.hpp index c1b87348a5a..6e157924258 100644 --- a/src/hotspot/share/classfile/systemDictionaryShared.hpp +++ b/src/hotspot/share/classfile/systemDictionaryShared.hpp @@ -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 @@ -300,13 +300,7 @@ public: } // Check if sharing is supported for the class loader. - static bool is_sharing_possible(ClassLoaderData* loader_data) { - oop class_loader = loader_data->class_loader(); - return (class_loader == NULL || - (UseAppCDS && (SystemDictionary::is_system_class_loader(class_loader) || - SystemDictionary::is_platform_class_loader(class_loader))) - ); - } + static bool is_sharing_possible(ClassLoaderData* loader_data); static bool is_shared_class_visible_for_classloader(InstanceKlass* ik, Handle class_loader, const char* pkg_string, diff --git a/src/hotspot/share/gc/g1/g1MonitoringSupport.cpp b/src/hotspot/share/gc/g1/g1MonitoringSupport.cpp index a696d46955c..a490dad2671 100644 --- a/src/hotspot/share/gc/g1/g1MonitoringSupport.cpp +++ b/src/hotspot/share/gc/g1/g1MonitoringSupport.cpp @@ -27,6 +27,7 @@ #include "gc/g1/g1MonitoringSupport.hpp" #include "gc/g1/g1Policy.hpp" #include "gc/shared/hSpaceCounters.hpp" +#include "memory/metaspaceCounters.hpp" G1GenerationCounters::G1GenerationCounters(G1MonitoringSupport* g1mm, const char* name, diff --git a/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp b/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp index ae68bff067a..03684211409 100644 --- a/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp +++ b/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp @@ -41,6 +41,7 @@ #include "gc/shared/gcLocker.hpp" #include "gc/shared/gcWhen.hpp" #include "logging/log.hpp" +#include "memory/metaspaceCounters.hpp" #include "oops/oop.inline.hpp" #include "runtime/handles.inline.hpp" #include "runtime/java.hpp" diff --git a/src/hotspot/share/gc/shared/genCollectedHeap.cpp b/src/hotspot/share/gc/shared/genCollectedHeap.cpp index 120093b7f7a..30754c90a83 100644 --- a/src/hotspot/share/gc/shared/genCollectedHeap.cpp +++ b/src/hotspot/share/gc/shared/genCollectedHeap.cpp @@ -49,6 +49,7 @@ #include "gc/shared/weakProcessor.hpp" #include "gc/shared/workgroup.hpp" #include "memory/filemap.hpp" +#include "memory/metaspaceCounters.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" #include "runtime/biasedLocking.hpp" diff --git a/src/hotspot/share/memory/heapInspection.cpp b/src/hotspot/share/memory/heapInspection.cpp index e51edcf32de..8571f8a31a3 100644 --- a/src/hotspot/share/memory/heapInspection.cpp +++ b/src/hotspot/share/memory/heapInspection.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 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 @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "classfile/classLoaderData.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/moduleEntry.hpp" #include "classfile/systemDictionary.hpp" #include "gc/shared/collectedHeap.hpp" diff --git a/src/hotspot/share/memory/universe.cpp b/src/hotspot/share/memory/universe.cpp index e25671a3152..6753728d9dd 100644 --- a/src/hotspot/share/memory/universe.cpp +++ b/src/hotspot/share/memory/universe.cpp @@ -46,6 +46,7 @@ #include "memory/filemap.hpp" #include "memory/metadataFactory.hpp" #include "memory/metaspaceClosure.hpp" +#include "memory/metaspaceCounters.hpp" #include "memory/metaspaceShared.hpp" #include "memory/oopFactory.hpp" #include "memory/resourceArea.hpp" diff --git a/src/hotspot/share/oops/instanceKlass.cpp b/src/hotspot/share/oops/instanceKlass.cpp index 4a2316cf13d..cb85198ee97 100644 --- a/src/hotspot/share/oops/instanceKlass.cpp +++ b/src/hotspot/share/oops/instanceKlass.cpp @@ -28,6 +28,7 @@ #include "classfile/classFileParser.hpp" #include "classfile/classFileStream.hpp" #include "classfile/classLoader.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/javaClasses.hpp" #include "classfile/moduleEntry.hpp" #include "classfile/systemDictionary.hpp" diff --git a/src/hotspot/share/oops/klass.cpp b/src/hotspot/share/oops/klass.cpp index c28ac415e56..245b3bcb8c2 100644 --- a/src/hotspot/share/oops/klass.cpp +++ b/src/hotspot/share/oops/klass.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/dictionary.hpp" #include "classfile/javaClasses.hpp" #include "classfile/systemDictionary.hpp" diff --git a/src/hotspot/share/oops/oopHandle.hpp b/src/hotspot/share/oops/oopHandle.hpp index 255d28bcc11..d2a1bd25151 100644 --- a/src/hotspot/share/oops/oopHandle.hpp +++ b/src/hotspot/share/oops/oopHandle.hpp @@ -46,7 +46,7 @@ public: inline oop resolve() const; // Used only for removing handle. - oop* ptr_raw() { return _obj; } + oop* ptr_raw() const { return _obj; } }; #endif // SHARE_VM_OOPS_OOPHANDLE_HPP diff --git a/src/hotspot/share/oops/weakHandle.cpp b/src/hotspot/share/oops/weakHandle.cpp index 3ba2bd15202..82072718334 100644 --- a/src/hotspot/share/oops/weakHandle.cpp +++ b/src/hotspot/share/oops/weakHandle.cpp @@ -51,8 +51,8 @@ template void WeakHandle::release() const { // Only release if the pointer to the object has been created. if (_obj != NULL) { - // Clear the WeakHandle. For class loader data race, the handle may not have - // been previously cleared by GC. + // Clear the WeakHandle. For race in creating ClassLoaderData, we can release this + // WeakHandle before it is cleared by GC. RootAccess::oop_store(_obj, (oop)NULL); get_storage()->release(_obj); } diff --git a/src/hotspot/share/prims/jvmtiGetLoadedClasses.cpp b/src/hotspot/share/prims/jvmtiGetLoadedClasses.cpp index 23f66e71758..61a616271f6 100644 --- a/src/hotspot/share/prims/jvmtiGetLoadedClasses.cpp +++ b/src/hotspot/share/prims/jvmtiGetLoadedClasses.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/systemDictionary.hpp" #include "gc/shared/collectedHeap.hpp" #include "memory/universe.hpp" diff --git a/src/hotspot/share/runtime/vmStructs.cpp b/src/hotspot/share/runtime/vmStructs.cpp index 8154a96a91f..8ad9b2a6e5d 100644 --- a/src/hotspot/share/runtime/vmStructs.cpp +++ b/src/hotspot/share/runtime/vmStructs.cpp @@ -543,7 +543,7 @@ typedef PaddedEnd PaddedObjectMonitor; /*******************/ \ /* ClassLoaderData */ \ /*******************/ \ - nonstatic_field(ClassLoaderData, _class_loader, oop) \ + nonstatic_field(ClassLoaderData, _class_loader, OopHandle) \ nonstatic_field(ClassLoaderData, _next, ClassLoaderData*) \ volatile_nonstatic_field(ClassLoaderData, _klasses, Klass*) \ nonstatic_field(ClassLoaderData, _is_anonymous, bool) \ diff --git a/src/hotspot/share/services/classLoadingService.cpp b/src/hotspot/share/services/classLoadingService.cpp index 276afe81c99..6fef0b71b61 100644 --- a/src/hotspot/share/services/classLoadingService.cpp +++ b/src/hotspot/share/services/classLoadingService.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 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 @@ -53,7 +53,7 @@ len = name->utf8_length(); \ } \ HOTSPOT_CLASS_##type( /* type = unloaded, loaded */ \ - data, len, (void*)(clss)->class_loader(), (shared)); \ + data, len, (void*)(clss)->class_loader_data(), (shared)); \ } #else // ndef DTRACE_ENABLED diff --git a/src/hotspot/share/trace/traceStream.cpp b/src/hotspot/share/trace/traceStream.cpp index 70291c0b209..78784b47865 100644 --- a/src/hotspot/share/trace/traceStream.cpp +++ b/src/hotspot/share/trace/traceStream.cpp @@ -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 @@ -25,7 +25,7 @@ #include "precompiled.hpp" #include "trace/traceStream.hpp" #if INCLUDE_TRACE -#include "classfile/classLoaderData.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/javaClasses.inline.hpp" #include "memory/resourceArea.hpp" #include "oops/klass.hpp" diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/classfile/ClassLoaderData.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/classfile/ClassLoaderData.java index 6423371e263..17acccf4231 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/classfile/ClassLoaderData.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/classfile/ClassLoaderData.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 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 @@ -42,14 +42,14 @@ public class ClassLoaderData extends VMObject { private static synchronized void initialize(TypeDataBase db) throws WrongTypeException { Type type = db.lookupType("ClassLoaderData"); - classLoaderField = type.getOopField("_class_loader"); + classLoaderField = type.getAddressField("_class_loader"); nextField = type.getAddressField("_next"); klassesField = new MetadataField(type.getAddressField("_klasses"), 0); isAnonymousField = new CIntField(type.getCIntegerField("_is_anonymous"), 0); dictionaryField = type.getAddressField("_dictionary"); } - private static sun.jvm.hotspot.types.OopField classLoaderField; + private static AddressField classLoaderField; private static AddressField nextField; private static MetadataField klassesField; private static CIntField isAnonymousField; @@ -72,7 +72,13 @@ public class ClassLoaderData extends VMObject { } public Oop getClassLoader() { - return VM.getVM().getObjectHeap().newOop(classLoaderField.getValue(getAddress())); + Address handle = classLoaderField.getValue(getAddress()); + if (handle != null) { + // Load through the handle + OopHandle refs = handle.getOopHandleAt(0); + return (Instance)VM.getVM().getObjectHeap().newOop(refs); + } + return null; } public boolean getIsAnonymous() {