8015505: Spurious inference error when return type of generic method requires unchecked conversion to target

Use check context compatibility during 15.12.2.8 check (only when JDK 8 inference is enabled)

Reviewed-by: jjg
This commit is contained in:
Maurizio Cimadamore 2013-06-04 11:34:31 +01:00
parent 1c6a680e66
commit 5d6029baac
4 changed files with 49 additions and 12 deletions

View File

@ -218,8 +218,8 @@ public class Infer {
//we need to skip capture? //we need to skip capture?
Warner retWarn = new Warner(); Warner retWarn = new Warner();
if (!resultInfo.checkContext.compatible(qtype1, resultInfo.checkContext.inferenceContext().asFree(to), retWarn) || if (!resultInfo.checkContext.compatible(qtype1, resultInfo.checkContext.inferenceContext().asFree(to), retWarn) ||
//unchecked conversion is not allowed //unchecked conversion is not allowed in source 7 mode
retWarn.hasLint(Lint.LintCategory.UNCHECKED)) { (!allowGraphInference && retWarn.hasLint(Lint.LintCategory.UNCHECKED))) {
throw inferenceException throw inferenceException
.setMessage("infer.no.conforming.instance.exists", .setMessage("infer.no.conforming.instance.exists",
inferenceContext.restvars(), mt.getReturnType(), to); inferenceContext.restvars(), mt.getReturnType(), to);

View File

@ -0,0 +1,18 @@
/**
* @test /nodynamiccopyright/
* @bug 8015505
* @summary Spurious inference error when return type of generic method requires unchecked conversion to target
* @compile/fail/ref=T8015505.out -Xlint:-options -source 7 -XDrawDiagnostics T8015505.java
* @compile T8015505.java
*/
import java.util.List;
class T8015505 {
<Z> List m() { return null; }
void test() {
List<?> l = m();
}
}

View File

@ -0,0 +1,2 @@
T8015505.java:16:22: compiler.err.prob.found.req: (compiler.misc.infer.no.conforming.instance.exists: Z, java.util.List, java.util.List<?>)
1 error

View File

@ -46,6 +46,17 @@ public class GenericOverrideTest
extends JavacTestingAbstractThreadedTest extends JavacTestingAbstractThreadedTest
implements Runnable { implements Runnable {
enum SourceLevel {
SOURCE_7("-source", "7"),
SOURCE_DEFAULT();
String[] opts;
SourceLevel(String... opts) {
this.opts = opts;
}
}
enum SignatureKind { enum SignatureKind {
NON_GENERIC(""), NON_GENERIC(""),
GENERIC("<X>"); GENERIC("<X>");
@ -112,12 +123,13 @@ public class GenericOverrideTest
} }
} }
boolean assignableTo(TypeArgumentKind that, SignatureKind sig) { boolean assignableTo(TypeArgumentKind that, SignatureKind sig, SourceLevel level) {
switch (this) { switch (this) {
case NONE: case NONE:
//this case needs to workaround to javac's impl of 15.12.2.8 being too strict //this case needs to workaround to javac's impl of 15.12.2.8 being too strict
//ideally should be just 'return true' (see 7067746) //ideally should be just 'return true' (see 7067746/8015505)
return sig == SignatureKind.NON_GENERIC || that == NONE; return level == SourceLevel.SOURCE_DEFAULT ||
sig == SignatureKind.NON_GENERIC || that == NONE;
case UNBOUND: case UNBOUND:
return that == this || that == NONE; return that == this || that == NONE;
case INTEGER: case INTEGER:
@ -143,10 +155,12 @@ public class GenericOverrideTest
for (TypeArgumentKind ta3 : TypeArgumentKind.values()) { for (TypeArgumentKind ta3 : TypeArgumentKind.values()) {
if (!ta3.compatibleWith(SignatureKind.NON_GENERIC)) if (!ta3.compatibleWith(SignatureKind.NON_GENERIC))
continue; continue;
pool.execute( for (SourceLevel level : SourceLevel.values()) {
new GenericOverrideTest(sig1, pool.execute(
rt1, ta1, sig2, rt2, new GenericOverrideTest(sig1,
ta2, rt3, ta3)); rt1, ta1, sig2, rt2,
ta2, rt3, ta3, level));
}
} }
} }
} }
@ -162,12 +176,13 @@ public class GenericOverrideTest
SignatureKind sig1, sig2; SignatureKind sig1, sig2;
ReturnTypeKind rt1, rt2, rt3; ReturnTypeKind rt1, rt2, rt3;
TypeArgumentKind ta1, ta2, ta3; TypeArgumentKind ta1, ta2, ta3;
SourceLevel level;
JavaSource source; JavaSource source;
DiagnosticChecker diagChecker; DiagnosticChecker diagChecker;
GenericOverrideTest(SignatureKind sig1, ReturnTypeKind rt1, TypeArgumentKind ta1, GenericOverrideTest(SignatureKind sig1, ReturnTypeKind rt1, TypeArgumentKind ta1,
SignatureKind sig2, ReturnTypeKind rt2, TypeArgumentKind ta2, SignatureKind sig2, ReturnTypeKind rt2, TypeArgumentKind ta2,
ReturnTypeKind rt3, TypeArgumentKind ta3) { ReturnTypeKind rt3, TypeArgumentKind ta3, SourceLevel level) {
this.sig1 = sig1; this.sig1 = sig1;
this.sig2 = sig2; this.sig2 = sig2;
this.rt1 = rt1; this.rt1 = rt1;
@ -176,6 +191,7 @@ public class GenericOverrideTest
this.ta1 = ta1; this.ta1 = ta1;
this.ta2 = ta2; this.ta2 = ta2;
this.ta3 = ta3; this.ta3 = ta3;
this.level = level;
this.source = new JavaSource(); this.source = new JavaSource();
this.diagChecker = new DiagnosticChecker(); this.diagChecker = new DiagnosticChecker();
} }
@ -213,7 +229,8 @@ public class GenericOverrideTest
@Override @Override
public void run() { public void run() {
JavacTask ct = (JavacTask)comp.getTask(null, fm.get(), diagChecker, JavacTask ct = (JavacTask)comp.getTask(null, fm.get(), diagChecker,
null, null, Arrays.asList(source)); level.opts != null ? Arrays.asList(level.opts) : null,
null, Arrays.asList(source));
try { try {
ct.analyze(); ct.analyze();
} catch (Throwable ex) { } catch (Throwable ex) {
@ -271,7 +288,7 @@ public class GenericOverrideTest
SignatureKind mssig = mostSpecific == 1 ? sig1 : sig2; SignatureKind mssig = mostSpecific == 1 ? sig1 : sig2;
if (!msrt.moreSpecificThan(rt3) || if (!msrt.moreSpecificThan(rt3) ||
!msta.assignableTo(ta3, mssig)) { !msta.assignableTo(ta3, mssig, level)) {
errorExpected = true; errorExpected = true;
} }
} }