Skip to content

Commit aec0618

Browse files
committed
SIP 61 - remove tasty hack by not "telescoping" forwarders
Now, all forwarders directly call the unrolled method. This means that there is no need to resolve an invisible method, so the "hack" using TERMREFdirect is no longer needed. This increases bytecode size but reduces stack depth. Also removed scala.annotation.internal.UnrolledForwarder, as it is only needed for the TASTy hack.
1 parent 4868397 commit aec0618

File tree

7 files changed

+81
-31
lines changed

7 files changed

+81
-31
lines changed

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

-1
Original file line numberDiff line numberDiff line change
@@ -1040,7 +1040,6 @@ class Definitions {
10401040
@tu lazy val NowarnAnnot: ClassSymbol = requiredClass("scala.annotation.nowarn")
10411041
@tu lazy val UnusedAnnot: ClassSymbol = requiredClass("scala.annotation.unused")
10421042
@tu lazy val UnrollAnnot: ClassSymbol = requiredClass("scala.annotation.unroll")
1043-
@tu lazy val UnrollForwarderAnnot: ClassSymbol = requiredClass("scala.annotation.internal.UnrollForwarder")
10441043
@tu lazy val TransparentTraitAnnot: ClassSymbol = requiredClass("scala.annotation.transparentTrait")
10451044
@tu lazy val NativeAnnot: ClassSymbol = requiredClass("scala.native")
10461045
@tu lazy val RepeatedAnnot: ClassSymbol = requiredClass("scala.annotation.internal.Repeated")

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

-3
Original file line numberDiff line numberDiff line change
@@ -490,9 +490,6 @@ class TreePickler(pickler: TastyPickler, attributes: Attributes) {
490490
writeByte(if name.isTypeName then SELECTtpt else SELECT)
491491
pickleNameAndSig(name, sig, ename)
492492
pickleTree(qual)
493-
else if sym.is(Invisible) && qual.isInstanceOf[This] && sym.hasAnnotation(defn.UnrollForwarderAnnot) then
494-
writeByte(TERMREFdirect)
495-
pickleSymRef(sym) // SIP-61 HACK: resolution from Signature filters out Invisible symbols
496493
else // select from owner
497494
writeByte(SELECTin)
498495
withLength {

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

+1-6
Original file line numberDiff line numberDiff line change
@@ -1259,12 +1259,7 @@ class TreeUnpickler(reader: TastyReader,
12591259
goto(start)
12601260
readType() match {
12611261
case path: TypeRef => TypeTree(path)
1262-
case path: TermRef =>
1263-
val sym = path.symbol
1264-
if sym.is(Invisible) && sym.hasAnnotation(defn.UnrollForwarderAnnot) then
1265-
This(sym.owner.asClass).select(sym)
1266-
else
1267-
ref(path)
1262+
case path: TermRef => ref(path)
12681263
case path: ThisType => untpd.This(untpd.EmptyTypeIdent).withType(path)
12691264
case path: ConstantType => Literal(path.value)
12701265
case path: ErrorType if isBestEffortTasty => TypeTree(path)

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

+7-11
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,7 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
116116
def generateSingleForwarder(defdef: DefDef,
117117
prevMethodType: Type,
118118
paramIndex: Int,
119+
paramCount: Int,
119120
nextParamIndex: Int,
120121
nextSymbol: Symbol,
121122
annotatedParamListIndex: Int,
@@ -125,18 +126,12 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
125126
val forwarderDefSymbol0 = Symbols.newSymbol(
126127
defdef.symbol.owner,
127128
defdef.name,
128-
(defdef.symbol.flags &~
129-
HasDefaultParams &~
130-
(if nextParamIndex == -1 then EmptyFlags else Deferred)) |
129+
defdef.symbol.flags &~ HasDefaultParams |
131130
Invisible | Synthetic,
132131
NoType, // fill in later
133132
coord = nextSymbol.span.shift(1) // shift by 1 to avoid "secondary constructor must call preceding" error
134133
).entered
135134

136-
// we need this such that when unpickling a TERMREFdirect, if we see this annotation,
137-
// we restore the tree to a Select
138-
forwarderDefSymbol0.addAnnotation(defn.UnrollForwarderAnnot)
139-
140135
val newParamSymMappings = extractParamSymss(copyParamSym(_, forwarderDefSymbol0))
141136
val (oldParams, newParams) = newParamSymMappings.flatten.unzip
142137

@@ -170,7 +165,7 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
170165
.map(_.size)
171166
.sum
172167

173-
val defaultCalls = Range(paramIndex, nextParamIndex).map(n =>
168+
val defaultCalls = Range(paramIndex, paramCount).map(n =>
174169

175170
def makeSelect(refTree: Tree, name: TermName): Tree =
176171
val sym = refTree.symbol
@@ -204,7 +199,8 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
204199
else Apply(lhs, newParams)
205200
)
206201

207-
val forwarderInner: Tree = This(defdef.symbol.owner.asClass).select(nextSymbol)
202+
val forwarderInner: Tree =
203+
This(defdef.symbol.owner.asClass).select(defdef.symbol)
208204

209205
val forwarderCallArgs =
210206
newParamSymLists.zipWithIndex.map{case (ps, i) =>
@@ -226,8 +222,7 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
226222
}
227223

228224
val forwarderDef =
229-
tpd.DefDef(forwarderDefSymbol,
230-
rhs = if nextParamIndex == -1 then EmptyTree else forwarderRhs())
225+
tpd.DefDef(forwarderDefSymbol, rhs = forwarderRhs())
231226

232227
forwarderDef.withSpan(nextSymbol.span.shift(1))
233228
}
@@ -297,6 +292,7 @@ class UnrollDefinitions extends MacroTransform, IdentityDenotTransformer {
297292
defdef,
298293
defdef.symbol.info,
299294
paramIndex,
295+
paramCount,
300296
nextParamIndex,
301297
nextSymbol,
302298
paramClauseIndex,

library/src/scala/annotation/internal/UnrollForwarder.scala

-9
This file was deleted.

tests/run-tasty-inspector/stdlibExperimentalDefinitions.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,7 @@ val experimentalDefinitionInLibrary = Set(
9494
"scala.Predef$.runtimeChecked", "scala.annotation.internal.RuntimeChecked",
9595

9696
// New feature: SIP 61 - @unroll annotation
97-
"scala.annotation.unroll", "scala.annotation.internal.UnrollForwarder"
97+
"scala.annotation.unroll"
9898
)
9999

100100

tests/run/unroll-multiple.scala

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
//> using options -experimental
2+
3+
import scala.annotation.unroll
4+
import scala.deriving.Mirror
5+
6+
class Unrolled {
7+
final def foo(
8+
s: String,
9+
@unroll y: Boolean = true,
10+
@unroll i: Int = 0,
11+
@unroll c: Char = '?'): String = s + y + i + c
12+
}
13+
14+
class Outer {
15+
class Unrolled {
16+
final def bar(
17+
s: String,
18+
@unroll y: Boolean = true,
19+
@unroll i: Int = 0,
20+
@unroll c: Char = '?'): String = s + y + i + c
21+
}
22+
23+
case class UnrolledCase(
24+
s: String,
25+
@unroll y: Boolean = true,
26+
@unroll i: Int = 0,
27+
@unroll c: Char = '?') {
28+
def baz: String = s + y + i + c
29+
}
30+
31+
class UnrolledSecondary {
32+
private var msg = ""
33+
34+
def qux: String = msg
35+
36+
def this(
37+
s: String,
38+
@unroll y: Boolean = true,
39+
@unroll i: Int = 0,
40+
@unroll c: Char = '?') = {
41+
this()
42+
msg = s + y + i + c
43+
}
44+
}
45+
}
46+
47+
@main def Test: Unit =
48+
assert(Unrolled().foo("foo") == "footrue0?")
49+
assert(Unrolled().foo("foo", false) == "foofalse0?")
50+
assert(Unrolled().foo("foo", false, 1) == "foofalse1?")
51+
assert(Unrolled().foo("foo", false, 1, '@') == "foofalse1@")
52+
val outer = new Outer()
53+
assert(new outer.Unrolled().bar("bar") == "bartrue0?")
54+
assert(new outer.Unrolled().bar("bar", false) == "barfalse0?")
55+
assert(new outer.Unrolled().bar("bar", false, 1) == "barfalse1?")
56+
assert(new outer.Unrolled().bar("bar", false, 1, '@') == "barfalse1@")
57+
assert(outer.UnrolledCase.apply("baz").baz == "baztrue0?")
58+
assert(outer.UnrolledCase.apply("baz", false).baz == "bazfalse0?")
59+
assert(outer.UnrolledCase.apply("baz", false, 1).baz == "bazfalse1?")
60+
assert(outer.UnrolledCase.apply("baz", false, 1, '@').baz == "bazfalse1@")
61+
assert((new outer.UnrolledCase("baz")).baz == "baztrue0?")
62+
assert((new outer.UnrolledCase("baz", false)).baz == "bazfalse0?")
63+
assert((new outer.UnrolledCase("baz", false, 1)).baz == "bazfalse1?")
64+
assert((new outer.UnrolledCase("baz", false, 1, '@')).baz == "bazfalse1@")
65+
assert(summon[Mirror.Of[outer.UnrolledCase]].fromProduct(Tuple("baz")).baz == "baztrue0?")
66+
assert(summon[Mirror.Of[outer.UnrolledCase]].fromProduct(("baz", false)).baz == "bazfalse0?")
67+
assert(summon[Mirror.Of[outer.UnrolledCase]].fromProduct(("baz", false, 1)).baz == "bazfalse1?")
68+
assert(summon[Mirror.Of[outer.UnrolledCase]].fromProduct(("baz", false, 1, '@')).baz == "bazfalse1@")
69+
assert(new outer.UnrolledSecondary("qux").qux == "quxtrue0?")
70+
assert(new outer.UnrolledSecondary("qux", false).qux == "quxfalse0?")
71+
assert(new outer.UnrolledSecondary("qux", false, 1).qux == "quxfalse1?")
72+
assert(new outer.UnrolledSecondary("qux", false, 1, '@').qux == "quxfalse1@")

0 commit comments

Comments
 (0)