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 b2553191ccf1..deb3d9f71f1e 100644 --- a/src/compiler/scala/tools/nsc/transform/Fields.scala +++ b/src/compiler/scala/tools/nsc/transform/Fields.scala @@ -422,12 +422,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) @@ -652,9 +652,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 706c07a6a7be..d5351cb41298 100644 --- a/src/reflect/scala/reflect/internal/StdNames.scala +++ b/src/reflect/scala/reflect/internal/StdNames.scala @@ -763,6 +763,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" diff --git a/src/reflect/scala/reflect/runtime/JavaUniverseForce.scala b/src/reflect/scala/reflect/runtime/JavaUniverseForce.scala index 09b4f26a72cf..016d4752f1d4 100644 --- a/src/reflect/scala/reflect/runtime/JavaUniverseForce.scala +++ b/src/reflect/scala/reflect/runtime/JavaUniverseForce.scala @@ -51,6 +51,7 @@ trait JavaUniverseForce { self: runtime.JavaUniverse => this.UseInvokeSpecial this.TypeParamVarargsAttachment this.KnownDirectSubclassesCalled + this.ConstructorNeedsFence this.noPrint this.typeDebug this.Range diff --git a/test/files/run/lazy-locals-2.scala b/test/files/run/lazy-locals-2.scala index d6c33cffcb85..526376fb4d91 100644 --- a/test/files/run/lazy-locals-2.scala +++ b/test/files/run/lazy-locals-2.scala @@ -281,8 +281,8 @@ object Test { lzyComputeMethods == expComputeMethods, s"wrong lzycompute methods. expected:\n$expComputeMethods\nfound:\n$lzyComputeMethods") - val fields = c.getClass.getDeclaredFields.toList.sortBy(_.getName).map(_.toString) - val expFields = List( + val fields: List[String] = c.getClass.getDeclaredFields.toList.sortBy(_.getName).map(_.toString) + val expFields = List[String]( "private volatile byte C.bitmap$0", "private int C.lvl1", "private java.lang.String C.lvl2", @@ -305,9 +305,22 @@ object Test { d.run() val dFields = d.getClass.getDeclaredFields.toList.sortBy(_.getName).map(_.toString) - assert( - dFields == expFields.map(_.replaceAll(" C.", " D.")), - s"wrong fields. expected:\n$expFields\nfound:\n$fields") + val expDFields = List[String]( + "private volatile byte D.bitmap$0", + "private int D.lvl1", + "private java.lang.String D.lvl2", + "private scala.runtime.BoxedUnit D.lvl3", + "private int D.t1", + "private java.lang.String D.t2", + "private scala.runtime.BoxedUnit D.t3", + "private int D.vl1", + "private java.lang.String D.vl2", + "private scala.runtime.BoxedUnit D.vl3", + "private int D.vr1", + "private java.lang.String D.vr2", + "private scala.runtime.BoxedUnit D.vr3") + assert(dFields == expDFields, + s"wrong fields. expected:\n$expDFields\nfound:\n$dFields") val d1 = new D1 diff --git a/test/files/run/t10075b.check b/test/files/run/t10075b.check index dc64e95ac7aa..c1801c3c7761 100644 --- a/test/files/run/t10075b.check +++ b/test/files/run/t10075b.check @@ -45,9 +45,9 @@ @RetainedAnnotation() public int TMix.lzyValGetterAnnotation() private int TMix.lzyValGetterAnnotation$lzycompute() @RetainedAnnotation() public int TMix.method() -@RetainedAnnotation() private final int TMix.valFieldAnnotation +@RetainedAnnotation() private int TMix.valFieldAnnotation public int TMix.valFieldAnnotation() - private final int TMix.valGetterAnnotation + private int TMix.valGetterAnnotation @RetainedAnnotation() public int TMix.valGetterAnnotation() @RetainedAnnotation() private int TMix.varFieldAnnotation public int TMix.varFieldAnnotation()