-
Notifications
You must be signed in to change notification settings - Fork 139
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Performance] Full build takes more time since 2024-09 (type inference) #3387
[Performance] Full build takes more time since 2024-09 (type inference) #3387
Conversation
(eclipse-jdt#3384) optimize 3: + never visit the same super type more than once Fixes eclipse-jdt#3327
For a moment I was tricked by the discussion about The requirement put forward by JLS (and thus not subject to our choosing) is similar to What still looks like some combinatorial explosion is in fact a difference between the number of paths vs nodes in the inheritance hierarchy. Viz, the original implementation traversed all paths from Limiting that traversal is what the cache achieves. In contrast to #3333 (comment) no cache of pairs is needed, since Additionally, using I'm still a bit uneasy about using |
FWIW, using the reproducer my poor man's measurements yield:
which is not such a significant gain over results from #3384:
|
@fedejeanne I marked this PR as draft, because I'd like to get feedback on results from #3384 in real life, before stacking more speculative optimization on top of the previous one. I mean we've seen the differences in counts, but not so much in compile time. |
@stephan-herrmann understood, I was about to check the performance gain anyway when I read your comments :-) I am setting up my IDE right now and I will measure performance with #3384 right away. |
|
FTR, even in its current state and with the bad implementations of I just happened to have VisualVM still open so I am adding also a screenshot of the Monitor view. FYI the build took time between ~9:23 to ~9:35 (sorry that I can not be more precise, I wasn't looking at the watch). This is the sample (sampled every 20ms): v20241203-1907 and 3387-Full build (clean workspace).zip And one can see that the hotspots changed:
|
Good, that's what i expected. BTW: you should not trust the Heap/Memory view while you are sampling. Sampling does too much memory overhead. |
I wasn't aware of that, thank you for the tip!
Hm, I didn't know that either. Do you have any hints about what should I look for when debugging these methods? I honestly assumed that these hotspots were "normal" since they were already there in the sampling I did with the good old fast 2024-06 in the original issue (#3327 (comment)) Also, this PR brings down the number of invocations on those 2 methods almost to those of 2024-06 so I don't know how much more there is to improve in there. I can certainly take a look at them after merging this PR though (since this PR improves both of them, I mean). I am always happy to help track and improve performance bottlenecks :-) |
no modification in sight
Please let's be extremely careful about "fixing". The current implementation is necessary for semantic correctness and has served this purpose without any reported issues for 20 years. In fact, scanning the compiler code, there are several locations using TypeBinding as keys in a HashMap. Again with no reported anomalies. Given those precedents, I'll stop worrying at this particular location. If desired, the general topic can be picked up in a follow-up ticket. |
I'm not sure I'm following the reasoning here. Perhaps there's an unintended "not" involved? Or should it read except instead of expect? As you see I'm clueless :) |
So, in the real world case there seems to be something that is not captured in the reproducer. With this gain I agree that there is value in the 3rd level optimization and will merge it, so it will get thorough field testing before release. |
Great, thank you! I'm looking forward to tomorrow's I-BUILD and I will use it ASAP. |
@@ -1277,11 +1283,11 @@ protected List<Pair<TypeBinding>> allSuperPairsWithCommonGenericType(TypeBinding | |||
if (tSuper != null && s.isParameterizedType() && tSuper.isParameterizedType()) { // optimization #1 again |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Assuming s.isParameterizedType
is a very cheap operation and findSuperTypeOriginatingFrom
is not:
The lookup of tSuper can be guarded by s.isParameterizedType():
TypeBinding tSuper = s.isParameterizedType() ? t.findSuperTypeOriginatingFrom(s) : null;
if (tSuper != null && tSuper.isParameterizedType()) {
result.add(new Pair<>(s, tSuper));
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can try this tomorrow and report back. Thank you for the hint!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Incredible... I had to sample twice because I couldn't believe my eyes 😄
Thank you, @szarnekow ! --> #3411
"equals() and hashCode() should always be in sync" - this really means when equals() is overridden, it is extremely likely a defect to not override hashCode() but not necessarily the other way around - no ? Particularly when equals() is coming from j.l.O ?? |
If equals() is overridden but hashCode() is not then it most likely violates the contract.
Which is impossible when using runtime depended hashcode from j.i.O. |
FWIW this is how it looks like in our big WS (Sample from #3411) JDT is in green. For what I see in these pictures (I haven't looked into the code in depth) If you guys want me to test something in our big WS, just let me know and I'll fire up VisualVM with any configuration you want :-) |
@fedejeanne i'd say you waste 11s in org.eclipse.jdt.internal.compiler.lookup.InferenceVariable.hashCode() |
You meant 11 ms - right ? |
yes, sorry. |
I would urge that we base any decisions on real data as opposed to intuition such as "will force performance to explode" - Even super talented engineers can be completely off when it comes to speculating about performance bottlenecks! :) This particular hashCode implementation dates back to 2004 and we haven't had complaints |
No no, that's actually 11 s. It's a "digit grouping symbol" and not a "Decimal symbol" ... I told you, it's a big WS 😅
With this debug output... ... I get always this stack trace for the initial 10% of the compilation: java.lang.RuntimeException: InferenceVariable.InferenceVarKey.hashCode() = <some_hashcode>
at org.eclipse.jdt.internal.compiler.lookup.InferenceVariable$InferenceVarKey.hashCode(InferenceVariable.java:42)
at java.base/java.util.HashMap.hash(HashMap.java:338)
at java.base/java.util.HashMap.put(HashMap.java:618)
at org.eclipse.jdt.internal.compiler.lookup.InferenceVariable.get(InferenceVariable.java:80)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.addInitialTypeVariableSubstitutions(InferenceContext18.java:307)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.createInitialBoundSet(InferenceContext18.java:224)
at org.eclipse.jdt.internal.compiler.lookup.ConstraintExpressionFormula.inferInvocationApplicability(ConstraintExpressionFormula.java:387)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.inferInvocationApplicability(InferenceContext18.java:369)
at org.eclipse.jdt.internal.compiler.lookup.ParameterizedGenericMethodBinding.computeCompatibleMethod18(ParameterizedGenericMethodBinding.java:252)
at org.eclipse.jdt.internal.compiler.lookup.ParameterizedGenericMethodBinding.computeCompatibleMethod(ParameterizedGenericMethodBinding.java:92)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:849)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:806)
at org.eclipse.jdt.internal.compiler.lookup.Scope.findMethod0(Scope.java:1775)
at org.eclipse.jdt.internal.compiler.lookup.Scope.findMethod(Scope.java:1677)
at org.eclipse.jdt.internal.compiler.lookup.Scope.getImplicitMethod(Scope.java:2669)
at org.eclipse.jdt.internal.compiler.ast.MessageSend.findMethodBinding(MessageSend.java:1136)
at org.eclipse.jdt.internal.compiler.ast.MessageSend.resolveType(MessageSend.java:872)
at org.eclipse.jdt.internal.compiler.ast.LocalDeclaration.resolve(LocalDeclaration.java:401)
at org.eclipse.jdt.internal.compiler.ast.LocalDeclaration.resolve(LocalDeclaration.java:257)
at org.eclipse.jdt.internal.compiler.ast.Statement.resolveWithBindings(Statement.java:507)
at org.eclipse.jdt.internal.compiler.ast.ASTNode.resolveStatements(ASTNode.java:735)
at org.eclipse.jdt.internal.compiler.ast.AbstractMethodDeclaration.resolveStatements(AbstractMethodDeclaration.java:721)
at org.eclipse.jdt.internal.compiler.ast.MethodDeclaration.resolveStatements(MethodDeclaration.java:397)
at org.eclipse.jdt.internal.compiler.ast.AbstractMethodDeclaration.resolve(AbstractMethodDeclaration.java:619)
at org.eclipse.jdt.internal.compiler.ast.TypeDeclaration.resolve(TypeDeclaration.java:1556)
at org.eclipse.jdt.internal.compiler.ast.TypeDeclaration.resolve(TypeDeclaration.java:1685)
at org.eclipse.jdt.internal.compiler.ast.CompilationUnitDeclaration.resolve(CompilationUnitDeclaration.java:661)
at org.eclipse.jdt.internal.compiler.Compiler.process(Compiler.java:957)
at org.eclipse.jdt.internal.compiler.ProcessTaskManager.processing(ProcessTaskManager.java:135)
at java.base/java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:572)
at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:317)
at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1144)
at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:642)
at java.base/java.lang.Thread.run(Thread.java:1583)
... Raw output: first10percent.txt |
Here's the output for the last 10%: last10percent.txt There is a bit more variation in there, e.g. this stack trace: java.lang.RuntimeException: InferenceVariable.InferenceVarKey.hashCode() = 15763705
at org.eclipse.jdt.internal.compiler.lookup.InferenceVariable$InferenceVarKey.hashCode(InferenceVariable.java:42)
at java.base/java.util.HashMap.hash(HashMap.java:338)
at java.base/java.util.HashMap.getNode(HashMap.java:576)
at java.base/java.util.HashMap.get(HashMap.java:564)
at org.eclipse.jdt.internal.compiler.lookup.InferenceVariable.get(InferenceVariable.java:74)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.addInitialTypeVariableSubstitutions(InferenceContext18.java:307)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.createInitialBoundSet(InferenceContext18.java:224)
at org.eclipse.jdt.internal.compiler.lookup.ConstraintExpressionFormula.inferInvocationApplicability(ConstraintExpressionFormula.java:387)
at org.eclipse.jdt.internal.compiler.lookup.InferenceContext18.inferInvocationApplicability(InferenceContext18.java:369)
at org.eclipse.jdt.internal.compiler.lookup.ParameterizedGenericMethodBinding.computeCompatibleMethod18(ParameterizedGenericMethodBinding.java:252)
at org.eclipse.jdt.internal.compiler.lookup.ParameterizedGenericMethodBinding.computeCompatibleMethod(ParameterizedGenericMethodBinding.java:92)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:849)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:806)
at org.eclipse.jdt.internal.compiler.lookup.Scope.getStaticFactory(Scope.java:5489)
at org.eclipse.jdt.internal.compiler.ast.AllocationExpression.inferDiamondConstructor(AllocationExpression.java:656)
at org.eclipse.jdt.internal.compiler.ast.AllocationExpression.inferConstructorOfElidedParameterizedType(AllocationExpression.java:632)
at org.eclipse.jdt.internal.compiler.ast.AllocationExpression.isCompatibleWith(AllocationExpression.java:613)
at org.eclipse.jdt.internal.compiler.lookup.PolyTypeBinding.isCompatibleWith(PolyTypeBinding.java:42)
at org.eclipse.jdt.internal.compiler.lookup.Scope.parameterCompatibilityLevel(Scope.java:5261)
at org.eclipse.jdt.internal.compiler.lookup.Scope.parameterCompatibilityLevel(Scope.java:5220)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:881)
at org.eclipse.jdt.internal.compiler.lookup.Scope.computeCompatibleMethod(Scope.java:806)
at org.eclipse.jdt.internal.compiler.lookup.Scope.findMethod0(Scope.java:1775)
at org.eclipse.jdt.internal.compiler.lookup.Scope.findMethod(Scope.java:1677)
at org.eclipse.jdt.internal.compiler.lookup.Scope.getMethod(Scope.java:3161)
at org.eclipse.jdt.internal.compiler.ast.MessageSend.findMethodBinding(MessageSend.java:1137)
at org.eclipse.jdt.internal.compiler.ast.MessageSend.resolveType(MessageSend.java:872)
at org.eclipse.jdt.internal.compiler.ast.Expression.resolve(Expression.java:1119)
at org.eclipse.jdt.internal.compiler.ast.Statement.resolveWithBindings(Statement.java:507)
at org.eclipse.jdt.internal.compiler.ast.ASTNode.resolveStatements(ASTNode.java:735)
at org.eclipse.jdt.internal.compiler.ast.Block.resolve(Block.java:143)
at org.eclipse.jdt.internal.compiler.ast.Statement.resolveWithBindings(Statement.java:507)
at org.eclipse.jdt.internal.compiler.ast.IfStatement.resolve(IfStatement.java:316)
at org.eclipse.jdt.internal.compiler.ast.Statement.resolveWithBindings(Statement.java:507)
at org.eclipse.jdt.internal.compiler.ast.IfStatement.resolve(IfStatement.java:319)
at org.eclipse.jdt.internal.compiler.ast.Statement.resolveWithBindings(Statement.java:507)
at org.eclipse.jdt.internal.compiler.ast.ASTNode.resolveStatements(ASTNode.java:735)
at org.eclipse.jdt.internal.compiler.ast.AbstractMethodDeclaration.resolveStatements(AbstractMethodDeclaration.java:721)
at org.eclipse.jdt.internal.compiler.ast.MethodDeclaration.resolveStatements(MethodDeclaration.java:397)
at org.eclipse.jdt.internal.compiler.ast.AbstractMethodDeclaration.resolve(AbstractMethodDeclaration.java:619)
at org.eclipse.jdt.internal.compiler.ast.TypeDeclaration.resolve(TypeDeclaration.java:1556)
at org.eclipse.jdt.internal.compiler.ast.TypeDeclaration.resolve(TypeDeclaration.java:1685)
at org.eclipse.jdt.internal.compiler.ast.CompilationUnitDeclaration.resolve(CompilationUnitDeclaration.java:661)
at org.eclipse.jdt.internal.compiler.Compiler.process(Compiler.java:957)
at org.eclipse.jdt.internal.compiler.ProcessTaskManager.processing(ProcessTaskManager.java:135)
at java.base/java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:572)
at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:317)
at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1144)
at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:642)
at java.base/java.lang.Thread.run(Thread.java:1583) |
@fedejeanne can you measure with |
@fedejeanne that output is not what i meant. The File InferenceVariable contains two |
Are we barking up the wrong tree ? If hashCode() distribution is poor and results in collision shouldn't we looking at whether equals() dominates runtime ?? |
May depend on what the caller does. Lets wait for the stacktrace. |
that InferenceVariable.hashCode() may relate to #3590 (comment) |
Added: And got: Adding the line made the compilation extremely slow (as expected) so I aborted the compilation after 10 minutes (at 14%). Moving on to measuring the compilation time as requested in #3387 (comment). I will report back. |
@jukzi I sampled again after removing Here's the raw data: 20250120-1800 without hashCode and equals in InferenceVariable.zip |
I created a separate issue for InferenceVariable.hashCode(): #3593 |
that file reads like you still have a problem in org.eclipse.jdt.internal.compiler.lookup.TypeBinding.findSuperTypeOriginatingFrom(TypeBinding), probably due to your redundant superinterfaces: |
optimize 3:
Fixes #3327