view test/tools/javac/lambda/FunctionalInterfaceConversionTest.java @ 3019:176472b94f2e

8129962: Investigate performance improvements in langtools combo tests Summary: New combo API that runs all combo instances in a shared javac context (whenever possible). Reviewed-by: jjg, jlahoda, vromero
author mcimadamore
date Mon, 31 Aug 2015 17:33:34 +0100
parents 49d207bf704d
children
line wrap: on
line source

/*
 * Copyright (c) 2012, 2015, 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 8003280 8004102 8006694 8129962
 * @summary Add lambda tests
 *  perform several automated checks in lambda conversion, esp. around accessibility
 *  temporarily workaround combo tests are causing time out in several platforms
 * @library /tools/javac/lib
 * @modules jdk.compiler/com.sun.tools.javac.api
 *          jdk.compiler/com.sun.tools.javac.code
 *          jdk.compiler/com.sun.tools.javac.comp
 *          jdk.compiler/com.sun.tools.javac.main
 *          jdk.compiler/com.sun.tools.javac.tree
 *          jdk.compiler/com.sun.tools.javac.util
 * @build combo.ComboTestHelper
 * @run main FunctionalInterfaceConversionTest
 */

import java.io.IOException;

import combo.ComboInstance;
import combo.ComboParameter;
import combo.ComboTask.Result;
import combo.ComboTestHelper;


public class FunctionalInterfaceConversionTest extends ComboInstance<FunctionalInterfaceConversionTest> {

    enum PackageKind implements ComboParameter {
        NO_PKG(""),
        PKG_A("a");

        String pkg;

        PackageKind(String pkg) {
            this.pkg = pkg;
        }

        @Override
        public String expand(String optParameter) {
            return this == NO_PKG ?
                "" :
                "package " + pkg + ";";
        }

        String getImportStat() {
            return this == NO_PKG ?
                "" :
                "import " + pkg + ".*;";
        }
    }

    enum SamKind implements ComboParameter {
        CLASS("public class Sam {  }"),
        ABSTACT_CLASS("public abstract class Sam {  }"),
        ANNOTATION("public @interface Sam {  }"),
        ENUM("public enum Sam { }"),
        INTERFACE("public interface Sam { \n #{METH1}; \n }");

        String sam_str;

        SamKind(String sam_str) {
            this.sam_str = sam_str;
        }

        @Override
        public String expand(String optParameter) {
            return sam_str;
        }
    }

    enum ModifierKind implements ComboParameter {
        PUBLIC("public"),
        PACKAGE("");

        String modifier_str;

        ModifierKind(String modifier_str) {
            this.modifier_str = modifier_str;
        }

        @Override
        public String expand(String optParameter) {
            return modifier_str;
        }
    }

    enum TypeKind implements ComboParameter {
        EXCEPTION("Exception"),
        PKG_CLASS("PackageClass");

        String typeStr;

        TypeKind(String typeStr) {
            this.typeStr = typeStr;
        }

        @Override
        public String expand(String optParameter) {
            return typeStr;
        }
    }

    enum ExprKind implements ComboParameter {
        LAMBDA("x -> null"),
        MREF("this::m");

        String exprStr;

        ExprKind(String exprStr) {
            this.exprStr = exprStr;
        }

        @Override
        public String expand(String optParameter) {
            return exprStr;
        }
    }

    enum MethodKind implements ComboParameter {
        NONE(""),
        NON_GENERIC("public abstract #{RET} m(#{ARG} s) throws #{THROWN};"),
        GENERIC("public abstract <X> #{RET} m(#{ARG} s) throws #{THROWN};");

        String methodTemplate;

        MethodKind(String methodTemplate) {
            this.methodTemplate = methodTemplate;
        }

        @Override
        public String expand(String optParameter) {
            return methodTemplate;
        }
    }

    public static void main(String[] args) throws Exception {
        new ComboTestHelper<FunctionalInterfaceConversionTest>()
                .withDimension("PKG", (x, pkg) -> x.samPkg = pkg, PackageKind.values())
                .withDimension("MOD", (x, mod) -> x.modKind = mod, ModifierKind.values())
                .withDimension("CLAZZ", (x, sam) -> x.samKind = sam, SamKind.values())
                .withDimension("METH1", (x, meth) -> x.samMeth = meth, MethodKind.values())
                .withDimension("METH2", (x, meth) -> x.clientMeth = meth, MethodKind.values())
                .withDimension("RET", (x, ret) -> x.retType = ret, TypeKind.values())
                .withDimension("ARG", (x, arg) -> x.argType = arg, TypeKind.values())
                .withDimension("THROWN", (x, thrown) -> x.thrownType = thrown, TypeKind.values())
                .withDimension("EXPR", (x, expr) -> x.exprKind = expr, ExprKind.values())
                .run(FunctionalInterfaceConversionTest::new);
    }

    PackageKind samPkg;
    ModifierKind modKind;
    SamKind samKind;
    MethodKind samMeth;
    MethodKind clientMeth;
    TypeKind retType;
    TypeKind argType;
    TypeKind thrownType;
    ExprKind exprKind;

    String samSource = "#{PKG} \n #{CLAZZ}";
    String pkgClassSource = "#{PKG}\n #{MOD} class PackageClass extends Exception { }";
    String clientSource = "#{IMP}\n abstract class Client { \n" +
                           "  Sam s = #{EXPR};\n" +
                           "  #{METH2} \n }";

    @Override
    public void doWork() throws IOException {
        check(newCompilationTask()
                .withSourceFromTemplate("Sam", samSource)
                .withSourceFromTemplate("PackageClass", pkgClassSource)
                .withSourceFromTemplate("Client", clientSource, this::importStmt)
                .analyze());
    }

    ComboParameter importStmt(String name) {
        switch (name) {
            case "IMP": return new ComboParameter.Constant<>(samPkg.getImportStat());
            default: return null;
        }
    }

    void check(Result<?> res) {
        if (res.hasErrors() == checkSamConversion()) {
            fail("Unexpected compilation result; " + res.compilationInfo());
        }
    }

    boolean checkSamConversion() {
        if (samKind != SamKind.INTERFACE) {
            //sam type must be an interface
            return false;
        } else if (samMeth == MethodKind.NONE) {
            //interface must have at least a method
            return false;
        } else if (exprKind == ExprKind.LAMBDA &&
                samMeth != MethodKind.NON_GENERIC) {
            //target method for lambda must be non-generic
            return false;
        } else if (exprKind == ExprKind.MREF &&
                clientMeth == MethodKind.NONE) {
            return false;
        } else if (samPkg != PackageKind.NO_PKG &&
                modKind != ModifierKind.PUBLIC &&
                (retType == TypeKind.PKG_CLASS ||
                argType == TypeKind.PKG_CLASS ||
                thrownType == TypeKind.PKG_CLASS)) {
            //target must not contain inaccessible types
            return false;
        } else {
            return true;
        }
    }
}