1 /*
2 * Copyright (c) 2018, 2024, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
24
25 #include "precompiled.hpp"
26 #include "cds/aotClassInitializer.hpp"
27 #include "cds/archiveBuilder.hpp"
28 #include "cds/archiveHeapLoader.hpp"
29 #include "cds/archiveHeapWriter.hpp"
30 #include "cds/archiveUtils.hpp"
31 #include "cds/cdsConfig.hpp"
32 #include "cds/cdsEnumKlass.hpp"
33 #include "cds/cdsHeapVerifier.hpp"
34 #include "cds/heapShared.hpp"
35 #include "cds/metaspaceShared.hpp"
36 #include "classfile/classLoaderData.hpp"
37 #include "classfile/classLoaderExt.hpp"
38 #include "classfile/javaClasses.inline.hpp"
39 #include "classfile/modules.hpp"
40 #include "classfile/stringTable.hpp"
41 #include "classfile/symbolTable.hpp"
42 #include "classfile/systemDictionary.hpp"
43 #include "classfile/systemDictionaryShared.hpp"
44 #include "classfile/vmClasses.hpp"
45 #include "classfile/vmSymbols.hpp"
46 #include "gc/shared/collectedHeap.hpp"
47 #include "gc/shared/gcLocker.hpp"
48 #include "gc/shared/gcVMOperations.hpp"
49 #include "logging/log.hpp"
50 #include "logging/logStream.hpp"
69 #include "gc/g1/g1CollectedHeap.hpp"
70 #endif
71
72 #if INCLUDE_CDS_JAVA_HEAP
73
74 struct ArchivableStaticFieldInfo {
75 const char* klass_name;
76 const char* field_name;
77 InstanceKlass* klass;
78 int offset;
79 BasicType type;
80
81 ArchivableStaticFieldInfo(const char* k, const char* f)
82 : klass_name(k), field_name(f), klass(nullptr), offset(0), type(T_ILLEGAL) {}
83
84 bool valid() {
85 return klass_name != nullptr;
86 }
87 };
88
89 bool HeapShared::_disable_writing = false;
90 DumpedInternedStrings *HeapShared::_dumped_interned_strings = nullptr;
91
92 size_t HeapShared::_alloc_count[HeapShared::ALLOC_STAT_SLOTS];
93 size_t HeapShared::_alloc_size[HeapShared::ALLOC_STAT_SLOTS];
94 size_t HeapShared::_total_obj_count;
95 size_t HeapShared::_total_obj_size;
96
97 #ifndef PRODUCT
98 #define ARCHIVE_TEST_FIELD_NAME "archivedObjects"
99 static Array<char>* _archived_ArchiveHeapTestClass = nullptr;
100 static const char* _test_class_name = nullptr;
101 static Klass* _test_class = nullptr;
102 static const ArchivedKlassSubGraphInfoRecord* _test_class_record = nullptr;
103 #endif
104
105
106 //
107 // If you add new entries to the following tables, you should know what you're doing!
108 //
109
110 static ArchivableStaticFieldInfo archive_subgraph_entry_fields[] = {
111 {"java/lang/Integer$IntegerCache", "archivedCache"},
112 {"java/lang/Long$LongCache", "archivedCache"},
113 {"java/lang/Byte$ByteCache", "archivedCache"},
114 {"java/lang/Short$ShortCache", "archivedCache"},
115 {"java/lang/Character$CharacterCache", "archivedCache"},
116 {"java/util/jar/Attributes$Name", "KNOWN_NAMES"},
117 {"sun/util/locale/BaseLocale", "constantBaseLocales"},
118 {"jdk/internal/module/ArchivedModuleGraph", "archivedModuleGraph"},
119 {"java/util/ImmutableCollections", "archivedObjects"},
120 {"java/lang/ModuleLayer", "EMPTY_LAYER"},
121 {"java/lang/module/Configuration", "EMPTY_CONFIGURATION"},
122 {"jdk/internal/math/FDBigInteger", "archivedCaches"},
123
124 #ifndef PRODUCT
125 {nullptr, nullptr}, // Extra slot for -XX:ArchiveHeapTestClass
126 #endif
127 {nullptr, nullptr},
128 };
129
130 // full module graph
131 static ArchivableStaticFieldInfo fmg_archive_subgraph_entry_fields[] = {
132 {"jdk/internal/loader/ArchivedClassLoaders", "archivedClassLoaders"},
133 {ARCHIVED_BOOT_LAYER_CLASS, ARCHIVED_BOOT_LAYER_FIELD},
134 {"java/lang/Module$ArchivedData", "archivedData"},
135 {nullptr, nullptr},
136 };
137
138 KlassSubGraphInfo* HeapShared::_dump_time_special_subgraph;
139 ArchivedKlassSubGraphInfoRecord* HeapShared::_run_time_special_subgraph;
140 GrowableArrayCHeap<oop, mtClassShared>* HeapShared::_pending_roots = nullptr;
141 GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_root_segments;
142 int HeapShared::_root_segment_max_size_elems;
143 OopHandle HeapShared::_scratch_basic_type_mirrors[T_VOID+1];
144 MetaspaceObjToOopHandleTable* HeapShared::_scratch_java_mirror_table = nullptr;
145 MetaspaceObjToOopHandleTable* HeapShared::_scratch_references_table = nullptr;
146
147 static bool is_subgraph_root_class_of(ArchivableStaticFieldInfo fields[], InstanceKlass* ik) {
148 for (int i = 0; fields[i].valid(); i++) {
149 if (fields[i].klass == ik) {
150 return true;
151 }
152 }
153 return false;
154 }
155
156 bool HeapShared::is_subgraph_root_class(InstanceKlass* ik) {
157 return is_subgraph_root_class_of(archive_subgraph_entry_fields, ik) ||
158 is_subgraph_root_class_of(fmg_archive_subgraph_entry_fields, ik);
159 }
160
208 vmSymbols::void_BuiltinClassLoader_signature(),
209 CHECK);
210 Handle boot_loader(THREAD, result.get_oop());
211 reset_states(boot_loader(), CHECK);
212 }
213
214 HeapShared::ArchivedObjectCache* HeapShared::_archived_object_cache = nullptr;
215
216 bool HeapShared::has_been_archived(oop obj) {
217 assert(CDSConfig::is_dumping_heap(), "dump-time only");
218 return archived_object_cache()->get(obj) != nullptr;
219 }
220
221 int HeapShared::append_root(oop obj) {
222 assert(CDSConfig::is_dumping_heap(), "dump-time only");
223
224 // No GC should happen since we aren't scanning _pending_roots.
225 assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
226
227 if (_pending_roots == nullptr) {
228 _pending_roots = new GrowableArrayCHeap<oop, mtClassShared>(500);
229 }
230
231 return _pending_roots->append(obj);
232 }
233
234 objArrayOop HeapShared::root_segment(int segment_idx) {
235 if (CDSConfig::is_dumping_heap()) {
236 assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
237 if (!HeapShared::can_write()) {
238 return nullptr;
239 }
240 } else {
241 assert(CDSConfig::is_using_archive(), "must be");
242 }
243
244 objArrayOop segment = (objArrayOop)_root_segments->at(segment_idx).resolve();
245 assert(segment != nullptr, "should have been initialized");
246 return segment;
247 }
248
249 void HeapShared::get_segment_indexes(int idx, int& seg_idx, int& int_idx) {
250 assert(_root_segment_max_size_elems > 0, "sanity");
251
252 // Try to avoid divisions for the common case.
253 if (idx < _root_segment_max_size_elems) {
254 seg_idx = 0;
255 int_idx = idx;
256 } else {
257 seg_idx = idx / _root_segment_max_size_elems;
258 int_idx = idx % _root_segment_max_size_elems;
259 }
260
261 assert(idx == seg_idx * _root_segment_max_size_elems + int_idx,
262 "sanity: %d index maps to %d segment and %d internal", idx, seg_idx, int_idx);
263 }
264
265 // Returns an objArray that contains all the roots of the archived objects
266 oop HeapShared::get_root(int index, bool clear) {
267 assert(index >= 0, "sanity");
268 assert(!CDSConfig::is_dumping_heap() && CDSConfig::is_using_archive(), "runtime only");
350 return nullptr;
351 }
352 }
353 void set_oop(MetaspaceObj* ptr, oop o) {
354 MutexLocker ml(ScratchObjects_lock, Mutex::_no_safepoint_check_flag);
355 OopHandle handle(Universe::vm_global(), o);
356 bool is_new = put(ptr, handle);
357 assert(is_new, "cannot set twice");
358 }
359 void remove_oop(MetaspaceObj* ptr) {
360 MutexLocker ml(ScratchObjects_lock, Mutex::_no_safepoint_check_flag);
361 OopHandle* handle = get(ptr);
362 if (handle != nullptr) {
363 handle->release(Universe::vm_global());
364 remove(ptr);
365 }
366 }
367 };
368
369 void HeapShared::add_scratch_resolved_references(ConstantPool* src, objArrayOop dest) {
370 _scratch_references_table->set_oop(src, dest);
371 }
372
373 objArrayOop HeapShared::scratch_resolved_references(ConstantPool* src) {
374 return (objArrayOop)_scratch_references_table->get_oop(src);
375 }
376
377 void HeapShared::init_scratch_objects(TRAPS) {
378 for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
379 BasicType bt = (BasicType)i;
380 if (!is_reference_type(bt)) {
381 oop m = java_lang_Class::create_basic_type_mirror(type2name(bt), bt, CHECK);
382 _scratch_basic_type_mirrors[i] = OopHandle(Universe::vm_global(), m);
383 }
384 }
385 _scratch_java_mirror_table = new (mtClass)MetaspaceObjToOopHandleTable();
386 _scratch_references_table = new (mtClass)MetaspaceObjToOopHandleTable();
387 }
388
389 // Given java_mirror that represents a (primitive or reference) type T,
390 // return the "scratch" version that represents the same type T.
391 // Note that if java_mirror will be returned if it's already a
392 // scratch mirror.
393 //
394 // See java_lang_Class::create_scratch_mirror() for more info.
395 oop HeapShared::scratch_java_mirror(oop java_mirror) {
396 assert(java_lang_Class::is_instance(java_mirror), "must be");
397
398 for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
399 BasicType bt = (BasicType)i;
400 if (!is_reference_type(bt)) {
401 if (_scratch_basic_type_mirrors[i].resolve() == java_mirror) {
402 return java_mirror;
403 }
404 }
405 }
406
407 if (java_lang_Class::is_primitive(java_mirror)) {
408 return scratch_java_mirror(java_lang_Class::as_BasicType(java_mirror));
409 } else {
410 return scratch_java_mirror(java_lang_Class::as_Klass(java_mirror));
411 }
412 }
413
414 oop HeapShared::scratch_java_mirror(BasicType t) {
415 assert((uint)t < T_VOID+1, "range check");
416 assert(!is_reference_type(t), "sanity");
417 return _scratch_basic_type_mirrors[t].resolve();
418 }
419
420 oop HeapShared::scratch_java_mirror(Klass* k) {
421 return _scratch_java_mirror_table->get_oop(k);
422 }
423
424 void HeapShared::set_scratch_java_mirror(Klass* k, oop mirror) {
425 _scratch_java_mirror_table->set_oop(k, mirror);
426 }
427
428 void HeapShared::remove_scratch_objects(Klass* k) {
429 // Klass is being deallocated. Java mirror can still be alive, and it should not
430 // point to dead klass. We need to break the link from mirror to the Klass.
431 // See how InstanceKlass::deallocate_contents does it for normal mirrors.
432 oop mirror = _scratch_java_mirror_table->get_oop(k);
433 if (mirror != nullptr) {
434 java_lang_Class::set_klass(mirror, nullptr);
435 }
436 _scratch_java_mirror_table->remove_oop(k);
437 if (k->is_instance_klass()) {
438 _scratch_references_table->remove(InstanceKlass::cast(k)->constants());
439 }
440 }
441
442 //TODO: we eventually want a more direct test for these kinds of things.
443 //For example the JVM could record some bit of context from the creation
444 //of the klass, such as who called the hidden class factory. Using
445 //string compares on names is fragile and will break as soon as somebody
446 //changes the names in the JDK code. See discussion in JDK-8342481 for
447 //related ideas about marking AOT-related classes.
448 bool HeapShared::is_lambda_form_klass(InstanceKlass* ik) {
449 return ik->is_hidden() &&
450 (ik->name()->starts_with("java/lang/invoke/LambdaForm$MH+") ||
451 ik->name()->starts_with("java/lang/invoke/LambdaForm$DMH+") ||
452 ik->name()->starts_with("java/lang/invoke/LambdaForm$BMH+") ||
453 ik->name()->starts_with("java/lang/invoke/LambdaForm$VH+"));
454 }
455
456 bool HeapShared::is_lambda_proxy_klass(InstanceKlass* ik) {
457 return ik->is_hidden() && (ik->name()->index_of_at(0, "$$Lambda+", 9) > 0);
458 }
459
754 mark_required_if_hidden_class(java_lang_Class::as_Klass(o));
755 } else if (java_lang_invoke_ResolvedMethodName::is_instance(o)) {
756 Method* m = java_lang_invoke_ResolvedMethodName::vmtarget(o);
757 if (m != nullptr) {
758 mark_required_if_hidden_class(m->method_holder());
759 }
760 }
761
762 o->oop_iterate(&c);
763 }
764 }
765 }
766
767 void HeapShared::archive_objects(ArchiveHeapInfo *heap_info) {
768 {
769 NoSafepointVerifier nsv;
770
771 // The special subgraph doesn't belong to any class. We use Object_klass() here just
772 // for convenience.
773 _dump_time_special_subgraph = init_subgraph_info(vmClasses::Object_klass(), false);
774
775 // Cache for recording where the archived objects are copied to
776 create_archived_object_cache();
777
778 if (UseCompressedOops || UseG1GC) {
779 log_info(cds)("Heap range = [" PTR_FORMAT " - " PTR_FORMAT "]",
780 UseCompressedOops ? p2i(CompressedOops::begin()) :
781 p2i((address)G1CollectedHeap::heap()->reserved().start()),
782 UseCompressedOops ? p2i(CompressedOops::end()) :
783 p2i((address)G1CollectedHeap::heap()->reserved().end()));
784 }
785 copy_objects();
786
787 CDSHeapVerifier::verify();
788 check_special_subgraph_classes();
789 }
790
791 ArchiveHeapWriter::write(_pending_roots, heap_info);
792 }
793
794 void HeapShared::copy_interned_strings() {
795 init_seen_objects_table();
796
797 auto copier = [&] (oop s, bool value_ignored) {
798 assert(s != nullptr, "sanity");
799 assert(!ArchiveHeapWriter::is_string_too_large_to_archive(s), "large strings must have been filtered");
800 bool success = archive_reachable_objects_from(1, _dump_time_special_subgraph, s);
801 assert(success, "must be");
802 // Prevent string deduplication from changing the value field to
803 // something not in the archive.
804 java_lang_String::set_deduplication_forbidden(s);
805 };
806 _dumped_interned_strings->iterate_all(copier);
807
808 delete_seen_objects_table();
809 }
810
811 void HeapShared::copy_special_subgraph() {
1132 return;
1133 }
1134 } else {
1135 assert(buffered_k->is_typeArray_klass(), "must be");
1136 // Primitive type arrays are created early during Universe::genesis.
1137 return;
1138 }
1139
1140 if (log_is_enabled(Debug, cds, heap)) {
1141 if (!_subgraph_object_klasses->contains(buffered_k)) {
1142 ResourceMark rm;
1143 log_debug(cds, heap)("Adding klass %s", orig_k->external_name());
1144 }
1145 }
1146
1147 _subgraph_object_klasses->append_if_missing(buffered_k);
1148 _has_non_early_klasses |= is_non_early_klass(orig_k);
1149 }
1150
1151 void KlassSubGraphInfo::check_allowed_klass(InstanceKlass* ik) {
1152 if (ik->module()->name() == vmSymbols::java_base()) {
1153 assert(ik->package() != nullptr, "classes in java.base cannot be in unnamed package");
1154 return;
1155 }
1156
1157 const char* lambda_msg = "";
1158 if (CDSConfig::is_dumping_invokedynamic()) {
1159 lambda_msg = ", or a lambda proxy class";
1160 if (HeapShared::is_lambda_proxy_klass(ik) &&
1161 (ik->class_loader() == nullptr ||
1162 ik->class_loader() == SystemDictionary::java_platform_loader() ||
1163 ik->class_loader() == SystemDictionary::java_system_loader())) {
1164 return;
1165 }
1166 }
1167
1168 #ifndef PRODUCT
1169 if (!ik->module()->is_named() && ik->package() == nullptr && ArchiveHeapTestClass != nullptr) {
1170 // This class is loaded by ArchiveHeapTestClass
1171 return;
1369 which, k->external_name());
1370 FlagSetting fs1(VerifyBeforeGC, true);
1371 FlagSetting fs2(VerifyDuringGC, true);
1372 FlagSetting fs3(VerifyAfterGC, true);
1373 Universe::heap()->collect(GCCause::_java_lang_system_gc);
1374 }
1375 }
1376 }
1377
1378 // Before GC can execute, we must ensure that all oops reachable from HeapShared::roots()
1379 // have a valid klass. I.e., oopDesc::klass() must have already been resolved.
1380 //
1381 // Note: if a ArchivedKlassSubGraphInfoRecord contains non-early classes, and JVMTI
1382 // ClassFileLoadHook is enabled, it's possible for this class to be dynamically replaced. In
1383 // this case, we will not load the ArchivedKlassSubGraphInfoRecord and will clear its roots.
1384 void HeapShared::resolve_classes(JavaThread* current) {
1385 assert(CDSConfig::is_using_archive(), "runtime only!");
1386 if (!ArchiveHeapLoader::is_in_use()) {
1387 return; // nothing to do
1388 }
1389 resolve_classes_for_subgraphs(current, archive_subgraph_entry_fields);
1390 resolve_classes_for_subgraphs(current, fmg_archive_subgraph_entry_fields);
1391 }
1392
1393 void HeapShared::resolve_classes_for_subgraphs(JavaThread* current, ArchivableStaticFieldInfo fields[]) {
1394 for (int i = 0; fields[i].valid(); i++) {
1395 ArchivableStaticFieldInfo* info = &fields[i];
1396 TempNewSymbol klass_name = SymbolTable::new_symbol(info->klass_name);
1397 InstanceKlass* k = SystemDictionaryShared::find_builtin_class(klass_name);
1398 assert(k != nullptr && k->is_shared_boot_class(), "sanity");
1399 resolve_classes_for_subgraph_of(current, k);
1400 }
1401 }
1402
1403 void HeapShared::resolve_classes_for_subgraph_of(JavaThread* current, Klass* k) {
1404 JavaThread* THREAD = current;
1405 ExceptionMark em(THREAD);
1406 const ArchivedKlassSubGraphInfoRecord* record =
1407 resolve_or_init_classes_for_subgraph_of(k, /*do_init=*/false, THREAD);
1408 if (HAS_PENDING_EXCEPTION) {
1737 oop referrer = (walker == nullptr) ? nullptr : walker->referencing_obj();
1738 PointsToOopsChecker points_to_oops_checker;
1739 obj->oop_iterate(&points_to_oops_checker);
1740 return CachedOopInfo(referrer, points_to_oops_checker.result());
1741 }
1742
1743 void HeapShared::init_box_classes(TRAPS) {
1744 if (ArchiveHeapLoader::is_in_use()) {
1745 vmClasses::Boolean_klass()->initialize(CHECK);
1746 vmClasses::Character_klass()->initialize(CHECK);
1747 vmClasses::Float_klass()->initialize(CHECK);
1748 vmClasses::Double_klass()->initialize(CHECK);
1749 vmClasses::Byte_klass()->initialize(CHECK);
1750 vmClasses::Short_klass()->initialize(CHECK);
1751 vmClasses::Integer_klass()->initialize(CHECK);
1752 vmClasses::Long_klass()->initialize(CHECK);
1753 vmClasses::Void_klass()->initialize(CHECK);
1754 }
1755 }
1756
1757 // (1) If orig_obj has not been archived yet, archive it.
1758 // (2) If orig_obj has not been seen yet (since start_recording_subgraph() was called),
1759 // trace all objects that are reachable from it, and make sure these objects are archived.
1760 // (3) Record the klasses of all orig_obj and all reachable objects.
1761 bool HeapShared::archive_reachable_objects_from(int level,
1762 KlassSubGraphInfo* subgraph_info,
1763 oop orig_obj) {
1764 assert(orig_obj != nullptr, "must be");
1765
1766 if (!JavaClasses::is_supported_for_archiving(orig_obj)) {
1767 // This object has injected fields that cannot be supported easily, so we disallow them for now.
1768 // If you get an error here, you probably made a change in the JDK library that has added
1769 // these objects that are referenced (directly or indirectly) by static fields.
1770 ResourceMark rm;
1771 log_error(cds, heap)("Cannot archive object " PTR_FORMAT " of class %s", p2i(orig_obj), orig_obj->klass()->external_name());
1772 debug_trace();
1773 MetaspaceShared::unrecoverable_writing_error();
1774 }
1775
1776 if (log_is_enabled(Debug, cds, heap) && java_lang_Class::is_instance(orig_obj)) {
1777 ResourceMark rm;
1778 LogTarget(Debug, cds, heap) log;
1779 LogStream out(log);
1780 out.print("Found java mirror " PTR_FORMAT " ", p2i(orig_obj));
1781 Klass* k = java_lang_Class::as_Klass(orig_obj);
1782 if (k != nullptr) {
1783 out.print("%s", k->external_name());
1784 } else {
1785 out.print("primitive");
1786 }
1787 out.print_cr("; scratch mirror = " PTR_FORMAT,
1788 p2i(scratch_java_mirror(orig_obj)));
1789 }
1790
1791 if (CDSConfig::is_initing_classes_at_dump_time()) {
1792 if (java_lang_Class::is_instance(orig_obj)) {
1793 orig_obj = scratch_java_mirror(orig_obj);
1829
1830 bool already_archived = has_been_archived(orig_obj);
1831 bool record_klasses_only = already_archived;
1832 if (!already_archived) {
1833 ++_num_new_archived_objs;
1834 if (!archive_object(orig_obj)) {
1835 // Skip archiving the sub-graph referenced from the current entry field.
1836 ResourceMark rm;
1837 log_error(cds, heap)(
1838 "Cannot archive the sub-graph referenced from %s object ("
1839 PTR_FORMAT ") size " SIZE_FORMAT ", skipped.",
1840 orig_obj->klass()->external_name(), p2i(orig_obj), orig_obj->size() * HeapWordSize);
1841 if (level == 1) {
1842 // Don't archive a subgraph root that's too big. For archives static fields, that's OK
1843 // as the Java code will take care of initializing this field dynamically.
1844 return false;
1845 } else {
1846 // We don't know how to handle an object that has been archived, but some of its reachable
1847 // objects cannot be archived. Bail out for now. We might need to fix this in the future if
1848 // we have a real use case.
1849 MetaspaceShared::unrecoverable_writing_error();
1850 }
1851 }
1852 }
1853
1854 Klass *orig_k = orig_obj->klass();
1855 subgraph_info->add_subgraph_object_klass(orig_k);
1856
1857 WalkOopAndArchiveClosure walker(level, record_klasses_only, subgraph_info, orig_obj);
1858 orig_obj->oop_iterate(&walker);
1859
1860 if (CDSConfig::is_initing_classes_at_dump_time()) {
1861 // The enum klasses are archived with aot-initialized mirror.
1862 // See AOTClassInitializer::can_archive_initialized_mirror().
1863 } else {
1864 if (CDSEnumKlass::is_enum_obj(orig_obj)) {
1865 CDSEnumKlass::handle_enum_obj(level + 1, subgraph_info, orig_obj);
1866 }
1867 }
1868
1869 return true;
2278
2279 void HeapShared::archive_object_subgraphs(ArchivableStaticFieldInfo fields[],
2280 bool is_full_module_graph) {
2281 _num_total_subgraph_recordings = 0;
2282 _num_total_walked_objs = 0;
2283 _num_total_archived_objs = 0;
2284 _num_total_recorded_klasses = 0;
2285 _num_total_verifications = 0;
2286
2287 // For each class X that has one or more archived fields:
2288 // [1] Dump the subgraph of each archived field
2289 // [2] Create a list of all the class of the objects that can be reached
2290 // by any of these static fields.
2291 // At runtime, these classes are initialized before X's archived fields
2292 // are restored by HeapShared::initialize_from_archived_subgraph().
2293 for (int i = 0; fields[i].valid(); ) {
2294 ArchivableStaticFieldInfo* info = &fields[i];
2295 const char* klass_name = info->klass_name;
2296 start_recording_subgraph(info->klass, klass_name, is_full_module_graph);
2297
2298 // If you have specified consecutive fields of the same klass in
2299 // fields[], these will be archived in the same
2300 // {start_recording_subgraph ... done_recording_subgraph} pass to
2301 // save time.
2302 for (; fields[i].valid(); i++) {
2303 ArchivableStaticFieldInfo* f = &fields[i];
2304 if (f->klass_name != klass_name) {
2305 break;
2306 }
2307
2308 archive_reachable_objects_from_static_field(f->klass, f->klass_name,
2309 f->offset, f->field_name);
2310 }
2311 done_recording_subgraph(info->klass, klass_name);
2312 }
2313
2314 log_info(cds, heap)("Archived subgraph records = %d",
2315 _num_total_subgraph_recordings);
2316 log_info(cds, heap)(" Walked %d objects", _num_total_walked_objs);
2317 log_info(cds, heap)(" Archived %d objects", _num_total_archived_objs);
2318 log_info(cds, heap)(" Recorded %d klasses", _num_total_recorded_klasses);
2319
2320 #ifndef PRODUCT
2321 for (int i = 0; fields[i].valid(); i++) {
2322 ArchivableStaticFieldInfo* f = &fields[i];
2323 verify_subgraph_from_static_field(f->klass, f->offset);
2324 }
2325 log_info(cds, heap)(" Verified %d references", _num_total_verifications);
2326 #endif
2327 }
|
1 /*
2 * Copyright (c) 2018, 2025, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
24
25 #include "precompiled.hpp"
26 #include "cds/aotClassInitializer.hpp"
27 #include "cds/archiveBuilder.hpp"
28 #include "cds/archiveHeapLoader.hpp"
29 #include "cds/archiveHeapWriter.hpp"
30 #include "cds/archiveUtils.hpp"
31 #include "cds/cdsAccess.hpp"
32 #include "cds/cdsConfig.hpp"
33 #include "cds/cdsEnumKlass.hpp"
34 #include "cds/cdsHeapVerifier.hpp"
35 #include "cds/heapShared.hpp"
36 #include "cds/metaspaceShared.hpp"
37 #include "classfile/classLoaderData.hpp"
38 #include "classfile/classLoaderExt.hpp"
39 #include "classfile/javaClasses.inline.hpp"
40 #include "classfile/modules.hpp"
41 #include "classfile/stringTable.hpp"
42 #include "classfile/symbolTable.hpp"
43 #include "classfile/systemDictionary.hpp"
44 #include "classfile/systemDictionaryShared.hpp"
45 #include "classfile/vmClasses.hpp"
46 #include "classfile/vmSymbols.hpp"
47 #include "gc/shared/collectedHeap.hpp"
48 #include "gc/shared/gcLocker.hpp"
49 #include "gc/shared/gcVMOperations.hpp"
50 #include "logging/log.hpp"
51 #include "logging/logStream.hpp"
70 #include "gc/g1/g1CollectedHeap.hpp"
71 #endif
72
73 #if INCLUDE_CDS_JAVA_HEAP
74
75 struct ArchivableStaticFieldInfo {
76 const char* klass_name;
77 const char* field_name;
78 InstanceKlass* klass;
79 int offset;
80 BasicType type;
81
82 ArchivableStaticFieldInfo(const char* k, const char* f)
83 : klass_name(k), field_name(f), klass(nullptr), offset(0), type(T_ILLEGAL) {}
84
85 bool valid() {
86 return klass_name != nullptr;
87 }
88 };
89
90 class HeapShared::ArchivingObjectMark : public StackObj {
91 public:
92 ArchivingObjectMark(oop obj) {
93 _trace->push(obj);
94 }
95 ~ArchivingObjectMark() {
96 _trace->pop();
97 }
98 };
99
100 class HeapShared::ContextMark : public StackObj {
101 ResourceMark rm;
102 public:
103 ContextMark(const char* c) : rm{} {
104 _context->push(c);
105 }
106 ~ContextMark() {
107 _context->pop();
108 }
109 };
110
111 bool HeapShared::_disable_writing = false;
112 DumpedInternedStrings *HeapShared::_dumped_interned_strings = nullptr;
113
114 size_t HeapShared::_alloc_count[HeapShared::ALLOC_STAT_SLOTS];
115 size_t HeapShared::_alloc_size[HeapShared::ALLOC_STAT_SLOTS];
116 size_t HeapShared::_total_obj_count;
117 size_t HeapShared::_total_obj_size;
118
119 #ifndef PRODUCT
120 #define ARCHIVE_TEST_FIELD_NAME "archivedObjects"
121 static Array<char>* _archived_ArchiveHeapTestClass = nullptr;
122 static const char* _test_class_name = nullptr;
123 static Klass* _test_class = nullptr;
124 static const ArchivedKlassSubGraphInfoRecord* _test_class_record = nullptr;
125 #endif
126
127
128 //
129 // If you add new entries to the following tables, you should know what you're doing!
130 //
131
132 static ArchivableStaticFieldInfo archive_subgraph_entry_fields[] = {
133 {"java/lang/Integer$IntegerCache", "archivedCache"},
134 {"java/lang/Long$LongCache", "archivedCache"},
135 {"java/lang/Byte$ByteCache", "archivedCache"},
136 {"java/lang/Short$ShortCache", "archivedCache"},
137 {"java/lang/Character$CharacterCache", "archivedCache"},
138 {"java/util/jar/Attributes$Name", "KNOWN_NAMES"},
139 {"sun/util/locale/BaseLocale", "constantBaseLocales"},
140 {"jdk/internal/module/ArchivedModuleGraph", "archivedModuleGraph"},
141 {"java/util/ImmutableCollections", "archivedObjects"},
142 {"java/lang/ModuleLayer", "EMPTY_LAYER"},
143 {"java/lang/module/Configuration", "EMPTY_CONFIGURATION"},
144 {"jdk/internal/math/FDBigInteger", "archivedCaches"},
145 {"java/lang/reflect/Proxy$ProxyBuilder", "archivedData"}, // FIXME -- requires AOTClassLinking
146
147 #ifndef PRODUCT
148 {nullptr, nullptr}, // Extra slot for -XX:ArchiveHeapTestClass
149 #endif
150 {nullptr, nullptr},
151 };
152
153 // full module graph
154 static ArchivableStaticFieldInfo fmg_archive_subgraph_entry_fields[] = {
155 {"jdk/internal/loader/ArchivedClassLoaders", "archivedClassLoaders"},
156 {ARCHIVED_BOOT_LAYER_CLASS, ARCHIVED_BOOT_LAYER_FIELD},
157 {"java/lang/Module$ArchivedData", "archivedData"},
158 {nullptr, nullptr},
159 };
160
161 KlassSubGraphInfo* HeapShared::_dump_time_special_subgraph;
162 ArchivedKlassSubGraphInfoRecord* HeapShared::_run_time_special_subgraph;
163 GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_pending_roots = nullptr;
164 GrowableArrayCHeap<oop, mtClassShared>* HeapShared::_trace = nullptr;
165 GrowableArrayCHeap<const char*, mtClassShared>* HeapShared::_context = nullptr;
166 GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_root_segments;
167 int HeapShared::_root_segment_max_size_elems;
168 OopHandle HeapShared::_scratch_basic_type_mirrors[T_VOID+1];
169 MetaspaceObjToOopHandleTable* HeapShared::_scratch_java_mirror_table = nullptr;
170 MetaspaceObjToOopHandleTable* HeapShared::_scratch_references_table = nullptr;
171
172 static bool is_subgraph_root_class_of(ArchivableStaticFieldInfo fields[], InstanceKlass* ik) {
173 for (int i = 0; fields[i].valid(); i++) {
174 if (fields[i].klass == ik) {
175 return true;
176 }
177 }
178 return false;
179 }
180
181 bool HeapShared::is_subgraph_root_class(InstanceKlass* ik) {
182 return is_subgraph_root_class_of(archive_subgraph_entry_fields, ik) ||
183 is_subgraph_root_class_of(fmg_archive_subgraph_entry_fields, ik);
184 }
185
233 vmSymbols::void_BuiltinClassLoader_signature(),
234 CHECK);
235 Handle boot_loader(THREAD, result.get_oop());
236 reset_states(boot_loader(), CHECK);
237 }
238
239 HeapShared::ArchivedObjectCache* HeapShared::_archived_object_cache = nullptr;
240
241 bool HeapShared::has_been_archived(oop obj) {
242 assert(CDSConfig::is_dumping_heap(), "dump-time only");
243 return archived_object_cache()->get(obj) != nullptr;
244 }
245
246 int HeapShared::append_root(oop obj) {
247 assert(CDSConfig::is_dumping_heap(), "dump-time only");
248
249 // No GC should happen since we aren't scanning _pending_roots.
250 assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
251
252 if (_pending_roots == nullptr) {
253 _pending_roots = new GrowableArrayCHeap<OopHandle, mtClassShared>(500);
254 }
255
256 OopHandle oh(Universe::vm_global(), obj);
257 return _pending_roots->append(oh);
258 }
259
260 objArrayOop HeapShared::root_segment(int segment_idx) {
261 if (CDSConfig::is_dumping_heap() && !CDSConfig::is_dumping_final_static_archive()) {
262 assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
263 if (!HeapShared::can_write()) {
264 return nullptr;
265 }
266 } else {
267 assert(CDSConfig::is_using_archive(), "must be");
268 }
269
270 objArrayOop segment = (objArrayOop)_root_segments->at(segment_idx).resolve();
271 assert(segment != nullptr, "should have been initialized");
272 return segment;
273 }
274
275 inline unsigned int oop_handle_hash(const OopHandle& oh) {
276 oop o = oh.resolve();
277 if (o == nullptr) {
278 return 0;
279 } else {
280 return o->identity_hash();
281 }
282 }
283
284 inline bool oop_handle_equals(const OopHandle& a, const OopHandle& b) {
285 return a.resolve() == b.resolve();
286 }
287
288 class OrigToScratchObjectTable: public ResourceHashtable<OopHandle, OopHandle,
289 36137, // prime number
290 AnyObj::C_HEAP,
291 mtClassShared,
292 oop_handle_hash,
293 oop_handle_equals> {};
294
295 static OrigToScratchObjectTable* _orig_to_scratch_object_table = nullptr;
296
297 void HeapShared::track_scratch_object(oop orig_obj, oop scratch_obj) {
298 MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
299 if (_orig_to_scratch_object_table == nullptr) {
300 _orig_to_scratch_object_table = new (mtClass)OrigToScratchObjectTable();
301 }
302
303 OopHandle orig_h(Universe::vm_global(), orig_obj);
304 OopHandle scratch_h(Universe::vm_global(), scratch_obj);
305 _orig_to_scratch_object_table->put_when_absent(orig_h, scratch_h);
306 }
307
308 oop HeapShared::orig_to_scratch_object(oop orig_obj) {
309 MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
310 if (_orig_to_scratch_object_table != nullptr) {
311 OopHandle orig(&orig_obj);
312 OopHandle* v = _orig_to_scratch_object_table->get(orig);
313 if (v != nullptr) {
314 return v->resolve();
315 }
316 }
317 return nullptr;
318 }
319
320 // Permanent oops are used to support AOT-compiled methods, which may have in-line references
321 // to Strings and MH oops.
322 //
323 // At runtime, these oops are stored in _runtime_permanent_oops (which keeps them alive forever)
324 // and are accssed vis CDSAccess::get_archived_object(int).
325 struct PermanentOopInfo {
326 int _index; // Gets assigned only if HeapShared::get_archived_object_permanent_index() has been called on the object
327 int _heap_offset; // Offset of the object from the bottom of the archived heap.
328 PermanentOopInfo(int index, int heap_offset) : _index(index), _heap_offset(heap_offset) {}
329 };
330
331 class PermanentOopTable: public ResourceHashtable<OopHandle, PermanentOopInfo,
332 36137, // prime number
333 AnyObj::C_HEAP,
334 mtClassShared,
335 oop_handle_hash,
336 oop_handle_equals> {};
337
338 static int _dumptime_permanent_oop_count = 0;
339 static PermanentOopTable* _dumptime_permanent_oop_table = nullptr;
340 static GrowableArrayCHeap<OopHandle, mtClassShared>* _runtime_permanent_oops = nullptr;
341
342 // ArchiveHeapWriter adds each archived heap object to _dumptime_permanent_oop_table,
343 // so we can remember their offset (from the bottom of the archived heap).
344 void HeapShared::add_to_permanent_oop_table(oop obj, int offset) {
345 assert_at_safepoint();
346 if (_dumptime_permanent_oop_table == nullptr) {
347 _dumptime_permanent_oop_table = new (mtClass)PermanentOopTable();
348 }
349
350 PermanentOopInfo info(-1, offset);
351 OopHandle oh(Universe::vm_global(), obj);
352 _dumptime_permanent_oop_table->put_when_absent(oh, info);
353 }
354
355 // A permanent index is assigned to an archived object ONLY when
356 // the AOT compiler calls this function.
357 int HeapShared::get_archived_object_permanent_index(oop obj) {
358 MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
359
360 if (!CDSConfig::is_dumping_heap()) {
361 return -1; // Called by the Leyden old workflow
362 }
363 if (_dumptime_permanent_oop_table == nullptr) {
364 return -1;
365 }
366
367 if (_orig_to_scratch_object_table != nullptr) {
368 OopHandle orig(&obj);
369 OopHandle* v = _orig_to_scratch_object_table->get(orig);
370 if (v != nullptr) {
371 obj = v->resolve();
372 }
373 }
374
375 OopHandle tmp(&obj);
376 PermanentOopInfo* info = _dumptime_permanent_oop_table->get(tmp);
377 if (info == nullptr) {
378 return -1;
379 } else {
380 if (info->_index < 0) {
381 info->_index = _dumptime_permanent_oop_count++;
382 }
383 return info->_index;
384 }
385 }
386
387 oop HeapShared::get_archived_object(int permanent_index) {
388 assert(permanent_index >= 0, "sanity");
389 assert(ArchiveHeapLoader::is_in_use(), "sanity");
390 assert(_runtime_permanent_oops != nullptr, "sanity");
391
392 return _runtime_permanent_oops->at(permanent_index).resolve();
393 }
394
395 // Remember all archived heap objects that have a permanent index.
396 // table[i] = offset of oop whose permanent index is i.
397 void CachedCodeDirectoryInternal::dumptime_init_internal() {
398 const int count = _dumptime_permanent_oop_count;
399 int* table = (int*)CDSAccess::allocate_from_code_cache(count * sizeof(int));
400 for (int i = 0; i < count; i++) {
401 table[count] = -1;
402 }
403 _dumptime_permanent_oop_table->iterate([&](OopHandle o, PermanentOopInfo& info) {
404 int index = info._index;
405 if (index >= 0) {
406 assert(index < count, "sanity");
407 table[index] = info._heap_offset;
408 }
409 return true; // continue
410 });
411
412 for (int i = 0; i < count; i++) {
413 assert(table[i] >= 0, "must be");
414 }
415
416 log_info(cds)("Dumped %d permanent oops", count);
417
418 _permanent_oop_count = count;
419 CDSAccess::set_pointer(&_permanent_oop_offsets, table);
420 }
421
422 // This is called during the bootstrap of the production run, before any GC can happen.
423 // Record each permanent oop in a OopHandle for GC safety.
424 void CachedCodeDirectoryInternal::runtime_init_internal() {
425 int count = _permanent_oop_count;
426 int* table = _permanent_oop_offsets;
427 _runtime_permanent_oops = new GrowableArrayCHeap<OopHandle, mtClassShared>();
428 for (int i = 0; i < count; i++) {
429 oop obj = ArchiveHeapLoader::oop_from_offset(table[i]);
430 OopHandle oh(Universe::vm_global(), obj);
431 _runtime_permanent_oops->append(oh);
432 }
433 };
434
435 void HeapShared::get_segment_indexes(int idx, int& seg_idx, int& int_idx) {
436 assert(_root_segment_max_size_elems > 0, "sanity");
437
438 // Try to avoid divisions for the common case.
439 if (idx < _root_segment_max_size_elems) {
440 seg_idx = 0;
441 int_idx = idx;
442 } else {
443 seg_idx = idx / _root_segment_max_size_elems;
444 int_idx = idx % _root_segment_max_size_elems;
445 }
446
447 assert(idx == seg_idx * _root_segment_max_size_elems + int_idx,
448 "sanity: %d index maps to %d segment and %d internal", idx, seg_idx, int_idx);
449 }
450
451 // Returns an objArray that contains all the roots of the archived objects
452 oop HeapShared::get_root(int index, bool clear) {
453 assert(index >= 0, "sanity");
454 assert(!CDSConfig::is_dumping_heap() && CDSConfig::is_using_archive(), "runtime only");
536 return nullptr;
537 }
538 }
539 void set_oop(MetaspaceObj* ptr, oop o) {
540 MutexLocker ml(ScratchObjects_lock, Mutex::_no_safepoint_check_flag);
541 OopHandle handle(Universe::vm_global(), o);
542 bool is_new = put(ptr, handle);
543 assert(is_new, "cannot set twice");
544 }
545 void remove_oop(MetaspaceObj* ptr) {
546 MutexLocker ml(ScratchObjects_lock, Mutex::_no_safepoint_check_flag);
547 OopHandle* handle = get(ptr);
548 if (handle != nullptr) {
549 handle->release(Universe::vm_global());
550 remove(ptr);
551 }
552 }
553 };
554
555 void HeapShared::add_scratch_resolved_references(ConstantPool* src, objArrayOop dest) {
556 if (_scratch_references_table == nullptr) {
557 _scratch_references_table = new (mtClass)MetaspaceObjToOopHandleTable();
558 }
559 _scratch_references_table->set_oop(src, dest);
560 }
561
562 objArrayOop HeapShared::scratch_resolved_references(ConstantPool* src) {
563 return (objArrayOop)_scratch_references_table->get_oop(src);
564 }
565
566 void HeapShared::init_scratch_objects(TRAPS) {
567 for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
568 BasicType bt = (BasicType)i;
569 if (!is_reference_type(bt)) {
570 oop m = java_lang_Class::create_basic_type_mirror(type2name(bt), bt, CHECK);
571 _scratch_basic_type_mirrors[i] = OopHandle(Universe::vm_global(), m);
572 track_scratch_object(Universe::java_mirror(bt), m);
573 }
574 }
575 _scratch_java_mirror_table = new (mtClass)MetaspaceObjToOopHandleTable();
576 if (_scratch_references_table == nullptr) {
577 _scratch_references_table = new (mtClass)MetaspaceObjToOopHandleTable();
578 }
579 }
580
581 // Given java_mirror that represents a (primitive or reference) type T,
582 // return the "scratch" version that represents the same type T.
583 // Note that if java_mirror will be returned if it's already a
584 // scratch mirror.
585 //
586 // See java_lang_Class::create_scratch_mirror() for more info.
587 oop HeapShared::scratch_java_mirror(oop java_mirror) {
588 assert(java_lang_Class::is_instance(java_mirror), "must be");
589
590 for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
591 BasicType bt = (BasicType)i;
592 if (!is_reference_type(bt)) {
593 if (_scratch_basic_type_mirrors[i].resolve() == java_mirror) {
594 return java_mirror;
595 }
596 }
597 }
598
599 if (java_lang_Class::is_primitive(java_mirror)) {
600 return scratch_java_mirror(java_lang_Class::as_BasicType(java_mirror));
601 } else {
602 return scratch_java_mirror(java_lang_Class::as_Klass(java_mirror));
603 }
604 }
605
606 oop HeapShared::scratch_java_mirror(BasicType t) {
607 assert((uint)t < T_VOID+1, "range check");
608 assert(!is_reference_type(t), "sanity");
609 return _scratch_basic_type_mirrors[t].resolve();
610 }
611
612 oop HeapShared::scratch_java_mirror(Klass* k) {
613 return _scratch_java_mirror_table->get_oop(k);
614 }
615
616 void HeapShared::set_scratch_java_mirror(Klass* k, oop mirror) {
617 track_scratch_object(k->java_mirror(), mirror);
618 _scratch_java_mirror_table->set_oop(k, mirror);
619 }
620
621 void HeapShared::remove_scratch_objects(Klass* k) {
622 // Klass is being deallocated. Java mirror can still be alive, and it should not
623 // point to dead klass. We need to break the link from mirror to the Klass.
624 // See how InstanceKlass::deallocate_contents does it for normal mirrors.
625 oop mirror = _scratch_java_mirror_table->get_oop(k);
626 if (mirror != nullptr) {
627 java_lang_Class::set_klass(mirror, nullptr);
628 }
629 _scratch_java_mirror_table->remove_oop(k);
630 if (k->is_instance_klass()) {
631 _scratch_references_table->remove(InstanceKlass::cast(k)->constants());
632 }
633 if (mirror != nullptr) {
634 OopHandle tmp(&mirror);
635 OopHandle* v = _orig_to_scratch_object_table->get(tmp);
636 if (v != nullptr) {
637 oop scratch_mirror = v->resolve();
638 java_lang_Class::set_klass(scratch_mirror, nullptr);
639 _orig_to_scratch_object_table->remove(tmp);
640 }
641 }
642 }
643
644 //TODO: we eventually want a more direct test for these kinds of things.
645 //For example the JVM could record some bit of context from the creation
646 //of the klass, such as who called the hidden class factory. Using
647 //string compares on names is fragile and will break as soon as somebody
648 //changes the names in the JDK code. See discussion in JDK-8342481 for
649 //related ideas about marking AOT-related classes.
650 bool HeapShared::is_lambda_form_klass(InstanceKlass* ik) {
651 return ik->is_hidden() &&
652 (ik->name()->starts_with("java/lang/invoke/LambdaForm$MH+") ||
653 ik->name()->starts_with("java/lang/invoke/LambdaForm$DMH+") ||
654 ik->name()->starts_with("java/lang/invoke/LambdaForm$BMH+") ||
655 ik->name()->starts_with("java/lang/invoke/LambdaForm$VH+"));
656 }
657
658 bool HeapShared::is_lambda_proxy_klass(InstanceKlass* ik) {
659 return ik->is_hidden() && (ik->name()->index_of_at(0, "$$Lambda+", 9) > 0);
660 }
661
956 mark_required_if_hidden_class(java_lang_Class::as_Klass(o));
957 } else if (java_lang_invoke_ResolvedMethodName::is_instance(o)) {
958 Method* m = java_lang_invoke_ResolvedMethodName::vmtarget(o);
959 if (m != nullptr) {
960 mark_required_if_hidden_class(m->method_holder());
961 }
962 }
963
964 o->oop_iterate(&c);
965 }
966 }
967 }
968
969 void HeapShared::archive_objects(ArchiveHeapInfo *heap_info) {
970 {
971 NoSafepointVerifier nsv;
972
973 // The special subgraph doesn't belong to any class. We use Object_klass() here just
974 // for convenience.
975 _dump_time_special_subgraph = init_subgraph_info(vmClasses::Object_klass(), false);
976 _trace = new GrowableArrayCHeap<oop, mtClassShared>(250);
977 _context = new GrowableArrayCHeap<const char*, mtClassShared>(250);
978
979 // Cache for recording where the archived objects are copied to
980 create_archived_object_cache();
981
982 if (UseCompressedOops || UseG1GC) {
983 log_info(cds)("Heap range = [" PTR_FORMAT " - " PTR_FORMAT "]",
984 UseCompressedOops ? p2i(CompressedOops::begin()) :
985 p2i((address)G1CollectedHeap::heap()->reserved().start()),
986 UseCompressedOops ? p2i(CompressedOops::end()) :
987 p2i((address)G1CollectedHeap::heap()->reserved().end()));
988 }
989 copy_objects();
990
991 if (!SkipArchiveHeapVerification) {
992 CDSHeapVerifier::verify();
993 }
994 check_special_subgraph_classes();
995 }
996
997 GrowableArrayCHeap<oop, mtClassShared>* roots = new GrowableArrayCHeap<oop, mtClassShared>(_pending_roots->length());
998 for (int i = 0; i < _pending_roots->length(); i++) {
999 roots->append(_pending_roots->at(i).resolve());
1000 }
1001 ArchiveHeapWriter::write(roots, heap_info);
1002 }
1003
1004 void HeapShared::copy_interned_strings() {
1005 init_seen_objects_table();
1006
1007 auto copier = [&] (oop s, bool value_ignored) {
1008 assert(s != nullptr, "sanity");
1009 assert(!ArchiveHeapWriter::is_string_too_large_to_archive(s), "large strings must have been filtered");
1010 bool success = archive_reachable_objects_from(1, _dump_time_special_subgraph, s);
1011 assert(success, "must be");
1012 // Prevent string deduplication from changing the value field to
1013 // something not in the archive.
1014 java_lang_String::set_deduplication_forbidden(s);
1015 };
1016 _dumped_interned_strings->iterate_all(copier);
1017
1018 delete_seen_objects_table();
1019 }
1020
1021 void HeapShared::copy_special_subgraph() {
1342 return;
1343 }
1344 } else {
1345 assert(buffered_k->is_typeArray_klass(), "must be");
1346 // Primitive type arrays are created early during Universe::genesis.
1347 return;
1348 }
1349
1350 if (log_is_enabled(Debug, cds, heap)) {
1351 if (!_subgraph_object_klasses->contains(buffered_k)) {
1352 ResourceMark rm;
1353 log_debug(cds, heap)("Adding klass %s", orig_k->external_name());
1354 }
1355 }
1356
1357 _subgraph_object_klasses->append_if_missing(buffered_k);
1358 _has_non_early_klasses |= is_non_early_klass(orig_k);
1359 }
1360
1361 void KlassSubGraphInfo::check_allowed_klass(InstanceKlass* ik) {
1362 if (CDSConfig::is_dumping_invokedynamic()) {
1363 // FIXME -- this allows LambdaProxy classes
1364 return;
1365 }
1366 if (ik->module()->name() == vmSymbols::java_base()) {
1367 assert(ik->package() != nullptr, "classes in java.base cannot be in unnamed package");
1368 return;
1369 }
1370
1371 const char* lambda_msg = "";
1372 if (CDSConfig::is_dumping_invokedynamic()) {
1373 lambda_msg = ", or a lambda proxy class";
1374 if (HeapShared::is_lambda_proxy_klass(ik) &&
1375 (ik->class_loader() == nullptr ||
1376 ik->class_loader() == SystemDictionary::java_platform_loader() ||
1377 ik->class_loader() == SystemDictionary::java_system_loader())) {
1378 return;
1379 }
1380 }
1381
1382 #ifndef PRODUCT
1383 if (!ik->module()->is_named() && ik->package() == nullptr && ArchiveHeapTestClass != nullptr) {
1384 // This class is loaded by ArchiveHeapTestClass
1385 return;
1583 which, k->external_name());
1584 FlagSetting fs1(VerifyBeforeGC, true);
1585 FlagSetting fs2(VerifyDuringGC, true);
1586 FlagSetting fs3(VerifyAfterGC, true);
1587 Universe::heap()->collect(GCCause::_java_lang_system_gc);
1588 }
1589 }
1590 }
1591
1592 // Before GC can execute, we must ensure that all oops reachable from HeapShared::roots()
1593 // have a valid klass. I.e., oopDesc::klass() must have already been resolved.
1594 //
1595 // Note: if a ArchivedKlassSubGraphInfoRecord contains non-early classes, and JVMTI
1596 // ClassFileLoadHook is enabled, it's possible for this class to be dynamically replaced. In
1597 // this case, we will not load the ArchivedKlassSubGraphInfoRecord and will clear its roots.
1598 void HeapShared::resolve_classes(JavaThread* current) {
1599 assert(CDSConfig::is_using_archive(), "runtime only!");
1600 if (!ArchiveHeapLoader::is_in_use()) {
1601 return; // nothing to do
1602 }
1603
1604 if (!CDSConfig::is_using_aot_linked_classes()) {
1605 assert( _run_time_special_subgraph != nullptr, "must be");
1606 Array<Klass*>* klasses = _run_time_special_subgraph->subgraph_object_klasses();
1607 if (klasses != nullptr) {
1608 for (int i = 0; i < klasses->length(); i++) {
1609 Klass* k = klasses->at(i);
1610 ExceptionMark em(current); // no exception can happen here
1611 resolve_or_init(k, /*do_init*/false, current);
1612 }
1613 }
1614 }
1615
1616 resolve_classes_for_subgraphs(current, archive_subgraph_entry_fields);
1617 resolve_classes_for_subgraphs(current, fmg_archive_subgraph_entry_fields);
1618 }
1619
1620 void HeapShared::resolve_classes_for_subgraphs(JavaThread* current, ArchivableStaticFieldInfo fields[]) {
1621 for (int i = 0; fields[i].valid(); i++) {
1622 ArchivableStaticFieldInfo* info = &fields[i];
1623 TempNewSymbol klass_name = SymbolTable::new_symbol(info->klass_name);
1624 InstanceKlass* k = SystemDictionaryShared::find_builtin_class(klass_name);
1625 assert(k != nullptr && k->is_shared_boot_class(), "sanity");
1626 resolve_classes_for_subgraph_of(current, k);
1627 }
1628 }
1629
1630 void HeapShared::resolve_classes_for_subgraph_of(JavaThread* current, Klass* k) {
1631 JavaThread* THREAD = current;
1632 ExceptionMark em(THREAD);
1633 const ArchivedKlassSubGraphInfoRecord* record =
1634 resolve_or_init_classes_for_subgraph_of(k, /*do_init=*/false, THREAD);
1635 if (HAS_PENDING_EXCEPTION) {
1964 oop referrer = (walker == nullptr) ? nullptr : walker->referencing_obj();
1965 PointsToOopsChecker points_to_oops_checker;
1966 obj->oop_iterate(&points_to_oops_checker);
1967 return CachedOopInfo(referrer, points_to_oops_checker.result());
1968 }
1969
1970 void HeapShared::init_box_classes(TRAPS) {
1971 if (ArchiveHeapLoader::is_in_use()) {
1972 vmClasses::Boolean_klass()->initialize(CHECK);
1973 vmClasses::Character_klass()->initialize(CHECK);
1974 vmClasses::Float_klass()->initialize(CHECK);
1975 vmClasses::Double_klass()->initialize(CHECK);
1976 vmClasses::Byte_klass()->initialize(CHECK);
1977 vmClasses::Short_klass()->initialize(CHECK);
1978 vmClasses::Integer_klass()->initialize(CHECK);
1979 vmClasses::Long_klass()->initialize(CHECK);
1980 vmClasses::Void_klass()->initialize(CHECK);
1981 }
1982 }
1983
1984 void HeapShared::exit_on_error() {
1985 if (_context != nullptr) {
1986 ResourceMark rm;
1987 LogStream ls(Log(cds, heap)::error());
1988 ls.print_cr("Context");
1989 for (int i = 0; i < _context->length(); i++) {
1990 const char* s = _context->at(i);
1991 ls.print_cr("- %s", s);
1992 }
1993 }
1994 if (_trace != nullptr) {
1995 ResourceMark rm;
1996 LogStream ls(Log(cds, heap)::error());
1997 ls.print_cr("Reference trace");
1998 for (int i = 0; i < _trace->length(); i++) {
1999 oop orig_obj = _trace->at(i);
2000 ls.print_cr("[%d] ========================================", i);
2001 orig_obj->print_on(&ls);
2002 ls.cr();
2003 }
2004 }
2005 MetaspaceShared::unrecoverable_writing_error();
2006 }
2007
2008 // (1) If orig_obj has not been archived yet, archive it.
2009 // (2) If orig_obj has not been seen yet (since start_recording_subgraph() was called),
2010 // trace all objects that are reachable from it, and make sure these objects are archived.
2011 // (3) Record the klasses of all orig_obj and all reachable objects.
2012 bool HeapShared::archive_reachable_objects_from(int level,
2013 KlassSubGraphInfo* subgraph_info,
2014 oop orig_obj) {
2015 ArchivingObjectMark mark(orig_obj);
2016 assert(orig_obj != nullptr, "must be");
2017
2018 if (!JavaClasses::is_supported_for_archiving(orig_obj)) {
2019 // This object has injected fields that cannot be supported easily, so we disallow them for now.
2020 // If you get an error here, you probably made a change in the JDK library that has added
2021 // these objects that are referenced (directly or indirectly) by static fields.
2022 ResourceMark rm;
2023 log_error(cds, heap)("Cannot archive object " PTR_FORMAT " of class %s", p2i(orig_obj), orig_obj->klass()->external_name());
2024 debug_trace();
2025 exit_on_error();
2026 }
2027
2028 if (log_is_enabled(Debug, cds, heap) && java_lang_Class::is_instance(orig_obj)) {
2029 ResourceMark rm;
2030 LogTarget(Debug, cds, heap) log;
2031 LogStream out(log);
2032 out.print("Found java mirror " PTR_FORMAT " ", p2i(orig_obj));
2033 Klass* k = java_lang_Class::as_Klass(orig_obj);
2034 if (k != nullptr) {
2035 out.print("%s", k->external_name());
2036 } else {
2037 out.print("primitive");
2038 }
2039 out.print_cr("; scratch mirror = " PTR_FORMAT,
2040 p2i(scratch_java_mirror(orig_obj)));
2041 }
2042
2043 if (CDSConfig::is_initing_classes_at_dump_time()) {
2044 if (java_lang_Class::is_instance(orig_obj)) {
2045 orig_obj = scratch_java_mirror(orig_obj);
2081
2082 bool already_archived = has_been_archived(orig_obj);
2083 bool record_klasses_only = already_archived;
2084 if (!already_archived) {
2085 ++_num_new_archived_objs;
2086 if (!archive_object(orig_obj)) {
2087 // Skip archiving the sub-graph referenced from the current entry field.
2088 ResourceMark rm;
2089 log_error(cds, heap)(
2090 "Cannot archive the sub-graph referenced from %s object ("
2091 PTR_FORMAT ") size " SIZE_FORMAT ", skipped.",
2092 orig_obj->klass()->external_name(), p2i(orig_obj), orig_obj->size() * HeapWordSize);
2093 if (level == 1) {
2094 // Don't archive a subgraph root that's too big. For archives static fields, that's OK
2095 // as the Java code will take care of initializing this field dynamically.
2096 return false;
2097 } else {
2098 // We don't know how to handle an object that has been archived, but some of its reachable
2099 // objects cannot be archived. Bail out for now. We might need to fix this in the future if
2100 // we have a real use case.
2101 exit_on_error();
2102 }
2103 }
2104 }
2105
2106 Klass *orig_k = orig_obj->klass();
2107 subgraph_info->add_subgraph_object_klass(orig_k);
2108
2109 WalkOopAndArchiveClosure walker(level, record_klasses_only, subgraph_info, orig_obj);
2110 orig_obj->oop_iterate(&walker);
2111
2112 if (CDSConfig::is_initing_classes_at_dump_time()) {
2113 // The enum klasses are archived with aot-initialized mirror.
2114 // See AOTClassInitializer::can_archive_initialized_mirror().
2115 } else {
2116 if (CDSEnumKlass::is_enum_obj(orig_obj)) {
2117 CDSEnumKlass::handle_enum_obj(level + 1, subgraph_info, orig_obj);
2118 }
2119 }
2120
2121 return true;
2530
2531 void HeapShared::archive_object_subgraphs(ArchivableStaticFieldInfo fields[],
2532 bool is_full_module_graph) {
2533 _num_total_subgraph_recordings = 0;
2534 _num_total_walked_objs = 0;
2535 _num_total_archived_objs = 0;
2536 _num_total_recorded_klasses = 0;
2537 _num_total_verifications = 0;
2538
2539 // For each class X that has one or more archived fields:
2540 // [1] Dump the subgraph of each archived field
2541 // [2] Create a list of all the class of the objects that can be reached
2542 // by any of these static fields.
2543 // At runtime, these classes are initialized before X's archived fields
2544 // are restored by HeapShared::initialize_from_archived_subgraph().
2545 for (int i = 0; fields[i].valid(); ) {
2546 ArchivableStaticFieldInfo* info = &fields[i];
2547 const char* klass_name = info->klass_name;
2548 start_recording_subgraph(info->klass, klass_name, is_full_module_graph);
2549
2550 ContextMark cm(klass_name);
2551 // If you have specified consecutive fields of the same klass in
2552 // fields[], these will be archived in the same
2553 // {start_recording_subgraph ... done_recording_subgraph} pass to
2554 // save time.
2555 for (; fields[i].valid(); i++) {
2556 ArchivableStaticFieldInfo* f = &fields[i];
2557 if (f->klass_name != klass_name) {
2558 break;
2559 }
2560
2561 ContextMark cm(f->field_name);
2562 archive_reachable_objects_from_static_field(f->klass, f->klass_name,
2563 f->offset, f->field_name);
2564 }
2565 done_recording_subgraph(info->klass, klass_name);
2566 }
2567
2568 log_info(cds, heap)("Archived subgraph records = %d",
2569 _num_total_subgraph_recordings);
2570 log_info(cds, heap)(" Walked %d objects", _num_total_walked_objs);
2571 log_info(cds, heap)(" Archived %d objects", _num_total_archived_objs);
2572 log_info(cds, heap)(" Recorded %d klasses", _num_total_recorded_klasses);
2573
2574 #ifndef PRODUCT
2575 for (int i = 0; fields[i].valid(); i++) {
2576 ArchivableStaticFieldInfo* f = &fields[i];
2577 verify_subgraph_from_static_field(f->klass, f->offset);
2578 }
2579 log_info(cds, heap)(" Verified %d references", _num_total_verifications);
2580 #endif
2581 }
|