![JAR search and dependency download from the Maven repository](/logo.png)
scala.tools.nsc.transform.patmat.ScalacPatternExpanders.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scala-compiler Show documentation
Show all versions of scala-compiler Show documentation
Compiler for the SubScript extension of the Scala Programming Language
The newest version!
/* NSC -- new Scala compiler
* Copyright 2005-2013 LAMP/EPFL
* @author Paul Phillips
*/
package scala
package tools
package nsc
package transform
package patmat
/** This is scalac-specific logic layered on top of the scalac-agnostic
* "matching products to patterns" logic defined in PatternExpander.
*/
trait ScalacPatternExpanders {
val global: Global
import global._
import definitions._
import treeInfo._
import analyzer._
type PatternAligned = ScalacPatternExpander#Aligned
implicit class AlignedOps(val aligned: PatternAligned) {
import aligned._
def expectedTypes = typedPatterns map (_.tpe)
def unexpandedFormals = extractor.varargsTypes
}
trait ScalacPatternExpander extends PatternExpander[Tree, Type] {
def NoPattern = EmptyTree
def NoType = global.NoType
def newPatterns(patterns: List[Tree]): Patterns = patterns match {
case init :+ last if isStar(last) => Patterns(init, last)
case _ => Patterns(patterns, NoPattern)
}
def elementTypeOf(tpe: Type) = {
val seq = repeatedToSeq(tpe)
( typeOfMemberNamedHead(seq)
orElse typeOfMemberNamedApply(seq)
orElse definitions.elementType(ArrayClass, seq)
)
}
def newExtractor(whole: Type, fixed: List[Type], repeated: Repeated): Extractor =
logResult(s"newExtractor($whole, $fixed, $repeated")(Extractor(whole, fixed, repeated))
// Turn Seq[A] into Repeated(Seq[A], A, A*)
def repeatedFromSeq(seqType: Type): Repeated = {
val elem = elementTypeOf(seqType)
val repeated = scalaRepeatedType(elem)
Repeated(seqType, elem, repeated)
}
// Turn A* into Repeated(Seq[A], A, A*)
def repeatedFromVarargs(repeated: Type): Repeated =
Repeated(repeatedToSeq(repeated), repeatedToSingle(repeated), repeated)
/** In this case we are basing the pattern expansion on a case class constructor.
* The argument is the MethodType carried by the primary constructor.
*/
def applyMethodTypes(method: Type): Extractor = {
val whole = method.finalResultType
method.paramTypes match {
case init :+ last if isScalaRepeatedParamType(last) => newExtractor(whole, init, repeatedFromVarargs(last))
case tps => newExtractor(whole, tps, NoRepeated)
}
}
/** In this case, expansion is based on an unapply or unapplySeq method.
* Unfortunately the MethodType does not carry the information of whether
* it was unapplySeq, so we have to funnel that information in separately.
*/
def unapplyMethodTypes(whole: Type, result: Type, isSeq: Boolean): Extractor = {
val expanded = (
if (result =:= BooleanTpe) Nil
else typeOfMemberNamedGet(result) match {
case rawGet if !hasSelectors(rawGet) => rawGet :: Nil
case rawGet => typesOfSelectors(rawGet)
}
)
expanded match {
case init :+ last if isSeq => newExtractor(whole, init, repeatedFromSeq(last))
case tps => newExtractor(whole, tps, NoRepeated)
}
}
}
object alignPatterns extends ScalacPatternExpander {
/** Converts a T => (A, B, C) extractor to a T => ((A, B, CC)) extractor.
*/
def tupleExtractor(extractor: Extractor): Extractor =
extractor.copy(fixed = tupleType(extractor.fixed) :: Nil)
private def validateAligned(context: Context, tree: Tree, aligned: Aligned): Aligned = {
import aligned._
def owner = tree.symbol.owner
def offering = extractor.offeringString
def symString = tree.symbol.fullLocationString
def offerString = if (extractor.isErroneous) "" else s" offering $offering"
def arityExpected = ( if (extractor.hasSeq) "at least " else "" ) + productArity
def err(msg: String) = context.error(tree.pos, msg)
def warn(msg: String) = context.warning(tree.pos, msg)
def arityError(what: String) = err(s"$what patterns for $owner$offerString: expected $arityExpected, found $totalArity")
if (isStar && !isSeq)
err("Star pattern must correspond with varargs or unapplySeq")
else if (elementArity < 0)
arityError("not enough")
else if (elementArity > 0 && !extractor.hasSeq)
arityError("too many")
aligned
}
def apply(context: Context, sel: Tree, args: List[Tree]): Aligned = {
val fn = sel match {
case Unapplied(fn) => fn
case _ => sel
}
val patterns = newPatterns(args)
val isUnapply = sel.symbol.name == nme.unapply
val extractor = sel.symbol.name match {
case nme.unapply => unapplyMethodTypes(firstParamType(fn.tpe), sel.tpe, isSeq = false)
case nme.unapplySeq => unapplyMethodTypes(firstParamType(fn.tpe), sel.tpe, isSeq = true)
case _ => applyMethodTypes(fn.tpe)
}
/** Rather than let the error that is SI-6675 pollute the entire matching
* process, we will tuple the extractor before creation Aligned so that
* it contains known good values.
*/
def productArity = extractor.productArity
def acceptMessage = if (extractor.isErroneous) "" else s" to hold ${extractor.offeringString}"
val requiresTupling = isUnapply && patterns.totalArity == 1 && productArity > 1
if (requiresTupling && effectivePatternArity(args) == 1) {
val sym = sel.symbol.owner
currentRun.reporting.deprecationWarning(sel.pos, sym, s"${sym} expects $productArity patterns$acceptMessage but crushing into $productArity-tuple to fit single pattern (SI-6675)")
}
val normalizedExtractor = if (requiresTupling) tupleExtractor(extractor) else extractor
validateAligned(context, fn, Aligned(patterns, normalizedExtractor))
}
def apply(context: Context, tree: Tree): Aligned = tree match {
case Apply(fn, args) => apply(context, fn, args)
case UnApply(fn, args) => apply(context, fn, args)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy