langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/Lambda.java
author mchung
Wed, 27 May 2015 13:25:18 -0700
changeset 30846 2b3f379840f0
parent 30730 d3ce7619db2c
permissions -rw-r--r--
8074432: Move jdeps and javap to jdk.jdeps module Reviewed-by: jjg, alanb, erikj

/*
 * Copyright (c) 2013, 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 8008077 8029721 8042451 8043974
 * @summary Test population of reference info for lambda expressions
 *          javac crash for annotated parameter type of lambda in a field
 * @modules jdk.jdeps/com.sun.tools.classfile
 * @ignore 8057687 emit correct byte code an attributes for type annotations
 * @compile -g Driver.java ReferenceInfoUtil.java Lambda.java
 * @run main Driver Lambda
 * @author Werner Dietl
 */

import static com.sun.tools.classfile.TypeAnnotation.TargetType.*;

public class Lambda {

    @TADescription(annotation = "TA", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "TB", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    public String returnMethodRef1() {
        return
                "class Lambda {" +
                "  public String getName() { return \"Lambda!\"; }" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  java.util.function.Function<Lambda, String> lambda() {" +
                "    return @TA @TB Lambda::getName;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "TA", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "TB", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "TC", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "TD", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1 })
    @TADescription(annotation = "TE", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1})
    public String returnMethodRef2() {
        return
                "class Lambda<S, T> {" +
                "  public String getName() { return \"Lambda!\"; }" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  java.util.function.Function<Lambda<Integer, Float>, String> lambda() {" +
                "    return @TA Lambda<@TB @TC Integer, @TD @TE Float>::getName;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "CTA", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "CTB", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "CTC", type = METHOD_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1    })
    public String returnMethodRef3() {
        return
                "class Lambda<S, T> {" +
                "  public String getName() { return \"Lambda!\"; }" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTA {" +
                "  String value();" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTB {" +
                "  int age();" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTC {" +
                "  String name();" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  java.util.function.Function<Lambda<Integer, Float>, String> lambda() {" +
                "    return @CTA(\"x\") Lambda<@CTB(age = 5) Integer, @CTC(name = \"y\") Float>::getName;" +
                "  }" +
                "}";
    }


    @TADescription(annotation = "TA", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "TB", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    public String returnConstructorRef1() {
        return
                "class Lambda {" +
                "  Lambda() { }" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  Runnable lambda() {" +
                "    return @TA @TB Lambda::new;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "TA", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "TB", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "TC", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "TD", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1 })
    @TADescription(annotation = "TE", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1    })
    public String returnConstructorRef2() {
        return
                "class Lambda<S, T> {" +
                "  Lambda() { }" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  Runnable lambda() {" +
                "    return @TA Lambda<@TB @TC Integer, @TD @TE Float>::new;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "CTA", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "CTB", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 0 })
    @TADescription(annotation = "CTC", type = CONSTRUCTOR_REFERENCE,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                genericLocation = { 3, 1    })
    public String returnConstructorRef3() {
        return
                "class Lambda<S, T> {" +
                "  Lambda() { }" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTA {" +
                "  String value();" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTB {" +
                "  int age();" +
                "}" +

                "@Target(ElementType.TYPE_USE)" +
                "@interface CTC {" +
                "  String name();" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  Runnable lambda() {" +
                "    return @CTA(\"x\") Lambda<@CTB(age = 5) Integer, @CTC(name = \"y\") Float>::new;" +
                "  }" +
                "}";
    }


    @TADescription(annotation = "TA", type = METHOD_REFERENCE_TYPE_ARGUMENT,
                 offset = ReferenceInfoUtil.IGNORE_VALUE,
                 typeIndex = 0)
    @TADescription(annotation = "TB", type = METHOD_REFERENCE_TYPE_ARGUMENT,
                 offset = ReferenceInfoUtil.IGNORE_VALUE,
                 typeIndex = 1)
    public String returnMethodRefTA1() {
        return
                "interface Lambda {" +
                "  <S, T> void generic(S p1, T p2);" +
                "}" +

                "class LambdaImpl implements Lambda {" +
                "  public <S, T> void generic(S p1, T p2) {}" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  Lambda lambda(LambdaImpl r) {" +
                "    return r::<@TA Object, @TB Object>generic;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "TA", type = CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT,
                 offset = ReferenceInfoUtil.IGNORE_VALUE,
                 typeIndex = 0)
    @TADescription(annotation = "TB", type = CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT,
                 offset = ReferenceInfoUtil.IGNORE_VALUE,
                 typeIndex = 1)
    public String returnConstructorRefTA2() {
        return
                "interface Lambda {" +
                "  <S, T> void generic(S p1, T p2);" +
                "}" +

                "class LambdaImpl implements Lambda {" +
                "  <S, T> LambdaImpl(S p1, T p2) {}" +
                "  public <S, T> void generic(S p1, T p2) {}" +
                "}" +

                "class %TEST_CLASS_NAME% {" +
                "  Lambda lambda() {" +
                "    return LambdaImpl::<@TA Object, @TB Object>new;" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "TA", type = METHOD_FORMAL_PARAMETER,
                paramIndex = 0)
    @TADescription(annotation = "TB", type = METHOD_FORMAL_PARAMETER,
                paramIndex = 1)
    @TADescription(annotation = "TC", type = METHOD_FORMAL_PARAMETER,
                paramIndex = 1, genericLocation = { 3, 0 })
    @TADescription(annotation = "TD", type = LOCAL_VARIABLE,
                lvarOffset = ReferenceInfoUtil.IGNORE_VALUE,
                lvarLength = ReferenceInfoUtil.IGNORE_VALUE,
                lvarIndex = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "TE", type = CAST,
                offset = ReferenceInfoUtil.IGNORE_VALUE,
                typeIndex = 0)
    public String returnLambdaExpr1() {
        return
                "interface LambdaInt {" +
                "  void lambda(Object p1, List<Object> p2);" +
                "}" +
                "class %TEST_CLASS_NAME% {" +
                "  LambdaInt getLambda() {" +
                "    return (@TA Object x, @TB List<@TC Object> y) -> { @TD Object l = null; System.out.println((@TE Object) l); };" +
                "  }" +
                "}";
    }

    @TADescription(annotation = "TA", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 0)
    public String lambdaField1() {
        return
            "class %TEST_CLASS_NAME% {" +
                " java.util.function.IntUnaryOperator field = (@TA int y) -> 1;" +
            "}";
    }

    @TADescription(annotation = "TA", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 0)
    public String lambdaField2() {
        return
            "class %TEST_CLASS_NAME% {" +
                " static java.util.function.IntUnaryOperator field = (@TA int y) -> 1;" +
            "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE)
    public String returnMethodRefRepeatableAnnotation1() {
        return
                "class Lambda {" +
                        "  public String getName() { return \"Lambda!\"; }" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  java.util.function.Function<Lambda, String> lambda() {" +
                        "    return @RTA @RTA Lambda::getName;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "RTBs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 0 })
    @TADescription(annotation = "RTCs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 0 })
    @TADescription(annotation = "RTDs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 1 })
    @TADescription(annotation = "RTEs", type = METHOD_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 1})
    public String returnMethodRefRepeatableAnnotation2() {
        return
                "class Lambda<S, T> {" +
                        "  public String getName() { return \"Lambda!\"; }" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  java.util.function.Function<Lambda<Integer, Float>, String> lambda() {" +
                        "    return @RTA @RTA Lambda<@RTB @RTB @RTC @RTC Integer, @RTD @RTD @RTE @RTE Float>::getName;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = CONSTRUCTOR_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE)
    public String returnConstructorRefRepeatable1() {
        return
                "class Lambda {" +
                        "  Lambda() { }" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  Runnable lambda() {" +
                        "    return @RTA @RTA Lambda::new;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = CONSTRUCTOR_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "RTBs", type = CONSTRUCTOR_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 0 })
    @TADescription(annotation = "RTCs", type = CONSTRUCTOR_REFERENCE,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            genericLocation = { 3, 1    })
    public String returnConstructorRefRepeatable2() {
        return
                "class Lambda<S, T> {" +
                        "  Lambda() { }" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  Runnable lambda() {" +
                        "    return @RTA @RTA Lambda<@RTB @RTB Integer, @RTC @RTC Float>::new;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_REFERENCE_TYPE_ARGUMENT,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            typeIndex = 0)
    @TADescription(annotation = "RTBs", type = METHOD_REFERENCE_TYPE_ARGUMENT,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            typeIndex = 1)
    public String returnMethodRefTARepeatableAnnotation1() {
        return
                "interface Lambda {" +
                        "  <S, T> void generic(S p1, T p2);" +
                        "}" +

                        "class LambdaImpl implements Lambda {" +
                        "  public <S, T> void generic(S p1, T p2) {}" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  Lambda lambda(LambdaImpl r) {" +
                        "    return r::<@RTA @RTA Object, @RTB @RTB Object>generic;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            typeIndex = 0)
    @TADescription(annotation = "RTBs", type = CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            typeIndex = 1)
    public String returnConstructorRefTARepeatableAnnotation2() {
        return
                "interface Lambda {" +
                        "  <S, T> void generic(S p1, T p2);" +
                        "}" +

                        "class LambdaImpl implements Lambda {" +
                        "  <S, T> LambdaImpl(S p1, T p2) {}" +
                        "  public <S, T> void generic(S p1, T p2) {}" +
                        "}" +

                        "class %TEST_CLASS_NAME% {" +
                        "  Lambda lambda() {" +
                        "    return LambdaImpl::<@RTA @RTA Object, @RTB @RTB Object>new;" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 0)
    @TADescription(annotation = "RTBs", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 1)
    @TADescription(annotation = "RTCs", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 1, genericLocation = { 3, 0 })
    @TADescription(annotation = "RTDs", type = LOCAL_VARIABLE,
            lvarOffset = ReferenceInfoUtil.IGNORE_VALUE,
            lvarLength = ReferenceInfoUtil.IGNORE_VALUE,
            lvarIndex = ReferenceInfoUtil.IGNORE_VALUE)
    @TADescription(annotation = "RTEs", type = CAST,
            offset = ReferenceInfoUtil.IGNORE_VALUE,
            typeIndex = 0)
    public String returnLambdaExprRepeatableAnnotation1() {
        return
                "interface LambdaInt {" +
                        "  void lambda(Object p1, List<Object> p2);" +
                        "}" +
                        "class %TEST_CLASS_NAME% {" +
                        "  LambdaInt getLambda() {" +
                        "    return (@RTA @RTA Object x, @RTB @RTB List<@RTC @RTC Object> y) ->" +
                        " { @RTD @RTD Object l = null; System.out.println((@RTE @RTE Object) l); };" +
                        "  }" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 0)
    public String lambdaFieldRepeatableAnnotation1() {
        return
                "class %TEST_CLASS_NAME% {" +
                        " java.util.function.IntUnaryOperator field = (@RTA @RTA int y) -> 1;" +
                        "}";
    }

    @TADescription(annotation = "RTAs", type = METHOD_FORMAL_PARAMETER,
            paramIndex = 0)
    public String lambdaFieldRepeatableAnnotation2() {
        return
                "class %TEST_CLASS_NAME% {" +
                        " static java.util.function.IntUnaryOperator field = (@RTA @RTA int y) -> 1;" +
                        "}";
    }
}