A babel plugin for performing tail call optimization in recursive functions.
Download the plugin from npm, using e.g. npm or yarn.
npm i babel-plugin-recursive-tail-calls
Add it to the plugins in babel.config.json
.
More information on using Babel here.
Tail call optimization was included in the ECMAScript 2015 language specification. However, some teams raised concerns about this feature, and therefore it is so far only supported by Safari.
In functional programming, recursion is the main way to simulate a loop. So for those wanting to take a functional approach to JavaScript, some kind of tail call optimization is needed to avoid exceeding the maximum stack size.
This plugin does not implement the feature as mentioned in the language specification. Rather, it aims to find all locations where a function calls itself in tail position, and rewrites those functions using a loop.
// before transpilation
function f(a) {
if (a <= 0) {
return a;
}
return f(a - 1);
}
// after transpilation
function f(a) {
let _continueRecursion = true;
_tailCallLoop: while (_continueRecursion) {
_continueRecursion = false;
if (a <= 0) {
return a;
}
[a] = [a - 1];
_continueRecursion = true;
continue _tailCallLoop;
}
}
This plugin aims to support all scenarios where a recursive tail call can
occur. Below are some examples.
See examples
directory for complete examples and their
transformations.
A tail call is a call that is part of a return statement, such that the call is the last thing evaluated before control is returned back to the calling function. This means that, in addition to being the only part of a return statement, a tail call can occur in the following places:
- last operand of logical and (
&&
) - last operand of logical or (
||
) - last operand of the nullish coalescing operator (
??
) - if or else-branch of the conditional opertor (
? :
)
When rewriting occurs in this cases, the evaluation of arguments is only done once.
In case of multiple recursive tail calls, all are rewritten. If function evaluation terminates without a return statement, the loop, to which the code is transpiled, is terminated accordingly.
The scope of the recursive call are checked to see if the function name has been shadowed by another function with the same. In this case optimization is not performed.
As array assignment, which is used in the transformation, exhibits much of the same properties as parameter assignment, all features of parameter assignment are supported.
- Spread syntax
- Rest parameters
- Default parameters
- Excessive arguments
- This is needed when the excessive argument is a function call with a side effect, in this case it must be evaluated.
Anonymous functions, e.g. arrow functions, are tricky in the context of
recursion since they are, well, anonymous, and identifying which identifier
corresponds to which function might not even be possible to do at compile time.
The only case where optimization with anonymous functions is done by this plugin
is when the function is directly assigned to a const
at creation and the
label of the const
declaration is used in the recursive call. This is
supported since it is a common pattern.
const functionName = () => {
// function body
return functionName();
};