From f0f6b0d919266b90bd91971f3cdb915f4e8c5300 Mon Sep 17 00:00:00 2001 From: Jan Lahoda Date: Thu, 29 Apr 2021 14:55:28 +0000 Subject: [PATCH] 8266027: The diamond finder does not find diamond candidates in field initializers Reviewed-by: jfranck, vromero --- .../com/sun/tools/javac/comp/Analyzer.java | 24 +++++++++++++-- .../tools/javac/analyzer/DiamondFields.java | 29 +++++++++++++++++++ .../tools/javac/analyzer/DiamondFields.out | 12 ++++++++ 3 files changed, 63 insertions(+), 2 deletions(-) create mode 100644 test/langtools/tools/javac/analyzer/DiamondFields.java create mode 100644 test/langtools/tools/javac/analyzer/DiamondFields.out diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java index a2d4e4727d1..9d94f67add3 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java @@ -35,6 +35,7 @@ import java.util.stream.Collectors; import com.sun.source.tree.LambdaExpressionTree; import com.sun.source.tree.NewClassTree; +import com.sun.source.tree.VariableTree; import com.sun.tools.javac.code.Flags; import com.sun.tools.javac.code.Kinds.Kind; import com.sun.tools.javac.code.Source; @@ -556,14 +557,17 @@ public class Analyzer { log.useSource(rewriting.env.toplevel.getSourceFile()); JCStatement treeToAnalyze = (JCStatement)rewriting.originalTree; + JCTree wrappedTree = null; + if (rewriting.env.info.scope.owner.kind == Kind.TYP) { //add a block to hoist potential dangling variable declarations treeToAnalyze = make.at(Position.NOPOS) .Block(Flags.SYNTHETIC, List.of((JCStatement)rewriting.originalTree)); + wrappedTree = rewriting.originalTree; } //TODO: to further refine the analysis, try all rewriting combinations - deferredAttr.attribSpeculative(treeToAnalyze, rewriting.env, attr.statInfo, new TreeRewriter(rewriting), + deferredAttr.attribSpeculative(treeToAnalyze, rewriting.env, attr.statInfo, new TreeRewriter(rewriting, wrappedTree), () -> rewriting.diagHandler(), AttributionMode.ANALYZER, argumentAttr.withLocalCacheContext()); rewriting.analyzer.process(rewriting.oldTree, rewriting.replacement, rewriting.erroneous); } catch (Throwable ex) { @@ -768,9 +772,11 @@ public class Analyzer { class TreeRewriter extends AnalyzerCopier { RewritingContext rewriting; + JCTree wrappedTree; - TreeRewriter(RewritingContext rewriting) { + TreeRewriter(RewritingContext rewriting, JCTree wrappedTree) { this.rewriting = rewriting; + this.wrappedTree = wrappedTree; } @Override @@ -783,5 +789,19 @@ public class Analyzer { } return newTree; } + + @Override + public JCTree visitVariable(VariableTree node, Void p) { + JCTree result = super.visitVariable(node, p); + if (node == wrappedTree) { + //The current tree is a field and has been wrapped by a block, so it effectivelly + //became local variable. If it has some modifiers (except for final), an error + //would be reported, causing the whole rewrite to fail. Removing the non-final + //modifiers from the variable here: + ((JCVariableDecl) result).mods.flags &= Flags.FINAL; + } + return result; + } + } } diff --git a/test/langtools/tools/javac/analyzer/DiamondFields.java b/test/langtools/tools/javac/analyzer/DiamondFields.java new file mode 100644 index 00000000000..291021c41d1 --- /dev/null +++ b/test/langtools/tools/javac/analyzer/DiamondFields.java @@ -0,0 +1,29 @@ +/** + * @test /nodynamiccopyright/ + * @bug 8266027 + * @summary Verify the diamond finder works on fields with modifiers. + * @compile/ref=DiamondFields.out -XDfind=diamond -XDrawDiagnostics DiamondFields.java + */ + +import java.util.LinkedList; +import java.util.List; + +public class DiamondFields { + List f1 = new LinkedList(); + private List f2 = new LinkedList(); + static List f3 = new LinkedList(); + @Deprecated List f4 = new LinkedList(); + final List f5 = new LinkedList(); + + DiamondFields() { + List l1 = new LinkedList(); + final List l2 = new LinkedList(); + @Deprecated List l3 = new LinkedList(); + } + + void t() { + List l1 = new LinkedList(); + final List l2 = new LinkedList(); + @Deprecated List l3 = new LinkedList(); + } +} diff --git a/test/langtools/tools/javac/analyzer/DiamondFields.out b/test/langtools/tools/javac/analyzer/DiamondFields.out new file mode 100644 index 00000000000..ddbc5a39334 --- /dev/null +++ b/test/langtools/tools/javac/analyzer/DiamondFields.out @@ -0,0 +1,12 @@ +DiamondFields.java:12:49: compiler.warn.diamond.redundant.args +DiamondFields.java:13:49: compiler.warn.diamond.redundant.args +DiamondFields.java:14:49: compiler.warn.diamond.redundant.args +DiamondFields.java:15:49: compiler.warn.diamond.redundant.args +DiamondFields.java:16:49: compiler.warn.diamond.redundant.args +DiamondFields.java:19:41: compiler.warn.diamond.redundant.args +DiamondFields.java:20:47: compiler.warn.diamond.redundant.args +DiamondFields.java:21:53: compiler.warn.diamond.redundant.args +DiamondFields.java:25:41: compiler.warn.diamond.redundant.args +DiamondFields.java:26:47: compiler.warn.diamond.redundant.args +DiamondFields.java:27:53: compiler.warn.diamond.redundant.args +11 warnings