package scala.tools.nsc
package symtab
import scala.collection.mutable.{ListBuffer, BitSet}
import math.max
import util.Statistics._
trait BaseTypeSeqs {
this: SymbolTable =>
import definitions._
class BaseTypeSeq(parents: List[Type], elems: Array[Type]) {
self =>
incCounter(baseTypeSeqCount)
incCounter(baseTypeSeqLenTotal, elems.length)
def length: Int = elems.length
val pending = new BitSet(length)
def apply(i: Int): Type =
if(pending contains i) {
pending.clear()
throw CyclicInheritance
} else
elems(i) match {
case rtp @ RefinedType(variants, decls) =>
pending += i
try {
mergePrefixAndArgs(variants, -1, lubDepth(variants)) match {
case Some(tp0) =>
pending(i) = false
elems(i) = tp0
tp0
case None =>
typeError(
"no common type instance of base types "+(variants mkString ", and ")+" exists.")
}
} catch {
case CyclicInheritance =>
typeError(
"computing the common type instance of base types "+(variants mkString ", and ")+" leads to a cycle.")
}
case tp =>
tp
}
def rawElem(i: Int) = elems(i)
def typeSymbol(i: Int): Symbol = {
elems(i) match {
case RefinedType(v :: vs, _) => v.typeSymbol
case tp => tp.typeSymbol
}
}
def toList: List[Type] = elems.toList
protected def copy(head: Type, offset: Int): BaseTypeSeq = {
val arr = new Array[Type](elems.length + offset)
compat.Platform.arraycopy(elems, 0, arr, offset, elems.length)
arr(0) = head
new BaseTypeSeq(parents, arr)
}
def prepend(tp: Type): BaseTypeSeq = copy(tp, 1)
def updateHead(tp: Type): BaseTypeSeq = copy(tp, 0)
def map(f: Type => Type): BaseTypeSeq = {
val len = length
var arr = new Array[Type](len)
var i = 0
while (i < len) {
arr(i) = f(elems(i))
i += 1
}
new BaseTypeSeq(parents, arr)
}
def lateMap(f: Type => Type): BaseTypeSeq = new BaseTypeSeq(parents map f, elems) {
override def apply(i: Int) = f(self.apply(i))
override def rawElem(i: Int) = f(self.rawElem(i))
override def typeSymbol(i: Int) = self.typeSymbol(i)
override def toList = self.toList map f
override protected def copy(head: Type, offset: Int) = (self map f).copy(head, offset)
override def map(g: Type => Type) = lateMap(g)
override def lateMap(g: Type => Type) = self.lateMap(x => g(f(x)))
override def exists(p: Type => Boolean) = elems exists (x => p(f(x)))
override protected def maxDepthOfElems: Int = elems map (x => maxDpth(f(x))) max
override def toString = elems.mkString("MBTS(", ",", ")")
}
def exists(p: Type => Boolean): Boolean = elems exists p
lazy val maxDepth: Int = maxDepthOfElems
protected def maxDepthOfElems = {
var d = 0
for (i <- 0 until length) d = max(d, maxDpth(elems(i)))
d
}
protected def maxDpth(tp: Type): Int = tp match {
case TypeRef(pre, sym, args) =>
max(maxDpth(pre), maxDpth(args) + 1)
case RefinedType(parents, decls) =>
max(maxDpth(parents), maxDpth(decls.toList.map(_.info)) + 1)
case TypeBounds(lo, hi) =>
max(maxDpth(lo), maxDpth(hi))
case MethodType(paramtypes, result) =>
maxDpth(result)
case NullaryMethodType(result) =>
maxDpth(result)
case PolyType(tparams, result) =>
max(maxDpth(result), maxDpth(tparams map (_.info)) + 1)
case ExistentialType(tparams, result) =>
max(maxDpth(result), maxDpth(tparams map (_.info)) + 1)
case _ =>
1
}
private def maxDpth(tps: Seq[Type]): Int = {
var d = 0
for (tp <- tps) d = max(d, maxDpth(tp))
d
}
override def toString = elems.mkString("BTS(", ",", ")")
private def typeError(msg: String): Nothing =
throw new TypeError(
"the type intersection "+(parents mkString " with ")+" is malformed"+
"\n --- because ---\n"+msg)
}
val undetBaseTypeSeq: BaseTypeSeq = new BaseTypeSeq(List(), Array())
def baseTypeSingletonSeq(tp: Type): BaseTypeSeq = new BaseTypeSeq(List(), Array(tp))
def compoundBaseTypeSeq(tp: Type): BaseTypeSeq = {
val tsym = tp.typeSymbol
val parents = tp.parents
val buf = new ListBuffer[Type]
buf += tsym.tpe
var btsSize = 1
if (parents.nonEmpty) {
val nparents = parents.length
val pbtss = new Array[BaseTypeSeq](nparents)
val index = new Array[Int](nparents)
var i = 0
for (p <- parents) {
pbtss(i) =
if (p.baseTypeSeq eq undetBaseTypeSeq) AnyClass.info.baseTypeSeq
else p.baseTypeSeq
index(i) = 0
i += 1
}
def nextTypeSymbol(i: Int): Symbol = {
val j = index(i)
val pbts = pbtss(i)
if (j < pbts.length) pbts.typeSymbol(j) else AnyClass
}
def nextRawElem(i: Int): Type = {
val j = index(i)
val pbts = pbtss(i)
if (j < pbts.length) pbts.rawElem(j) else AnyClass.tpe
}
var minSym: Symbol = NoSymbol
while (minSym != AnyClass) {
minSym = nextTypeSymbol(0)
i = 1
while (i < nparents) {
val nextSym = nextTypeSymbol(i)
if (nextSym isLess minSym)
minSym = nextSym
i += 1
}
var minTypes: List[Type] = List()
i = 0
while (i < nparents) {
if (nextTypeSymbol(i) == minSym) {
nextRawElem(i) match {
case RefinedType(variants, decls) =>
for (tp <- variants)
if (!(minTypes exists (tp =:=))) minTypes = tp :: minTypes
case tp =>
if (!(minTypes exists (tp =:=))) minTypes = tp :: minTypes
}
index(i) = index(i) + 1
}
i += 1
}
buf += intersectionType(minTypes)
btsSize += 1
}
}
val elems = new Array[Type](btsSize)
buf.copyToArray(elems, 0)
new BaseTypeSeq(parents, elems)
}
val CyclicInheritance = new Throwable
}