Skip to content

Commit 82deb5c

Browse files
committedJan 4, 2023
8298601: Refactor archiving of java.lang.Module objects
Reviewed-by: coleenp, ccheung
1 parent 77ff197 commit 82deb5c

8 files changed

+144
-63
lines changed
 

‎src/hotspot/share/cds/heapShared.cpp

+5-24
Original file line numberDiff line numberDiff line change
@@ -30,9 +30,8 @@
3030
#include "cds/heapShared.hpp"
3131
#include "cds/metaspaceShared.hpp"
3232
#include "classfile/classLoaderData.hpp"
33-
#include "classfile/classLoaderDataShared.hpp"
3433
#include "classfile/javaClasses.inline.hpp"
35-
#include "classfile/moduleEntry.hpp"
34+
#include "classfile/modules.hpp"
3635
#include "classfile/stringTable.hpp"
3736
#include "classfile/symbolTable.hpp"
3837
#include "classfile/systemDictionary.hpp"
@@ -554,7 +553,7 @@ void HeapShared::copy_open_objects(GrowableArray<MemRegion>* open_regions) {
554553
archive_object_subgraphs(fmg_open_archive_subgraph_entry_fields,
555554
false /* is_closed_archive */,
556555
true /* is_full_module_graph */);
557-
ClassLoaderDataShared::init_archived_oops();
556+
Modules::verify_archived_modules();
558557
}
559558

560559
copy_roots();
@@ -1189,25 +1188,6 @@ void HeapShared::check_closed_region_object(InstanceKlass* k) {
11891188
}
11901189
}
11911190

1192-
void HeapShared::check_module_oop(oop orig_module_obj) {
1193-
assert(DumpSharedSpaces, "must be");
1194-
assert(java_lang_Module::is_instance(orig_module_obj), "must be");
1195-
ModuleEntry* orig_module_ent = java_lang_Module::module_entry_raw(orig_module_obj);
1196-
if (orig_module_ent == NULL) {
1197-
// These special Module objects are created in Java code. They are not
1198-
// defined via Modules::define_module(), so they don't have a ModuleEntry:
1199-
// java.lang.Module::ALL_UNNAMED_MODULE
1200-
// java.lang.Module::EVERYONE_MODULE
1201-
// jdk.internal.loader.ClassLoaders$BootClassLoader::unnamedModule
1202-
assert(java_lang_Module::name(orig_module_obj) == NULL, "must be unnamed");
1203-
log_info(cds, heap)("Module oop with No ModuleEntry* @[" PTR_FORMAT "]", p2i(orig_module_obj));
1204-
} else {
1205-
ClassLoaderData* loader_data = orig_module_ent->loader_data();
1206-
assert(loader_data->is_builtin_class_loader_data(), "must be");
1207-
}
1208-
}
1209-
1210-
12111191
// (1) If orig_obj has not been archived yet, archive it.
12121192
// (2) If orig_obj has not been seen yet (since start_recording_subgraph() was called),
12131193
// trace all objects that are reachable from it, and make sure these objects are archived.
@@ -1276,9 +1256,10 @@ oop HeapShared::archive_reachable_objects_from(int level,
12761256
}
12771257

