8234899: Compiler reports AssertionError for conditional operator ? : and pattern matching for instanceof
Reviewed-by: mcimadamore
This commit is contained in:
parent
fb830f6a53
commit
f26bdf8476
@ -46,6 +46,7 @@ import javax.tools.JavaFileObject;
|
||||
|
||||
import com.sun.source.tree.ClassTree;
|
||||
import com.sun.source.tree.CompilationUnitTree;
|
||||
import com.sun.source.tree.Tree;
|
||||
import com.sun.source.util.JavacTask;
|
||||
import com.sun.source.util.TaskEvent;
|
||||
import com.sun.source.util.TaskEvent.Kind;
|
||||
@ -68,6 +69,7 @@ import com.sun.tools.javac.main.Arguments;
|
||||
import com.sun.tools.javac.main.JavaCompiler;
|
||||
import com.sun.tools.javac.model.JavacElements;
|
||||
import com.sun.tools.javac.tree.JCTree.JCClassDecl;
|
||||
import com.sun.tools.javac.tree.JCTree.LetExpr;
|
||||
import com.sun.tools.javac.util.Context;
|
||||
import com.sun.tools.javac.util.DefinedBy;
|
||||
import com.sun.tools.javac.util.DefinedBy.Api;
|
||||
@ -278,6 +280,18 @@ public class JavacTaskPool {
|
||||
* (typically because of cyclic inheritance) the symbol kind of a core class has been touched.
|
||||
*/
|
||||
TreeScanner<Void, Symtab> pollutionScanner = new TreeScanner<Void, Symtab>() {
|
||||
@Override @DefinedBy(Api.COMPILER_TREE)
|
||||
public Void scan(Tree tree, Symtab syms) {
|
||||
if (tree instanceof LetExpr) {
|
||||
LetExpr le = (LetExpr) tree;
|
||||
scan(le.defs, syms);
|
||||
scan(le.expr, syms);
|
||||
return null;
|
||||
} else {
|
||||
return super.scan(tree, syms);
|
||||
}
|
||||
}
|
||||
|
||||
@Override @DefinedBy(Api.COMPILER_TREE)
|
||||
public Void visitClass(ClassTree node, Symtab syms) {
|
||||
Symbol sym = ((JCClassDecl)node).sym;
|
||||
|
@ -727,6 +727,8 @@ public class Gen extends JCTree.Visitor {
|
||||
if (markBranches) result.tree = tree.falsepart;
|
||||
return result;
|
||||
} else if (inner_tree.hasTag(SWITCH_EXPRESSION)) {
|
||||
code.resolvePending();
|
||||
|
||||
boolean prevInCondSwitchExpression = inCondSwitchExpression;
|
||||
Chain prevSwitchExpressionTrueChain = switchExpressionTrueChain;
|
||||
Chain prevSwitchExpressionFalseChain = switchExpressionFalseChain;
|
||||
@ -751,6 +753,8 @@ public class Gen extends JCTree.Visitor {
|
||||
switchExpressionFalseChain = prevSwitchExpressionFalseChain;
|
||||
}
|
||||
} else if (inner_tree.hasTag(LETEXPR) && ((LetExpr) inner_tree).needsCond) {
|
||||
code.resolvePending();
|
||||
|
||||
LetExpr tree = (LetExpr) inner_tree;
|
||||
int limit = code.nextreg;
|
||||
int prevLetExprStart = code.setLetExprStackPos(code.state.stacksize);
|
||||
|
@ -0,0 +1,219 @@
|
||||
/*
|
||||
* Copyright (c) 2019, 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 8234899
|
||||
* @summary Verify behavior w.r.t. preview feature API errors and warnings
|
||||
* @library /tools/lib /tools/javac/lib
|
||||
* @modules
|
||||
* java.base/jdk.internal
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.file
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.compiler/com.sun.tools.javac.util
|
||||
* @build toolbox.ToolBox toolbox.JavacTask
|
||||
* @build combo.ComboTestHelper
|
||||
* @compile --enable-preview -source ${jdk.version} ConditionalExpressionResolvePending.java
|
||||
* @run main/othervm --enable-preview ConditionalExpressionResolvePending
|
||||
*/
|
||||
|
||||
import combo.ComboInstance;
|
||||
import combo.ComboParameter;
|
||||
import combo.ComboTask;
|
||||
import combo.ComboTestHelper;
|
||||
import java.nio.file.Path;
|
||||
import java.nio.file.Paths;
|
||||
import java.util.Iterator;
|
||||
import java.util.Objects;
|
||||
import java.util.function.BiPredicate;
|
||||
import toolbox.ToolBox;
|
||||
|
||||
import javax.tools.JavaFileObject;
|
||||
|
||||
public class ConditionalExpressionResolvePending extends ComboInstance<ConditionalExpressionResolvePending> {
|
||||
protected ToolBox tb;
|
||||
|
||||
ConditionalExpressionResolvePending() {
|
||||
super();
|
||||
tb = new ToolBox();
|
||||
}
|
||||
|
||||
public static void main(String... args) throws Exception {
|
||||
new ComboTestHelper<ConditionalExpressionResolvePending>()
|
||||
.withDimension("METHOD", (x, method) -> x.method = method, Method.values())
|
||||
.withDimension("EXPRESSION", (x, expression) -> x.expression = expression, Expression.values())
|
||||
.withDimension("TRUE", (x, True) -> x.True = True, TestOrDummy.values())
|
||||
.withDimension("FALSE", (x, False) -> x.False = False, TestOrDummy.values())
|
||||
.withDimension("SNIPPET", (x, snippet) -> x.snippet = snippet, Snippet.values())
|
||||
.run(ConditionalExpressionResolvePending::new);
|
||||
}
|
||||
|
||||
private Method method;
|
||||
private Expression expression;
|
||||
private TestOrDummy True;
|
||||
private TestOrDummy False;
|
||||
private Snippet snippet;
|
||||
|
||||
private static final String MAIN_TEMPLATE =
|
||||
"""
|
||||
public class Test {
|
||||
public static boolean doTest(boolean c, Object input) {
|
||||
String clazzName = input.getClass().getName();
|
||||
int len = clazzName.length();
|
||||
#{METHOD}
|
||||
}
|
||||
}
|
||||
""";
|
||||
|
||||
@Override
|
||||
protected void doWork() throws Throwable {
|
||||
Path base = Paths.get(".");
|
||||
|
||||
ComboTask task = newCompilationTask()
|
||||
.withSourceFromTemplate(MAIN_TEMPLATE, pname -> switch (pname) {
|
||||
case "METHOD" -> method;
|
||||
case "EXPRESSION" -> expression;
|
||||
case "TRUE" -> True;
|
||||
case "FALSE" -> False;
|
||||
case "SNIPPET" -> snippet;
|
||||
default -> throw new UnsupportedOperationException(pname);
|
||||
})
|
||||
.withOption("--enable-preview")
|
||||
.withOption("-source")
|
||||
.withOption(String.valueOf(Runtime.version().feature()));
|
||||
|
||||
task.generate(result -> {
|
||||
try {
|
||||
Iterator<? extends JavaFileObject> filesIt = result.get().iterator();
|
||||
JavaFileObject file = filesIt.next();
|
||||
if (filesIt.hasNext()) {
|
||||
throw new IllegalStateException("More than one classfile returned!");
|
||||
}
|
||||
byte[] data = file.openInputStream().readAllBytes();
|
||||
ClassLoader inMemoryLoader = new ClassLoader() {
|
||||
protected Class<?> findClass(String name) throws ClassNotFoundException {
|
||||
if ("Test".equals(name)) {
|
||||
return defineClass(name, data, 0, data.length);
|
||||
}
|
||||
return super.findClass(name);
|
||||
}
|
||||
};
|
||||
Class<?> test = Class.forName("Test", false, inMemoryLoader);
|
||||
java.lang.reflect.Method doTest = test.getDeclaredMethod("doTest", boolean.class, Object.class);
|
||||
runTest((c, input) -> {
|
||||
try {
|
||||
return (boolean) doTest.invoke(null, c, input);
|
||||
} catch (Exception ex) {
|
||||
throw new IllegalStateException(ex);
|
||||
}
|
||||
});
|
||||
} catch (Throwable ex) {
|
||||
throw new IllegalStateException(ex);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void runTest(BiPredicate<Boolean, Object> test) {
|
||||
assertEquals(false, test.test(true, ""));
|
||||
assertEquals(true, test.test(true, 1));
|
||||
assertEquals(false, test.test(false, ""));
|
||||
assertEquals(true, test.test(false, 1));
|
||||
}
|
||||
|
||||
private void assertEquals(Object o1, Object o2) {
|
||||
if (!Objects.equals(o1, o2)) {
|
||||
throw new AssertionError();
|
||||
}
|
||||
}
|
||||
|
||||
public enum Method implements ComboParameter {
|
||||
VARIABLE("""
|
||||
boolean b = #{EXPRESSION};
|
||||
return b;
|
||||
"""),
|
||||
IF("""
|
||||
boolean b;
|
||||
if (#{EXPRESSION}) b = true;
|
||||
else b = false;
|
||||
return b;
|
||||
"""),
|
||||
RETURN("""
|
||||
return #{EXPRESSION};
|
||||
""");
|
||||
private final String body;
|
||||
|
||||
private Method(String body) {
|
||||
this.body = body;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String expand(String optParameter) {
|
||||
return body;
|
||||
}
|
||||
|
||||
}
|
||||
public enum Expression implements ComboParameter {
|
||||
CONDITIONAL("c ? #{TRUE} : #{FALSE}"),
|
||||
AND("(c && #{TRUE}) || (!c && #{FALSE})");
|
||||
private final String expression;
|
||||
|
||||
private Expression(String expression) {
|
||||
this.expression = expression;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String expand(String optParameter) {
|
||||
return expression;
|
||||
}
|
||||
}
|
||||
public enum TestOrDummy implements ComboParameter {
|
||||
TEST("!(#{SNIPPET})"),
|
||||
DUMMY("input.getClass() == Integer.class");
|
||||
private final String code;
|
||||
private TestOrDummy(String code) {
|
||||
this.code = code;
|
||||
}
|
||||
@Override
|
||||
public String expand(String optParameter) {
|
||||
return code;
|
||||
}
|
||||
}
|
||||
public enum Snippet implements ComboParameter {
|
||||
PATTERN("input instanceof String sX"),
|
||||
SWITCH_EXPRESSION("switch (len) { case 16 -> {boolean r = true; yield r; } default -> {boolean r = false; yield r; } }"),
|
||||
SWITCH_EXPRESSION_STRING("switch (clazzName) { case \"java.lang.String\"-> {boolean r = true; yield r; } default -> {boolean r = false; yield r; } }");
|
||||
private static int idx;
|
||||
private final String snippet;
|
||||
|
||||
private Snippet(String snippet) {
|
||||
this.snippet = snippet;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String expand(String optParameter) {
|
||||
return snippet.replace("sX", "s" + idx++);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
@ -161,6 +161,9 @@ public class BindingsTest1 {
|
||||
String s2 = s;
|
||||
}
|
||||
|
||||
boolean result = (o1 instanceof String a1) ? (o1 instanceof String a2) : (!(o1 instanceof String a3));
|
||||
boolean result2 = (o1 instanceof String a1) ? (o1 instanceof String a2) : (!(switch (0) { default -> false; }));
|
||||
|
||||
System.out.println("BindingsTest1 complete");
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user