Skip to content

Commit 40c3249

Browse files
committedMar 31, 2021
8264149: BreakpointInfo::set allocates metaspace object in VM thread
Reviewed-by: dholmes, iklam
1 parent 999c134 commit 40c3249

10 files changed

+97
-47
lines changed
 

‎src/hotspot/share/interpreter/interpreterRuntime.cpp

+1-7
Original file line numberDiff line numberDiff line change
@@ -1115,13 +1115,7 @@ JRT_ENTRY(void, InterpreterRuntime::update_mdp_for_ret(JavaThread* thread, int r
11151115
JRT_END
11161116

11171117
JRT_ENTRY(MethodCounters*, InterpreterRuntime::build_method_counters(JavaThread* thread, Method* m))
1118-
MethodCounters* mcs = Method::build_method_counters(m, thread);
1119-
if (HAS_PENDING_EXCEPTION) {
1120-
// Only metaspace OOM is expected. No Java code executed.
1121-
assert((PENDING_EXCEPTION->is_a(vmClasses::OutOfMemoryError_klass())), "we expect only an OOM error here");
1122-
CLEAR_PENDING_EXCEPTION;
1123-
}
1124-
return mcs;
1118+
return Method::build_method_counters(thread, m);
11251119
JRT_END
11261120

11271121

‎src/hotspot/share/interpreter/interpreterRuntime.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -158,7 +158,7 @@ class InterpreterRuntime: AllStatic {
158158
#ifdef ASSERT
159159
static void verify_mdp(Method* method, address bcp, address mdp);
160160
#endif // ASSERT
161-
static MethodCounters* build_method_counters(JavaThread* thread, Method* m);
161+
static MethodCounters* build_method_counters(JavaThread* current, Method* m);
162162
};
163163

164164

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

+8-1
Original file line numberDiff line numberDiff line change
@@ -79,10 +79,17 @@ void StackObj::operator delete [](void* p) { ShouldNotCallThis(); }
7979
void* MetaspaceObj::operator new(size_t size, ClassLoaderData* loader_data,
8080
size_t word_size,
8181
MetaspaceObj::Type type, TRAPS) throw() {
82-
// Klass has it's own operator new
82+
// Klass has its own operator new
8383
return Metaspace::allocate(loader_data, word_size, type, THREAD);
8484
}
8585

86+
void* MetaspaceObj::operator new(size_t size, ClassLoaderData* loader_data,
87+
size_t word_size,
88+
MetaspaceObj::Type type) throw() {
89+
assert(!Thread::current()->is_Java_thread(), "only allowed by non-Java thread");
90+
return Metaspace::allocate(loader_data, word_size, type);
91+
}
92+
8693
bool MetaspaceObj::is_valid(const MetaspaceObj* p) {
8794
// Weed out obvious bogus values first without traversing metaspace
8895
if ((size_t)p < os::min_page_size()) {

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

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1997, 2021, 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
@@ -345,6 +345,9 @@ class MetaspaceObj {
345345
size_t word_size,
346346
Type type, Thread* thread) throw();
347347
// can't use TRAPS from this header file.
348+
void* operator new(size_t size, ClassLoaderData* loader_data,
349+
size_t word_size,
350+
Type type) throw();
348351
void operator delete(void* p) { ShouldNotCallThis(); }
349352

350353
// Declare a *static* method with the same signature in any subclass of MetaspaceObj

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

+36-16
Original file line numberDiff line numberDiff line change
@@ -789,17 +789,14 @@ size_t Metaspace::max_allocation_word_size() {
789789
return metaspace::chunklevel::MAX_CHUNK_WORD_SIZE - max_overhead_words;
790790
}
791791

792+
// This version of Metaspace::allocate does not throw OOM but simply returns NULL, and
793+
// is suitable for calling from non-Java threads.
794+
// Callers are responsible for checking null.
792795
MetaWord* Metaspace::allocate(ClassLoaderData* loader_data, size_t word_size,
793-
MetaspaceObj::Type type, TRAPS) {
796+
MetaspaceObj::Type type) {
794797
assert(word_size <= Metaspace::max_allocation_word_size(),
795798
"allocation size too large (" SIZE_FORMAT ")", word_size);
796799
assert(!_frozen, "sanity");
797-
assert(!(DumpSharedSpaces && THREAD->is_VM_thread()), "sanity");
798-
799-
if (HAS_PENDING_EXCEPTION) {
800-
assert(false, "Should not allocate with exception pending");
801-
return NULL; // caller does a CHECK_NULL too
802-
}
803800

804801
assert(loader_data != NULL, "Should never pass around a NULL loader_data. "
805802
"ClassLoaderData::the_null_class_loader_data() should have been used.");
@@ -809,7 +806,30 @@ MetaWord* Metaspace::allocate(ClassLoaderData* loader_data, size_t word_size,
809806
// Try to allocate metadata.
810807
MetaWord* result = loader_data->metaspace_non_null()->allocate(word_size, mdtype);
811808

809+
if (result != NULL) {
810+
// Zero initialize.
811+
Copy::fill_to_words((HeapWord*)result, word_size, 0);
812+
813+
log_trace(metaspace)("Metaspace::allocate: type %d return " PTR_FORMAT ".", (int)type, p2i(result));
814+
}
815+
816+
return result;
817+
}
818+
819+
MetaWord* Metaspace::allocate(ClassLoaderData* loader_data, size_t word_size,
820+
MetaspaceObj::Type type, TRAPS) {
821+
822+
assert(THREAD->is_Java_thread(), "can't allocate in non-Java thread because we cannot throw exception");
823+
824+
if (HAS_PENDING_EXCEPTION) {
825+
assert(false, "Should not allocate with exception pending");
826+
return NULL; // caller does a CHECK_NULL too
827+
}
828+
829+
MetaWord* result = allocate(loader_data, word_size, type);
830+
812831
if (result == NULL) {
832+
MetadataType mdtype = (type == MetaspaceObj::ClassType) ? ClassType : NonClassType;
813833
tracer()->report_metaspace_allocation_failure(loader_data, word_size, type, mdtype);
814834

815835
// Allocation failed.
@@ -819,18 +839,18 @@ MetaWord* Metaspace::allocate(ClassLoaderData* loader_data, size_t word_size,
819839
// expansion of the metaspace.
820840
result = Universe::heap()->satisfy_failed_metadata_allocation(loader_data, word_size, mdtype);
821841
}
822-
}
823842

824-
if (result == NULL) {
825-
report_metadata_oome(loader_data, word_size, type, mdtype, THREAD);
826-
assert(HAS_PENDING_EXCEPTION, "sanity");
827-
return NULL;
828-
}
843+
if (result == NULL) {
844+
report_metadata_oome(loader_data, word_size, type, mdtype, THREAD);
845+
assert(HAS_PENDING_EXCEPTION, "sanity");
846+
return NULL;
847+
}
829848

830-
// Zero initialize.
831-
Copy::fill_to_words((HeapWord*)result, word_size, 0);
849+
// Zero initialize.
850+
Copy::fill_to_words((HeapWord*)result, word_size, 0);
832851

833-
log_trace(metaspace)("Metaspace::allocate: type %d return " PTR_FORMAT ".", (int)type, p2i(result));
852+
log_trace(metaspace)("Metaspace::allocate: type %d return " PTR_FORMAT ".", (int)type, p2i(result));
853+
}
834854

835855
return result;
836856
}

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

+5
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,11 @@ class Metaspace : public AllStatic {
124124
static MetaWord* allocate(ClassLoaderData* loader_data, size_t word_size,
125125
MetaspaceObj::Type type, TRAPS);
126126

127+
// Non-TRAPS version of allocate which can be called by a non-Java thread, that returns
128+
// NULL on failure.
129+
static MetaWord* allocate(ClassLoaderData* loader_data, size_t word_size,
130+
MetaspaceObj::Type type);
131+
127132
static bool contains(const void* ptr);
128133
static bool contains_non_shared(const void* ptr);
129134

‎src/hotspot/share/oops/method.cpp

+23-8
Original file line numberDiff line numberDiff line change
@@ -557,25 +557,40 @@ void Method::build_interpreter_method_data(const methodHandle& method, TRAPS) {
557557
}
558558
}
559559

560-
MethodCounters* Method::build_method_counters(Method* m, TRAPS) {
560+
MethodCounters* Method::build_method_counters(Thread* current, Method* m) {
561561
// Do not profile the method if metaspace has hit an OOM previously
562562
if (ClassLoaderDataGraph::has_metaspace_oom()) {
563563
return NULL;
564564
}
565565

566-
methodHandle mh(THREAD, m);
567-
MethodCounters* counters = MethodCounters::allocate(mh, THREAD);
568-
if (HAS_PENDING_EXCEPTION) {
566+
methodHandle mh(current, m);
567+
MethodCounters* counters;
568+
if (current->is_Java_thread()) {
569+
Thread* THREAD = current;
570+
// Use the TRAPS version for a JavaThread so it will adjust the GC threshold
571+
// if needed.
572+
counters = MethodCounters::allocate_with_exception(mh, THREAD);
573+
if (HAS_PENDING_EXCEPTION) {
574+
CLEAR_PENDING_EXCEPTION;
575+
}
576+
} else {
577+
// Call metaspace allocation that doesn't throw exception if the
578+
// current thread isn't a JavaThread, ie. the VMThread.
579+
counters = MethodCounters::allocate_no_exception(mh);
580+
}
581+
582+
if (counters == NULL) {
569583
CompileBroker::log_metaspace_failure();
570584
ClassLoaderDataGraph::set_metaspace_oom(true);
571-
return NULL; // return the exception (which is cleared)
585+
return NULL;
572586
}
587+
573588
if (!mh->init_method_counters(counters)) {
574589
MetadataFactory::free_metadata(mh->method_holder()->class_loader_data(), counters);
575590
}
576591

577592
if (LogTouchedMethods) {
578-
mh->log_touched(CHECK_NULL);
593+
mh->log_touched(current);
579594
}
580595

581596
return mh->method_counters();
@@ -2374,7 +2389,7 @@ class TouchedMethodRecord : CHeapObj<mtTracing> {
23742389
static const int TOUCHED_METHOD_TABLE_SIZE = 20011;
23752390
static TouchedMethodRecord** _touched_method_table = NULL;
23762391

2377-
void Method::log_touched(TRAPS) {
2392+
void Method::log_touched(Thread* current) {
23782393

23792394
const int table_size = TOUCHED_METHOD_TABLE_SIZE;
23802395
Symbol* my_class = klass_name();
@@ -2386,7 +2401,7 @@ void Method::log_touched(TRAPS) {
23862401
my_sig->identity_hash();
23872402
juint index = juint(hash) % table_size;
23882403

2389-
MutexLocker ml(THREAD, TouchedMethodLog_lock);
2404+
MutexLocker ml(current, TouchedMethodLog_lock);
23902405
if (_touched_method_table == NULL) {
23912406
_touched_method_table = NEW_C_HEAP_ARRAY2(TouchedMethodRecord*, table_size,
23922407
mtTracing, CURRENT_PC);

‎src/hotspot/share/oops/method.hpp

+11-11
Original file line numberDiff line numberDiff line change
@@ -228,28 +228,28 @@ class Method : public Metadata {
228228
void clear_all_breakpoints();
229229
// Tracking number of breakpoints, for fullspeed debugging.
230230
// Only mutated by VM thread.
231-
u2 number_of_breakpoints() const {
231+
u2 number_of_breakpoints() const {
232232
MethodCounters* mcs = method_counters();
233233
if (mcs == NULL) {
234234
return 0;
235235
} else {
236236
return mcs->number_of_breakpoints();
237237
}
238238
}
239-
void incr_number_of_breakpoints(TRAPS) {
240-
MethodCounters* mcs = get_method_counters(CHECK);
239+
void incr_number_of_breakpoints(Thread* current) {
240+
MethodCounters* mcs = get_method_counters(current);
241241
if (mcs != NULL) {
242242
mcs->incr_number_of_breakpoints();
243243
}
244244
}
245-
void decr_number_of_breakpoints(TRAPS) {
246-
MethodCounters* mcs = get_method_counters(CHECK);
245+
void decr_number_of_breakpoints(Thread* current) {
246+
MethodCounters* mcs = get_method_counters(current);
247247
if (mcs != NULL) {
248248
mcs->decr_number_of_breakpoints();
249249
}
250250
}
251251
// Initialization only
252-
void clear_number_of_breakpoints() {
252+
void clear_number_of_breakpoints() {
253253
MethodCounters* mcs = method_counters();
254254
if (mcs != NULL) {
255255
mcs->clear_number_of_breakpoints();
@@ -292,8 +292,8 @@ class Method : public Metadata {
292292

293293
#if COMPILER2_OR_JVMCI
294294
// Count of times method was exited via exception while interpreting
295-
void interpreter_throwout_increment(TRAPS) {
296-
MethodCounters* mcs = get_method_counters(CHECK);
295+
void interpreter_throwout_increment(Thread* current) {
296+
MethodCounters* mcs = get_method_counters(current);
297297
if (mcs != NULL) {
298298
mcs->interpreter_throwout_increment();
299299
}
@@ -432,7 +432,7 @@ class Method : public Metadata {
432432

433433
static void build_interpreter_method_data(const methodHandle& method, TRAPS);
434434

435-
static MethodCounters* build_method_counters(Method* m, TRAPS);
435+
static MethodCounters* build_method_counters(Thread* current, Method* m);
436436

437437
int interpreter_invocation_count() { return invocation_count(); }
438438

@@ -944,9 +944,9 @@ class Method : public Metadata {
944944
void print_made_not_compilable(int comp_level, bool is_osr, bool report, const char* reason);
945945

946946
public:
947-
MethodCounters* get_method_counters(TRAPS) {
947+
MethodCounters* get_method_counters(Thread* current) {
948948
if (_method_counters == NULL) {
949-
build_method_counters(this, CHECK_AND_CLEAR_NULL);
949+
build_method_counters(current, this);
950950
}
951951
return _method_counters;
952952
}

‎src/hotspot/share/oops/methodCounters.cpp

+6-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,12 @@ MethodCounters::MethodCounters(const methodHandle& mh) :
5555
_backedge_mask = right_n_bits(CompilerConfig::scaled_freq_log(Tier0BackedgeNotifyFreqLog, scale)) << InvocationCounter::count_shift;
5656
}
5757

58-
MethodCounters* MethodCounters::allocate(const methodHandle& mh, TRAPS) {
58+
MethodCounters* MethodCounters::allocate_no_exception(const methodHandle& mh) {
59+
ClassLoaderData* loader_data = mh->method_holder()->class_loader_data();
60+
return new(loader_data, method_counters_size(), MetaspaceObj::MethodCountersType) MethodCounters(mh);
61+
}
62+
63+
MethodCounters* MethodCounters::allocate_with_exception(const methodHandle& mh, TRAPS) {
5964
ClassLoaderData* loader_data = mh->method_holder()->class_loader_data();
6065
return new(loader_data, method_counters_size(), MetaspaceObj::MethodCountersType, THREAD) MethodCounters(mh);
6166
}

‎src/hotspot/share/oops/methodCounters.hpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,8 @@ class MethodCounters : public Metadata {
6969
public:
7070
virtual bool is_methodCounters() const { return true; }
7171

72-
static MethodCounters* allocate(const methodHandle& mh, TRAPS);
72+
static MethodCounters* allocate_no_exception(const methodHandle& mh);
73+
static MethodCounters* allocate_with_exception(const methodHandle& mh, TRAPS);
7374

7475
void deallocate_contents(ClassLoaderData* loader_data) {}
7576

0 commit comments

Comments
 (0)
Please sign in to comment.