From 49901e0cb3e19168f602882d38bbebbd8ab6b1cd Mon Sep 17 00:00:00 2001 From: mcimadamore Date: Tue, 25 Sep 2012 11:55:34 +0100 Subject: [PATCH] 7175433: Inference cleanup: add helper class to handle inference variables Summary: Add class to handle inference variables instantiation and associated info Reviewed-by: jjg, dlsmith --- .../com/sun/tools/javac/comp/Check.java | 23 +- .../com/sun/tools/javac/comp/Infer.java | 463 ++++++++++++------ .../com/sun/tools/javac/comp/Resolve.java | 67 ++- .../generics/inference/6638712/T6638712c.out | 2 +- .../tools/javac/varargs/6313164/T7175433.java | 59 +++ 5 files changed, 424 insertions(+), 190 deletions(-) create mode 100644 test/tools/javac/varargs/6313164/T7175433.java 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 d3a9e3bc..50885920 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -40,6 +40,8 @@ import com.sun.tools.javac.code.Lint; import com.sun.tools.javac.code.Lint.LintCategory; import com.sun.tools.javac.code.Type.*; import com.sun.tools.javac.code.Symbol.*; +import com.sun.tools.javac.comp.Infer.InferenceContext; +import com.sun.tools.javac.comp.Infer.InferenceContext.FreeTypeListener; import static com.sun.tools.javac.code.Flags.*; import static com.sun.tools.javac.code.Flags.ANNOTATION; @@ -429,6 +431,8 @@ public class Check { * Obtain a warner for this check context */ public Warner checkWarner(DiagnosticPosition pos, Type found, Type req); + + public Infer.InferenceContext inferenceContext(); } /** @@ -455,6 +459,10 @@ public class Check { public Warner checkWarner(DiagnosticPosition pos, Type found, Type req) { return enclosingContext.checkWarner(pos, found, req); } + + public Infer.InferenceContext inferenceContext() { + return enclosingContext.inferenceContext(); + } } /** @@ -471,6 +479,10 @@ public class Check { public Warner checkWarner(DiagnosticPosition pos, Type found, Type req) { return convertWarner(pos, found, req); } + + public InferenceContext inferenceContext() { + return infer.emptyContext; + } }; /** Check that a given type is assignable to a given proto-type. @@ -483,7 +495,16 @@ public class Check { return checkType(pos, found, req, basicHandler); } - Type checkType(final DiagnosticPosition pos, Type found, Type req, CheckContext checkContext) { + Type checkType(final DiagnosticPosition pos, final Type found, final Type req, final CheckContext checkContext) { + final Infer.InferenceContext inferenceContext = checkContext.inferenceContext(); + if (inferenceContext.free(req)) { + inferenceContext.addFreeTypeListener(List.of(req), new FreeTypeListener() { + @Override + public void typesInferred(InferenceContext inferenceContext) { + checkType(pos, found, inferenceContext.asInstType(req, types), checkContext); + } + }); + } if (req.tag == ERROR) return req; if (req.tag == NONE) diff --git a/src/share/classes/com/sun/tools/javac/comp/Infer.java b/src/share/classes/com/sun/tools/javac/comp/Infer.java index 659eeda2..b1a6ab8d 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Infer.java +++ b/src/share/classes/com/sun/tools/javac/comp/Infer.java @@ -25,18 +25,21 @@ package com.sun.tools.javac.comp; +import com.sun.tools.javac.code.*; +import com.sun.tools.javac.code.Symbol.*; +import com.sun.tools.javac.code.Type.*; +import com.sun.tools.javac.comp.Resolve.InapplicableMethodException; +import com.sun.tools.javac.comp.Resolve.VerboseResolutionMode; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.tree.JCTree.JCTypeCast; import com.sun.tools.javac.tree.TreeInfo; import com.sun.tools.javac.util.*; import com.sun.tools.javac.util.List; -import com.sun.tools.javac.code.*; -import com.sun.tools.javac.code.Type.*; -import com.sun.tools.javac.code.Symbol.*; -import com.sun.tools.javac.comp.Resolve.InapplicableMethodException; -import com.sun.tools.javac.comp.Resolve.VerboseResolutionMode; import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition; +import java.util.HashMap; +import java.util.Map; + import static com.sun.tools.javac.code.TypeTags.*; /** Helper class for type parameter inference, used by the attribution phase. @@ -76,40 +79,38 @@ public class Infer { chk = Check.instance(context); diags = JCDiagnostic.Factory.instance(context); inferenceException = new InferenceException(diags); - } + /** + * This exception class is design to store a list of diagnostics corresponding + * to inference errors that can arise during a method applicability check. + */ public static class InferenceException extends InapplicableMethodException { private static final long serialVersionUID = 0; + List messages = List.nil(); + InferenceException(JCDiagnostic.Factory diags) { super(diags); } - } - private final InferenceException inferenceException; + @Override + InapplicableMethodException setMessage(JCDiagnostic diag) { + messages = messages.append(diag); + return this; + } -/*************************************************************************** - * Auxiliary type values and classes - ***************************************************************************/ + @Override + public JCDiagnostic getDiagnostic() { + return messages.head; + } - /** A mapping that turns type variables into undetermined type variables. - */ - List makeUndetvars(List tvars) { - List undetvars = Type.map(tvars, fromTypeVarFun); - for (Type t : undetvars) { - UndetVar uv = (UndetVar)t; - uv.hibounds = types.getBounds((TypeVar)uv.qtype); + void clear() { + messages = List.nil(); } - return undetvars; } - //where - Mapping fromTypeVarFun = new Mapping("fromTypeVarFun") { - public Type apply(Type t) { - if (t.tag == TYPEVAR) return new UndetVar(t); - else return t.map(this); - } - }; + + private final InferenceException inferenceException; /*************************************************************************** * Mini/Maximization of UndetVars @@ -118,8 +119,8 @@ public class Infer { /** Instantiate undetermined type variable to its minimal upper bound. * Throw a NoInstanceException if this not possible. */ - void maximizeInst(UndetVar that, Warner warn) throws InferenceException { - List hibounds = Type.filter(that.hibounds, errorFilter); + void maximizeInst(UndetVar that, Warner warn) throws InferenceException { + List hibounds = Type.filter(that.hibounds, boundFilter); if (that.eq.isEmpty()) { if (hibounds.isEmpty()) that.inst = syms.objectType; @@ -137,10 +138,10 @@ public class Infer { that.qtype, hibounds); } - private Filter errorFilter = new Filter() { + private Filter boundFilter = new Filter() { @Override public boolean accepts(Type t) { - return !t.isErroneous(); + return !t.isErroneous() && t.tag != BOT; } }; @@ -148,11 +149,12 @@ public class Infer { * Throw a NoInstanceException if this not possible. */ void minimizeInst(UndetVar that, Warner warn) throws InferenceException { - List lobounds = Type.filter(that.lobounds, errorFilter); + List lobounds = Type.filter(that.lobounds, boundFilter); if (that.eq.isEmpty()) { - if (lobounds.isEmpty()) - that.inst = syms.botType; - else if (lobounds.tail.isEmpty()) + if (lobounds.isEmpty()) { + //do nothing - the inference variable is under-constrained + return; + } else if (lobounds.tail.isEmpty()) that.inst = lobounds.head.isPrimitive() ? syms.errType : lobounds.head; else { that.inst = types.lub(lobounds); @@ -166,90 +168,70 @@ public class Infer { } } - Type asUndetType(Type t, List undetvars) { - return types.subst(t, inferenceVars(undetvars), undetvars); - } - - List inferenceVars(List undetvars) { - ListBuffer tvars = ListBuffer.lb(); - for (Type uv : undetvars) { - tvars.append(((UndetVar)uv).qtype); - } - return tvars.toList(); - } - /*************************************************************************** * Exported Methods ***************************************************************************/ - /** Try to instantiate expression type `that' to given type `to'. - * If a maximal instantiation exists which makes this type - * a subtype of type `to', return the instantiated type. - * If no instantiation exists, or if several incomparable - * best instantiations exist throw a NoInstanceException. + /** + * Instantiate uninferred inference variables (JLS 15.12.2.8). First + * if the method return type is non-void, we derive constraints from the + * expected type - then we use declared bound well-formedness to derive additional + * constraints. If no instantiation exists, or if several incomparable + * best instantiations exist throw a NoInstanceException. */ - public List instantiateUninferred(DiagnosticPosition pos, - List undetvars, - List tvars, - MethodType mtype, - Attr.ResultInfo resultInfo, - Warner warn) throws InferenceException { + public void instantiateUninferred(DiagnosticPosition pos, + InferenceContext inferenceContext, + MethodType mtype, + Attr.ResultInfo resultInfo, + Warner warn) throws InferenceException { Type to = resultInfo.pt; if (to.tag == NONE) { to = mtype.getReturnType().tag <= VOID ? mtype.getReturnType() : syms.objectType; } - Type qtype1 = types.subst(mtype.getReturnType(), tvars, undetvars); + Type qtype1 = inferenceContext.asFree(mtype.getReturnType(), types); if (!types.isSubtype(qtype1, qtype1.tag == UNDETVAR ? types.boxedTypeOrType(to) : to)) { throw inferenceException - .setMessage("infer.no.conforming.instance.exists", - tvars, mtype.getReturnType(), to); + .setMessage("infer.no.conforming.instance.exists", + inferenceContext.restvars(), mtype.getReturnType(), to); } - List insttypes; while (true) { boolean stuck = true; - insttypes = List.nil(); - for (Type t : undetvars) { + for (Type t : inferenceContext.undetvars) { UndetVar uv = (UndetVar)t; - if (uv.inst == null && (uv.eq.nonEmpty() || !Type.containsAny(uv.hibounds, tvars))) { + if (uv.inst == null && (uv.eq.nonEmpty() || !inferenceContext.free(uv.hibounds))) { maximizeInst((UndetVar)t, warn); stuck = false; } - insttypes = insttypes.append(uv.inst == null ? uv.qtype : uv.inst); } - if (!Type.containsAny(insttypes, tvars)) { + if (inferenceContext.restvars().isEmpty()) { //all variables have been instantiated - exit break; } else if (stuck) { //some variables could not be instantiated because of cycles in //upper bounds - provide a (possibly recursive) default instantiation - insttypes = types.subst(insttypes, - tvars, - instantiateAsUninferredVars(undetvars, tvars)); + instantiateAsUninferredVars(inferenceContext); break; } else { //some variables have been instantiated - replace newly instantiated //variables in remaining upper bounds and continue - for (Type t : undetvars) { + for (Type t : inferenceContext.undetvars) { UndetVar uv = (UndetVar)t; - uv.hibounds = types.subst(uv.hibounds, tvars, insttypes); + uv.hibounds = inferenceContext.asInstTypes(uv.hibounds, types); } } } - return insttypes; } /** * Infer cyclic inference variables as described in 15.12.2.8. */ - private List instantiateAsUninferredVars(List undetvars, List tvars) { - Assert.check(undetvars.length() == tvars.length()); - ListBuffer insttypes = ListBuffer.lb(); + private void instantiateAsUninferredVars(InferenceContext inferenceContext) { ListBuffer todo = ListBuffer.lb(); //step 1 - create fresh tvars - for (Type t : undetvars) { + for (Type t : inferenceContext.undetvars) { UndetVar uv = (UndetVar)t; if (uv.inst == null) { TypeSymbol fresh_tvar = new TypeSymbol(Flags.SYNTHETIC, uv.qtype.tsym.name, null, uv.qtype.tsym.owner); @@ -257,25 +239,23 @@ public class Infer { todo.append(uv); uv.inst = fresh_tvar.type; } - insttypes.append(uv.inst); } //step 2 - replace fresh tvars in their bounds - List formals = tvars; + List formals = inferenceContext.inferenceVars(); for (Type t : todo) { UndetVar uv = (UndetVar)t; TypeVar ct = (TypeVar)uv.inst; - ct.bound = types.glb(types.subst(types.getBounds(ct), tvars, insttypes.toList())); + ct.bound = types.glb(inferenceContext.asInstTypes(types.getBounds(ct), types)); if (ct.bound.isErroneous()) { //report inference error if glb fails reportBoundError(uv, BoundErrorKind.BAD_UPPER); } formals = formals.tail; } - return insttypes.toList(); } - /** Instantiate method type `mt' by finding instantiations of - * `tvars' so that method can be applied to `argtypes'. + /** Instantiate a generic method type by finding instantiations for all its + * inference variables so that it can be applied to a given argument type list. */ public Type instantiateMethod(Env env, List tvars, @@ -287,83 +267,61 @@ public class Infer { boolean useVarargs, Warner warn) throws InferenceException { //-System.err.println("instantiateMethod(" + tvars + ", " + mt + ", " + argtypes + ")"); //DEBUG - List undetvars = makeUndetvars(tvars); + final InferenceContext inferenceContext = new InferenceContext(tvars, types); + inferenceException.clear(); - List capturedArgs = - rs.checkRawArgumentsAcceptable(env, undetvars, argtypes, mt.getParameterTypes(), - allowBoxing, useVarargs, warn, new InferenceCheckHandler(undetvars)); + try { + rs.checkRawArgumentsAcceptable(env, inferenceContext, argtypes, mt.getParameterTypes(), + allowBoxing, useVarargs, warn, new InferenceCheckHandler(inferenceContext)); - // minimize as yet undetermined type variables - for (Type t : undetvars) - minimizeInst((UndetVar) t, warn); - - /** Type variables instantiated to bottom */ - ListBuffer restvars = new ListBuffer(); - - /** Undet vars instantiated to bottom */ - final ListBuffer restundet = new ListBuffer(); - - /** Instantiated types or TypeVars if under-constrained */ - ListBuffer insttypes = new ListBuffer(); + // minimize as yet undetermined type variables + for (Type t : inferenceContext.undetvars) { + minimizeInst((UndetVar)t, warn); + } - /** Instantiated types or UndetVars if under-constrained */ - ListBuffer undettypes = new ListBuffer(); + checkWithinBounds(inferenceContext, warn); - for (Type t : undetvars) { - UndetVar uv = (UndetVar)t; - if (uv.inst.tag == BOT) { - restvars.append(uv.qtype); - restundet.append(uv); - insttypes.append(uv.qtype); - undettypes.append(uv); - uv.inst = null; - } else { - insttypes.append(uv.inst); - undettypes.append(uv.inst); - } - } - checkWithinBounds(tvars, undetvars, insttypes.toList(), warn); + mt = (MethodType)inferenceContext.asInstType(mt, types); - mt = (MethodType)types.subst(mt, tvars, insttypes.toList()); + List restvars = inferenceContext.restvars(); - if (!restvars.isEmpty() && resultInfo != null) { - List restInferred = - instantiateUninferred(env.tree.pos(), restundet.toList(), restvars.toList(), mt, resultInfo, warn); - checkWithinBounds(tvars, undetvars, - types.subst(insttypes.toList(), restvars.toList(), restInferred), warn); - mt = (MethodType)types.subst(mt, restvars.toList(), restInferred); - if (rs.verboseResolutionMode.contains(VerboseResolutionMode.DEFERRED_INST)) { - log.note(env.tree.pos, "deferred.method.inst", msym, mt, resultInfo.pt); + if (!restvars.isEmpty()) { + if (resultInfo != null) { + instantiateUninferred(env.tree.pos(), inferenceContext, mt, resultInfo, warn); + checkWithinBounds(inferenceContext, warn); + mt = (MethodType)inferenceContext.asInstType(mt, types); + if (rs.verboseResolutionMode.contains(VerboseResolutionMode.DEFERRED_INST)) { + log.note(env.tree.pos, "deferred.method.inst", msym, mt, resultInfo.pt); + } + } } - } - if (restvars.isEmpty() || resultInfo != null) { - // check that actuals conform to inferred formals - checkArgumentsAcceptable(env, capturedArgs, mt.getParameterTypes(), allowBoxing, useVarargs, warn); + // return instantiated version of method type + return mt; + } finally { + inferenceContext.notifyChange(types); } - // return instantiated version of method type - return mt; } //where /** inference check handler **/ class InferenceCheckHandler implements Resolve.MethodCheckHandler { - List undetvars; + InferenceContext inferenceContext; - public InferenceCheckHandler(List undetvars) { - this.undetvars = undetvars; + public InferenceCheckHandler(InferenceContext inferenceContext) { + this.inferenceContext = inferenceContext; } public InapplicableMethodException arityMismatch() { - return inferenceException.setMessage("infer.arg.length.mismatch", inferenceVars(undetvars)); + return inferenceException.setMessage("infer.arg.length.mismatch", inferenceContext.inferenceVars()); } public InapplicableMethodException argumentMismatch(boolean varargs, JCDiagnostic details) { String key = varargs ? "infer.varargs.argument.mismatch" : "infer.no.conforming.assignment.exists"; return inferenceException.setMessage(key, - inferenceVars(undetvars), details); + inferenceContext.inferenceVars(), details); } public InapplicableMethodException inaccessibleVarargs(Symbol location, Type expected) { return inferenceException.setMessage("inaccessible.varargs.type", @@ -371,51 +329,37 @@ public class Infer { } } - private void checkArgumentsAcceptable(Env env, List actuals, List formals, - boolean allowBoxing, boolean useVarargs, Warner warn) { - try { - rs.checkRawArgumentsAcceptable(env, actuals, formals, - allowBoxing, useVarargs, warn); - } - catch (InapplicableMethodException ex) { - // inferred method is not applicable - throw inferenceException.setMessage(ex.getDiagnostic()); - } - } - /** check that type parameters are within their bounds. */ - void checkWithinBounds(List tvars, - List undetvars, - List arguments, + void checkWithinBounds(InferenceContext inferenceContext, Warner warn) throws InferenceException { - List args = arguments; - for (Type t : undetvars) { + List tvars = inferenceContext.inferenceVars(); + for (Type t : inferenceContext.undetvars) { UndetVar uv = (UndetVar)t; - uv.hibounds = types.subst(uv.hibounds, tvars, arguments); - uv.lobounds = types.subst(uv.lobounds, tvars, arguments); - uv.eq = types.subst(uv.eq, tvars, arguments); - checkCompatibleUpperBounds(uv, tvars); - if (args.head.tag != TYPEVAR || !args.head.containsAny(tvars)) { - Type inst = args.head; + uv.hibounds = inferenceContext.asInstTypes(uv.hibounds, types); + uv.lobounds = inferenceContext.asInstTypes(uv.lobounds, types); + uv.eq = inferenceContext.asInstTypes(uv.eq, types); + checkCompatibleUpperBounds(uv, inferenceContext.inferenceVars()); + if (!inferenceContext.restvars().contains(tvars.head)) { + Type inst = inferenceContext.asInstType(t, types); for (Type u : uv.hibounds) { - if (!types.isSubtypeUnchecked(inst, types.subst(u, tvars, undetvars), warn)) { + if (!types.isSubtypeUnchecked(inst, inferenceContext.asFree(u, types), warn)) { reportBoundError(uv, BoundErrorKind.UPPER); } } for (Type l : uv.lobounds) { - if (!types.isSubtypeUnchecked(types.subst(l, tvars, undetvars), inst, warn)) { + if (!types.isSubtypeUnchecked(inferenceContext.asFree(l, types), inst, warn)) { reportBoundError(uv, BoundErrorKind.LOWER); } } for (Type e : uv.eq) { - if (!types.isSameType(inst, types.subst(e, tvars, undetvars))) { + if (!types.isSameType(inst, inferenceContext.asFree(e, types))) { reportBoundError(uv, BoundErrorKind.EQ); } } } - args = args.tail; + tvars = tvars.tail; } } @@ -423,7 +367,7 @@ public class Infer { // VGJ: sort of inlined maximizeInst() below. Adding // bounds can cause lobounds that are above hibounds. ListBuffer hiboundsNoVars = ListBuffer.lb(); - for (Type t : Type.filter(uv.hibounds, errorFilter)) { + for (Type t : Type.filter(uv.hibounds, boundFilter)) { if (!t.containsAny(tvars)) { hiboundsNoVars.append(t); } @@ -531,4 +475,199 @@ public class Infer { return t; } }; + + /** + * Mapping that turns inference variables into undet vars + * (used by inference context) + */ + static Mapping fromTypeVarFun = new Mapping("fromTypeVarFun") { + public Type apply(Type t) { + if (t.tag == TYPEVAR) return new UndetVar(t); + else return t.map(this); + } + }; + + /** + * An inference context keeps track of the set of variables that are free + * in the current context. It provides utility methods for opening/closing + * types to their corresponding free/closed forms. It also provide hooks for + * attaching deferred post-inference action (see PendingCheck). Finally, + * it can be used as an entry point for performing upper/lower bound inference + * (see InferenceKind). + */ + static class InferenceContext { + + /** + * Single-method-interface for defining inference callbacks. Certain actions + * (i.e. subtyping checks) might need to be redone after all inference variables + * have been fixed. + */ + interface FreeTypeListener { + void typesInferred(InferenceContext inferenceContext); + } + + /** list of inference vars as undet vars */ + List undetvars; + + /** list of inference vars in this context */ + List inferencevars; + + java.util.Map> freeTypeListeners = + new java.util.HashMap>(); + + List freetypeListeners = List.nil(); + + public InferenceContext(List inferencevars, Types types) { + this.undetvars = Type.map(inferencevars, fromTypeVarFun); + this.inferencevars = inferencevars; + for (Type t : this.undetvars) { + UndetVar uv = (UndetVar)t; + uv.hibounds = types.getBounds((TypeVar)uv.qtype); + } + } + + /** + * returns the list of free variables (as type-variables) in this + * inference context + */ + List inferenceVars() { + return inferencevars; + } + + /** + * returns the list of uninstantiated variables (as type-variables) in this + * inference context (usually called after instantiate()) + */ + List restvars() { + List undetvars = this.undetvars; + ListBuffer restvars = ListBuffer.lb(); + for (Type t : instTypes()) { + UndetVar uv = (UndetVar)undetvars.head; + if (uv.qtype == t) { + restvars.append(t); + } + undetvars = undetvars.tail; + } + return restvars.toList(); + } + + /** + * is this type free? + */ + final boolean free(Type t) { + return t.containsAny(inferencevars); + } + + final boolean free(List ts) { + for (Type t : ts) { + if (free(t)) return true; + } + return false; + } + + /** + * Returns a list of free variables in a given type + */ + final List freeVarsIn(Type t) { + ListBuffer buf = ListBuffer.lb(); + for (Type iv : inferenceVars()) { + if (t.contains(iv)) { + buf.add(iv); + } + } + return buf.toList(); + } + + final List freeVarsIn(List ts) { + ListBuffer buf = ListBuffer.lb(); + for (Type t : ts) { + buf.appendList(freeVarsIn(t)); + } + ListBuffer buf2 = ListBuffer.lb(); + for (Type t : buf) { + if (!buf2.contains(t)) { + buf2.add(t); + } + } + return buf2.toList(); + } + + /** + * Replace all free variables in a given type with corresponding + * undet vars (used ahead of subtyping/compatibility checks to allow propagation + * of inference constraints). + */ + final Type asFree(Type t, Types types) { + return types.subst(t, inferencevars, undetvars); + } + + final List asFree(List ts, Types types) { + ListBuffer buf = ListBuffer.lb(); + for (Type t : ts) { + buf.append(asFree(t, types)); + } + return buf.toList(); + } + + List instTypes() { + ListBuffer buf = ListBuffer.lb(); + for (Type t : undetvars) { + UndetVar uv = (UndetVar)t; + buf.append(uv.inst != null ? uv.inst : uv.qtype); + } + return buf.toList(); + } + + /** + * Replace all free variables in a given type with corresponding + * instantiated types - if one or more free variable has not been + * fully instantiated, it will still be available in the resulting type. + */ + Type asInstType(Type t, Types types) { + return types.subst(t, inferencevars, instTypes()); + } + + List asInstTypes(List ts, Types types) { + ListBuffer buf = ListBuffer.lb(); + for (Type t : ts) { + buf.append(asInstType(t, types)); + } + return buf.toList(); + } + + /** + * Add custom hook for performing post-inference action + */ + void addFreeTypeListener(List types, FreeTypeListener ftl) { + freeTypeListeners.put(ftl, freeVarsIn(types)); + } + + /** + * Mark the inference context as complete and trigger evaluation + * of all deferred checks. + */ + void notifyChange(Types types) { + InferenceException thrownEx = null; + for (Map.Entry> entry : + new HashMap>(freeTypeListeners).entrySet()) { + if (!Type.containsAny(entry.getValue(), restvars())) { + try { + entry.getKey().typesInferred(this); + freeTypeListeners.remove(entry.getKey()); + } catch (InferenceException ex) { + if (thrownEx == null) { + thrownEx = ex; + } + } + } + } + //inference exception multiplexing - present any inference exception + //thrown when processing listeners as a single one + if (thrownEx != null) { + throw thrownEx; + } + } } + + final InferenceContext emptyContext = new InferenceContext(List.nil(), types); +} diff --git a/src/share/classes/com/sun/tools/javac/comp/Resolve.java b/src/share/classes/com/sun/tools/javac/comp/Resolve.java index 63941f15..eb38ce66 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Resolve.java +++ b/src/share/classes/com/sun/tools/javac/comp/Resolve.java @@ -31,6 +31,8 @@ import com.sun.tools.javac.code.Type.*; import com.sun.tools.javac.code.Symbol.*; import com.sun.tools.javac.comp.Attr.ResultInfo; import com.sun.tools.javac.comp.Check.CheckContext; +import com.sun.tools.javac.comp.Infer.InferenceContext; +import com.sun.tools.javac.comp.Infer.InferenceContext.FreeTypeListener; import com.sun.tools.javac.comp.Resolve.MethodResolutionContext.Candidate; import com.sun.tools.javac.jvm.*; import com.sun.tools.javac.tree.*; @@ -586,7 +588,7 @@ public class Resolve { boolean allowBoxing, boolean useVarargs, Warner warn) { - checkRawArgumentsAcceptable(env, List.nil(), argtypes, formals, + checkRawArgumentsAcceptable(env, infer.emptyContext, argtypes, formals, allowBoxing, useVarargs, warn, resolveHandler); } @@ -606,8 +608,8 @@ public class Resolve { * * A method check handler (see above) is used in order to report errors. */ - List checkRawArgumentsAcceptable(Env env, - List undetvars, + void checkRawArgumentsAcceptable(final Env env, + final Infer.InferenceContext inferenceContext, List argtypes, List formals, boolean allowBoxing, @@ -623,7 +625,7 @@ public class Resolve { } while (argtypes.nonEmpty() && formals.head != varargsFormal) { - ResultInfo resultInfo = methodCheckResult(formals.head, allowBoxing, false, undetvars, handler, warn); + ResultInfo resultInfo = methodCheckResult(formals.head, allowBoxing, false, inferenceContext, handler, warn); checkedArgs.append(resultInfo.check(env.tree.pos(), argtypes.head)); argtypes = argtypes.tail; formals = formals.tail; @@ -638,17 +640,29 @@ public class Resolve { //the last argument of a varargs is _not_ an array type (see JLS 15.12.2.5) Type elt = types.elemtype(varargsFormal); while (argtypes.nonEmpty()) { - ResultInfo resultInfo = methodCheckResult(elt, allowBoxing, true, undetvars, handler, warn); + ResultInfo resultInfo = methodCheckResult(elt, allowBoxing, true, inferenceContext, handler, warn); checkedArgs.append(resultInfo.check(env.tree.pos(), argtypes.head)); argtypes = argtypes.tail; } //check varargs element type accessibility - if (undetvars.isEmpty() && !isAccessible(env, elt)) { + varargsAccessible(env, elt, handler, inferenceContext); + } + } + + void varargsAccessible(final Env env, final Type t, final Resolve.MethodCheckHandler handler, final InferenceContext inferenceContext) { + if (inferenceContext.free(t)) { + inferenceContext.addFreeTypeListener(List.of(t), new FreeTypeListener() { + @Override + public void typesInferred(InferenceContext inferenceContext) { + varargsAccessible(env, inferenceContext.asInstType(t, types), handler, inferenceContext); + } + }); + } else { + if (!isAccessible(env, t)) { Symbol location = env.enclClass.sym; - throw handler.inaccessibleVarargs(location, elt); + throw handler.inaccessibleVarargs(location, t); } } - return checkedArgs.toList(); } /** @@ -659,13 +673,13 @@ public class Resolve { MethodCheckHandler handler; boolean useVarargs; - List undetvars; + Infer.InferenceContext inferenceContext; Warner rsWarner; - public MethodCheckContext(MethodCheckHandler handler, boolean useVarargs, List undetvars, Warner rsWarner) { + public MethodCheckContext(MethodCheckHandler handler, boolean useVarargs, Infer.InferenceContext inferenceContext, Warner rsWarner) { this.handler = handler; this.useVarargs = useVarargs; - this.undetvars = undetvars; + this.inferenceContext = inferenceContext; this.rsWarner = rsWarner; } @@ -676,6 +690,10 @@ public class Resolve { public Warner checkWarner(DiagnosticPosition pos, Type found, Type req) { return rsWarner; } + + public InferenceContext inferenceContext() { + return inferenceContext; + } } /** @@ -684,12 +702,12 @@ public class Resolve { */ class StrictMethodContext extends MethodCheckContext { - public StrictMethodContext(MethodCheckHandler handler, boolean useVarargs, List undetvars, Warner rsWarner) { - super(handler, useVarargs, undetvars, rsWarner); + public StrictMethodContext(MethodCheckHandler handler, boolean useVarargs, Infer.InferenceContext inferenceContext, Warner rsWarner) { + super(handler, useVarargs, inferenceContext, rsWarner); } public boolean compatible(Type found, Type req, Warner warn) { - return types.isSubtypeUnchecked(found, infer.asUndetType(req, undetvars), warn); + return types.isSubtypeUnchecked(found, inferenceContext.asFree(req, types), warn); } } @@ -699,12 +717,12 @@ public class Resolve { */ class LooseMethodContext extends MethodCheckContext { - public LooseMethodContext(MethodCheckHandler handler, boolean useVarargs, List undetvars, Warner rsWarner) { - super(handler, useVarargs, undetvars, rsWarner); + public LooseMethodContext(MethodCheckHandler handler, boolean useVarargs, Infer.InferenceContext inferenceContext, Warner rsWarner) { + super(handler, useVarargs, inferenceContext, rsWarner); } public boolean compatible(Type found, Type req, Warner warn) { - return types.isConvertible(found, infer.asUndetType(req, undetvars), warn); + return types.isConvertible(found, inferenceContext.asFree(req, types), warn); } } @@ -712,10 +730,10 @@ public class Resolve { * Create a method check context to be used during method applicability check */ ResultInfo methodCheckResult(Type to, boolean allowBoxing, boolean useVarargs, - List undetvars, MethodCheckHandler methodHandler, Warner rsWarner) { + Infer.InferenceContext inferenceContext, MethodCheckHandler methodHandler, Warner rsWarner) { MethodCheckContext checkContext = allowBoxing ? - new LooseMethodContext(methodHandler, useVarargs, undetvars, rsWarner) : - new StrictMethodContext(methodHandler, useVarargs, undetvars, rsWarner); + new LooseMethodContext(methodHandler, useVarargs, inferenceContext, rsWarner) : + new StrictMethodContext(methodHandler, useVarargs, inferenceContext, rsWarner); return attr.new ResultInfo(VAL, to, checkContext) { @Override protected Type check(DiagnosticPosition pos, Type found) { @@ -735,16 +753,13 @@ public class Resolve { this.diags = diags; } InapplicableMethodException setMessage() { - this.diagnostic = null; - return this; + return setMessage((JCDiagnostic)null); } InapplicableMethodException setMessage(String key) { - this.diagnostic = key != null ? diags.fragment(key) : null; - return this; + return setMessage(key != null ? diags.fragment(key) : null); } InapplicableMethodException setMessage(String key, Object... args) { - this.diagnostic = key != null ? diags.fragment(key, args) : null; - return this; + return setMessage(key != null ? diags.fragment(key, args) : null); } InapplicableMethodException setMessage(JCDiagnostic diag) { this.diagnostic = diag; diff --git a/test/tools/javac/generics/inference/6638712/T6638712c.out b/test/tools/javac/generics/inference/6638712/T6638712c.out index 9ea180cb..66286e3f 100644 --- a/test/tools/javac/generics/inference/6638712/T6638712c.out +++ b/test/tools/javac/generics/inference/6638712/T6638712c.out @@ -1,2 +1,2 @@ -T6638712c.java:16:9: compiler.err.cant.apply.symbol.1: kindname.method, sort, T[],java.util.Comparator, java.lang.Enum[],java.util.Comparator>, kindname.class, T6638712c, (compiler.misc.no.conforming.assignment.exists: (compiler.misc.inconvertible.types: java.util.Comparator>, java.util.Comparator)) +T6638712c.java:16:9: compiler.err.cant.apply.symbol.1: kindname.method, sort, T[],java.util.Comparator, java.lang.Enum[],java.util.Comparator>, kindname.class, T6638712c, (compiler.misc.infer.no.conforming.assignment.exists: T, (compiler.misc.inconvertible.types: java.util.Comparator>, java.util.Comparator)) 1 error diff --git a/test/tools/javac/varargs/6313164/T7175433.java b/test/tools/javac/varargs/6313164/T7175433.java new file mode 100644 index 00000000..b4627eb7 --- /dev/null +++ b/test/tools/javac/varargs/6313164/T7175433.java @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2012, 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 7175433 6313164 + * @summary Inference cleanup: add helper class to handle inference variables + * + */ + +import java.util.List; + +class Bar { + + private class Foo { } + + List m(Object... o) { T7175433.assertTrue(true); return null; } + List m(Foo... o) { T7175433.assertTrue(false); return null; } + + Foo getFoo() { return null; } +} + +public class T7175433 { + + static int assertionCount; + + static void assertTrue(boolean b) { + assertionCount++; + if (!b) { + throw new AssertionError(); + } + } + + public static void main(String[] args) { + Bar b = new Bar(); + b.m(b.getFoo()); + assertTrue(assertionCount == 1); + } +} -- GitLab