RFR: 8221393: ResolvedMethodTable too small for StackWalking applications

coleen.phillimore at oracle.com coleen.phillimore at oracle.com
Tue Apr 9 14:29:32 UTC 2019


http://cr.openjdk.java.net/~stefank/8221393/webrev.01/src/hotspot/share/classfile/javaClasses.cpp.udiff.html

+ if (method->is_old()) {
+ // Replace method with redefined version
+
+ method = holder->method_with_idnum(method->method_idnum());
+ if (method == NULL) {
+ // Replace deleted method with NSME.
+ method = Universe::throw_no_such_method_error();
+ }
+ }


This code that came from resolvedMethodTable.cpp was just changed with 
https://bugs.openjdk.java.net/browse/JDK-8221992.  Can you use the new 
code here.

But make sure this holder is the one of the new method selected:

+ holder->set_has_resolved_methods();


http://cr.openjdk.java.net/~stefank/8221393/webrev.01/test/hotspot/jtreg/runtime/testlibrary/ClassWithManyMethodsClassLoader.java.html

    2  * Copyright (c) 2013, 2018, Oracle and/or its affiliates. All rights reserved.


Copyright is wrong.   Looks like a good utility class we can use.

The rest looks great!


On 4/5/19 8:24 AM, Stefan Karlsson wrote:
> Hi all,
>
> Please review this patch to replace the hashtable in 
> ResolvedMethodTable, with the ConcurrentHashTable (currently used by 
> the StringTable and SymbolTable).
>
> http://cr.openjdk.java.net/~stefank/8221393/webrev.01/
> https://bugs.openjdk.java.net/browse/JDK-8221393
>
> This solves at least two scalability problems with the 
> ResolvedMethodTable:
>
> 1) The old table had a fixed, small size that could cause the table to 
> be overpopulated when StackWalking APIs were used. See the bug report. 
> The new table automatically grows when it gets filled up.
>
> 2) The old table used locks for inserts, lookups, and cleanups. This 
> could cause large performance degradations when multiple threads used 
> the StackWalking APIs, or resolved MethodHandle constants. The new 
> table uses lock-free inserts and lookups, and thereby scales much better.
>
> A stack walking micro benchmark (not yet pushed) shows this.
>
> Running with 8 threads, on my 16 core machine:
> java -XX:+UseG1GC -Xmx1g -Xms1g -jar benchmarks.jar -wi 0 -i 5 -p 
> depth=4 -t 8 StackWalkBench.swClassNamesDefaultOpts
>
> Baseline: avgt 25 91777.142 ± 2760.507  ns/op
> Change:   avgt 25  8149.368 ± 212.863   ns/op
>
> Note that a lot of the code to use the ConcurrentHashTable is copied 
> from the StringTable and/or SymbolTable. There are unifications, and 
> maybe simplifications that we can do as follow-up RFEs.
>

I agree.  I'll file some RFEs unless you do it.

> There's an open question regarding the verification code in 
> ResolvedMethodTable::finish_dead_counter(). It's a quadratic 
> verification operation, and has the potential to take a long time. I 
> left it in for testing, but I don't think we should have it always 
> enabled. For the StringTable we have VerifyStringTableAtExit, to do 
> verification before the JVM shuts down. Any suggestions on what to do 
> for the ResolvedMethodTable?
>

For the most part, RMT is pretty small unless stackwalk API is used.  I 
would take the verification out of GC, since we don't verify string 
table that way.  Since VerifyStringTableAtExit is diagnostic, we could 
remove it and make it VerifyConcurrentHashtablesAtExit.  I'm not sure 
how useful it is here.  Maybe we could VerifyConcurrentHashtables at 
full GCs instead?

Thank you for making this change!
Coleen

> Tested with tier1-3, tier1-7 on Linux.
>
> Thanks,
> StefanK



More information about the hotspot-dev mailing list