6131ab1edd
This CL implements PrepareForTailCall() mentioned in ES6 spec for full codegen, Crankshaft and Turbofan. When debugger is active tail calls are disabled. Tail calling can be enabled by --harmony-tailcalls flag. BUG=v8:4698 LOG=Y TBR=rossberg@chromium.org Review URL: https://codereview.chromium.org/1609893003 Cr-Commit-Position: refs/heads/master@{#33509}
178 lines
4.4 KiB
JavaScript
178 lines
4.4 KiB
JavaScript
// Copyright 2016 the V8 project authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
// Flags: --allow-natives-syntax --harmony-tailcalls
|
|
"use strict";
|
|
|
|
Error.prepareStackTrace = (e,s) => s;
|
|
|
|
function CheckStackTrace(expected) {
|
|
var stack = (new Error()).stack;
|
|
assertEquals("CheckStackTrace", stack[0].getFunctionName());
|
|
for (var i = 0; i < expected.length; i++) {
|
|
assertEquals(expected[i].name, stack[i + 1].getFunctionName());
|
|
}
|
|
}
|
|
|
|
|
|
// Tail call when caller does not have an arguments adaptor frame.
|
|
(function test() {
|
|
// Caller and callee have same number of arguments.
|
|
function f1(a) {
|
|
CheckStackTrace([f1, test]);
|
|
return 10 + a;
|
|
}
|
|
function g1(a) { return f1(2); }
|
|
assertEquals(12, g1(1));
|
|
|
|
// Caller has more arguments than callee.
|
|
function f2(a) {
|
|
CheckStackTrace([f2, test]);
|
|
return 10 + a;
|
|
}
|
|
function g2(a, b, c) { return f2(2); }
|
|
assertEquals(12, g2(1, 2, 3));
|
|
|
|
// Caller has less arguments than callee.
|
|
function f3(a, b, c) {
|
|
CheckStackTrace([f3, test]);
|
|
return 10 + a + b + c;
|
|
}
|
|
function g3(a) { return f3(2, 3, 4); }
|
|
assertEquals(19, g3(1));
|
|
|
|
// Callee has arguments adaptor frame.
|
|
function f4(a, b, c) {
|
|
CheckStackTrace([f4, test]);
|
|
return 10 + a;
|
|
}
|
|
function g4(a) { return f4(2); }
|
|
assertEquals(12, g4(1));
|
|
})();
|
|
|
|
|
|
// Tail call when caller has an arguments adaptor frame.
|
|
(function test() {
|
|
// Caller and callee have same number of arguments.
|
|
function f1(a) {
|
|
CheckStackTrace([f1, test]);
|
|
return 10 + a;
|
|
}
|
|
function g1(a) { return f1(2); }
|
|
assertEquals(12, g1());
|
|
|
|
// Caller has more arguments than callee.
|
|
function f2(a) {
|
|
CheckStackTrace([f2, test]);
|
|
return 10 + a;
|
|
}
|
|
function g2(a, b, c) { return f2(2); }
|
|
assertEquals(12, g2());
|
|
|
|
// Caller has less arguments than callee.
|
|
function f3(a, b, c) {
|
|
CheckStackTrace([f3, test]);
|
|
return 10 + a + b + c;
|
|
}
|
|
function g3(a) { return f3(2, 3, 4); }
|
|
assertEquals(19, g3());
|
|
|
|
// Callee has arguments adaptor frame.
|
|
function f4(a, b, c) {
|
|
CheckStackTrace([f4, test]);
|
|
return 10 + a;
|
|
}
|
|
function g4(a) { return f4(2); }
|
|
assertEquals(12, g4());
|
|
})();
|
|
|
|
|
|
// Tail call bound function when caller does not have an arguments
|
|
// adaptor frame.
|
|
(function test() {
|
|
// Caller and callee have same number of arguments.
|
|
function f1(a) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f1, test]);
|
|
return 10 + a;
|
|
}
|
|
var b1 = f1.bind({a: 153});
|
|
function g1(a) { return b1(2); }
|
|
assertEquals(12, g1(1));
|
|
|
|
// Caller has more arguments than callee.
|
|
function f2(a) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f2, test]);
|
|
return 10 + a;
|
|
}
|
|
var b2 = f2.bind({a: 153});
|
|
function g2(a, b, c) { return b2(2); }
|
|
assertEquals(12, g2(1, 2, 3));
|
|
|
|
// Caller has less arguments than callee.
|
|
function f3(a, b, c) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f3, test]);
|
|
return 10 + a + b + c;
|
|
}
|
|
var b3 = f3.bind({a: 153});
|
|
function g3(a) { return b3(2, 3, 4); }
|
|
assertEquals(19, g3(1));
|
|
|
|
// Callee has arguments adaptor frame.
|
|
function f4(a, b, c) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f4, test]);
|
|
return 10 + a;
|
|
}
|
|
var b4 = f4.bind({a: 153});
|
|
function g4(a) { return b4(2); }
|
|
assertEquals(12, g4(1));
|
|
})();
|
|
|
|
|
|
// Tail call bound function when caller has an arguments adaptor frame.
|
|
(function test() {
|
|
// Caller and callee have same number of arguments.
|
|
function f1(a) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f1, test]);
|
|
return 10 + a;
|
|
}
|
|
var b1 = f1.bind({a: 153});
|
|
function g1(a) { return b1(2); }
|
|
assertEquals(12, g1());
|
|
|
|
// Caller has more arguments than callee.
|
|
function f2(a) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f2, test]);
|
|
return 10 + a;
|
|
}
|
|
var b2 = f2.bind({a: 153});
|
|
function g2(a, b, c) { return b2(2); }
|
|
assertEquals(12, g2());
|
|
|
|
// Caller has less arguments than callee.
|
|
function f3(a, b, c) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f3, test]);
|
|
return 10 + a + b + c;
|
|
}
|
|
var b3 = f3.bind({a: 153});
|
|
function g3(a) { return b3(2, 3, 4); }
|
|
assertEquals(19, g3());
|
|
|
|
// Callee has arguments adaptor frame.
|
|
function f4(a, b, c) {
|
|
assertEquals(153, this.a);
|
|
CheckStackTrace([f4, test]);
|
|
return 10 + a;
|
|
}
|
|
var b4 = f4.bind({a: 153});
|
|
function g4(a) { return b4(2); }
|
|
assertEquals(12, g4());
|
|
})();
|