From 806a31318fcdb70d0fcc7ac5877d2e207fdbd013 Mon Sep 17 00:00:00 2001 From: mcimadamore Date: Tue, 11 Aug 2009 01:13:14 +0100 Subject: [PATCH] 6521805: Regression: JDK5/JDK6 javac allows write access to outer class reference Summary: javac should warn/complain about identifiers with the same name as synthetic symbol Reviewed-by: jjg --- .../com/sun/tools/javac/comp/Check.java | 31 +++++++++ .../com/sun/tools/javac/comp/Lower.java | 63 +++++++++++++------ .../sun/tools/javac/main/JavaCompiler.java | 22 +++++-- .../tools/javac/resources/compiler.properties | 2 + test/tools/javac/6521805/T6521805a.java | 45 +++++++++++++ test/tools/javac/6521805/T6521805a_1.out | 2 + test/tools/javac/6521805/T6521805a_2.out | 2 + test/tools/javac/6521805/T6521805b.java | 44 +++++++++++++ test/tools/javac/6521805/T6521805c.java | 44 +++++++++++++ test/tools/javac/6521805/T6521805d.java | 50 +++++++++++++++ test/tools/javac/6521805/T6521805d.out | 2 + test/tools/javac/6521805/T6521805e.java | 32 ++++++++++ test/tools/javac/6521805/T6521805e.out | 2 + test/tools/javac/6521805/p/Outer.java | 5 ++ test/tools/javac/6521805/p/Sub.java | 13 ++++ test/tools/javac/6734819/T6734819a.out | 4 +- test/tools/javac/6734819/T6734819b.out | 2 +- test/tools/javac/policy/test2/byfile.AB.out | 2 +- test/tools/javac/policy/test2/bytodo.AB.out | 2 +- 19 files changed, 339 insertions(+), 30 deletions(-) create mode 100644 test/tools/javac/6521805/T6521805a.java create mode 100644 test/tools/javac/6521805/T6521805a_1.out create mode 100644 test/tools/javac/6521805/T6521805a_2.out create mode 100644 test/tools/javac/6521805/T6521805b.java create mode 100644 test/tools/javac/6521805/T6521805c.java create mode 100644 test/tools/javac/6521805/T6521805d.java create mode 100644 test/tools/javac/6521805/T6521805d.out create mode 100644 test/tools/javac/6521805/T6521805e.java create mode 100644 test/tools/javac/6521805/T6521805e.out create mode 100644 test/tools/javac/6521805/p/Outer.java create mode 100644 test/tools/javac/6521805/p/Sub.java diff --git a/src/share/classes/com/sun/tools/javac/comp/Check.java b/src/share/classes/com/sun/tools/javac/comp/Check.java index d46a5070..337dce9d 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -65,6 +65,7 @@ public class Check { private final Types types; private final JCDiagnostic.Factory diags; private final boolean skipAnnotations; + private boolean warnOnSyntheticConflicts; private final TreeInfo treeinfo; // The set of lint options currently in effect. It is initialized @@ -99,6 +100,7 @@ public class Check { allowAnnotations = source.allowAnnotations(); complexInference = options.get("-complexinference") != null; skipAnnotations = options.get("skipAnnotations") != null; + warnOnSyntheticConflicts = options.get("warnOnSyntheticConflicts") != null; boolean verboseDeprecated = lint.isEnabled(LintCategory.DEPRECATION); boolean verboseUnchecked = lint.isEnabled(LintCategory.UNCHECKED); @@ -1709,6 +1711,35 @@ public class Check { checkCompatibleConcretes(pos, c); } + void checkConflicts(DiagnosticPosition pos, Symbol sym, TypeSymbol c) { + for (Type ct = c.type; ct != Type.noType ; ct = types.supertype(ct)) { + for (Scope.Entry e = ct.tsym.members().lookup(sym.name); e.scope == ct.tsym.members(); e = e.next()) { + // VM allows methods and variables with differing types + if (sym.kind == e.sym.kind && + types.isSameType(types.erasure(sym.type), types.erasure(e.sym.type)) && + sym != e.sym && + (sym.flags() & Flags.SYNTHETIC) != (e.sym.flags() & Flags.SYNTHETIC) && + (sym.flags() & BRIDGE) == 0 && (e.sym.flags() & BRIDGE) == 0) { + syntheticError(pos, (e.sym.flags() & SYNTHETIC) == 0 ? e.sym : sym); + return; + } + } + } + } + + /** Report a conflict between a user symbol and a synthetic symbol. + */ + private void syntheticError(DiagnosticPosition pos, Symbol sym) { + if (!sym.type.isErroneous()) { + if (warnOnSyntheticConflicts) { + log.warning(pos, "synthetic.name.conflict", sym, sym.location()); + } + else { + log.error(pos, "synthetic.name.conflict", sym, sym.location()); + } + } + } + /** Check that class c does not implement directly or indirectly * the same parameterized interface with two different argument lists. * @param pos Position to be used for error reporting. diff --git a/src/share/classes/com/sun/tools/javac/comp/Lower.java b/src/share/classes/com/sun/tools/javac/comp/Lower.java index 03fb500b..ad6b049d 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Lower.java +++ b/src/share/classes/com/sun/tools/javac/comp/Lower.java @@ -596,35 +596,57 @@ public class Lower extends TreeTranslator { * Symbol manipulation utilities *************************************************************************/ - /** Report a conflict between a user symbol and a synthetic symbol. - */ - private void duplicateError(DiagnosticPosition pos, Symbol sym) { - if (!sym.type.isErroneous()) { - log.error(pos, "synthetic.name.conflict", sym, sym.location()); - } - } - /** Enter a synthetic symbol in a given scope, but complain if there was already one there. * @param pos Position for error reporting. * @param sym The symbol. * @param s The scope. */ private void enterSynthetic(DiagnosticPosition pos, Symbol sym, Scope s) { - if (sym.name != names.error && sym.name != names.empty) { - for (Scope.Entry e = s.lookup(sym.name); e.scope == s; e = e.next()) { - if (sym != e.sym && sym.kind == e.sym.kind) { - // VM allows methods and variables with differing types - if ((sym.kind & (MTH|VAR)) != 0 && - !types.erasure(sym.type).equals(types.erasure(e.sym.type))) - continue; - duplicateError(pos, e.sym); - break; - } - } - } s.enter(sym); } + /** Check whether synthetic symbols generated during lowering conflict + * with user-defined symbols. + * + * @param translatedTrees lowered class trees + */ + void checkConflicts(List translatedTrees) { + for (JCTree t : translatedTrees) { + t.accept(conflictsChecker); + } + } + + JCTree.Visitor conflictsChecker = new TreeScanner() { + + TypeSymbol currentClass; + + @Override + public void visitMethodDef(JCMethodDecl that) { + chk.checkConflicts(that.pos(), that.sym, currentClass); + super.visitMethodDef(that); + } + + @Override + public void visitVarDef(JCVariableDecl that) { + if (that.sym.owner.kind == TYP) { + chk.checkConflicts(that.pos(), that.sym, currentClass); + } + super.visitVarDef(that); + } + + @Override + public void visitClassDef(JCClassDecl that) { + TypeSymbol prevCurrentClass = currentClass; + currentClass = that.sym; + try { + super.visitClassDef(that); + } + finally { + currentClass = prevCurrentClass; + } + } + }; + /** Look up a synthetic name in a given scope. * @param scope The scope. * @param name The name. @@ -3192,6 +3214,7 @@ public class Lower extends TreeTranslator { makeAccessible(l.head); for (EnumMapping map : enumSwitchMap.values()) map.translate(); + checkConflicts(this.translated.toList()); translated = this.translated; } finally { // note that recursive invocations of this method fail hard diff --git a/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java b/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java index 2de2c537..e93d82fd 100644 --- a/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java +++ b/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java @@ -1207,6 +1207,9 @@ public class JavaCompiler implements ClassReader.SourceCompleter { return stopIfError(CompileState.FLOW, results); } + HashMap, Queue, JCClassDecl>>> desugaredEnvs = + new HashMap, Queue, JCClassDecl>>>(); + /** * Prepare attributed parse trees, in conjunction with their attribution contexts, * for source or code generation. If the file was not listed on the command line, @@ -1222,10 +1225,17 @@ public class JavaCompiler implements ClassReader.SourceCompleter { return; } + if (compileStates.isDone(env, CompileState.LOWER)) { + results.addAll(desugaredEnvs.get(env)); + return; + } + /** - * As erasure (TransTypes) destroys information needed in flow analysis, - * including information in supertypes, we need to ensure that supertypes - * are processed through attribute and flow before subtypes are translated. + * Ensure that superclasses of C are desugared before C itself. This is + * required for two reasons: (i) as erasure (TransTypes) destroys + * information needed in flow analysis and (ii) as some checks carried + * out during lowering require that all synthetic fields/methods have + * already been added to C and its superclasses. */ class ScanNested extends TreeScanner { Set> dependencies = new LinkedHashSet>(); @@ -1246,8 +1256,8 @@ public class JavaCompiler implements ClassReader.SourceCompleter { ScanNested scanner = new ScanNested(); scanner.scan(env.tree); for (Env dep: scanner.dependencies) { - if (!compileStates.isDone(dep, CompileState.FLOW)) - flow(attribute(dep)); + if (!compileStates.isDone(dep, CompileState.FLOW)) + desugaredEnvs.put(dep, desugar(flow(attribute(dep)))); } //We need to check for error another time as more classes might @@ -1298,6 +1308,7 @@ public class JavaCompiler implements ClassReader.SourceCompleter { return; env.tree = transTypes.translateTopLevelClass(env.tree, localMake); + compileStates.put(env, CompileState.TRANSTYPES); if (shouldStop(CompileState.LOWER)) return; @@ -1315,6 +1326,7 @@ public class JavaCompiler implements ClassReader.SourceCompleter { //translate out inner classes List cdefs = lower.translateTopLevelClass(env, env.tree, localMake); + compileStates.put(env, CompileState.LOWER); if (shouldStop(CompileState.LOWER)) return; diff --git a/src/share/classes/com/sun/tools/javac/resources/compiler.properties b/src/share/classes/com/sun/tools/javac/resources/compiler.properties index 1a0c6bdb..d006968e 100644 --- a/src/share/classes/com/sun/tools/javac/resources/compiler.properties +++ b/src/share/classes/com/sun/tools/javac/resources/compiler.properties @@ -431,6 +431,8 @@ compiler.err.static.imp.only.classes.and.interfaces=\ static import only from classes and interfaces compiler.err.synthetic.name.conflict=\ the symbol {0} conflicts with a compiler-synthesized symbol in {1} +compiler.warn.synthetic.name.conflict=\ + the symbol {0} conflicts with a compiler-synthesized symbol in {1} compiler.err.throws.not.allowed.in.intf.annotation=\ throws clause not allowed in @interface members diff --git a/test/tools/javac/6521805/T6521805a.java b/test/tools/javac/6521805/T6521805a.java new file mode 100644 index 00000000..d2241df0 --- /dev/null +++ b/test/tools/javac/6521805/T6521805a.java @@ -0,0 +1,45 @@ +/* + * Copyright 2009 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6521805 + * @summary Regression: JDK5/JDK6 javac allows write access to outer class reference + * @author mcimadamore + * + * @compile/fail/ref=T6521805a_1.out T6521805a.java -XDrawDiagnostics + * @compile/ref=T6521805a_2.out T6521805a.java -XDwarnOnSyntheticConflicts -XDrawDiagnostics + */ + +class T6521805a { + + static class Outer { + T6521805a this$0 = null; + } + + public class Inner extends Outer { + public void foo() { + this$0 = new T6521805a(); + } + } +} diff --git a/test/tools/javac/6521805/T6521805a_1.out b/test/tools/javac/6521805/T6521805a_1.out new file mode 100644 index 00000000..0246b3d6 --- /dev/null +++ b/test/tools/javac/6521805/T6521805a_1.out @@ -0,0 +1,2 @@ +T6521805a.java:40:12: compiler.err.synthetic.name.conflict: this$0, T6521805a.Outer +1 error diff --git a/test/tools/javac/6521805/T6521805a_2.out b/test/tools/javac/6521805/T6521805a_2.out new file mode 100644 index 00000000..f99aa99c --- /dev/null +++ b/test/tools/javac/6521805/T6521805a_2.out @@ -0,0 +1,2 @@ +T6521805a.java:40:12: compiler.warn.synthetic.name.conflict: this$0, T6521805a.Outer +1 warning diff --git a/test/tools/javac/6521805/T6521805b.java b/test/tools/javac/6521805/T6521805b.java new file mode 100644 index 00000000..ff9c6071 --- /dev/null +++ b/test/tools/javac/6521805/T6521805b.java @@ -0,0 +1,44 @@ +/* + * Copyright 2009 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6521805 + * @summary Regression: JDK5/JDK6 javac allows write access to outer class reference + * @author mcimadamore + * + * @compile T6521805b.java + */ + +class T6521805b { + + static class Outer { + String this$0 = null; + } + + public class Inner extends Outer { + public void foo() { + this$0 = "Hello!"; + } + } +} diff --git a/test/tools/javac/6521805/T6521805c.java b/test/tools/javac/6521805/T6521805c.java new file mode 100644 index 00000000..2da6d4d5 --- /dev/null +++ b/test/tools/javac/6521805/T6521805c.java @@ -0,0 +1,44 @@ +/* + * Copyright 2009 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6521805 + * @summary Regression: JDK5/JDK6 javac allows write access to outer class reference + * @author mcimadamore + * + * @compile T6521805c.java + */ + +class T6521805c { + + static class Outer { + T6521805c this$0() { return null;} + } + + public class Inner extends Outer { + public void foo() { + this$0(); + } + } +} diff --git a/test/tools/javac/6521805/T6521805d.java b/test/tools/javac/6521805/T6521805d.java new file mode 100644 index 00000000..a1df158e --- /dev/null +++ b/test/tools/javac/6521805/T6521805d.java @@ -0,0 +1,50 @@ +/* + * Copyright 2009 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6521805 + * @summary Regression: JDK5/JDK6 javac allows write access to outer class reference + * @author mcimadamore + * + * @compile/fail/ref=T6521805d.out T6521805d.java -XDrawDiagnostics + */ + +class T6521805 { + + static class Inner extends T6521805.Outer { + + Inner(T6521805 t) { + t.super(); + } + + T6521805 this$0 = null; + + public void foo() { + this$0 = new T6521805(); + } + } + + class Outer {} + +} diff --git a/test/tools/javac/6521805/T6521805d.out b/test/tools/javac/6521805/T6521805d.out new file mode 100644 index 00000000..f3656247 --- /dev/null +++ b/test/tools/javac/6521805/T6521805d.out @@ -0,0 +1,2 @@ +T6521805d.java:41:18: compiler.err.synthetic.name.conflict: this$0, T6521805.Inner +1 error diff --git a/test/tools/javac/6521805/T6521805e.java b/test/tools/javac/6521805/T6521805e.java new file mode 100644 index 00000000..4af4649d --- /dev/null +++ b/test/tools/javac/6521805/T6521805e.java @@ -0,0 +1,32 @@ +/* + * Copyright 2009 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6521805 + * @summary Regression: JDK5/JDK6 javac allows write access to outer class reference + * @author mcimadamore + * + * @compile/fail/ref=T6521805e.out p/Outer.java p/Sub.java -XDrawDiagnostics + * @compile/fail/ref=T6521805e.out p/Sub.java p/Outer.java -XDrawDiagnostics + */ diff --git a/test/tools/javac/6521805/T6521805e.out b/test/tools/javac/6521805/T6521805e.out new file mode 100644 index 00000000..04c71cd8 --- /dev/null +++ b/test/tools/javac/6521805/T6521805e.out @@ -0,0 +1,2 @@ +Sub.java:8:11: compiler.err.synthetic.name.conflict: this$0, p.Inner +1 error diff --git a/test/tools/javac/6521805/p/Outer.java b/test/tools/javac/6521805/p/Outer.java new file mode 100644 index 00000000..3dd1de56 --- /dev/null +++ b/test/tools/javac/6521805/p/Outer.java @@ -0,0 +1,5 @@ +package p; + +class Outer { + class Super {} +} diff --git a/test/tools/javac/6521805/p/Sub.java b/test/tools/javac/6521805/p/Sub.java new file mode 100644 index 00000000..7a9f7889 --- /dev/null +++ b/test/tools/javac/6521805/p/Sub.java @@ -0,0 +1,13 @@ +package p; + +class Inner extends Outer.Super { + Inner(Outer t) { + t.super(); + } + + Outer this$0 = null; + + public void foo() { + this$0 = new Outer(); + } +} diff --git a/test/tools/javac/6734819/T6734819a.out b/test/tools/javac/6734819/T6734819a.out index 73437aee..a3ccb07e 100644 --- a/test/tools/javac/6734819/T6734819a.out +++ b/test/tools/javac/6734819/T6734819a.out @@ -4,9 +4,9 @@ [flow W] [attribute Z] [flow Z] +[desugar Z] +[desugar W] [desugar Y] [generate code Y] -[desugar W] [generate code W] -[desugar Z] [generate code Z] diff --git a/test/tools/javac/6734819/T6734819b.out b/test/tools/javac/6734819/T6734819b.out index 3d715333..0c962900 100644 --- a/test/tools/javac/6734819/T6734819b.out +++ b/test/tools/javac/6734819/T6734819b.out @@ -2,8 +2,8 @@ [flow A] [attribute B] [flow B] +[desugar B] [desugar A] [generate code A] -[desugar B] [generate code B.C] [generate code B] diff --git a/test/tools/javac/policy/test2/byfile.AB.out b/test/tools/javac/policy/test2/byfile.AB.out index fdc7d872..6d152c87 100644 --- a/test/tools/javac/policy/test2/byfile.AB.out +++ b/test/tools/javac/policy/test2/byfile.AB.out @@ -2,6 +2,7 @@ [flow A] [attribute B] [flow B] +[desugar B] [desugar A] [generate code A.A1] [generate code A.A2] @@ -10,6 +11,5 @@ [generate code ] [generate code A.A4] [generate code A] -[desugar B] [generate code B.Inner] [generate code B] diff --git a/test/tools/javac/policy/test2/bytodo.AB.out b/test/tools/javac/policy/test2/bytodo.AB.out index fdc7d872..6d152c87 100644 --- a/test/tools/javac/policy/test2/bytodo.AB.out +++ b/test/tools/javac/policy/test2/bytodo.AB.out @@ -2,6 +2,7 @@ [flow A] [attribute B] [flow B] +[desugar B] [desugar A] [generate code A.A1] [generate code A.A2] @@ -10,6 +11,5 @@ [generate code ] [generate code A.A4] [generate code A] -[desugar B] [generate code B.Inner] [generate code B] -- GitLab