From c4ebb7f3c0e8a9b546fb5b7e3e1d994ac0353763 Mon Sep 17 00:00:00 2001 From: Srikanth Adayapalam Date: Fri, 13 Mar 2015 13:15:24 +0530 Subject: [PATCH 1/6] 8071847: java.lang.NullPointerException at com.sun.tools.javac.code.Types.elemtype(Types.java:1870) Class file reader should recover from malformed class files that contain methods incorrectly flagged as variable arity methods. Reviewed-by: jlahoda --- .../com/sun/tools/javac/jvm/ClassReader.java | 10 ++ .../tools/javac/resources/compiler.properties | 3 + .../test/tools/javac/T8071847/T8071847.java | 108 ++++++++++++++++++ .../tools/javac/diags/examples.not-yet.txt | 1 + 4 files changed, 122 insertions(+) create mode 100644 langtools/test/tools/javac/T8071847/T8071847.java diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/ClassReader.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/ClassReader.java index 307119aa9bd..641098642ae 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/ClassReader.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/ClassReader.java @@ -56,6 +56,7 @@ import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition; import static com.sun.tools.javac.code.Flags.*; import static com.sun.tools.javac.code.Kinds.Kind.*; +import static com.sun.tools.javac.code.TypeTag.ARRAY; import static com.sun.tools.javac.code.TypeTag.CLASS; import static com.sun.tools.javac.code.TypeTag.TYPEVAR; import static com.sun.tools.javac.jvm.ClassFile.*; @@ -2006,6 +2007,15 @@ public class ClassReader { } if (saveParameterNames) setParameterNames(m, type); + + if ((flags & VARARGS) != 0) { + final Type last = type.getParameterTypes().last(); + if (last == null || !last.hasTag(ARRAY)) { + m.flags_field &= ~VARARGS; + throw badClassFile("malformed.vararg.method", m); + } + } + return m; } diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties index c1bb9da153e..c3a8900c57c 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties @@ -1847,6 +1847,9 @@ compiler.misc.unicode.str.not.supported=\ compiler.misc.undecl.type.var=\ undeclared type variable: {0} +compiler.misc.malformed.vararg.method=\ + class file contains malformed variable arity method: {0} + compiler.misc.wrong.version=\ class file has wrong version {0}.{1}, should be {2}.{3} diff --git a/langtools/test/tools/javac/T8071847/T8071847.java b/langtools/test/tools/javac/T8071847/T8071847.java new file mode 100644 index 00000000000..1d691e15b02 --- /dev/null +++ b/langtools/test/tools/javac/T8071847/T8071847.java @@ -0,0 +1,108 @@ +/* + * Copyright (c) 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 8071847 + * @summary Verify proper termination when instance initialization method uses invalid flags + * @compile T8071847.java + * @run main T8071847 +*/ + + +import java.io.*; +import java.util.*; + +public class T8071847 { + String testclass="invalidFlags.class"; + String testclassHexString = + "CAFEBABE00000031000D0A0003000A07000B07000C0100063C696E69743E0100" + + "03282956010004436F646501000F4C696E654E756D6265725461626C6501000A" + + "536F7572636546696C65010009546573742E6A6176610C0004000501000C696E" + + "76616C6964466C6167730100106A6176612F6C616E672F4F626A656374002000" + + "02000300000000000100A000040005000100060000001D00010001000000052A" + + "B70001B10000000100070000000600010000000100010008000000020009"; + + String testJavaFile = "testInvalidFlags.java"; + String testJavaSource ="public class testInvalidFlags extends invalidFlags {" + + "invalidFlags c = null;" + + "public testInvalidFlags() { c = new invalidFlags(); }" + + "public static void main(String... args) { " + + "new testInvalidFlags();}}"; + + public static void main(String[] args) throws Exception { + new T8071847().run(); + } + + public void run() throws IOException { + writeHexFile(testclass,testclassHexString); + writeTestFile(testJavaFile, testJavaSource); + javac(testJavaFile); + } + + File writeTestFile(String fname, String source) throws IOException { + File f = new File(fname); + PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(f))); + out.println(source); + out.close(); + return f; + } + + byte[] hexToByte(String str) { + char[] CA = str.toCharArray(); + byte[] byteArry = new byte[str.length()/2]; + int bi = 0; + for (int i = 0; i javacArgs = new ArrayList<>(); + javacArgs.addAll(Arrays.asList("-XDrawDiagnostics", "-cp", ".", "-d", ".", className)); + rc = com.sun.tools.javac.Main.compile( + javacArgs.toArray(new String[javacArgs.size()]),out); + out.close(); + if (rc > 1) { + System.out.println(sw.toString()); + throw new Error("javac " + className + " failed. rc=" + rc); + } + if (rc != 1 || !sw.toString().contains("compiler.misc.malformed.vararg.method")) + throw new RuntimeException("Unexpected output" + sw.toString()); + return sw.toString(); + } +} diff --git a/langtools/test/tools/javac/diags/examples.not-yet.txt b/langtools/test/tools/javac/diags/examples.not-yet.txt index 9f58cf88154..795f7166008 100644 --- a/langtools/test/tools/javac/diags/examples.not-yet.txt +++ b/langtools/test/tools/javac/diags/examples.not-yet.txt @@ -87,6 +87,7 @@ compiler.misc.type.req.exact compiler.misc.unable.to.access.file # ClassFile compiler.misc.undecl.type.var # ClassReader compiler.misc.unicode.str.not.supported # ClassReader +compiler.misc.malformed.vararg.method # ClassReader compiler.misc.verbose.retro # UNUSED compiler.misc.verbose.retro.with # UNUSED compiler.misc.verbose.retro.with.list # UNUSED From f1f1f0f7e110a60035673b2173bf7619581c1101 Mon Sep 17 00:00:00 2001 From: Srikanth Adayapalam Date: Fri, 13 Mar 2015 10:25:46 +0530 Subject: [PATCH 2/6] 8074148: Attr.visitBinary flags error at wrong position Reviewed-by: vromero --- .../classes/com/sun/tools/javac/comp/Attr.java | 2 +- .../test/tools/javac/expression/BinopVoidTest.java | 14 ++++++++++++++ .../test/tools/javac/expression/BinopVoidTest.out | 5 +++++ 3 files changed, 20 insertions(+), 1 deletion(-) create mode 100644 langtools/test/tools/javac/expression/BinopVoidTest.java create mode 100644 langtools/test/tools/javac/expression/BinopVoidTest.out diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index 25d25c3b863..953c6129f12 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -2942,7 +2942,7 @@ public class Attr extends JCTree.Visitor { public void visitBinary(JCBinary tree) { // Attribute arguments. Type left = chk.checkNonVoid(tree.lhs.pos(), attribExpr(tree.lhs, env)); - Type right = chk.checkNonVoid(tree.lhs.pos(), attribExpr(tree.rhs, env)); + Type right = chk.checkNonVoid(tree.rhs.pos(), attribExpr(tree.rhs, env)); // Find operator. Symbol operator = tree.operator = operators.resolveBinary(tree, tree.getTag(), left, right); Type owntype = types.createErrorType(tree.type); diff --git a/langtools/test/tools/javac/expression/BinopVoidTest.java b/langtools/test/tools/javac/expression/BinopVoidTest.java new file mode 100644 index 00000000000..399845a54f7 --- /dev/null +++ b/langtools/test/tools/javac/expression/BinopVoidTest.java @@ -0,0 +1,14 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8074148 + * @summary Attr.visitBinary flags error at wrong position + * + * @compile/fail/ref=BinopVoidTest.out -XDrawDiagnostics BinopVoidTest.java + */ + +class BinopVoidTest { + void foo() {} + int x = 10 + foo(); + int y = foo() + 10; + int z = foo() + foo(); +} diff --git a/langtools/test/tools/javac/expression/BinopVoidTest.out b/langtools/test/tools/javac/expression/BinopVoidTest.out new file mode 100644 index 00000000000..1d8265b99a5 --- /dev/null +++ b/langtools/test/tools/javac/expression/BinopVoidTest.out @@ -0,0 +1,5 @@ +BinopVoidTest.java:11:21: compiler.err.void.not.allowed.here +BinopVoidTest.java:12:16: compiler.err.void.not.allowed.here +BinopVoidTest.java:13:16: compiler.err.void.not.allowed.here +BinopVoidTest.java:13:24: compiler.err.void.not.allowed.here +4 errors From c03dd06d9a011baa8397da3a7b94da48dd3effe3 Mon Sep 17 00:00:00 2001 From: Maurizio Cimadamore Date: Thu, 19 Mar 2015 11:39:36 +0000 Subject: [PATCH 3/6] 8074381: java.lang.AssertionError during compiling Add extra functional interface check to prevent crash during code generation Reviewed-by: vromero --- .../com/sun/tools/javac/comp/Attr.java | 12 +++++ .../tools/javac/lambda/8074381/T8074381a.java | 33 ++++++++++++++ .../tools/javac/lambda/8074381/T8074381a.out | 4 ++ .../tools/javac/lambda/8074381/T8074381b.java | 44 +++++++++++++++++++ .../tools/javac/lambda/8074381/T8074381b.out | 2 + 5 files changed, 95 insertions(+) create mode 100644 langtools/test/tools/javac/lambda/8074381/T8074381a.java create mode 100644 langtools/test/tools/javac/lambda/8074381/T8074381a.out create mode 100644 langtools/test/tools/javac/lambda/8074381/T8074381b.java create mode 100644 langtools/test/tools/javac/lambda/8074381/T8074381b.out diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index 953c6129f12..ce01abdc960 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -40,11 +40,13 @@ import com.sun.tools.javac.code.Lint.LintCategory; import com.sun.tools.javac.code.Scope.WriteableScope; import com.sun.tools.javac.code.Symbol.*; import com.sun.tools.javac.code.Type.*; +import com.sun.tools.javac.code.Types.FunctionDescriptorLookupError; import com.sun.tools.javac.comp.Check.CheckContext; import com.sun.tools.javac.comp.DeferredAttr.AttrMode; import com.sun.tools.javac.comp.Infer.InferenceContext; import com.sun.tools.javac.comp.Infer.FreeTypeListener; import com.sun.tools.javac.jvm.*; +import com.sun.tools.javac.resources.CompilerProperties.Fragments; import com.sun.tools.javac.tree.*; import com.sun.tools.javac.tree.JCTree.*; import com.sun.tools.javac.tree.JCTree.JCPolyExpression.*; @@ -2871,6 +2873,16 @@ public class Attr extends JCTree.Visitor { names.empty, List.of(fExpr.targets.head), ABSTRACT); if (csym != null) { chk.checkImplementations(env.tree, csym, csym); + try { + //perform an additional functional interface check on the synthetic class, + //as there may be spurious errors for raw targets - because of existing issues + //with membership and inheritance (see JDK-8074570). + csym.flags_field |= INTERFACE; + types.findDescriptorType(csym.type); + } catch (FunctionDescriptorLookupError err) { + resultInfo.checkContext.report(fExpr, + diags.fragment(Fragments.NoSuitableFunctionalIntfInst(fExpr.targets.head))); + } } } catch (Types.FunctionDescriptorLookupError ex) { JCDiagnostic cause = ex.getDiagnostic(); diff --git a/langtools/test/tools/javac/lambda/8074381/T8074381a.java b/langtools/test/tools/javac/lambda/8074381/T8074381a.java new file mode 100644 index 00000000000..7b7b9dd6cc2 --- /dev/null +++ b/langtools/test/tools/javac/lambda/8074381/T8074381a.java @@ -0,0 +1,33 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8074381 + * @summary java.lang.AssertionError during compiling + * @compile/fail/ref=T8074381a.out -XDrawDiagnostics T8074381a.java + */ +class T8074381a { + interface Sup { + boolean m(X x); + } + + interface Sub extends Sup { + boolean m(String s); + } + + void testRaw() { + Sub s1 = c -> true; + Sub s2 = Boolean::new; + Sub s3 = new Sub() { + @Override + public boolean m(String o) { return true; } + }; + } + + void testNonRaw() { + Sub s1 = c -> true; + Sub s2 = Boolean::new; + Sub s3 = new Sub() { + @Override + public boolean m(String o) { return true; } + }; + } +} diff --git a/langtools/test/tools/javac/lambda/8074381/T8074381a.out b/langtools/test/tools/javac/lambda/8074381/T8074381a.out new file mode 100644 index 00000000000..f7ba487c586 --- /dev/null +++ b/langtools/test/tools/javac/lambda/8074381/T8074381a.out @@ -0,0 +1,4 @@ +T8074381a.java:17:18: compiler.err.prob.found.req: (compiler.misc.no.suitable.functional.intf.inst: T8074381a.Sub) +T8074381a.java:18:18: compiler.err.prob.found.req: (compiler.misc.no.suitable.functional.intf.inst: T8074381a.Sub) +T8074381a.java:19:28: compiler.err.does.not.override.abstract: compiler.misc.anonymous.class: T8074381a$1, m(java.lang.Object), T8074381a.Sup +3 errors diff --git a/langtools/test/tools/javac/lambda/8074381/T8074381b.java b/langtools/test/tools/javac/lambda/8074381/T8074381b.java new file mode 100644 index 00000000000..3c28b8095e1 --- /dev/null +++ b/langtools/test/tools/javac/lambda/8074381/T8074381b.java @@ -0,0 +1,44 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8074381 + * @summary java.lang.AssertionError during compiling + * @compile/fail/ref=T8074381b.out -XDrawDiagnostics T8074381b.java + */ +import java.util.function.BiConsumer; +import java.util.function.Consumer; + +class T8074381b { + + @SuppressWarnings("unchecked") + public Invocation resolve(Handler handler) { + return new Invocation((t) -> handler.handle((String) t)); + } + + public static class Handler { + public void handle(String s) { + System.out.println(s); + } + } + + public static class Invocation { + public final ThrowingConsumer consumer; + + public Invocation(final ThrowingConsumer consumer) { + this.consumer = consumer; + } + } + + @FunctionalInterface + public interface ThrowingConsumer extends BiConsumer> { + @Override + default void accept(final T elem, final Consumer errorHandler) { + try { + acceptThrows(elem); + } catch (final Throwable e) { + errorHandler.accept(e); + } + } + + void acceptThrows(T elem) throws Throwable; + } +} diff --git a/langtools/test/tools/javac/lambda/8074381/T8074381b.out b/langtools/test/tools/javac/lambda/8074381/T8074381b.out new file mode 100644 index 00000000000..7965e2e3734 --- /dev/null +++ b/langtools/test/tools/javac/lambda/8074381/T8074381b.out @@ -0,0 +1,2 @@ +T8074381b.java:14:16: compiler.err.cant.apply.symbol: kindname.constructor, Invocation, T8074381b.ThrowingConsumer, @383, kindname.class, T8074381b.Invocation, (compiler.misc.no.conforming.assignment.exists: (compiler.misc.no.suitable.functional.intf.inst: T8074381b.ThrowingConsumer)) +1 error From 582a4ebd92e60f95ae77d25eb00c6a146e727fab Mon Sep 17 00:00:00 2001 From: Maurizio Cimadamore Date: Thu, 19 Mar 2015 11:40:07 +0000 Subject: [PATCH 4/6] 8074100: Turn Type.Mapping into a true visitor Replace Type.Mapping with a true visitor in Types Reviewed-by: jlahoda, vromero --- .../com/sun/tools/javac/code/Type.java | 140 +++++++++--------- .../com/sun/tools/javac/code/Types.java | 51 +++---- .../com/sun/tools/javac/comp/Attr.java | 6 +- .../sun/tools/javac/comp/DeferredAttr.java | 26 +--- .../com/sun/tools/javac/comp/Infer.java | 40 ++--- .../com/sun/tools/javac/comp/Resolve.java | 5 +- .../com/sun/tools/javac/util/List.java | 9 ++ 7 files changed, 137 insertions(+), 140 deletions(-) diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java index 502fc412e50..e2d3208c211 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java @@ -31,10 +31,12 @@ import java.util.EnumMap; import java.util.EnumSet; import java.util.Map; import java.util.Set; +import java.util.function.Function; import javax.lang.model.type.*; import com.sun.tools.javac.code.Symbol.*; +import com.sun.tools.javac.code.Types.MapVisitor; import com.sun.tools.javac.util.*; import com.sun.tools.javac.util.DefinedBy.Api; import static com.sun.tools.javac.code.BoundKind.*; @@ -218,33 +220,81 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { /** An abstract class for mappings from types to types */ - public static abstract class Mapping { - private String name; - public Mapping(String name) { - this.name = name; + public static abstract class TypeMapping extends Types.MapVisitor implements Function { + + @Override + public Type apply(Type type) { + return visit(type); } - public abstract Type apply(Type t); - public String toString() { - return name; + + List visit(List ts, S s) { + return ts.map(t -> visit(t, s)); + } + + @Override + public Type visitClassType(ClassType t, S s) { + Type outer = t.getEnclosingType(); + Type outer1 = visit(outer, s); + List typarams = t.getTypeArguments(); + List typarams1 = visit(typarams, s); + if (outer1 == outer && typarams1 == typarams) return t; + else return new ClassType(outer1, typarams1, t.tsym, t.metadata); + } + + @Override + public Type visitWildcardType(WildcardType wt, S s) { + Type t = wt.type; + if (t != null) + t = visit(t, s); + if (t == wt.type) + return wt; + else + return new WildcardType(t, wt.kind, wt.tsym, wt.bound, wt.metadata); + } + + @Override + public Type visitArrayType(ArrayType t, S s) { + Type elemtype = t.elemtype; + Type elemtype1 = visit(elemtype, s); + if (elemtype1 == elemtype) return t; + else return new ArrayType(elemtype1, t.tsym, t.metadata); + } + + @Override + public Type visitMethodType(MethodType t, S s) { + List argtypes = t.argtypes; + Type restype = t.restype; + List thrown = t.thrown; + List argtypes1 = visit(argtypes, s); + Type restype1 = visit(restype, s); + List thrown1 = visit(thrown, s); + if (argtypes1 == argtypes && + restype1 == restype && + thrown1 == thrown) return t; + else return new MethodType(argtypes1, restype1, thrown1, t.tsym); + } + + @Override + public Type visitCapturedType(CapturedType t, S s) { + return visitTypeVar(t, s); + } + + @Override + public Type visitForAll(ForAll t, S s) { + return visit(t.qtype, s); } } /** map a type function over all immediate descendants of this type */ - public Type map(Mapping f) { - return this; + public Type map(TypeMapping mapping, Z arg) { + return mapping.visit(this, arg); } - /** map a type function over a list of types + /** map a type function over all immediate descendants of this type (no arg version) */ - public static List map(List ts, Mapping f) { - if (ts.nonEmpty()) { - List tail1 = map(ts.tail, f); - Type t = f.apply(ts.head); - if (tail1 != ts.tail || t != ts.head) - return tail1.prepend(t); - } - return ts; + public Type map(TypeMapping mapping) { + return mapping.visit(this, null); } /** Define a constant type, of the same kind as this type @@ -775,17 +825,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { return s.toString(); } - public Type map(Mapping f) { - //- System.err.println(" (" + this + ").map(" + f + ")");//DEBUG - Type t = type; - if (t != null) - t = f.apply(t); - if (t == type) - return this; - else - return new WildcardType(t, kind, tsym, bound, metadata); - } - @DefinedBy(Api.LANGUAGE_MODEL) public Type getExtendsBound() { if (kind == EXTENDS) @@ -1009,15 +1048,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { allparams().isEmpty(); } - public Type map(Mapping f) { - Type outer = getEnclosingType(); - Type outer1 = f.apply(outer); - List typarams = getTypeArguments(); - List typarams1 = map(typarams, f); - if (outer1 == outer && typarams1 == typarams) return this; - else return new ClassType(outer1, typarams1, tsym, metadata); - } - public boolean contains(Type elem) { return elem == this @@ -1248,12 +1278,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { }; } - public Type map(Mapping f) { - Type elemtype1 = f.apply(elemtype); - if (elemtype1 == elemtype) return this; - else return new ArrayType(elemtype1, tsym, metadata); - } - public boolean contains(Type elem) { return elem == this || elemtype.contains(elem); } @@ -1345,16 +1369,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { restype != null && restype.isErroneous(); } - public Type map(Mapping f) { - List argtypes1 = map(argtypes, f); - Type restype1 = f.apply(restype); - List thrown1 = map(thrown, f); - if (argtypes1 == argtypes && - restype1 == restype && - thrown1 == thrown) return this; - else return new MethodType(argtypes1, restype1, thrown1, tsym); - } - public boolean contains(Type elem) { return elem == this || contains(argtypes, elem) || restype.contains(elem) || contains(thrown, elem); } @@ -1647,10 +1661,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { return qtype.isErroneous(); } - public Type map(Mapping f) { - return f.apply(qtype); - } - public boolean contains(Type elem) { return qtype.contains(elem); } @@ -1820,7 +1830,7 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { } protected void addBound(InferenceBound ib, Type bound, Types types, boolean update) { - Type bound2 = toTypeVarMap.apply(bound).baseType(); + Type bound2 = bound.map(toTypeVarMap).baseType(); List prevBounds = bounds.get(ib); for (Type b : prevBounds) { //check for redundancy - use strict version of isSameType on tvars @@ -1831,15 +1841,10 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { notifyChange(EnumSet.of(ib)); } //where - Type.Mapping toTypeVarMap = new Mapping("toTypeVarMap") { + TypeMapping toTypeVarMap = new TypeMapping() { @Override - public Type apply(Type t) { - if (t.hasTag(UNDETVAR)) { - UndetVar uv = (UndetVar)t; - return uv.inst != null ? uv.inst : uv.qtype; - } else { - return t.map(this); - } + public Type visitUndetVar(UndetVar uv, Void _unused) { + return uv.inst != null ? uv.inst : uv.qtype; } }; @@ -2110,7 +2115,6 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { public Type getEnclosingType() { return this; } public Type getReturnType() { return this; } public Type asSub(Symbol sym) { return this; } - public Type map(Mapping f) { return this; } public boolean isGenType(Type t) { return true; } public boolean isErroneous() { return true; } diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java index b1386a94e0f..3964371b7dd 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java @@ -1766,10 +1766,11 @@ public class Types { // public List cvarLowerBounds(List ts) { - return map(ts, cvarLowerBoundMapping); + return ts.map(cvarLowerBoundMapping); } - private final Mapping cvarLowerBoundMapping = new Mapping("cvarLowerBound") { - public Type apply(Type t) { + private final TypeMapping cvarLowerBoundMapping = new TypeMapping() { + @Override + public Type visitCapturedType(CapturedType t, Void _unused) { return cvarLowerBound(t); } }; @@ -1879,9 +1880,15 @@ public class Types { /** * Mapping to take element type of an arraytype */ - private Mapping elemTypeFun = new Mapping ("elemTypeFun") { - public Type apply(Type t) { - return elemtype(skipTypeVars(t, false)); + private TypeMapping elemTypeFun = new TypeMapping() { + @Override + public Type visitArrayType(ArrayType t, Void _unused) { + return t.elemtype; + } + + @Override + public Type visitTypeVar(TypeVar t, Void _unused) { + return visit(skipTypeVars(t, false)); } }; @@ -2177,7 +2184,7 @@ public class Types { } } // where - private SimpleVisitor erasure = new SimpleVisitor() { + private TypeMapping erasure = new TypeMapping() { private Type combineMetadata(final Type ty, final TypeMetadata md) { if (!md.isEmpty()) { @@ -2202,8 +2209,8 @@ public class Types { if (t.isPrimitive()) return t; /*fast special case*/ else { - Type erased = t.map(recurse ? erasureRecFun : erasureFun); - return combineMetadata(erased, t.getMetadata()); + //other cases already handled + return combineMetadata(t, t.getMetadata()); } } @@ -2223,23 +2230,10 @@ public class Types { Type erased = erasure(t.bound, recurse); return combineMetadata(erased, t.getMetadata()); } - - @Override - public Type visitErrorType(ErrorType t, Boolean recurse) { - return t; - } }; - private Mapping erasureFun = new Mapping ("erasure") { - public Type apply(Type t) { return erasure(t); } - }; - - private Mapping erasureRecFun = new Mapping ("erasureRecursive") { - public Type apply(Type t) { return erasureRecursive(t); } - }; - public List erasure(List ts) { - return Type.map(ts, erasureFun); + return erasure.visit(ts, false); } public Type erasureRecursive(Type t) { @@ -2247,7 +2241,7 @@ public class Types { } public List erasureRecursive(List ts) { - return Type.map(ts, erasureRecFun); + return erasure.visit(ts, true); } // @@ -3177,15 +3171,18 @@ public class Types { * changing all recursive bounds from old to new list. */ public List newInstances(List tvars) { - List tvars1 = Type.map(tvars, newInstanceFun); + List tvars1 = tvars.map(newInstanceFun); for (List l = tvars1; l.nonEmpty(); l = l.tail) { TypeVar tv = (TypeVar) l.head; tv.bound = subst(tv.bound, tvars, tvars1); } return tvars1; } - private static final Mapping newInstanceFun = new Mapping("newInstanceFun") { - public Type apply(Type t) { return new TypeVar(t.tsym, t.getUpperBound(), t.getLowerBound(), t.getMetadata()); } + private static final TypeMapping newInstanceFun = new TypeMapping() { + @Override + public TypeVar visitTypeVar(TypeVar t, Void _unused) { + return new TypeVar(t.tsym, t.getUpperBound(), t.getLowerBound(), t.getMetadata()); + } }; // diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index ce01abdc960..c3b467de0e5 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -3734,7 +3734,7 @@ public class Attr extends JCTree.Visitor { DeferredAttr.DeferredTypeMap checkDeferredMap = deferredAttr.new DeferredTypeMap(DeferredAttr.AttrMode.CHECK, sym, env.info.pendingResolutionPhase); - argtypes = Type.map(argtypes, checkDeferredMap); + argtypes = argtypes.map(checkDeferredMap); if (noteWarner.hasNonSilentLint(LintCategory.UNCHECKED)) { chk.warnUnchecked(env.tree.pos(), @@ -3742,7 +3742,7 @@ public class Attr extends JCTree.Visitor { kindName(sym), sym.name, rs.methodArguments(sym.type.getParameterTypes()), - rs.methodArguments(Type.map(argtypes, checkDeferredMap)), + rs.methodArguments(argtypes.map(checkDeferredMap)), kindName(sym.location()), sym.location()); owntype = new MethodType(owntype.getParameterTypes(), @@ -3766,7 +3766,7 @@ public class Attr extends JCTree.Visitor { return new Pair<>(sym, diag); } }; - List argtypes2 = Type.map(argtypes, + List argtypes2 = argtypes.map( rs.new ResolveDeferredRecoveryMap(AttrMode.CHECK, sym, env.info.pendingResolutionPhase)); JCDiagnostic errDiag = errSym.getDiagnostic(JCDiagnostic.DiagnosticType.ERROR, env.tree, sym, site, sym.name, argtypes2, typeargtypes); diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/DeferredAttr.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/DeferredAttr.java index 84c59427cc2..8a7156155fd 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/DeferredAttr.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/DeferredAttr.java @@ -27,6 +27,7 @@ package com.sun.tools.javac.comp; import com.sun.source.tree.LambdaExpressionTree.BodyKind; import com.sun.tools.javac.code.*; +import com.sun.tools.javac.code.Type.TypeMapping; import com.sun.tools.javac.comp.Resolve.ResolveError; import com.sun.tools.javac.resources.CompilerProperties.Fragments; import com.sun.tools.javac.tree.*; @@ -44,7 +45,6 @@ import com.sun.tools.javac.util.Log.DeferredDiagnosticHandler; import java.util.ArrayList; import java.util.Collections; -import java.util.EnumMap; import java.util.EnumSet; import java.util.LinkedHashMap; import java.util.LinkedHashSet; @@ -300,13 +300,6 @@ public class DeferredAttr extends JCTree.Visitor { } }; - DeferredTypeCompleter dummyCompleter = new DeferredTypeCompleter() { - public Type complete(DeferredType dt, ResultInfo resultInfo, DeferredAttrContext deferredAttrContext) { - Assert.check(deferredAttrContext.mode == AttrMode.CHECK); - return dt.tree.type = Type.stuckType; - } - }; - /** * Policy for detecting stuck expressions. Different criteria might cause * an expression to be judged as stuck, depending on whether the check @@ -849,33 +842,24 @@ public class DeferredAttr extends JCTree.Visitor { /** an empty deferred attribution context - all methods throw exceptions */ final DeferredAttrContext emptyDeferredAttrContext; - /** The AttrMode to descriptive name mapping */ - private static final EnumMap deferredTypeMapDescriptions; - static { - deferredTypeMapDescriptions = new EnumMap<>(AttrMode.class); - deferredTypeMapDescriptions.put(AttrMode.CHECK, "deferredTypeMap[CHECK]"); - deferredTypeMapDescriptions.put(AttrMode.SPECULATIVE, "deferredTypeMap[SPECULATIVE]"); - } - /** * Map a list of types possibly containing one or more deferred types * into a list of ordinary types. Each deferred type D is mapped into a type T, * where T is computed by retrieving the type that has already been * computed for D during a previous deferred attribution round of the given kind. */ - class DeferredTypeMap extends Type.Mapping { + class DeferredTypeMap extends TypeMapping { DeferredAttrContext deferredAttrContext; protected DeferredTypeMap(AttrMode mode, Symbol msym, MethodResolutionPhase phase) { - super(deferredTypeMapDescriptions.get(mode)); this.deferredAttrContext = new DeferredAttrContext(mode, msym, phase, infer.emptyContext, emptyDeferredAttrContext, types.noWarnings); } @Override - public Type apply(Type t) { + public Type visitType(Type t, Void _unused) { if (!t.hasTag(DEFERRED)) { - return t.map(this); + return super.visitType(t, null); } else { DeferredType dt = (DeferredType)t; return typeOf(dt); @@ -928,7 +912,7 @@ public class DeferredAttr extends JCTree.Visitor { return chk.checkNonVoid(pos, super.check(pos, found)); } }); - return super.apply(dt); + return super.visit(dt); } } diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java index d2da81aefb5..40d94858b54 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java @@ -25,6 +25,7 @@ package com.sun.tools.javac.comp; +import com.sun.tools.javac.code.Type.TypeMapping; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.tree.JCTree.JCTypeCast; import com.sun.tools.javac.tree.TreeInfo; @@ -477,7 +478,7 @@ public class Infer { restype = syms.objectType; } - List paramtypes = Type.map(argtypes, new ImplicitArgType(spMethod, resolveContext.step)); + List paramtypes = argtypes.map(new ImplicitArgType(spMethod, resolveContext.step)); List exType = spMethod != null ? spMethod.getThrownTypes() : List.of(syms.throwableType); // make it throw all exceptions @@ -495,9 +496,16 @@ public class Infer { (rs.deferredAttr).super(AttrMode.SPECULATIVE, msym, phase); } - public Type apply(Type t) { - t = types.erasure(super.apply(t)); - if (t.hasTag(BOT)) + @Override + public Type visitClassType(ClassType t, Void aVoid) { + return types.erasure(t); + } + + @Override + public Type visitType(Type t, Void _unused) { + if (t.hasTag(DEFERRED)) { + return visit(super.visitType(t, null)); + } else if (t.hasTag(BOT)) // nulls type as the marker type Null (which has no instances) // infer as java.lang.Void for now t = types.boxedClass(syms.voidType).type; @@ -2046,23 +2054,19 @@ public class Infer { List freetypeListeners = List.nil(); public InferenceContext(List inferencevars) { - this.undetvars = Type.map(inferencevars, fromTypeVarFun); + this.undetvars = inferencevars.map(fromTypeVarFun); this.inferencevars = inferencevars; } //where - Mapping fromTypeVarFun = new Mapping("fromTypeVarFunWithBounds") { - // mapping that turns inference variables into undet vars - public Type apply(Type t) { - if (t.hasTag(TYPEVAR)) { - TypeVar tv = (TypeVar)t; - if (tv.isCaptured()) { - return new CapturedUndetVar((CapturedType)tv, types); - } else { - return new UndetVar(tv, types); - } - } else { - return t.map(this); - } + TypeMapping fromTypeVarFun = new TypeMapping() { + @Override + public Type visitTypeVar(TypeVar tv, Void aVoid) { + return new UndetVar(tv, types); + } + + @Override + public Type visitCapturedType(CapturedType t, Void aVoid) { + return new CapturedUndetVar(t, types); } }; diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java index ceefa7584ae..5e1fb9e8040 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java @@ -232,8 +232,7 @@ public class Resolve { } } String key = success ? "verbose.resolve.multi" : "verbose.resolve.multi.1"; - List argtypes2 = Type.map(argtypes, - deferredAttr.new RecoveryDeferredTypeMap(AttrMode.SPECULATIVE, bestSoFar, currentResolutionContext.step)); + List argtypes2 = argtypes.map(deferredAttr.new RecoveryDeferredTypeMap(AttrMode.SPECULATIVE, bestSoFar, currentResolutionContext.step)); JCDiagnostic main = diags.note(log.currentSource(), dpos, key, name, site.tsym, mostSpecificPos, currentResolutionContext.step, methodArguments(argtypes2), @@ -2259,7 +2258,7 @@ public class Resolve { (typeargtypes == null || !Type.isErroneous(typeargtypes)); } public List getArgumentTypes(ResolveError errSym, Symbol accessedSym, Name name, List argtypes) { - return Type.map(argtypes, new ResolveDeferredRecoveryMap(AttrMode.SPECULATIVE, accessedSym, currentResolutionContext.step)); + return argtypes.map(new ResolveDeferredRecoveryMap(AttrMode.SPECULATIVE, accessedSym, currentResolutionContext.step)); } }; diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java index ee2cb9acab4..cab4f18eb0b 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java @@ -33,6 +33,7 @@ import java.util.Iterator; import java.util.AbstractCollection; import java.util.ListIterator; import java.util.NoSuchElementException; +import java.util.function.Function; import java.util.stream.Collector; /** A class for generic linked lists. Links are supposed to be @@ -416,6 +417,14 @@ public class List extends AbstractCollection implements java.util.List return last; } + public List map(Function mapper) { + ListBuffer buf = new ListBuffer<>(); + for (A a : this) { + buf.add(mapper.apply(a)); + } + return buf.toList(); + } + @SuppressWarnings("unchecked") public static List convert(Class klass, List list) { if (list == null) From 24c51e1e1b95b8f94f917895b12e9f6a1083ce21 Mon Sep 17 00:00:00 2001 From: Maurizio Cimadamore Date: Thu, 19 Mar 2015 11:40:47 +0000 Subject: [PATCH 5/6] 8048838: type inference performance regression Reduce redundant bounds before attempting to do pairwise lub computation during bound incorporation. Reviewed-by: vromero --- .../com/sun/tools/javac/code/Types.java | 65 ++++++++++++++--- .../com/sun/tools/javac/comp/Infer.java | 3 +- .../generics/inference/8048838/T8048838.java | 72 +++++++++++++++++++ 3 files changed, 130 insertions(+), 10 deletions(-) create mode 100644 langtools/test/tools/javac/generics/inference/8048838/T8048838.java diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java index 3964371b7dd..70ce0435abb 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java @@ -32,6 +32,8 @@ import java.util.Locale; import java.util.Map; import java.util.Set; import java.util.WeakHashMap; +import java.util.function.BiPredicate; +import java.util.stream.Collector; import javax.tools.JavaFileObject; @@ -3405,42 +3407,87 @@ public class Types { return cl; } + /** + * Collect types into a new closure (using a @code{ClosureHolder}) + */ + public Collector> closureCollector(boolean minClosure, BiPredicate shouldSkip) { + return Collector.of(() -> new ClosureHolder(minClosure, shouldSkip), + ClosureHolder::add, + ClosureHolder::merge, + ClosureHolder::closure); + } + //where + class ClosureHolder { + List closure; + final boolean minClosure; + final BiPredicate shouldSkip; + + ClosureHolder(boolean minClosure, BiPredicate shouldSkip) { + this.closure = List.nil(); + this.minClosure = minClosure; + this.shouldSkip = shouldSkip; + } + + void add(Type type) { + closure = insert(closure, type, shouldSkip); + } + + ClosureHolder merge(ClosureHolder other) { + closure = union(closure, other.closure, shouldSkip); + return this; + } + + List closure() { + return minClosure ? closureMin(closure) : closure; + } + } + + BiPredicate basicClosureSkip = (t1, t2) -> t1.tsym == t2.tsym; + /** * Insert a type in a closure */ - public List insert(List cl, Type t) { + public List insert(List cl, Type t, BiPredicate shouldSkip) { if (cl.isEmpty()) { return cl.prepend(t); - } else if (t.tsym == cl.head.tsym) { + } else if (shouldSkip.test(t, cl.head)) { return cl; } else if (t.tsym.precedes(cl.head.tsym, this)) { return cl.prepend(t); } else { // t comes after head, or the two are unrelated - return insert(cl.tail, t).prepend(cl.head); + return insert(cl.tail, t, shouldSkip).prepend(cl.head); } } + public List insert(List cl, Type t) { + return insert(cl, t, basicClosureSkip); + } + /** * Form the union of two closures */ - public List union(List cl1, List cl2) { + public List union(List cl1, List cl2, BiPredicate shouldSkip) { if (cl1.isEmpty()) { return cl2; } else if (cl2.isEmpty()) { return cl1; - } else if (cl1.head.tsym == cl2.head.tsym) { - return union(cl1.tail, cl2.tail).prepend(cl1.head); + } else if (shouldSkip.test(cl1.head, cl2.head)) { + return union(cl1.tail, cl2.tail, shouldSkip).prepend(cl1.head); } else if (cl1.head.tsym.precedes(cl2.head.tsym, this)) { - return union(cl1.tail, cl2).prepend(cl1.head); + return union(cl1.tail, cl2, shouldSkip).prepend(cl1.head); } else if (cl2.head.tsym.precedes(cl1.head.tsym, this)) { - return union(cl1, cl2.tail).prepend(cl2.head); + return union(cl1, cl2.tail, shouldSkip).prepend(cl2.head); } else { // unrelated types - return union(cl1.tail, cl2).prepend(cl1.head); + return union(cl1.tail, cl2, shouldSkip).prepend(cl1.head); } } + public List union(List cl1, List cl2) { + return union(cl1, cl2, basicClosureSkip); + } + /** * Intersect two closures */ diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java index 40d94858b54..23c961ec638 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java @@ -873,7 +873,8 @@ public class Infer { @Override public void apply(UndetVar uv, InferenceContext inferenceContext, Warner warn) { Infer infer = inferenceContext.infer(); - List boundList = uv.getBounds(InferenceBound.UPPER); + List boundList = uv.getBounds(InferenceBound.UPPER).stream() + .collect(infer.types.closureCollector(true, infer.types::isSameType)); List boundListTail = boundList.tail; while (boundList.nonEmpty()) { List tmpTail = boundListTail; diff --git a/langtools/test/tools/javac/generics/inference/8048838/T8048838.java b/langtools/test/tools/javac/generics/inference/8048838/T8048838.java new file mode 100644 index 00000000000..162d94c5f97 --- /dev/null +++ b/langtools/test/tools/javac/generics/inference/8048838/T8048838.java @@ -0,0 +1,72 @@ +/* + * Copyright (c) 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 8048838 + * @summary type inference performance regression + * @compile T8048838.java + */ +class T8048838 { + + + T1 foo(T1 x1, T2 x2, T3 x3, T4 x4, T5 x5, T6 x6, T7 x7, T8 x8, T9 x9, T10 x10, T11 x11, T12 x12, + T13 x13, T14 x14, T15 x15, T16 x16, T17 x17, T18 x18, T19 x19, T20 x20, T21 x21, T22 x22, + T23 x23, T24 x24, T25 x25, T26 x26, T27 x27, T28 x28, T29 x29, T30 x30, T31 x31, T32 x32, + T33 x33, T34 x34, T35 x35, T36 x36, T37 x37, T38 x38, T39 x39, T40 x40, T41 x41, T42 x42, + T43 x43, T44 x44, T45 x45, T46 x46, T47 x47, T48 x48, T49 x49, T50 x50, T51 x51, T52 x52, + T53 x53, T54 x54, T55 x55, T56 x56, T57 x57, T58 x58, T59 x59, T60 x60, T61 x61, T62 x62, + T63 x63, T64 x64, T65 x65, T66 x66, T67 x67, T68 x68, T69 x69, T70 x70, T71 x71, T72 x72, + T73 x73, T74 x74, T75 x75, T76 x76, T77 x77, T78 x78, T79 x79, T80 x80, T81 x81, T82 x82, + T83 x83, T84 x84, T85 x85, T86 x86, T87 x87, T88 x88, T89 x89, T90 x90, T91 x91, T92 x92, + T93 x93, T94 x94, T95 x95, T96 x96, T97 x97, T98 x98, T99 x99, T100 x100) { return null; } + + Object test() { + return foo(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, + 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, + 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, + 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, + 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100); // type inference expected + } +} From c399e03fe203aa5b82b0dbff26acb17a8fe7ed04 Mon Sep 17 00:00:00 2001 From: Maurizio Cimadamore Date: Thu, 19 Mar 2015 16:23:21 +0000 Subject: [PATCH 6/6] 8075509: List.map should return itself if list is unchanged Fix List.map to match semantics of old Type.map Reviewed-by: jlahoda --- .../share/classes/com/sun/tools/javac/util/List.java | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java index cab4f18eb0b..53990dc74e7 100644 --- a/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java +++ b/langtools/src/jdk.compiler/share/classes/com/sun/tools/javac/util/List.java @@ -417,12 +417,15 @@ public class List extends AbstractCollection implements java.util.List return last; } + @SuppressWarnings("unchecked") public List map(Function mapper) { - ListBuffer buf = new ListBuffer<>(); - for (A a : this) { - buf.add(mapper.apply(a)); + if (nonEmpty()) { + List tail1 = tail.map(mapper); + Z head1 = mapper.apply(head); + if (tail1 != tail || head1 != head) + return tail1.prepend(head1); } - return buf.toList(); + return (List)this; } @SuppressWarnings("unchecked")