Index: test/mjsunit/es6/tail-call.js |
diff --git a/test/mjsunit/es6/tail-call.js b/test/mjsunit/es6/tail-call.js |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a728a2ce41cb010b6040e1d89be6287d5e6dd4fd |
--- /dev/null |
+++ b/test/mjsunit/es6/tail-call.js |
@@ -0,0 +1,177 @@ |
+// 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()); |
+})(); |