From f4151fdbc0102e90ab63a9fd665e38bc160821d8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9=20Bargull?= <andre.bargull@gmail.com> Date: Fri, 8 Dec 2017 13:22:03 -0800 Subject: [PATCH] Add tests for TCO with eval and cross-realm cases --- .../call/tco-cross-realm-class-construct.js | 39 ++++++++++++++++ ...tco-cross-realm-class-derived-construct.js | 39 ++++++++++++++++ .../call/tco-cross-realm-fun-call.js | 38 +++++++++++++++ .../call/tco-cross-realm-fun-construct.js | 38 +++++++++++++++ .../call/tco-non-eval-function-dynamic.js | 44 ++++++++++++++++++ .../expressions/call/tco-non-eval-function.js | 44 ++++++++++++++++++ .../expressions/call/tco-non-eval-global.js | 43 +++++++++++++++++ .../expressions/call/tco-non-eval-with.js | 46 +++++++++++++++++++ 8 files changed, 331 insertions(+) create mode 100644 test/language/expressions/call/tco-cross-realm-class-construct.js create mode 100644 test/language/expressions/call/tco-cross-realm-class-derived-construct.js create mode 100644 test/language/expressions/call/tco-cross-realm-fun-call.js create mode 100644 test/language/expressions/call/tco-cross-realm-fun-construct.js create mode 100644 test/language/expressions/call/tco-non-eval-function-dynamic.js create mode 100644 test/language/expressions/call/tco-non-eval-function.js create mode 100644 test/language/expressions/call/tco-non-eval-global.js create mode 100644 test/language/expressions/call/tco-non-eval-with.js diff --git a/test/language/expressions/call/tco-cross-realm-class-construct.js b/test/language/expressions/call/tco-cross-realm-class-construct.js new file mode 100644 index 0000000000..8d74cee29a --- /dev/null +++ b/test/language/expressions/call/tco-cross-realm-class-construct.js @@ -0,0 +1,39 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Check TypeError is thrown from correct realm with tco-call to class constructor from derived + class [[Construct]] invocation. +description: | + 12.3.4.3 Runtime Semantics: EvaluateDirectCall( func, thisValue, arguments, tailPosition ) + ... + 4. If tailPosition is true, perform PrepareForTailCall(). + 5. Let result be Call(func, thisValue, argList). + 6. Assert: If tailPosition is true, the above call will not return here, but instead evaluation will continue as if the following return has already occurred. + 7. Assert: If result is not an abrupt completion, then Type(result) is an ECMAScript language type. + 8. Return result. + + 9.2.1 [[Call]] ( thisArgument, argumentsList) + ... + 2. If F.[[FunctionKind]] is "classConstructor", throw a TypeError exception. + 3. Let callerContext be the running execution context. + 4. Let calleeContext be PrepareForOrdinaryCall(F, undefined). + 5. Assert: calleeContext is now the running execution context. + ... + +features: [tail-call-optimization, class] +---*/ + +// - The class constructor call is in a valid tail-call position, which means PrepareForTailCall is performed. +// - The function call returns from `otherRealm` and proceeds the tail-call in this realm. +// - Calling the class constructor throws a TypeError from the current realm, that means this realm and not `otherRealm`. +var code = "(class { constructor() { return (class {})(); } });"; + +var otherRealm = $262.createRealm(); +var tco = otherRealm.evalScript(code); + +assert.throws(TypeError, function() { + new tco(); +}); diff --git a/test/language/expressions/call/tco-cross-realm-class-derived-construct.js b/test/language/expressions/call/tco-cross-realm-class-derived-construct.js new file mode 100644 index 0000000000..f55a8fe9b9 --- /dev/null +++ b/test/language/expressions/call/tco-cross-realm-class-derived-construct.js @@ -0,0 +1,39 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Check TypeError is thrown from correct realm with tco-call to class constructor from + class [[Construct]] invocation. +description: | + 12.3.4.3 Runtime Semantics: EvaluateDirectCall( func, thisValue, arguments, tailPosition ) + ... + 4. If tailPosition is true, perform PrepareForTailCall(). + 5. Let result be Call(func, thisValue, argList). + 6. Assert: If tailPosition is true, the above call will not return here, but instead evaluation will continue as if the following return has already occurred. + 7. Assert: If result is not an abrupt completion, then Type(result) is an ECMAScript language type. + 8. Return result. + + 9.2.1 [[Call]] ( thisArgument, argumentsList) + ... + 2. If F.[[FunctionKind]] is "classConstructor", throw a TypeError exception. + 3. Let callerContext be the running execution context. + 4. Let calleeContext be PrepareForOrdinaryCall(F, undefined). + 5. Assert: calleeContext is now the running execution context. + ... + +features: [tail-call-optimization, class] +---*/ + +// - The class constructor call is in a valid tail-call position, which means PrepareForTailCall is performed. +// - The function call returns from `otherRealm` and proceeds the tail-call in this realm. +// - Calling the class constructor throws a TypeError from the current realm, that means this realm and not `otherRealm`. +var code = "(class extends Object { constructor() { return (class {})(); } });"; + +var otherRealm = $262.createRealm(); +var tco = otherRealm.evalScript(code); + +assert.throws(TypeError, function() { + new tco(); +}); diff --git a/test/language/expressions/call/tco-cross-realm-fun-call.js b/test/language/expressions/call/tco-cross-realm-fun-call.js new file mode 100644 index 0000000000..6c0b3e30a4 --- /dev/null +++ b/test/language/expressions/call/tco-cross-realm-fun-call.js @@ -0,0 +1,38 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Check TypeError is thrown from correct realm with tco-call to class constructor from [[Call]] invocation. +description: | + 12.3.4.3 Runtime Semantics: EvaluateDirectCall( func, thisValue, arguments, tailPosition ) + ... + 4. If tailPosition is true, perform PrepareForTailCall(). + 5. Let result be Call(func, thisValue, argList). + 6. Assert: If tailPosition is true, the above call will not return here, but instead evaluation will continue as if the following return has already occurred. + 7. Assert: If result is not an abrupt completion, then Type(result) is an ECMAScript language type. + 8. Return result. + + 9.2.1 [[Call]] ( thisArgument, argumentsList) + ... + 2. If F.[[FunctionKind]] is "classConstructor", throw a TypeError exception. + 3. Let callerContext be the running execution context. + 4. Let calleeContext be PrepareForOrdinaryCall(F, undefined). + 5. Assert: calleeContext is now the running execution context. + ... + +features: [tail-call-optimization, class] +---*/ + +// - The class constructor call is in a valid tail-call position, which means PrepareForTailCall is performed. +// - The function call returns from `otherRealm` and proceeds the tail-call in this realm. +// - Calling the class constructor throws a TypeError from the current realm, that means this realm and not `otherRealm`. +var code = "'use strict'; (function() { return (class {})(); });"; + +var otherRealm = $262.createRealm(); +var tco = otherRealm.evalScript(code); + +assert.throws(TypeError, function() { + tco(); +}); diff --git a/test/language/expressions/call/tco-cross-realm-fun-construct.js b/test/language/expressions/call/tco-cross-realm-fun-construct.js new file mode 100644 index 0000000000..e26165ec86 --- /dev/null +++ b/test/language/expressions/call/tco-cross-realm-fun-construct.js @@ -0,0 +1,38 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Check TypeError is thrown from correct realm with tco-call to class constructor from [[Construct]] invocation. +description: | + 12.3.4.3 Runtime Semantics: EvaluateDirectCall( func, thisValue, arguments, tailPosition ) + ... + 4. If tailPosition is true, perform PrepareForTailCall(). + 5. Let result be Call(func, thisValue, argList). + 6. Assert: If tailPosition is true, the above call will not return here, but instead evaluation will continue as if the following return has already occurred. + 7. Assert: If result is not an abrupt completion, then Type(result) is an ECMAScript language type. + 8. Return result. + + 9.2.1 [[Call]] ( thisArgument, argumentsList) + ... + 2. If F.[[FunctionKind]] is "classConstructor", throw a TypeError exception. + 3. Let callerContext be the running execution context. + 4. Let calleeContext be PrepareForOrdinaryCall(F, undefined). + 5. Assert: calleeContext is now the running execution context. + ... + +features: [tail-call-optimization, class] +---*/ + +// - The class constructor call is in a valid tail-call position, which means PrepareForTailCall is performed. +// - The function call returns from `otherRealm` and proceeds the tail-call in this realm. +// - Calling the class constructor throws a TypeError from the current realm, that means this realm and not `otherRealm`. +var code = "'use strict'; (function() { return (class {})(); });"; + +var otherRealm = $262.createRealm(); +var tco = otherRealm.evalScript(code); + +assert.throws(TypeError, function() { + new tco(); +}); diff --git a/test/language/expressions/call/tco-non-eval-function-dynamic.js b/test/language/expressions/call/tco-non-eval-function-dynamic.js new file mode 100644 index 0000000000..a570c791cd --- /dev/null +++ b/test/language/expressions/call/tco-non-eval-function-dynamic.js @@ -0,0 +1,44 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Tail-call with identifier named "eval" in function environment, local "eval" binding dynamically added. +description: | + 12.3.4.1 Runtime Semantics: Evaluation + ... + 6. If Type(ref) is Reference and IsPropertyReference(ref) is false and + GetReferencedName(ref) is "eval", then + a. If SameValue(func, %eval%) is true, then + ... + ... + 9. Return ? EvaluateCall(func, ref, arguments, tailCall). + + 12.3.4.2 Runtime Semantics: EvaluateCall( func, ref, arguments, tailPosition ) + ... + 7. If tailPosition is true, perform PrepareForTailCall(). + 8. Let result be Call(func, thisValue, argList). + ... + +flags: [noStrict] +features: [tail-call-optimization] +includes: [tcoHelper.js] +---*/ + +var callCount = 0; + +(function() { + function f(n) { + "use strict"; + if (n === 0) { + callCount += 1 + return; + } + return eval(n - 1); + } + eval("var eval = f;"); + f($MAX_ITERATIONS); +})(); + +assert.sameValue(callCount, 1); diff --git a/test/language/expressions/call/tco-non-eval-function.js b/test/language/expressions/call/tco-non-eval-function.js new file mode 100644 index 0000000000..623a101273 --- /dev/null +++ b/test/language/expressions/call/tco-non-eval-function.js @@ -0,0 +1,44 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Tail-call with identifier named "eval" in function environment. +description: | + 12.3.4.1 Runtime Semantics: Evaluation + ... + 6. If Type(ref) is Reference and IsPropertyReference(ref) is false and + GetReferencedName(ref) is "eval", then + a. If SameValue(func, %eval%) is true, then + ... + ... + 9. Return ? EvaluateCall(func, ref, arguments, tailCall). + + 12.3.4.2 Runtime Semantics: EvaluateCall( func, ref, arguments, tailPosition ) + ... + 7. If tailPosition is true, perform PrepareForTailCall(). + 8. Let result be Call(func, thisValue, argList). + ... + +flags: [noStrict] +features: [tail-call-optimization] +includes: [tcoHelper.js] +---*/ + +var callCount = 0; + +(function() { + function f(n) { + "use strict"; + if (n === 0) { + callCount += 1 + return; + } + return eval(n - 1); + } + var eval = f; + f($MAX_ITERATIONS); +})(); + +assert.sameValue(callCount, 1); diff --git a/test/language/expressions/call/tco-non-eval-global.js b/test/language/expressions/call/tco-non-eval-global.js new file mode 100644 index 0000000000..8152b82538 --- /dev/null +++ b/test/language/expressions/call/tco-non-eval-global.js @@ -0,0 +1,43 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Tail-call with identifier named "eval" in global environment. +description: | + 12.3.4.1 Runtime Semantics: Evaluation + ... + 6. If Type(ref) is Reference and IsPropertyReference(ref) is false and + GetReferencedName(ref) is "eval", then + a. If SameValue(func, %eval%) is true, then + ... + ... + 9. Return ? EvaluateCall(func, ref, arguments, tailCall). + + 12.3.4.2 Runtime Semantics: EvaluateCall( func, ref, arguments, tailPosition ) + ... + 7. If tailPosition is true, perform PrepareForTailCall(). + 8. Let result be Call(func, thisValue, argList). + ... + +flags: [noStrict] +features: [tail-call-optimization] +includes: [tcoHelper.js] +---*/ + +var callCount = 0; + +function f(n) { + "use strict"; + if (n === 0) { + callCount += 1 + return; + } + return eval(n - 1); +} +eval = f; + +f($MAX_ITERATIONS); + +assert.sameValue(callCount, 1); diff --git a/test/language/expressions/call/tco-non-eval-with.js b/test/language/expressions/call/tco-non-eval-with.js new file mode 100644 index 0000000000..6797874429 --- /dev/null +++ b/test/language/expressions/call/tco-non-eval-with.js @@ -0,0 +1,46 @@ +// Copyright (C) 2017 André Bargull. All rights reserved. +// This code is governed by the BSD license found in the LICENSE file. + +/*--- +id: sec-function-calls-runtime-semantics-evaluation +info: > + Tail-call with identifier named "eval" in object environment. +description: | + 12.3.4.1 Runtime Semantics: Evaluation + ... + 6. If Type(ref) is Reference and IsPropertyReference(ref) is false and + GetReferencedName(ref) is "eval", then + a. If SameValue(func, %eval%) is true, then + ... + ... + 9. Return ? EvaluateCall(func, ref, arguments, tailCall). + + 12.3.4.2 Runtime Semantics: EvaluateCall( func, ref, arguments, tailPosition ) + ... + 7. If tailPosition is true, perform PrepareForTailCall(). + 8. Let result be Call(func, thisValue, argList). + ... + +flags: [noStrict] +features: [tail-call-optimization] +includes: [tcoHelper.js] +---*/ + +var callCount = 0; + +var f, scope = {}; +with (scope) { + f = function (n) { + "use strict"; + if (n === 0) { + callCount += 1 + return; + } + return eval(n - 1); + } +} +scope.eval = f; + +f($MAX_ITERATIONS); + +assert.sameValue(callCount, 1); -- GitLab