Skip to content

Commit d3fc8df

Browse files
committedApr 2, 2024
8329135: Store Universe::*exception_instance() in CDS archive
Reviewed-by: vlivanov, ccheung
1 parent a85c849 commit d3fc8df

File tree

5 files changed

+90
-35
lines changed

5 files changed

+90
-35
lines changed
 

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

+16-5
Original file line numberDiff line numberDiff line change
@@ -430,6 +430,12 @@ void HeapShared::archive_strings() {
430430
StringTable::set_shared_strings_array_index(append_root(shared_strings_array));
431431
}
432432

433+
int HeapShared::archive_exception_instance(oop exception) {
434+
bool success = archive_reachable_objects_from(1, _default_subgraph_info, exception);
435+
assert(success, "sanity");
436+
return append_root(exception);
437+
}
438+
433439
void HeapShared::mark_native_pointers(oop orig_obj) {
434440
if (java_lang_Class::is_instance(orig_obj)) {
435441
ArchiveHeapWriter::mark_native_pointer(orig_obj, java_lang_Class::klass_offset());
@@ -589,6 +595,7 @@ void HeapShared::copy_special_objects() {
589595
init_seen_objects_table();
590596
archive_java_mirrors();
591597
archive_strings();
598+
Universe::archive_exception_instances();
592599
delete_seen_objects_table();
593600
}
594601

@@ -1387,11 +1394,15 @@ void HeapShared::check_default_subgraph_classes() {
13871394
i, subgraph_k->external_name());
13881395
}
13891396

1390-
guarantee(subgraph_k->name()->equals("java/lang/Class") ||
1391-
subgraph_k->name()->equals("java/lang/String") ||
1392-
subgraph_k->name()->equals("[Ljava/lang/Object;") ||
1393-
subgraph_k->name()->equals("[C") ||
1394-
subgraph_k->name()->equals("[B"),
1397+
Symbol* name = ArchiveBuilder::current()->get_source_addr(subgraph_k->name());
1398+
guarantee(name == vmSymbols::java_lang_Class() ||
1399+
name == vmSymbols::java_lang_String() ||
1400+
name == vmSymbols::java_lang_ArithmeticException() ||
1401+
name == vmSymbols::java_lang_NullPointerException() ||
1402+
name == vmSymbols::java_lang_VirtualMachineError() ||
1403+
name == vmSymbols::object_array_signature() ||
1404+
name == vmSymbols::byte_array_signature() ||
1405+
name == vmSymbols::char_array_signature(),
13951406
"default subgraph can have only these objects");
13961407
}
13971408
}

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

+1
Original file line numberDiff line numberDiff line change
@@ -365,6 +365,7 @@ class HeapShared: AllStatic {
365365
return _archived_object_cache;
366366
}
367367

368+
static int archive_exception_instance(oop exception);
368369
static void archive_objects(ArchiveHeapInfo* heap_info);
369370
static void copy_objects();
370371
static void copy_special_objects();

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

