Skip to content

Commit 7bfb027

Browse files
authored
Implement SIP-61 @unroll annotation (#21693)
The main implementation follows [com-lihaoyi/unroll](https://github.com/com-lihaoyi/unroll) but with some changes: - `@unroll` annotation is `@experimental` - run before pickling - increase validation checks for incorrect usage of `@unroll` - `Underscore` not `EmptyTree` in pattern match default case - ensure correct spans in TASTy - ensure symbols are correctly substituted in types There is one main library addition: `scala.annotation.unroll`, i.e. the `@unroll` annotation that appears on parameters, fixes #21728
2 parents 3f5c570 + 2a04670 commit 7bfb027

File tree

186 files changed

+2919
-31
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

186 files changed

+2919
-31
lines changed

compiler/src/dotty/tools/dotc/CompilationUnit.scala

+3
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,9 @@ class CompilationUnit protected (val source: SourceFile, val info: CompilationUn
5959

6060
var hasMacroAnnotations: Boolean = false
6161

62+
def hasUnrollDefs: Boolean = unrolledClasses.nonEmpty
63+
var unrolledClasses: Set[Symbol] = Set.empty
64+
6265
/** Set to `true` if inliner added anonymous mirrors that need to be completed */
6366
var needsMirrorSupport: Boolean = false
6467

compiler/src/dotty/tools/dotc/Compiler.scala

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ class Compiler {
4040
List(new sbt.ExtractDependencies) :: // Sends information on classes' dependencies to sbt via callbacks
4141
List(new semanticdb.ExtractSemanticDB.ExtractSemanticInfo) :: // Extract info into .semanticdb files
4242
List(new PostTyper) :: // Additional checks and cleanups after type checking
43+
List(new UnrollDefinitions) :: // Unroll annotated methods if detected in PostTyper
4344
List(new sjs.PrepJSInterop) :: // Additional checks and transformations for Scala.js (Scala.js only)
4445
List(new SetRootTree) :: // Set the `rootTreeOrProvider` on class symbols
4546
Nil

compiler/src/dotty/tools/dotc/core/Definitions.scala

+1
Original file line numberDiff line numberDiff line change
@@ -1036,6 +1036,7 @@ class Definitions {
10361036
@tu lazy val MigrationAnnot: ClassSymbol = requiredClass("scala.annotation.migration")
10371037
@tu lazy val NowarnAnnot: ClassSymbol = requiredClass("scala.annotation.nowarn")
10381038
@tu lazy val UnusedAnnot: ClassSymbol = requiredClass("scala.annotation.unused")
1039+
@tu lazy val UnrollAnnot: ClassSymbol = requiredClass("scala.annotation.unroll")
10391040
@tu lazy val TransparentTraitAnnot: ClassSymbol = requiredClass("scala.annotation.transparentTrait")
10401041
@tu lazy val NativeAnnot: ClassSymbol = requiredClass("scala.native")
10411042
@tu lazy val RepeatedAnnot: ClassSymbol = requiredClass("scala.annotation.internal.Repeated")

compiler/src/dotty/tools/dotc/core/Denotations.scala

+3-1
Original file line numberDiff line numberDiff line change
@@ -1071,7 +1071,9 @@ object Denotations {
10711071
def filterDisjoint(denots: PreDenotation)(using Context): SingleDenotation =
10721072
if (denots.exists && denots.matches(this)) NoDenotation else this
10731073
def filterWithFlags(required: FlagSet, excluded: FlagSet)(using Context): SingleDenotation =
1074-
val realExcluded = if ctx.isAfterTyper then excluded else excluded | Invisible
1074+
val realExcluded =
1075+
if ctx.isAfterTyper || ctx.mode.is(Mode.ResolveFromTASTy) then excluded
1076+
else excluded | Invisible
10751077
def symd: SymDenotation = this match
10761078
case symd: SymDenotation => symd
10771079
case _ => symbol.denot

compiler/src/dotty/tools/dotc/core/Mode.scala

+3
Original file line numberDiff line numberDiff line change
@@ -125,6 +125,9 @@ object Mode {
125125
/** Read original positions when unpickling from TASTY */
126126
val ReadPositions: Mode = newMode(17, "ReadPositions")
127127

128+
/** We are resolving a SELECT name from TASTy */
129+
val ResolveFromTASTy: Mode = newMode(18, "ResolveFromTASTy")
130+
128131
/** We are elaborating the fully qualified name of a package clause.
129132
* In this case, identifiers should never be imported.
130133
*/

compiler/src/dotty/tools/dotc/core/SymDenotations.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -617,7 +617,7 @@ object SymDenotations {
617617
case _ =>
618618
// Otherwise, no completion is necessary, see the preconditions of `markAbsent()`.
619619
(myInfo `eq` NoType)
620-
|| is(Invisible) && ctx.isTyper
620+
|| (is(Invisible) && !ctx.mode.is(Mode.ResolveFromTASTy)) && ctx.isTyper
621621
|| is(ModuleVal, butNot = Package) && moduleClass.isAbsent(canForce)
622622
}
623623

compiler/src/dotty/tools/dotc/core/tasty/TreeUnpickler.scala

+6-2
Original file line numberDiff line numberDiff line change
@@ -1563,7 +1563,7 @@ class TreeUnpickler(reader: TastyReader,
15631563
* - sbt-test/tasty-compat/remove-override
15641564
* - sbt-test/tasty-compat/move-method
15651565
*/
1566-
def lookupInSuper =
1566+
def lookupInSuper(using Context) =
15671567
val cls = ownerTpe.classSymbol
15681568
if cls.exists then
15691569
cls.asClass.classDenot
@@ -1572,14 +1572,18 @@ class TreeUnpickler(reader: TastyReader,
15721572
else
15731573
NoDenotation
15741574

1575-
val denot =
1575+
1576+
def searchDenot(using Context): Denotation =
15761577
if owner.is(JavaAnnotation) && name == nme.CONSTRUCTOR then
15771578
// #19951 Fix up to read TASTy produced before 3.5.0 -- ignore the signature
15781579
ownerTpe.nonPrivateDecl(name).asSeenFrom(prefix)
15791580
else
15801581
val d = ownerTpe.decl(name).atSignature(sig, target)
15811582
(if !d.exists then lookupInSuper else d).asSeenFrom(prefix)
15821583

1584+
val denot = inContext(ctx.addMode(Mode.ResolveFromTASTy)):
1585+
searchDenot // able to resolve Invisible members
1586+
15831587
makeSelect(qual, name, denot)
15841588
case REPEATED =>
15851589
val elemtpt = readTpt()

compiler/src/dotty/tools/dotc/reporting/ErrorMessageID.scala

+1
Original file line numberDiff line numberDiff line change
@@ -220,6 +220,7 @@ enum ErrorMessageID(val isActive: Boolean = true) extends java.lang.Enum[ErrorMe
220220
case DeprecatedInfixNamedArgumentSyntaxID // errorNumber: 204
221221
case GivenSearchPriorityID // errorNumber: 205
222222
case EnumMayNotBeValueClassesID // errorNumber: 206
223+
case IllegalUnrollPlacementID // errorNumber: 207
223224

224225
def errorNumber = ordinal - 1
225226

compiler/src/dotty/tools/dotc/reporting/messages.scala

+27-4
Original file line numberDiff line numberDiff line change
@@ -3334,14 +3334,14 @@ final class QuotedTypeMissing(tpe: Type)(using Context) extends StagingMessage(Q
33343334

33353335
private def witness = defn.QuotedTypeClass.typeRef.appliedTo(tpe)
33363336

3337-
override protected def msg(using Context): String =
3337+
override protected def msg(using Context): String =
33383338
i"Reference to $tpe within quotes requires a given ${witness} in scope"
33393339

33403340
override protected def explain(using Context): String =
3341-
i"""Referencing `$tpe` inside a quoted expression requires a `${witness}` to be in scope.
3341+
i"""Referencing `$tpe` inside a quoted expression requires a `${witness}` to be in scope.
33423342
|Since Scala is subject to erasure at runtime, the type information will be missing during the execution of the code.
3343-
|`${witness}` is therefore needed to carry `$tpe`'s type information into the quoted code.
3344-
|Without an implicit `${witness}`, the type `$tpe` cannot be properly referenced within the expression.
3343+
|`${witness}` is therefore needed to carry `$tpe`'s type information into the quoted code.
3344+
|Without an implicit `${witness}`, the type `$tpe` cannot be properly referenced within the expression.
33453345
|To resolve this, ensure that a `${witness}` is available, either through a context-bound or explicitly.
33463346
|"""
33473347

@@ -3408,3 +3408,26 @@ final class EnumMayNotBeValueClasses(sym: Symbol)(using Context) extends SyntaxM
34083408

34093409
def explain(using Context) = ""
34103410
end EnumMayNotBeValueClasses
3411+
3412+
class IllegalUnrollPlacement(origin: Option[Symbol])(using Context)
3413+
extends DeclarationMsg(IllegalUnrollPlacementID):
3414+
def msg(using Context) = origin match
3415+
case None => "@unroll is only allowed on a method parameter"
3416+
case Some(method) =>
3417+
val isCtor = method.isConstructor
3418+
def what = if isCtor then i"a ${if method.owner.is(Trait) then "trait" else "class"} constructor" else i"method ${method.name}"
3419+
val prefix = s"Cannot unroll parameters of $what"
3420+
if method.is(Deferred) then
3421+
i"$prefix: it must not be abstract"
3422+
else if isCtor && method.owner.is(Trait) then
3423+
i"implementation restriction: $prefix"
3424+
else if !(isCtor || method.is(Final) || method.owner.is(ModuleClass)) then
3425+
i"$prefix: it is not final"
3426+
else if method.owner.companionClass.is(CaseClass) then
3427+
i"$prefix of a case class companion object: please annotate the class constructor instead"
3428+
else
3429+
assert(method.owner.is(CaseClass))
3430+
i"$prefix of a case class: please annotate the class constructor instead"
3431+
3432+
def explain(using Context) = ""
3433+
end IllegalUnrollPlacement

compiler/src/dotty/tools/dotc/transform/PostTyper.scala

+31
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import reporting.*
2121
import NameKinds.WildcardParamName
2222
import cc.*
2323
import dotty.tools.dotc.transform.MacroAnnotations.hasMacroAnnotation
24+
import dotty.tools.dotc.core.NameKinds.DefaultGetterName
2425

2526
object PostTyper {
2627
val name: String = "posttyper"
@@ -119,8 +120,31 @@ class PostTyper extends MacroTransform with InfoTransformer { thisPhase =>
119120

120121
private var inJavaAnnot: Boolean = false
121122

123+
private val seenUnrolledMethods: util.EqHashMap[Symbol, Boolean] = new util.EqHashMap[Symbol, Boolean]
124+
122125
private var noCheckNews: Set[New] = Set()
123126

127+
def isValidUnrolledMethod(method: Symbol, origin: SrcPos)(using Context): Boolean =
128+
seenUnrolledMethods.getOrElseUpdate(method, {
129+
val isCtor = method.isConstructor
130+
if
131+
method.name.is(DefaultGetterName)
132+
then
133+
false // not an error, but not an expandable unrolled method
134+
else if
135+
method.is(Deferred)
136+
|| isCtor && method.owner.is(Trait)
137+
|| !(isCtor || method.is(Final) || method.owner.is(ModuleClass))
138+
|| method.owner.companionClass.is(CaseClass)
139+
&& (method.name == nme.apply || method.name == nme.fromProduct)
140+
|| method.owner.is(CaseClass) && method.name == nme.copy
141+
then
142+
report.error(IllegalUnrollPlacement(Some(method)), origin)
143+
false
144+
else
145+
true
146+
})
147+
124148
def withNoCheckNews[T](ts: List[New])(op: => T): T = {
125149
val saved = noCheckNews
126150
noCheckNews ++= ts
@@ -199,6 +223,12 @@ class PostTyper extends MacroTransform with InfoTransformer { thisPhase =>
199223
tree
200224
}
201225

226+
private def registerIfUnrolledParam(sym: Symbol)(using Context): Unit =
227+
if sym.hasAnnotation(defn.UnrollAnnot) && isValidUnrolledMethod(sym.owner, sym.sourcePos) then
228+
val cls = sym.enclosingClass
229+
val additions = Array(cls, cls.linkedClass).filter(_ != NoSymbol)
230+
ctx.compilationUnit.unrolledClasses ++= additions
231+
202232
private def processValOrDefDef(tree: Tree)(using Context): tree.type =
203233
val sym = tree.symbol
204234
tree match
@@ -215,6 +245,7 @@ class PostTyper extends MacroTransform with InfoTransformer { thisPhase =>
215245
++ sym.annotations)
216246
else
217247
if sym.is(Param) then
248+
registerIfUnrolledParam(sym)
218249
sym.keepAnnotationsCarrying(thisPhase, Set(defn.ParamMetaAnnot), orNoneOf = defn.NonBeanMetaAnnots)
219250
else if sym.is(ParamAccessor) then
220251
// @publicInBinary is not a meta-annotation and therefore not kept by `keepAnnotationsCarrying`

0 commit comments

Comments
 (0)