From 66c76689eb1c815cf8d95aa86607675da0dae6bf Mon Sep 17 00:00:00 2001 From: mcimadamore Date: Thu, 23 Oct 2008 17:59:16 +0100 Subject: [PATCH] 6732484: Bound error on wildcard code Summary: Check.checkExtends should cope with captured type-variables Reviewed-by: jjg --- .../com/sun/tools/javac/code/Type.java | 15 ---- .../com/sun/tools/javac/comp/Check.java | 90 ++++++++++--------- .../javac/generics/wildcards/T6732484.java | 37 ++++++++ 3 files changed, 84 insertions(+), 58 deletions(-) create mode 100644 test/tools/javac/generics/wildcards/T6732484.java diff --git a/src/share/classes/com/sun/tools/javac/code/Type.java b/src/share/classes/com/sun/tools/javac/code/Type.java index b2a3dae9..891a346f 100644 --- a/src/share/classes/com/sun/tools/javac/code/Type.java +++ b/src/share/classes/com/sun/tools/javac/code/Type.java @@ -360,17 +360,6 @@ public class Type implements PrimitiveType { public boolean isUnbound() { return false; } public Type withTypeVar(Type t) { return this; } - public static List removeBounds(List ts) { - ListBuffer result = new ListBuffer(); - for(;ts.nonEmpty(); ts = ts.tail) { - result.append(ts.head.removeBounds()); - } - return result.toList(); - } - public Type removeBounds() { - return this; - } - /** The underlying method type of this type. */ public MethodType asMethodType() { throw new AssertionError(); } @@ -489,10 +478,6 @@ public class Type implements PrimitiveType { return new WildcardType(t, kind, tsym, bound); } - public Type removeBounds() { - return isUnbound() ? this : type; - } - public Type getExtendsBound() { if (kind == EXTENDS) return type; diff --git a/src/share/classes/com/sun/tools/javac/comp/Check.java b/src/share/classes/com/sun/tools/javac/comp/Check.java index d901d867..a0d5cf32 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -424,43 +424,43 @@ public class Check { * @param bs The bound. */ private void checkExtends(DiagnosticPosition pos, Type a, TypeVar bs) { - if (a.tag == TYPEVAR && ((TypeVar)a).isCaptured()) { - CapturedType ct = (CapturedType)a; - boolean ok; - if (ct.bound.isErroneous()) {//capture doesn't exist - ok = false; - } - else { - switch (ct.wildcard.kind) { - case EXTENDS: - ok = types.isCastable(bs.getUpperBound(), - types.upperBound(a), - Warner.noWarnings); - break; - case SUPER: - ok = !types.notSoftSubtype(types.lowerBound(a), - bs.getUpperBound()); - break; - case UNBOUND: - ok = true; - break; - default: - throw new AssertionError("Invalid bound kind"); - } - } - if (!ok) - log.error(pos, "not.within.bounds", a); - } - else { - a = types.upperBound(a); - for (List l = types.getBounds(bs); l.nonEmpty(); l = l.tail) { - if (!types.isSubtype(a, l.head)) { - log.error(pos, "not.within.bounds", a); - return; - } + if (a.isUnbound()) { + return; + } else if (a.tag != WILDCARD) { + a = types.upperBound(a); + for (List l = types.getBounds(bs); l.nonEmpty(); l = l.tail) { + if (!types.isSubtype(a, l.head)) { + log.error(pos, "not.within.bounds", a); + return; + } + } + } else if (a.isExtendsBound()) { + if (!types.isCastable(bs.getUpperBound(), types.upperBound(a), Warner.noWarnings)) + log.error(pos, "not.within.bounds", a); + } else if (a.isSuperBound()) { + if (types.notSoftSubtype(types.lowerBound(a), bs.getUpperBound())) + log.error(pos, "not.within.bounds", a); + } + } + + /** Check that a type is within some bounds. + * + * Used in TypeApply to verify that, e.g., X in V is a valid + * type argument. + * @param pos Position to be used for error reporting. + * @param a The type that should be bounded by bs. + * @param bs The bound. + */ + private void checkCapture(JCTypeApply tree) { + List args = tree.getTypeArguments(); + for (Type arg : types.capture(tree.type).getTypeArguments()) { + if (arg.tag == TYPEVAR && arg.getUpperBound().isErroneous()) { + log.error(args.head.pos, "not.within.bounds", args.head.type); + break; } + args = args.tail; } - } + } /** Check that type is different from 'void'. * @param pos Position to be used for error reporting. @@ -803,7 +803,7 @@ public class Check { public void visitTypeApply(JCTypeApply tree) { if (tree.type.tag == CLASS) { List formals = tree.type.tsym.type.getTypeArguments(); - List actuals = types.capture(tree.type).getTypeArguments(); + List actuals = tree.type.getTypeArguments(); List args = tree.arguments; List forms = formals; ListBuffer tvars_buf = new ListBuffer(); @@ -826,25 +826,29 @@ public class Check { } args = tree.arguments; - List tvars = tvars_buf.toList(); - while (args.nonEmpty() && tvars.nonEmpty()) { + List tvars_cap = types.substBounds(formals, + formals, + types.capture(tree.type).getTypeArguments()); + while (args.nonEmpty() && tvars_cap.nonEmpty()) { // Let the actual arguments know their bound - args.head.type.withTypeVar(tvars.head); + args.head.type.withTypeVar((TypeVar)tvars_cap.head); args = args.tail; - tvars = tvars.tail; + tvars_cap = tvars_cap.tail; } args = tree.arguments; - tvars = tvars_buf.toList(); + List tvars = tvars_buf.toList(); + while (args.nonEmpty() && tvars.nonEmpty()) { checkExtends(args.head.pos(), - actuals.head, + args.head.type, tvars.head); args = args.tail; tvars = tvars.tail; - actuals = actuals.tail; } + checkCapture(tree); + // Check that this type is either fully parameterized, or // not parameterized at all. if (tree.type.getEnclosingType().isRaw()) diff --git a/test/tools/javac/generics/wildcards/T6732484.java b/test/tools/javac/generics/wildcards/T6732484.java new file mode 100644 index 00000000..42906e92 --- /dev/null +++ b/test/tools/javac/generics/wildcards/T6732484.java @@ -0,0 +1,37 @@ +/* + * Copyright 2008 Sun Microsystems, Inc. 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + */ + +/* + * @test + * @bug 6732484 + * @summary Bound error on wildcard code + * @author Maurizio Cimadamore + * @compile T6732484.java + */ + +class T6732484 { + class A> {} + class B extends A {} + + A f; +} \ No newline at end of file -- GitLab