+1-2
Original file line numberDiff line numberDiff line change
@@ -1466,8 +1466,7 @@ void MetaspaceShared::initialize_shared_spaces() {
14661466
// done after ReadClosure.
14671467
static_mapinfo->patch_heap_embedded_pointers();
14681468
ArchiveHeapLoader::finish_initialization();
1469-
1470-
CDS_JAVA_HEAP_ONLY(Universe::update_archived_basic_type_mirrors());
1469+
Universe::load_archived_object_instances();
14711470

14721471
// Close the mapinfo file
14731472
static_mapinfo->close();

‎src/hotspot/share/memory/universe.cpp

+70-23
Original file line numberDiff line numberDiff line change
@@ -149,10 +149,6 @@ volatile jint Universe::_preallocated_out_of_memory_error_avail_count = 0;
149149
OopHandle Universe::_msg_metaspace;
150150
OopHandle Universe::_msg_class_metaspace;
151151

152-
OopHandle Universe::_null_ptr_exception_instance;
153-
OopHandle Universe::_arithmetic_exception_instance;
154-
OopHandle Universe::_virtual_machine_error_instance;
155-
156152
OopHandle Universe::_reference_pending_list;
157153

158154
Array<Klass*>* Universe::_the_array_interfaces_array = nullptr;
@@ -186,6 +182,52 @@ OopStorage* Universe::_vm_global = nullptr;
186182

187183
CollectedHeap* Universe::_collectedHeap = nullptr;
188184

185+
// These are the exceptions that are always created and are guatanteed to exist.
186+
// If possible, they can be stored as CDS archived objects to speed up AOT code.
187+
class BuiltinException {
188+
OopHandle _instance;
189+
CDS_JAVA_HEAP_ONLY(int _archived_root_index;)
190+
191+
public:
192+
BuiltinException() : _instance() {
193+
CDS_JAVA_HEAP_ONLY(_archived_root_index = 0);
194+
}
195+
196+
void init_if_empty(Symbol* symbol, TRAPS) {
197+
if (_instance.is_empty()) {
198+
Klass* k = SystemDictionary::resolve_or_fail(symbol, true, CHECK);
199+
oop obj = InstanceKlass::cast(k)->allocate_instance(CHECK);
200+
_instance = OopHandle(Universe::vm_global(), obj);
201+
}
202+
}
203+
204+
oop instance() {
205+
return _instance.resolve();
206+
}
207+
208+
#if INCLUDE_CDS_JAVA_HEAP
209+
void store_in_cds() {
210+
_archived_root_index = HeapShared::archive_exception_instance(instance());
211+
}
212+
213+
void load_from_cds() {
214+
if (_archived_root_index >= 0) {
215+
oop obj = HeapShared::get_root(_archived_root_index);
216+
assert(obj != nullptr, "must be");
217+
_instance = OopHandle(Universe::vm_global(), obj);
218+
}
219+
}
220+
221+
void serialize(SerializeClosure *f) {
222+
f->do_int(&_archived_root_index);
223+
}
224+
#endif
225+
};
226+
227+
static BuiltinException _null_ptr_exception;
228+
static BuiltinException _arithmetic_exception;
229+
static BuiltinException _virtual_machine_error;
230+
189231
objArrayOop Universe::the_empty_class_array () {
190232
return (objArrayOop)_the_empty_class_array.resolve();
191233
}
@@ -199,9 +241,9 @@ void Universe::set_system_thread_group(oop group) { _system_thread_group = OopHa
199241
oop Universe::the_null_string() { return _the_null_string.resolve(); }
200242
oop Universe::the_min_jint_string() { return _the_min_jint_string.resolve(); }
201243

202-
oop Universe::null_ptr_exception_instance() { return _null_ptr_exception_instance.resolve(); }
203-
oop Universe::arithmetic_exception_instance() { return _arithmetic_exception_instance.resolve(); }
204-
oop Universe::virtual_machine_error_instance() { return _virtual_machine_error_instance.resolve(); }
244+
oop Universe::null_ptr_exception_instance() { return _null_ptr_exception.instance(); }
245+
oop Universe::arithmetic_exception_instance() { return _arithmetic_exception.instance(); }
246+
oop Universe::virtual_machine_error_instance() { return _virtual_machine_error.instance(); }
205247

206248
oop Universe::the_null_sentinel() { return _the_null_sentinel.resolve(); }
207249

@@ -254,7 +296,13 @@ void Universe::set_archived_basic_type_mirror_index(BasicType t, int index) {
254296
_archived_basic_type_mirror_indices[t] = index;
255297
}
256298

257-
void Universe::update_archived_basic_type_mirrors() {
299+
void Universe::archive_exception_instances() {
300+
_null_ptr_exception.store_in_cds();
301+
_arithmetic_exception.store_in_cds();
302+
_virtual_machine_error.store_in_cds();
303+
}
304+
305+
void Universe::load_archived_object_instances() {
258306
if (ArchiveHeapLoader::is_in_use()) {
259307
for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
260308
int index = _archived_basic_type_mirror_indices[i];
@@ -264,6 +312,10 @@ void Universe::update_archived_basic_type_mirrors() {
264312
_basic_type_mirrors[i] = OopHandle(vm_global(), mirror_oop);
265313
}
266314
}
315+
316+
_null_ptr_exception.load_from_cds();
317+
_arithmetic_exception.load_from_cds();
318+
_virtual_machine_error.load_from_cds();
267319
}
268320
}
269321
#endif
@@ -275,8 +327,11 @@ void Universe::serialize(SerializeClosure* f) {
275327
f->do_int(&_archived_basic_type_mirror_indices[i]);
276328
// if f->reading(): We can't call HeapShared::get_root() yet, as the heap
277329
// contents may need to be relocated. _basic_type_mirrors[i] will be
278-
// updated later in Universe::update_archived_basic_type_mirrors().
330+
// updated later in Universe::load_archived_object_instances().
279331
}
332+
_null_ptr_exception.serialize(f);
333+
_arithmetic_exception.serialize(f);
334+
_virtual_machine_error.serialize(f);
280335
#endif
281336

282337
f->do_ptr(&_fillerArrayKlassObj);
@@ -1021,27 +1076,19 @@ bool universe_post_init() {
10211076
Universe::_delayed_stack_overflow_error_message = OopHandle(Universe::vm_global(), instance);
10221077
}
10231078

1024-
// Setup preallocated NullPointerException
1025-
// (this is currently used for a cheap & dirty solution in compiler exception handling)
1026-
Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_NullPointerException(), true, CHECK_false);
1027-
instance = InstanceKlass::cast(k)->allocate_instance(CHECK_false);
1028-
Universe::_null_ptr_exception_instance = OopHandle(Universe::vm_global(), instance);
1029-
1030-
// Setup preallocated ArithmeticException
1031-
// (this is currently used for a cheap & dirty solution in compiler exception handling)
1032-
k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_ArithmeticException(), true, CHECK_false);
1033-
instance = InstanceKlass::cast(k)->allocate_instance(CHECK_false);
1034-
Universe::_arithmetic_exception_instance = OopHandle(Universe::vm_global(), instance);
1079+
// Setup preallocated NullPointerException/ArithmeticException
1080+
// (used for a cheap & dirty solution in compiler exception handling)
1081+
_null_ptr_exception.init_if_empty(vmSymbols::java_lang_NullPointerException(), CHECK_false);
1082+
_arithmetic_exception.init_if_empty(vmSymbols::java_lang_ArithmeticException(), CHECK_false);
10351083

