Skip to content

Commit

Permalink
Merge pull request scala#3730 from lrytz/checkinit
Browse files Browse the repository at this point in the history
Fix checkinit build
  • Loading branch information
retronym committed May 9, 2014
2 parents 7289a26 + f0846e7 commit d079e76
Show file tree
Hide file tree
Showing 10 changed files with 87 additions and 35 deletions.
86 changes: 54 additions & 32 deletions src/compiler/scala/tools/nsc/transform/Mixin.scala
Original file line number Diff line number Diff line change
Expand Up @@ -1004,24 +1004,56 @@ abstract class Mixin extends InfoTransform with ast.TreeDSL {
buildBitmapOffsets()
var stats1 = addCheckedGetters(clazz, stats)

def accessedReference(sym: Symbol) = sym.tpe match {
case MethodType(Nil, ConstantType(c)) => Literal(c)
case _ =>
// if it is a mixed-in lazy value, complete the accessor
if (sym.isLazy && sym.isGetter) {
val isUnit = sym.tpe.resultType.typeSymbol == UnitClass
val initCall = Apply(staticRef(initializer(sym)), gen.mkAttributedThis(clazz) :: Nil)
val selection = Select(This(clazz), sym.accessed)
val init = if (isUnit) initCall else atPos(sym.pos)(Assign(selection, initCall))
val returns = if (isUnit) UNIT else selection

mkLazyDef(clazz, sym, List(init), returns, fieldOffset(sym))
}
else sym.getter(sym.owner).tpe.resultType.typeSymbol match {
case UnitClass => UNIT
case _ => Select(This(clazz), sym.accessed)
}
def getterBody(getter: Symbol) = {
assert(getter.isGetter)
val readValue = getter.tpe match {
// A field "final val f = const" in a trait generates a getter with a ConstantType.
case MethodType(Nil, ConstantType(c)) =>
Literal(c)
case _ =>
// if it is a mixed-in lazy value, complete the accessor
if (getter.isLazy) {
val isUnit = isUnitGetter(getter)
val initCall = Apply(staticRef(initializer(getter)), gen.mkAttributedThis(clazz) :: Nil)
val selection = fieldAccess(getter)
val init = if (isUnit) initCall else atPos(getter.pos)(Assign(selection, initCall))
val returns = if (isUnit) UNIT else selection
mkLazyDef(clazz, getter, List(init), returns, fieldOffset(getter))
}
// For a field of type Unit in a trait, no actual field is generated when being mixed in.
else if (isUnitGetter(getter)) UNIT
else fieldAccess(getter)
}
if (!needsInitFlag(getter)) readValue
else mkCheckedAccessor(clazz, readValue, fieldOffset(getter), getter.pos, getter)
}

def setterBody(setter: Symbol) = {
val getter = setter.getterIn(clazz)

// A trait with a field of type Unit creates a trait setter (invoked by the
// implementation class constructor), like for any other trait field.
// However, no actual field is created in the class that mixes in the trait.
// Therefore the setter does nothing (except setting the -Xcheckinit flag).

val setInitFlag =
if (!needsInitFlag(getter)) Nil
else List(mkSetFlag(clazz, fieldOffset(getter), getter, bitmapKind(getter)))

val fieldInitializer =
if (isUnitGetter(getter)) Nil
else List(Assign(fieldAccess(setter), Ident(setter.firstParam)))

(fieldInitializer ::: setInitFlag) match {
case Nil => UNIT
// If there's only one statement, the Block factory does not actually create a Block.
case stats => Block(stats: _*)
}
}

def isUnitGetter(getter: Symbol) = getter.tpe.resultType.typeSymbol == UnitClass
def fieldAccess(accessor: Symbol) = Select(This(clazz), accessor.accessed)

def isOverriddenSetter(sym: Symbol) =
nme.isTraitSetterName(sym.name) && {
val other = sym.nextOverriddenSymbol
Expand All @@ -1036,27 +1068,17 @@ abstract class Mixin extends InfoTransform with ast.TreeDSL {
}
// if class is not a trait add accessor definitions
else if (!clazz.isTrait) {
// This needs to be a def to avoid sharing trees
def accessedRef = accessedReference(sym)
if (isConcreteAccessor(sym)) {
// add accessor definitions
addDefDef(sym, {
if (sym.isSetter) {
// If this is a setter of a mixed-in field which is overridden by another mixin,
// the trait setter of the overridden one does not need to do anything - the
// trait setter of the overriding field will initialize the field.
if (isOverriddenSetter(sym)) UNIT
else accessedRef match {
case ref @ Literal(_) => ref
case ref =>
val init = Assign(ref, Ident(sym.firstParam))
val getter = sym.getter(clazz)

if (!needsInitFlag(getter)) init
else Block(init, mkSetFlag(clazz, fieldOffset(getter), getter, bitmapKind(getter)), UNIT)
}
else setterBody(sym)
}
else if (needsInitFlag(sym))
mkCheckedAccessor(clazz, accessedRef, fieldOffset(sym), sym.pos, sym)
else
accessedRef
else getterBody(sym)
})
}
else if (sym.isModule && !(sym hasFlag LIFTED | BRIDGE)) {
Expand Down
3 changes: 2 additions & 1 deletion test/files/run/t3569.scala
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,8 @@ object Test {
s.x += 1
println(s.x)

(classOf[X].getDeclaredFields map ("" + _)).sorted foreach println
// under -Xcheckinit there's an additional $init$ field
(classOf[X].getDeclaredFields map ("" + _)).sorted.filter(_ != "private volatile byte Test$X.bitmap$init$0") foreach println
(classOf[Y].getDeclaredFields map ("" + _)).sorted foreach println
}
}
3 changes: 2 additions & 1 deletion test/files/run/t5256h.scala
Original file line number Diff line number Diff line change
Expand Up @@ -6,5 +6,6 @@ object Test extends App {
val c = cm.classSymbol(mutant.getClass)
println(c)
println(c.fullName)
println(c.info)
// under -Xcheckinit there's an additional $init$ field
c.info.toString.lines.filter(_ != " private var bitmap$init$0: Boolean") foreach println
}
1 change: 1 addition & 0 deletions test/files/run/t7974.flags
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
-Xcheckinit:false
2 changes: 1 addition & 1 deletion test/files/run/t7974/Test.scala
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ object Test extends BytecodeTest {
val classNode = loadClassNode("Symbols", skipDebugInfo = false)
val textifier = new Textifier
classNode.accept(new TraceClassVisitor(null, textifier, null))

val classString = stringFromWriter(w => textifier.print(w))
val result =
classString.split('\n')
Expand Down
1 change: 1 addition & 0 deletions test/files/run/t8570.flags
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
-Xcheckinit
10 changes: 10 additions & 0 deletions test/files/run/t8570.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
trait Trait40_1 {
val value37_2 = ()
def run = { value37_2 }
}

object Test {
def main(args: Array[String]) {
(new Trait40_1 {}).run
}
}
1 change: 1 addition & 0 deletions test/files/run/t8570a.check
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
()
1 change: 1 addition & 0 deletions test/files/run/t8570a.flags
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
-Xcheckinit
14 changes: 14 additions & 0 deletions test/files/run/t8570a.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
trait Trait40_1 {
val value37_2 = ()
def run = { value37_2 }
}

trait T1 extends Trait40_1 {
override val value37_2 = ()
}

object Test {
def main(args: Array[String]) {
println((new T1 {}).run)
}
}

0 comments on commit d079e76

Please sign in to comment.