package scala.tools.nsc
package typechecker
import symtab.Flags
import symtab.Flags._
import scala.collection.mutable
import scala.collection.mutable.ListBuffer
trait SyntheticMethods extends ast.TreeDSL {
self: Analyzer =>
import global._
import definitions._
private val createdMethodSymbols = new mutable.HashSet[Symbol]
def resetSynthetics() {
createdMethodSymbols.clear()
}
def addSyntheticMethods(templ: Template, clazz: Symbol, context: Context): Template = {
val localTyper = newTyper(
if (reporter.hasErrors) context makeSilent false else context
)
def hasOverridingImplementation(meth: Symbol): Boolean = {
val sym = clazz.info nonPrivateMember meth.name
def isOverride(s: Symbol) = {
s != meth && !s.isDeferred && !s.isSynthetic && !createdMethodSymbols(s) &&
(clazz.thisType.memberType(s) matches clazz.thisType.memberType(meth))
}
sym.alternatives exists isOverride
}
def syntheticMethod(name: Name, flags: Int, tpeCons: Symbol => Type) =
newSyntheticMethod(name, flags | OVERRIDE, tpeCons)
def newSyntheticMethod(name: Name, flags: Int, tpeCons: Symbol => Type) = {
val method = clazz.newMethod(clazz.pos.focus, name.toTermName) setFlag flags
createdMethodSymbols += method
method setInfo tpeCons(method)
clazz.info.decls.enter(method)
}
def makeNoArgConstructor(res: Type) =
(sym: Symbol) => MethodType(Nil, res)
def makeTypeConstructor(args: List[Type], res: Type) =
(sym: Symbol) => MethodType(sym newSyntheticValueParams args, res)
def makeEqualityMethod(name: Name) =
syntheticMethod(name, 0, makeTypeConstructor(List(AnyClass.tpe), BooleanClass.tpe))
import CODE._
def productPrefixMethod: Tree = typer.typed {
val method = syntheticMethod(nme.productPrefix, 0, sym => NullaryMethodType(StringClass.tpe))
DEF(method) === LIT(clazz.name.decode)
}
def productArityMethod(nargs: Int): Tree = {
val method = syntheticMethod(nme.productArity, 0, sym => NullaryMethodType(IntClass.tpe))
typer typed { DEF(method) === LIT(nargs) }
}
def perElementMethod(accs: List[Symbol], methodName: Name, resType: Type, caseFn: Symbol => Tree): Tree = {
val symToTpe = makeTypeConstructor(List(IntClass.tpe), resType)
val method = syntheticMethod(methodName, 0, symToTpe)
val arg = method ARG 0
val default = List(DEFAULT ==> THROW(IndexOutOfBoundsExceptionClass, arg))
val cases =
for ((sym, i) <- accs.zipWithIndex) yield
CASE(LIT(i)) ==> caseFn(sym)
typer typed {
DEF(method) === {
arg MATCH { cases ::: default : _* }
}
}
}
def productElementMethod(accs: List[Symbol]): Tree =
perElementMethod(accs, nme.productElement, AnyClass.tpe, x => Ident(x))
def moduleToStringMethod: Tree = {
val method = syntheticMethod(nme.toString_, FINAL, makeNoArgConstructor(StringClass.tpe))
typer typed { DEF(method) === LIT(clazz.name.decode) }
}
def moduleHashCodeMethod: Tree = {
val method = syntheticMethod(nme.hashCode_, FINAL, makeNoArgConstructor(IntClass.tpe))
val code = clazz.name.decode.hashCode
typer typed { DEF(method) === LIT(code) }
}
def forwardingMethod(name: Name, targetName: Name): Tree = {
val target = getMember(ScalaRunTimeModule, targetName)
val paramtypes = target.tpe.paramTypes drop 1
val method = syntheticMethod(name, 0, makeTypeConstructor(paramtypes, target.tpe.resultType))
typer typed {
DEF(method) === {
Apply(REF(target), This(clazz) :: (method ARGNAMES))
}
}
}
def equalsModuleMethod: Tree = {
val method = makeEqualityMethod(nme.equals_)
val that = method ARG 0
localTyper typed {
DEF(method) === (This(clazz) ANY_EQ that)
}
}
def canEqualMethod: Tree = {
val method = makeEqualityMethod(nme.canEqual_)
val that = method ARG 0
typer typed (DEF(method) === (that IS_OBJ clazz.tpe))
}
def equalsClassMethod: Tree = {
val method = makeEqualityMethod(nme.equals_)
val that = method ARG 0
val constrParamTypes = clazz.primaryConstructor.tpe.paramTypes
def makeTrees(acc: Symbol, cpt: Type): (Tree, Bind) = {
val varName = context.unit.freshTermName(acc.name + "$")
val isRepeated = isRepeatedParamType(cpt)
val binding = if (isRepeated) Star(WILD()) else WILD()
val eqMethod: Tree =
if (isRepeated) gen.mkRuntimeCall(nme.sameElements, List(Ident(varName), Ident(acc)))
else (Ident(varName) DOT nme.EQ)(Ident(acc))
(eqMethod, Bind(varName, binding))
}
val (guards, params) = (clazz.caseFieldAccessors, constrParamTypes).zipped map makeTrees unzip
def canEqualCheck() = {
val that: Tree = (method ARG 0) AS clazz.tpe
val canEqualOther: Symbol = clazz.info nonPrivateMember nme.canEqual_
typer typed {
(that DOT canEqualOther)(This(clazz))
}
}
val (guard, pat) = (AND(guards: _*), Ident(clazz.name.toTermName) APPLY params)
localTyper typed {
DEF(method) === {
(This(clazz) ANY_EQ that) OR (that MATCH(
(CASE(pat) IF guard) ==> canEqualCheck() ,
DEFAULT ==> FALSE
))
}
}
}
def newAccessorMethod(tree: Tree): Tree = tree match {
case DefDef(_, _, _, _, _, rhs) =>
var newAcc = tree.symbol.cloneSymbol
newAcc.name = context.unit.freshTermName(tree.symbol.name + "$")
newAcc setFlag SYNTHETIC resetFlag (ACCESSOR | PARAMACCESSOR | PRIVATE | PROTECTED)
newAcc.privateWithin = NoSymbol
newAcc = newAcc.owner.info.decls enter newAcc
val result = typer typed { DEF(newAcc) === rhs.duplicate }
log("new accessor method " + result)
result
}
def needsReadResolve = (
clazz.isSerializable && (clazz.owner.isPackageClass || clazz.owner.isModuleClass)
)
val ts = new ListBuffer[Tree]
if (!phase.erasedTypes) try {
if (clazz.isCase) {
val isTop = clazz.ancestors forall (x => !x.isCase)
if (isTop) {
def needsService(s: Symbol) = s.isMethod && s.isCaseAccessor && !s.isPublic
for (stat <- templ.body ; if stat.isDef && needsService(stat.symbol)) {
ts += newAccessorMethod(stat)
stat.symbol resetFlag CASEACCESSOR
}
}
def classMethods = List(
Object_hashCode -> (() => forwardingMethod(nme.hashCode_, "_" + nme.hashCode_)),
Object_toString -> (() => forwardingMethod(nme.toString_, "_" + nme.toString_)),
Object_equals -> (() => equalsClassMethod)
)
def objectMethods = List(
Object_hashCode -> (() => moduleHashCodeMethod),
Object_toString -> (() => moduleToStringMethod)
)
def everywhereMethods = {
val accessors = clazz.caseFieldAccessors
List(
Product_productPrefix -> (() => productPrefixMethod),
Product_productArity -> (() => productArityMethod(accessors.length)),
Product_productElement -> (() => productElementMethod(accessors)),
Product_canEqual -> (() => canEqualMethod)
)
}
if (clazz.isModuleClass) {
val otherEquals = clazz.info.nonPrivateMember(Object_equals.name)
if (otherEquals.owner != clazz && createdMethodSymbols(otherEquals)) ts += equalsModuleMethod
}
val methods = (if (clazz.isModuleClass) objectMethods else classMethods) ++ everywhereMethods
for ((m, impl) <- methods ; if !hasOverridingImplementation(m))
ts += impl()
}
if (clazz.isModuleClass) {
def hasReadResolve = {
val sym = clazz.info member nme.readResolve
sym.isTerm && !sym.isDeferred
}
if (!hasReadResolve && needsReadResolve){
val method = newSyntheticMethod(nme.readResolve, PROTECTED, makeNoArgConstructor(ObjectClass.tpe))
ts += typer typed (DEF(method) === REF(clazz.sourceModule))
}
}
} catch {
case ex: TypeError =>
if (!reporter.hasErrors) throw ex
}
if (phase.id <= currentRun.typerPhase.id) {
treeCopy.Template(templ, templ.parents, templ.self,
if (ts.isEmpty) templ.body else templ.body ++ ts
)
}
else templ
}
}