mirror of
https://github.com/openjdk/jdk.git
synced 2025-09-17 09:34:38 +02:00
207 lines
8.3 KiB
C++
207 lines
8.3 KiB
C++
/*
|
|
* Copyright (c) 2020, 2023, 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
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*
|
|
*/
|
|
|
|
#include "precompiled.hpp"
|
|
#include "cds/metaspaceShared.hpp"
|
|
#include "classfile/classLoaderData.inline.hpp"
|
|
#include "classfile/classLoaderDataShared.hpp"
|
|
#include "classfile/moduleEntry.hpp"
|
|
#include "classfile/packageEntry.hpp"
|
|
#include "classfile/systemDictionary.hpp"
|
|
#include "logging/log.hpp"
|
|
#include "runtime/handles.inline.hpp"
|
|
|
|
#if INCLUDE_CDS_JAVA_HEAP
|
|
|
|
bool ClassLoaderDataShared::_full_module_graph_loaded = false;
|
|
|
|
class ArchivedClassLoaderData {
|
|
Array<PackageEntry*>* _packages;
|
|
Array<ModuleEntry*>* _modules;
|
|
|
|
void assert_valid(ClassLoaderData* loader_data) {
|
|
// loader_data may be null if the boot layer has loaded no modules for the platform or
|
|
// system loaders (e.g., if you create a custom JDK image with only java.base).
|
|
if (loader_data != nullptr) {
|
|
assert(!loader_data->has_class_mirror_holder(),
|
|
"loaders for non-strong hidden classes not supported");
|
|
}
|
|
}
|
|
public:
|
|
ArchivedClassLoaderData() : _packages(nullptr), _modules(nullptr) {}
|
|
|
|
void iterate_symbols(ClassLoaderData* loader_data, MetaspaceClosure* closure);
|
|
void allocate(ClassLoaderData* loader_data);
|
|
void init_archived_entries(ClassLoaderData* loader_data);
|
|
|
|
void serialize(SerializeClosure* f) {
|
|
f->do_ptr(&_packages);
|
|
f->do_ptr(&_modules);
|
|
}
|
|
|
|
void restore(ClassLoaderData* loader_data, bool do_entries, bool do_oops);
|
|
void clear_archived_oops();
|
|
};
|
|
|
|
static ArchivedClassLoaderData _archived_boot_loader_data;
|
|
static ArchivedClassLoaderData _archived_platform_loader_data;
|
|
static ArchivedClassLoaderData _archived_system_loader_data;
|
|
static ModuleEntry* _archived_javabase_moduleEntry = nullptr;
|
|
|
|
void ArchivedClassLoaderData::iterate_symbols(ClassLoaderData* loader_data, MetaspaceClosure* closure) {
|
|
assert(DumpSharedSpaces, "must be");
|
|
assert_valid(loader_data);
|
|
if (loader_data != nullptr) {
|
|
loader_data->packages()->iterate_symbols(closure);
|
|
loader_data->modules() ->iterate_symbols(closure);
|
|
}
|
|
}
|
|
|
|
void ArchivedClassLoaderData::allocate(ClassLoaderData* loader_data) {
|
|
assert(DumpSharedSpaces, "must be");
|
|
assert_valid(loader_data);
|
|
if (loader_data != nullptr) {
|
|
// We can't create hashtables at dump time because the hashcode depends on the
|
|
// address of the Symbols, which may be relocated at runtime due to ASLR.
|
|
// So we store the packages/modules in Arrays. At runtime, we create
|
|
// the hashtables using these arrays.
|
|
_packages = loader_data->packages()->allocate_archived_entries();
|
|
_modules = loader_data->modules() ->allocate_archived_entries();
|
|
}
|
|
}
|
|
|
|
void ArchivedClassLoaderData::init_archived_entries(ClassLoaderData* loader_data) {
|
|
assert(DumpSharedSpaces, "must be");
|
|
assert_valid(loader_data);
|
|
if (loader_data != nullptr) {
|
|
loader_data->packages()->init_archived_entries(_packages);
|
|
loader_data->modules() ->init_archived_entries(_modules);
|
|
}
|
|
}
|
|
|
|
void ArchivedClassLoaderData::restore(ClassLoaderData* loader_data, bool do_entries, bool do_oops) {
|
|
assert(UseSharedSpaces, "must be");
|
|
assert_valid(loader_data);
|
|
if (_modules != nullptr) { // Could be null if we have archived no modules for platform/system loaders
|
|
ModuleEntryTable* modules = loader_data->modules();
|
|
PackageEntryTable* packages = loader_data->packages();
|
|
|
|
MutexLocker m1(Module_lock);
|
|
if (do_entries) {
|
|
modules->load_archived_entries(loader_data, _modules);
|
|
packages->load_archived_entries(_packages);
|
|
}
|
|
if (do_oops) {
|
|
modules->restore_archived_oops(loader_data, _modules);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ArchivedClassLoaderData::clear_archived_oops() {
|
|
assert(UseSharedSpaces, "must be");
|
|
if (_modules != nullptr) {
|
|
for (int i = 0; i < _modules->length(); i++) {
|
|
_modules->at(i)->clear_archived_oops();
|
|
}
|
|
}
|
|
}
|
|
|
|
// ------------------------------
|
|
|
|
static ClassLoaderData* null_class_loader_data() {
|
|
ClassLoaderData* loader_data = ClassLoaderData::the_null_class_loader_data();
|
|
assert(loader_data != nullptr, "must be");
|
|
return loader_data;
|
|
}
|
|
|
|
static ClassLoaderData* java_platform_loader_data_or_null() {
|
|
return ClassLoaderData::class_loader_data_or_null(SystemDictionary::java_platform_loader());
|
|
}
|
|
|
|
static ClassLoaderData* java_system_loader_data_or_null() {
|
|
return ClassLoaderData::class_loader_data_or_null(SystemDictionary::java_system_loader());
|
|
}
|
|
|
|
void ClassLoaderDataShared::iterate_symbols(MetaspaceClosure* closure) {
|
|
assert(DumpSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_boot_loader_data.iterate_symbols (null_class_loader_data(), closure);
|
|
_archived_platform_loader_data.iterate_symbols(java_platform_loader_data_or_null(), closure);
|
|
_archived_system_loader_data.iterate_symbols (java_system_loader_data_or_null(), closure);
|
|
}
|
|
|
|
void ClassLoaderDataShared::allocate_archived_tables() {
|
|
assert(DumpSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_boot_loader_data.allocate (null_class_loader_data());
|
|
_archived_platform_loader_data.allocate(java_platform_loader_data_or_null());
|
|
_archived_system_loader_data.allocate (java_system_loader_data_or_null());
|
|
}
|
|
|
|
void ClassLoaderDataShared::init_archived_tables() {
|
|
assert(DumpSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_boot_loader_data.init_archived_entries (null_class_loader_data());
|
|
_archived_platform_loader_data.init_archived_entries(java_platform_loader_data_or_null());
|
|
_archived_system_loader_data.init_archived_entries (java_system_loader_data_or_null());
|
|
_archived_javabase_moduleEntry = ModuleEntry::get_archived_entry(ModuleEntryTable::javabase_moduleEntry());
|
|
}
|
|
|
|
void ClassLoaderDataShared::serialize(SerializeClosure* f) {
|
|
_archived_boot_loader_data.serialize(f);
|
|
_archived_platform_loader_data.serialize(f);
|
|
_archived_system_loader_data.serialize(f);
|
|
f->do_ptr(&_archived_javabase_moduleEntry);
|
|
|
|
if (f->reading() && MetaspaceShared::use_full_module_graph()) {
|
|
// Must be done before ClassLoader::create_javabase()
|
|
_archived_boot_loader_data.restore(null_class_loader_data(), true, false);
|
|
ModuleEntryTable::set_javabase_moduleEntry(_archived_javabase_moduleEntry);
|
|
log_info(cds)("use_full_module_graph = true; java.base = " INTPTR_FORMAT,
|
|
p2i(_archived_javabase_moduleEntry));
|
|
}
|
|
}
|
|
|
|
void ClassLoaderDataShared::clear_archived_oops() {
|
|
assert(UseSharedSpaces && !MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_boot_loader_data.clear_archived_oops();
|
|
_archived_platform_loader_data.clear_archived_oops();
|
|
_archived_system_loader_data.clear_archived_oops();
|
|
}
|
|
|
|
oop ClassLoaderDataShared::restore_archived_oops_for_null_class_loader_data() {
|
|
assert(UseSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_boot_loader_data.restore(null_class_loader_data(), false, true);
|
|
return _archived_javabase_moduleEntry->module();
|
|
}
|
|
|
|
void ClassLoaderDataShared::restore_java_platform_loader_from_archive(ClassLoaderData* loader_data) {
|
|
assert(UseSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_platform_loader_data.restore(loader_data, true, true);
|
|
}
|
|
|
|
void ClassLoaderDataShared::restore_java_system_loader_from_archive(ClassLoaderData* loader_data) {
|
|
assert(UseSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
|
|
_archived_system_loader_data.restore(loader_data, true, true);
|
|
_full_module_graph_loaded = true;
|
|
}
|
|
|
|
#endif // INCLUDE_CDS_JAVA_HEAP
|