2011-11-28 15:56:42 +00:00
|
|
|
/*
|
2020-06-09 13:37:53 +00:00
|
|
|
* Copyright (c) 2011, 2020, Oracle and/or its affiliates. All rights reserved.
|
2011-11-28 15:56:42 +00:00
|
|
|
* 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
|
2015-08-31 16:33:34 +00:00
|
|
|
* @bug 7115050 8003280 8005852 8006694 8129962
|
2012-11-17 19:01:03 +00:00
|
|
|
* @summary Add lambda tests
|
|
|
|
* Add parser support for lambda expressions
|
2013-01-23 20:57:40 +00:00
|
|
|
* temporarily workaround combo tests are causing time out in several platforms
|
2015-08-31 16:33:34 +00:00
|
|
|
* @library /tools/javac/lib
|
|
|
|
* @modules jdk.compiler/com.sun.tools.javac.api
|
2020-01-28 08:13:27 +00:00
|
|
|
* jdk.compiler/com.sun.tools.javac.file
|
2015-08-31 16:33:34 +00:00
|
|
|
* jdk.compiler/com.sun.tools.javac.util
|
|
|
|
* @build combo.ComboTestHelper
|
|
|
|
|
|
|
|
* @run main LambdaParserTest
|
2011-11-28 15:56:42 +00:00
|
|
|
*/
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
import java.io.IOException;
|
2018-02-22 20:49:32 +00:00
|
|
|
import java.util.Arrays;
|
2013-01-23 20:57:40 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
import combo.ComboInstance;
|
|
|
|
import combo.ComboParameter;
|
|
|
|
import combo.ComboTask.Result;
|
|
|
|
import combo.ComboTestHelper;
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
public class LambdaParserTest extends ComboInstance<LambdaParserTest> {
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum LambdaKind implements ComboParameter {
|
2011-11-28 15:56:42 +00:00
|
|
|
NILARY_EXPR("()->x"),
|
|
|
|
NILARY_STMT("()->{ return x; }"),
|
2015-08-31 16:33:34 +00:00
|
|
|
ONEARY_SHORT_EXPR("#{NAME}->x"),
|
|
|
|
ONEARY_SHORT_STMT("#{NAME}->{ return x; }"),
|
|
|
|
ONEARY_EXPR("(#{MOD[0]} #{TYPE[0]} #{NAME})->x"),
|
|
|
|
ONEARY_STMT("(#{MOD[0]} #{TYPE[0]} #{NAME})->{ return x; }"),
|
|
|
|
TWOARY_EXPR("(#{MOD[0]} #{TYPE[0]} #{NAME}, #{MOD[1]} #{TYPE[1]} y)->x"),
|
|
|
|
TWOARY_STMT("(#{MOD[0]} #{TYPE[0]} #{NAME}, #{MOD[1]} #{TYPE[1]} y)->{ return x; }");
|
2011-11-28 15:56:42 +00:00
|
|
|
|
|
|
|
String lambdaTemplate;
|
|
|
|
|
|
|
|
LambdaKind(String lambdaTemplate) {
|
|
|
|
this.lambdaTemplate = lambdaTemplate;
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return lambdaTemplate;
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int arity() {
|
|
|
|
switch (this) {
|
|
|
|
case NILARY_EXPR:
|
|
|
|
case NILARY_STMT: return 0;
|
|
|
|
case ONEARY_SHORT_EXPR:
|
|
|
|
case ONEARY_SHORT_STMT:
|
|
|
|
case ONEARY_EXPR:
|
|
|
|
case ONEARY_STMT: return 1;
|
|
|
|
case TWOARY_EXPR:
|
|
|
|
case TWOARY_STMT: return 2;
|
|
|
|
default: throw new AssertionError("Invalid lambda kind " + this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean isShort() {
|
|
|
|
return this == ONEARY_SHORT_EXPR ||
|
|
|
|
this == ONEARY_SHORT_STMT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum LambdaParameterName implements ComboParameter {
|
2013-01-17 18:15:20 +00:00
|
|
|
IDENT("x"),
|
|
|
|
UNDERSCORE("_");
|
|
|
|
|
|
|
|
String nameStr;
|
|
|
|
|
|
|
|
LambdaParameterName(String nameStr) {
|
|
|
|
this.nameStr = nameStr;
|
|
|
|
}
|
2015-08-31 16:33:34 +00:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return nameStr;
|
|
|
|
}
|
2013-01-17 18:15:20 +00:00
|
|
|
}
|
|
|
|
|
2018-02-22 20:49:32 +00:00
|
|
|
enum SourceKind {
|
2018-10-04 15:37:08 +00:00
|
|
|
SOURCE_10("10"),
|
|
|
|
SOURCE_11("11");
|
2018-02-22 20:49:32 +00:00
|
|
|
|
|
|
|
String sourceNumber;
|
|
|
|
|
|
|
|
SourceKind(String sourceNumber) {
|
|
|
|
this.sourceNumber = sourceNumber;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum LambdaParameterKind implements ComboParameter {
|
2018-02-22 20:49:32 +00:00
|
|
|
|
|
|
|
IMPLICIT_1("", ExplicitKind.IMPLICIT),
|
|
|
|
IMPLICIT_2("var", ExplicitKind.IMPLICIT_VAR),
|
2018-10-04 15:37:08 +00:00
|
|
|
EXPLICIT_SIMPLE("A", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_SIMPLE_ARR1("A[]", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_SIMPLE_ARR2("A[][]", ExplicitKind.EXPLICIT),
|
2018-02-22 20:49:32 +00:00
|
|
|
EXPLICIT_VARARGS("A...", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_GENERIC1("A<X>", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_GENERIC2("A<? extends X, ? super Y>", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_GENERIC2_VARARGS("A<? extends X, ? super Y>...", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_GENERIC2_ARR1("A<? extends X, ? super Y>[]", ExplicitKind.EXPLICIT),
|
|
|
|
EXPLICIT_GENERIC2_ARR2("A<? extends X, ? super Y>[][]", ExplicitKind.EXPLICIT);
|
|
|
|
|
|
|
|
enum ExplicitKind {
|
|
|
|
IMPLICIT,
|
|
|
|
IMPLICIT_VAR,
|
|
|
|
EXPLICIT;
|
|
|
|
}
|
2011-11-28 15:56:42 +00:00
|
|
|
|
|
|
|
String parameterType;
|
2018-02-22 20:49:32 +00:00
|
|
|
ExplicitKind explicitKind;
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2018-02-21 05:29:04 +00:00
|
|
|
|
2018-02-22 20:49:32 +00:00
|
|
|
LambdaParameterKind(String parameterType, ExplicitKind ekind) {
|
|
|
|
this.parameterType = parameterType;
|
|
|
|
this.explicitKind = ekind;
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
2012-11-30 15:14:48 +00:00
|
|
|
|
|
|
|
boolean isVarargs() {
|
|
|
|
return this == EXPLICIT_VARARGS ||
|
|
|
|
this == EXPLICIT_GENERIC2_VARARGS;
|
|
|
|
}
|
2015-08-31 16:33:34 +00:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return parameterType;
|
|
|
|
}
|
2018-02-22 20:49:32 +00:00
|
|
|
|
|
|
|
ExplicitKind explicitKind(SourceKind sk) {
|
2018-10-04 15:37:08 +00:00
|
|
|
return explicitKind;
|
2018-02-22 20:49:32 +00:00
|
|
|
}
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum ModifierKind implements ComboParameter {
|
2011-11-28 15:56:42 +00:00
|
|
|
NONE(""),
|
|
|
|
FINAL("final"),
|
2018-02-22 20:49:32 +00:00
|
|
|
PUBLIC("public"),
|
|
|
|
ANNO("@A");
|
2011-11-28 15:56:42 +00:00
|
|
|
|
|
|
|
String modifier;
|
|
|
|
|
|
|
|
ModifierKind(String modifier) {
|
|
|
|
this.modifier = modifier;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean compatibleWith(LambdaParameterKind pk) {
|
|
|
|
switch (this) {
|
|
|
|
case PUBLIC: return false;
|
2018-02-22 20:49:32 +00:00
|
|
|
case ANNO:
|
|
|
|
case FINAL: return pk != LambdaParameterKind.IMPLICIT_1;
|
2011-11-28 15:56:42 +00:00
|
|
|
case NONE: return true;
|
|
|
|
default: throw new AssertionError("Invalid modifier kind " + this);
|
|
|
|
}
|
|
|
|
}
|
2015-08-31 16:33:34 +00:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return modifier;
|
|
|
|
}
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum ExprKind implements ComboParameter {
|
|
|
|
NONE("#{LAMBDA}#{SUBEXPR}"),
|
|
|
|
SINGLE_PAREN1("(#{LAMBDA}#{SUBEXPR})"),
|
|
|
|
SINGLE_PAREN2("(#{LAMBDA})#{SUBEXPR}"),
|
|
|
|
DOUBLE_PAREN1("((#{LAMBDA}#{SUBEXPR}))"),
|
|
|
|
DOUBLE_PAREN2("((#{LAMBDA})#{SUBEXPR})"),
|
|
|
|
DOUBLE_PAREN3("((#{LAMBDA}))#{SUBEXPR}");
|
2011-11-28 15:56:42 +00:00
|
|
|
|
|
|
|
String expressionTemplate;
|
|
|
|
|
|
|
|
ExprKind(String expressionTemplate) {
|
|
|
|
this.expressionTemplate = expressionTemplate;
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return expressionTemplate;
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
enum SubExprKind implements ComboParameter {
|
2011-11-28 15:56:42 +00:00
|
|
|
NONE(""),
|
|
|
|
SELECT_FIELD(".f"),
|
|
|
|
SELECT_METHOD(".f()"),
|
|
|
|
SELECT_NEW(".new Foo()"),
|
|
|
|
POSTINC("++"),
|
|
|
|
POSTDEC("--");
|
|
|
|
|
|
|
|
String subExpression;
|
|
|
|
|
|
|
|
SubExprKind(String subExpression) {
|
|
|
|
this.subExpression = subExpression;
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
@Override
|
|
|
|
public String expand(String optParameter) {
|
|
|
|
return subExpression;
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
2015-08-31 16:33:34 +00:00
|
|
|
}
|
2013-01-08 13:47:57 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
public static void main(String... args) throws Exception {
|
|
|
|
new ComboTestHelper<LambdaParserTest>()
|
|
|
|
.withFilter(LambdaParserTest::redundantTestFilter)
|
2018-02-22 20:49:32 +00:00
|
|
|
.withDimension("SOURCE", (x, sk) -> x.sk = sk, SourceKind.values())
|
2015-08-31 16:33:34 +00:00
|
|
|
.withDimension("LAMBDA", (x, lk) -> x.lk = lk, LambdaKind.values())
|
|
|
|
.withDimension("NAME", (x, name) -> x.pn = name, LambdaParameterName.values())
|
|
|
|
.withArrayDimension("TYPE", (x, type, idx) -> x.pks[idx] = type, 2, LambdaParameterKind.values())
|
|
|
|
.withArrayDimension("MOD", (x, mod, idx) -> x.mks[idx] = mod, 2, ModifierKind.values())
|
2020-06-09 13:37:53 +00:00
|
|
|
.withDimension("EXPR", (x, exp) -> x.exp = exp, ExprKind.values())
|
|
|
|
.withDimension("SUBEXPR", (x, sub) -> x.sub = sub, SubExprKind.values())
|
2015-08-31 16:33:34 +00:00
|
|
|
.run(LambdaParserTest::new);
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
LambdaParameterKind[] pks = new LambdaParameterKind[2];
|
|
|
|
ModifierKind[] mks = new ModifierKind[2];
|
2011-11-28 15:56:42 +00:00
|
|
|
LambdaKind lk;
|
2013-01-17 18:15:20 +00:00
|
|
|
LambdaParameterName pn;
|
2018-02-22 20:49:32 +00:00
|
|
|
SourceKind sk;
|
2020-06-09 13:37:53 +00:00
|
|
|
ExprKind exp;
|
|
|
|
SubExprKind sub;
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
boolean redundantTestFilter() {
|
2020-06-09 13:37:53 +00:00
|
|
|
if (sub == SubExprKind.NONE) {
|
|
|
|
switch (exp) {
|
|
|
|
//followings combinations with empty sub-expressions produces the same source
|
|
|
|
case SINGLE_PAREN2, DOUBLE_PAREN2, DOUBLE_PAREN3: return false;
|
2015-08-31 16:33:34 +00:00
|
|
|
}
|
2020-06-09 13:37:53 +00:00
|
|
|
} else {
|
|
|
|
switch (lk) {
|
|
|
|
//any non-empty subexpression does not combine with lambda statements
|
|
|
|
case NILARY_STMT, ONEARY_SHORT_STMT, ONEARY_STMT, TWOARY_STMT: return false;
|
2015-08-31 16:33:34 +00:00
|
|
|
}
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
2020-06-09 13:37:53 +00:00
|
|
|
switch (lk) {
|
|
|
|
//parameters not present in the expression are redundant
|
|
|
|
case NILARY_EXPR, NILARY_STMT:
|
|
|
|
if (pn.ordinal() != 0) return false;
|
|
|
|
case ONEARY_SHORT_EXPR, ONEARY_SHORT_STMT:
|
|
|
|
if (pks[0].ordinal() != 0 || mks[0].ordinal() != 0) return false;
|
|
|
|
case ONEARY_EXPR, ONEARY_STMT :
|
|
|
|
if (pks[1].ordinal() != 0 || mks[1].ordinal() != 0) return false;
|
|
|
|
}
|
2015-08-31 16:33:34 +00:00
|
|
|
return true;
|
2011-11-28 15:56:42 +00:00
|
|
|
}
|
|
|
|
|
2018-02-22 20:49:32 +00:00
|
|
|
String template = "@interface A { }\n" +
|
|
|
|
"class Test {\n" +
|
|
|
|
" SAM s = #{EXPR};\n" +
|
|
|
|
"}";
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
@Override
|
|
|
|
public void doWork() throws IOException {
|
2017-09-01 12:04:20 +00:00
|
|
|
newCompilationTask()
|
2018-02-22 20:49:32 +00:00
|
|
|
.withOptions(Arrays.asList("-source", sk.sourceNumber))
|
2015-08-31 16:33:34 +00:00
|
|
|
.withSourceFromTemplate(template)
|
2017-09-01 12:04:20 +00:00
|
|
|
.parse(this::check);
|
2015-08-31 16:33:34 +00:00
|
|
|
}
|
2011-11-28 15:56:42 +00:00
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
void check(Result<?> res) {
|
|
|
|
boolean errorExpected = (lk.arity() > 0 && !mks[0].compatibleWith(pks[0])) ||
|
|
|
|
(lk.arity() > 1 && !mks[1].compatibleWith(pks[1]));
|
2011-11-28 15:56:42 +00:00
|
|
|
|
|
|
|
if (lk.arity() == 2 &&
|
2018-02-22 20:49:32 +00:00
|
|
|
(pks[0].explicitKind(sk) != pks[1].explicitKind(sk) ||
|
2015-08-31 16:33:34 +00:00
|
|
|
pks[0].isVarargs())) {
|
2011-11-28 15:56:42 +00:00
|
|
|
errorExpected = true;
|
|
|
|
}
|
|
|
|
|
2013-01-17 18:15:20 +00:00
|
|
|
errorExpected |= pn == LambdaParameterName.UNDERSCORE &&
|
|
|
|
lk.arity() > 0;
|
|
|
|
|
2018-10-04 15:37:08 +00:00
|
|
|
for (int i = 0; i < lk.arity(); i++) {
|
|
|
|
if (!lk.isShort() &&
|
|
|
|
pks[i].explicitKind(sk) == LambdaParameterKind.ExplicitKind.IMPLICIT_VAR &&
|
|
|
|
sk == SourceKind.SOURCE_10) {
|
|
|
|
errorExpected = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-31 16:33:34 +00:00
|
|
|
if (errorExpected != res.hasErrors()) {
|
|
|
|
fail("invalid diagnostics for source:\n" +
|
|
|
|
res.compilationInfo() +
|
|
|
|
"\nFound error: " + res.hasErrors() +
|
2011-11-28 15:56:42 +00:00
|
|
|
"\nExpected error: " + errorExpected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|