From 47a084264698884cae544e85cee2712836d6fadb Mon Sep 17 00:00:00 2001 From: "Daniel D. Daugherty" Date: Fri, 26 Feb 2021 15:41:13 +0000 Subject: [PATCH] 8241403: JavaThread::get_thread_name() should be ThreadSMR-aware Reviewed-by: rehn, coleenp, dholmes --- src/hotspot/share/compiler/compileBroker.cpp | 12 ++-- src/hotspot/share/prims/jvm.cpp | 2 - src/hotspot/share/runtime/thread.cpp | 62 ++++++++++++++++---- src/hotspot/share/runtime/thread.hpp | 4 ++ 4 files changed, 63 insertions(+), 17 deletions(-) diff --git a/src/hotspot/share/compiler/compileBroker.cpp b/src/hotspot/share/compiler/compileBroker.cpp index 95a783820dd..129798a96c2 100644 --- a/src/hotspot/share/compiler/compileBroker.cpp +++ b/src/hotspot/share/compiler/compileBroker.cpp @@ -1006,7 +1006,8 @@ void CompileBroker::init_compiler_sweeper_threads() { _compilers[1]->set_num_compiler_threads(i + 1); if (TraceCompilerThreads) { ResourceMark rm; - MutexLocker mu(Threads_lock); + ThreadsListHandle tlh; // get_thread_name() depends on the TLH. + assert(tlh.includes(ct), "ct=" INTPTR_FORMAT " exited unexpectedly.", p2i(ct)); tty->print_cr("Added initial compiler thread %s", ct->get_thread_name()); } } @@ -1026,7 +1027,8 @@ void CompileBroker::init_compiler_sweeper_threads() { _compilers[0]->set_num_compiler_threads(i + 1); if (TraceCompilerThreads) { ResourceMark rm; - MutexLocker mu(Threads_lock); + ThreadsListHandle tlh; // get_thread_name() depends on the TLH. + assert(tlh.includes(ct), "ct=" INTPTR_FORMAT " exited unexpectedly.", p2i(ct)); tty->print_cr("Added initial compiler thread %s", ct->get_thread_name()); } } @@ -1112,7 +1114,8 @@ void CompileBroker::possibly_add_compiler_threads(Thread* THREAD) { _compilers[1]->set_num_compiler_threads(i + 1); if (TraceCompilerThreads) { ResourceMark rm; - MutexLocker mu(Threads_lock); + ThreadsListHandle tlh; // get_thread_name() depends on the TLH. + assert(tlh.includes(ct), "ct=" INTPTR_FORMAT " exited unexpectedly.", p2i(ct)); tty->print_cr("Added compiler thread %s (available memory: %dMB, available non-profiled code cache: %dMB)", ct->get_thread_name(), (int)(available_memory/M), (int)(available_cc_np/M)); } @@ -1132,7 +1135,8 @@ void CompileBroker::possibly_add_compiler_threads(Thread* THREAD) { _compilers[0]->set_num_compiler_threads(i + 1); if (TraceCompilerThreads) { ResourceMark rm; - MutexLocker mu(Threads_lock); + ThreadsListHandle tlh; // get_thread_name() depends on the TLH. + assert(tlh.includes(ct), "ct=" INTPTR_FORMAT " exited unexpectedly.", p2i(ct)); tty->print_cr("Added compiler thread %s (available memory: %dMB, available profiled code cache: %dMB)", ct->get_thread_name(), (int)(available_memory/M), (int)(available_cc_p/M)); } diff --git a/src/hotspot/share/prims/jvm.cpp b/src/hotspot/share/prims/jvm.cpp index 3a8ebd78808..3fea3a5743c 100644 --- a/src/hotspot/share/prims/jvm.cpp +++ b/src/hotspot/share/prims/jvm.cpp @@ -2933,8 +2933,6 @@ JVM_END // but is thought to be reliable and simple. In the case, where the receiver is the // same thread as the sender, no VM_Operation is needed. JVM_ENTRY(void, JVM_StopThread(JNIEnv* env, jobject jthread, jobject throwable)) - // A nested ThreadsListHandle will grab the Threads_lock so create - // tlh before we resolve throwable. ThreadsListHandle tlh(thread); oop java_throwable = JNIHandles::resolve(throwable); if (java_throwable == NULL) { diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index a9fa059cd42..c0d3b81575e 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -482,6 +482,50 @@ void Thread::check_for_dangling_thread_pointer(Thread *thread) { } #endif +// Is the target JavaThread protected by the calling Thread +// or by some other mechanism: +bool Thread::is_JavaThread_protected(const JavaThread* p) { + // Do the simplest check first: + if (SafepointSynchronize::is_at_safepoint()) { + // The target is protected since JavaThreads cannot exit + // while we're at a safepoint. + return true; + } + + // Now make the simple checks based on who the caller is: + Thread* current_thread = Thread::current(); + if (current_thread == p || Threads_lock->owner() == current_thread) { + // Target JavaThread is self or calling thread owns the Threads_lock. + // Second check is the same as Threads_lock->owner_is_self(), + // but we already have the current thread so check directly. + return true; + } + + // Check the ThreadsLists associated with the calling thread (if any) + // to see if one of them protects the target JavaThread: + for (SafeThreadsListPtr* stlp = current_thread->_threads_list_ptr; + stlp != NULL; stlp = stlp->previous()) { + if (stlp->list()->includes(p)) { + // The target JavaThread is protected by this ThreadsList: + return true; + } + } + + // Use this debug code with -XX:+UseNewCode to diagnose locations that + // are missing a ThreadsListHandle or other protection mechanism: + // guarantee(!UseNewCode, "current_thread=" INTPTR_FORMAT " is not protecting p=" + // INTPTR_FORMAT, p2i(current_thread), p2i(p)); + + // Note: Since 'p' isn't protected by a TLH, the call to + // p->is_handshake_safe_for() may crash, but we have debug bits so + // we'll be able to figure out what protection mechanism is missing. + assert(p->is_handshake_safe_for(current_thread), "JavaThread=" INTPTR_FORMAT + " is not protected and not handshake safe.", p2i(p)); + + // The target JavaThread is not protected so it is not safe to query: + return false; +} + ThreadPriority Thread::get_priority(const Thread* const thread) { ThreadPriority priority; // Can return an error! @@ -2509,21 +2553,17 @@ void JavaThread::verify() { // Most callers of this method assume that it can't return NULL but a // thread may not have a name whilst it is in the process of attaching to // the VM - see CR 6412693, and there are places where a JavaThread can be -// seen prior to having it's threadObj set (eg JNI attaching threads and +// seen prior to having its threadObj set (e.g., JNI attaching threads and // if vm exit occurs during initialization). These cases can all be accounted // for such that this method never returns NULL. const char* JavaThread::get_thread_name() const { -#ifdef ASSERT - // early safepoints can hit while current thread does not yet have TLS - if (!SafepointSynchronize::is_at_safepoint()) { - // Current JavaThreads are allowed to get their own name without - // the Threads_lock. - if (Thread::current() != this) { - assert_locked_or_safepoint_or_handshake(Threads_lock, this); - } + if (Thread::is_JavaThread_protected(this)) { + // The target JavaThread is protected so get_thread_name_string() is safe: + return get_thread_name_string(); } -#endif // ASSERT - return get_thread_name_string(); + + // The target JavaThread is not protected so we return the default: + return Thread::name(); } // Returns a non-NULL representation of this thread's name, or a suitable diff --git a/src/hotspot/share/runtime/thread.hpp b/src/hotspot/share/runtime/thread.hpp index 54cc4d38f30..1ac4a564196 100644 --- a/src/hotspot/share/runtime/thread.hpp +++ b/src/hotspot/share/runtime/thread.hpp @@ -201,6 +201,10 @@ class Thread: public ThreadShadow { } public: + // Is the target JavaThread protected by the calling Thread + // or by some other mechanism: + static bool is_JavaThread_protected(const JavaThread* p); + void* operator new(size_t size) throw() { return allocate(size, true); } void* operator new(size_t size, const std::nothrow_t& nothrow_constant) throw() { return allocate(size, false); }