/* * Copyright (c) 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. */ /* * @test * @bug 8242293 8246774 * @summary allow for local interfaces and enums plus nested records, interfaces and enums * @library /tools/javac/lib * @modules jdk.compiler/com.sun.tools.javac.api * jdk.compiler/com.sun.tools.javac.file * jdk.compiler/com.sun.tools.javac.util * @build combo.ComboTestHelper * @run main LocalStaticDeclarations */ import javax.lang.model.element.Element; import javax.tools.Diagnostic; import javax.tools.JavaFileObject; import com.sun.tools.javac.util.Assert; import com.sun.tools.javac.api.ClientCodeWrapper; import com.sun.tools.javac.util.JCDiagnostic; import com.sun.tools.javac.util.List; import combo.ComboInstance; import combo.ComboParameter; 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 = """ import java.lang.annotation.*; class Test { int INSTANCE_FIELD = 0; static int STATIC_FIELD = 0; // instance initializer { int LOCAL_VARIABLE = 0; #{CONTAINER} } Test() { int LOCAL_VARIABLE = 0; #{CONTAINER} } void m() { int LOCAL_VARIABLE = 0; #{CONTAINER} } static void foo() { int LOCAL_VARIABLE = 0; #{CONTAINER} } } """; enum Container implements ComboParameter { NO_CONTAINER("#{STATIC_LOCAL}"), INTERFACE("interface CI { #{STATIC_LOCAL} }"), ANONYMOUS( """ new Object() { // instance initializer { #{STATIC_LOCAL} } void m() { #{STATIC_LOCAL} } }; """ ), RECORD("record CR() { #{STATIC_LOCAL} }"), CLASS("class CC { #{STATIC_LOCAL} }"), ENUM("enum CE { CE1; #{STATIC_LOCAL} }"), LAMBDA("Runnable run = () -> { #{STATIC_LOCAL} };"); String container; Container(String container) { this.container = container; } public String expand(String optParameter) { return container; } } enum StaticLocalDecl implements ComboParameter { ENUM("enum E { E1; #{MEMBER} }"), RECORD("record R() { #{MEMBER} }"), INTERFACE("interface I { #{MEMBER} }"); String localDecl; StaticLocalDecl(String localDecl) { this.localDecl = localDecl; } public String expand(String optParameter) { return localDecl; } } enum Member implements ComboParameter { METHOD("int foo() { return #{EXPR}; }"), DEFAULT_METHOD("default int foo() { return #{EXPR}; }"); String member; Member(String member) { this.member = member; } public String expand(String optParameter) { return member; } } enum Expression implements ComboParameter { LITERAL("1"), STATIC_FIELD("STATIC_FIELD"), LOCAL_VARIABLE("LOCAL_VARIABLE"), INSTANCE_FIELD("INSTANCE_FIELD"); String expr; Expression(String expr) { this.expr = expr; } public String expand(String optParameter) { return expr; } } public static void main(String... args) throws Exception { new combo.ComboTestHelper() .withFilter(LocalStaticDeclarations::notTriviallyIncorrect) .withDimension("CONTAINER", (x, t) -> { x.container = t; }, Container.values()) .withDimension("STATIC_LOCAL", (x, t) -> { x.decl = t; }, StaticLocalDecl.values()) .withDimension("MEMBER", (x, t) -> { x.member = t; }, Member.values()) .withDimension("EXPR", (x, expr) -> x.expr = expr, Expression.values()) .run(LocalStaticDeclarations::new); } Container container; StaticLocalDecl decl; Member member; Expression expr; @Override public void doWork() throws Throwable { newCompilationTask() .withSourceFromTemplate("Test", sourceTemplate) .generate(this::check); } boolean notTriviallyIncorrect() { return decl == StaticLocalDecl.INTERFACE && member == Member.DEFAULT_METHOD || decl != StaticLocalDecl.INTERFACE && member == Member.METHOD; } void check(ComboTask.Result> result) { if (shouldFail()) { Assert.check(result.hasErrors(), "unexpected compilation\n" + result.compilationInfo()); if (!expectedDiagFound(result)) { fail("test failing with unexpected error message\n" + result.compilationInfo()); } } else { Assert.check(!result.hasErrors(), result.compilationInfo()); } } boolean shouldFail() { return (expr == Expression.LOCAL_VARIABLE || expr == Expression.INSTANCE_FIELD); } boolean acceptableExpr() { return (expr == Expression.LITERAL || expr == Expression.STATIC_FIELD); } boolean expectedDiagFound(ComboTask.Result> result) { if (expr == Expression.LOCAL_VARIABLE || expr == Expression.INSTANCE_FIELD) { return result.containsKey("compiler.err.non-static.cant.be.ref"); } return false; } }