-
Notifications
You must be signed in to change notification settings - Fork 1.1k
/
Copy pathAnnotations.scala
311 lines (257 loc) · 12 KB
/
Annotations.scala
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
package dotty.tools
package dotc
package core
import Symbols.*, Types.*, Contexts.*, Constants.*, Phases.*
import ast.tpd, tpd.*
import util.Spans.Span
import printing.{Showable, Printer}
import printing.Texts.Text
import scala.annotation.internal.sharable
object Annotations {
def annotClass(tree: Tree)(using Context) =
if (tree.symbol.isConstructor) tree.symbol.owner
else tree.tpe.typeSymbol
abstract class Annotation extends Showable {
def tree(using Context): Tree
def symbol(using Context): Symbol = annotClass(tree)
def hasSymbol(sym: Symbol)(using Context) = symbol == sym
def matches(cls: Symbol)(using Context): Boolean = symbol.derivesFrom(cls)
def appliesToModule: Boolean = true // for now; see remark in SymDenotations
def derivedAnnotation(tree: Tree)(using Context): Annotation =
if (tree eq this.tree) this else Annotation(tree)
/** All term arguments of this annotation in a single flat list */
def arguments(using Context): List[Tree] = tpd.allTermArguments(tree)
def argument(i: Int)(using Context): Option[Tree] = {
val args = arguments
if (i < args.length) Some(args(i)) else None
}
def argumentConstant(i: Int)(using Context): Option[Constant] =
for (case ConstantType(c) <- argument(i) map (_.tpe.widenTermRefExpr.normalized)) yield c
def argumentConstantString(i: Int)(using Context): Option[String] =
for (case Constant(s: String) <- argumentConstant(i)) yield s
/** All type and term argument trees of this annotation in a single flat list */
private def allArguments(using Context): List[Tree] = tpd.allArguments(tree)
/** The tree evaluation is in progress. */
def isEvaluating: Boolean = false
/** The tree evaluation has finished. */
def isEvaluated: Boolean = true
/** Normally, applies a type map to all tree nodes of this annotation, but can
* be overridden. Returns EmptyAnnotation if type type map produces a range
* type, since ranges cannot be types of trees.
*/
def mapWith(tm: TypeMap)(using Context) =
val args = tpd.allArguments(tree)
if args.isEmpty then this
else
// Checks if `tm` would result in any change by applying it to types
// inside the annotations' arguments and checking if the resulting types
// are different.
val findDiff = new TreeAccumulator[Type]:
def apply(x: Type, tree: Tree)(using Context): Type =
if tm.isRange(x) then x
else
val tp1 = tm(tree.tpe)
foldOver(if !tp1.exists || tp1.eql(tree.tpe) then x else tp1, tree)
val diff = findDiff(NoType, args)
if tm.isRange(diff) then EmptyAnnotation
else if diff.exists then derivedAnnotation(tm.mapOver(tree))
else this
/** Does this annotation refer to a parameter of `tl`? */
def refersToParamOf(tl: TermLambda)(using Context): Boolean =
val args = tpd.allArguments(tree)
if args.isEmpty then false
else tree.existsSubTree:
case id: (Ident | This) => id.tpe.stripped match
case TermParamRef(tl1, _) => tl eq tl1
case _ => false
case _ => false
/** A string representation of the annotation. Overridden in BodyAnnotation.
*/
def toText(printer: Printer): Text = printer.annotText(this)
def ensureCompleted(using Context): Unit = tree
def sameAnnotation(that: Annotation)(using Context): Boolean =
def sameArg(arg1: Tree, arg2: Tree): Boolean = tpd.stripNamedArg(arg1).sameTree(tpd.stripNamedArg(arg2))
symbol == that.symbol && allArguments.corresponds(that.allArguments)(sameArg)
def hasOneOfMetaAnnotation(metaSyms: Set[Symbol], orNoneOf: Set[Symbol] = Set.empty)(using Context): Boolean = atPhaseNoLater(erasurePhase) {
def go(metaSyms: Set[Symbol]) =
def recTp(tp: Type): Boolean = tp.dealiasKeepAnnots match
case AnnotatedType(parent, metaAnnot) => metaSyms.exists(metaAnnot.matches) || recTp(parent)
case _ => false
def rec(tree: Tree): Boolean = methPart(tree) match
case New(tpt) => rec(tpt)
case Select(qual, _) => rec(qual)
case Annotated(arg, metaAnnot) => metaSyms.exists(metaAnnot.tpe.classSymbol.derivesFrom) || rec(arg)
case t @ Ident(_) => recTp(t.tpe)
case Typed(expr, _) => rec(expr)
case _ => false
metaSyms.exists(symbol.hasAnnotation) || rec(tree)
go(metaSyms) || orNoneOf.nonEmpty && !go(orNoneOf)
}
/** Operations for hash-consing, can be overridden */
def hash: Int = System.identityHashCode(this)
def eql(that: Annotation) = this eq that
}
case class ConcreteAnnotation(t: Tree) extends Annotation:
def tree(using Context): Tree = t
abstract class LazyAnnotation extends Annotation {
protected var mySym: Symbol | (Context ?=> Symbol) | Null
override def symbol(using parentCtx: Context): Symbol =
assert(mySym != null)
mySym match {
case symFn: (Context ?=> Symbol) @unchecked =>
mySym = null
mySym = atPhaseBeforeTransforms(symFn)
// We should always produce the same annotation tree, no matter when the
// annotation is evaluated. Setting the phase to a pre-transformation phase
// seems to be enough to ensure this (note that after erasure, `ctx.typer`
// will be the Erasure typer, but that doesn't seem to affect the annotation
// trees we create, so we leave it as is)
case sym: Symbol if sym.defRunId != parentCtx.runId =>
mySym = sym.denot.current.symbol
case _ =>
}
mySym.asInstanceOf[Symbol]
protected var myTree: Tree | (Context ?=> Tree) | Null
def tree(using Context): Tree =
assert(myTree != null)
myTree match {
case treeFn: (Context ?=> Tree) @unchecked =>
myTree = null
myTree = atPhaseBeforeTransforms(treeFn)
case _ =>
}
myTree.asInstanceOf[Tree]
override def isEvaluating: Boolean = myTree == null
override def isEvaluated: Boolean = myTree.isInstanceOf[Tree @unchecked]
}
class DeferredSymAndTree(symFn: Context ?=> Symbol, treeFn: Context ?=> Tree)
extends LazyAnnotation:
protected var mySym: Symbol | (Context ?=> Symbol) | Null = ctx ?=> symFn(using ctx)
protected var myTree: Tree | (Context ?=> Tree) | Null = ctx ?=> treeFn(using ctx)
/** An annotation indicating the body of a right-hand side,
* typically of an inline method. Treated specially in
* pickling/unpickling and TypeTreeMaps
*/
abstract class BodyAnnotation extends Annotation {
override def symbol(using Context): ClassSymbol = defn.BodyAnnot
override def derivedAnnotation(tree: Tree)(using Context): Annotation =
if (tree eq this.tree) this else ConcreteBodyAnnotation(tree)
override def arguments(using Context): List[Tree] = Nil
override def ensureCompleted(using Context): Unit = ()
override def toText(printer: Printer): Text = "@Body"
}
class ConcreteBodyAnnotation(body: Tree) extends BodyAnnotation {
def tree(using Context): Tree = body
}
abstract class LazyBodyAnnotation extends BodyAnnotation {
// Copy-pasted from LazyAnnotation to avoid having to turn it into a trait
protected var myTree: Tree | (Context ?=> Tree) | Null
def tree(using Context): Tree =
assert(myTree != null)
myTree match {
case treeFn: (Context ?=> Tree) @unchecked =>
myTree = null
myTree = atPhaseBeforeTransforms(treeFn)
case _ =>
}
myTree.asInstanceOf[Tree]
override def isEvaluating: Boolean = myTree == null
override def isEvaluated: Boolean = myTree.isInstanceOf[Tree @unchecked]
}
object LazyBodyAnnotation {
def apply(bodyFn: Context ?=> Tree): LazyBodyAnnotation =
new LazyBodyAnnotation:
protected var myTree: Tree | (Context ?=> Tree) | Null = ctx ?=> bodyFn(using ctx)
}
object Annotation {
def apply(tree: Tree): ConcreteAnnotation = ConcreteAnnotation(tree)
def apply(cls: ClassSymbol, span: Span)(using Context): Annotation =
apply(cls, Nil, span)
def apply(cls: ClassSymbol, arg: Tree, span: Span)(using Context): Annotation =
apply(cls, arg :: Nil, span)
def apply(cls: ClassSymbol, args: List[Tree], span: Span)(using Context): Annotation =
apply(cls.typeRef, args, span)
def apply(atp: Type, arg: Tree, span: Span)(using Context): Annotation =
apply(atp, arg :: Nil, span)
def apply(atp: Type, args: List[Tree], span: Span)(using Context): Annotation =
apply(New(atp, args).withSpan(span))
/** Create an annotation where the tree is computed lazily. */
def deferred(sym: Symbol)(treeFn: Context ?=> Tree): Annotation =
new LazyAnnotation {
protected var myTree: Tree | (Context ?=> Tree) | Null = ctx ?=> treeFn(using ctx)
protected var mySym: Symbol | (Context ?=> Symbol) | Null = sym
}
/** Create an annotation where the symbol and the tree are computed lazily. */
def deferredSymAndTree(symFn: Context ?=> Symbol)(treeFn: Context ?=> Tree): Annotation =
DeferredSymAndTree(symFn, treeFn)
/** Extractor for child annotations */
object Child {
/** A deferred annotation to the result of a given child computation */
def later(delayedSym: Context ?=> Symbol, span: Span)(using Context): Annotation = {
def makeChildLater(using Context) = {
val sym = delayedSym
New(defn.ChildAnnot.typeRef.appliedTo(sym.owner.thisType.select(sym.name, sym)), Nil)
.withSpan(span)
}
deferred(defn.ChildAnnot)(makeChildLater)
}
/** A regular, non-deferred Child annotation */
def apply(sym: Symbol, span: Span)(using Context): Annotation = later(sym, span)
def unapply(ann: Annotation)(using Context): Option[Symbol] =
if (ann.symbol == defn.ChildAnnot) {
val AppliedType(_, (arg: NamedType) :: Nil) = ann.tree.tpe: @unchecked
Some(arg.symbol)
}
else None
}
}
@sharable val EmptyAnnotation = Annotation(EmptyTree)
def ThrowsAnnotation(cls: ClassSymbol)(using Context): Annotation = {
val tref = cls.typeRef
Annotation(defn.ThrowsAnnot.typeRef.appliedTo(tref), Ident(tref), cls.span)
}
/** Extracts the type of the thrown exception from an annotation.
*
* Supports both "old-style" `@throws(classOf[Exception])`
* as well as "new-style" `@throws[Exception]("cause")` annotations.
*/
object ThrownException {
def unapply(a: Annotation)(using Context): Option[Type] =
if (a.symbol ne defn.ThrowsAnnot)
None
else a.argumentConstant(0) match {
// old-style: @throws(classOf[Exception]) (which is throws[T](classOf[Exception]))
case Some(Constant(tpe: Type)) =>
Some(tpe)
// new-style: @throws[Exception], @throws[Exception]("cause")
case _ =>
stripApply(a.tree) match {
case TypeApply(_, List(tpt)) =>
Some(tpt.tpe)
case _ =>
None
}
}
}
object ExperimentalAnnotation {
/** Create an instance of `@experimental(<msg>)` */
def apply(msg: String, span: Span)(using Context): Annotation =
Annotation(defn.ExperimentalAnnot, Literal(Constant(msg)), span)
/** Matches and extracts the message from an instance of `@experimental(msg)`
* Returns `Some("")` for `@experimental` with no message.
*/
def unapply(a: Annotation)(using Context): Option[String] =
if a.symbol ne defn.ExperimentalAnnot then
None
else a.argumentConstant(0) match
case Some(Constant(msg: String)) => Some(msg)
case _ => Some("")
/** Makes a copy of the `@experimental(msg)` annotation on `sym`
* None is returned if the symbol does not have an `@experimental` annotation.
*/
def copy(sym: Symbol)(using Context): Option[Annotation] =
sym.getAnnotation(defn.ExperimentalAnnot).map {
case annot @ ExperimentalAnnotation(msg) => ExperimentalAnnotation(msg, annot.tree.span)
}
}
}