package scala.tools.nsc
package backend
package icode
trait TypeKinds { self: ICodes =>
import global._
import definitions.{ ArrayClass, AnyRefClass, ObjectClass, NullClass, NothingClass, arrayType }
import icodes.{ checkerDebug, NothingReference, NullReference }
lazy val primitiveTypeMap: Map[Symbol, TypeKind] = {
import definitions._
Map(
UnitClass -> UNIT,
BooleanClass -> BOOL,
CharClass -> CHAR,
ByteClass -> BYTE,
ShortClass -> SHORT,
IntClass -> INT,
LongClass -> LONG,
FloatClass -> FLOAT,
DoubleClass -> DOUBLE
)
}
private lazy val reversePrimitiveMap: Map[TypeKind, Symbol] =
primitiveTypeMap map (_.swap) toMap
sealed abstract class TypeKind {
def maxType(other: TypeKind): TypeKind
def toType: Type = reversePrimitiveMap get this map (_.tpe) getOrElse {
this match {
case REFERENCE(cls) => cls.tpe
case ARRAY(elem) => arrayType(elem.toType)
case _ => abort("Unknown type kind.")
}
}
def isReferenceType = false
def isArrayType = false
def isValueType = false
def isBoxedType = false
final def isRefOrArrayType = isReferenceType || isArrayType
final def isRefArrayOrBoxType = isRefOrArrayType || isBoxedType
final def isNothingType = this == NothingReference
final def isNullType = this == NullReference
final def isInterfaceType = this match {
case REFERENCE(cls) if cls.isInterface || cls.isTrait => true
case _ => false
}
def isIntSizedType: Boolean = this match {
case BOOL | CHAR | BYTE | SHORT | INT => true
case _ => false
}
def isIntegralType: Boolean = this match {
case BYTE | SHORT | INT | LONG | CHAR => true
case _ => false
}
def isRealType: Boolean = this match {
case FLOAT | DOUBLE => true
case _ => false
}
def isNumericType: Boolean = isIntegralType | isRealType
def <:<(other: TypeKind): Boolean = (this eq other) || (this match {
case BOOL | BYTE | SHORT | CHAR => other == INT || other == LONG
case _ => this eq other
})
def isWideType: Boolean = this match {
case DOUBLE | LONG => true
case _ => false
}
def dimensions: Int = 0
protected def uncomparable(thisKind: String, other: TypeKind): Nothing =
abort("Uncomparable type kinds: " + thisKind + " with " + other)
protected def uncomparable(other: TypeKind): Nothing =
uncomparable(this.toString, other)
}
sealed abstract class ValueTypeKind extends TypeKind {
override def isValueType = true
override def toString = {
this.getClass.getName stripSuffix "$" dropWhile (_ != '$') drop 1
}
}
var lubs0 = 0
def lub(a: TypeKind, b: TypeKind): TypeKind = {
def lub0(tk1: TypeKind, tk2: TypeKind): Type = atPhase(currentRun.uncurryPhase) {
import definitions._
val tp = global.lub(List(tk1.toType, tk2.toType))
val (front, rest) = tp.parents span (_.typeSymbol.hasTraitFlag)
if (front.isEmpty) tp
else if (rest.isEmpty) front.head
else rest.head match {
case AnyRefClass | ObjectClass => tp
case x => x
}
}
def isIntLub = (
(a == INT && b.isIntSizedType) ||
(b == INT && a.isIntSizedType)
)
if (a == b) a
else if (a.isNothingType) b
else if (b.isNothingType) a
else if (a.isBoxedType || b.isBoxedType) AnyRefReference
else if (isIntLub) INT
else if (a.isRefOrArrayType && b.isRefOrArrayType) {
if (a.isNullType) b
else if (b.isNullType) a
else toTypeKind(lub0(a, b))
}
else throw new CheckerException("Incompatible types: " + a + " with " + b)
}
case object UNIT extends ValueTypeKind {
def maxType(other: TypeKind) = other match {
case UNIT | REFERENCE(NothingClass) => UNIT
case _ => uncomparable(other)
}
}
case object BOOL extends ValueTypeKind {
def maxType(other: TypeKind) = other match {
case BOOL | REFERENCE(NothingClass) => BOOL
case _ => uncomparable(other)
}
}
case object BYTE extends ValueTypeKind {
def maxType(other: TypeKind) = {
if (other == BYTE || other.isNothingType) BYTE
else if (other == CHAR) INT
else if (other.isNumericType) other
else uncomparable(other)
}
}
case object SHORT extends ValueTypeKind {
override def maxType(other: TypeKind) = other match {
case BYTE | SHORT | REFERENCE(NothingClass) => SHORT
case CHAR => INT
case INT | LONG | FLOAT | DOUBLE => other
case _ => uncomparable(other)
}
}
case object CHAR extends ValueTypeKind {
override def maxType(other: TypeKind) = other match {
case CHAR | REFERENCE(NothingClass) => CHAR
case BYTE | SHORT => INT
case INT | LONG | FLOAT | DOUBLE => other
case _ => uncomparable(other)
}
}
case object INT extends ValueTypeKind {
override def maxType(other: TypeKind) = other match {
case BYTE | SHORT | CHAR | INT | REFERENCE(NothingClass) => INT
case LONG | FLOAT | DOUBLE => other
case _ => uncomparable(other)
}
}
case object LONG extends ValueTypeKind {
override def maxType(other: TypeKind): TypeKind =
if (other.isIntegralType || other.isNothingType) LONG
else if (other.isRealType) DOUBLE
else uncomparable(other)
}
case object FLOAT extends ValueTypeKind {
override def maxType(other: TypeKind): TypeKind =
if (other == DOUBLE) DOUBLE
else if (other.isNumericType || other.isNothingType) FLOAT
else uncomparable(other)
}
case object DOUBLE extends ValueTypeKind {
override def maxType(other: TypeKind): TypeKind =
if (other.isNumericType || other.isNothingType) DOUBLE
else uncomparable(other)
}
final case class REFERENCE(cls: Symbol) extends TypeKind {
override def toString = "REF(" + cls + ")"
assert(cls ne null,
"REFERENCE to null class symbol.")
assert(cls != ArrayClass,
"REFERENCE to Array is not allowed, should be ARRAY[..] instead")
assert(cls != NoSymbol,
"REFERENCE to NoSymbol not allowed!")
override def maxType(other: TypeKind) = other match {
case REFERENCE(_) | ARRAY(_) => AnyRefReference
case _ => uncomparable("REFERENCE", other)
}
override def <:<(other: TypeKind) = isNothingType || (other match {
case REFERENCE(cls2) => cls.tpe <:< cls2.tpe
case ARRAY(_) => cls == NullClass
case _ => false
})
override def isReferenceType = true
}
def ArrayN(elem: TypeKind, dims: Int): ARRAY = {
assert(dims > 0)
if (dims == 1) ARRAY(elem)
else ARRAY(ArrayN(elem, dims - 1))
}
final case class ARRAY(val elem: TypeKind) extends TypeKind {
override def toString = "ARRAY[" + elem + "]"
override def isArrayType = true
override def dimensions = 1 + elem.dimensions
def elementKind: TypeKind = elem match {
case a @ ARRAY(_) => a.elementKind
case k => k
}
override def maxType(other: TypeKind) = other match {
case ARRAY(elem2) if elem == elem2 => ARRAY(elem)
case ARRAY(_) | REFERENCE(_) => AnyRefReference
case _ => uncomparable("ARRAY", other)
}
override def <:<(other: TypeKind) = other match {
case ARRAY(elem2) => elem <:< elem2
case REFERENCE(AnyRefClass | ObjectClass) => true
case _ => false
}
}
case class BOXED(kind: TypeKind) extends TypeKind {
override def isBoxedType = true
override def maxType(other: TypeKind) = other match {
case BOXED(`kind`) => this
case REFERENCE(_) | ARRAY(_) | BOXED(_) => AnyRefReference
case _ => uncomparable("BOXED", other)
}
override def <:<(other: TypeKind) = other match {
case BOXED(`kind`) => true
case REFERENCE(AnyRefClass | ObjectClass) => true
case _ => false
}
}
case object ConcatClass extends TypeKind {
override def toString = "ConcatClass"
override def maxType(other: TypeKind) = other match {
case REFERENCE(_) => AnyRefReference
case _ => uncomparable(other)
}
override def <:<(other: TypeKind) = this eq other
}
def toTypeKind(t: Type): TypeKind = t.normalize match {
case ThisType(ArrayClass) => ObjectReference
case ThisType(sym) => REFERENCE(sym)
case SingleType(_, sym) => primitiveOrRefType(sym)
case ConstantType(_) => toTypeKind(t.underlying)
case TypeRef(_, sym, args) => primitiveOrClassType(sym, args)
case ClassInfoType(_, _, ArrayClass) => abort("ClassInfoType to ArrayClass!")
case ClassInfoType(_, _, sym) => primitiveOrRefType(sym)
case ExistentialType(_, t) => toTypeKind(t)
case AnnotatedType(_, t, _) => toTypeKind(t)
case RefinedType(parents, _) => parents map toTypeKind reduceLeft lub
case norm => abort(
"Unknown type: %s, %s [%s, %s] TypeRef? %s".format(
t, norm, t.getClass, norm.getClass, t.isInstanceOf[TypeRef]
)
)
}
private def arrayOrClassType(sym: Symbol, targs: List[Type]) = sym match {
case ArrayClass => ARRAY(toTypeKind(targs.head))
case _ if sym.isClass => newReference(sym)
case _ =>
assert(sym.isType, sym)
ObjectReference
}
private def newReference(sym: Symbol): TypeKind = {
if (sym.isImplClass) {
val traitSym = sym.owner.info.decl(nme.interfaceName(sym.name))
if (traitSym != NoSymbol)
return REFERENCE(traitSym)
}
REFERENCE(sym)
}
private def primitiveOrRefType(sym: Symbol) =
primitiveTypeMap.getOrElse(sym, newReference(sym))
private def primitiveOrClassType(sym: Symbol, targs: List[Type]) =
primitiveTypeMap.getOrElse(sym, arrayOrClassType(sym, targs))
def msil_mgdptr(tk: TypeKind): TypeKind = (tk: @unchecked) match {
case REFERENCE(cls) => REFERENCE(loaders.clrTypes.mdgptrcls4clssym(cls))
case _ => abort("cannot obtain a managed pointer for " + tk)
}
}