8269466: Factor out the common code for initializing and starting internal VM JavaThreads
Reviewed-by: sspitsyn, pchilanomate, dcubed, coleenp, kvn
This commit is contained in:
parent
2baf498eb5
commit
4107dcf6ec
@ -875,8 +875,7 @@ void DeoptimizeObjectsALotThread::deoptimize_objects_alot_loop_all() {
|
||||
|
||||
JavaThread* CompileBroker::make_thread(ThreadType type, jobject thread_handle, CompileQueue* queue, AbstractCompiler* comp, JavaThread* THREAD) {
|
||||
JavaThread* new_thread = NULL;
|
||||
{
|
||||
MutexLocker mu(THREAD, Threads_lock);
|
||||
|
||||
switch (type) {
|
||||
case compiler_t:
|
||||
assert(comp != NULL, "Compiler instance missing.");
|
||||
@ -898,27 +897,21 @@ JavaThread* CompileBroker::make_thread(ThreadType type, jobject thread_handle, C
|
||||
}
|
||||
|
||||
// At this point the new CompilerThread data-races with this startup
|
||||
// thread (which I believe is the primoridal thread and NOT the VM
|
||||
// thread). This means Java bytecodes being executed at startup can
|
||||
// thread (which is the main thread and NOT the VM thread).
|
||||
// This means Java bytecodes being executed at startup can
|
||||
// queue compile jobs which will run at whatever default priority the
|
||||
// newly created CompilerThread runs at.
|
||||
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory. We would have to throw an exception
|
||||
// in that case. However, since this must work and we do not allow
|
||||
// exceptions anyway, check and abort if this fails. But first release the
|
||||
// lock.
|
||||
|
||||
// JavaThread due to lack of resources. We will handle that failure below.
|
||||
// Also check new_thread so that static analysis is happy.
|
||||
if (new_thread != NULL && new_thread->osthread() != NULL) {
|
||||
Handle thread_oop(THREAD, JNIHandles::resolve_non_null(thread_handle));
|
||||
|
||||
java_lang_Thread::set_thread(JNIHandles::resolve_non_null(thread_handle), new_thread);
|
||||
|
||||
// Note that this only sets the JavaThread _priority field, which by
|
||||
// definition is limited to Java priorities and not OS priorities.
|
||||
// The os-priority is set in the CompilerThread startup code itself
|
||||
|
||||
java_lang_Thread::set_priority(JNIHandles::resolve_non_null(thread_handle), NearMaxPriority);
|
||||
if (type == compiler_t) {
|
||||
CompilerThread::cast(new_thread)->set_compiler(comp);
|
||||
}
|
||||
|
||||
// Note that we cannot call os::set_priority because it expects Java
|
||||
// priorities and we are *explicitly* using OS priorities so that it's
|
||||
@ -935,30 +928,22 @@ JavaThread* CompileBroker::make_thread(ThreadType type, jobject thread_handle, C
|
||||
}
|
||||
os::set_native_priority(new_thread, native_prio);
|
||||
|
||||
java_lang_Thread::set_daemon(JNIHandles::resolve_non_null(thread_handle));
|
||||
// Note that this only sets the JavaThread _priority field, which by
|
||||
// definition is limited to Java priorities and not OS priorities.
|
||||
JavaThread::start_internal_daemon(THREAD, new_thread, thread_oop, NearMaxPriority);
|
||||
|
||||
new_thread->set_threadObj(JNIHandles::resolve_non_null(thread_handle));
|
||||
if (type == compiler_t) {
|
||||
CompilerThread::cast(new_thread)->set_compiler(comp);
|
||||
}
|
||||
Threads::add(new_thread);
|
||||
Thread::start(new_thread);
|
||||
}
|
||||
}
|
||||
|
||||
// First release lock before aborting VM.
|
||||
if (new_thread == NULL || new_thread->osthread() == NULL) {
|
||||
if (UseDynamicNumberOfCompilerThreads && type == compiler_t && comp->num_compiler_threads() > 0) {
|
||||
if (new_thread != NULL) {
|
||||
new_thread->smr_delete();
|
||||
}
|
||||
} else { // osthread initialization failure
|
||||
if (UseDynamicNumberOfCompilerThreads && type == compiler_t
|
||||
&& comp->num_compiler_threads() > 0) {
|
||||
// The new thread is not known to Thread-SMR yet so we can just delete.
|
||||
delete new_thread;
|
||||
return NULL;
|
||||
}
|
||||
} else {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
}
|
||||
|
||||
// Let go of Threads_lock before yielding
|
||||
os::naked_yield(); // make sure that the compiler thread is started early (especially helpful on SOLARIS)
|
||||
|
||||
return new_thread;
|
||||
|
@ -46,29 +46,18 @@ static Thread* start_thread(instanceHandle thread_oop, ThreadFunction proc, TRAP
|
||||
assert(thread_oop.not_null(), "invariant");
|
||||
assert(proc != NULL, "invariant");
|
||||
|
||||
bool allocation_failed = false;
|
||||
JavaThread* new_thread = NULL;
|
||||
{
|
||||
MutexLocker mu(THREAD, Threads_lock);
|
||||
new_thread = new JavaThread(proc);
|
||||
JavaThread* new_thread = new JavaThread(proc);
|
||||
|
||||
// At this point it may be possible that no
|
||||
// osthread was created for the JavaThread due to lack of memory.
|
||||
if (new_thread == NULL || new_thread->osthread() == NULL) {
|
||||
// osthread was created for the JavaThread due to lack of resources.
|
||||
if (new_thread->osthread() == NULL) {
|
||||
delete new_thread;
|
||||
allocation_failed = true;
|
||||
JfrJavaSupport::throw_out_of_memory_error("Unable to create native recording thread for JFR", THREAD);
|
||||
return NULL;
|
||||
} else {
|
||||
java_lang_Thread::set_thread(thread_oop(), new_thread);
|
||||
java_lang_Thread::set_priority(thread_oop(), NormPriority);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
new_thread->set_threadObj(thread_oop());
|
||||
Threads::add(new_thread);
|
||||
}
|
||||
}
|
||||
if (allocation_failed) {
|
||||
JfrJavaSupport::throw_out_of_memory_error("Unable to create native recording thread for JFR", CHECK_NULL);
|
||||
}
|
||||
Thread::start(new_thread);
|
||||
JavaThread::start_internal_daemon(THREAD, new_thread, thread_oop, NormPriority);
|
||||
return new_thread;
|
||||
}
|
||||
}
|
||||
|
||||
JfrPostBox* JfrRecorderThread::_post_box = NULL;
|
||||
|
@ -1326,28 +1326,19 @@ JvmtiEnv::RunAgentThread(jthread thread, jvmtiStartFunction proc, const void* ar
|
||||
}
|
||||
|
||||
Handle thread_hndl(current_thread, thread_oop);
|
||||
{
|
||||
MutexLocker mu(current_thread, Threads_lock); // grab Threads_lock
|
||||
|
||||
JvmtiAgentThread *new_thread = new JvmtiAgentThread(this, proc, arg);
|
||||
JvmtiAgentThread* new_thread = new JvmtiAgentThread(this, proc, arg);
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory.
|
||||
if (new_thread == NULL || new_thread->osthread() == NULL) {
|
||||
if (new_thread != NULL) {
|
||||
new_thread->smr_delete();
|
||||
}
|
||||
// JavaThread due to lack of resources.
|
||||
if (new_thread->osthread() == NULL) {
|
||||
// The new thread is not known to Thread-SMR yet so we can just delete.
|
||||
delete new_thread;
|
||||
return JVMTI_ERROR_OUT_OF_MEMORY;
|
||||
}
|
||||
|
||||
java_lang_Thread::set_thread(thread_hndl(), new_thread);
|
||||
java_lang_Thread::set_priority(thread_hndl(), (ThreadPriority)priority);
|
||||
java_lang_Thread::set_daemon(thread_hndl());
|
||||
|
||||
new_thread->set_threadObj(thread_hndl());
|
||||
Threads::add(new_thread);
|
||||
Thread::start(new_thread);
|
||||
} // unlock Threads_lock
|
||||
JavaThread::start_internal_daemon(current_thread, new_thread, thread_hndl,
|
||||
(ThreadPriority)priority);
|
||||
|
||||
return JVMTI_ERROR_NONE;
|
||||
} /* end RunAgentThread */
|
||||
|
@ -33,8 +33,6 @@
|
||||
#include "runtime/monitorDeflationThread.hpp"
|
||||
#include "runtime/mutexLocker.hpp"
|
||||
|
||||
MonitorDeflationThread* MonitorDeflationThread::_instance = NULL;
|
||||
|
||||
void MonitorDeflationThread::initialize() {
|
||||
EXCEPTION_MARK;
|
||||
|
||||
@ -50,28 +48,10 @@ void MonitorDeflationThread::initialize() {
|
||||
string,
|
||||
CHECK);
|
||||
|
||||
{
|
||||
MutexLocker mu(THREAD, Threads_lock);
|
||||
MonitorDeflationThread* thread = new MonitorDeflationThread(&monitor_deflation_thread_entry);
|
||||
JavaThread::vm_exit_on_osthread_failure(thread);
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory. We would have to throw an exception
|
||||
// in that case. However, since this must work and we do not allow
|
||||
// exceptions anyway, check and abort if this fails.
|
||||
if (thread == NULL || thread->osthread() == NULL) {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), thread);
|
||||
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
thread->set_threadObj(thread_oop());
|
||||
_instance = thread;
|
||||
|
||||
Threads::add(thread);
|
||||
Thread::start(thread);
|
||||
}
|
||||
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
|
||||
}
|
||||
|
||||
void MonitorDeflationThread::monitor_deflation_thread_entry(JavaThread* jt, TRAPS) {
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -32,7 +32,6 @@
|
||||
class MonitorDeflationThread : public JavaThread {
|
||||
friend class VMStructs;
|
||||
private:
|
||||
static MonitorDeflationThread* _instance;
|
||||
|
||||
static void monitor_deflation_thread_entry(JavaThread* thread, TRAPS);
|
||||
MonitorDeflationThread(ThreadFunction entry_point) : JavaThread(entry_point) {};
|
||||
|
@ -35,8 +35,6 @@
|
||||
#include "services/gcNotifier.hpp"
|
||||
#include "services/lowMemoryDetector.hpp"
|
||||
|
||||
NotificationThread* NotificationThread::_instance = NULL;
|
||||
|
||||
void NotificationThread::initialize() {
|
||||
EXCEPTION_MARK;
|
||||
|
||||
@ -61,28 +59,11 @@ void NotificationThread::initialize() {
|
||||
vmSymbols::thread_void_signature(),
|
||||
thread_oop,
|
||||
THREAD);
|
||||
{
|
||||
MutexLocker mu(THREAD, Threads_lock);
|
||||
|
||||
NotificationThread* thread = new NotificationThread(¬ification_thread_entry);
|
||||
JavaThread::vm_exit_on_osthread_failure(thread);
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory. We would have to throw an exception
|
||||
// in that case. However, since this must work and we do not allow
|
||||
// exceptions anyway, check and abort if this fails.
|
||||
if (thread == NULL || thread->osthread() == NULL) {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), thread);
|
||||
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
thread->set_threadObj(thread_oop());
|
||||
_instance = thread;
|
||||
|
||||
Threads::add(thread);
|
||||
Thread::start(thread);
|
||||
}
|
||||
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
|
||||
}
|
||||
|
||||
|
||||
@ -128,4 +109,3 @@ void NotificationThread::notification_thread_entry(JavaThread* jt, TRAPS) {
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2019, 2021, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -36,8 +36,6 @@ class NotificationThread : public JavaThread {
|
||||
friend class VMStructs;
|
||||
private:
|
||||
|
||||
static NotificationThread* _instance;
|
||||
|
||||
static void notification_thread_entry(JavaThread* thread, TRAPS);
|
||||
NotificationThread(ThreadFunction entry_point) : JavaThread(entry_point) {};
|
||||
|
||||
|
@ -490,26 +490,11 @@ void os::initialize_jdk_signal_support(TRAPS) {
|
||||
thread_oop,
|
||||
CHECK);
|
||||
|
||||
{ MutexLocker mu(THREAD, Threads_lock);
|
||||
JavaThread* signal_thread = new JavaThread(&signal_thread_entry);
|
||||
JavaThread* thread = new JavaThread(&signal_thread_entry);
|
||||
JavaThread::vm_exit_on_osthread_failure(thread);
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory. We would have to throw an exception
|
||||
// in that case. However, since this must work and we do not allow
|
||||
// exceptions anyway, check and abort if this fails.
|
||||
if (signal_thread == NULL || signal_thread->osthread() == NULL) {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), signal_thread);
|
||||
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
|
||||
signal_thread->set_threadObj(thread_oop());
|
||||
Threads::add(signal_thread);
|
||||
Thread::start(signal_thread);
|
||||
}
|
||||
// Handle ^BREAK
|
||||
os::signal(SIGBREAK, os::user_handler());
|
||||
}
|
||||
|
@ -51,7 +51,7 @@
|
||||
#include "services/lowMemoryDetector.hpp"
|
||||
#include "services/threadIdTable.hpp"
|
||||
|
||||
ServiceThread* ServiceThread::_instance = NULL;
|
||||
DEBUG_ONLY(JavaThread* ServiceThread::_instance = NULL;)
|
||||
JvmtiDeferredEvent* ServiceThread::_jvmti_event = NULL;
|
||||
// The service thread has it's own static deferred event queue.
|
||||
// Events can be posted before JVMTI vm_start, so it's too early to call JvmtiThreadState::state_for
|
||||
@ -103,28 +103,11 @@ void ServiceThread::initialize() {
|
||||
string,
|
||||
CHECK);
|
||||
|
||||
{
|
||||
MutexLocker mu(THREAD, Threads_lock);
|
||||
ServiceThread* thread = new ServiceThread(&service_thread_entry);
|
||||
JavaThread::vm_exit_on_osthread_failure(thread);
|
||||
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of memory. We would have to throw an exception
|
||||
// in that case. However, since this must work and we do not allow
|
||||
// exceptions anyway, check and abort if this fails.
|
||||
if (thread == NULL || thread->osthread() == NULL) {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), thread);
|
||||
java_lang_Thread::set_priority(thread_oop(), NearMaxPriority);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
thread->set_threadObj(thread_oop());
|
||||
_instance = thread;
|
||||
|
||||
Threads::add(thread);
|
||||
Thread::start(thread);
|
||||
}
|
||||
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NearMaxPriority);
|
||||
DEBUG_ONLY(_instance = thread;)
|
||||
}
|
||||
|
||||
static void cleanup_oopstorages() {
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2011, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2011, 2021, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -36,7 +36,7 @@ class JvmtiDeferredEvent;
|
||||
class ServiceThread : public JavaThread {
|
||||
friend class VMStructs;
|
||||
private:
|
||||
static ServiceThread* _instance;
|
||||
DEBUG_ONLY(static JavaThread* _instance;)
|
||||
static JvmtiDeferredEvent* _jvmti_event;
|
||||
static JvmtiDeferredEventQueue _jvmti_service_queue;
|
||||
|
||||
|
@ -3900,3 +3900,44 @@ void JavaThread::verify_cross_modify_fence_failure(JavaThread *thread) {
|
||||
report_vm_error(__FILE__, __LINE__, "Cross modify fence failure", "%p", thread);
|
||||
}
|
||||
#endif
|
||||
|
||||
// Starts the target JavaThread as a daemon of the given priority, and
|
||||
// bound to the given java.lang.Thread instance.
|
||||
// The Threads_lock is held for the duration.
|
||||
void JavaThread::start_internal_daemon(JavaThread* current, JavaThread* target,
|
||||
Handle thread_oop, ThreadPriority prio) {
|
||||
|
||||
assert(target->osthread()!= NULL, "target thread is not properly initialized");
|
||||
|
||||
MutexLocker mu(current, Threads_lock);
|
||||
|
||||
// Initialize the fields of the thread_oop first.
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), target); // isAlive == true now
|
||||
|
||||
if (prio != NoPriority) {
|
||||
java_lang_Thread::set_priority(thread_oop(), prio);
|
||||
// Note: we don't call os::set_priority here. Possibly we should,
|
||||
// else all threads should call it themselves when they first run.
|
||||
}
|
||||
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
|
||||
// Now bind the thread_oop to the target JavaThread.
|
||||
target->set_threadObj(thread_oop());
|
||||
|
||||
Threads::add(target); // target is now visible for safepoint/handshake
|
||||
Thread::start(target);
|
||||
}
|
||||
|
||||
void JavaThread::vm_exit_on_osthread_failure(JavaThread* thread) {
|
||||
// At this point it may be possible that no osthread was created for the
|
||||
// JavaThread due to lack of resources. However, since this must work
|
||||
// for critical system threads just check and abort if this fails.
|
||||
if (thread->osthread() == nullptr) {
|
||||
// This isn't really an OOM condition, but historically this is what
|
||||
// we report.
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
}
|
||||
|
@ -54,7 +54,6 @@
|
||||
#include "jfr/support/jfrThreadExtension.hpp"
|
||||
#endif
|
||||
|
||||
|
||||
class SafeThreadsListPtr;
|
||||
class ThreadSafepointState;
|
||||
class ThreadsList;
|
||||
@ -1594,6 +1593,15 @@ public:
|
||||
static OopStorage* thread_oop_storage();
|
||||
|
||||
static void verify_cross_modify_fence_failure(JavaThread *thread) PRODUCT_RETURN;
|
||||
|
||||
// Helper function to start a VM-internal daemon thread.
|
||||
// E.g. ServiceThread, NotificationThread, CompilerThread etc.
|
||||
static void start_internal_daemon(JavaThread* current, JavaThread* target,
|
||||
Handle thread_oop, ThreadPriority prio);
|
||||
|
||||
// Helper function to do vm_exit_on_initialization for osthread
|
||||
// resource allocation failure.
|
||||
static void vm_exit_on_osthread_failure(JavaThread* thread);
|
||||
};
|
||||
|
||||
// The active thread queue. It also keeps track of the current used
|
||||
|
@ -484,22 +484,10 @@ void AttachListener::init() {
|
||||
return;
|
||||
}
|
||||
|
||||
{ MutexLocker mu(THREAD, Threads_lock);
|
||||
JavaThread* listener_thread = new JavaThread(&attach_listener_thread_entry);
|
||||
JavaThread* thread = new JavaThread(&attach_listener_thread_entry);
|
||||
JavaThread::vm_exit_on_osthread_failure(thread);
|
||||
|
||||
// Check that thread and osthread were created
|
||||
if (listener_thread == NULL || listener_thread->osthread() == NULL) {
|
||||
vm_exit_during_initialization("java.lang.OutOfMemoryError",
|
||||
os::native_thread_creation_failed_msg());
|
||||
}
|
||||
|
||||
java_lang_Thread::set_thread(thread_oop(), listener_thread);
|
||||
java_lang_Thread::set_daemon(thread_oop());
|
||||
|
||||
listener_thread->set_threadObj(thread_oop());
|
||||
Threads::add(listener_thread);
|
||||
Thread::start(listener_thread);
|
||||
}
|
||||
JavaThread::start_internal_daemon(THREAD, thread, thread_oop, NoPriority);
|
||||
}
|
||||
|
||||
// Performs clean-up tasks on platforms where we can detect that the last
|
||||
|
Loading…
x
Reference in New Issue
Block a user