8185005: Improve performance of ThreadMXBean.getThreadInfo(long ids[], int maxDepth)
Daniil Titov
daniil.x.titov at oracle.com
Mon Jul 29 15:37:25 UTC 2019
Hi Serguei,
Thank you for catching this!
Regarding the testing, at this stage, I run all tier1- tier3 tests . I will check why this specific case was not included in the tests and will add it if required.
Thanks,
Daniil
From: "serguei.spitsyn at oracle.com" <serguei.spitsyn at oracle.com>
Date: Monday, July 29, 2019 at 2:12 AM
To: Daniil Titov <daniil.x.titov at oracle.com>, <daniel.daugherty at oracle.com>, OpenJDK Serviceability <serviceability-dev at openjdk.java.net>, "hotspot-runtime-dev at openjdk.java.net" <hotspot-runtime-dev at openjdk.java.net>, "jmx-dev at openjdk.java.net" <jmx-dev at openjdk.java.net>, David Holmes <david.holmes at oracle.com>
Subject: Re: RFR: 8185005: Improve performance of ThreadMXBean.getThreadInfo(long ids[], int maxDepth)
Hi Daniil,
Probably, it'd make sense to re-iterate on this after you resolve David's comments.
Now, just one comment though as it looks dangerous.
http://cr.openjdk.java.net/~dtitov/8185005/webrev.03/src/hotspot/share/services/management.cpp.udiff.html
} else {
// reset contention statistics for a given thread
+ JavaThread* java_thread = NULL;
+ if (THREAD->is_Java_thread()) {
+ JavaThread* current_thread = (JavaThread*)THREAD;
+ if (tid == java_lang_Thread::thread_id(current_thread->threadObj())) {
+ java_thread = current_thread;
+ }
+ }
+ if (java_thread == NULL) {
JavaThread* java_thread = jtiwh.list()->find_JavaThread_from_java_tid(tid);
+ }
if (java_thread == NULL) {
return false;
}
Now, the definition of java_thread below:
+ if (java_thread == NULL) {
JavaThread* java_thread = jtiwh.list()->find_JavaThread_from_java_tid(tid);
+ }
becomes completely useless because the block of its definition is ended right away.
Here, overriding the local variable 'java_thread' does not look as a good idea.
Then how was it really tested?
Thanks,
Serguei
On 7/24/19 10:21, Daniil Titov wrote:
Hi David, Daniel, and Serguei,
Please review the new version of the fix, that makes the thread table initialization on demand and
moves it inside ThreadsList::find_JavaThread_from_java_tid(). At the creation time the thread table
is initialized with the threads from the current thread list. We don't want to hold Threads_lock
inside find_JavaThread_from_java_tid(), thus new threads still could be created while the thread
table is being initialized . Such threads will be found by the linear search and added to the thread table
later, in ThreadsList::find_JavaThread_from_java_tid().
The change also includes additional optimization for some callers of find_JavaThread_from_java_tid()
as Daniel suggested.
That is correct that ResolvedMethodTable was used as a blueprint for the thread table, however, I tried
to strip it of the all functionality that is not required in the thread table case.
We need to have the thread table resizable and allow it to grow as the number of threads increases to avoid
reserving excessive memory a-priori or deteriorating lookup times. The ServiceThread is responsible for
growing the thread table when required.
There is no ConcurrentHashTable available in Java 8 and for backporting this fix to Java 8 another implementation
of the hash table, probably originally suggested in the patch attached to the JBS issue, should be used. It will make
the backporting more complicated, however, adding a new Implementation of the hash table in Java 14 while it
already has ConcurrentHashTable doesn't seem reasonable for me.
Webrev: http://cr.openjdk.java.net/~dtitov/8185005/webrev.03
Bug: https://bugs.openjdk.java.net/browse/JDK-8185005
Thanks!
--Daniil
On 7/8/19, 3:24 PM, "Daniel D. Daugherty" <daniel.daugherty at oracle.com> wrote:
On 6/29/19 12:06 PM, Daniil Titov wrote:
> Hi Serguei and David,
>
> Serguei is right, ThreadTable::find_thread(java_tid) cannot return a JavaThread with an unmatched java_tid.
>
> Please find a new version of the fix that includes the changes Serguei suggested.
>
> Regarding the concern about the maintaining the thread table when it may never even be queried, one of
> the options could be to add ThreadTable ::isEnabled flag, set it to "false" by default, and wrap the calls to the thread table
> in ThreadsSMRSupport add_thread() and remove_thread() methods to check this flag.
>
> When ThreadsList::find_JavaThread_from_java_tid() is called for the first time it could check if ThreadTable ::isEnabled
> Is on and if not then set it on and populate the thread table with all existing threads from the thread list.
I have the same concerns as David H. about this new ThreadTable.
ThreadsList::find_JavaThread_from_java_tid() is only called from code
in src/hotspot/share/services/management.cpp so I think that table
needs to enabled and populated only if it is going to be used.
I've taken a look at the webrev below and I see that David has
followed up with additional comments. Before I do a crawl through
code review for this, I would like to see the ThreadTable stuff
made optional and David's other comments addressed.
Another possible optimization is for callers of
find_JavaThread_from_java_tid() to save the calling thread's
tid value before they loop and if the current tid == saved_tid
then use the current JavaThread* instead of calling
find_JavaThread_from_java_tid() to get the JavaThread*.
Dan
>
> Webrev: https://cr.openjdk.java.net/~dtitov/8185005/webrev.02/
> Bug: https://bugs.openjdk.java.net/browse/JDK-8185005
>
> Thanks!
> --Daniil
>
> From: <serguei.spitsyn at oracle.com>
> Organization: Oracle Corporation
> Date: Friday, June 28, 2019 at 7:56 PM
> To: Daniil Titov <daniil.x.titov at oracle.com>, OpenJDK Serviceability <serviceability-dev at openjdk.java.net>, "hotspot-runtime-dev at openjdk.java.net" <hotspot-runtime-dev at openjdk.java.net>, "jmx-dev at openjdk.java.net" <jmx-dev at openjdk.java.net>
> Subject: Re: RFR: 8185005: Improve performance of ThreadMXBean.getThreadInfo(long ids[], int maxDepth)
>
> Hi Daniil,
>
> I have several quick comments.
>
> The indent in the hotspot c/c++ files has to be 2, not 4.
>
> https://cr.openjdk.java.net/~dtitov/8185005/webrev.01/src/hotspot/share/runtime/threadSMR.cpp.frames.html
> 614 JavaThread* ThreadsList::find_JavaThread_from_java_tid(jlong java_tid) const {
> 615 JavaThread* java_thread = ThreadTable::find_thread(java_tid);
> 616 if (java_thread == NULL && java_tid == PMIMORDIAL_JAVA_TID) {
> 617 // ThreadsSMRSupport::add_thread() is not called for the primordial
> 618 // thread. Thus, we find this thread with a linear search and add it
> 619 // to the thread table.
> 620 for (uint i = 0; i < length(); i++) {
> 621 JavaThread* thread = thread_at(i);
> 622 if (is_valid_java_thread(java_tid,thread)) {
> 623 ThreadTable::add_thread(java_tid, thread);
> 624 return thread;
> 625 }
> 626 }
> 627 } else if (java_thread != NULL && is_valid_java_thread(java_tid, java_thread)) {
> 628 return java_thread;
> 629 }
> 630 return NULL;
> 631 }
> 632 bool ThreadsList::is_valid_java_thread(jlong java_tid, JavaThread* java_thread) {
> 633 oop tobj = java_thread->threadObj();
> 634 // Ignore the thread if it hasn't run yet, has exited
> 635 // or is starting to exit.
> 636 return (tobj != NULL && !java_thread->is_exiting() &&
> 637 java_tid == java_lang_Thread::thread_id(tobj));
> 638 }
>
> 615 JavaThread* java_thread = ThreadTable::find_thread(java_tid);
>
> I'd suggest to rename find_thread() to find_thread_by_tid().
>
> A space is missed after the comma:
> 622 if (is_valid_java_thread(java_tid,thread)) {
>
> An empty line is needed before L632.
>
> The name 'is_valid_java_thread' looks wrong (or confusing) to me.
> Something like 'is_alive_java_thread_with_tid()' would be better.
> It'd better to list parameters in the opposite order.
>
> The call to is_valid_java_thread() is confusing:
> 627 } else if (java_thread != NULL && is_valid_java_thread(java_tid, java_thread)) {
>
> Why would the call ThreadTable::find_thread(java_tid) return a JavaThread with an unmatched java_tid?
>
>
> Thanks,
> Serguei
>
> On 6/28/19, 9:40 PM, "David Holmes" <david.holmes at oracle.com> wrote:
>
> Hi Daniil,
>
> The definition and use of this hashtable (yet another hashtable
> implementation!) will need careful examination. We have to be concerned
> about the cost of maintaining it when it may never even be queried. You
> would need to look at footprint cost and performance impact.
>
> Unfortunately I'm just about to board a plane and will be out for the
> next few days. I will try to look at this asap next week, but we will
> need a lot more data on it.
>
> Thanks,
> David
>
> On 6/28/19 3:31 PM, Daniil Titov wrote:
> Please review the change that improves performance of ThreadMXBean MXBean methods returning the
> information for specific threads. The change introduces the thread table that uses ConcurrentHashTable
> to store one-to-one the mapping between the thread ids and JavaThread objects and replaces the linear
> search over the thread list in ThreadsList::find_JavaThread_from_java_tid(jlong tid) method with the lookup
> in the thread table.
>
> Testing: Mach5 tier1,tier2 and tier3 tests successfully passed.
>
> Webrev: https://cr.openjdk.java.net/~dtitov/8185005/webrev.01/
> Bug: https://bugs.openjdk.java.net/browse/JDK-8185005
>
> Thanks!
>
> Best regards,
> Daniil
>
>
>
>
>
>
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.java.net/pipermail/serviceability-dev/attachments/20190729/8ad10ef0/attachment-0001.html>
More information about the serviceability-dev
mailing list