A Tail Rec Transformer.
What it does:
Finds method calls in tail-position and replaces them with jumps. A call is in a tail-position if it is the last instruction to be executed in the body of a method. This includes being in tail-position of a return
from a Labeled
block which is itself in tail-position (which is critical for tail-recursive calls in the cases of a match
). To identify tail positions, we recurse over the trees that may contain calls in tail-position (trees that can't contain such calls are not transformed).
When a method contains at least one tail-recursive call, its rhs is wrapped in the following structure:
var localForParam1: T1 = param1
...
while (<empty>) {
tailResult[ResultType]: {
return {
// original rhs with tail recursive calls transformed (see below)
}
}
}
Self-recursive calls in tail-position are then replaced by (a) reassigning the local var
s substituting formal parameters and (b) a return
from the tailResult
labeled block, which has the net effect of looping back to the beginning of the method. If the receiver is modifed in a recursive call, an additional var
is used to replace this
.
As a complete example of the transformation, the classical fact
function, defined as:
def fact(n: Int, acc: Int): Int =
if (n == 0) acc
else fact(n - 1, acc * n)
is rewritten as:
def fact(n: Int, acc: Int): Int = {
var acc$tailLocal1: Int = acc
var n$tailLocal1: Int = n
while (<empty>) {
tailLabel1[Unit]: {
return {
if (n$tailLocal1 == 0)
acc$tailLocal1
else {
val n$tailLocal1$tmp1: Int = n$tailLocal1 - 1
val acc$tailLocal1$tmp1: Int = acc$tailLocal1 * n$tailLocal1
n$tailLocal1 = n$tailLocal1$tmp1
acc$tailLocal1 = acc$tailLocal1$tmp1
(return[tailLabel1] ()): Int
}
}
}
}
}
As the JVM provides no way to jump from a method to another one, non-recursive calls in tail-position are not optimized.
A method call is self-recursive if it calls the current method and the method is final (otherwise, it could be a call to an overridden method in a subclass). Recursive calls on a different instance are optimized.
This phase has been moved after erasure to allow the use of vars for the parameters combined with a WhileDo
. This is also beneficial to support polymorphic tail-recursive calls.
In scalac, if the method had type parameters, the call must contain the same parameters as type arguments. This is no longer the case in dotc thanks to being located after erasure. In scalac, this is named tailCall but it does only provide optimization for self recursive functions, that's why it's renamed to tailrec
Attributes
- Authors:
-
Erik Stenman, Iulian Dragos, ported and heavily modified for dotty by Dmitry Petrashko moved after erasure and adapted to emit
Labeled
blocks by Sébastien Doeraene - Companion
- object
- Graph
-
- Supertypes
Members list
Type members
Classlikes
Value members
Concrete methods
Attributes
- Definition Classes
A name given to the Phase
that can be used to debug the compiler. For instance, it is possible to print trees after a given phase using:
A name given to the Phase
that can be used to debug the compiler. For instance, it is possible to print trees after a given phase using:
$ ./bin/scalac -Xprint:<phaseNameHere> sourceFile.scala
Attributes
- Definition Classes
List of names of phases that should precede this phase
Attributes
- Definition Classes
Inherited methods
If set, implicit search is enabled
Do not run if compile progress has been cancelled
Can this transform change the base types of a type?
Can this transform create or delete non-private members?
Can this transform change the parents of a class?
Check what the phase achieves, to be called at any point after it is finished.
Check what the phase achieves, to be called at any point after it is finished.
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
True for all phases except NoPhase
Attributes
- Inherited from:
- Phase
The sequence position of this phase in the given context where 0 is reserved for NoPhase and the first real phase is at position 1. -1 if the phase is not installed in the context.
The sequence position of this phase in the given context where 0 is reserved for NoPhase and the first real phase is at position 1. -1 if the phase is not installed in the context.
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
Output should be checkable by TreeChecker
This property is queried when phases are first assembled. If it is false, the phase will be dropped from the set of phases to traverse.
This property is queried when phases are first assembled. If it is false, the phase will be dropped from the set of phases to traverse.
Attributes
- Inherited from:
- Phase
This property is queried before a phase is run. If it is false, the phase is skipped.
This property is queried before a phase is run. If it is false, the phase is skipped.
Attributes
- Inherited from:
- Phase
Is this phase the standard typerphase? True for TyperPhase, but not for other first phases (such as FromTasty or Parser). The predicate is tested in some places that perform checks and corrections. It's different from ctx.isAfterTyper (and cheaper to test).
Is this phase the standard typerphase? True for TyperPhase, but not for other first phases (such as FromTasty or Parser). The predicate is tested in some places that perform checks and corrections. It's different from ctx.isAfterTyper (and cheaper to test).
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
Cancellable region, if not cancelled, run the body in the context of the current compilation unit. Enrich crash messages.
Cancellable region, if not cancelled, run the body in the context of the current compilation unit. Enrich crash messages.
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
A hook that allows to transform the usual context passed to the function that prints a compilation unit after a phase
A hook that allows to transform the usual context passed to the function that prints a compilation unit after a phase
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
If set, use relaxed typing for all phases in group
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
List of names of phases that should have finished their processing of all compilation units before this phase starts
List of names of phases that should have finished their processing of all compilation units before this phase starts
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
Attributes
- Inherited from:
- Phase
Convert a compilation unit's tree to a string; can be overridden
Attributes
- Inherited from:
- MiniPhase
skip the phase for a Java compilation unit, may depend on -Yjava-tasty
skip the phase for a Java compilation unit, may depend on -Yjava-tasty
Attributes
- Inherited from:
- Phase
for purposes of progress tracking, overridden in TyperPhase
Transform tree using all transforms of current group (including this one)
Transform tree using all transforms of current group (including this one)
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Transform single node using all transforms following the current one in this group
Transform single node using all transforms following the current one in this group
Attributes
- Inherited from:
- MiniPhase
Transform tree using all transforms following the current one in this group
Transform tree using all transforms following the current one in this group
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- MiniPhase
Attributes
- Inherited from:
- Phase
Inherited fields
Attributes
- Inherited from:
- MiniPhase