8234899: Compiler reports AssertionError for conditional operator ? : and pattern matching for instanceof

Reviewed-by: mcimadamore
This commit is contained in:
Jan Lahoda 2019-12-04 09:38:31 +01:00
parent fb830f6a53
commit f26bdf8476
4 changed files with 240 additions and 0 deletions

View File

@ -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;

View File

@ -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);

View File

@ -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++);
}
}
}

View File

@ -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");
}
}