12781258
if (java_lang_Module::is_instance(orig_obj)) {
1279-
check_module_oop(orig_obj);
1259+
if (Modules::check_module_oop(orig_obj)) {
1260+
Modules::update_oops_in_archived_module(orig_obj, append_root(archived_obj));
1261+
}
12801262
java_lang_Module::set_module_entry(archived_obj, NULL);
1281-
java_lang_Module::set_loader(archived_obj, NULL);
12821263
} else if (java_lang_ClassLoader::is_instance(orig_obj)) {
12831264
// class_data will be restored explicitly at run time.
12841265
guarantee(orig_obj == SystemDictionary::java_platform_loader() ||

‎src/hotspot/share/cds/heapShared.hpp

-1
Original file line numberDiff line numberDiff line change
@@ -301,7 +301,6 @@ class HeapShared: AllStatic {
301301
static bool has_been_seen_during_subgraph_recording(oop obj);
302302
static void set_has_been_seen_during_subgraph_recording(oop obj);
303303

304-
static void check_module_oop(oop orig_module_obj);
305304
static void copy_roots();
306305

307306
static void resolve_classes_for_subgraphs(JavaThread* current, ArchivableStaticFieldInfo fields[]);

‎src/hotspot/share/classfile/classLoaderDataShared.cpp

+1-17
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -54,7 +54,6 @@ class ArchivedClassLoaderData {
5454
void iterate_symbols(ClassLoaderData* loader_data, MetaspaceClosure* closure);
5555
void allocate(ClassLoaderData* loader_data);
5656
void init_archived_entries(ClassLoaderData* loader_data);
57-
void init_archived_oops(ClassLoaderData* loader_data);
5857

5958
void serialize(SerializeClosure* f) {
6059
f->do_ptr((void**)&_packages);
@@ -101,14 +100,6 @@ void ArchivedClassLoaderData::init_archived_entries(ClassLoaderData* loader_data
101100
}
102101
}
103102

104-
void ArchivedClassLoaderData::init_archived_oops(ClassLoaderData* loader_data) {
105-
assert(DumpSharedSpaces, "must be");
106-
assert_valid(loader_data);
107-
if (loader_data != NULL) {
108-
loader_data->modules()->init_archived_oops(_modules);
109-
}
110-
}
111-
112103
void ArchivedClassLoaderData::restore(ClassLoaderData* loader_data, bool do_entries, bool do_oops) {
113104
assert(UseSharedSpaces, "must be");
114105
assert_valid(loader_data);
@@ -174,13 +165,6 @@ void ClassLoaderDataShared::init_archived_tables() {
174165
_archived_javabase_moduleEntry = ModuleEntry::get_archived_entry(ModuleEntryTable::javabase_moduleEntry());
175166
}
176167

177-
void ClassLoaderDataShared::init_archived_oops() {
178-
assert(DumpSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
179-
_archived_boot_loader_data.init_archived_oops (null_class_loader_data());
180-
_archived_platform_loader_data.init_archived_oops(java_platform_loader_data_or_null());
181-
_archived_system_loader_data.init_archived_oops (java_system_loader_data_or_null());
182-
}
183-
184168
void ClassLoaderDataShared::serialize(SerializeClosure* f) {
185169
_archived_boot_loader_data.serialize(f);
186170
_archived_platform_loader_data.serialize(f);

‎src/hotspot/share/classfile/classLoaderDataShared.hpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -38,7 +38,6 @@ class ClassLoaderDataShared : AllStatic {
3838
static void allocate_archived_tables();
3939
static void iterate_symbols(MetaspaceClosure* closure);
4040
static void init_archived_tables();
41-
static void init_archived_oops();
4241
static void serialize(SerializeClosure* f);
4342
static void clear_archived_oops();
4443
static oop restore_archived_oops_for_null_class_loader_data();

‎src/hotspot/share/classfile/moduleEntry.cpp

+48-17
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,10 @@
3131
#include "classfile/classLoaderData.inline.hpp"
3232
#include "classfile/javaClasses.inline.hpp"
3333
#include "classfile/moduleEntry.hpp"
34+
#include "classfile/systemDictionary.hpp"
3435
#include "jni.h"
3536
#include "logging/log.hpp"
37+
#include "logging/logStream.hpp"
3638
#include "memory/resourceArea.hpp"
3739
#include "memory/universe.hpp"
3840
#include "oops/oopHandle.inline.hpp"
@@ -386,20 +388,38 @@ typedef ResourceHashtable<
386388
AnyObj::C_HEAP> ArchivedModuleEntries;
387389
static ArchivedModuleEntries* _archive_modules_entries = NULL;
388390

391+
#ifndef PRODUCT
392+
static int _num_archived_module_entries = 0;
393+
static int _num_inited_module_entries = 0;
394+
#endif
395+
389396
ModuleEntry* ModuleEntry::allocate_archived_entry() const {
390397
assert(is_named(), "unnamed packages/modules are not archived");
391398
ModuleEntry* archived_entry = (ModuleEntry*)ArchiveBuilder::rw_region_alloc(sizeof(ModuleEntry));
392399
memcpy((void*)archived_entry, (void*)this, sizeof(ModuleEntry));
400+
archived_entry->_archived_module_index = -1;
393401

394402
if (_archive_modules_entries == NULL) {
395403
_archive_modules_entries = new (mtClass)ArchivedModuleEntries();
396404
}
397405
assert(_archive_modules_entries->get(this) == NULL, "Each ModuleEntry must not be shared across ModuleEntryTables");
398406
_archive_modules_entries->put(this, archived_entry);
407+
DEBUG_ONLY(_num_archived_module_entries++);
399408

409+
if (log_is_enabled(Info, cds, module)) {
410+
ResourceMark rm;
411+
LogStream ls(Log(cds, module)::info());
412+
ls.print("Stored in archive: ");
413+
archived_entry->print(&ls);
414+
}
400415
return archived_entry;
401416
}
402417

418+
bool ModuleEntry::has_been_archived() {
419+
assert(!ArchiveBuilder::current()->is_in_buffer_space(this), "must be called on original ModuleEntry");
420+
return _archive_modules_entries->contains(this);
421+
}
422+
403423
ModuleEntry* ModuleEntry::get_archived_entry(ModuleEntry* orig_entry) {
404424
ModuleEntry** ptr = _archive_modules_entries->get(orig_entry);
405425
assert(ptr != NULL && *ptr != NULL, "must have been allocated");
@@ -467,27 +487,39 @@ void ModuleEntry::init_as_archived_entry() {
467487
ArchivePtrMarker::mark_pointer((address*)&_location);
468488
}
469489

470-
void ModuleEntry::init_archived_oops() {
490+
void ModuleEntry::update_oops_in_archived_module(int root_oop_index) {
471491
assert(DumpSharedSpaces, "static dump only");
472-
oop module_obj = module();
473-
if (module_obj != NULL) {
474-
oop m = HeapShared::find_archived_heap_object(module_obj);
475-
assert(m != NULL, "sanity");
476-
_archived_module_index = HeapShared::append_root(m);
477-
}
492+
assert(_archived_module_index == -1, "must be set exactly once");
493+
assert(root_oop_index >= 0, "sanity");
494+
495+
_archived_module_index = root_oop_index;
496+
478497
assert(shared_protection_domain() == NULL, "never set during -Xshare:dump");
479498
// Clear handles and restore at run time. Handles cannot be archived.
480499
OopHandle null_handle;
481500
_module = null_handle;
501+
502+
// For verify_archived_module_entries()
503+
DEBUG_ONLY(_num_inited_module_entries++);
482504
}
483505

506+
#ifndef PRODUCT
507+
void ModuleEntry::verify_archived_module_entries() {
508+
assert(_num_archived_module_entries == _num_inited_module_entries,
509+
"%d ModuleEntries have been archived but %d of them have been properly initialized with archived java.lang.Module objects",
510+
_num_archived_module_entries, _num_inited_module_entries);
511+
}
512+
#endif // PRODUCT
513+
484514
void ModuleEntry::load_from_archive(ClassLoaderData* loader_data) {
515+
assert(UseSharedSpaces, "runtime only");
485516
set_loader_data(loader_data);
486517
_reads = restore_growable_array((Array<ModuleEntry*>*)_reads);
487518
JFR_ONLY(INIT_ID(this);)
488519
}
489520

490521
void ModuleEntry::restore_archived_oops(ClassLoaderData* loader_data) {
522+
assert(UseSharedSpaces, "runtime only");
491523
Handle module_handle(Thread::current(), HeapShared::get_root(_archived_module_index, /*clear=*/true));
492524
assert(module_handle.not_null(), "huh");
493525
set_module(loader_data->add_handle(module_handle));
@@ -496,12 +528,19 @@ void ModuleEntry::restore_archived_oops(ClassLoaderData* loader_data) {
496528
// because it may be affected by archive relocation.
497529
java_lang_Module::set_module_entry(module_handle(), this);
498530

499-
if (loader_data->class_loader() != NULL) {
500-
java_lang_Module::set_loader(module_handle(), loader_data->class_loader());
531+
assert(java_lang_Module::loader(module_handle()) == loader_data->class_loader(),
532+
"must be set in dump time");
533+
534+
if (log_is_enabled(Info, cds, module)) {
535+
ResourceMark rm;
536+
LogStream ls(Log(cds, module)::info());
537+
ls.print("Restored from archive: ");
538+
print(&ls);
501539
}
502540
}
503541

504542
void ModuleEntry::clear_archived_oops() {
543+
assert(UseSharedSpaces, "runtime only");
505544
HeapShared::clear_root(_archived_module_index);
506545
}
507546

@@ -544,14 +583,6 @@ void ModuleEntryTable::init_archived_entries(Array<ModuleEntry*>* archived_modul
544583
}
545584
}
546585

547-
void ModuleEntryTable::init_archived_oops(Array<ModuleEntry*>* archived_modules) {
548-
assert(DumpSharedSpaces, "dump time only");
549-
for (int i = 0; i < archived_modules->length(); i++) {
550-
ModuleEntry* archived_entry = archived_modules->at(i);
551-
archived_entry->init_archived_oops();
552-
}
553-
}
554-
555586
void ModuleEntryTable::load_archived_entries(ClassLoaderData* loader_data,
556587
Array<ModuleEntry*>* archived_modules) {
557588
assert(UseSharedSpaces, "runtime only");

‎src/hotspot/share/classfile/moduleEntry.hpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -175,13 +175,15 @@ class ModuleEntry : public CHeapObj<mtModule> {
175175
void iterate_symbols(MetaspaceClosure* closure);
176176
ModuleEntry* allocate_archived_entry() const;
177177
void init_as_archived_entry();
178-
void init_archived_oops();
179178
static ModuleEntry* get_archived_entry(ModuleEntry* orig_entry);
179+
bool has_been_archived();
180180
static Array<ModuleEntry*>* write_growable_array(GrowableArray<ModuleEntry*>* array);
181181
static GrowableArray<ModuleEntry*>* restore_growable_array(Array<ModuleEntry*>* archived_array);
182182
void load_from_archive(ClassLoaderData* loader_data);
183183
void restore_archived_oops(ClassLoaderData* loader_data);
184184
void clear_archived_oops();
185+
void update_oops_in_archived_module(int root_oop_index);
186+
static void verify_archived_module_entries() PRODUCT_RETURN;
185187
#endif
186188
};
187189

@@ -250,7 +252,6 @@ class ModuleEntryTable : public CHeapObj<mtModule> {
250252
void iterate_symbols(MetaspaceClosure* closure);
251253
Array<ModuleEntry*>* allocate_archived_entries();
252254
void init_archived_entries(Array<ModuleEntry*>* archived_modules);
253-
void init_archived_oops(Array<ModuleEntry*>* archived_modules);
254255
void load_archived_entries(ClassLoaderData* loader_data,
255256
Array<ModuleEntry*>* archived_modules);
256257
void restore_archived_oops(ClassLoaderData* loader_data,

‎src/hotspot/share/classfile/modules.cpp

+82
Original file line numberDiff line numberDiff line change
@@ -477,6 +477,88 @@ void Modules::define_module(Handle module, jboolean is_open, jstring version,
477477
}
478478

479479
#if INCLUDE_CDS_JAVA_HEAP
480+
static bool _seen_platform_unnamed_module = false;
481+
static bool _seen_system_unnamed_module = false;
482+
483+
// Validate the states of an java.lang.Module oop to be archived.
484+
//
485+
// Returns true iff the oop has an archived ModuleEntry.
486+
bool Modules::check_module_oop(oop orig_module_obj) {
487+
assert(DumpSharedSpaces, "must be");
488+
assert(MetaspaceShared::use_full_module_graph(), "must be");
489+
assert(java_lang_Module::is_instance(orig_module_obj), "must be");
490+
491+
ModuleEntry* orig_module_ent = java_lang_Module::module_entry_raw(orig_module_obj);
492+
if (orig_module_ent == nullptr) {
493+
// These special java.lang.Module oops are created in Java code. They are not
494+
// defined via Modules::define_module(), so they don't have a ModuleEntry:
495+
// java.lang.Module::ALL_UNNAMED_MODULE
496+
// java.lang.Module::EVERYONE_MODULE
497+
// jdk.internal.loader.ClassLoaders$BootClassLoader::unnamedModule
498+
log_info(cds, module)("Archived java.lang.Module oop " PTR_FORMAT " with no ModuleEntry*", p2i(orig_module_obj));
499+
assert(java_lang_Module::name(orig_module_obj) == nullptr, "must be unnamed");
500+
return false;
501+
} else {
502+
// This java.lang.Module oop has an ModuleEntry*. Check if the latter is archived.
503+
if (log_is_enabled(Info, cds, module)) {
504+
ResourceMark rm;
505+
LogStream ls(Log(cds, module)::info());
506+
ls.print("Archived java.lang.Module oop " PTR_FORMAT " for ", p2i(orig_module_obj));
507+
orig_module_ent->print(&ls);
508+
}
509+
510+
// We only archive the default module graph, which should contain only java.lang.Module oops
511+
// for the 3 built-in loaders (boot/platform/system)
512+
ClassLoaderData* loader_data = orig_module_ent->loader_data();
513+
assert(loader_data->is_builtin_class_loader_data(), "must be");
514+
515+
if (orig_module_ent->name() != nullptr) {
516+
// For each named module, we archive both the java.lang.Module oop and the ModuleEntry.
517+
assert(orig_module_ent->has_been_archived(), "sanity");
518+
return true;
519+
} else {
520+
// We only archive two unnamed module oops (for platform and system loaders). These do NOT have an archived
521+
// ModuleEntry.
522+
//
523+
// At runtime, these oops are fetched from java_lang_ClassLoader::unnamedModule(loader) and
524+
// are initialized in ClassLoaderData::ClassLoaderData() => ModuleEntry::create_unnamed_module(), where
525+
// a new ModuleEntry is allocated.
526+
assert(!loader_data->is_boot_class_loader_data(), "unnamed module for boot loader should be not archived");
527+
assert(!orig_module_ent->has_been_archived(), "sanity");
528+
529+
if (SystemDictionary::is_platform_class_loader(loader_data->class_loader())) {
530+
assert(!_seen_platform_unnamed_module, "only once");
531+
_seen_platform_unnamed_module = true;
532+
} else if (SystemDictionary::is_system_class_loader(loader_data->class_loader())) {
533+
assert(!_seen_system_unnamed_module, "only once");
534+
_seen_system_unnamed_module = true;
535+
} else {
536+
// The java.lang.Module oop and ModuleEntry of the unnamed module of the boot loader are
537+
// not in the archived module graph. These are always allocated at runtime.
538+
ShouldNotReachHere();
539+
}
540+
return false;
541+
}
542+
}
543+
}
544+
545+
void Modules::update_oops_in_archived_module(oop orig_module_obj, int archived_module_root_index) {
546+
// This java.lang.Module oop must have an archived ModuleEntry
547+
assert(check_module_oop(orig_module_obj) == true, "sanity");
548+
549+
// We remember the oop inside the ModuleEntry::_archived_module_index. At runtime, we use
550+
// this index to reinitialize the ModuleEntry inside ModuleEntry::restore_archived_oops().
551+
//
552+
// ModuleEntry::verify_archived_module_entries(), called below, ensures that every archived
553+
// ModuleEntry has been assigned an _archived_module_index.
554+
ModuleEntry* orig_module_ent = java_lang_Module::module_entry_raw(orig_module_obj);
555+
ModuleEntry::get_archived_entry(orig_module_ent)->update_oops_in_archived_module(archived_module_root_index);
556+
}
557+
558+
void Modules::verify_archived_modules() {
559+
ModuleEntry::verify_archived_module_entries();
560+
}
561+
480562
void Modules::define_archived_modules(Handle h_platform_loader, Handle h_system_loader, TRAPS) {
481563
assert(UseSharedSpaces && MetaspaceShared::use_full_module_graph(), "must be");
482564

‎src/hotspot/share/classfile/modules.hpp

+4
Original file line numberDiff line numberDiff line change
@@ -53,8 +53,12 @@ class Modules : AllStatic {
5353
static void define_module(Handle module, jboolean is_open, jstring version,
5454
jstring location, jobjectArray packages, TRAPS);
5555

56+
static bool check_module_oop(oop orig_module_obj) NOT_CDS_JAVA_HEAP_RETURN_(false);
57+
static void update_oops_in_archived_module(oop orig_module_obj, int archived_module_root_index)
58+
NOT_CDS_JAVA_HEAP_RETURN;
5659
static void define_archived_modules(Handle h_platform_loader, Handle h_system_loader,
5760
TRAPS) NOT_CDS_JAVA_HEAP_RETURN;
61+
static void verify_archived_modules() NOT_CDS_JAVA_HEAP_RETURN;
5862

5963
// Provides the java.lang.Module for the unnamed module defined
6064
// to the boot loader.

0 commit comments

Comments
 (0)
Please sign in to comment.