Skip to content

Commit 4107dcf

Browse files
author
David Holmes
committedJul 2, 2021
8269466: Factor out the common code for initializing and starting internal VM JavaThreads
Reviewed-by: sspitsyn, pchilanomate, dcubed, coleenp, kvn
1 parent 2baf498 commit 4107dcf

13 files changed

+146
-219
lines changed
 

‎src/hotspot/share/compiler/compileBroker.cpp

+54-69
Original file line numberDiff line numberDiff line change
@@ -875,90 +875,75 @@ void DeoptimizeObjectsALotThread::deoptimize_objects_alot_loop_all() {
875875

876876
JavaThread* CompileBroker::make_thread(ThreadType type, jobject thread_handle, CompileQueue* queue, AbstractCompiler* comp, JavaThread* THREAD) {
877877
JavaThread* new_thread = NULL;
878-
{
879-
MutexLocker mu(THREAD, Threads_lock);
880-
switch (type) {
881-
case compiler_t:
882-
assert(comp != NULL, "Compiler instance missing.");
883-
if (!InjectCompilerCreationFailure || comp->num_compiler_threads() == 0) {
884-
CompilerCounters* counters = new CompilerCounters();
885-
new_thread = new CompilerThread(queue, counters);
886-
}
887-
break;
888-
case sweeper_t:
889-
new_thread = new CodeCacheSweeperThread();
890-
break;
878+
879+
switch (type) {
880+
case compiler_t:
881+
assert(comp != NULL, "Compiler instance missing.");
882+
if (!InjectCompilerCreationFailure || comp->num_compiler_threads() == 0) {
883+
CompilerCounters* counters = new CompilerCounters();
884+
new_thread = new CompilerThread(queue, counters);
885+
}
886+
break;
887+
case sweeper_t:
888+
new_thread = new CodeCacheSweeperThread();
889+
break;
891890
#if defined(ASSERT) && COMPILER2_OR_JVMCI
892-
case deoptimizer_t:
893-
new_thread = new DeoptimizeObjectsALotThread();
894-
break;
891+
case deoptimizer_t:
892+
new_thread = new DeoptimizeObjectsALotThread();
893+
break;
895894
#endif // ASSERT
896-
default:
897-
ShouldNotReachHere();
898-
}
899-
900-
// At this point the new CompilerThread data-races with this startup
901-
// thread (which I believe is the primoridal thread and NOT the VM
902-
// thread). This means Java bytecodes being executed at startup can
903-
// queue compile jobs which will run at whatever default priority the
904-
// newly created CompilerThread runs at.
905-
906-
907-
// At this point it may be possible that no osthread was created for the
908-
// JavaThread due to lack of memory. We would have to throw an exception
909-
// in that case. However, since this must work and we do not allow
910-
// exceptions anyway, check and abort if this fails. But first release the
911-
// lock.
912-
913-
if (new_thread != NULL && new_thread->osthread() != NULL) {
914-
915-
java_lang_Thread::set_thread(JNIHandles::resolve_non_null(thread_handle), new_thread);
895+
default:
896+
ShouldNotReachHere();
897+
}
916898

917-
// Note that this only sets the JavaThread _priority field, which by
918-
// definition is limited to Java priorities and not OS priorities.
919-
// The os-priority is set in the CompilerThread startup code itself
899+
// At this point the new CompilerThread data-races with this startup
900+
// thread (which is the main thread and NOT the VM thread).
901+
// This means Java bytecodes being executed at startup can
902+
// queue compile jobs which will run at whatever default priority the
903+
// newly created CompilerThread runs at.
920904

921-
java_lang_Thread::set_priority(JNIHandles::resolve_non_null(thread_handle), NearMaxPriority);
922905

923-
// Note that we cannot call os::set_priority because it expects Java
924-
// priorities and we are *explicitly* using OS priorities so that it's
925-
// possible to set the compiler thread priority higher than any Java
926-
// thread.
906+
// At this point it may be possible that no osthread was created for the
907+
// JavaThread due to lack of resources. We will handle that failure below.
908+
// Also check new_thread so that static analysis is happy.
909+
if (new_thread != NULL && new_thread->osthread() != NULL) {
910+
Handle thread_oop(THREAD, JNIHandles::resolve_non_null(thread_handle));
927911

928-
int native_prio = CompilerThreadPriority;
929-
if (native_prio == -1) {
930-
if (UseCriticalCompilerThreadPriority) {
931-
native_prio = os::java_to_os_priority[CriticalPriority];
932-
} else {
933-
native_prio = os::java_to_os_priority[NearMaxPriority];
934-
}
935-
}
936-
os::set_native_priority(new_thread, native_prio);
912+
if (type == compiler_t) {
913+
CompilerThread::cast(new_thread)->set_compiler(comp);
914+
}
937915

938-
java_lang_Thread::set_daemon(JNIHandles::resolve_non_null(thread_handle));
916+
// Note that we cannot call os::set_priority because it expects Java
917+
// priorities and we are *explicitly* using OS priorities so that it's
918+
// possible to set the compiler thread priority higher than any Java
919+
// thread.
939920

940-
new_thread->set_threadObj(JNIHandles::resolve_non_null(thread_handle));
941-
if (type == compiler_t) {
942-
CompilerThread::cast(new_thread)->set_compiler(comp);
921+
int native_prio = CompilerThreadPriority;
922+
if (native_prio == -1) {
923+
if (UseCriticalCompilerThreadPriority) {
924+
native_prio = os::java_to_os_priority[CriticalPriority];
925+
} else {
926+
native_prio = os::java_to_os_priority[NearMaxPriority];
943927
}
944-
Threads::add(new_thread);
945-
Thread::start(new_thread);
946928
}
947-
}
929+
os::set_native_priority(new_thread, native_prio);
948930

949-
// First release lock before aborting VM.
950-
if (new_thread == NULL || new_thread->osthread() == NULL) {
951-
if (UseDynamicNumberOfCompilerThreads && type == compiler_t && comp->num_compiler_threads() > 0) {
952-
if (new_thread != NULL) {
953-
new_thread->smr_delete();
954-
}
931+
// Note that this only sets the JavaThread _priority field, which by
932+
// definition is limited to Java priorities and not OS priorities.
933+
JavaThread::start_internal_daemon(THREAD, new_thread, thread_oop, NearMaxPriority);
934+
935+
} else { // osthread initialization failure
936+
if (UseDynamicNumberOfCompilerThreads && type == compiler_t
937+
&& comp->num_compiler_threads() > 0) {
938+
// The new thread is not known to Thread-SMR yet so we can just delete.
939+
delete new_thread;
955940
return NULL;
941+
} else {
942+
vm_exit_during_initialization("java.lang.OutOfMemoryError",
943+
os::native_thread_creation_failed_msg());
956944
}
957-
vm_exit_during_initialization("java.lang.OutOfMemoryError",
958-
os::native_thread_creation_failed_msg());
959945
}
960946

961-
// Let go of Threads_lock before yielding
962947
os::naked_yield(); // make sure that the compiler thread is started early (especially helpful on SOLARIS)
963948

964949
return new_thread;

‎src/hotspot/share/jfr/recorder/service/jfrRecorderThread.cpp

+11-22
Original file line numberDiff line numberDiff line change
@@ -46,29 +46,18 @@ static Thread* start_thread(instanceHandle thread_oop, ThreadFunction proc, TRAP
4646
assert(thread_oop.not_null(), "invariant");
4747
assert(proc != NULL, "invariant");
4848

49-
bool allocation_failed = false;
50-
JavaThread* new_thread = NULL;
51-
{
52-
MutexLocker mu(THREAD, Threads_lock);
53-
new_thread = new JavaThread(proc);
54-
// At this point it may be possible that no
55-
// osthread was created for the JavaThread due to lack of memory.
56-
if (new_thread == NULL || new_thread->osthread() == NULL) {
57-
delete new_thread;
58-
allocation_failed = true;
59-
} else {
60-
java_lang_Thread::set_thread(thread_oop(), new_thread);
61-
java_lang_Thread::set_priority(thread_oop(), NormPriority);
62-
java_lang_Thread::set_daemon(thread_oop());
63-
new_thread->set_threadObj(thread_oop());
64-
Threads::add(new_thread);
65-
}
66-
}
67-
if (allocation_failed) {
68-
JfrJavaSupport::throw_out_of_memory_error("Unable to create native recording thread for JFR", CHECK_NULL);
49+
JavaThread* new_thread = new JavaThread(proc);
50+
51+
// At this point it may be possible that no
52+
// osthread was created for the JavaThread due to lack of resources.
53+
if (new_thread->osthread() == NULL) {
54+
delete new_thread;
55+
JfrJavaSupport::throw_out_of_memory_error("Unable to create native recording thread for JFR", THREAD);
56+
return NULL;
57+
} else {
58+
JavaThread::start_internal_daemon(THREAD, new_thread, thread_oop, NormPriority);
59+
return new_thread;
6960
}
70-
Thread::start(new_thread);
71-
return new_thread;
7261
}
7362

7463
JfrPostBox* JfrRecorderThread::_post_box = NULL;

‎src/hotspot/share/prims/jvmtiEnv.cpp

+10-19
Original file line numberDiff line numberDiff line change
@@ -1326,28 +1326,19 @@ JvmtiEnv::RunAgentThread(jthread thread, jvmtiStartFunction proc, const void* ar
13261326
}
13271327

13281328
Handle thread_hndl(current_thread, thread_oop);
1329-
{
1330-
MutexLocker mu(current_thread, Threads_lock); // grab Threads_lock
13311329

1332-
JvmtiAgentThread *new_thread = new JvmtiAgentThread(this, proc, arg);
1330+
JvmtiAgentThread* new_thread = new JvmtiAgentThread(this, proc, arg);
13331331

1334-
// At this point it may be possible that no osthread was created for the
1335-
// JavaThread due to lack of memory.
1336-
if (new_thread == NULL || new_thread->osthread() == NULL) {
1337-
if (new_thread != NULL) {
1338-
new_thread->smr_delete();
1339-
}
1340-
return JVMTI_ERROR_OUT_OF_MEMORY;
1341-
}
1342-
1343-
java_lang_Thread::set_thread(thread_hndl(), new_thread);
1344-
java_lang_Thread::set_priority(thread_hndl(), (ThreadPriority)priority);
1345-
java_lang_Thread::set_daemon(thread_hndl());
1332+
// At this point it may be possible that no osthread was created for the
1333+
// JavaThread due to lack of resources.
1334+
if (new_thread->osthread() == NULL) {
1335+
// The new thread is not known to Thread-SMR yet so we can just delete.
1336+
delete new_thread;
1337+
return JVMTI_ERROR_OUT_OF_MEMORY;
1338+
}
13461339

1347-
new_thread->set_threadObj(thread_hndl());
1348-
Threads::add(new_thread);
1349-
Thread::start(new_thread);
1350-
} // unlock Threads_lock
1340+
JavaThread::start_internal_daemon(current_thread, new_thread, thread_hndl,
1341+
(ThreadPriority)priority);
13511342

13521343
return JVMTI_ERROR_NONE;
13531344
} /* end RunAgentThread */

‎src/hotspot/share/runtime/monitorDeflationThread.cpp

+3-23
Original file line numberDiff line numberDiff line change
@@ -33,8 +33,6 @@
3333
#include "runtime/monitorDeflationThread.hpp"
3434
#include "runtime/mutexLocker.hpp"
3535

36-
MonitorDeflationThread* MonitorDeflationThread::_instance = NULL;
37-
3836
void MonitorDeflationThread::initialize() {
3937
EXCEPTION_MARK;
4038

@@ -50,28 +48,10 @@ void MonitorDeflationThread::initialize() {
5048
string,
5149
CHECK);
5250

53-
{
54-
MutexLocker mu(THREAD, Threads_lock);
55-
MonitorDeflationThread* thread = new MonitorDeflationThread(&monitor_deflation_thread_entry);
56-
57-
// At this point it may be possible that no osthread was created for the
58-
// JavaThread due to lack of memory. We would have to throw an exception
59-
// in that case. However, since this must work and we do not allow
60-
// exceptions anyway, check and abort if this fails.
61-
if (thread == NULL || thread->osthread() == NULL) {
62-
vm_exit_during_initialization("java.lang.OutOfMemoryError",
63-
os::native_thread_creation_failed_msg());
64-
}
51+
MonitorDeflationThread* thread = new MonitorDeflationThread(&monitor_deflation_thread_entry);
52+
JavaThread::vm_exit_on_osthread_failure(thread);
6553

66-
java_lang_Thread::set_thread(thread_oop(), thread);
67-
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
68-
java_lang_Thread::set_daemon(thread_oop());
69-
thread->set_threadObj(thread_oop());
70-
_instance = thread;
71-
72-
Threads::add(thread);
73-
Thread::start(thread);
74-
}
54+
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
7555
}
7656

7757
void MonitorDeflationThread::monitor_deflation_thread_entry(JavaThread* jt, TRAPS) {

‎src/hotspot/share/runtime/monitorDeflationThread.hpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 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
@@ -32,7 +32,6 @@
3232
class MonitorDeflationThread : public JavaThread {
3333
friend class VMStructs;
3434
private:
35-
static MonitorDeflationThread* _instance;
3635

3736
static void monitor_deflation_thread_entry(JavaThread* thread, TRAPS);
3837
MonitorDeflationThread(ThreadFunction entry_point) : JavaThread(entry_point) {};

‎src/hotspot/share/runtime/notificationThread.cpp

+3-23
Original file line numberDiff line numberDiff line change
@@ -35,8 +35,6 @@
3535
#include "services/gcNotifier.hpp"
3636
#include "services/lowMemoryDetector.hpp"
3737

38-
NotificationThread* NotificationThread::_instance = NULL;
39-
4038
void NotificationThread::initialize() {
4139
EXCEPTION_MARK;
4240

@@ -61,28 +59,11 @@ void NotificationThread::initialize() {
6159
vmSymbols::thread_void_signature(),
6260
thread_oop,
6361
THREAD);
64-
{
65-
MutexLocker mu(THREAD, Threads_lock);
66-
NotificationThread* thread = new NotificationThread(&notification_thread_entry);
67-
68-
// At this point it may be possible that no osthread was created for the
69-
// JavaThread due to lack of memory. We would have to throw an exception
70-
// in that case. However, since this must work and we do not allow
71-
// exceptions anyway, check and abort if this fails.
72-
if (thread == NULL || thread->osthread() == NULL) {
73-
vm_exit_during_initialization("java.lang.OutOfMemoryError",
74-
os::native_thread_creation_failed_msg());
75-
}
7662

77-
java_lang_Thread::set_thread(thread_oop(), thread);
78-
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
79-
java_lang_Thread::set_daemon(thread_oop());
80-
thread->set_threadObj(thread_oop());
81-
_instance = thread;
63+
NotificationThread* thread = new NotificationThread(&notification_thread_entry);
64+
JavaThread::vm_exit_on_osthread_failure(thread);
8265

83-
Threads::add(thread);
84-
Thread::start(thread);
85-
}
66+
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
8667
}
8768

8869

@@ -128,4 +109,3 @@ void NotificationThread::notification_thread_entry(JavaThread* jt, TRAPS) {
128109

129110
}
130111
}
131-

‎src/hotspot/share/runtime/notificationThread.hpp

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2019, 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
@@ -36,8 +36,6 @@ class NotificationThread : public JavaThread {
3636
friend class VMStructs;
3737
private:
3838

39-
static NotificationThread* _instance;
40-
4139
static void notification_thread_entry(JavaThread* thread, TRAPS);
4240
NotificationThread(ThreadFunction entry_point) : JavaThread(entry_point) {};
4341

‎src/hotspot/share/runtime/os.cpp

+3-18
Original file line numberDiff line numberDiff line change
@@ -490,26 +490,11 @@ void os::initialize_jdk_signal_support(TRAPS) {
490490
thread_oop,
491491
CHECK);
492492

493-
{ MutexLocker mu(THREAD, Threads_lock);
494-
JavaThread* signal_thread = new JavaThread(&signal_thread_entry);
495-
496-
// At this point it may be possible that no osthread was created for the
497-
// JavaThread due to lack of memory. We would have to throw an exception
498-
// in that case. However, since this must work and we do not allow
499-
// exceptions anyway, check and abort if this fails.
500-
if (signal_thread == NULL || signal_thread->osthread() == NULL) {
501-
vm_exit_during_initialization("java.lang.OutOfMemoryError",
502-
os::native_thread_creation_failed_msg());
503-
}
493+
JavaThread* thread = new JavaThread(&signal_thread_entry);
494+
JavaThread::vm_exit_on_osthread_failure(thread);
504495

505-
java_lang_Thread::set_thread(thread_oop(), signal_thread);
506-
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
507-
java_lang_Thread::set_daemon(thread_oop());
496+
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
508497

509-
signal_thread->set_threadObj(thread_oop());
510-
Threads::add(signal_thread);
511-
Thread::start(signal_thread);
512-
}
513498
// Handle ^BREAK
514499
os::signal(SIGBREAK, os::user_handler());
515500
}

0 commit comments

Comments
 (0)
Please sign in to comment.