summaryrefslogtreecommitdiff
path: root/src/compiler/scala/tools/nsc/typechecker/Typers.scala
diff options
context:
space:
mode:
authorAdriaan Moors <adriaan.moors@epfl.ch>2012-07-23 14:14:24 +0200
committerAdriaan Moors <adriaan.moors@epfl.ch>2012-07-23 17:42:13 +0200
commitee5721e864de6fff2d54b9fb5452123bcca82483 (patch)
tree4d40dfff5f13984badf035d090bfcd6fea235ab0 /src/compiler/scala/tools/nsc/typechecker/Typers.scala
parent7b62eeea86c62f5e068c366065f8f4c2e6624eb7 (diff)
downloadscala-ee5721e864de6fff2d54b9fb5452123bcca82483.tar.gz
scala-ee5721e864de6fff2d54b9fb5452123bcca82483.tar.bz2
scala-ee5721e864de6fff2d54b9fb5452123bcca82483.zip
SI-6111 accept single-subpattern unapply pattern
An extractor pattern `X(p)` should type check for any `X.unapply`/`X.unapplySeq` that returns an `Option[_]` -- previously we were confused about the case where it was an `Option[(T1, ... , Tn)]`. In this case, the expected type for the pattern `p` is simply `(T1, ... , Tn)`. While I was at it, tried to clean up unapplyTypeList and friends (by replacing them by extractorFormalTypes). From the spec: 8.1.8 ExtractorPatterns An extractor pattern x(p1, ..., pn) where n ≥ 0 is of the same syntactic form as a constructor pattern. However, instead of a case class, the stable identifier x denotes an object which has a member method named unapply or unapplySeq that matches the pattern. An unapply method in an object x matches the pattern x(p1, ..., pn) if it takes exactly one argument and one of the following applies: n = 0 and unapply’s result type is Boolean. n = 1 and unapply’s result type is Option[T], for some type T. the (only) argument pattern p1 is typed in turn with expected type T n > 1 and unapply’s result type is Option[(T1, ..., Tn)], for some types T1, ..., Tn. the argument patterns p1, ..., pn are typed in turn with expected types T1, ..., Tn
Diffstat (limited to 'src/compiler/scala/tools/nsc/typechecker/Typers.scala')
-rw-r--r--src/compiler/scala/tools/nsc/typechecker/Typers.scala12
1 files changed, 7 insertions, 5 deletions
diff --git a/src/compiler/scala/tools/nsc/typechecker/Typers.scala b/src/compiler/scala/tools/nsc/typechecker/Typers.scala
index dbe65c16d8..8361ea9586 100644
--- a/src/compiler/scala/tools/nsc/typechecker/Typers.scala
+++ b/src/compiler/scala/tools/nsc/typechecker/Typers.scala
@@ -2749,6 +2749,7 @@ trait Typers extends Modes with Adaptations with Tags {
def typedArgs(args: List[Tree], mode: Int) =
args mapConserve (arg => typedArg(arg, mode, 0, WildcardType))
+ // [adriaan] as far as I can tell, formals0 is only supplied to detect whether the last formal was originally a vararg
def typedArgs(args0: List[Tree], mode: Int, formals0: List[Type], adapted0: List[Type]): List[Tree] = {
val sticky = onlyStickyModes(mode)
def loop(args: List[Tree], formals: List[Type], adapted: List[Type]): List[Tree] = {
@@ -3157,12 +3158,13 @@ trait Typers extends Modes with Adaptations with Tags {
if (fun1.tpe.isErroneous) duplErrTree
else {
- val formals0 = unapplyTypeList(fun1.symbol, fun1.tpe)
- val formals1 = formalTypes(formals0, args.length)
+ val resTp = fun1.tpe.finalResultType.normalize
+ val nbSubPats = args.length
- if (!sameLength(formals1, args)) duplErrorTree(WrongNumberArgsPatternError(tree, fun))
+ val (formals, formalsExpanded) = extractorFormalTypes(resTp, nbSubPats, fun1.symbol)
+ if (formals == null) duplErrorTree(WrongNumberArgsPatternError(tree, fun))
else {
- val args1 = typedArgs(args, mode, formals0, formals1)
+ val args1 = typedArgs(args, mode, formals, formalsExpanded)
// This used to be the following (failing) assert:
// assert(isFullyDefined(pt), tree+" ==> "+UnApply(fun1, args1)+", pt = "+pt)
// I modified as follows. See SI-1048.
@@ -4880,7 +4882,7 @@ trait Typers extends Modes with Adaptations with Tags {
case UnApply(fun, args) =>
val fun1 = typed(fun)
- val tpes = formalTypes(unapplyTypeList(fun.symbol, fun1.tpe), args.length)
+ val tpes = formalTypes(unapplyTypeList(fun.symbol, fun1.tpe, args.length), args.length)
val args1 = map2(args, tpes)(typedPattern)
treeCopy.UnApply(tree, fun1, args1) setType pt