From 9a19eb6918e1f766ccf1b1671ea1161a76fee571 Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Sat, 21 Nov 2020 03:17:57 +0000 Subject: [PATCH] 8254105: allow static nested declarations Reviewed-by: mcimadamore --- .../com/sun/tools/javac/code/Flags.java | 47 +- .../com/sun/tools/javac/comp/Attr.java | 16 +- .../com/sun/tools/javac/comp/Check.java | 15 +- .../tools/javac/AnonStaticMember_1.java | 5 +- .../tools/javac/AnonStaticMember_1.out | 4 + .../tools/javac/AnonStaticMember_2.java | 5 +- .../tools/javac/AnonStaticMember_2.out | 4 +- .../tools/javac/InnerNamedConstant_2.java | 3 +- .../tools/javac/InnerNamedConstant_2.out | 5 - .../tools/javac/InnerNamedConstant_2_A.out | 7 + .../tools/javac/InnerNamedConstant_2_B.out | 2 + .../tools/javac/InterfaceInInner.java | 5 +- .../tools/javac/InterfaceInInner.out | 4 +- .../javac/T8222035/MinContextOpTest.java | 3 +- .../javac/T8222035/MinContextOpTest_A.out | 6 + ...ntextOpTest.out => MinContextOpTest_B.out} | 6 +- .../diags/examples/EnumsMustBeStatic.java | 2 + .../examples/InnerClassCantHaveStatic.java | 4 +- .../RecordsNotAllowedInInnerClasses.java | 30 - .../tools/javac/enum/NestedEnum.java | 5 +- .../langtools/tools/javac/enum/NestedEnum.out | 4 +- test/langtools/tools/javac/enum/T5081785.java | 5 +- test/langtools/tools/javac/enum/T5081785.out | 10 +- .../records/LocalStaticDeclarations.java | 33 +- .../javac/records/RecordCompilationTests.java | 810 ++++++++++-------- 25 files changed, 581 insertions(+), 459 deletions(-) create mode 100644 test/langtools/tools/javac/AnonStaticMember_1.out delete mode 100644 test/langtools/tools/javac/InnerNamedConstant_2.out create mode 100644 test/langtools/tools/javac/InnerNamedConstant_2_A.out create mode 100644 test/langtools/tools/javac/InnerNamedConstant_2_B.out create mode 100644 test/langtools/tools/javac/T8222035/MinContextOpTest_A.out rename test/langtools/tools/javac/T8222035/{MinContextOpTest.out => MinContextOpTest_B.out} (54%) delete mode 100644 test/langtools/tools/javac/diags/examples/RecordsNotAllowedInInnerClasses.java diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Flags.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Flags.java index aebbc78da36..b87a1e41298 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Flags.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Flags.java @@ -383,30 +383,31 @@ public class Flags { /** Modifier masks. */ public static final int - AccessFlags = PUBLIC | PROTECTED | PRIVATE, - LocalClassFlags = FINAL | ABSTRACT | STRICTFP | ENUM | SYNTHETIC, - StaticLocalFlags = LocalClassFlags | STATIC | INTERFACE, - MemberClassFlags = LocalClassFlags | INTERFACE | AccessFlags, - MemberRecordFlags = MemberClassFlags | STATIC, - ClassFlags = LocalClassFlags | INTERFACE | PUBLIC | ANNOTATION, - InterfaceVarFlags = FINAL | STATIC | PUBLIC, - VarFlags = AccessFlags | FINAL | STATIC | - VOLATILE | TRANSIENT | ENUM, - ConstructorFlags = AccessFlags, - InterfaceMethodFlags = ABSTRACT | PUBLIC, - MethodFlags = AccessFlags | ABSTRACT | STATIC | NATIVE | - SYNCHRONIZED | FINAL | STRICTFP, - RecordMethodFlags = AccessFlags | ABSTRACT | STATIC | - SYNCHRONIZED | FINAL | STRICTFP; + AccessFlags = PUBLIC | PROTECTED | PRIVATE, + LocalClassFlags = FINAL | ABSTRACT | STRICTFP | ENUM | SYNTHETIC, + StaticLocalFlags = LocalClassFlags | STATIC | INTERFACE, + MemberClassFlags = LocalClassFlags | INTERFACE | AccessFlags, + MemberStaticClassFlags = MemberClassFlags | STATIC, + ClassFlags = LocalClassFlags | INTERFACE | PUBLIC | ANNOTATION, + InterfaceVarFlags = FINAL | STATIC | PUBLIC, + VarFlags = AccessFlags | FINAL | STATIC | + VOLATILE | TRANSIENT | ENUM, + ConstructorFlags = AccessFlags, + InterfaceMethodFlags = ABSTRACT | PUBLIC, + MethodFlags = AccessFlags | ABSTRACT | STATIC | NATIVE | + SYNCHRONIZED | FINAL | STRICTFP, + RecordMethodFlags = AccessFlags | ABSTRACT | STATIC | + SYNCHRONIZED | FINAL | STRICTFP; public static final long - ExtendedStandardFlags = (long)StandardFlags | DEFAULT | SEALED | NON_SEALED, - ExtendedMemberClassFlags = (long)MemberClassFlags | SEALED | NON_SEALED, - ExtendedClassFlags = (long)ClassFlags | SEALED | NON_SEALED, - ModifierFlags = ((long)StandardFlags & ~INTERFACE) | DEFAULT | SEALED | NON_SEALED, - InterfaceMethodMask = ABSTRACT | PRIVATE | STATIC | PUBLIC | STRICTFP | DEFAULT, - AnnotationTypeElementMask = ABSTRACT | PUBLIC, - LocalVarFlags = FINAL | PARAMETER, - ReceiverParamFlags = PARAMETER; + ExtendedStandardFlags = (long)StandardFlags | DEFAULT | SEALED | NON_SEALED, + ExtendedMemberClassFlags = (long)MemberClassFlags | SEALED | NON_SEALED, + ExtendedMemberStaticClassFlags = (long) MemberStaticClassFlags | SEALED | NON_SEALED, + ExtendedClassFlags = (long)ClassFlags | SEALED | NON_SEALED, + ModifierFlags = ((long)StandardFlags & ~INTERFACE) | DEFAULT | SEALED | NON_SEALED, + InterfaceMethodMask = ABSTRACT | PRIVATE | STATIC | PUBLIC | STRICTFP | DEFAULT, + AnnotationTypeElementMask = ABSTRACT | PUBLIC, + LocalVarFlags = FINAL | PARAMETER, + ReceiverParamFlags = PARAMETER; @SuppressWarnings("preview") public static Set asModifierSet(long flags) { diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index 0041b402449..df21524229c 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -171,6 +171,7 @@ public class Attr extends JCTree.Visitor { allowReifiableTypesInInstanceof = Feature.REIFIABLE_TYPES_INSTANCEOF.allowedInSource(source) && (!preview.isPreview(Feature.REIFIABLE_TYPES_INSTANCEOF) || preview.isEnabled()); + allowRecords = Feature.RECORDS.allowedInSource(source); sourceName = source.name; useBeforeDeclarationWarning = options.isSet("useBeforeDeclarationWarning"); @@ -207,6 +208,10 @@ public class Attr extends JCTree.Visitor { */ boolean allowReifiableTypesInInstanceof; + /** Are records allowed + */ + private final boolean allowRecords; + /** * Switch: warn about use of variable before declaration? * RFE: 6425594 @@ -5309,14 +5314,15 @@ public class Attr extends JCTree.Visitor { attribStat(l.head, env); // Check that declarations in inner classes are not static (JLS 8.1.2) // Make an exception for static constants. - if (c.owner.kind != PCK && - ((c.flags() & STATIC) == 0 || c.name == names.empty) && - (TreeInfo.flags(l.head) & (STATIC | INTERFACE)) != 0) { + if (!allowRecords && + c.owner.kind != PCK && + ((c.flags() & STATIC) == 0 || c.name == names.empty) && + (TreeInfo.flags(l.head) & (STATIC | INTERFACE)) != 0) { Symbol sym = null; if (l.head.hasTag(VARDEF)) sym = ((JCVariableDecl) l.head).sym; if (sym == null || - sym.kind != VAR || - ((VarSymbol) sym).getConstValue() == null) + sym.kind != VAR || + ((VarSymbol) sym).getConstValue() == null) log.error(l.head.pos(), Errors.IclsCantHaveStaticDecl(c)); } } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java index baff49db6d7..18c4c2b8658 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java @@ -1216,23 +1216,20 @@ public class Check { implicit |= sym.owner.flags_field & STRICTFP; break; case TYP: - if (sym.isLocal()) { + if (sym.owner.kind.matches(KindSelector.VAL_MTH)) { boolean implicitlyStatic = !sym.isAnonymous() && ((flags & RECORD) != 0 || (flags & ENUM) != 0 || (flags & INTERFACE) != 0); boolean staticOrImplicitlyStatic = (flags & STATIC) != 0 || implicitlyStatic; + // local statics are allowed only if records are allowed too mask = staticOrImplicitlyStatic && allowRecords && (flags & ANNOTATION) == 0 ? StaticLocalFlags : LocalClassFlags; implicit = implicitlyStatic ? STATIC : implicit; - if (staticOrImplicitlyStatic) { - if (sym.owner.kind == TYP) { - log.error(pos, Errors.StaticDeclarationNotAllowedInInnerClasses); - } - } } else if (sym.owner.kind == TYP) { - mask = (flags & RECORD) != 0 ? MemberRecordFlags : ExtendedMemberClassFlags; + // statics in inner classes are allowed only if records are allowed too + mask = ((flags & STATIC) != 0) && allowRecords ? ExtendedMemberStaticClassFlags : ExtendedMemberClassFlags; if (sym.owner.owner.kind == PCK || - (sym.owner.flags_field & STATIC) != 0) + (sym.owner.flags_field & STATIC) != 0) { mask |= STATIC; - else if ((flags & ENUM) != 0 || (flags & RECORD) != 0) { + } else if (!allowRecords && ((flags & ENUM) != 0 || (flags & RECORD) != 0)) { log.error(pos, Errors.StaticDeclarationNotAllowedInInnerClasses); } // Nested interfaces and enums are always STATIC (Spec ???) diff --git a/test/langtools/tools/javac/AnonStaticMember_1.java b/test/langtools/tools/javac/AnonStaticMember_1.java index 7b6602a48ba..bc7b2542e34 100644 --- a/test/langtools/tools/javac/AnonStaticMember_1.java +++ b/test/langtools/tools/javac/AnonStaticMember_1.java @@ -24,10 +24,11 @@ /* * @test * @bug 4279339 - * @summary Verify that an anonymous class cannot contain a static field. + * @summary Verify that an anonymous class can contain a static field only if source >= 16 * @author maddox * - * @run compile/fail AnonStaticMember_1.java + * @compile/fail/ref=AnonStaticMember_1.out -source 15 -XDrawDiagnostics AnonStaticMember_1.java + * @compile AnonStaticMember_1.java */ class AnonStaticMember_1 { diff --git a/test/langtools/tools/javac/AnonStaticMember_1.out b/test/langtools/tools/javac/AnonStaticMember_1.out new file mode 100644 index 00000000000..7398987790e --- /dev/null +++ b/test/langtools/tools/javac/AnonStaticMember_1.out @@ -0,0 +1,4 @@ +- compiler.warn.source.no.system.modules.path: 15 +AnonStaticMember_1.java:36:20: compiler.err.icls.cant.have.static.decl: compiler.misc.anonymous.class: AnonStaticMember_1$1 +1 error +1 warning diff --git a/test/langtools/tools/javac/AnonStaticMember_2.java b/test/langtools/tools/javac/AnonStaticMember_2.java index 964f7f007a7..3d50baf83af 100644 --- a/test/langtools/tools/javac/AnonStaticMember_2.java +++ b/test/langtools/tools/javac/AnonStaticMember_2.java @@ -1,10 +1,11 @@ /* * @test /nodynamiccopyright/ * @bug 4279339 6969184 - * @summary Verify that an anonymous class cannot contain a static method. + * @summary Verify that an anonymous class can contain a static method only if source >= 16 * @author maddox * - * @run compile/fail/ref=AnonStaticMember_2.out -XDrawDiagnostics AnonStaticMember_2.java + * @compile/fail/ref=AnonStaticMember_2.out -source 15 -XDrawDiagnostics AnonStaticMember_2.java + * @compile AnonStaticMember_2.java */ class AnonStaticMember_2 { diff --git a/test/langtools/tools/javac/AnonStaticMember_2.out b/test/langtools/tools/javac/AnonStaticMember_2.out index 40f8821bf7f..197e2c09692 100644 --- a/test/langtools/tools/javac/AnonStaticMember_2.out +++ b/test/langtools/tools/javac/AnonStaticMember_2.out @@ -1,2 +1,4 @@ -AnonStaticMember_2.java:12:21: compiler.err.icls.cant.have.static.decl: compiler.misc.anonymous.class: AnonStaticMember_2$1 +- compiler.warn.source.no.system.modules.path: 15 +AnonStaticMember_2.java:13:21: compiler.err.icls.cant.have.static.decl: compiler.misc.anonymous.class: AnonStaticMember_2$1 1 error +1 warning diff --git a/test/langtools/tools/javac/InnerNamedConstant_2.java b/test/langtools/tools/javac/InnerNamedConstant_2.java index 3383800321a..b75a0b07abb 100644 --- a/test/langtools/tools/javac/InnerNamedConstant_2.java +++ b/test/langtools/tools/javac/InnerNamedConstant_2.java @@ -4,7 +4,8 @@ * @summary Verify rejection of illegal static variables in inner classes. * @author William Maddox (maddox) * - * @compile/fail/ref=InnerNamedConstant_2.out -XDrawDiagnostics InnerNamedConstant_2.java + * @compile/fail/ref=InnerNamedConstant_2_A.out -XDrawDiagnostics -source 15 InnerNamedConstant_2.java + * @compile/fail/ref=InnerNamedConstant_2_B.out -XDrawDiagnostics InnerNamedConstant_2.java */ public class InnerNamedConstant_2 { diff --git a/test/langtools/tools/javac/InnerNamedConstant_2.out b/test/langtools/tools/javac/InnerNamedConstant_2.out deleted file mode 100644 index 58e5ff55c7a..00000000000 --- a/test/langtools/tools/javac/InnerNamedConstant_2.out +++ /dev/null @@ -1,5 +0,0 @@ -InnerNamedConstant_2.java:22:20: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner2 -InnerNamedConstant_2.java:23:29: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner2 -InnerNamedConstant_2.java:25:13: compiler.err.cant.assign.val.to.final.var: z -InnerNamedConstant_2.java:34:26: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner3 -4 errors diff --git a/test/langtools/tools/javac/InnerNamedConstant_2_A.out b/test/langtools/tools/javac/InnerNamedConstant_2_A.out new file mode 100644 index 00000000000..deb216e858f --- /dev/null +++ b/test/langtools/tools/javac/InnerNamedConstant_2_A.out @@ -0,0 +1,7 @@ +- compiler.warn.source.no.system.modules.path: 15 +InnerNamedConstant_2.java:23:20: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner2 +InnerNamedConstant_2.java:24:29: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner2 +InnerNamedConstant_2.java:26:13: compiler.err.cant.assign.val.to.final.var: z +InnerNamedConstant_2.java:35:26: compiler.err.icls.cant.have.static.decl: InnerNamedConstant_2.Inner3 +4 errors +1 warning diff --git a/test/langtools/tools/javac/InnerNamedConstant_2_B.out b/test/langtools/tools/javac/InnerNamedConstant_2_B.out new file mode 100644 index 00000000000..3be59c3d59f --- /dev/null +++ b/test/langtools/tools/javac/InnerNamedConstant_2_B.out @@ -0,0 +1,2 @@ +InnerNamedConstant_2.java:26:13: compiler.err.cant.assign.val.to.final.var: z +1 error diff --git a/test/langtools/tools/javac/InterfaceInInner.java b/test/langtools/tools/javac/InterfaceInInner.java index ca025399101..41b4ea04273 100644 --- a/test/langtools/tools/javac/InterfaceInInner.java +++ b/test/langtools/tools/javac/InterfaceInInner.java @@ -1,10 +1,11 @@ /* * @test /nodynamiccopyright/ * @bug 4063740 6969184 - * @summary Interfaces may only be declared in top level classes. + * @summary Interfaces can be declared in inner classes only for source >= 16 * @author turnidge * - * @compile/fail/ref=InterfaceInInner.out -XDrawDiagnostics InterfaceInInner.java + * @compile/fail/ref=InterfaceInInner.out -XDrawDiagnostics -source 15 InterfaceInInner.java + * @compile InterfaceInInner.java */ class InterfaceInInner { InterfaceInInner() { diff --git a/test/langtools/tools/javac/InterfaceInInner.out b/test/langtools/tools/javac/InterfaceInInner.out index 34fd184c004..b81cc11e1cb 100644 --- a/test/langtools/tools/javac/InterfaceInInner.out +++ b/test/langtools/tools/javac/InterfaceInInner.out @@ -1,2 +1,4 @@ -InterfaceInInner.java:12:13: compiler.err.static.declaration.not.allowed.in.inner.classes +- compiler.warn.source.no.system.modules.path: 15 +InterfaceInInner.java:13:13: compiler.err.icls.cant.have.static.decl: foo 1 error +1 warning diff --git a/test/langtools/tools/javac/T8222035/MinContextOpTest.java b/test/langtools/tools/javac/T8222035/MinContextOpTest.java index fe61cae8e2a..a9dfdd998e4 100644 --- a/test/langtools/tools/javac/T8222035/MinContextOpTest.java +++ b/test/langtools/tools/javac/T8222035/MinContextOpTest.java @@ -25,7 +25,8 @@ * @test * @bug 8222035 * @summary minimal inference context optimization is forcing resolution with incomplete constraints - * @compile/fail/ref=MinContextOpTest.out -XDrawDiagnostics MinContextOpTest.java + * @compile/fail/ref=MinContextOpTest_A.out -XDrawDiagnostics -source 15 MinContextOpTest.java + * @compile/fail/ref=MinContextOpTest_B.out -XDrawDiagnostics MinContextOpTest.java */ import java.util.Map; diff --git a/test/langtools/tools/javac/T8222035/MinContextOpTest_A.out b/test/langtools/tools/javac/T8222035/MinContextOpTest_A.out new file mode 100644 index 00000000000..3d2dfd5fd9a --- /dev/null +++ b/test/langtools/tools/javac/T8222035/MinContextOpTest_A.out @@ -0,0 +1,6 @@ +- compiler.warn.source.no.system.modules.path: 15 +MinContextOpTest.java:39:25: compiler.err.mod.not.allowed.here: static +MinContextOpTest.java:45:25: compiler.err.mod.not.allowed.here: static +MinContextOpTest.java:51:34: compiler.err.prob.found.req: (compiler.misc.infer.no.conforming.assignment.exists: T,K,V,E, (compiler.misc.inconvertible.types: java.util.function.Function, java.util.function.Function>)) +3 errors +1 warning diff --git a/test/langtools/tools/javac/T8222035/MinContextOpTest.out b/test/langtools/tools/javac/T8222035/MinContextOpTest_B.out similarity index 54% rename from test/langtools/tools/javac/T8222035/MinContextOpTest.out rename to test/langtools/tools/javac/T8222035/MinContextOpTest_B.out index 741ff52ff57..658fcd8318c 100644 --- a/test/langtools/tools/javac/T8222035/MinContextOpTest.out +++ b/test/langtools/tools/javac/T8222035/MinContextOpTest_B.out @@ -1,4 +1,2 @@ -MinContextOpTest.java:38:25: compiler.err.mod.not.allowed.here: static -MinContextOpTest.java:44:25: compiler.err.mod.not.allowed.here: static -MinContextOpTest.java:50:34: compiler.err.prob.found.req: (compiler.misc.infer.no.conforming.assignment.exists: T,K,V,E, (compiler.misc.inconvertible.types: java.util.function.Function, java.util.function.Function>)) -3 errors +MinContextOpTest.java:51:34: compiler.err.prob.found.req: (compiler.misc.infer.no.conforming.assignment.exists: T,K,V,E, (compiler.misc.inconvertible.types: java.util.function.Function, java.util.function.Function>)) +1 error diff --git a/test/langtools/tools/javac/diags/examples/EnumsMustBeStatic.java b/test/langtools/tools/javac/diags/examples/EnumsMustBeStatic.java index dcfdce8650e..222f2eb2903 100644 --- a/test/langtools/tools/javac/diags/examples/EnumsMustBeStatic.java +++ b/test/langtools/tools/javac/diags/examples/EnumsMustBeStatic.java @@ -22,6 +22,8 @@ */ // key: compiler.err.static.declaration.not.allowed.in.inner.classes +// key: compiler.warn.source.no.system.modules.path +// options: -source 15 class EnumsMustBeStatic { class Nested { diff --git a/test/langtools/tools/javac/diags/examples/InnerClassCantHaveStatic.java b/test/langtools/tools/javac/diags/examples/InnerClassCantHaveStatic.java index 96dc72f223a..721126b663a 100644 --- a/test/langtools/tools/javac/diags/examples/InnerClassCantHaveStatic.java +++ b/test/langtools/tools/javac/diags/examples/InnerClassCantHaveStatic.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2010, 2020 Oracle and/or its affiliates. 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 @@ -22,6 +22,8 @@ */ // key: compiler.err.icls.cant.have.static.decl +// key: compiler.warn.source.no.system.modules.path +// options: -source 15 class InnerClassCantHaveStatic { class Inner { diff --git a/test/langtools/tools/javac/diags/examples/RecordsNotAllowedInInnerClasses.java b/test/langtools/tools/javac/diags/examples/RecordsNotAllowedInInnerClasses.java deleted file mode 100644 index 1016a874e00..00000000000 --- a/test/langtools/tools/javac/diags/examples/RecordsNotAllowedInInnerClasses.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (c) 2019, 2020, Oracle and/or its affiliates. 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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ - -// key: compiler.err.static.declaration.not.allowed.in.inner.classes - -class Outer { - class Inner { - record R(int a) {} - } -} diff --git a/test/langtools/tools/javac/enum/NestedEnum.java b/test/langtools/tools/javac/enum/NestedEnum.java index 446dd5110ea..225ec4db5de 100644 --- a/test/langtools/tools/javac/enum/NestedEnum.java +++ b/test/langtools/tools/javac/enum/NestedEnum.java @@ -1,10 +1,11 @@ /* * @test /nodynamiccopyright/ * @bug 5071831 - * @summary javac allows enum in an inner class + * @summary javac allows enum in an inner class for source >= 16 * @author gafter * - * @compile/fail/ref=NestedEnum.out -XDrawDiagnostics NestedEnum.java + * @compile/fail/ref=NestedEnum.out -XDrawDiagnostics -source 15 NestedEnum.java + * @compile NestedEnum.java */ class NestedEnum { diff --git a/test/langtools/tools/javac/enum/NestedEnum.out b/test/langtools/tools/javac/enum/NestedEnum.out index e7d49eef73b..0be9622c1e5 100644 --- a/test/langtools/tools/javac/enum/NestedEnum.out +++ b/test/langtools/tools/javac/enum/NestedEnum.out @@ -1,2 +1,4 @@ -NestedEnum.java:12:9: compiler.err.static.declaration.not.allowed.in.inner.classes +- compiler.warn.source.no.system.modules.path: 15 +NestedEnum.java:13:9: compiler.err.static.declaration.not.allowed.in.inner.classes 1 error +1 warning diff --git a/test/langtools/tools/javac/enum/T5081785.java b/test/langtools/tools/javac/enum/T5081785.java index 0359619f670..35330018480 100644 --- a/test/langtools/tools/javac/enum/T5081785.java +++ b/test/langtools/tools/javac/enum/T5081785.java @@ -1,9 +1,10 @@ /* * @test /nodynamiccopyright/ * @bug 5081785 - * @summary Empty Enums allowed in non-static contexts + * @summary enums should be allowed in non-static contexts * @author Peter von der Ah\u00e9 - * @compile/fail/ref=T5081785.out -XDrawDiagnostics T5081785.java + * @compile/fail/ref=T5081785.out -XDrawDiagnostics -source 15 T5081785.java + * @compile T5081785.java */ class A1 { diff --git a/test/langtools/tools/javac/enum/T5081785.out b/test/langtools/tools/javac/enum/T5081785.out index c873019913c..d85c8395ed0 100644 --- a/test/langtools/tools/javac/enum/T5081785.out +++ b/test/langtools/tools/javac/enum/T5081785.out @@ -1,5 +1,7 @@ -T5081785.java:29:9: compiler.err.static.declaration.not.allowed.in.inner.classes -T5081785.java:12:13: compiler.err.static.declaration.not.allowed.in.inner.classes -T5081785.java:19:27: compiler.err.static.declaration.not.allowed.in.inner.classes -T5081785.java:24:31: compiler.err.static.declaration.not.allowed.in.inner.classes +- compiler.warn.source.no.system.modules.path: 15 +T5081785.java:30:9: compiler.err.static.declaration.not.allowed.in.inner.classes +T5081785.java:13:13: compiler.err.static.declaration.not.allowed.in.inner.classes +T5081785.java:20:27: compiler.err.static.declaration.not.allowed.in.inner.classes +T5081785.java:25:31: compiler.err.static.declaration.not.allowed.in.inner.classes 4 errors +1 warning diff --git a/test/langtools/tools/javac/records/LocalStaticDeclarations.java b/test/langtools/tools/javac/records/LocalStaticDeclarations.java index 53dd9854231..6e6571df805 100644 --- a/test/langtools/tools/javac/records/LocalStaticDeclarations.java +++ b/test/langtools/tools/javac/records/LocalStaticDeclarations.java @@ -48,6 +48,12 @@ import combo.ComboTask; import combo.ComboTask.Result; import combo.ComboTestHelper; +/** this test checks two thinks: + * 1 - that static declarations are allowed inside inner classes + * 2 - and in addtion that non-static variables can't be captured + * by static contexts + */ + public class LocalStaticDeclarations extends ComboInstance { static final String sourceTemplate = @@ -57,10 +63,12 @@ public class LocalStaticDeclarations extends ComboInstance { #{STATIC_LOCAL} };"); String container; @@ -124,7 +132,6 @@ public class LocalStaticDeclarations extends ComboInstance> result) { if (shouldFail()) { - Assert.check(result.hasErrors(), result.compilationInfo()); + Assert.check(result.hasErrors(), "unexpected compilation\n" + result.compilationInfo()); if (!expectedDiagFound(result)) { fail("test failing with unexpected error message\n" + result.compilationInfo()); } @@ -195,10 +202,7 @@ public class LocalStaticDeclarations extends ComboInstance> result) { - if ((container == Container.NO_CONTAINER || - container == Container.LAMBDA || - container == Container.ANONYMOUS) && - !acceptableExpr()) { + if (expr == Expression.LOCAL_VARIABLE || expr == Expression.INSTANCE_FIELD) { return result.containsKey("compiler.err.non-static.cant.be.ref"); - } else if (container == Container.ENUM) { - return result.containsKey("compiler.err.enum.constant.expected" ); } - return result.containsKey("compiler.err.static.declaration.not.allowed.in.inner.classes" ); + return false; } } diff --git a/test/langtools/tools/javac/records/RecordCompilationTests.java b/test/langtools/tools/javac/records/RecordCompilationTests.java index bc3be98c72d..11940d92a98 100644 --- a/test/langtools/tools/javac/records/RecordCompilationTests.java +++ b/test/langtools/tools/javac/records/RecordCompilationTests.java @@ -477,365 +477,483 @@ public class RecordCompilationTests extends CompilationTestCase { assertFail("compiler.err.already.defined", template); } - public void testStaticLocalTypes() { - // local records can also be final - assertOK("class R { \n" + - " void m() { \n" + - " final record RR(int x) { };\n" + - " }\n" + - "}"); + public void testStaticLocals() { + // static locals can't capture local variables, instance fields or type variables + for (String s : List.of( + "record RR(int x) { public int x() { return y; }};", + "record RR(int x) { public int x() { return z; }};", + "record RR(int x) { public int x() { return instance; }};", + "record RR(T t) {};", + "record RR(U u) {};", - // Can't capture locals - assertFail("compiler.err.non-static.cant.be.ref", - "class R { \n" + - " void m(int y) { \n" + - " record RR(int x) { public int x() { return y; }};\n" + - " }\n" + - "}"); + "interface I { default int x() { return y; }};", + "interface I { default int x() { return z; }};", + "interface I { default int x() { return instance; }};", + "interface I { default int x(T t) { return 0; }};", + "interface I { default int x(U u) { return 0; }};", - assertFail("compiler.err.non-static.cant.be.ref", - "class R { \n" + - " void m() {\n" + - " int y;\n" + - " record RR(int x) { public int x() { return y; }};\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m() {\n" + - " String hello = \"hello\";\n" + - " interface I {\n" + - " public default void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(hello);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m() {\n" + - " String hello = \"hello\";\n" + - " record R(int i) {\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(hello);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m() {\n" + - " String hello = \"hello\";\n" + - " enum E {\n" + - " A;\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(hello);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m(String param) {\n" + - " interface I {\n" + - " public default void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(param);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m(String param) {\n" + - " record R(int i) {\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(param);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void m(String param) {\n" + - " enum E {\n" + - " A;\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(param);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " String instanceField = \"instance\";\n" + - " public static void m() {\n" + - " interface I {\n" + - " public default void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(instanceField);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " String instanceField = \"instance\";\n" + - " public static void m(String param) {\n" + - " record R(int i) {\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(instanceField);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " String instanceField = \"instance\";\n" + - " public static void m(String param) {\n" + - " enum E {\n" + - " A;\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " System.err.println(instanceField);\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - // instance fields - assertFail("compiler.err.non-static.cant.be.ref", - "class R { \n" + - " int z = 0;\n" + - " void m() { \n" + - " record RR(int x) { public int x() { return z; }};\n" + - " }\n" + - "}"); - - // or type variables - assertFail("compiler.err.non-static.cant.be.ref", - "class R { \n" + - " void m() { \n" + - " record RR(T t) {};\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class R {\n" + - " static U make(U u) { //method is static\n" + - " interface Checker {\n" + - " void check(U u);\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class LocalEnum {\n" + - " static U getAndSet(U u) { //method is static\n" + - " enum X {\n" + - " A;\n" + - " U u;\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}\n"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class R {\n" + - " static U make(U u) { //method is static\n" + - " record Checker() {\n" + - " void check(U u);\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class R {\n" + - " U make(U u) { // enclosing method is not static\n" + - " interface Checker {\n" + - " void check(U u);\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class LocalEnum {\n" + - " U getAndSet(U u) { // enclosing method is not static\n" + - " enum X {\n" + - " A;\n" + - " U u;\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}\n"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class R {\n" + - " U make(U u) { // enclosing method is not static\n" + - " record Checker() {\n" + - " void check(U u);\n" + - " }\n" + - " return null;\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void main(String[] args) {\n" + - " interface I {\n" + - " public default void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " T t = null;\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void main(String[] args) {\n" + - " record R(int i) {\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " T t = null;\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - assertFail("compiler.err.non-static.cant.be.ref", - "class C {\n" + - " public static void main(String[] args) {\n" + - " enum E {\n" + - " A;\n" + - " public void test1() {\n" + - " class X {\n" + - " public void test2() {\n" + - " T t = null;\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - " }\n" + - "}"); - - // but static fields are OK - assertOK("class R { \n" + - " static int z = 0;\n" + - " void m() { \n" + - " record RR(int x) { public int x() { return z; }};\n" + - " }\n" + - "}"); - // Can't self-shadow - assertFail("compiler.err.already.defined", + "enum E { A; int x() { return y; }};", + "enum E { A; int x() { return z; }};", + "enum E { A; int x() { return instance; }};", + "enum E { A; int x(T t) { return 0; }};", + "enum E { A; int x(U u) { return 0; }};" + )) { + assertFail("compiler.err.non-static.cant.be.ref", """ - class R { - void m() { - record R(int x) { }; + class R { + int instance = 0; + U m(int y) { + int z; + #S + return null; } } - """ - ); + """.replaceFirst("#S", s)); + } + + // a similar example but a bit more complex + for (String s : List.of( + "record R() { void test1() { class X { void test2() { System.err.println(localVar); } } } }", + "record R() { void test1() { class X { void test2() { System.err.println(param); } } } }", + "record R() {void test1() { class X { void test2() { System.err.println(instanceField); } } } }", + "record R() { void test1() { class X { T t; } } }", + "record R() { void test1() { class X { U u; } } }", + + "interface I { default void test1() { class X { void test2() { System.err.println(localVar); } } } }", + "interface I() { default void test1() { class X { void test2() {System.err.println(param);} } } }", + "interface I { default void test1() { class X { void test2() { System.err.println(instanceField); } } } }", + "interface I { default void test1() { class X { T t; } } }", + "interface I() { default void test1() { class X {U u;} } }", + + "enum E { A; void test1() { class X { void test2() { System.err.println(localVar); } } } }", + "enum E { A; void test1() { class X { void test2() {System.err.println(param);} } } }", + "enum E { A; void test1() { class X { void test2() { System.err.println(instanceField); } } } }", + "enum E { A; void test1() { class X { T t; } } }", + "enum E { A; void test1() { class X {U u;} } }" + )) { + assertFail("compiler.err.non-static.cant.be.ref", + """ + class C { + String instanceField = "instance"; + static U m(String param) { + String localVar = "local"; + #S + return null; + } + } + """.replaceFirst("#S", s)); + } + + // can't self-shadow + for (String s : List.of("record R() {}", "interface R {}", "enum R { A }")) { + assertFail("compiler.err.already.defined", "class R { void m() { #S } }".replaceFirst("#S", s)); + } + // can't be explicitly static - assertFail("compiler.err.illegal.start.of.expr", + for (String s : List.of("static record RR() { }", "static interface I {}", "static enum E { A }")) { + assertFail("compiler.err.illegal.start.of.expr", "class R { void m() { #S } }".replaceFirst("#S", s)); + } + + // but static fields can be accessed + for (String s : List.of( + "record RR() { public int x() { return z; } };", + "interface I { default int x() { return z; } }", + "enum E { A; int x() { return z; } }" + )) { + assertOK("class R { static int z = 0; void m() { #S } }".replaceFirst("#S", s)); + } + + // local records can also be final + assertOK("class R { void m() { final record RR(int x) { }; } }"); + } + + public void testStaticDefinitionsInInnerClasses() { + // static defs in inner classes can't capture instance fields or type variables + for (String s : List.of( """ - class R { - void m() { - static record RR(int x) { }; + record R() { + void test() { System.err.println(field); } + } + """, + """ + record R() { + void test(T t) {} + } + """, + """ + record R() { + void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + record R() { + void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + interface I { + default void test() { System.err.println(field); } + } + """, + """ + interface I { + default void test(T t) {} + } + """, + """ + interface I { + default void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + interface I { + default void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + enum E { + A; + void test() { System.err.println(field); } + } + """, + """ + enum E { + A; + void test(T t) {} + } + """, + """ + enum E { + A; + void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + enum E { + A; + void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + static class SC { + void test() { System.err.println(field); } + } + """, + """ + static class SC { + void test(T t) {} + } + """, + """ + static class SC { + void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + static class SC { + void test1() { + class X { void test2(T t) {} } } } """ - ); + )) { + assertFail("compiler.err.non-static.cant.be.ref", + """ + class C { + String field = "field"; + class Inner { + #S + } + } + """.replaceFirst("#S", s)); + } - // positive cases - assertOK( + // another, more complex, example + // static defs in inner classes can't capture instance locals, fields or type variables + for (String s : List.of( """ - import java.security.*; - class Test { - static Test newInstance(Object provider) { - return new Test() { - private final PrivilegedExceptionAction action = new PrivilegedExceptionAction() { - public KeyStore run() throws Exception { - if (provider == null) {} - return null; - } - }; - }; + record R() { + void test() { System.err.println(field); } + } + """, + """ + record R() { + void test1() { + class X { void test2() { System.err.println(field); } } + } + } + """, + """ + record R() { + void test() { System.err.println(param); } + } + """, + """ + record R() { + void test1() { + class X { void test2() { System.err.println(param); } } + } + } + """, + """ + record R() { + void test() { System.err.println(local); } + } + """, + """ + record R() { + void test1() { + class X { void test2() { System.err.println(local); } } + } + } + """, + """ + record R() { + void test(T t) {} + } + """, + """ + record R() { + void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + interface I { + default void test() { System.err.println(field); } + } + """, + """ + interface I { + default void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + interface I { + default void test() { System.err.println(param); } + } + """, + """ + interface I { + default void test1() { + class X { + void test2() { System.err.println(param); } + } + } + } + """, + """ + interface I { + default void test() { System.err.println(local); } + } + """, + """ + interface I { + default void test1() { + class X { + void test2() { System.err.println(local); } + } + } + } + """, + """ + interface I { + default void test(T t) {} + } + """, + """ + interface I { + default void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + enum E { + A; + void test() { System.err.println(field); } + } + """, + """ + enum E { + A; + void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + enum E { + A; + void test() { System.err.println(param); } + } + """, + """ + enum E { + A; + void test1() { + class X { + void test2() { System.err.println(param); } + } + } + } + """, + """ + enum E { + A; + void test() { System.err.println(local); } + } + """, + """ + enum E { + A; + void test1() { + class X { + void test2() { System.err.println(local); } + } + } + } + """, + """ + enum E { + A; + void test(T t) {} + } + """, + """ + enum E { + A; + void test1() { + class X { void test2(T t) {} } + } + } + """, + + """ + static class SC { + void test() { System.err.println(field); } + } + """, + """ + static class SC { + void test1() { + class X { + void test2() { System.err.println(field); } + } + } + } + """, + """ + static class SC { + void test() { System.err.println(param); } + } + """, + """ + static class SC { + void test1() { + class X { + void test2() { System.err.println(param); } + } + } + } + """, + """ + static class SC { + void test() { System.err.println(local); } + } + """, + """ + static class SC { + void test1() { + class X { + void test2() { System.err.println(local); } + } + } + } + """, + """ + static class SC { + void test(T t) {} + } + """, + """ + static class SC { + void test1() { + class X { void test2(T t) {} } } } """ - ); - - assertOK( - """ - import java.security.*; - class Test { - static Test newInstance(Object provider) { - return new Test() { - int m(PrivilegedExceptionAction a) { return 0; } - { - m( - new PrivilegedExceptionAction() { - public KeyStore run() throws Exception { - if (provider == null) {} - return null; - } - } - ); + )) { + assertFail("compiler.err.non-static.cant.be.ref", + """ + class C { + String field = "field"; + U m(String param) { + String local = "local"; + class Local { + class Inner { #S } } - }; + return null; + } } + """.replaceFirst("#S", s)); + } + + // inner classes can contain static methods too + assertOK( + """ + class C { + class Inner { + // static method inside inner class + static void m() {} + } + } + """ + ); + + assertOK( + """ + class C { + void m() { + new Object() { + // static method inside inner class + static void m() {} + }; + } } """ ); @@ -869,7 +987,7 @@ public class RecordCompilationTests extends CompilationTestCase { } public void testRecordsInsideInner() { - assertFail("compiler.err.static.declaration.not.allowed.in.inner.classes", + assertOK( """ class Outer { class Inner { @@ -878,7 +996,7 @@ public class RecordCompilationTests extends CompilationTestCase { } """ ); - assertFail("compiler.err.static.declaration.not.allowed.in.inner.classes", + assertOK( """ class Outer { public void test() { @@ -888,7 +1006,7 @@ public class RecordCompilationTests extends CompilationTestCase { } } """); - assertFail("compiler.err.static.declaration.not.allowed.in.inner.classes", + assertOK( """ class Outer { Runnable run = new Runnable() { @@ -897,7 +1015,7 @@ public class RecordCompilationTests extends CompilationTestCase { }; } """); - assertFail("compiler.err.static.declaration.not.allowed.in.inner.classes", + assertOK( """ class Outer { void m() {