diff --git a/src/compiler/scala/tools/nsc/transform/Constructors.scala b/src/compiler/scala/tools/nsc/transform/Constructors.scala index e3d33fb2b548..99b3315b69c0 100644 --- a/src/compiler/scala/tools/nsc/transform/Constructors.scala +++ b/src/compiler/scala/tools/nsc/transform/Constructors.scala @@ -756,11 +756,16 @@ abstract class Constructors extends Statics with Transform with TypingTransforme if (isDelayedInitSubclass && remainingConstrStats.nonEmpty) delayedInitDefsAndConstrStats(defs, remainingConstrStats) else (Nil, remainingConstrStats) + val fence = if (clazz.primaryConstructor.hasAttachment[ConstructorNeedsFence.type]) { + val tree = localTyper.typedPos(clazz.primaryConstructor.pos)(gen.mkRuntimeCall(nme.releaseFence, Nil)) + tree :: Nil + } else Nil + // Assemble final constructor val primaryConstructor = deriveDefDef(primaryConstr)(_ => { treeCopy.Block( primaryConstrBody, - paramInits ::: constructorPrefix ::: uptoSuperStats ::: guardSpecializedInitializer(remainingConstrStatsDelayedInit), + paramInits ::: constructorPrefix ::: uptoSuperStats ::: guardSpecializedInitializer(remainingConstrStatsDelayedInit) ::: fence, primaryConstrBody.expr) }) diff --git a/src/compiler/scala/tools/nsc/transform/Fields.scala b/src/compiler/scala/tools/nsc/transform/Fields.scala index 39323abacf8c..2d50c76d3224 100644 --- a/src/compiler/scala/tools/nsc/transform/Fields.scala +++ b/src/compiler/scala/tools/nsc/transform/Fields.scala @@ -423,12 +423,12 @@ abstract class Fields extends InfoTransform with ast.TreeDSL with TypingTransfor // println(s"expanded modules for $clazz: $expandedModules") // afterOwnPhase, so traits receive trait setters for vals (needs to be at finest grain to avoid looping) - val synthInSubclass = + val synthInSubclass: List[Symbol] = clazz.mixinClasses.flatMap(mixin => afterOwnPhase{mixin.info}.decls.toList.filter(accessorImplementedInSubclass)) // mixin field accessors -- // invariant: (accessorsMaybeNeedingImpl, mixedInAccessorAndFields).zipped.forall(case (acc, clone :: _) => `clone` is clone of `acc` case _ => true) - val mixedInAccessorAndFields = synthInSubclass.map{ member => + val mixedInAccessorAndFields: List[List[Symbol]] = synthInSubclass.map{ member => def cloneAccessor() = { val clonedAccessor = (member cloneSymbol clazz) setPos clazz.pos setMixedinAccessorFlags(member, clonedAccessor) @@ -653,9 +653,15 @@ abstract class Fields extends InfoTransform with ast.TreeDSL with TypingTransfor // trait val/var setter mixed into class else fieldAccess(setter) match { case NoSymbol => EmptyTree - case fieldSel => afterOwnPhase { // the assign only type checks after our phase (assignment to val) - mkAccessor(setter)(Assign(Select(This(clazz), fieldSel), castHack(Ident(setter.firstParam), fieldSel.info))) - } + case fieldSel => + if (!fieldSel.hasFlag(MUTABLE)) { + fieldSel.setFlag(MUTABLE) + fieldSel.owner.primaryConstructor.updateAttachment(ConstructorNeedsFence) + } + + afterOwnPhase { // the assign only type checks after our phase (assignment to val) + mkAccessor(setter)(Assign(Select(This(clazz), fieldSel), castHack(Ident(setter.firstParam), fieldSel.info))) + } } def moduleAccessorBody(module: Symbol): Tree = diff --git a/src/reflect/scala/reflect/internal/StdAttachments.scala b/src/reflect/scala/reflect/internal/StdAttachments.scala index e704632b4991..bd560c7ef1b0 100644 --- a/src/reflect/scala/reflect/internal/StdAttachments.scala +++ b/src/reflect/scala/reflect/internal/StdAttachments.scala @@ -107,4 +107,6 @@ trait StdAttachments { case object KnownDirectSubclassesCalled extends PlainAttachment class QualTypeSymAttachment(val sym: Symbol) + + case object ConstructorNeedsFence extends PlainAttachment } diff --git a/src/reflect/scala/reflect/internal/StdNames.scala b/src/reflect/scala/reflect/internal/StdNames.scala index 882d7160d364..c611d7707900 100644 --- a/src/reflect/scala/reflect/internal/StdNames.scala +++ b/src/reflect/scala/reflect/internal/StdNames.scala @@ -756,6 +756,7 @@ trait StdNames { val productPrefix: NameType = "productPrefix" val raw_ : NameType = "raw" val readResolve: NameType = "readResolve" + val releaseFence: NameType = "releaseFence" val reify : NameType = "reify" val reificationSupport : NameType = "reificationSupport" val rootMirror : NameType = "rootMirror"