10361084
// Virtual Machine Error for when we get into a situation we can't resolve
1037-
k = vmClasses::VirtualMachineError_klass();
1085+
Klass* k = vmClasses::VirtualMachineError_klass();
10381086
bool linked = InstanceKlass::cast(k)->link_class_or_fail(CHECK_false);
10391087
if (!linked) {
10401088
tty->print_cr("Unable to link/verify VirtualMachineError class");
10411089
return false; // initialization failed
10421090
}
1043-
instance = InstanceKlass::cast(k)->allocate_instance(CHECK_false);
1044-
Universe::_virtual_machine_error_instance = OopHandle(Universe::vm_global(), instance);
1091+
_virtual_machine_error.init_if_empty(vmSymbols::java_lang_VirtualMachineError(), CHECK_false);
10451092

10461093
Handle msg = java_lang_String::create_from_str("/ by zero", CHECK_false);
10471094
java_lang_Throwable::set_message(Universe::arithmetic_exception_instance(), msg());

‎src/hotspot/share/memory/universe.hpp

+2-5
Original file line numberDiff line numberDiff line change
@@ -106,10 +106,6 @@ class Universe: AllStatic {
106106
static OopHandle _msg_metaspace;
107107
static OopHandle _msg_class_metaspace;
108108

109-
static OopHandle _null_ptr_exception_instance; // preallocated exception object
110-
static OopHandle _arithmetic_exception_instance; // preallocated exception object
111-
static OopHandle _virtual_machine_error_instance; // preallocated exception object
112-
113109
// References waiting to be transferred to the ReferenceHandler
114110
static OopHandle _reference_pending_list;
115111

@@ -211,9 +207,10 @@ class Universe: AllStatic {
211207

212208
static oop java_mirror(BasicType t);
213209

210+
static void load_archived_object_instances() NOT_CDS_JAVA_HEAP_RETURN;
214211
#if INCLUDE_CDS_JAVA_HEAP
215212
static void set_archived_basic_type_mirror_index(BasicType t, int index);
216-
static void update_archived_basic_type_mirrors();
213+
static void archive_exception_instances();
217214
#endif
218215

219216
static oop main_thread_group();

0 commit comments

Comments
 (0)
Please sign in to comment.