aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorodersky <odersky@gmail.com>2017-04-11 09:29:25 +0200
committerGitHub <noreply@github.com>2017-04-11 09:29:25 +0200
commit4ff656138a2e4e127b763adeee3f0f72d515f6b6 (patch)
treecd65d0f412217ccf5fb9cb082514fb6a0444e94b
parent0bd782195b7bcaa2d270b6306fd02c8a649822e9 (diff)
parent0cf17c5f63b3ec37a05da920a81337067ac335db (diff)
downloaddotty-4ff656138a2e4e127b763adeee3f0f72d515f6b6.tar.gz
dotty-4ff656138a2e4e127b763adeee3f0f72d515f6b6.tar.bz2
dotty-4ff656138a2e4e127b763adeee3f0f72d515f6b6.zip
Merge pull request #2215 from dotty-staging/#2142
Fix #2142: Skolemize arguments of dependent methods if necessary
-rw-r--r--compiler/src/dotty/tools/dotc/core/Types.scala5
-rw-r--r--compiler/src/dotty/tools/dotc/printing/Formatting.scala14
-rw-r--r--compiler/src/dotty/tools/dotc/printing/PlainPrinter.scala12
-rw-r--r--compiler/src/dotty/tools/dotc/printing/RefinedPrinter.scala2
-rw-r--r--compiler/src/dotty/tools/dotc/typer/Applications.scala5
-rw-r--r--compiler/src/dotty/tools/dotc/typer/TypeAssigner.scala20
-rw-r--r--tests/neg/i2142.scala28
7 files changed, 71 insertions, 15 deletions
diff --git a/compiler/src/dotty/tools/dotc/core/Types.scala b/compiler/src/dotty/tools/dotc/core/Types.scala
index e7130ee2f..4c69c9484 100644
--- a/compiler/src/dotty/tools/dotc/core/Types.scala
+++ b/compiler/src/dotty/tools/dotc/core/Types.scala
@@ -107,10 +107,11 @@ object Types {
final def isValueTypeOrLambda: Boolean = isValueType || this.isInstanceOf[TypeLambda]
/** Does this type denote a stable reference (i.e. singleton type)? */
- @tailrec final def isStable(implicit ctx: Context): Boolean = stripTypeVar match {
- case tp: TermRef => tp.termSymbol.isStable && tp.prefix.isStable
+ final def isStable(implicit ctx: Context): Boolean = stripTypeVar match {
+ case tp: TermRef => tp.termSymbol.isStable && tp.prefix.isStable || tp.info.isStable
case _: SingletonType | NoPrefix => true
case tp: RefinedOrRecType => tp.parent.isStable
+ case tp: ExprType => tp.resultType.isStable
case _ => false
}
diff --git a/compiler/src/dotty/tools/dotc/printing/Formatting.scala b/compiler/src/dotty/tools/dotc/printing/Formatting.scala
index e8fa45403..aa25880c2 100644
--- a/compiler/src/dotty/tools/dotc/printing/Formatting.scala
+++ b/compiler/src/dotty/tools/dotc/printing/Formatting.scala
@@ -107,7 +107,7 @@ object Formatting {
else nonSensicalStartTag + str + nonSensicalEndTag
}
- private type Recorded = AnyRef /*Symbol | TypeParamRef*/
+ private type Recorded = AnyRef /*Symbol | TypeParamRef | SkolemType */
private class Seen extends mutable.HashMap[String, List[Recorded]] {
@@ -135,8 +135,13 @@ object Formatting {
if ((sym is ModuleClass) && sym.sourceModule.exists) simpleNameString(sym.sourceModule)
else seen.record(super.simpleNameString(sym), sym)
- override def TypeParamRefNameString(param: TypeParamRef): String =
- seen.record(super.TypeParamRefNameString(param), param)
+ override def ParamRefNameString(param: ParamRef): String =
+ seen.record(super.ParamRefNameString(param), param)
+
+ override def toTextRef(tp: SingletonType): Text = tp match {
+ case tp: SkolemType => seen.record(tp.repr, tp)
+ case _ => super.toTextRef(tp)
+ }
}
/** Create explanation for single `Recorded` type or symbol */
@@ -165,6 +170,8 @@ object Formatting {
s"is a type variable${addendum("constraint", ctx.typeComparer.bounds(param))}"
case sym: Symbol =>
s"is a ${ctx.printer.kindString(sym)}${sym.showExtendedLocation}${addendum("bounds", sym.info)}"
+ case tp: SkolemType =>
+ s"is an unknown value of type ${tp.widen.show}"
}
}
@@ -176,6 +183,7 @@ object Formatting {
private def explanations(seen: Seen)(implicit ctx: Context): String = {
def needsExplanation(entry: Recorded) = entry match {
case param: TypeParamRef => ctx.typerState.constraint.contains(param)
+ case skolem: SkolemType => true
case _ => false
}
diff --git a/compiler/src/dotty/tools/dotc/printing/PlainPrinter.scala b/compiler/src/dotty/tools/dotc/printing/PlainPrinter.scala
index d5014b547..c762bbeaf 100644
--- a/compiler/src/dotty/tools/dotc/printing/PlainPrinter.scala
+++ b/compiler/src/dotty/tools/dotc/printing/PlainPrinter.scala
@@ -134,7 +134,7 @@ class PlainPrinter(_ctx: Context) extends Printer {
case tp: TypeType =>
toTextRHS(tp)
case tp: TermRef
- if !tp.denotationIsCurrent && !homogenizedView || // always print underyling when testing picklers
+ if !tp.denotationIsCurrent && !homogenizedView || // always print underlying when testing picklers
tp.symbol.is(Module) ||
tp.symbol.name.isImportName =>
toTextRef(tp) ~ ".type"
@@ -183,7 +183,9 @@ class PlainPrinter(_ctx: Context) extends Printer {
toTextGlobal(tp.resultType)
}
case tp: TypeParamRef =>
- TypeParamRefNameString(tp) ~ lambdaHash(tp.binder)
+ ParamRefNameString(tp) ~ lambdaHash(tp.binder)
+ case tp: TermParamRef =>
+ ParamRefNameString(tp) ~ ".type"
case AnnotatedType(tpe, annot) =>
toTextLocal(tpe) ~ " " ~ toText(annot)
case HKApply(tycon, args) =>
@@ -206,10 +208,10 @@ class PlainPrinter(_ctx: Context) extends Printer {
}
}.close
- protected def TypeParamRefNameString(name: TypeName): String = name.toString
+ protected def ParamRefNameString(name: Name): String = name.toString
- protected def TypeParamRefNameString(param: TypeParamRef): String =
- TypeParamRefNameString(param.binder.paramNames(param.paramNum))
+ protected def ParamRefNameString(param: ParamRef): String =
+ ParamRefNameString(param.binder.paramNames(param.paramNum))
/** The name of the symbol without a unique id. Under refined printing,
* the decoded original name.
diff --git a/compiler/src/dotty/tools/dotc/printing/RefinedPrinter.scala b/compiler/src/dotty/tools/dotc/printing/RefinedPrinter.scala
index 76bce7920..0b683d90c 100644
--- a/compiler/src/dotty/tools/dotc/printing/RefinedPrinter.scala
+++ b/compiler/src/dotty/tools/dotc/printing/RefinedPrinter.scala
@@ -604,7 +604,7 @@ class RefinedPrinter(_ctx: Context) extends PlainPrinter(_ctx) {
def optText[T >: Untyped](tree: List[Tree[T]])(encl: Text => Text): Text =
if (tree.exists(!_.isEmpty)) encl(blockText(tree)) else ""
- override protected def TypeParamRefNameString(name: TypeName): String =
+ override protected def ParamRefNameString(name: Name): String =
name.unexpandedName.toString
override protected def treatAsTypeParam(sym: Symbol): Boolean = sym is TypeParam
diff --git a/compiler/src/dotty/tools/dotc/typer/Applications.scala b/compiler/src/dotty/tools/dotc/typer/Applications.scala
index 4e43e429b..1fcebf4f0 100644
--- a/compiler/src/dotty/tools/dotc/typer/Applications.scala
+++ b/compiler/src/dotty/tools/dotc/typer/Applications.scala
@@ -395,9 +395,8 @@ trait Applications extends Compatibility { self: Typer with Dynamic =>
def addTyped(arg: Arg, formal: Type): Type => Type = {
addArg(typedArg(arg, formal), formal)
if (methodType.isParamDependent)
- _.substParam(methodType.newParamRef(n), typeOfArg(arg))
- else
- identity
+ safeSubstParam(_, methodType.paramRefs(n), typeOfArg(arg))
+ else identity
}
def missingArg(n: Int): Unit = {
diff --git a/compiler/src/dotty/tools/dotc/typer/TypeAssigner.scala b/compiler/src/dotty/tools/dotc/typer/TypeAssigner.scala
index 2aa7036b4..ead4ad5cb 100644
--- a/compiler/src/dotty/tools/dotc/typer/TypeAssigner.scala
+++ b/compiler/src/dotty/tools/dotc/typer/TypeAssigner.scala
@@ -315,10 +315,28 @@ trait TypeAssigner {
}
}
+ /** Substitute argument type `argType` for parameter `pref` in type `tp`,
+ * skolemizing the argument type if it is not stable and `pref` occurs in `tp`.
+ */
+ def safeSubstParam(tp: Type, pref: ParamRef, argType: Type)(implicit ctx: Context) = {
+ val tp1 = tp.substParam(pref, argType)
+ if ((tp1 eq tp) || argType.isStable) tp1
+ else tp.substParam(pref, SkolemType(argType.widen))
+ }
+
def assignType(tree: untpd.Apply, fn: Tree, args: List[Tree])(implicit ctx: Context) = {
val ownType = fn.tpe.widen match {
case fntpe: MethodType =>
- if (sameLength(fntpe.paramInfos, args) || ctx.phase.prev.relaxedTyping) fntpe.instantiate(args.tpes)
+ def safeSubstParams(tp: Type, params: List[ParamRef], args: List[Tree]): Type = params match {
+ case param :: params1 =>
+ val tp1 = safeSubstParam(tp, param, args.head.tpe)
+ safeSubstParams(tp1, params1, args.tail)
+ case Nil =>
+ tp
+ }
+ if (sameLength(fntpe.paramInfos, args) || ctx.phase.prev.relaxedTyping)
+ if (fntpe.isDependent) safeSubstParams(fntpe.resultType, fntpe.paramRefs, args)
+ else fntpe.resultType
else
errorType(i"wrong number of arguments for $fntpe: ${fn.tpe}, expected: ${fntpe.paramInfos.length}, found: ${args.length}", tree.pos)
case t =>
diff --git a/tests/neg/i2142.scala b/tests/neg/i2142.scala
new file mode 100644
index 000000000..7aeef95f0
--- /dev/null
+++ b/tests/neg/i2142.scala
@@ -0,0 +1,28 @@
+object Foo {
+
+class A
+val a1 = new A()
+val a2 = new A()
+
+def f(x: A, y: x.type) = ()
+f(a1, a1) // ok
+f(a1, a2) // error
+f(new A(), new A()) // error
+f(new A(), a1) // error
+
+def g(x: A)(y: x.type) = ()
+g(a1)(a1) // ok
+g(a1)(a2) // error
+g(new A())(new A()) // error
+g(new A())(a1) // error
+
+val x0 = g(new A()) _
+x0 (new A()) // error
+
+class C[T]
+
+def h(x: A): C[x.type] = ???
+val x = h(a1)
+val y = h(new A())
+
+}