mirror of https://github.com/tc39/test262.git
Merge pull request #1722 from leobalter/curation-20180822
JSC Tests Curation, the stress folder
This commit is contained in:
commit
31340cbd9a
|
@ -17,4 +17,90 @@
|
|||
- set-*.js / map-*.js / weakmap-set-*.js / weakset-*.js:
|
||||
- Test static return values for before and after adding an item in an WeakSet.
|
||||
- zero-to-string.js:
|
||||
- Number#toString is not checked over literal numbers but number objects only.
|
||||
- Number#toString is not checked over literal numbers but number objects only.
|
||||
|
||||
- assign-argument-in-inlined-call.js
|
||||
- No coverage for fn_name.arguments going to inner scopes.
|
||||
- assignment-in-function-call-bracket-node.js
|
||||
- Tests obj computer propety names containing expressions that evaluates to `null`.
|
||||
- async-arrow-function-in-class-heritage.js
|
||||
- Tests class heritage failing with arrow functions (no constructor). Might have coverage, not found yet.
|
||||
- async-arrow-functions-*.js
|
||||
- No tests checking for lexical arguments/new.target/etc in async arrow functions
|
||||
- Async arrow functions needs some refactoring
|
||||
- inferred-names.js
|
||||
- Needs to add tests for multiple anonymous fn forms, verify the .name property is properly set
|
||||
- Web Compat fix: https://github.com/Microsoft/ChakraCore/issues/3407
|
||||
- builtin-function-is-construct-type-none.js
|
||||
- Add a forEach test with a new.target call
|
||||
- Extend it to other methods
|
||||
- >
|
||||
"Built-in Function Objects
|
||||
|
||||
...
|
||||
Built-in function objects that are not identified as constructors do not implement the [[Construct]] internal method unless otherwise specified in the description of a particular function. When a built-in constructor is called as part of a new expression the argumentsList parameter of the invoked [[Construct]] internal method provides the values for the built-in constructor's named parameters.
|
||||
|
||||
Built-in functions that are not constructors do not have a prototype property unless otherwise specified in the description of a particular function."
|
||||
- builtin-function-length.js
|
||||
- Add length property check for function forms in the language folder
|
||||
- computed-accessor.js / computed-accessor-parsing.js
|
||||
- Review the coverage of test/language/computed-property-names
|
||||
- good to use the test generation tool to create cases to be reused for objects and classes, etc
|
||||
- should check type coercion as well, using composed objects too
|
||||
- completion-value.js
|
||||
- v8: add tests for completion value resolution in try/catch/finally blocks (anything returning -2)
|
||||
- `eval('99; do { -99; try { 42 } catch (e) { -1 } finally { -2; break; -3 }; } while (false);')`
|
||||
- `eval('99; do { -99; try { [].x.x } catch (e) { 42; } finally { -2; break; -3 }; } while (false);')`
|
||||
- `eval('99; do { -99; try { 42 } catch (e) { -1 } finally { -2; break; -3 }; -77 } while (false);')`
|
||||
- `eval('99; do { -99; try { [].x.x } catch (e) { 42; } finally { -2; break; -3 }; -77 } while (false);')`
|
||||
- `eval('99; do { -99; try { 42 } catch (e) { -1 } finally { -2; continue; -3 }; } while (false);')`
|
||||
- `eval('99; do { -99; try { [].x.x } catch (e) { 42; } finally { -2; continue; -3 }; } while (false);')`
|
||||
- `eval('99; do { -99; try { 42 } catch (e) { -1 } finally { -2; continue; -3 }; -77 } while (false);')`
|
||||
- `eval('99; do { -99; try { [].x.x } catch (e) { 42; } finally { -2; continue; -3 }; -77 } while (false);')`
|
||||
- expected: 42
|
||||
- actual: -2
|
||||
- sm cases are already covered
|
||||
- concat-append-one-with-sparse-array.js
|
||||
- Append a small value in a very long array - not any specific part of the specs to check - test is slow but maybe worth considering?
|
||||
- create-subclass-structure-may-throw-exception-when-getting-prototype.js
|
||||
- tests are inconsistent in different engines
|
||||
- verify this on the usage for Reflect.construct
|
||||
- custom-iterators.js
|
||||
- it would be interesting to create templates and cases for iteration over objects. Using Symbol.iterator (and Symbol.asyncIterator) in for-of loops, for-of-await loops, Array.from, etc etc etc
|
||||
- The examples in that test file would be interesting to create __iterator like__ objects
|
||||
- ChakraCore is apparently reporting of a wrong error constructor
|
||||
- custom-prototype-may-be-same-to-original-one.js
|
||||
- Tests that Reflect.constructor can use a custom prototype with the same reference for the original one
|
||||
- error-description-on-symbols-should-not-crash.js
|
||||
- Symbols are not callable...
|
||||
- exception-in-to-property-key-should-be-handled-early-in-object-methods.js
|
||||
- Review coverage for Object#hasOwnProperty, add coercions for the this value
|
||||
- function-caller-*.js
|
||||
- Verify .caller behavior for nested functions, retrieving the function body.
|
||||
- Results are diverging from JSC (abrupt completion) to other engines (no error)
|
||||
- global-environment-does-not-trap-unscopables.js
|
||||
- duplicate `language/with/binding-blocked-by-unscopables.js` to assert binding block of the global in inner scopes, deferring references to the local scope.
|
||||
- ignore-promise-species.js
|
||||
- add coverage for Promise.all and race handling a local `@@species`
|
||||
- modify-set-during-iteration.js
|
||||
- Add coverage for iterating Set/Map/WeakSet/WeakMap objects with values being changed during the iteration. (e.g. for-of loops)
|
||||
- module-namespace-access*js
|
||||
- review these tests for import()
|
||||
- private-name-as-anonymous-builtin.js
|
||||
- Review name property for the executor functions in Promise.resolve calls
|
||||
- regress-187006.js
|
||||
- check if we have coverage for regexp `match` setting array values.
|
||||
- regress-179634.js
|
||||
- review coverage for Object.defineProperties
|
||||
- regress-170732.js
|
||||
- review coverage for valid and invalid await expressions
|
||||
- regress-159883.js
|
||||
- review common forms for invalid dates
|
||||
- regress-189317.js
|
||||
- AnnexB defineGetter/Setter being used with built-in functions
|
||||
- cross engine errors
|
||||
- string-to-string-error.js / string-value-of-error.js
|
||||
- Needs coverage for the this value in `String#toString` and `String#valueOf`
|
||||
- Also verify with ordinary objects containing toPrimitive methods.
|
||||
- reserved-word-with-escape.js
|
||||
- Add coverage for checking reserved names - mostly `var` - as global variables, using escape sequences.
|
||||
|
|
|
@ -1,35 +0,0 @@
|
|||
|
||||
function assert(b) {
|
||||
if (!b)
|
||||
throw new Error("Bad assertion");
|
||||
}
|
||||
|
||||
for (var i = 0; i < 1000; i++) {
|
||||
|
||||
;(function foo(x = 20) {
|
||||
assert(typeof foo === "function");
|
||||
})();
|
||||
|
||||
;(function foo(x = 20) {
|
||||
function bar() { return foo; }
|
||||
assert(typeof foo === "function");
|
||||
})();
|
||||
|
||||
;(function foo(x = foo) {
|
||||
var foo = 20;
|
||||
assert(foo === 20);
|
||||
assert(typeof x === "function");
|
||||
})();
|
||||
|
||||
;(function foo(capFoo = function() { return foo; }) {
|
||||
var foo = 20;
|
||||
assert(foo === 20);
|
||||
assert(typeof capFoo() === "function");
|
||||
})();
|
||||
|
||||
;(function foo(x = eval("foo")) {
|
||||
var foo = 20;
|
||||
assert(foo === 20);
|
||||
assert(typeof x === "function");
|
||||
})();
|
||||
}
|
|
@ -1,38 +0,0 @@
|
|||
function assert(b) {
|
||||
if (!b)
|
||||
throw new Error("Bad assertion");
|
||||
}
|
||||
|
||||
for (var i = 0; i < 1000; i++) {
|
||||
;(function foo() {
|
||||
foo = 20;
|
||||
assert(foo !== 20);
|
||||
assert(typeof foo === "function");
|
||||
})();
|
||||
|
||||
;(function foo() {
|
||||
var bar = function() { return foo; }
|
||||
foo = 20;
|
||||
assert(foo !== 20);
|
||||
assert(bar() !== 20);
|
||||
assert(typeof foo === "function");
|
||||
assert(typeof bar() === "function");
|
||||
})();
|
||||
|
||||
;(function foo() {
|
||||
eval("foo = 20;");
|
||||
assert(foo !== 20);
|
||||
assert(typeof foo === "function");
|
||||
})();
|
||||
|
||||
;(function foo() {
|
||||
eval("var foo = 20;");
|
||||
assert(foo === 20);
|
||||
})();
|
||||
|
||||
;(function foo() {
|
||||
"use strict";
|
||||
assert(foo !== 20);
|
||||
assert(typeof foo === "function");
|
||||
})();
|
||||
}
|
|
@ -1,100 +0,0 @@
|
|||
// This test should not hang, and should call the reviver function the expected number of times.
|
||||
|
||||
function shouldBe(actualExpr, expectedExpr) {
|
||||
function toString(x) {
|
||||
return "" + x;
|
||||
}
|
||||
|
||||
let actual = eval(actualExpr);
|
||||
let expected = eval(expectedExpr);
|
||||
if (typeof actual != typeof expected)
|
||||
throw Error("expected type " + typeof expected + " actual type " + typeof actual);
|
||||
if (toString(actual) != toString(expected))
|
||||
throw Error("expected: " + expected + " actual: " + actual);
|
||||
}
|
||||
|
||||
let result;
|
||||
let visited;
|
||||
|
||||
function test(parseString, clearLength) {
|
||||
visited = [];
|
||||
var result = JSON.parse(parseString, function (key, value) {
|
||||
visited.push('{' + key + ':' + value + '}');
|
||||
if (clearLength)
|
||||
this.length = 0;
|
||||
return; // returning undefined deletes the value.
|
||||
});
|
||||
return result;
|
||||
}
|
||||
|
||||
result = test('[10]', false);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{:}']");
|
||||
shouldBe("visited.length", "2");
|
||||
|
||||
result = test('[10]', true);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{:}']");
|
||||
shouldBe("visited.length", "2");
|
||||
|
||||
result = test('[ [ 10, 11 ], 12, [13, 14, 15], 16, 17]', false);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{1:11}','{0:,}','{1:12}','{0:13}','{1:14}','{2:15}','{2:,,}','{3:16}','{4:17}','{:,,,,}']");
|
||||
shouldBe("visited.length", "11");
|
||||
|
||||
result = test('[ [ 10, 11 ], 12, [13, 14, 15], 16, 17]', true);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{1:undefined}','{0:}','{1:undefined}','{2:undefined}','{3:undefined}','{4:undefined}','{:}']");
|
||||
shouldBe("visited.length", "8");
|
||||
|
||||
result = test('[ { "a": [ 10, 11 ], "b": 12 } , [ 13, { "c": 14 }, 15], 16, 17]', false);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{1:11}','{a:,}','{b:12}','{0:[object Object]}','{0:13}','{c:14}','{1:[object Object]}','{2:15}','{1:,,}','{2:16}','{3:17}','{:,,,}']");
|
||||
shouldBe("visited.length", "13");
|
||||
|
||||
result = test('[ { "a": [ 10, 11 ], "b": 12 } , [ 13, { "c": 14 }, 15], 16, 17]', true);
|
||||
shouldBe("result", "undefined");
|
||||
shouldBe("visited", "['{0:10}','{1:undefined}','{a:}','{b:12}','{0:[object Object]}','{1:undefined}','{2:undefined}','{3:undefined}','{:}']");
|
||||
shouldBe("visited.length", "9");
|
||||
|
||||
|
||||
function test2(parseString, clearLength) {
|
||||
visited = [];
|
||||
var result = JSON.parse(parseString, function (key, value) {
|
||||
visited.push('{' + key + ':' + value + '}');
|
||||
if (clearLength)
|
||||
this.length = 0;
|
||||
return (typeof value === "number") ? value * 2 : value;
|
||||
});
|
||||
return result;
|
||||
}
|
||||
|
||||
result = test2('[10]', false);
|
||||
shouldBe("result", "[20]");
|
||||
shouldBe("visited", "['{0:10}','{:20}']");
|
||||
shouldBe("visited.length", "2");
|
||||
|
||||
result = test2('[10]', true);
|
||||
shouldBe("result", "[20]");
|
||||
shouldBe("visited", "['{0:10}','{:20}']");
|
||||
shouldBe("visited.length", "2");
|
||||
|
||||
result = test2('[ [ 10, 11 ], 12, [13, 14, 15], 16, 17]', false);
|
||||
shouldBe("result", "[20,22,24,26,28,30,32,34]");
|
||||
shouldBe("visited", "['{0:10}','{1:11}','{0:20,22}','{1:12}','{0:13}','{1:14}','{2:15}','{2:26,28,30}','{3:16}','{4:17}','{:20,22,24,26,28,30,32,34}']");
|
||||
shouldBe("visited.length", "11");
|
||||
|
||||
result = test2('[ [ 10, 11 ], 12, [13, 14, 15], 16, 17]', true);
|
||||
shouldBe("result", "[]");
|
||||
shouldBe("visited", "['{0:10}','{1:undefined}','{0:}','{1:undefined}','{2:undefined}','{3:undefined}','{4:undefined}','{:}']");
|
||||
shouldBe("visited.length", "8");
|
||||
|
||||
result = test2('[ { "a": [ 10, 11 ], "b": 12 } , [ 13, { "c": 14 }, 15], 16, 17]', false);
|
||||
shouldBe("result", "['[object Object]',26,'[object Object]',30,32,34]");
|
||||
shouldBe("visited", "['{0:10}','{1:11}','{a:20,22}','{b:12}','{0:[object Object]}','{0:13}','{c:14}','{1:[object Object]}','{2:15}','{1:26,[object Object],30}','{2:16}','{3:17}','{:[object Object],26,[object Object],30,32,34}']");
|
||||
shouldBe("visited.length", "13");
|
||||
|
||||
result = test2('[ { "a": [ 10, 11 ], "b": 12 } , [ 13, { "c": 14 }, 15], 16, 17]', true);
|
||||
shouldBe("result", "[]");
|
||||
shouldBe("visited", "['{0:10}','{1:undefined}','{a:}','{b:12}','{0:[object Object]}','{1:undefined}','{2:undefined}','{3:undefined}','{:}']");
|
||||
shouldBe("visited.length", "9");
|
|
@ -1,34 +0,0 @@
|
|||
function tryGetOwnPropertyDescriptorGetName(obj, property, expectedName)
|
||||
{
|
||||
let descriptor = Object.getOwnPropertyDescriptor(obj, property);
|
||||
if (!descriptor)
|
||||
throw "Couldn't find property descriptor on object " + obj.toString() + " for property " + property.toString();
|
||||
|
||||
let getter = descriptor.get;
|
||||
if (!getter)
|
||||
throw "Property " + property.toString() + " on object " + obj.toString() + " is not a getter";
|
||||
|
||||
let getterName = getter.name;
|
||||
if (getterName !== expectedName)
|
||||
throw "Wrong getter name for property " + property.toString() + " on object " + obj.toString() + " expected " + expectedName + " got " + getterName;
|
||||
}
|
||||
|
||||
tryGetOwnPropertyDescriptorGetName(Array, Symbol.species, "get [Symbol.species]");
|
||||
tryGetOwnPropertyDescriptorGetName(Map, Symbol.species, "get [Symbol.species]");
|
||||
tryGetOwnPropertyDescriptorGetName(Set, Symbol.species, "get [Symbol.species]");
|
||||
tryGetOwnPropertyDescriptorGetName(RegExp, Symbol.species, "get [Symbol.species]");
|
||||
tryGetOwnPropertyDescriptorGetName(Promise, Symbol.species, "get [Symbol.species]");
|
||||
tryGetOwnPropertyDescriptorGetName(Map.prototype, "size", "get size");
|
||||
tryGetOwnPropertyDescriptorGetName(Set.prototype, "size", "get size");
|
||||
tryGetOwnPropertyDescriptorGetName(RegExp.prototype, "flags", "get flags");
|
||||
tryGetOwnPropertyDescriptorGetName(RegExp.prototype, "sticky", "get sticky");
|
||||
tryGetOwnPropertyDescriptorGetName(RegExp.prototype, "source", "get source");
|
||||
|
||||
if (Object.__lookupGetter__("__proto__").name !== "get __proto__")
|
||||
throw "Expected Object __proto__ getter to be named \"get __proto\"";
|
||||
|
||||
if (Object.__lookupSetter__("__proto__").name !== "set __proto__")
|
||||
throw "Expected Object __proto__ setter to be named \"set __proto\"";
|
||||
|
||||
if (Int32Array.prototype.__lookupGetter__("byteOffset").name !== "get byteOffset")
|
||||
throw "Expected TypedArray.prototype byteOffset getter to be named \"get byteOffset\"";
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function assert(a, message) {
|
||||
if (!a)
|
||||
throw new Error(message);
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function test1() {
|
||||
var o1;
|
||||
for (let i = 0; i < 1000000; ++i) {
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
//@ runDefault
|
||||
|
||||
function assert(x) {
|
||||
|
|
|
@ -1,25 +0,0 @@
|
|||
function foo(x) {
|
||||
Object.defineProperty(arguments, 0, {configurable: true, enumerable: true, writable:true, value:42});
|
||||
return [x, arguments[0], arguments]
|
||||
}
|
||||
|
||||
var result = foo(1);
|
||||
|
||||
if (result[0] !== 42)
|
||||
throw new Error();
|
||||
|
||||
if (result[1] !== 42)
|
||||
throw new Error();
|
||||
|
||||
if (Array.prototype.join.call(result[2], ",") != "42")
|
||||
throw new Error();
|
||||
|
||||
var array = [];
|
||||
for (var s in result[2])
|
||||
array.push(s);
|
||||
|
||||
if (array.join(",") != "0")
|
||||
throw new Error();
|
||||
|
||||
if (Object.keys(result[2]).join(",") != "0")
|
||||
throw new Error();
|
|
@ -1,28 +0,0 @@
|
|||
function foo(x) {
|
||||
Object.defineProperty(arguments, 0, {configurable: true, enumerable: false, writable:true, value:42});
|
||||
return [x, arguments[0], arguments]
|
||||
}
|
||||
|
||||
var result = foo(1);
|
||||
|
||||
if (result[0] !== 42)
|
||||
throw new Error();
|
||||
|
||||
if (result[1] !== 42)
|
||||
throw new Error();
|
||||
|
||||
if (Array.prototype.join.call(result[2], ",") != "42")
|
||||
throw new Error();
|
||||
|
||||
var array = [];
|
||||
for (var s in result[2])
|
||||
array.push(s);
|
||||
|
||||
if (array.join(",") != "0")
|
||||
throw new Error();
|
||||
|
||||
if (Object.keys(result[2]).join(",") != "0")
|
||||
throw new Error();
|
||||
|
||||
if (Object.getOwnPropertyDescriptor(result[2], 0).enumerable === true)
|
||||
throw new Error();
|
|
@ -1,139 +0,0 @@
|
|||
function negativeRange(results)
|
||||
{
|
||||
for (var i = -1; i > -10; --i) {
|
||||
results[Math.abs(i)] = i;
|
||||
}
|
||||
}
|
||||
noInline(negativeRange);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
negativeRange(results);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 10; ++j) {
|
||||
if (j < 1) {
|
||||
if (results[j] !== undefined)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
continue;
|
||||
}
|
||||
if (results[j] !== -j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
function negativeRangeIncludingZero(results)
|
||||
{
|
||||
for (var i = 0; i > -10; --i) {
|
||||
results[Math.abs(i)] = i;
|
||||
}
|
||||
}
|
||||
noInline(negativeRangeIncludingZero);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
negativeRangeIncludingZero(results);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 10; ++j) {
|
||||
if (results[j] !== -j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
function negativeRangeWithOverflow(results, limit)
|
||||
{
|
||||
var i = -2147483648 + 10;
|
||||
do {
|
||||
results.push(Math.abs(i));
|
||||
--i;
|
||||
} while (i !== limit);
|
||||
}
|
||||
noInline(negativeRangeWithOverflow);
|
||||
|
||||
// First, we warm up without overflow.
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
negativeRangeWithOverflow(results, -2147483647);
|
||||
if (results.length != 9)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 9; ++j) {
|
||||
if (results[j] !== 2147483638 + j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
// Then we overflow.
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
negativeRangeWithOverflow(results, -2147483648);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 10; ++j) {
|
||||
if (results[j] !== 2147483638 + j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
function positiveRange(results)
|
||||
{
|
||||
for (var i = 1; i < 10; ++i) {
|
||||
results[Math.abs(i)] = i;
|
||||
}
|
||||
}
|
||||
noInline(positiveRange);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
positiveRange(results);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 10; ++j) {
|
||||
if (j < 1) {
|
||||
if (results[j] !== undefined)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
continue;
|
||||
}
|
||||
if (results[j] !== j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
function positiveRangeIncludingZero(results)
|
||||
{
|
||||
for (var i = 0; i < 10; ++i) {
|
||||
results[Math.abs(i)] = i;
|
||||
}
|
||||
}
|
||||
noInline(positiveRangeIncludingZero);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
positiveRangeIncludingZero(results);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = 0; j < 10; ++j) {
|
||||
if (results[j] !== j)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
||||
|
||||
function rangeWithoutOverflow(results)
|
||||
{
|
||||
for (var i = -10; i < 10; ++i) {
|
||||
results[i] = Math.abs(i);
|
||||
}
|
||||
}
|
||||
noInline(rangeWithoutOverflow);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
var results = [];
|
||||
rangeWithoutOverflow(results);
|
||||
if (results.length != 10)
|
||||
throw "Wrong result length: " + results.length;
|
||||
for (var j = -10; j < 10; ++j) {
|
||||
var expected = (j < 0) ? -j : j;
|
||||
if (results[j] !== expected)
|
||||
throw "Wrong result, results[j] = " + results[j] + " at j = " + j;
|
||||
}
|
||||
}
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
|
|
@ -1,237 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let asinOfHalf = Math.asin(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["1", "" + Math.asin(1)],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "" + asinOfHalf],
|
||||
["Math.PI", "" + Math.asin(Math.PI)],
|
||||
["Infinity", "NaN"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + asinOfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + asinOfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.asin() without arguments.
|
||||
function opaqueASinNoArgument() {
|
||||
return Math.asin();
|
||||
}
|
||||
noInline(opaqueASinNoArgument);
|
||||
noOSRExitFuzzing(opaqueASinNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueASinNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueASinNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASinNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.asin() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesASin(argument) {
|
||||
return Math.asin(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesASin);
|
||||
noOSRExitFuzzing(opaqueAllTypesASin);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesASin(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesASin) > 2)
|
||||
throw "We should have detected asin() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.asin() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueASin(argument) {
|
||||
return Math.asin(argument);
|
||||
}
|
||||
noInline(opaqueASin);
|
||||
noOSRExitFuzzing(opaqueASin);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueASin(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASin) > 1)
|
||||
throw "We should have compiled a single asin for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.asin() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueASinOnConstant() {
|
||||
return Math.asin(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueASinOnConstant);
|
||||
noOSRExitFuzzing(opaqueASinOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueASinOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASinOnConstant) > 1)
|
||||
throw "We should have compiled a single asin for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueASinForSideEffects(argument) {
|
||||
return Math.asin(argument);
|
||||
}
|
||||
noInline(opaqueASinForSideEffects);
|
||||
noOSRExitFuzzing(opaqueASinForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let asinResult = Math.asin(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueASinForSideEffects(testObject) !== asinResult)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueASinForSideEffects) > 1)
|
||||
throw "opaqueASinForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify asin() is not subject to CSE if the argument has side effects.
|
||||
function opaqueASinForCSE(argument) {
|
||||
return Math.asin(argument) + Math.asin(argument) + Math.asin(argument);
|
||||
}
|
||||
noInline(opaqueASinForCSE);
|
||||
noOSRExitFuzzing(opaqueASinForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let asinResult = Math.asin(0.2);
|
||||
let threeasinResult = asinResult + asinResult + asinResult;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueASinForCSE(testObject) !== threeasinResult)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueASinForCSE) > 1)
|
||||
throw "opaqueASinForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify asin() is not subject to DCE if the argument has side effects.
|
||||
function opaqueASinForDCE(argument) {
|
||||
Math.asin(argument);
|
||||
}
|
||||
noInline(opaqueASinForDCE);
|
||||
noOSRExitFuzzing(opaqueASinForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueASinForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueASinForDCE) > 1)
|
||||
throw "opaqueASinForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueASinWithException(argument) {
|
||||
let result = Math.asin(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueASinWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let asinResult = Math.asin(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueASinWithException(testObject) !== asinResult)
|
||||
throw "Incorrect result in opaqueASinWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueASinWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueASinWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let asinhOfFour = Math.asinh(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + asinhOfFour],
|
||||
["Math.PI", "" + Math.asinh(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + asinhOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + asinhOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.asinh() without arguments.
|
||||
function opaqueASinhNoArgument() {
|
||||
return Math.asinh();
|
||||
}
|
||||
noInline(opaqueASinhNoArgument);
|
||||
noOSRExitFuzzing(opaqueASinhNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueASinhNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueASinhNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASinhNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.asinh() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesASinh(argument) {
|
||||
return Math.asinh(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesASinh);
|
||||
noOSRExitFuzzing(opaqueAllTypesASinh);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesASinh(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesASinh) > 2)
|
||||
throw "We should have detected asinh() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.asinh() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueASinh(argument) {
|
||||
return Math.asinh(argument);
|
||||
}
|
||||
noInline(opaqueASinh);
|
||||
noOSRExitFuzzing(opaqueASinh);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueASinh(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASinh) > 1)
|
||||
throw "We should have compiled a single asinh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.asinh() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueASinhOnConstant() {
|
||||
return Math.asinh(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueASinhOnConstant);
|
||||
noOSRExitFuzzing(opaqueASinhOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueASinhOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueASinhOnConstant) > 1)
|
||||
throw "We should have compiled a single asinh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueASinhForSideEffects(argument) {
|
||||
return Math.asinh(argument);
|
||||
}
|
||||
noInline(opaqueASinhForSideEffects);
|
||||
noOSRExitFuzzing(opaqueASinhForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let asinh16 = Math.asinh(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueASinhForSideEffects(testObject) !== asinh16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueASinhForSideEffects) > 1)
|
||||
throw "opaqueASinhForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify asinh() is not subject to CSE if the argument has side effects.
|
||||
function opaqueASinhForCSE(argument) {
|
||||
return Math.asinh(argument) + Math.asinh(argument) + Math.asinh(argument);
|
||||
}
|
||||
noInline(opaqueASinhForCSE);
|
||||
noOSRExitFuzzing(opaqueASinhForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let asinh16 = Math.asinh(16);
|
||||
let threeASinh16 = asinh16 + asinh16 + asinh16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueASinhForCSE(testObject) !== threeASinh16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueASinhForCSE) > 1)
|
||||
throw "opaqueASinhForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify asinh() is not subject to DCE if the argument has side effects.
|
||||
function opaqueASinhForDCE(argument) {
|
||||
Math.asinh(argument);
|
||||
}
|
||||
noInline(opaqueASinhForDCE);
|
||||
noOSRExitFuzzing(opaqueASinhForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueASinhForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueASinhForDCE) > 1)
|
||||
throw "opaqueASinhForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueASinhWithException(argument) {
|
||||
let result = Math.asinh(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueASinhWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let asinh64 = Math.asinh(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueASinhWithException(testObject) !== asinh64)
|
||||
throw "Incorrect result in opaqueASinhWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueASinhWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueASinhWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let atanOfFour = Math.atan(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + atanOfFour],
|
||||
["Math.PI", "" + Math.atan(Math.PI)],
|
||||
["Infinity", "" + Math.atan(Infinity)],
|
||||
["-Infinity", "-" + Math.atan(Infinity)],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + atanOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + atanOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.atan() without arguments.
|
||||
function opaqueATanNoArgument() {
|
||||
return Math.atan();
|
||||
}
|
||||
noInline(opaqueATanNoArgument);
|
||||
noOSRExitFuzzing(opaqueATanNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueATanNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueATanNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATanNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.atan() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesATan(argument) {
|
||||
return Math.atan(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesATan);
|
||||
noOSRExitFuzzing(opaqueAllTypesATan);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesATan(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesATan) > 2)
|
||||
throw "We should have detected atan() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.atan() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueATan(argument) {
|
||||
return Math.atan(argument);
|
||||
}
|
||||
noInline(opaqueATan);
|
||||
noOSRExitFuzzing(opaqueATan);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueATan(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATan) > 1)
|
||||
throw "We should have compiled a single atan for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.atan() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueATanOnConstant() {
|
||||
return Math.atan(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueATanOnConstant);
|
||||
noOSRExitFuzzing(opaqueATanOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueATanOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATanOnConstant) > 1)
|
||||
throw "We should have compiled a single atan for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueATanForSideEffects(argument) {
|
||||
return Math.atan(argument);
|
||||
}
|
||||
noInline(opaqueATanForSideEffects);
|
||||
noOSRExitFuzzing(opaqueATanForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let AtanResult = Math.atan(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueATanForSideEffects(testObject) !== AtanResult)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueATanForSideEffects) > 1)
|
||||
throw "opaqueATanForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify atan() is not subject to CSE if the argument has side effects.
|
||||
function opaqueATanForCSE(argument) {
|
||||
return Math.atan(argument) + Math.atan(argument) + Math.atan(argument);
|
||||
}
|
||||
noInline(opaqueATanForCSE);
|
||||
noOSRExitFuzzing(opaqueATanForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let AtanResult = Math.atan(16);
|
||||
let threeAtanResult = AtanResult + AtanResult + AtanResult;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueATanForCSE(testObject) !== threeAtanResult)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueATanForCSE) > 1)
|
||||
throw "opaqueATanForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify atan() is not subject to DCE if the argument has side effects.
|
||||
function opaqueATanForDCE(argument) {
|
||||
Math.atan(argument);
|
||||
}
|
||||
noInline(opaqueATanForDCE);
|
||||
noOSRExitFuzzing(opaqueATanForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueATanForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueATanForDCE) > 1)
|
||||
throw "opaqueATanForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueATanWithException(argument) {
|
||||
let result = Math.atan(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueATanWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let atanResult = Math.atan(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueATanWithException(testObject) !== atanResult)
|
||||
throw "Incorrect result in opaqueATanWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueATanWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueATanWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,237 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let atanhOfHalf = Math.atanh(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["1", "Infinity"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "" + atanhOfHalf],
|
||||
["Math.PI", "" + Math.atanh(Math.PI)],
|
||||
["Infinity", "NaN"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + atanhOfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + atanhOfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.atanh() without arguments.
|
||||
function opaqueATanhNoArgument() {
|
||||
return Math.atanh();
|
||||
}
|
||||
noInline(opaqueATanhNoArgument);
|
||||
noOSRExitFuzzing(opaqueATanhNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueATanhNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueATanhNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATanhNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.atanh() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesATanh(argument) {
|
||||
return Math.atanh(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesATanh);
|
||||
noOSRExitFuzzing(opaqueAllTypesATanh);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesATanh(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesATanh) > 2)
|
||||
throw "We should have detected atanh() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.atanh() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueATanh(argument) {
|
||||
return Math.atanh(argument);
|
||||
}
|
||||
noInline(opaqueATanh);
|
||||
noOSRExitFuzzing(opaqueATanh);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueATanh(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATanh) > 1)
|
||||
throw "We should have compiled a single atanh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.atanh() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueATanhOnConstant() {
|
||||
return Math.atanh(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueATanhOnConstant);
|
||||
noOSRExitFuzzing(opaqueATanhOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueATanhOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueATanhOnConstant) > 1)
|
||||
throw "We should have compiled a single atanh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueATanhForSideEffects(argument) {
|
||||
return Math.atanh(argument);
|
||||
}
|
||||
noInline(opaqueATanhForSideEffects);
|
||||
noOSRExitFuzzing(opaqueATanhForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let atanhResult = Math.atanh(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueATanhForSideEffects(testObject) !== atanhResult)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueATanhForSideEffects) > 1)
|
||||
throw "opaqueATanhForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify atanh() is not subject to CSE if the argument has side effects.
|
||||
function opaqueATanhForCSE(argument) {
|
||||
return Math.atanh(argument) + Math.atanh(argument) + Math.atanh(argument);
|
||||
}
|
||||
noInline(opaqueATanhForCSE);
|
||||
noOSRExitFuzzing(opaqueATanhForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let atanhResult = Math.atanh(0.2);
|
||||
let threeatanhResult = atanhResult + atanhResult + atanhResult;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueATanhForCSE(testObject) !== threeatanhResult)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueATanhForCSE) > 1)
|
||||
throw "opaqueATanhForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify atanh() is not subject to DCE if the argument has side effects.
|
||||
function opaqueATanhForDCE(argument) {
|
||||
Math.atanh(argument);
|
||||
}
|
||||
noInline(opaqueATanhForDCE);
|
||||
noOSRExitFuzzing(opaqueATanhForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueATanhForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueATanhForDCE) > 1)
|
||||
throw "opaqueATanhForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueATanhWithException(argument) {
|
||||
let result = Math.atanh(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueATanhWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let atanhResult = Math.atanh(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueATanhWithException(testObject) !== atanhResult)
|
||||
throw "Incorrect result in opaqueATanhWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueATanhWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueATanhWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,238 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let cbrtOfHalf = Math.cbrt(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["1", "1"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["-42.", "" + Math.cbrt(-42)],
|
||||
["0.5", "" + cbrtOfHalf],
|
||||
["Math.PI", "" + Math.cbrt(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + cbrtOfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + cbrtOfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.cbrt() without arguments.
|
||||
function opaqueCbrtNoArgument() {
|
||||
return Math.cbrt();
|
||||
}
|
||||
noInline(opaqueCbrtNoArgument);
|
||||
noOSRExitFuzzing(opaqueCbrtNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueCbrtNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueCbrtNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCbrtNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.cbrt() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesCbrt(argument) {
|
||||
return Math.cbrt(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesCbrt);
|
||||
noOSRExitFuzzing(opaqueAllTypesCbrt);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesCbrt(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesCbrt) > 2)
|
||||
throw "We should have detected cbrt() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.cbrt() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCbrt(argument) {
|
||||
return Math.cbrt(argument);
|
||||
}
|
||||
noInline(opaqueCbrt);
|
||||
noOSRExitFuzzing(opaqueCbrt);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCbrt(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCbrt) > 1)
|
||||
throw "We should have compiled a single cbrt for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.cbrt() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCbrtOnConstant() {
|
||||
return Math.cbrt(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueCbrtOnConstant);
|
||||
noOSRExitFuzzing(opaqueCbrtOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCbrtOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCbrtOnConstant) > 1)
|
||||
throw "We should have compiled a single cbrt for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueCbrtForSideEffects(argument) {
|
||||
return Math.cbrt(argument);
|
||||
}
|
||||
noInline(opaqueCbrtForSideEffects);
|
||||
noOSRExitFuzzing(opaqueCbrtForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let cbrtResult = Math.cbrt(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCbrtForSideEffects(testObject) !== cbrtResult)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueCbrtForSideEffects) > 1)
|
||||
throw "opaqueCbrtForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify cbrt() is not subject to CSE if the argument has side effects.
|
||||
function opaqueCbrtForCSE(argument) {
|
||||
return Math.cbrt(argument) + Math.cbrt(argument) + Math.cbrt(argument);
|
||||
}
|
||||
noInline(opaqueCbrtForCSE);
|
||||
noOSRExitFuzzing(opaqueCbrtForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let cbrtResult = Math.cbrt(0.2);
|
||||
let threecbrtResult = cbrtResult + cbrtResult + cbrtResult;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCbrtForCSE(testObject) !== threecbrtResult)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueCbrtForCSE) > 1)
|
||||
throw "opaqueCbrtForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify cbrt() is not subject to DCE if the argument has side effects.
|
||||
function opaqueCbrtForDCE(argument) {
|
||||
Math.cbrt(argument);
|
||||
}
|
||||
noInline(opaqueCbrtForDCE);
|
||||
noOSRExitFuzzing(opaqueCbrtForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueCbrtForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueCbrtForDCE) > 1)
|
||||
throw "opaqueCbrtForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueCbrtWithException(argument) {
|
||||
let result = Math.cbrt(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueCbrtWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let cbrtResult = Math.cbrt(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueCbrtWithException(testObject) !== cbrtResult)
|
||||
throw "Incorrect result in opaqueCbrtWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueCbrtWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueCbrtWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,308 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "1"],
|
||||
["-0.5", "-0"],
|
||||
["4", "4"],
|
||||
["42.1", "43"],
|
||||
["42.5", "43"],
|
||||
["42.9", "43"],
|
||||
["-42.1", "-42"],
|
||||
["-42.5", "-42"],
|
||||
["-42.9", "-42"],
|
||||
["Math.PI", "4"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "4"],
|
||||
["\"42.5\"", "43"],
|
||||
["{ valueOf: () => { return 4; } }", "4"],
|
||||
["{ valueOf: () => { return 0; } }", "0"],
|
||||
["{ valueOf: () => { return -0; } }", "-0"],
|
||||
["{ valueOf: () => { return 0.5; } }", "1"],
|
||||
["{ valueOf: () => { return -0.5; } }", "-0"],
|
||||
["{ valueOf: () => { return Number.MIN_SAFE_INTEGER; } }", "-9007199254740991"],
|
||||
["{ valueOf: () => { return Number.MAX_SAFE_INTEGER; } }", "9007199254740991"],
|
||||
["{ valueOf: () => { return 0x80000000|0; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return 0x7fffffff|0; } }", "2147483647"],
|
||||
["{ valueOf: () => { return (0x80000000|0) - 0.5; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return (0x7fffffff|0) + 0.5; } }", "2147483648"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && (1 / expected) !== (1 / result))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.ceil() without arguments.
|
||||
function opaqueCeilNoArgument() {
|
||||
return Math.ceil();
|
||||
}
|
||||
noInline(opaqueCeilNoArgument);
|
||||
noOSRExitFuzzing(opaqueCeilNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueCeilNoArgument();
|
||||
if (!isIdentical(output, NaN)) {
|
||||
throw "Failed opaqueCeilNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCeilNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.ceil() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesCeil(argument) {
|
||||
return Math.ceil(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesCeil);
|
||||
noOSRExitFuzzing(opaqueAllTypesCeil);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesCeil(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesCeil) > 3)
|
||||
throw "We should have detected ceil() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Polymorphic input but negative zero is not observable.
|
||||
function opaqueAllTypesCeilWithoutNegativeZero(argument) {
|
||||
return Math.ceil(argument) + 0;
|
||||
}
|
||||
noInline(opaqueAllTypesCeilWithoutNegativeZero);
|
||||
noOSRExitFuzzing(opaqueAllTypesCeilWithoutNegativeZero);
|
||||
|
||||
function testAllTypesWithoutNegativeZeroCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesCeilWithoutNegativeZero(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1] + 0))
|
||||
throw "Failed testAllTypesWithoutNegativeZeroCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesCeil) > 3)
|
||||
throw "We should have detected ceil() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.ceil() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCeil(argument) {
|
||||
return Math.ceil(argument);
|
||||
}
|
||||
noInline(opaqueCeil);
|
||||
noOSRExitFuzzing(opaqueCeil);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCeil(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCeil) > 1)
|
||||
throw "We should have compiled a single ceil for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
function checkCompileCountForUselessNegativeZero(testFunction)
|
||||
{
|
||||
if (jscOptions().useMaximalFlushInsertionPhase) {
|
||||
// If we forced a flush after the operation, the negative zero becomes
|
||||
// observable and we may be overly optimistic.
|
||||
return numberOfDFGCompiles(testFunction) <= 2;
|
||||
}
|
||||
return numberOfDFGCompiles(testFunction) <= 1;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.ceil() on a completely typed input, but without negative zero.
|
||||
function testSingleTypeWithoutNegativeZeroCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCeil(argument) {
|
||||
return Math.ceil(argument) + 0;
|
||||
}
|
||||
noInline(opaqueCeil);
|
||||
noOSRExitFuzzing(opaqueCeil);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCeil(${testCaseInput[0]}), ${testCaseInput[1]} + 0)) {
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall()";
|
||||
}
|
||||
}
|
||||
if (!checkCompileCountForUselessNegativeZero(opaqueCeil))
|
||||
throw "We should have compiled a single ceil for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.ceil() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCeilOnConstant() {
|
||||
return Math.ceil(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueCeilOnConstant);
|
||||
noOSRExitFuzzing(opaqueCeilOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCeilOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCeilOnConstant) > 1)
|
||||
throw "We should have compiled a single ceil for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueCeilForSideEffects(argument) {
|
||||
return Math.ceil(argument);
|
||||
}
|
||||
noInline(opaqueCeilForSideEffects);
|
||||
noOSRExitFuzzing(opaqueCeilForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let ceil16 = Math.ceil(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCeilForSideEffects(testObject) !== ceil16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueCeilForSideEffects) > 1)
|
||||
throw "opaqueCeilForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify ceil() is not subject to CSE if the argument has side effects.
|
||||
function opaqueCeilForCSE(argument) {
|
||||
return Math.ceil(argument) + Math.ceil(argument) + Math.ceil(argument);
|
||||
}
|
||||
noInline(opaqueCeilForCSE);
|
||||
noOSRExitFuzzing(opaqueCeilForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let ceil16 = Math.ceil(16);
|
||||
let threeCeil16 = ceil16 + ceil16 + ceil16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCeilForCSE(testObject) !== threeCeil16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueCeilForCSE) > 1)
|
||||
throw "opaqueCeilForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify ceil() is not subject to DCE if the argument has side effects.
|
||||
function opaqueCeilForDCE(argument) {
|
||||
Math.ceil(argument);
|
||||
}
|
||||
noInline(opaqueCeilForDCE);
|
||||
noOSRExitFuzzing(opaqueCeilForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueCeilForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueCeilForDCE) > 1)
|
||||
throw "opaqueCeilForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueCeilWithException(argument) {
|
||||
let result = Math.ceil(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueCeilWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let ceil64 = Math.ceil(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueCeilWithException(testObject) !== ceil64)
|
||||
throw "Incorrect result in opaqueCeilWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueCeilWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueCeilWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,234 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "32"],
|
||||
["null", "32"],
|
||||
["0", "32"],
|
||||
["-0.", "32"],
|
||||
["4", "29"],
|
||||
["Math.PI", "30"],
|
||||
["Infinity", "32"],
|
||||
["-Infinity", "32"],
|
||||
["NaN", "32"],
|
||||
["\"WebKit\"", "32"],
|
||||
["\"4\"", "29"],
|
||||
["{ valueOf: () => { return 4; } }", "29"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.clz32() without arguments.
|
||||
function opaqueClz32NoArgument() {
|
||||
return Math.clz32();
|
||||
}
|
||||
noInline(opaqueClz32NoArgument);
|
||||
noOSRExitFuzzing(opaqueClz32NoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueClz32NoArgument();
|
||||
if (output !== 32) {
|
||||
throw "Failed opaqueClz32NoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueClz32NoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.clz32() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesClz32(argument) {
|
||||
return Math.clz32(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesClz32);
|
||||
noOSRExitFuzzing(opaqueAllTypesClz32);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesClz32(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesClz32) > 2)
|
||||
throw "We should have detected clz32() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.clz32() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueClz32(argument) {
|
||||
return Math.clz32(argument);
|
||||
}
|
||||
noInline(opaqueClz32);
|
||||
noOSRExitFuzzing(opaqueClz32);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueClz32(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueClz32) > 1)
|
||||
throw "We should have compiled a single clz32 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.clz32() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueClz32OnConstant() {
|
||||
return Math.clz32(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueClz32OnConstant);
|
||||
noOSRExitFuzzing(opaqueClz32OnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueClz32OnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueClz32OnConstant) > 1)
|
||||
throw "We should have compiled a single clz32 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueClz32ForSideEffects(argument) {
|
||||
return Math.clz32(argument);
|
||||
}
|
||||
noInline(opaqueClz32ForSideEffects);
|
||||
noOSRExitFuzzing(opaqueClz32ForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let clz3216 = Math.clz32(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueClz32ForSideEffects(testObject) !== clz3216)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueClz32ForSideEffects) > 1)
|
||||
throw "opaqueClz32ForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify clz32() is not subject to CSE if the argument has side effects.
|
||||
function opaqueClz32ForCSE(argument) {
|
||||
return Math.clz32(argument) + Math.clz32(argument) + Math.clz32(argument);
|
||||
}
|
||||
noInline(opaqueClz32ForCSE);
|
||||
noOSRExitFuzzing(opaqueClz32ForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let clz3216 = Math.clz32(16);
|
||||
let threeClz3216 = clz3216 + clz3216 + clz3216;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueClz32ForCSE(testObject) !== threeClz3216)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueClz32ForCSE) > 1)
|
||||
throw "opaqueClz32ForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify clz32() is not subject to DCE if the argument has side effects.
|
||||
function opaqueClz32ForDCE(argument) {
|
||||
Math.clz32(argument);
|
||||
}
|
||||
noInline(opaqueClz32ForDCE);
|
||||
noOSRExitFuzzing(opaqueClz32ForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueClz32ForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueClz32ForDCE) > 1)
|
||||
throw "opaqueClz32ForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueClz32WithException(argument) {
|
||||
let result = Math.clz32(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueClz32WithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let clz3264 = Math.clz32(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueClz32WithException(testObject) !== clz3264)
|
||||
throw "Incorrect result in opaqueClz32WithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueClz32WithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueClz32WithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let cosOfFour = Math.cos(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "1"],
|
||||
["0", "1"],
|
||||
["-0.", "1"],
|
||||
["4", "" + cosOfFour],
|
||||
["Math.PI", "-1"],
|
||||
["Infinity", "NaN"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + cosOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + cosOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.cos() without arguments.
|
||||
function opaqueCosNoArgument() {
|
||||
return Math.cos();
|
||||
}
|
||||
noInline(opaqueCosNoArgument);
|
||||
noOSRExitFuzzing(opaqueCosNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueCosNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueCosNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCosNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.cos() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesCos(argument) {
|
||||
return Math.cos(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesCos);
|
||||
noOSRExitFuzzing(opaqueAllTypesCos);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesCos(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesCos) > 2)
|
||||
throw "We should have detected cos() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.cos() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCos(argument) {
|
||||
return Math.cos(argument);
|
||||
}
|
||||
noInline(opaqueCos);
|
||||
noOSRExitFuzzing(opaqueCos);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCos(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCos) > 1)
|
||||
throw "We should have compiled a single cos for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.cos() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCosOnConstant() {
|
||||
return Math.cos(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueCosOnConstant);
|
||||
noOSRExitFuzzing(opaqueCosOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCosOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCosOnConstant) > 1)
|
||||
throw "We should have compiled a single cos for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueCosForSideEffects(argument) {
|
||||
return Math.cos(argument);
|
||||
}
|
||||
noInline(opaqueCosForSideEffects);
|
||||
noOSRExitFuzzing(opaqueCosForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let cos16 = Math.cos(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCosForSideEffects(testObject) !== cos16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueCosForSideEffects) > 1)
|
||||
throw "opaqueCosForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify cos() is not subject to CSE if the argument has side effects.
|
||||
function opaqueCosForCSE(argument) {
|
||||
return Math.cos(argument) + Math.cos(argument) + Math.cos(argument);
|
||||
}
|
||||
noInline(opaqueCosForCSE);
|
||||
noOSRExitFuzzing(opaqueCosForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let cos16 = Math.cos(16);
|
||||
let threeCos16 = cos16 + cos16 + cos16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCosForCSE(testObject) !== threeCos16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueCosForCSE) > 1)
|
||||
throw "opaqueCosForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify cos() is not subject to DCE if the argument has side effects.
|
||||
function opaqueCosForDCE(argument) {
|
||||
Math.cos(argument);
|
||||
}
|
||||
noInline(opaqueCosForDCE);
|
||||
noOSRExitFuzzing(opaqueCosForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueCosForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueCosForDCE) > 1)
|
||||
throw "opaqueCosForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueCosWithException(argument) {
|
||||
let result = Math.cos(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueCosWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let cos64 = Math.cos(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueCosWithException(testObject) !== cos64)
|
||||
throw "Incorrect result in opaqueCosWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueCosWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueCosWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let coshOfFour = Math.cosh(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "1"],
|
||||
["0", "1"],
|
||||
["-0.", "1"],
|
||||
["4", "" + coshOfFour],
|
||||
["Math.PI", "" + Math.cosh(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + coshOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + coshOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.cosh() without arguments.
|
||||
function opaqueCoshNoArgument() {
|
||||
return Math.cosh();
|
||||
}
|
||||
noInline(opaqueCoshNoArgument);
|
||||
noOSRExitFuzzing(opaqueCoshNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueCoshNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueCoshNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCoshNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.cosh() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesCosh(argument) {
|
||||
return Math.cosh(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesCosh);
|
||||
noOSRExitFuzzing(opaqueAllTypesCosh);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesCosh(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesCosh) > 2)
|
||||
throw "We should have detected cosh() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.cosh() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCosh(argument) {
|
||||
return Math.cosh(argument);
|
||||
}
|
||||
noInline(opaqueCosh);
|
||||
noOSRExitFuzzing(opaqueCosh);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCosh(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCosh) > 1)
|
||||
throw "We should have compiled a single cosh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.cosh() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueCoshOnConstant() {
|
||||
return Math.cosh(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueCoshOnConstant);
|
||||
noOSRExitFuzzing(opaqueCoshOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueCoshOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueCoshOnConstant) > 1)
|
||||
throw "We should have compiled a single cosh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueCoshForSideEffects(argument) {
|
||||
return Math.cosh(argument);
|
||||
}
|
||||
noInline(opaqueCoshForSideEffects);
|
||||
noOSRExitFuzzing(opaqueCoshForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let cosh16 = Math.cosh(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCoshForSideEffects(testObject) !== cosh16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueCoshForSideEffects) > 1)
|
||||
throw "opaqueCoshForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify cosh() is not subject to CSE if the argument has side effects.
|
||||
function opaqueCoshForCSE(argument) {
|
||||
return Math.cosh(argument) + Math.cosh(argument) + Math.cosh(argument);
|
||||
}
|
||||
noInline(opaqueCoshForCSE);
|
||||
noOSRExitFuzzing(opaqueCoshForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let cosh16 = Math.cosh(16);
|
||||
let threeCosh16 = cosh16 + cosh16 + cosh16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueCoshForCSE(testObject) !== threeCosh16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueCoshForCSE) > 1)
|
||||
throw "opaqueCoshForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify cosh() is not subject to DCE if the argument has side effects.
|
||||
function opaqueCoshForDCE(argument) {
|
||||
Math.cosh(argument);
|
||||
}
|
||||
noInline(opaqueCoshForDCE);
|
||||
noOSRExitFuzzing(opaqueCoshForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueCoshForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueCoshForDCE) > 1)
|
||||
throw "opaqueCoshForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueCoshWithException(argument) {
|
||||
let result = Math.cosh(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueCoshWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let cosh64 = Math.cosh(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueCoshWithException(testObject) !== cosh64)
|
||||
throw "Incorrect result in opaqueCoshWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueCoshWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueCoshWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,237 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let expm1OfHalf = Math.expm1(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["1", "" + Math.expm1(1)],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "" + expm1OfHalf],
|
||||
["Math.PI", "" + Math.expm1(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-1"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + expm1OfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + expm1OfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.expm1() without arguments.
|
||||
function opaqueExpm1NoArgument() {
|
||||
return Math.expm1();
|
||||
}
|
||||
noInline(opaqueExpm1NoArgument);
|
||||
noOSRExitFuzzing(opaqueExpm1NoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueExpm1NoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueExpm1NoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueExpm1NoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.expm1() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesExpm1(argument) {
|
||||
return Math.expm1(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesExpm1);
|
||||
noOSRExitFuzzing(opaqueAllTypesExpm1);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesExpm1(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesExpm1) > 2)
|
||||
throw "We should have detected expm1() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.expm1() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueExpm1(argument) {
|
||||
return Math.expm1(argument);
|
||||
}
|
||||
noInline(opaqueExpm1);
|
||||
noOSRExitFuzzing(opaqueExpm1);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueExpm1(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueExpm1) > 1)
|
||||
throw "We should have compiled a single expm1 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.expm1() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueExpm1OnConstant() {
|
||||
return Math.expm1(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueExpm1OnConstant);
|
||||
noOSRExitFuzzing(opaqueExpm1OnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueExpm1OnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueExpm1OnConstant) > 1)
|
||||
throw "We should have compiled a single expm1 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueExpm1ForSideEffects(argument) {
|
||||
return Math.expm1(argument);
|
||||
}
|
||||
noInline(opaqueExpm1ForSideEffects);
|
||||
noOSRExitFuzzing(opaqueExpm1ForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let expm1Result = Math.expm1(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueExpm1ForSideEffects(testObject) !== expm1Result)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueExpm1ForSideEffects) > 1)
|
||||
throw "opaqueExpm1ForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify expm1() is not subject to CSE if the argument has side effects.
|
||||
function opaqueExpm1ForCSE(argument) {
|
||||
return Math.expm1(argument) + Math.expm1(argument) + Math.expm1(argument);
|
||||
}
|
||||
noInline(opaqueExpm1ForCSE);
|
||||
noOSRExitFuzzing(opaqueExpm1ForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let expm1Result = Math.expm1(0.2);
|
||||
let threeexpm1Result = expm1Result + expm1Result + expm1Result;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueExpm1ForCSE(testObject) !== threeexpm1Result)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueExpm1ForCSE) > 1)
|
||||
throw "opaqueExpm1ForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify expm1() is not subject to DCE if the argument has side effects.
|
||||
function opaqueExpm1ForDCE(argument) {
|
||||
Math.expm1(argument);
|
||||
}
|
||||
noInline(opaqueExpm1ForDCE);
|
||||
noOSRExitFuzzing(opaqueExpm1ForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueExpm1ForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueExpm1ForDCE) > 1)
|
||||
throw "opaqueExpm1ForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueExpm1WithException(argument) {
|
||||
let result = Math.expm1(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueExpm1WithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let expm1Result = Math.expm1(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueExpm1WithException(testObject) !== expm1Result)
|
||||
throw "Incorrect result in opaqueExpm1WithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueExpm1WithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueExpm1WithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,308 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "0"],
|
||||
["-0.5", "-1"],
|
||||
["4", "4"],
|
||||
["42.1", "42"],
|
||||
["42.5", "42"],
|
||||
["42.9", "42"],
|
||||
["-42.1", "-43"],
|
||||
["-42.5", "-43"],
|
||||
["-42.9", "-43"],
|
||||
["Math.PI", "3"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "4"],
|
||||
["\"42.5\"", "42"],
|
||||
["{ valueOf: () => { return 4; } }", "4"],
|
||||
["{ valueOf: () => { return 0; } }", "0"],
|
||||
["{ valueOf: () => { return -0; } }", "-0"],
|
||||
["{ valueOf: () => { return 0.5; } }", "0"],
|
||||
["{ valueOf: () => { return -0.5; } }", "-1"],
|
||||
["{ valueOf: () => { return Number.MIN_SAFE_INTEGER; } }", "-9007199254740991"],
|
||||
["{ valueOf: () => { return Number.MAX_SAFE_INTEGER; } }", "9007199254740991"],
|
||||
["{ valueOf: () => { return 0x80000000|0; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return 0x7fffffff|0; } }", "2147483647"],
|
||||
["{ valueOf: () => { return (0x80000000|0) - 0.5; } }", "-2147483649"],
|
||||
["{ valueOf: () => { return (0x7fffffff|0) + 0.5; } }", "2147483647"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && (1 / expected) !== (1 / result))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.floor() without arguments.
|
||||
function opaqueFloorNoArgument() {
|
||||
return Math.floor();
|
||||
}
|
||||
noInline(opaqueFloorNoArgument);
|
||||
noOSRExitFuzzing(opaqueFloorNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueFloorNoArgument();
|
||||
if (!isIdentical(output, NaN)) {
|
||||
throw "Failed opaqueFloorNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFloorNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.floor() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesFloor(argument) {
|
||||
return Math.floor(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesFloor);
|
||||
noOSRExitFuzzing(opaqueAllTypesFloor);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesFloor(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesFloor) > 3)
|
||||
throw "We should have detected floor() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Polymorphic input but negative zero is not observable.
|
||||
function opaqueAllTypesFloorWithoutNegativeZero(argument) {
|
||||
return Math.floor(argument) + 0;
|
||||
}
|
||||
noInline(opaqueAllTypesFloorWithoutNegativeZero);
|
||||
noOSRExitFuzzing(opaqueAllTypesFloorWithoutNegativeZero);
|
||||
|
||||
function testAllTypesWithoutNegativeZeroCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesFloorWithoutNegativeZero(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1] + 0))
|
||||
throw "Failed testAllTypesWithoutNegativeZeroCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesFloor) > 3)
|
||||
throw "We should have detected floor() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.floor() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueFloor(argument) {
|
||||
return Math.floor(argument);
|
||||
}
|
||||
noInline(opaqueFloor);
|
||||
noOSRExitFuzzing(opaqueFloor);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueFloor(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFloor) > 1)
|
||||
throw "We should have compiled a single floor for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
function checkCompileCountForUselessNegativeZero(testFunction)
|
||||
{
|
||||
if (jscOptions().useMaximalFlushInsertionPhase) {
|
||||
// If we forced a flush after the operation, the negative zero becomes
|
||||
// observable and we may be overly optimistic.
|
||||
return numberOfDFGCompiles(testFunction) <= 2;
|
||||
}
|
||||
return numberOfDFGCompiles(testFunction) <= 1;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.floor() on a completely typed input, but without negative zero.
|
||||
function testSingleTypeWithoutNegativeZeroCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueFloor(argument) {
|
||||
return Math.floor(argument) + 0;
|
||||
}
|
||||
noInline(opaqueFloor);
|
||||
noOSRExitFuzzing(opaqueFloor);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueFloor(${testCaseInput[0]}), ${testCaseInput[1]} + 0)) {
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall()";
|
||||
}
|
||||
}
|
||||
if (!checkCompileCountForUselessNegativeZero(opaqueFloor))
|
||||
throw "We should have compiled a single floor for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.floor() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueFloorOnConstant() {
|
||||
return Math.floor(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueFloorOnConstant);
|
||||
noOSRExitFuzzing(opaqueFloorOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueFloorOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFloorOnConstant) > 1)
|
||||
throw "We should have compiled a single floor for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueFloorForSideEffects(argument) {
|
||||
return Math.floor(argument);
|
||||
}
|
||||
noInline(opaqueFloorForSideEffects);
|
||||
noOSRExitFuzzing(opaqueFloorForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let floor16 = Math.floor(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueFloorForSideEffects(testObject) !== floor16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueFloorForSideEffects) > 1)
|
||||
throw "opaqueFloorForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify floor() is not subject to CSE if the argument has side effects.
|
||||
function opaqueFloorForCSE(argument) {
|
||||
return Math.floor(argument) + Math.floor(argument) + Math.floor(argument);
|
||||
}
|
||||
noInline(opaqueFloorForCSE);
|
||||
noOSRExitFuzzing(opaqueFloorForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let floor16 = Math.floor(16);
|
||||
let threeFloor16 = floor16 + floor16 + floor16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueFloorForCSE(testObject) !== threeFloor16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueFloorForCSE) > 1)
|
||||
throw "opaqueFloorForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify floor() is not subject to DCE if the argument has side effects.
|
||||
function opaqueFloorForDCE(argument) {
|
||||
Math.floor(argument);
|
||||
}
|
||||
noInline(opaqueFloorForDCE);
|
||||
noOSRExitFuzzing(opaqueFloorForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueFloorForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueFloorForDCE) > 1)
|
||||
throw "opaqueFloorForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueFloorWithException(argument) {
|
||||
let result = Math.floor(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueFloorWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let floor64 = Math.floor(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueFloorWithException(testObject) !== floor64)
|
||||
throw "Incorrect result in opaqueFloorWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueFloorWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueFloorWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,238 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let froundOfPi = Math.fround(Math.PI);
|
||||
let froundOfE = Math.fround(Math.E);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["1.", "1"],
|
||||
["Math.PI", "" + froundOfPi],
|
||||
["Math.E", "" + froundOfE],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"" + Math.PI + "\"", "" + froundOfPi],
|
||||
["{ valueOf: () => { return Math.E; } }", "" + froundOfE],
|
||||
["{ valueOf: () => { return 1; } }", "1"],
|
||||
["{ valueOf: () => { return Math.PI; } }", "" + froundOfPi],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
// Test Math.fround() without arguments.
|
||||
function opaqueFRoundNoArgument() {
|
||||
return Math.fround();
|
||||
}
|
||||
noInline(opaqueFRoundNoArgument);
|
||||
noOSRExitFuzzing(opaqueFRoundNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueFRoundNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueFRoundNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFRoundNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
// Test Math.fround() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesFround(argument) {
|
||||
return Math.fround(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesFround);
|
||||
noOSRExitFuzzing(opaqueAllTypesFround);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesFround(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesFround) > 2)
|
||||
throw "We should have detected fround() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.fround() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueFround(argument) {
|
||||
return Math.fround(argument);
|
||||
}
|
||||
noInline(opaqueFround);
|
||||
noOSRExitFuzzing(opaqueFround);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueFround(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFround) > 1)
|
||||
throw "We should have compiled a single fround for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.fround() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueFroundOnConstant() {
|
||||
return Math.fround(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueFroundOnConstant);
|
||||
noOSRExitFuzzing(opaqueFroundOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueFroundOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueFroundOnConstant) > 1)
|
||||
throw "We should have compiled a single fround for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueFroundForSideEffects(argument) {
|
||||
return Math.fround(argument);
|
||||
}
|
||||
noInline(opaqueFroundForSideEffects);
|
||||
noOSRExitFuzzing(opaqueFroundForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let fround16 = Math.fround(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueFroundForSideEffects(testObject) !== fround16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueFroundForSideEffects) > 1)
|
||||
throw "opaqueFroundForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify fround() is not subject to CSE if the argument has side effects.
|
||||
function opaqueFroundForCSE(argument) {
|
||||
return Math.fround(argument) + Math.fround(argument) + Math.fround(argument);
|
||||
}
|
||||
noInline(opaqueFroundForCSE);
|
||||
noOSRExitFuzzing(opaqueFroundForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let fround16 = Math.fround(16);
|
||||
let threeFround16 = fround16 + fround16 + fround16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueFroundForCSE(testObject) !== threeFround16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueFroundForCSE) > 1)
|
||||
throw "opaqueFroundForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify fround() is not subject to DCE if the argument has side effects.
|
||||
function opaqueFroundForDCE(argument) {
|
||||
Math.fround(argument);
|
||||
}
|
||||
noInline(opaqueFroundForDCE);
|
||||
noOSRExitFuzzing(opaqueFroundForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueFroundForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueFroundForDCE) > 1)
|
||||
throw "opaqueFroundForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueFroundWithException(argument) {
|
||||
let result = Math.fround(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueFroundWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let fround64 = Math.fround(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueFroundWithException(testObject) !== fround64)
|
||||
throw "Incorrect result in opaqueFroundWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueFroundWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueFroundWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,240 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ runNoCJITValidatePhases
|
||||
//@ runFTLNoCJITValidate
|
||||
"use strict";
|
||||
|
||||
let logOfFour = Math.log(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "-Infinity"],
|
||||
["0", "-Infinity"],
|
||||
["-0.", "-Infinity"],
|
||||
["1.", "0"],
|
||||
["4", "" + logOfFour],
|
||||
["Math.E", "1"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + logOfFour],
|
||||
["{ valueOf: () => { return Math.E; } }", "1"],
|
||||
["{ valueOf: () => { return 1; } }", "0"],
|
||||
["{ valueOf: () => { return 4; } }", "" + logOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.log() without arguments.
|
||||
function opaqueLogNoArgument() {
|
||||
return Math.log();
|
||||
}
|
||||
noInline(opaqueLogNoArgument);
|
||||
noOSRExitFuzzing(opaqueLogNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueLogNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueLogNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLogNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.log() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesLog(argument) {
|
||||
return Math.log(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesLog);
|
||||
noOSRExitFuzzing(opaqueAllTypesLog);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesLog(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesLog) > 2)
|
||||
throw "We should have detected log() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.log() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLog(argument) {
|
||||
return Math.log(argument);
|
||||
}
|
||||
noInline(opaqueLog);
|
||||
noOSRExitFuzzing(opaqueLog);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLog(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog) > 1)
|
||||
throw "We should have compiled a single log for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.log() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLogOnConstant() {
|
||||
return Math.log(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueLogOnConstant);
|
||||
noOSRExitFuzzing(opaqueLogOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLogOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLogOnConstant) > 1)
|
||||
throw "We should have compiled a single log for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueLogForSideEffects(argument) {
|
||||
return Math.log(argument);
|
||||
}
|
||||
noInline(opaqueLogForSideEffects);
|
||||
noOSRExitFuzzing(opaqueLogForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let log16 = Math.log(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLogForSideEffects(testObject) !== log16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueLogForSideEffects) > 1)
|
||||
throw "opaqueLogForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify log() is not subject to CSE if the argument has side effects.
|
||||
function opaqueLogForCSE(argument) {
|
||||
return Math.log(argument) + Math.log(argument) + Math.log(argument);
|
||||
}
|
||||
noInline(opaqueLogForCSE);
|
||||
noOSRExitFuzzing(opaqueLogForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let log16 = Math.log(16);
|
||||
let threeLog16 = log16 + log16 + log16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLogForCSE(testObject) !== threeLog16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueLogForCSE) > 1)
|
||||
throw "opaqueLogForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify log() is not subject to DCE if the argument has side effects.
|
||||
function opaqueLogForDCE(argument) {
|
||||
Math.log(argument);
|
||||
}
|
||||
noInline(opaqueLogForDCE);
|
||||
noOSRExitFuzzing(opaqueLogForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueLogForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueLogForDCE) > 1)
|
||||
throw "opaqueLogForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueLogWithException(argument) {
|
||||
let result = Math.log(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueLogWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let log64 = Math.log(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueLogWithException(testObject) !== log64)
|
||||
throw "Incorrect result in opaqueLogWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueLogWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueLogWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,237 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let log10OfHalf = Math.log10(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "-Infinity"],
|
||||
["1", "0"],
|
||||
["0", "-Infinity"],
|
||||
["-0.", "-Infinity"],
|
||||
["0.5", "" + log10OfHalf],
|
||||
["Math.PI", "" + Math.log10(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + log10OfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + log10OfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.log10() without arguments.
|
||||
function opaqueLog10NoArgument() {
|
||||
return Math.log10();
|
||||
}
|
||||
noInline(opaqueLog10NoArgument);
|
||||
noOSRExitFuzzing(opaqueLog10NoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueLog10NoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueLog10NoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog10NoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.log10() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesLog10(argument) {
|
||||
return Math.log10(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesLog10);
|
||||
noOSRExitFuzzing(opaqueAllTypesLog10);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesLog10(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesLog10) > 2)
|
||||
throw "We should have detected log10() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.log10() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLog10(argument) {
|
||||
return Math.log10(argument);
|
||||
}
|
||||
noInline(opaqueLog10);
|
||||
noOSRExitFuzzing(opaqueLog10);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLog10(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog10) > 1)
|
||||
throw "We should have compiled a single log10 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.log10() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLog10OnConstant() {
|
||||
return Math.log10(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueLog10OnConstant);
|
||||
noOSRExitFuzzing(opaqueLog10OnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLog10OnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog10OnConstant) > 1)
|
||||
throw "We should have compiled a single log10 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueLog10ForSideEffects(argument) {
|
||||
return Math.log10(argument);
|
||||
}
|
||||
noInline(opaqueLog10ForSideEffects);
|
||||
noOSRExitFuzzing(opaqueLog10ForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let log10Result = Math.log10(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLog10ForSideEffects(testObject) !== log10Result)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueLog10ForSideEffects) > 1)
|
||||
throw "opaqueLog10ForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify log10() is not subject to CSE if the argument has side effects.
|
||||
function opaqueLog10ForCSE(argument) {
|
||||
return Math.log10(argument) + Math.log10(argument) + Math.log10(argument);
|
||||
}
|
||||
noInline(opaqueLog10ForCSE);
|
||||
noOSRExitFuzzing(opaqueLog10ForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let log10Result = Math.log10(0.2);
|
||||
let threelog10Result = log10Result + log10Result + log10Result;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLog10ForCSE(testObject) !== threelog10Result)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueLog10ForCSE) > 1)
|
||||
throw "opaqueLog10ForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify log10() is not subject to DCE if the argument has side effects.
|
||||
function opaqueLog10ForDCE(argument) {
|
||||
Math.log10(argument);
|
||||
}
|
||||
noInline(opaqueLog10ForDCE);
|
||||
noOSRExitFuzzing(opaqueLog10ForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueLog10ForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueLog10ForDCE) > 1)
|
||||
throw "opaqueLog10ForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueLog10WithException(argument) {
|
||||
let result = Math.log10(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueLog10WithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let log10Result = Math.log10(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueLog10WithException(testObject) !== log10Result)
|
||||
throw "Incorrect result in opaqueLog10WithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueLog10WithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueLog10WithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,237 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let log2OfHalf = Math.log2(0.5);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "-Infinity"],
|
||||
["1", "0"],
|
||||
["0", "-Infinity"],
|
||||
["-0.", "-Infinity"],
|
||||
["0.5", "" + log2OfHalf],
|
||||
["Math.PI", "" + Math.log2(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"0.5\"", "" + log2OfHalf],
|
||||
["{ valueOf: () => { return 0.5; } }", "" + log2OfHalf],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.log2() without arguments.
|
||||
function opaqueLog2NoArgument() {
|
||||
return Math.log2();
|
||||
}
|
||||
noInline(opaqueLog2NoArgument);
|
||||
noOSRExitFuzzing(opaqueLog2NoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueLog2NoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueLog2NoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog2NoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.log2() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesLog2(argument) {
|
||||
return Math.log2(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesLog2);
|
||||
noOSRExitFuzzing(opaqueAllTypesLog2);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesLog2(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesLog2) > 2)
|
||||
throw "We should have detected log2() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.log2() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLog2(argument) {
|
||||
return Math.log2(argument);
|
||||
}
|
||||
noInline(opaqueLog2);
|
||||
noOSRExitFuzzing(opaqueLog2);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLog2(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog2) > 1)
|
||||
throw "We should have compiled a single log2 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.log2() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueLog2OnConstant() {
|
||||
return Math.log2(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueLog2OnConstant);
|
||||
noOSRExitFuzzing(opaqueLog2OnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueLog2OnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueLog2OnConstant) > 1)
|
||||
throw "We should have compiled a single log2 for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueLog2ForSideEffects(argument) {
|
||||
return Math.log2(argument);
|
||||
}
|
||||
noInline(opaqueLog2ForSideEffects);
|
||||
noOSRExitFuzzing(opaqueLog2ForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let log2Result = Math.log2(0.2);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLog2ForSideEffects(testObject) !== log2Result)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueLog2ForSideEffects) > 1)
|
||||
throw "opaqueLog2ForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify log2() is not subject to CSE if the argument has side effects.
|
||||
function opaqueLog2ForCSE(argument) {
|
||||
return Math.log2(argument) + Math.log2(argument) + Math.log2(argument);
|
||||
}
|
||||
noInline(opaqueLog2ForCSE);
|
||||
noOSRExitFuzzing(opaqueLog2ForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
let log2Result = Math.log2(0.2);
|
||||
let threelog2Result = log2Result + log2Result + log2Result;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueLog2ForCSE(testObject) !== threelog2Result)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueLog2ForCSE) > 1)
|
||||
throw "opaqueLog2ForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify log2() is not subject to DCE if the argument has side effects.
|
||||
function opaqueLog2ForDCE(argument) {
|
||||
Math.log2(argument);
|
||||
}
|
||||
noInline(opaqueLog2ForDCE);
|
||||
noOSRExitFuzzing(opaqueLog2ForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 0.2; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueLog2ForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueLog2ForDCE) > 1)
|
||||
throw "opaqueLog2ForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueLog2WithException(argument) {
|
||||
let result = Math.log2(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueLog2WithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 0.1; } };
|
||||
let log2Result = Math.log2(0.1);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueLog2WithException(testObject) !== log2Result)
|
||||
throw "Incorrect result in opaqueLog2WithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 0.1; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueLog2WithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueLog2WithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,284 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "-0"],
|
||||
["0", "-0"],
|
||||
["-0.", "0"],
|
||||
["0.5", "-0.5"],
|
||||
["-0.5", "0.5"],
|
||||
["4", "-4"],
|
||||
["42.1", "-42.1"],
|
||||
["42.5", "-42.5"],
|
||||
["42.9", "-42.9"],
|
||||
["-42.1", "42.1"],
|
||||
["-42.5", "42.5"],
|
||||
["-42.9", "42.9"],
|
||||
["Math.PI", "-Math.PI"],
|
||||
["Infinity", "-Infinity"],
|
||||
["-Infinity", "Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "-4"],
|
||||
["\"42.5\"", "-42.5"],
|
||||
["{ valueOf: () => { return 4; } }", "-4"],
|
||||
["{ valueOf: () => { return 0; } }", "-0"],
|
||||
["{ valueOf: () => { return -0; } }", "0"],
|
||||
["{ valueOf: () => { return 0.5; } }", "-0.5"],
|
||||
["{ valueOf: () => { return -0.5; } }", "0.5"],
|
||||
["{ valueOf: () => { return Number.MIN_SAFE_INTEGER; } }", "9007199254740991"],
|
||||
["{ valueOf: () => { return Number.MAX_SAFE_INTEGER; } }", "-9007199254740991"],
|
||||
["{ valueOf: () => { return 0x80000000|0; } }", "2147483648"],
|
||||
["{ valueOf: () => { return 0x7fffffff|0; } }", "-2147483647"],
|
||||
["{ valueOf: () => { return (0x80000000|0) - 0.5; } }", "2147483648.5"],
|
||||
["{ valueOf: () => { return (0x7fffffff|0) + 0.5; } }", "-2147483647.5"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && (1 / expected) !== (1 / result))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test negate with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesNegate(argument) {
|
||||
return -(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesNegate);
|
||||
noOSRExitFuzzing(opaqueAllTypesNegate);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesNegate(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesNegate) > 3)
|
||||
throw "We should have detected negate was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Polymorphic input but negative zero is not observable.
|
||||
function opaqueAllTypesNegateWithoutNegativeZero(argument) {
|
||||
return -(argument) + 0;
|
||||
}
|
||||
noInline(opaqueAllTypesNegateWithoutNegativeZero);
|
||||
noOSRExitFuzzing(opaqueAllTypesNegateWithoutNegativeZero);
|
||||
|
||||
function testAllTypesWithoutNegativeZeroCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesNegateWithoutNegativeZero(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1] + 0))
|
||||
throw "Failed testAllTypesWithoutNegativeZeroCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesNegate) > 3)
|
||||
throw "We should have detected negate was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test negate on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueNegate(argument) {
|
||||
return -(argument);
|
||||
}
|
||||
noInline(opaqueNegate);
|
||||
noOSRExitFuzzing(opaqueNegate);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueNegate(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueNegate) > 1)
|
||||
throw "Failed testSingleTypeCall(). We should have compiled a single negate for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
function checkCompileCountForUselessNegativeZero(testFunction)
|
||||
{
|
||||
if (jscOptions().useMaximalFlushInsertionPhase) {
|
||||
// If we forced a flush after the operation, the negative zero becomes
|
||||
// observable and we may be overly optimistic.
|
||||
return numberOfDFGCompiles(testFunction) <= 2;
|
||||
}
|
||||
return numberOfDFGCompiles(testFunction) <= 1;
|
||||
}
|
||||
|
||||
|
||||
// Test negate on a completely typed input, but without negative zero.
|
||||
function testSingleTypeWithoutNegativeZeroCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueNegate(argument) {
|
||||
return -(argument) + 0;
|
||||
}
|
||||
noInline(opaqueNegate);
|
||||
noOSRExitFuzzing(opaqueNegate);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueNegate(${testCaseInput[0]}), ${testCaseInput[1]} + 0)) {
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall()";
|
||||
}
|
||||
}
|
||||
if (!checkCompileCountForUselessNegativeZero(opaqueNegate))
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall(). We should have compiled a single negate for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test negate on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueNegateOnConstant() {
|
||||
return -(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueNegateOnConstant);
|
||||
noOSRExitFuzzing(opaqueNegateOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueNegateOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueNegateOnConstant) > 1)
|
||||
throw "Failed testConstant(). We should have compiled a single negate for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueNegateForSideEffects(argument) {
|
||||
return -(argument);
|
||||
}
|
||||
noInline(opaqueNegateForSideEffects);
|
||||
noOSRExitFuzzing(opaqueNegateForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueNegateForSideEffects(testObject) !== -16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueNegateForSideEffects) > 1)
|
||||
throw "opaqueNegateForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify negate is not subject to CSE if the argument has side effects.
|
||||
function opaqueNegateForCSE(argument) {
|
||||
return -(argument) + -(argument) + -(argument);
|
||||
}
|
||||
noInline(opaqueNegateForCSE);
|
||||
noOSRExitFuzzing(opaqueNegateForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueNegateForCSE(testObject) !== -48)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueNegateForCSE) > 1)
|
||||
throw "opaqueNegateForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify negate is not subject to DCE if the argument has side effects.
|
||||
function opaqueNegateForDCE(argument) {
|
||||
-(argument);
|
||||
}
|
||||
noInline(opaqueNegateForDCE);
|
||||
noOSRExitFuzzing(opaqueNegateForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueNegateForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueNegateForDCE) > 1)
|
||||
throw "opaqueNegateForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueNegateWithException(argument) {
|
||||
let result = -(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueNegateWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueNegateWithException(testObject) !== -64)
|
||||
throw "Incorrect result in opaqueNegateWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueNegateWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueNegateWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,308 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "1"],
|
||||
["-0.5", "-0"],
|
||||
["4", "4"],
|
||||
["42.1", "42"],
|
||||
["42.5", "43"],
|
||||
["42.9", "43"],
|
||||
["-42.1", "-42"],
|
||||
["-42.5", "-42"],
|
||||
["-42.9", "-43"],
|
||||
["Math.PI", "3"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "4"],
|
||||
["\"42.5\"", "43"],
|
||||
["{ valueOf: () => { return 4; } }", "4"],
|
||||
["{ valueOf: () => { return 0; } }", "0"],
|
||||
["{ valueOf: () => { return -0; } }", "-0"],
|
||||
["{ valueOf: () => { return 0.5; } }", "1"],
|
||||
["{ valueOf: () => { return -0.5; } }", "-0"],
|
||||
["{ valueOf: () => { return Number.MIN_SAFE_INTEGER; } }", "-9007199254740991"],
|
||||
["{ valueOf: () => { return Number.MAX_SAFE_INTEGER; } }", "9007199254740991"],
|
||||
["{ valueOf: () => { return 0x80000000|0; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return 0x7fffffff|0; } }", "2147483647"],
|
||||
["{ valueOf: () => { return (0x80000000|0) - 0.5; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return (0x7fffffff|0) + 0.5; } }", "2147483648"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && (1 / expected) !== (1 / result))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.round() without arguments.
|
||||
function opaqueRoundNoArgument() {
|
||||
return Math.round();
|
||||
}
|
||||
noInline(opaqueRoundNoArgument);
|
||||
noOSRExitFuzzing(opaqueRoundNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueRoundNoArgument();
|
||||
if (!isIdentical(output, NaN)) {
|
||||
throw "Failed opaqueRoundNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueRoundNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.round() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesRound(argument) {
|
||||
return Math.round(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesRound);
|
||||
noOSRExitFuzzing(opaqueAllTypesRound);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesRound(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesRound) > 3)
|
||||
throw "We should have detected round() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Polymorphic input but negative zero is not observable.
|
||||
function opaqueAllTypesRoundWithoutNegativeZero(argument) {
|
||||
return Math.round(argument) + 0;
|
||||
}
|
||||
noInline(opaqueAllTypesRoundWithoutNegativeZero);
|
||||
noOSRExitFuzzing(opaqueAllTypesRoundWithoutNegativeZero);
|
||||
|
||||
function testAllTypesWithoutNegativeZeroCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesRoundWithoutNegativeZero(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1] + 0))
|
||||
throw "Failed testAllTypesWithoutNegativeZeroCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesRound) > 3)
|
||||
throw "We should have detected round() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.round() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueRound(argument) {
|
||||
return Math.round(argument);
|
||||
}
|
||||
noInline(opaqueRound);
|
||||
noOSRExitFuzzing(opaqueRound);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueRound(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueRound) > 1)
|
||||
throw "We should have compiled a single round for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
function checkCompileCountForUselessNegativeZero(testFunction)
|
||||
{
|
||||
if (jscOptions().useMaximalFlushInsertionPhase) {
|
||||
// If we forced a flush after the operation, the negative zero becomes
|
||||
// observable and we may be overly optimistic.
|
||||
return numberOfDFGCompiles(testFunction) <= 2;
|
||||
}
|
||||
return numberOfDFGCompiles(testFunction) <= 1;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.round() on a completely typed input, but without negative zero.
|
||||
function testSingleTypeWithoutNegativeZeroCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueRound(argument) {
|
||||
return Math.round(argument) + 0;
|
||||
}
|
||||
noInline(opaqueRound);
|
||||
noOSRExitFuzzing(opaqueRound);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueRound(${testCaseInput[0]}), ${testCaseInput[1]} + 0)) {
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall()";
|
||||
}
|
||||
}
|
||||
if (!checkCompileCountForUselessNegativeZero(opaqueRound))
|
||||
throw "We should have compiled a single round for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.round() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueRoundOnConstant() {
|
||||
return Math.round(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueRoundOnConstant);
|
||||
noOSRExitFuzzing(opaqueRoundOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueRoundOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueRoundOnConstant) > 1)
|
||||
throw "We should have compiled a single round for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueRoundForSideEffects(argument) {
|
||||
return Math.round(argument);
|
||||
}
|
||||
noInline(opaqueRoundForSideEffects);
|
||||
noOSRExitFuzzing(opaqueRoundForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let round16 = Math.round(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueRoundForSideEffects(testObject) !== round16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueRoundForSideEffects) > 1)
|
||||
throw "opaqueRoundForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify round() is not subject to CSE if the argument has side effects.
|
||||
function opaqueRoundForCSE(argument) {
|
||||
return Math.round(argument) + Math.round(argument) + Math.round(argument);
|
||||
}
|
||||
noInline(opaqueRoundForCSE);
|
||||
noOSRExitFuzzing(opaqueRoundForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let round16 = Math.round(16);
|
||||
let threeRound16 = round16 + round16 + round16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueRoundForCSE(testObject) !== threeRound16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueRoundForCSE) > 1)
|
||||
throw "opaqueRoundForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify round() is not subject to DCE if the argument has side effects.
|
||||
function opaqueRoundForDCE(argument) {
|
||||
Math.round(argument);
|
||||
}
|
||||
noInline(opaqueRoundForDCE);
|
||||
noOSRExitFuzzing(opaqueRoundForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueRoundForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueRoundForDCE) > 1)
|
||||
throw "opaqueRoundForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueRoundWithException(argument) {
|
||||
let result = Math.round(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueRoundWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let round64 = Math.round(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueRoundWithException(testObject) !== round64)
|
||||
throw "Incorrect result in opaqueRoundWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueRoundWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueRoundWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let sinOfFour = Math.sin(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + sinOfFour],
|
||||
["Math.PI", "" + Math.sin(Math.PI)],
|
||||
["Infinity", "NaN"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + sinOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + sinOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.sin() without arguments.
|
||||
function opaqueSinNoArgument() {
|
||||
return Math.sin();
|
||||
}
|
||||
noInline(opaqueSinNoArgument);
|
||||
noOSRExitFuzzing(opaqueSinNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueSinNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueSinNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSinNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.sin() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesSin(argument) {
|
||||
return Math.sin(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesSin);
|
||||
noOSRExitFuzzing(opaqueAllTypesSin);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesSin(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesSin) > 2)
|
||||
throw "We should have detected sin() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.sin() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSin(argument) {
|
||||
return Math.sin(argument);
|
||||
}
|
||||
noInline(opaqueSin);
|
||||
noOSRExitFuzzing(opaqueSin);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSin(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSin) > 1)
|
||||
throw "We should have compiled a single sin for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.sin() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSinOnConstant() {
|
||||
return Math.sin(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueSinOnConstant);
|
||||
noOSRExitFuzzing(opaqueSinOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSinOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSinOnConstant) > 1)
|
||||
throw "We should have compiled a single sin for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueSinForSideEffects(argument) {
|
||||
return Math.sin(argument);
|
||||
}
|
||||
noInline(opaqueSinForSideEffects);
|
||||
noOSRExitFuzzing(opaqueSinForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let sin16 = Math.sin(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSinForSideEffects(testObject) !== sin16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueSinForSideEffects) > 1)
|
||||
throw "opaqueSinForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify sin() is not subject to CSE if the argument has side effects.
|
||||
function opaqueSinForCSE(argument) {
|
||||
return Math.sin(argument) + Math.sin(argument) + Math.sin(argument);
|
||||
}
|
||||
noInline(opaqueSinForCSE);
|
||||
noOSRExitFuzzing(opaqueSinForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let sin16 = Math.sin(16);
|
||||
let threeSin16 = sin16 + sin16 + sin16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSinForCSE(testObject) !== threeSin16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueSinForCSE) > 1)
|
||||
throw "opaqueSinForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify sin() is not subject to DCE if the argument has side effects.
|
||||
function opaqueSinForDCE(argument) {
|
||||
Math.sin(argument);
|
||||
}
|
||||
noInline(opaqueSinForDCE);
|
||||
noOSRExitFuzzing(opaqueSinForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueSinForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueSinForDCE) > 1)
|
||||
throw "opaqueSinForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueSinWithException(argument) {
|
||||
let result = Math.sin(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueSinWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let sin64 = Math.sin(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueSinWithException(testObject) !== sin64)
|
||||
throw "Incorrect result in opaqueSinWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueSinWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueSinWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let sinhOfFour = Math.sinh(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + sinhOfFour],
|
||||
["Math.PI", "" + Math.sinh(Math.PI)],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + sinhOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + sinhOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.sinh() without arguments.
|
||||
function opaqueSinhNoArgument() {
|
||||
return Math.sinh();
|
||||
}
|
||||
noInline(opaqueSinhNoArgument);
|
||||
noOSRExitFuzzing(opaqueSinhNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueSinhNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueSinhNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSinhNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.sinh() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesSinh(argument) {
|
||||
return Math.sinh(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesSinh);
|
||||
noOSRExitFuzzing(opaqueAllTypesSinh);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesSinh(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesSinh) > 2)
|
||||
throw "We should have detected sinh() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.sinh() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSinh(argument) {
|
||||
return Math.sinh(argument);
|
||||
}
|
||||
noInline(opaqueSinh);
|
||||
noOSRExitFuzzing(opaqueSinh);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSinh(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSinh) > 1)
|
||||
throw "We should have compiled a single sinh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.sinh() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSinhOnConstant() {
|
||||
return Math.sinh(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueSinhOnConstant);
|
||||
noOSRExitFuzzing(opaqueSinhOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSinhOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSinhOnConstant) > 1)
|
||||
throw "We should have compiled a single sinh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueSinhForSideEffects(argument) {
|
||||
return Math.sinh(argument);
|
||||
}
|
||||
noInline(opaqueSinhForSideEffects);
|
||||
noOSRExitFuzzing(opaqueSinhForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let sinh16 = Math.sinh(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSinhForSideEffects(testObject) !== sinh16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueSinhForSideEffects) > 1)
|
||||
throw "opaqueSinhForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify sinh() is not subject to CSE if the argument has side effects.
|
||||
function opaqueSinhForCSE(argument) {
|
||||
return Math.sinh(argument) + Math.sinh(argument) + Math.sinh(argument);
|
||||
}
|
||||
noInline(opaqueSinhForCSE);
|
||||
noOSRExitFuzzing(opaqueSinhForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let sinh16 = Math.sinh(16);
|
||||
let threeSinh16 = sinh16 + sinh16 + sinh16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSinhForCSE(testObject) !== threeSinh16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueSinhForCSE) > 1)
|
||||
throw "opaqueSinhForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify sinh() is not subject to DCE if the argument has side effects.
|
||||
function opaqueSinhForDCE(argument) {
|
||||
Math.sinh(argument);
|
||||
}
|
||||
noInline(opaqueSinhForDCE);
|
||||
noOSRExitFuzzing(opaqueSinhForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueSinhForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueSinhForDCE) > 1)
|
||||
throw "opaqueSinhForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueSinhWithException(argument) {
|
||||
let result = Math.sinh(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueSinhWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let sinh64 = Math.sinh(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueSinhWithException(testObject) !== sinh64)
|
||||
throw "Incorrect result in opaqueSinhWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueSinhWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueSinhWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,230 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0."],
|
||||
["4", "2"],
|
||||
["42.5", "6.519202405202649"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "2"],
|
||||
["{ valueOf: () => { return 4; } }", "2"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && 1 / expected === -Infinity && 1 / result !== -Infinity)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.sqrt() without arguments.
|
||||
function opaqueSqrtNoArgument() {
|
||||
return Math.sqrt();
|
||||
}
|
||||
noInline(opaqueSqrtNoArgument);
|
||||
noOSRExitFuzzing(opaqueSqrtNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueSqrtNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueSqrtNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSqrtNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.sqrt() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesSqrt(argument) {
|
||||
return Math.sqrt(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesSqrt);
|
||||
noOSRExitFuzzing(opaqueAllTypesSqrt);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesSqrt(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesSqrt) > 2)
|
||||
throw "We should have detected sqrt() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.sqrt() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSqrt(argument) {
|
||||
return Math.sqrt(argument);
|
||||
}
|
||||
noInline(opaqueSqrt);
|
||||
noOSRExitFuzzing(opaqueSqrt);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSqrt(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSqrt) > 1)
|
||||
throw "We should have compiled a single sqrt for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.sqrt() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueSqrtOnConstant() {
|
||||
return Math.sqrt(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueSqrtOnConstant);
|
||||
noOSRExitFuzzing(opaqueSqrtOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueSqrtOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueSqrtOnConstant) > 1)
|
||||
throw "We should have compiled a single sqrt for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueSqrtForSideEffects(argument) {
|
||||
return Math.sqrt(argument);
|
||||
}
|
||||
noInline(opaqueSqrtForSideEffects);
|
||||
noOSRExitFuzzing(opaqueSqrtForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSqrtForSideEffects(testObject) !== 4)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueSqrtForSideEffects) > 1)
|
||||
throw "opaqueSqrtForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify sqrt() is not subject to CSE if the argument has side effects.
|
||||
function opaqueSqrtForCSE(argument) {
|
||||
return Math.sqrt(argument) + Math.sqrt(argument) + Math.sqrt(argument);
|
||||
}
|
||||
noInline(opaqueSqrtForCSE);
|
||||
noOSRExitFuzzing(opaqueSqrtForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueSqrtForCSE(testObject) !== 12)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueSqrtForCSE) > 1)
|
||||
throw "opaqueSqrtForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify sqrt() is not subject to DCE if the argument has side effects.
|
||||
function opaqueSqrtForDCE(argument) {
|
||||
Math.sqrt(argument);
|
||||
}
|
||||
noInline(opaqueSqrtForDCE);
|
||||
noOSRExitFuzzing(opaqueSqrtForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueSqrtForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueSqrtForDCE) > 1)
|
||||
throw "opaqueSqrtForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueSqrtWithException(argument) {
|
||||
let result = Math.sqrt(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueSqrtWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueSqrtWithException(testObject) !== 8)
|
||||
throw "Incorrect result in opaqueSqrtWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueSqrtWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueSqrtWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let tanOfFour = Math.tan(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + tanOfFour],
|
||||
["Math.PI", "" + Math.tan(Math.PI)],
|
||||
["Infinity", "NaN"],
|
||||
["-Infinity", "NaN"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + tanOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + tanOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.tan() without arguments.
|
||||
function opaqueTanNoArgument() {
|
||||
return Math.tan();
|
||||
}
|
||||
noInline(opaqueTanNoArgument);
|
||||
noOSRExitFuzzing(opaqueTanNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueTanNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueTanNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTanNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.tan() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesTan(argument) {
|
||||
return Math.tan(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesTan);
|
||||
noOSRExitFuzzing(opaqueAllTypesTan);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesTan(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesTan) > 2)
|
||||
throw "We should have detected tan() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.tan() on a completely typed input. Every call see only one type.
|
||||
function testTangleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTan(argument) {
|
||||
return Math.tan(argument);
|
||||
}
|
||||
noInline(opaqueTan);
|
||||
noOSRExitFuzzing(opaqueTan);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTan(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testTangleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTan) > 1)
|
||||
throw "We should have compiled a tangle tan for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testTangleTypeCall();
|
||||
|
||||
|
||||
// Test Math.tan() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTanOnConstant() {
|
||||
return Math.tan(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueTanOnConstant);
|
||||
noOSRExitFuzzing(opaqueTanOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTanOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTanOnConstant) > 1)
|
||||
throw "We should have compiled a tangle tan for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueTanForSideEffects(argument) {
|
||||
return Math.tan(argument);
|
||||
}
|
||||
noInline(opaqueTanForSideEffects);
|
||||
noOSRExitFuzzing(opaqueTanForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let tan16 = Math.tan(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTanForSideEffects(testObject) !== tan16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueTanForSideEffects) > 1)
|
||||
throw "opaqueTanForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify tan() is not subject to CSE if the argument has side effects.
|
||||
function opaqueTanForCSE(argument) {
|
||||
return Math.tan(argument) + Math.tan(argument) + Math.tan(argument);
|
||||
}
|
||||
noInline(opaqueTanForCSE);
|
||||
noOSRExitFuzzing(opaqueTanForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let tan16 = Math.tan(16);
|
||||
let threeTan16 = tan16 + tan16 + tan16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTanForCSE(testObject) !== threeTan16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueTanForCSE) > 1)
|
||||
throw "opaqueTanForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify tan() is not subject to DCE if the argument has side effects.
|
||||
function opaqueTanForDCE(argument) {
|
||||
Math.tan(argument);
|
||||
}
|
||||
noInline(opaqueTanForDCE);
|
||||
noOSRExitFuzzing(opaqueTanForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueTanForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueTanForDCE) > 1)
|
||||
throw "opaqueTanForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueTanWithException(argument) {
|
||||
let result = Math.tan(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueTanWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let tan64 = Math.tan(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueTanWithException(testObject) !== tan64)
|
||||
throw "Incorrect result in opaqueTanWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueTanWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueTanWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,236 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let tanhOfFour = Math.tanh(4);
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["4", "" + tanhOfFour],
|
||||
["Math.PI", "" + Math.tanh(Math.PI)],
|
||||
["Infinity", "1"],
|
||||
["-Infinity", "-1"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "" + tanhOfFour],
|
||||
["{ valueOf: () => { return 4; } }", "" + tanhOfFour],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected)
|
||||
return (1 / expected) === (1 / result);
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.tanh() without arguments.
|
||||
function opaqueTanhNoArgument() {
|
||||
return Math.tanh();
|
||||
}
|
||||
noInline(opaqueTanhNoArgument);
|
||||
noOSRExitFuzzing(opaqueTanhNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueTanhNoArgument();
|
||||
if (output === output) {
|
||||
throw "Failed opaqueTanhNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTanhNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.tanh() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesTanh(argument) {
|
||||
return Math.tanh(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesTanh);
|
||||
noOSRExitFuzzing(opaqueAllTypesTanh);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesTanh(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesTanh) > 2)
|
||||
throw "We should have detected tanh() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Test Math.tanh() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTanh(argument) {
|
||||
return Math.tanh(argument);
|
||||
}
|
||||
noInline(opaqueTanh);
|
||||
noOSRExitFuzzing(opaqueTanh);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTanh(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTanh) > 1)
|
||||
throw "We should have compiled a single tanh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
// Test Math.tanh() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTanhOnConstant() {
|
||||
return Math.tanh(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueTanhOnConstant);
|
||||
noOSRExitFuzzing(opaqueTanhOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTanhOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTanhOnConstant) > 1)
|
||||
throw "We should have compiled a single tanh for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueTanhForSideEffects(argument) {
|
||||
return Math.tanh(argument);
|
||||
}
|
||||
noInline(opaqueTanhForSideEffects);
|
||||
noOSRExitFuzzing(opaqueTanhForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let tanh16 = Math.tanh(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTanhForSideEffects(testObject) !== tanh16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueTanhForSideEffects) > 1)
|
||||
throw "opaqueTanhForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify tanh() is not subject to CSE if the argument has side effects.
|
||||
function opaqueTanhForCSE(argument) {
|
||||
return Math.tanh(argument) + Math.tanh(argument) + Math.tanh(argument);
|
||||
}
|
||||
noInline(opaqueTanhForCSE);
|
||||
noOSRExitFuzzing(opaqueTanhForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let tanh16 = Math.tanh(16);
|
||||
let threeTanh16 = tanh16 + tanh16 + tanh16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTanhForCSE(testObject) !== threeTanh16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueTanhForCSE) > 1)
|
||||
throw "opaqueTanhForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify tanh() is not subject to DCE if the argument has side effects.
|
||||
function opaqueTanhForDCE(argument) {
|
||||
Math.tanh(argument);
|
||||
}
|
||||
noInline(opaqueTanhForDCE);
|
||||
noOSRExitFuzzing(opaqueTanhForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueTanhForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueTanhForDCE) > 1)
|
||||
throw "opaqueTanhForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueTanhWithException(argument) {
|
||||
let result = Math.tanh(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueTanhWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let tanh64 = Math.tanh(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueTanhWithException(testObject) !== tanh64)
|
||||
throw "Incorrect result in opaqueTanhWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueTanhWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueTanhWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,308 +0,0 @@
|
|||
//@ skip if not $jitTests
|
||||
//@ defaultNoEagerRun
|
||||
"use strict";
|
||||
|
||||
let validInputTestCases = [
|
||||
// input as string, expected result as string.
|
||||
["undefined", "NaN"],
|
||||
["null", "0"],
|
||||
["0", "0"],
|
||||
["-0.", "-0"],
|
||||
["0.5", "0"],
|
||||
["-0.5", "-0"],
|
||||
["4", "4"],
|
||||
["42.1", "42"],
|
||||
["42.5", "42"],
|
||||
["42.9", "42"],
|
||||
["-42.1", "-42"],
|
||||
["-42.5", "-42"],
|
||||
["-42.9", "-42"],
|
||||
["Math.PI", "3"],
|
||||
["Infinity", "Infinity"],
|
||||
["-Infinity", "-Infinity"],
|
||||
["NaN", "NaN"],
|
||||
["\"WebKit\"", "NaN"],
|
||||
["\"4\"", "4"],
|
||||
["\"42.5\"", "42"],
|
||||
["{ valueOf: () => { return 4; } }", "4"],
|
||||
["{ valueOf: () => { return 0; } }", "0"],
|
||||
["{ valueOf: () => { return -0; } }", "-0"],
|
||||
["{ valueOf: () => { return 0.5; } }", "0"],
|
||||
["{ valueOf: () => { return -0.5; } }", "-0"],
|
||||
["{ valueOf: () => { return Number.MIN_SAFE_INTEGER; } }", "-9007199254740991"],
|
||||
["{ valueOf: () => { return Number.MAX_SAFE_INTEGER; } }", "9007199254740991"],
|
||||
["{ valueOf: () => { return 0x80000000|0; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return 0x7fffffff|0; } }", "2147483647"],
|
||||
["{ valueOf: () => { return (0x80000000|0) - 0.5; } }", "-2147483648"],
|
||||
["{ valueOf: () => { return (0x7fffffff|0) + 0.5; } }", "2147483647"],
|
||||
];
|
||||
|
||||
let validInputTypedTestCases = validInputTestCases.map((element) => { return [eval("(" + element[0] + ")"), eval(element[1])] });
|
||||
|
||||
function isIdentical(result, expected)
|
||||
{
|
||||
if (expected === expected) {
|
||||
if (result !== expected)
|
||||
return false;
|
||||
if (!expected && (1 / expected) !== (1 / result))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
return result !== result;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.trunc() without arguments.
|
||||
function opaqueTruncNoArgument() {
|
||||
return Math.trunc();
|
||||
}
|
||||
noInline(opaqueTruncNoArgument);
|
||||
noOSRExitFuzzing(opaqueTruncNoArgument);
|
||||
|
||||
function testNoArgument() {
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
let output = opaqueTruncNoArgument();
|
||||
if (!isIdentical(output, NaN)) {
|
||||
throw "Failed opaqueTruncNoArgument";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTruncNoArgument) > 1)
|
||||
throw "The call without arguments should never exit.";
|
||||
}
|
||||
testNoArgument();
|
||||
|
||||
|
||||
// Test Math.trunc() with a very polymorphic input. All test cases are seen at each iteration.
|
||||
function opaqueAllTypesTrunc(argument) {
|
||||
return Math.trunc(argument);
|
||||
}
|
||||
noInline(opaqueAllTypesTrunc);
|
||||
noOSRExitFuzzing(opaqueAllTypesTrunc);
|
||||
|
||||
function testAllTypesCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesTrunc(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1]))
|
||||
throw "Failed testAllTypesCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesTrunc) > 3)
|
||||
throw "We should have detected trunc() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesCall();
|
||||
|
||||
|
||||
// Polymorphic input but negative zero is not observable.
|
||||
function opaqueAllTypesTruncWithoutNegativeZero(argument) {
|
||||
return Math.trunc(argument) + 0;
|
||||
}
|
||||
noInline(opaqueAllTypesTruncWithoutNegativeZero);
|
||||
noOSRExitFuzzing(opaqueAllTypesTruncWithoutNegativeZero);
|
||||
|
||||
function testAllTypesWithoutNegativeZeroCall() {
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
for (let testCaseInput of validInputTypedTestCases) {
|
||||
let output = opaqueAllTypesTruncWithoutNegativeZero(testCaseInput[0]);
|
||||
if (!isIdentical(output, testCaseInput[1] + 0))
|
||||
throw "Failed testAllTypesWithoutNegativeZeroCall for input " + testCaseInput[0] + " expected " + testCaseInput[1] + " got " + output;
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueAllTypesTrunc) > 3)
|
||||
throw "We should have detected trunc() was polymorphic and generated a generic version.";
|
||||
}
|
||||
testAllTypesWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.trunc() on a completely typed input. Every call see only one type.
|
||||
function testSingleTypeCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTrunc(argument) {
|
||||
return Math.trunc(argument);
|
||||
}
|
||||
noInline(opaqueTrunc);
|
||||
noOSRExitFuzzing(opaqueTrunc);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTrunc(${testCaseInput[0]}), ${testCaseInput[1]})) {
|
||||
throw "Failed testSingleTypeCall()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTrunc) > 1)
|
||||
throw "Failed testSingleTypeCall(). We should have compiled a single trunc for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeCall();
|
||||
|
||||
|
||||
function checkCompileCountForUselessNegativeZero(testFunction)
|
||||
{
|
||||
if (jscOptions().useMaximalFlushInsertionPhase) {
|
||||
// If we forced a flush after the operation, the negative zero becomes
|
||||
// observable and we may be overly optimistic.
|
||||
return numberOfDFGCompiles(testFunction) <= 2;
|
||||
}
|
||||
return numberOfDFGCompiles(testFunction) <= 1;
|
||||
}
|
||||
|
||||
|
||||
// Test Math.trunc() on a completely typed input, but without negative zero.
|
||||
function testSingleTypeWithoutNegativeZeroCall() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTrunc(argument) {
|
||||
return Math.trunc(argument) + 0;
|
||||
}
|
||||
noInline(opaqueTrunc);
|
||||
noOSRExitFuzzing(opaqueTrunc);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTrunc(${testCaseInput[0]}), ${testCaseInput[1]} + 0)) {
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall()";
|
||||
}
|
||||
}
|
||||
if (!checkCompileCountForUselessNegativeZero(opaqueTrunc))
|
||||
throw "Failed testSingleTypeWithoutNegativeZeroCall(). We should have compiled a single trunc for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testSingleTypeWithoutNegativeZeroCall();
|
||||
|
||||
|
||||
// Test Math.trunc() on constants
|
||||
function testConstant() {
|
||||
for (let testCaseInput of validInputTestCases) {
|
||||
eval(`
|
||||
function opaqueTruncOnConstant() {
|
||||
return Math.trunc(${testCaseInput[0]});
|
||||
}
|
||||
noInline(opaqueTruncOnConstant);
|
||||
noOSRExitFuzzing(opaqueTruncOnConstant);
|
||||
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (!isIdentical(opaqueTruncOnConstant(), ${testCaseInput[1]})) {
|
||||
throw "Failed testConstant()";
|
||||
}
|
||||
}
|
||||
if (numberOfDFGCompiles(opaqueTruncOnConstant) > 1)
|
||||
throw "Failed testConstant(). We should have compiled a single trunc for the expected type.";
|
||||
`);
|
||||
}
|
||||
}
|
||||
testConstant();
|
||||
|
||||
|
||||
// Verify we call valueOf() exactly once per call.
|
||||
function opaqueTruncForSideEffects(argument) {
|
||||
return Math.trunc(argument);
|
||||
}
|
||||
noInline(opaqueTruncForSideEffects);
|
||||
noOSRExitFuzzing(opaqueTruncForSideEffects);
|
||||
|
||||
function testSideEffect() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let trunc16 = Math.trunc(16);
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTruncForSideEffects(testObject) !== trunc16)
|
||||
throw "Incorrect result in testSideEffect()";
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testSideEffect()";
|
||||
if (numberOfDFGCompiles(opaqueTruncForSideEffects) > 1)
|
||||
throw "opaqueTruncForSideEffects() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testSideEffect();
|
||||
|
||||
|
||||
// Verify trunc() is not subject to CSE if the argument has side effects.
|
||||
function opaqueTruncForCSE(argument) {
|
||||
return Math.trunc(argument) + Math.trunc(argument) + Math.trunc(argument);
|
||||
}
|
||||
noInline(opaqueTruncForCSE);
|
||||
noOSRExitFuzzing(opaqueTruncForCSE);
|
||||
|
||||
function testCSE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
let trunc16 = Math.trunc(16);
|
||||
let threeTrunc16 = trunc16 + trunc16 + trunc16;
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
if (opaqueTruncForCSE(testObject) !== threeTrunc16)
|
||||
throw "Incorrect result in testCSE()";
|
||||
}
|
||||
if (testObject.counter !== 3e4)
|
||||
throw "Failed testCSE()";
|
||||
if (numberOfDFGCompiles(opaqueTruncForCSE) > 1)
|
||||
throw "opaqueTruncForCSE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testCSE();
|
||||
|
||||
|
||||
// Verify trunc() is not subject to DCE if the argument has side effects.
|
||||
function opaqueTruncForDCE(argument) {
|
||||
Math.trunc(argument);
|
||||
}
|
||||
noInline(opaqueTruncForDCE);
|
||||
noOSRExitFuzzing(opaqueTruncForDCE);
|
||||
|
||||
function testDCE() {
|
||||
let testObject = {
|
||||
counter: 0,
|
||||
valueOf: function() { ++this.counter; return 16; }
|
||||
};
|
||||
for (let i = 0; i < 1e4; ++i) {
|
||||
opaqueTruncForDCE(testObject);
|
||||
}
|
||||
if (testObject.counter !== 1e4)
|
||||
throw "Failed testDCE()";
|
||||
if (numberOfDFGCompiles(opaqueTruncForDCE) > 1)
|
||||
throw "opaqueTruncForDCE() is predictable, it should only be compiled once.";
|
||||
}
|
||||
testDCE();
|
||||
|
||||
|
||||
// Test exceptions in the argument.
|
||||
function testException() {
|
||||
let counter = 0;
|
||||
function opaqueTruncWithException(argument) {
|
||||
let result = Math.trunc(argument);
|
||||
++counter;
|
||||
return result;
|
||||
}
|
||||
noInline(opaqueTruncWithException);
|
||||
|
||||
let testObject = { valueOf: () => { return 64; } };
|
||||
let trunc64 = Math.trunc(64);
|
||||
|
||||
// Warm up without exception.
|
||||
for (let i = 0; i < 1e3; ++i) {
|
||||
if (opaqueTruncWithException(testObject) !== trunc64)
|
||||
throw "Incorrect result in opaqueTruncWithException()";
|
||||
}
|
||||
|
||||
let testThrowObject = { valueOf: () => { throw testObject; return 64; } };
|
||||
|
||||
for (let i = 0; i < 1e2; ++i) {
|
||||
try {
|
||||
if (opaqueTruncWithException(testThrowObject) !== 8)
|
||||
throw "This code should not be reached!!";
|
||||
} catch (e) {
|
||||
if (e !== testObject) {
|
||||
throw "Wrong object thrown from opaqueTruncWithException."
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (counter !== 1e3) {
|
||||
throw "Invalid count in testException()";
|
||||
}
|
||||
}
|
||||
testException();
|
|
@ -1,27 +0,0 @@
|
|||
function f1(a) {
|
||||
return a[0];
|
||||
}
|
||||
|
||||
function getF2Arguments() {
|
||||
return f2.arguments;
|
||||
}
|
||||
|
||||
noInline(getF2Arguments);
|
||||
|
||||
function f2(a) {
|
||||
a = f1(getF2Arguments());
|
||||
return a;
|
||||
}
|
||||
|
||||
function f3(a) {
|
||||
return f2(a);
|
||||
}
|
||||
|
||||
noInline(f3);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var result = f3(42);
|
||||
if (result != 42)
|
||||
throw "Error: bad result: " + result;
|
||||
}
|
||||
|
|
@ -1,21 +0,0 @@
|
|||
function f1(a) {
|
||||
return a[0];
|
||||
}
|
||||
|
||||
function f2(a) {
|
||||
a = f1(arguments);
|
||||
return a;
|
||||
}
|
||||
|
||||
function f3(a) {
|
||||
return f2(a);
|
||||
}
|
||||
|
||||
noInline(f3);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var result = f3(42);
|
||||
if (result != 42)
|
||||
throw "Error: bad result: " + result;
|
||||
}
|
||||
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
|
|
|
@ -1,3 +1,9 @@
|
|||
// Reviewed
|
||||
/**
|
||||
* reviewed
|
||||
* Tests class heritage failing with arrow functions (no constructor). Might have coverage, not found yet.
|
||||
*/
|
||||
|
||||
function testSyntax(script) {
|
||||
try {
|
||||
eval(script);
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(expected, actual, msg) {
|
||||
if (msg === void 0)
|
||||
msg = "";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(expected, actual, msg) {
|
||||
if (msg === void 0)
|
||||
msg = "";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(expected, actual, msg) {
|
||||
if (msg === void 0)
|
||||
msg = "";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(expected, actual, msg) {
|
||||
if (msg === void 0)
|
||||
msg = "";
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(expected, actual, msg) {
|
||||
if (msg === void 0)
|
||||
msg = "";
|
||||
|
|
|
@ -1,348 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function shouldBeAsync(expected, run, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
run().then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (hadError)
|
||||
throw actual;
|
||||
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
function shouldThrow(run, errorType, message) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
|
||||
try {
|
||||
actual = run();
|
||||
} catch (e) {
|
||||
hadError = true;
|
||||
actual = e;
|
||||
}
|
||||
|
||||
if (!hadError)
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but did not throw.");
|
||||
if (!(actual instanceof errorType))
|
||||
throw new Error("Expeced " + run + "() to throw " + errorType.name + " , but threw '" + actual + "'");
|
||||
if (message !== void 0 && actual.message !== message)
|
||||
throw new Error("Expected " + run + "() to throw '" + message + "', but threw '" + actual.message + "'");
|
||||
}
|
||||
|
||||
function shouldThrowAsync(run, errorType, message) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
run().then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (!hadError)
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but did not throw.");
|
||||
if (!(actual instanceof errorType))
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but threw '" + actual + "'");
|
||||
if (message !== void 0 && actual.message !== message)
|
||||
throw new Error("Expected " + run + "() to throw '" + message + "', but threw '" + actual.message + "'");
|
||||
}
|
||||
|
||||
function shouldThrowSyntaxError(str, message) {
|
||||
try {
|
||||
eval(str);
|
||||
throw new Error("Expected `" + str + "` to throw a SyntaxError, but did not throw.")
|
||||
} catch (e) {
|
||||
if (e.constructor !== SyntaxError)
|
||||
throw new Error("Expected `" + str + "` to throw a SyntaxError, but threw '" + e + "'");
|
||||
if (message !== void 0 && e.message !== message)
|
||||
throw new Error("Expected `" + str + "` to throw SyntaxError: '" + message + "', but threw '" + e + "'");
|
||||
}
|
||||
}
|
||||
|
||||
// Do not install `AsyncFunction` constructor on global object
|
||||
shouldBe(undefined, this.AsyncFunction);
|
||||
let AsyncFunction = (async function() {}).constructor;
|
||||
|
||||
// Let functionPrototype be the intrinsic object %AsyncFunctionPrototype%.
|
||||
async function asyncFunctionForProto() {}
|
||||
shouldBe(AsyncFunction.prototype, Object.getPrototypeOf(asyncFunctionForProto));
|
||||
shouldBe(AsyncFunction.prototype, Object.getPrototypeOf(async function() {}));
|
||||
shouldBe(AsyncFunction.prototype, Object.getPrototypeOf(async () => {}));
|
||||
shouldBe(AsyncFunction.prototype, Object.getPrototypeOf({ async method() {} }.method));
|
||||
|
||||
// FIXME: AsyncFunction constructor should build functions with correct prototype.
|
||||
// shouldBe(AsyncFunction.prototype, Object.getPrototypeOf(AsyncFunction()));
|
||||
|
||||
// AsyncFunctionCreate does not produce an object with a Prototype
|
||||
shouldBe(undefined, asyncFunctionForProto.prototype);
|
||||
shouldBe(false, asyncFunctionForProto.hasOwnProperty("prototype"));
|
||||
shouldBe(undefined, (async function() {}).prototype);
|
||||
shouldBe(false, (async function() {}).hasOwnProperty("prototype"));
|
||||
shouldBe(undefined, (async() => {}).prototype);
|
||||
shouldBe(false, (async() => {}).hasOwnProperty("prototype"));
|
||||
shouldBe(undefined, ({ async method() {} }).method.prototype);
|
||||
shouldBe(false, ({ async method() {} }).method.hasOwnProperty("prototype"));
|
||||
shouldBe(undefined, AsyncFunction().prototype);
|
||||
shouldBe(false, AsyncFunction().hasOwnProperty("prototype"));
|
||||
|
||||
// AsyncFunction.prototype[ @@toStringTag ]
|
||||
var descriptor = Object.getOwnPropertyDescriptor(AsyncFunction.prototype, Symbol.toStringTag);
|
||||
shouldBe("AsyncFunction", descriptor.value);
|
||||
shouldBe(false, descriptor.enumerable);
|
||||
shouldBe(false, descriptor.writable);
|
||||
shouldBe(true, descriptor.configurable);
|
||||
|
||||
shouldBe(1, AsyncFunction.length);
|
||||
|
||||
// Let F be ! FunctionAllocate(functionPrototype, Strict, "non-constructor")
|
||||
async function asyncNonConstructorDecl() {}
|
||||
shouldThrow(() => new asyncNonConstructorDecl(), TypeError);
|
||||
shouldThrow(() => new (async function() {}), TypeError);
|
||||
shouldThrow(() => new ({ async nonConstructor() {} }).nonConstructor(), TypeError);
|
||||
shouldThrow(() => new (() => "not a constructor!"), TypeError);
|
||||
shouldThrow(() => new (AsyncFunction()), TypeError);
|
||||
|
||||
// Normal completion
|
||||
async function asyncDecl() { return "test"; }
|
||||
shouldBeAsync("test", asyncDecl);
|
||||
shouldBeAsync("test2", async function() { return "test2"; });
|
||||
shouldBeAsync("test3", async () => "test3");
|
||||
shouldBeAsync("test4", () => ({ async f() { return "test4"; } }).f());
|
||||
|
||||
class MyError extends Error {};
|
||||
|
||||
// Throw completion
|
||||
async function asyncDeclThrower(e) { throw new MyError(e); }
|
||||
shouldThrowAsync(() => asyncDeclThrower("boom!"), MyError, "boom!");
|
||||
shouldThrowAsync(() => (async function(e) { throw new MyError(e); })("boom!!!"), MyError, "boom!!!");
|
||||
shouldThrowAsync(() => (async e => { throw new MyError(e) })("boom!!"), MyError, "boom!!");
|
||||
shouldThrowAsync(() => ({ async thrower(e) { throw new MyError(e); } }).thrower("boom!!!!"), MyError, "boom!!!!");
|
||||
|
||||
function resolveLater(value) { return Promise.resolve(value); }
|
||||
function rejectLater(error) { return Promise.reject(error); }
|
||||
|
||||
// Resume after Normal completion
|
||||
var log = [];
|
||||
async function resumeAfterNormal(value) {
|
||||
log.push("start:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
return value + 1;
|
||||
}
|
||||
|
||||
shouldBeAsync(4, () => resumeAfterNormal(1));
|
||||
shouldBe("start:1 resume:2 resume:3", log.join(" "));
|
||||
|
||||
var O = {
|
||||
async resumeAfterNormal(value) {
|
||||
log.push("start:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
return value + 1;
|
||||
}
|
||||
};
|
||||
log = [];
|
||||
shouldBeAsync(5, () => O.resumeAfterNormal(2));
|
||||
shouldBe("start:2 resume:3 resume:4", log.join(" "));
|
||||
|
||||
var resumeAfterNormalArrow = async (value) => {
|
||||
log.push("start:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
return value + 1;
|
||||
};
|
||||
log = [];
|
||||
shouldBeAsync(6, () => resumeAfterNormalArrow(3));
|
||||
shouldBe("start:3 resume:4 resume:5", log.join(" "));
|
||||
|
||||
var resumeAfterNormalEval = AsyncFunction("value", `
|
||||
log.push("start:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
value = await resolveLater(value + 1);
|
||||
log.push("resume:" + value);
|
||||
return value + 1;
|
||||
`);
|
||||
log = [];
|
||||
shouldBeAsync(7, () => resumeAfterNormalEval(4));
|
||||
shouldBe("start:4 resume:5 resume:6", log.join(" "));
|
||||
|
||||
// Resume after Throw completion
|
||||
async function resumeAfterThrow(value) {
|
||||
log.push("start:" + value);
|
||||
try {
|
||||
value = await rejectLater("throw1");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
try {
|
||||
value = await rejectLater("throw2");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
return value + 1;
|
||||
}
|
||||
|
||||
log = [];
|
||||
shouldBeAsync(2, () => resumeAfterThrow(1));
|
||||
shouldBe("start:1 resume:throw1 resume:throw2", log.join(" "));
|
||||
|
||||
var O = {
|
||||
async resumeAfterThrow(value) {
|
||||
log.push("start:" + value);
|
||||
try {
|
||||
value = await rejectLater("throw1");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
try {
|
||||
value = await rejectLater("throw2");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
return value + 1;
|
||||
}
|
||||
}
|
||||
log = [];
|
||||
shouldBeAsync(3, () => O.resumeAfterThrow(2));
|
||||
shouldBe("start:2 resume:throw1 resume:throw2", log.join(" "));
|
||||
|
||||
var resumeAfterThrowArrow = async (value) => {
|
||||
log.push("start:" + value);
|
||||
try {
|
||||
value = await rejectLater("throw1");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
try {
|
||||
value = await rejectLater("throw2");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
return value + 1;
|
||||
};
|
||||
log = [];
|
||||
shouldBeAsync(4, () => resumeAfterThrowArrow(3));
|
||||
shouldBe("start:3 resume:throw1 resume:throw2", log.join(" "));
|
||||
|
||||
var resumeAfterThrowEval = AsyncFunction("value", `
|
||||
log.push("start:" + value);
|
||||
try {
|
||||
value = await rejectLater("throw1");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
try {
|
||||
value = await rejectLater("throw2");
|
||||
} catch (e) {
|
||||
log.push("resume:" + e);
|
||||
}
|
||||
return value + 1;
|
||||
`);
|
||||
log = [];
|
||||
shouldBeAsync(5, () => resumeAfterThrowEval(4));
|
||||
shouldBe("start:4 resume:throw1 resume:throw2", log.join(" "));
|
||||
|
||||
var awaitEpression = async (value) => {
|
||||
log.push("start:" + value);
|
||||
if (!await false)
|
||||
log.push('step 1');
|
||||
var t = ~await true;
|
||||
log.push('step 2 ' + t);
|
||||
|
||||
var t1 = +await Promise.resolve(12345);
|
||||
log.push('step 3 ' + t1);
|
||||
|
||||
var t2 = -await 54321;
|
||||
log.push('step 4 ' + t2);
|
||||
|
||||
var t3 = !!!!!await Promise.resolve(true);
|
||||
log.push('step 5 ' + t3);
|
||||
|
||||
try {
|
||||
var t4 = ++await 1;
|
||||
} catch(e) {
|
||||
if (e instanceof ReferenceError) {
|
||||
log.push('step 6 ');
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
var t5 = --await 1;
|
||||
} catch(e) {
|
||||
if (e instanceof ReferenceError) {
|
||||
log.push('step 7');
|
||||
}
|
||||
}
|
||||
|
||||
return void await 'test';
|
||||
};
|
||||
log = [];
|
||||
|
||||
shouldBeAsync(undefined, () => awaitEpression(5));
|
||||
shouldBe("start:5 step 1 step 2 -2 step 3 12345 step 4 -54321 step 5 false step 6 step 7", log.join(" "));
|
||||
|
||||
// MethoodDefinition SyntaxErrors
|
||||
shouldThrowSyntaxError("var obj = { async foo : true };", "Unexpected token ':'. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async foo = true };", "Unexpected token '='. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async foo , bar };", "Unexpected token ','. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async foo }", "Unexpected token '}'. Expected a parenthesis for argument list.");
|
||||
|
||||
shouldThrowSyntaxError("var obj = { async 0 : true };", "Unexpected token ':'. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 0 = true };", "Unexpected token '='. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 0 , bar };", "Unexpected token ','. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 0 }", "Unexpected token '}'. Expected a parenthesis for argument list.");
|
||||
|
||||
shouldThrowSyntaxError("var obj = { async 'foo' : true };", "Unexpected token ':'. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 'foo' = true };", "Unexpected token '='. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 'foo' , bar };", "Unexpected token ','. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async 'foo' }", "Unexpected token '}'. Expected a parenthesis for argument list.");
|
||||
|
||||
shouldThrowSyntaxError("var obj = { async ['foo'] : true };", "Unexpected token ':'. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async ['foo'] = true };", "Unexpected token '='. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async ['foo'] , bar };", "Unexpected token ','. Expected a parenthesis for argument list.");
|
||||
shouldThrowSyntaxError("var obj = { async ['foo'] }", "Unexpected token '}'. Expected a parenthesis for argument list.");
|
||||
|
||||
shouldThrowSyntaxError("class C { async foo : true };", "Unexpected token ':'. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async foo = true };", "Unexpected token '='. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async foo , bar };", "Unexpected token ','. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async foo }", "Unexpected token '}'. Expected an opening '(' before a async method's parameter list.");
|
||||
|
||||
shouldThrowSyntaxError("class C { async 0 : true };", "Unexpected token ':'. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 0 = true };", "Unexpected token '='. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 0 , bar };", "Unexpected token ','. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 0 }", "Unexpected token '}'. Expected an opening '(' before a async method's parameter list.");
|
||||
|
||||
shouldThrowSyntaxError("class C { async 'foo' : true };", "Unexpected token ':'. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 'foo' = true };", "Unexpected token '='. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 'foo' , bar };", "Unexpected token ','. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async 'foo' }", "Unexpected token '}'. Expected an opening '(' before a async method's parameter list.");
|
||||
|
||||
shouldThrowSyntaxError("class C { async ['foo'] : true };", "Unexpected token ':'. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async ['foo'] = true };", "Unexpected token '='. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async ['foo'] , bar };", "Unexpected token ','. Expected an opening '(' before a async method's parameter list.");
|
||||
shouldThrowSyntaxError("class C { async ['foo'] }", "Unexpected token '}'. Expected an opening '(' before a async method's parameter list.");
|
||||
|
||||
// Ensure awaited builtin Promise objects are themselves wrapped in a new Promise,
|
||||
// per https://tc39.github.io/ecma262/#sec-async-functions-abstract-operations-async-function-await
|
||||
log = [];
|
||||
async function awaitedPromisesAreWrapped() {
|
||||
log.push("before");
|
||||
await Promise.resolve();
|
||||
log.push("after");
|
||||
}
|
||||
awaitedPromisesAreWrapped();
|
||||
Promise.resolve().then(() => log.push("Promise.resolve()"));
|
||||
drainMicrotasks();
|
||||
shouldBe("before|Promise.resolve()|after", log.join("|"));
|
|
@ -1,26 +0,0 @@
|
|||
// Copyright (C) Copyright 2016 the v8 project authors. All rights reserved.
|
||||
|
||||
//@ runNoCJIT("--gcMaxHeapSize=2000000")
|
||||
|
||||
|
||||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
|
||||
let out;
|
||||
|
||||
async function longLoop() {
|
||||
for (let i = 0; i < 10000; i++)
|
||||
await undefined;
|
||||
out = 1;
|
||||
}
|
||||
|
||||
longLoop();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
shouldBe(out, 1);
|
|
@ -1,69 +0,0 @@
|
|||
function shouldThrow(func, errorMessage) {
|
||||
var errorThrown = false;
|
||||
var error = null;
|
||||
try {
|
||||
func();
|
||||
} catch (e) {
|
||||
errorThrown = true;
|
||||
error = e;
|
||||
}
|
||||
if (!errorThrown)
|
||||
throw new Error('not thrown');
|
||||
if (String(error) !== errorMessage)
|
||||
throw new Error(`bad error: ${String(error)}`);
|
||||
}
|
||||
|
||||
function checkModuleSyntaxError(source, errorMessage) {
|
||||
shouldThrow(() => checkModuleSyntax(source), errorMessage);
|
||||
}
|
||||
|
||||
checkModuleSyntaxError(String.raw`
|
||||
var await;
|
||||
`, `SyntaxError: Cannot use 'await' as a variable name in a module.:2`);
|
||||
checkModuleSyntaxError(`
|
||||
export var await;
|
||||
`, `SyntaxError: Cannot use 'await' as a variable name in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
let await;
|
||||
`, `SyntaxError: Cannot use 'await' as a lexical variable name in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
export let await;
|
||||
`, `SyntaxError: Cannot use 'await' as a lexical variable name in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
const await = 1
|
||||
`, `SyntaxError: Cannot use 'await' as a lexical variable name in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
export const await = 1
|
||||
`, `SyntaxError: Cannot use 'await' as a lexical variable name in a module.:2`);
|
||||
|
||||
checkModuleSyntaxError(String.raw`
|
||||
function await() {}
|
||||
`, `SyntaxError: Cannot declare function named 'await' in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
function* await() {}
|
||||
`, `SyntaxError: Cannot declare function named 'await' in a module.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
async function await() {}
|
||||
`, `SyntaxError: Cannot declare function named 'await' in a module.:2`);
|
||||
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import {await} from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import {foo as await} from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import * as await from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import await, {x, y, z} from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import await, {x, y, z,} from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import await from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
||||
checkModuleSyntaxError(String.raw`
|
||||
import await, * as foo from 'foo';
|
||||
`, `SyntaxError: Cannot use 'await' as an imported binding name.:2`);
|
|
@ -1,302 +0,0 @@
|
|||
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||
* file, You can obtain one at http://moz.org/MPL/2.0/. */
|
||||
|
||||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function shouldBeAsync(expected, run, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
run().then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
if (hadError)
|
||||
throw actual;
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
function shouldThrow(run, errorType, message) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
try {
|
||||
actual = run();
|
||||
} catch (e) {
|
||||
hadError = true;
|
||||
actual = e;
|
||||
}
|
||||
if (!hadError)
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but did not throw.");
|
||||
if (!(actual instanceof errorType))
|
||||
throw new Error("Expeced " + run + "() to throw " + errorType.name + " , but threw '" + actual + "'");
|
||||
if (message !== void 0 && actual.message !== message)
|
||||
throw new Error("Expected " + run + "() to throw '" + message + "', but threw '" + actual.message + "'");
|
||||
}
|
||||
|
||||
function shouldThrowAsync(run, errorType, message) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
run().then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
if (!hadError)
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but did not throw.");
|
||||
if (!(actual instanceof errorType))
|
||||
throw new Error("Expected " + run + "() to throw " + errorType.name + ", but threw '" + actual + "'");
|
||||
if (message !== void 0 && actual.message !== message)
|
||||
throw new Error("Expected " + run + "() to throw '" + message + "', but threw '" + actual.message + "'");
|
||||
}
|
||||
|
||||
function assert(cond, msg = "") {
|
||||
if (!cond)
|
||||
throw new Error(msg);
|
||||
}
|
||||
|
||||
function shouldThrowSyntaxError(str, message) {
|
||||
var hadError = false;
|
||||
try {
|
||||
eval(str);
|
||||
} catch (e) {
|
||||
if (e instanceof SyntaxError) {
|
||||
hadError = true;
|
||||
if (typeof message === "string")
|
||||
assert(e.message === message, "Expected '" + message + "' but threw '" + e.message + "'");
|
||||
}
|
||||
}
|
||||
assert(hadError, "Did not throw syntax error");
|
||||
}
|
||||
|
||||
// semantics.js
|
||||
(function mozSemantics() {
|
||||
|
||||
async function empty() {
|
||||
}
|
||||
|
||||
async function simpleReturn() {
|
||||
return 1;
|
||||
}
|
||||
|
||||
async function simpleAwait() {
|
||||
var result = await 2;
|
||||
return result;
|
||||
}
|
||||
|
||||
async function simpleAwaitAsync() {
|
||||
var result = await simpleReturn();
|
||||
return 2 + result;
|
||||
}
|
||||
|
||||
async function returnOtherAsync() {
|
||||
return 1 + await simpleAwaitAsync();
|
||||
}
|
||||
|
||||
async function simpleThrower() {
|
||||
throw new Error();
|
||||
}
|
||||
|
||||
async function delegatedThrower() {
|
||||
var val = await simpleThrower();
|
||||
return val;
|
||||
}
|
||||
|
||||
async function tryCatch() {
|
||||
try {
|
||||
await delegatedThrower();
|
||||
return 'FAILED';
|
||||
} catch (_) {
|
||||
return 5;
|
||||
}
|
||||
}
|
||||
|
||||
async function tryCatchThrow() {
|
||||
try {
|
||||
await delegatedThrower();
|
||||
return 'FAILED';
|
||||
} catch (_) {
|
||||
return delegatedThrower();
|
||||
}
|
||||
}
|
||||
|
||||
async function wellFinally() {
|
||||
try {
|
||||
await delegatedThrower();
|
||||
} catch (_) {
|
||||
return 'FAILED';
|
||||
} finally {
|
||||
return 6;
|
||||
}
|
||||
}
|
||||
|
||||
async function finallyMayFail() {
|
||||
try {
|
||||
await delegatedThrower();
|
||||
} catch (_) {
|
||||
return 5;
|
||||
} finally {
|
||||
return delegatedThrower();
|
||||
}
|
||||
}
|
||||
|
||||
async function embedded() {
|
||||
async function inner() {
|
||||
return 7;
|
||||
}
|
||||
return await inner();
|
||||
}
|
||||
|
||||
// recursion, it works!
|
||||
async function fib(n) {
|
||||
return (n == 0 || n == 1) ? n : await fib(n - 1) + await fib(n - 2);
|
||||
}
|
||||
|
||||
// mutual recursion
|
||||
async function isOdd(n) {
|
||||
async function isEven(n) {
|
||||
return n === 0 || await isOdd(n - 1);
|
||||
}
|
||||
return n !== 0 && await isEven(n - 1);
|
||||
}
|
||||
|
||||
// recursion, take three!
|
||||
var hardcoreFib = async function fib2(n) {
|
||||
return (n == 0 || n == 1) ? n : await fib2(n - 1) + await fib2(n - 2);
|
||||
}
|
||||
|
||||
var asyncExpr = async function() {
|
||||
return 10;
|
||||
}
|
||||
|
||||
var namedAsyncExpr = async function simple() {
|
||||
return 11;
|
||||
}
|
||||
|
||||
async function executionOrder() {
|
||||
var value = 0;
|
||||
async function first() {
|
||||
return (value = value === 0 ? 1 : value);
|
||||
}
|
||||
async function second() {
|
||||
return (value = value === 0 ? 2 : value);
|
||||
}
|
||||
async function third() {
|
||||
return (value = value === 0 ? 3 : value);
|
||||
}
|
||||
return await first() + await second() + await third() + 6;
|
||||
}
|
||||
|
||||
async function miscellaneous() {
|
||||
if (arguments.length === 3 &&
|
||||
arguments.callee.name === "miscellaneous")
|
||||
return 14;
|
||||
}
|
||||
|
||||
function thrower() {
|
||||
throw 15;
|
||||
}
|
||||
|
||||
async function defaultArgs(arg = thrower()) {
|
||||
}
|
||||
|
||||
// Async functions are not constructible
|
||||
shouldThrow(() => {
|
||||
async function Person() {
|
||||
}
|
||||
new Person();
|
||||
}, TypeError);
|
||||
|
||||
shouldBeAsync(undefined, empty);
|
||||
shouldBeAsync(1, simpleReturn);
|
||||
shouldBeAsync(2, simpleAwait);
|
||||
shouldBeAsync(3, simpleAwaitAsync);
|
||||
shouldBeAsync(4, returnOtherAsync);
|
||||
shouldThrowAsync(simpleThrower, Error);
|
||||
shouldBeAsync(5, tryCatch);
|
||||
shouldBeAsync(6, wellFinally);
|
||||
shouldThrowAsync(finallyMayFail, Error);
|
||||
shouldBeAsync(7, embedded);
|
||||
shouldBeAsync(8, () => fib(6));
|
||||
shouldBeAsync(9, executionOrder);
|
||||
shouldBeAsync(10, asyncExpr);
|
||||
shouldBeAsync(11, namedAsyncExpr);
|
||||
shouldBeAsync(12, () => isOdd(12).then(v => v ? "oops" : 12));
|
||||
shouldBeAsync(13, () => hardcoreFib(7));
|
||||
shouldBeAsync(14, () => miscellaneous(1, 2, 3));
|
||||
shouldBeAsync(15, () => defaultArgs().catch(e => e));
|
||||
|
||||
})();
|
||||
|
||||
// methods.js
|
||||
(function mozMethods() {
|
||||
|
||||
class X {
|
||||
constructor() {
|
||||
this.value = 42;
|
||||
}
|
||||
async getValue() {
|
||||
return this.value;
|
||||
}
|
||||
setValue(value) {
|
||||
this.value = value;
|
||||
}
|
||||
async increment() {
|
||||
var value = await this.getValue();
|
||||
this.setValue(value + 1);
|
||||
return this.getValue();
|
||||
}
|
||||
async getBaseClassName() {
|
||||
return 'X';
|
||||
}
|
||||
static async getStaticValue() {
|
||||
return 44;
|
||||
}
|
||||
}
|
||||
|
||||
class Y extends X {
|
||||
async getBaseClassName() {
|
||||
return super.getBaseClassName();
|
||||
}
|
||||
}
|
||||
|
||||
var objLiteral = {
|
||||
async get() {
|
||||
return 45;
|
||||
},
|
||||
someStuff: 5
|
||||
};
|
||||
|
||||
var x = new X();
|
||||
var y = new Y();
|
||||
|
||||
shouldBeAsync(42, () => x.getValue());
|
||||
shouldBeAsync(43, () => x.increment());
|
||||
shouldBeAsync(44, () => X.getStaticValue());
|
||||
shouldBeAsync(45, () => objLiteral.get());
|
||||
shouldBeAsync('X', () => y.getBaseClassName());
|
||||
|
||||
})();
|
||||
|
||||
(function mozFunctionNameInferrence() {
|
||||
|
||||
async function test() { }
|
||||
|
||||
var anon = async function() { }
|
||||
|
||||
shouldBe("test", test.name);
|
||||
shouldBe("anon", anon.name);
|
||||
|
||||
})();
|
||||
|
||||
(function mozSyntaxErrors() {
|
||||
|
||||
shouldThrowSyntaxError("'use strict'; async function eval() {}");
|
||||
shouldThrowSyntaxError("'use strict'; async function arguments() {}");
|
||||
shouldThrowSyntaxError("async function a(k = super.prop) { }");
|
||||
shouldThrowSyntaxError("async function a() { super.prop(); }");
|
||||
shouldThrowSyntaxError("async function a() { super(); }");
|
||||
shouldThrowSyntaxError("async function a(k = await 3) {}");
|
||||
|
||||
})();
|
|
@ -1,163 +0,0 @@
|
|||
function assert(cond, msg = "") {
|
||||
if (!cond)
|
||||
throw new Error(msg);
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
function shouldThrowSyntaxError(str, message) {
|
||||
var hadError = false;
|
||||
try {
|
||||
eval(str);
|
||||
} catch (e) {
|
||||
if (e instanceof SyntaxError) {
|
||||
hadError = true;
|
||||
if (typeof message === "string")
|
||||
assert(e.message === message, "Expected '" + message + "' but threw '" + e.message + "'");
|
||||
}
|
||||
}
|
||||
assert(hadError, "Did not throw syntax error");
|
||||
}
|
||||
noInline(shouldThrowSyntaxError);
|
||||
|
||||
var AsyncFunction = (async function() {}).constructor;
|
||||
|
||||
// AsyncFunctionExpression
|
||||
shouldThrowSyntaxError("(async function() { var await; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { var [await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { var [...await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { var {await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let await; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let [await] = []; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let [...await] = []; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let {await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let {isAsync: await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { let {isAsync: await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const await; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const [await] = []; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const [...await] = []; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const {await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const {isAsync: await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { const {isAsync: await} = {}; })", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { function await() {} })", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("(async function() { async function await() {} })", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("(async function(await) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("(async function f([await]) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("(async function f([...await]) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("(async function f(...await) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("(async function f({await}) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("(async function f({isAsync: await}) {})", "Cannot use 'await' as a parameter name in an async function.");
|
||||
|
||||
// AsyncFunctionDeclaration
|
||||
shouldThrowSyntaxError("async function f() { var await; }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { var [await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { var [...await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { var {await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let await; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let [await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let [...await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let {await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { let {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const await; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const [await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const [...await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const {await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { const {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { function await() {} }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("async function f() { async function await() {} }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("async function f(await) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("async function f([await]) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("async function f([...await]) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("async function f(...await) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("async function f({await}) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("async function f({isAsync: await}) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
|
||||
// AsyncArrowFunction
|
||||
shouldThrowSyntaxError("var f = async () => { var await; }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { var [await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { var [...await] = []; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { var {await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { var {isAsync: await} = {}; })", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let await; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let [await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let [...await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let {await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { let {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const await; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const [await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const [...await] = []; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const {await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { const {isAsync: await} = {}; }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { function await() {} }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("var f = async () => { async function await() {} }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("var f = async (await) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async ([await]) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async ([...await]) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async (...await) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async ({await}) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async ({isAsync: await}) => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var f = async await => {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
|
||||
// AsyncMethod
|
||||
shouldThrowSyntaxError("var O = { async f() { var await; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { var [await] = []; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { var [...await] = []; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { var {await} = {}; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { var {isAsync: await} = {}; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { var {isAsync: await} = {}; } }", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let await; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let [await] = []; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let [...await] = []; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let {await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let {isAsync: await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { let {isAsync: await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const await; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const [await] = []; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const [...await] = []; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const {await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const {isAsync: await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { const {isAsync: await} = {}; } }", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { function await() {} }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f() { async function await() {} } }", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f(await) {} } ", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f([await]) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f([...await]) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f(...await) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f({await}) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("var O = { async f({isAsync: await}) {}", "Cannot use 'await' as a parameter name in an async function.");
|
||||
|
||||
// AsyncFunction constructor
|
||||
shouldThrowSyntaxError("AsyncFunction('var await;')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('var [await] = [];')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('var [...await] = [];')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('var {await} = {};')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('var {isAsync: await} = {};')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('var {isAsync: await} = {};')", "Cannot use 'await' as a variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let await;')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let [await] = [];')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let [...await] = [];')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let {await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let {isAsync: await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('let {isAsync: await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const await;')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const [await] = [];')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const [...await] = [];')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const {await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const {isAsync: await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('const {isAsync: await} = {};')", "Cannot use 'await' as a lexical variable name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('function await() {}')", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('async function await() {}')", "Cannot declare function named 'await' in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('await', '')", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('[await]', '')", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('[...await]', '')", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('...await', '')", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('{await}', '')", "Cannot use 'await' as a parameter name in an async function.");
|
||||
shouldThrowSyntaxError("AsyncFunction('{isAsync: await}', '')", "Cannot use 'await' as a parameter name in an async function.");
|
|
@ -1,582 +0,0 @@
|
|||
// Copyright (C) 2016 the V8 project authors. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function testSyntax(script) {
|
||||
try {
|
||||
eval(script);
|
||||
} catch (error) {
|
||||
if (error instanceof SyntaxError)
|
||||
throw new Error("Bad error: " + String(error) + "\n evaluating `" + script + "`");
|
||||
}
|
||||
}
|
||||
|
||||
function testSyntaxError(script, message) {
|
||||
var error = null;
|
||||
try {
|
||||
eval(script);
|
||||
} catch (e) {
|
||||
error = e;
|
||||
}
|
||||
if (!error)
|
||||
throw new Error("Expected syntax error not thrown\n evaluating `" + script + "`");
|
||||
|
||||
if (typeof message === "string" && String(error) !== message)
|
||||
throw new Error("Bad error: " + String(error) + "\n evaluating `" + script + "`");
|
||||
}
|
||||
|
||||
(function testTopLevelAsyncAwaitSyntaxSloppyMode() {
|
||||
testSyntax(`({async: 1})`);
|
||||
testSyntax(`var asyncFn = async function() { await 1; };`);
|
||||
|
||||
testSyntax(`var asyncFn = async function() { var t = !await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = ~await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = typeof await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = void await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = !(await 1); };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = ~(await 1); };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = typeof (await 1); };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = void (await 1); };`);
|
||||
|
||||
|
||||
testSyntax(`var asyncFn = async function() { var t = !!await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = ~~await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = typeof typeof await 1; };`);
|
||||
testSyntax(`var asyncFn = async function() { var t = void void await 1; };`);
|
||||
|
||||
testSyntax(`var asyncFn = async function withName() { await 1; };`);
|
||||
testSyntax(`var asyncFn = async () => await 'test';`);
|
||||
|
||||
testSyntax(`var asyncFn = async () => !await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => ~await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => typeof await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => void await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => - await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => + await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => delete await 'test';`);
|
||||
|
||||
testSyntax(`var asyncFn = async () => !!await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => ~~await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => typeof typeof await 'test';`);
|
||||
testSyntax(`var asyncFn = async () => void void await 'test';`);
|
||||
|
||||
testSyntax(`var asyncFn = async x => await x + 'test';`);
|
||||
testSyntax(`function foo(fn) { fn({ async: true }); }`);
|
||||
testSyntax(`async function asyncFn() { await 1; }`);
|
||||
testSyntax(`var O = { async method() { await 1; } };`);
|
||||
testSyntax(`var O = { async ['meth' + 'od']() { await 1; } };`);
|
||||
testSyntax(`var O = { async 'method'() { await 1; } };`);
|
||||
testSyntax(`var O = { async 0() { await 1; } };`);
|
||||
testSyntax(`var O = { async function() {} };`);
|
||||
testSyntax(`class C { async method() { await 1; } };`);
|
||||
testSyntax(`class C { async ['meth' + 'od']() { await 1; } };`);
|
||||
testSyntax(`class C { async 'method'() { await 1; } };`);
|
||||
testSyntax(`class C { async 0() { await 1; } };`);
|
||||
testSyntax(`var asyncFn = async({ foo = 1 }) => foo;`);
|
||||
testSyntax(`var asyncFn = async({ foo = 1 } = {}) => foo;`);
|
||||
testSyntax(`function* g() { var f = async(yield); }`);
|
||||
testSyntax(`function* g() { var f = async(x = yield); }`);
|
||||
testSyntax(`class C { async ['function']() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { async ['function']() {} }`);
|
||||
testSyntax(`class C { static async ['function']() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { static async ['function']() {} }`);
|
||||
testSyntax(`class C { async function() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { async function() {} }`);
|
||||
testSyntax(`class C { static async function() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { static async function() {} }`);
|
||||
testSyntax(`class C { async 'function'() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { async 'function'() {} }`);
|
||||
testSyntax(`class C { static async 'function'() {} }`);
|
||||
testSyntax(`class C {}; class C2 extends C { static async 'function'() {} }`);
|
||||
})();
|
||||
|
||||
(function testTopLevelAsyncAwaitSyntaxStrictMode() {
|
||||
testSyntax(`"use strict"; ({async: 1})`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { await 1; };`);
|
||||
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = ~await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = +await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = -await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = typeof await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = void await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !(await 1); };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = ~(await 1); };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = -(await 1); };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = +(await 1); };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = typeof (await 1); };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = void (await 1); };`);
|
||||
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !!await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = ~~await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = typeof typeof await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = void void await 1; };`);
|
||||
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !!await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !~await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !typeof typeof await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async function() { var t = !void void await 1; };`);
|
||||
|
||||
|
||||
testSyntax(`"use strict"; var asyncFn = async function withName() { await 1; };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async () => await 'test';`);
|
||||
testSyntax(`"use strict"; var asyncFn = async x => await x + 'test';`);
|
||||
testSyntax(`"use strict"; function foo(fn) { fn({ async: true }); }`);
|
||||
testSyntax(`"use strict"; async function asyncFn() { await 1; }`);
|
||||
testSyntax(`"use strict"; var O = { async method() { await 1; } };`);
|
||||
testSyntax(`"use strict"; var O = { async ['meth' + 'od']() { await 1; } };`);
|
||||
testSyntax(`"use strict"; var O = { async 'method'() { await 1; } };`);
|
||||
testSyntax(`"use strict"; var O = { async 0() { await 1; } };`);
|
||||
testSyntax(`"use strict"; class C { async method() { await 1; } };`);
|
||||
testSyntax(`"use strict"; class C { async ['meth' + 'od']() { await 1; } };`);
|
||||
testSyntax(`"use strict"; class C { async 'method'() { await 1; } };`);
|
||||
testSyntax(`"use strict"; class C { async 0() { await 1; } };`);
|
||||
testSyntax(`"use strict"; var asyncFn = async({ foo = 1 }) => foo;`);
|
||||
testSyntax(`"use strict"; var asyncFn = async({ foo = 1 } = {}) => foo;`);
|
||||
testSyntax(`"use strict"; function* g() { var f = async(yield); }`);
|
||||
testSyntax(`"use strict"; function* g() { var f = async(x = yield); }`);
|
||||
testSyntax(`"use strict"; class C { async ['function']() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { async ['function']() {} }`);
|
||||
testSyntax(`"use strict"; class C { static async ['function']() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { static async ['function']() {} }`);
|
||||
testSyntax(`"use strict"; class C { async function() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { async function() {} }`);
|
||||
testSyntax(`"use strict"; class C { async function() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { async function() {} }`);
|
||||
testSyntax(`"use strict"; class C { static async function() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { static async function() {} }`);
|
||||
testSyntax(`"use strict"; class C { async 'function'() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { async 'function'() {} }`);
|
||||
testSyntax(`"use strict"; class C { static async 'function'() {} }`);
|
||||
testSyntax(`"use strict"; class C {}; class C2 extends C { static async 'function'() {} }`);
|
||||
})();
|
||||
|
||||
(function testNestedAsyncAwaitSyntax() {
|
||||
var contextData = [
|
||||
{ prefix: "function outerFunction() { ", suffix: " }" },
|
||||
{ prefix: "function* outerGenerator() { ", suffix: " }" },
|
||||
{ prefix: "var outerFuncExpr = function() { ", suffix: " };" },
|
||||
{ prefix: "var outerGenExpr = function*() { ", suffix: " };" },
|
||||
{ prefix: "var outerObject = { outerMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerObject = { *outerGenMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { outerMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { *outerGenMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { static outerStaticMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { static *outerStaticGenMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { outerMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { *outerGenMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { static outerStaticMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { static *outerStaticGenMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerArrow = () => { ", suffix: " };" },
|
||||
{ prefix: "async function outerAsyncFunction() { ", suffix: " }" },
|
||||
{ prefix: "var outerAsyncFuncExpr = async function() { ", suffix: " };" },
|
||||
{ prefix: "var outerAsyncArrowFunc = async () => { ", suffix: " };" },
|
||||
{ prefix: "var outerObject = { async outerAsyncMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { async outerAsyncMethod() { ", suffix: " } };" },
|
||||
{ prefix: "var outerClassExpr = class C { static async outerStaticAsyncMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { async outerAsyncMethod() { ", suffix: " } };" },
|
||||
{ prefix: "class outerClass { static async outerStaticAsyncMethod() { ", suffix: " } };" },
|
||||
];
|
||||
|
||||
var testData = [
|
||||
`var async = 1; return async;`,
|
||||
`let async = 1; return async;`,
|
||||
`const async = 1; return async;`,
|
||||
`function async() {} return async();`,
|
||||
`var async = async => async; return async();`,
|
||||
`function foo() { var await = 1; return await; }`,
|
||||
`function foo(await) { return await; }`,
|
||||
`function* foo() { var await = 1; return await; }`,
|
||||
`function* foo(await) { return await; }`,
|
||||
`var f = () => { var await = 1; return await; }`,
|
||||
`var O = { method() { var await = 1; return await; } };`,
|
||||
`var O = { method(await) { return await; } };`,
|
||||
`var O = { *method() { var await = 1; return await; } };`,
|
||||
`var O = { *method(await) { return await; } };`,
|
||||
|
||||
`(function await() {})`,
|
||||
];
|
||||
|
||||
for (let context of contextData) {
|
||||
for (let test of testData) {
|
||||
let script = context.prefix + test + context.suffix;
|
||||
testSyntax(script);
|
||||
testSyntax(`"use strict"; ${script}`);
|
||||
}
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
(function testTopLevelAsyncAwaitSyntaxSloppyMode() {
|
||||
testSyntaxError(`({ async
|
||||
foo() {} })`);
|
||||
testSyntaxError(`({async = 1})`);
|
||||
testSyntaxError(`var asyncFn = async function await() {}`);
|
||||
testSyntaxError(`var asyncFn = async () => var await = 'test';`);
|
||||
testSyntaxError(`var asyncFn = async () => { var await = 'test'; };`);
|
||||
testSyntaxError(`var asyncFn = async await => await + 'test'`);
|
||||
testSyntaxError(`var asyncFn = async function(await) {}`);
|
||||
testSyntaxError(`var asyncFn = async function withName(await) {}`);
|
||||
testSyntaxError(`var asyncFn = async (await) => 'test';`);
|
||||
testSyntaxError(`async function asyncFunctionDeclaration(await) {}`);
|
||||
|
||||
testSyntaxError(`var outerObject = { async method(a, a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async ['meth' + 'od'](a, a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async 'method'(a, a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async 0(a, a) {} }`);
|
||||
|
||||
testSyntaxError(`var outerObject = { async method(a, {a}) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method({a}, a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method({a}, {a}) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method(a, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method({a}, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method(a, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { async method({a, ...a}) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function(a, {a}) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function({a}, a) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function({a}, {a}) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function(a, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function({a}, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function(a, ...a) {} }`);
|
||||
testSyntaxError(`var outerObject = { func: async function({a, ...a}) {} }`);
|
||||
|
||||
testSyntaxError(`var asyncArrowFn = async() => await;`);
|
||||
|
||||
testSyntaxError(`var O = { *async asyncGeneratorMethod() {} };`);
|
||||
testSyntaxError(`var O = { async asyncGeneratorMethod*() {} };`);
|
||||
|
||||
testSyntaxError(`var asyncFn = async function(x = await 1) { return x; }`);
|
||||
testSyntaxError(`async function f(x = await 1) { return x; }`);
|
||||
testSyntaxError(`var f = async(x = await 1) => x;`);
|
||||
testSyntaxError(`var O = { async method(x = await 1) { return x; } };`);
|
||||
|
||||
testSyntaxError(`function* outerGenerator() { var asyncArrowFn = async yield => 1; }`);
|
||||
testSyntaxError(`function* outerGenerator() { var asyncArrowFn = async(yield) => 1; }`);
|
||||
testSyntaxError(`function* outerGenerator() { var asyncArrowFn = async(x = yield) => 1; }`);
|
||||
testSyntaxError(`function* outerGenerator() { var asyncArrowFn = async({x = yield}) => 1; }`);
|
||||
|
||||
testSyntaxError(`class C { async constructor() {} }`);
|
||||
testSyntaxError(`class C {}; class C2 extends C { async constructor() {} }`);
|
||||
testSyntaxError(`class C { static async prototype() {} }`);
|
||||
testSyntaxError(`class C {}; class C2 extends C { static async prototype() {} }`);
|
||||
|
||||
testSyntaxError(`var f = async() => ((async(x = await 1) => x)();`);
|
||||
|
||||
// Henrique Ferreiro's bug (tm)
|
||||
testSyntaxError(`(async function foo1() { } foo2 => 1)`);
|
||||
testSyntaxError(`(async function foo3() { } () => 1)`);
|
||||
testSyntaxError(`(async function foo4() { } => 1)`);
|
||||
testSyntaxError(`(async function() { } foo5 => 1)`);
|
||||
testSyntaxError(`(async function() { } () => 1)`);
|
||||
testSyntaxError(`(async function() { } => 1)`);
|
||||
testSyntaxError(`(async.foo6 => 1)`);
|
||||
testSyntaxError(`(async.foo7 foo8 => 1)`);
|
||||
testSyntaxError(`(async.foo9 () => 1)`);
|
||||
testSyntaxError(`(async().foo10 => 1)`);
|
||||
testSyntaxError(`(async().foo11 foo12 => 1)`);
|
||||
testSyntaxError(`(async().foo13 () => 1)`);
|
||||
testSyntaxError(`(async['foo14'] => 1)`);
|
||||
testSyntaxError(`(async['foo15'] foo16 => 1)`);
|
||||
testSyntaxError(`(async['foo17'] () => 1)`);
|
||||
testSyntaxError(`(async()['foo18'] => 1)`);
|
||||
testSyntaxError(`(async()['foo19'] foo20 => 1)`);
|
||||
testSyntaxError(`(async()['foo21'] () => 1`);
|
||||
testSyntaxError("(async`foo22` => 1)");
|
||||
testSyntaxError("(async`foo23` foo24 => 1)");
|
||||
testSyntaxError("(async`foo25` () => 1)");
|
||||
testSyntaxError("(async`foo26`.bar27 => 1)");
|
||||
testSyntaxError("(async`foo28`.bar29 foo30 => 1)");
|
||||
testSyntaxError("(async`foo31`.bar32 () => 1)");
|
||||
|
||||
// assert that errors are still thrown for calls that may have been async functions
|
||||
testSyntaxError(`function async() {}
|
||||
async({ foo33 = 1 })`);
|
||||
})();
|
||||
|
||||
(function testTopLevelAsyncAwaitSyntaxStrictMode() {
|
||||
testSyntaxError(`"use strict"; ({ async
|
||||
foo() {} })`);
|
||||
testSyntaxError(`"use strict"; ({async = 1})`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async function await() {}`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async () => var await = 'test';`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async () => { var await = 'test'; };`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async await => await + 'test'`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async function(await) {}`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async function withName(await) {}`);
|
||||
testSyntaxError(`"use strict"; var asyncFn = async (await) => 'test';`);
|
||||
testSyntaxError(`"use strict"; async function asyncFunctionDeclaration(await) {}`);
|
||||
|
||||
testSyntaxError(`"use strict"; var outerObject = { async method(a, a) {} }`);
|
||||
testSyntaxError(`"use strict"; var outerObject = { async ['meth' + 'od'](a, a) {} }`);
|
||||
testSyntaxError(`"use strict"; var outerObject = { async 'method'(a, a) {} }`);
|
||||
testSyntaxError(`"use strict"; var outerObject = { async 0(a, a) {} }`);
|
||||
|
||||
testSyntaxError(`"use strict"; var asyncArrowFn = async() => await;`);
|
||||
|
||||
testSyntaxError(`"use strict"; var O = { *async asyncGeneratorMethod() {} };`);
|
||||
testSyntaxError(`"use strict"; var O = { async asyncGeneratorMethod*() {} };`);
|
||||
|
||||
testSyntax(`"use strict"; var O = { async function() {} };`);
|
||||
|
||||
testSyntaxError(`"use strict"; var asyncFn = async function(x = await 1) { return x; }`);
|
||||
testSyntaxError(`"use strict"; async function f(x = await 1) { return x; }`);
|
||||
testSyntaxError(`"use strict"; var f = async(x = await 1) => x;`);
|
||||
testSyntaxError(`"use strict"; var O = { async method(x = await 1) { return x; } };`);
|
||||
|
||||
testSyntaxError(`"use strict"; function* outerGenerator() { var asyncArrowFn = async yield => 1; }`);
|
||||
testSyntaxError(`"use strict"; function* outerGenerator() { var asyncArrowFn = async(yield) => 1; }`);
|
||||
testSyntaxError(`"use strict"; function* outerGenerator() { var asyncArrowFn = async(x = yield) => 1; }`);
|
||||
testSyntaxError(`"use strict"; function* outerGenerator() { var asyncArrowFn = async({x = yield}) => 1; }`);
|
||||
|
||||
testSyntaxError(`"use strict"; class C { async constructor() {} }`);
|
||||
testSyntaxError(`"use strict"; class C {}; class C2 extends C { async constructor() {} }`);
|
||||
testSyntaxError(`"use strict"; class C { static async prototype() {} }`);
|
||||
testSyntaxError(`"use strict"; class C {}; class C2 extends C { static async prototype() {} }`);
|
||||
|
||||
testSyntaxError(`"use strict"; var f = async() => ((async(x = await 1) => x)();`);
|
||||
|
||||
// Henrique Ferreiro's bug (tm)
|
||||
testSyntaxError(`"use strict"; (async function foo1() { } foo2 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async function foo3() { } () => 1)`);
|
||||
testSyntaxError(`"use strict"; (async function foo4() { } => 1)`);
|
||||
testSyntaxError(`"use strict"; (async function() { } foo5 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async function() { } () => 1)`);
|
||||
testSyntaxError(`"use strict"; (async function() { } => 1)`);
|
||||
testSyntaxError(`"use strict"; (async.foo6 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async.foo7 foo8 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async.foo9 () => 1)`);
|
||||
testSyntaxError(`"use strict"; (async().foo10 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async().foo11 foo12 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async().foo13 () => 1)`);
|
||||
testSyntaxError(`"use strict"; (async['foo14'] => 1)`);
|
||||
testSyntaxError(`"use strict"; (async['foo15'] foo16 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async['foo17'] () => 1)`);
|
||||
testSyntaxError(`"use strict"; (async()['foo18'] => 1)`);
|
||||
testSyntaxError(`"use strict"; (async()['foo19'] foo20 => 1)`);
|
||||
testSyntaxError(`"use strict"; (async()['foo21'] () => 1)`);
|
||||
testSyntaxError('"use strict"; (async`foo22` => 1)');
|
||||
testSyntaxError('"use strict"; (async`foo23` foo24 => 1)');
|
||||
testSyntaxError('"use strict"; (async`foo25` () => 1)');
|
||||
testSyntaxError('"use strict"; (async`foo26`.bar27 => 1)');
|
||||
testSyntaxError('"use strict"; (async`foo28`.bar29 foo30 => 1)');
|
||||
testSyntaxError('"use strict"; (async`foo31`.bar32 () => 1)');
|
||||
|
||||
// assert that errors are still thrown for calls that may have been async functions
|
||||
testSyntaxError(`"use strict"; function async() {}
|
||||
async({ foo33 = 1 })`);
|
||||
|
||||
testSyntaxError(`"use strict"; var O = { async method(eval) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async ['meth' + 'od'](eval) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async 'method'(eval) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async 0(eval) {} }`);
|
||||
|
||||
testSyntaxError(`"use strict"; var O = { async method(arguments) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async ['meth' + 'od'](arguments) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async 'method'(arguments) {} }`);
|
||||
testSyntaxError(`"use strict"; var O = { async 0(arguments) {} }`);
|
||||
|
||||
testSyntaxError(`"use strict"; var O = { async method(dupe, dupe) {} }`);
|
||||
})();
|
||||
|
||||
(function testAwaitInFormalParameters() {
|
||||
var testData = [
|
||||
`async function f(await) {}`,
|
||||
`async function f(...await) {}`,
|
||||
`async function f(await = 1) {}`,
|
||||
`async function f([await]) {}`,
|
||||
`async function f([await = 1]) {}`,
|
||||
`async function f({ await }) {}`,
|
||||
`async function f({ await = 1 }) {}`,
|
||||
`async function f({ } = await) {}`,
|
||||
|
||||
`(async function(await) {})`,
|
||||
`(async function(...await) {})`,
|
||||
`(async function(await = 1) {})`,
|
||||
`(async function([await]) {})`,
|
||||
`(async function([await = 1]) {})`,
|
||||
`(async function({ await }) {})`,
|
||||
`(async function({ await = 1 }) {})`,
|
||||
`(async function({ } = await) {})`,
|
||||
|
||||
`var asyncArrow = async(await) => 1;`,
|
||||
`var asyncArrow = async(await) => {};`,
|
||||
`var asyncArrow = async(...await) => 1;`,
|
||||
`var asyncArrow = async(...await) => {};`,
|
||||
`var asyncArrow = async(await = 1) => 1;`,
|
||||
`var asyncArrow = async(await = 1) => {};`,
|
||||
`var asyncArrow = async([await]) => 1;`,
|
||||
`var asyncArrow = async([await]) => {};`,
|
||||
`var asyncArrow = async([await = 1]) => 1;`,
|
||||
`var asyncArrow = async([await = 1]) => {};`,
|
||||
`var asyncArrow = async([] = await) => 1;`,
|
||||
`var asyncArrow = async([] = await) => {};`,
|
||||
`var asyncArrow = async({ await }) => 1;`,
|
||||
`var asyncArrow = async({ await } ) => {};`,
|
||||
`var asyncArrow = async({ await = 1}) => 1;`,
|
||||
`var asyncArrow = async({ await = 1}) => {};`,
|
||||
`var asyncArrow = async({ } = await) => 1;`,
|
||||
`var asyncArrow = async({ } = await) => {};`,
|
||||
|
||||
`({ async method(await) {} })`,
|
||||
`({ async method(...await) {} })`,
|
||||
`({ async method(await = 1) {} })`,
|
||||
`({ async method([await]) {} })`,
|
||||
`({ async method([await = 1]) {} })`,
|
||||
`({ async method({ await }) {} })`,
|
||||
`({ async method({ await = 1 }) {} })`,
|
||||
`({ async method({ } = await) {} })`,
|
||||
|
||||
`(class { async method(await) {} })`,
|
||||
`(class { async method(...await) {} })`,
|
||||
`(class { async method(await = 1) {} })`,
|
||||
`(class { async method([await]) {} })`,
|
||||
`(class { async method([await = 1]) {} })`,
|
||||
`(class { async method({ await }) {} })`,
|
||||
`(class { async method({ await = 1 }) {} })`,
|
||||
`(class { async method({ } = await) {} })`,
|
||||
|
||||
`(class { static async method(await) {} })`,
|
||||
`(class { static async method(...await) {} })`,
|
||||
`(class { static async method(await = 1) {} })`,
|
||||
`(class { static async method([await]) {} })`,
|
||||
`(class { static async method([await = 1]) {} })`,
|
||||
`(class { static async method({ await }) {} })`,
|
||||
`(class { static async method({ await = 1 }) {} })`,
|
||||
`(class { static async method({ } = await) {} })`,
|
||||
];
|
||||
|
||||
for (let script of testData) {
|
||||
testSyntaxError(script);
|
||||
testSyntaxError(`"use strict"; ${script}`);
|
||||
|
||||
var nested = `var await; var f = (async function() { ${script} });`;
|
||||
testSyntaxError(nested);
|
||||
testSyntaxError(`"use strict"; ${nested}`);
|
||||
}
|
||||
})();
|
||||
|
||||
testSyntaxError(`
|
||||
async function fn(arguments) {
|
||||
"use strict";
|
||||
}
|
||||
`, `SyntaxError: Invalid parameters or function name in strict mode.`);
|
||||
|
||||
testSyntaxError(`
|
||||
async function fn(eval) {
|
||||
"use strict";
|
||||
}
|
||||
`, `SyntaxError: Invalid parameters or function name in strict mode.`);
|
||||
|
||||
testSyntaxError(`
|
||||
async function arguments() {
|
||||
"use strict";
|
||||
}
|
||||
`, `SyntaxError: 'arguments' is not a valid function name in strict mode.`);
|
||||
|
||||
testSyntaxError(`
|
||||
async function eval() {
|
||||
"use strict";
|
||||
}
|
||||
`, `SyntaxError: 'eval' is not a valid function name in strict mode.`);
|
||||
|
||||
testSyntaxError(`
|
||||
async function fn(a) {
|
||||
let a = 1;
|
||||
}
|
||||
`, `SyntaxError: Cannot declare a let variable twice: 'a'.`);
|
||||
|
||||
testSyntaxError(`
|
||||
async function fn(b) {
|
||||
const b = 1;
|
||||
}
|
||||
`, `SyntaxError: Cannot declare a const variable twice: 'b'.`);
|
||||
|
||||
(function testMethodDefinition() {
|
||||
testSyntax("({ async [foo]() {} })");
|
||||
testSyntax("({ async [Symbol.asyncIterator]() {} })");
|
||||
testSyntax("({ async 0() {} })");
|
||||
testSyntax("({ async 'string'() {} })");
|
||||
testSyntax("({ async ident() {} })");
|
||||
|
||||
testSyntax("(class { async [foo]() {} })");
|
||||
testSyntax("(class { async [Symbol.asyncIterator]() {} })");
|
||||
testSyntax("(class { async 0() {} })");
|
||||
testSyntax("(class { async 'string'() {} })");
|
||||
testSyntax("(class { async ident() {} })");
|
||||
|
||||
testSyntax("(class { static async [foo]() {} })");
|
||||
testSyntax("(class { static async [Symbol.asyncIterator]() {} })");
|
||||
testSyntax("(class { static async 0() {} })");
|
||||
testSyntax("(class { static async 'string'() {} })");
|
||||
testSyntax("(class { static async ident() {} })");
|
||||
})();
|
||||
|
||||
(function testLineTerminator() {
|
||||
let testLineFeedErrors = (prefix, suffix) => {
|
||||
testSyntaxError(`${prefix}// comment
|
||||
${suffix}`);
|
||||
testSyntaxError(`${prefix}/* comment
|
||||
*/ ${suffix}`);
|
||||
testSyntaxError(`${prefix}
|
||||
${suffix}`);
|
||||
testSyntaxError(`"use strict";${prefix}// comment
|
||||
${suffix}`);
|
||||
testSyntaxError(`"use strict";${prefix}/* comment
|
||||
*/ ${suffix}`);
|
||||
testSyntaxError(`"use strict";${prefix}
|
||||
${suffix}`);
|
||||
};
|
||||
|
||||
let testLineFeeds = (prefix, suffix) => {
|
||||
testSyntax(`${prefix}// comment
|
||||
${suffix}`);
|
||||
testSyntax(`${prefix}/* comment
|
||||
*/${suffix}`);
|
||||
testSyntax(`${prefix}
|
||||
${suffix}`);
|
||||
testSyntax(`"use strict";${prefix}// comment
|
||||
${suffix}`);
|
||||
testSyntax(`"use strict";${prefix}/* comment
|
||||
*/${suffix}`);
|
||||
testSyntax(`"use strict";${prefix}
|
||||
${suffix}`);
|
||||
};
|
||||
|
||||
let tests = [
|
||||
// ObjectLiteral AsyncMethodDefinition
|
||||
{ prefix: "({ async", suffix: "method() {} }).method" },
|
||||
|
||||
// ClassLiteral AsyncMethodDefinition
|
||||
{ prefix: "(class { async", suffix: "method() {} }).prototype.method" },
|
||||
|
||||
// AsyncArrowFunctions
|
||||
{ prefix: "(async", suffix: "param => 1)" },
|
||||
{ prefix: "(async", suffix: "(param) => 1)" },
|
||||
{ prefix: "(async", suffix: "param => {})" },
|
||||
{ prefix: "(async", suffix: "(param) => {})" },
|
||||
];
|
||||
|
||||
for (let { prefix, suffix } of tests) {
|
||||
testSyntax(`${prefix} ${suffix}`);
|
||||
testSyntax(`"use strict";${prefix} ${suffix}`);
|
||||
shouldBe("function", typeof eval(`${prefix} ${suffix}`));
|
||||
shouldBe("function", typeof eval(`"use strict";${prefix} ${suffix}`));
|
||||
testLineFeedErrors(prefix, suffix);
|
||||
}
|
||||
|
||||
// AsyncFunctionDeclaration
|
||||
testSyntax("async function foo() {}");
|
||||
testLineFeeds("async", "function foo() {}");
|
||||
|
||||
// AsyncFunctionExpression
|
||||
testSyntax("var x = async function foo() {}");
|
||||
testSyntax("'use strict';var x = async function foo() {}");
|
||||
testLineFeeds("var x = async", "function foo() {}");
|
||||
testLineFeedErrors("var x = async", "function() {}");
|
||||
})();
|
|
@ -1,32 +0,0 @@
|
|||
// Copyright (C) Copyright 2016 the v8 project authors. All rights reserved.
|
||||
|
||||
//@ runNoCJIT("--gcMaxHeapSize=2000000")
|
||||
|
||||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
|
||||
let out;
|
||||
|
||||
async function thrower() { throw undefined; }
|
||||
|
||||
async function throwLoop() {
|
||||
for (let i = 0; i < 8000; i++) {
|
||||
try {
|
||||
await thrower();
|
||||
} catch (error) {
|
||||
void 0;
|
||||
}
|
||||
}
|
||||
out = 2;
|
||||
}
|
||||
|
||||
throwLoop();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
shouldBe(out, 2);
|
|
@ -1,13 +0,0 @@
|
|||
function test(cond)
|
||||
{
|
||||
if (cond) {
|
||||
var func = async function () {};
|
||||
return func;
|
||||
}
|
||||
|
||||
return 42;
|
||||
}
|
||||
noInline(test);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i)
|
||||
test();
|
|
@ -1,44 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
}
|
||||
|
||||
var AsyncFunctionPrototype = async function() {}.__proto__;
|
||||
|
||||
function testAsyncFunctionExpression()
|
||||
{
|
||||
return async function()
|
||||
{
|
||||
await 42;
|
||||
return 1;
|
||||
};
|
||||
}
|
||||
noInline(testAsyncFunctionExpression);
|
||||
|
||||
function testAsyncFunctionDeclaration()
|
||||
{
|
||||
async function asyncFn()
|
||||
{
|
||||
await 42;
|
||||
return 1;
|
||||
}
|
||||
|
||||
return asyncFn;
|
||||
}
|
||||
noInline(testAsyncFunctionDeclaration);
|
||||
|
||||
function testAsyncArrowFunction()
|
||||
{
|
||||
return async() =>
|
||||
{
|
||||
await 42;
|
||||
return 1;
|
||||
};
|
||||
}
|
||||
noInline(testAsyncArrowFunction);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
shouldBe(testAsyncFunctionExpression().__proto__, AsyncFunctionPrototype);
|
||||
shouldBe(testAsyncFunctionDeclaration().__proto__, AsyncFunctionPrototype);
|
||||
shouldBe(testAsyncArrowFunction().__proto__, AsyncFunctionPrototype);
|
||||
}
|
|
@ -1,38 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function call(o) { o.x = 3; }
|
||||
noInline(call);
|
||||
|
||||
function sink (p, q) {
|
||||
async function f() { };
|
||||
if (p) {
|
||||
call(f); // Force allocation of f
|
||||
if (q) {
|
||||
OSRExit();
|
||||
}
|
||||
return f;
|
||||
}
|
||||
return { 'x': 2 };
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 100000; ++i) {
|
||||
var o = sink(true, false);
|
||||
shouldBe(o.__proto__, AsyncFunctionPrototype);
|
||||
if (o.x != 3)
|
||||
throw "Error: expected o.x to be 2 but is " + result;
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Check that the function is properly allocated on OSR exit
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
if (f.x != 3)
|
||||
throw "Error: expected o.x to be 3 but is " + result;
|
|
@ -1,42 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function shouldBeAsync(expected, promise, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
promise.then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (hadError)
|
||||
throw actual;
|
||||
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function sink (p, q) {
|
||||
async function g(x) { return x; };
|
||||
if (p) { if (q) OSRExit(); return g; }
|
||||
async function f(x) { return x; };
|
||||
return f;
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var f = sink(true, false);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Check that the function is properly allocated on OSR exit
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
|
@ -1,49 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function shouldBeAsync(expected, promise, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
promise.then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (hadError)
|
||||
throw actual;
|
||||
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function sink (p, q) {
|
||||
async function g(x) { return x; };
|
||||
if (p) { if (q) g.inner = 42; return g; }
|
||||
async function f(x) { return x; };
|
||||
return f;
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Test the allocation on the implicit inner else branch
|
||||
var f = sink(true, false);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(12, f(12));
|
||||
|
||||
// Check that the allocation did not sink beyond the property assignment
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
var result = f.inner;
|
||||
if (result !== 42)
|
||||
throw "Error: inner should be 42 but is " + result;
|
|
@ -1,36 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
}
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function call(o) { o.x = 3; }
|
||||
noInline(call);
|
||||
|
||||
function sink (p, q) {
|
||||
var f = async function() { };
|
||||
if (p) {
|
||||
call(f); // Force allocation of f
|
||||
if (q) {
|
||||
OSRExit();
|
||||
}
|
||||
return f;
|
||||
}
|
||||
return { 'x': 2 };
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 100000; ++i) {
|
||||
var o = sink(true, false);
|
||||
shouldBe(o.__proto__, AsyncFunctionPrototype);
|
||||
if (o.x != 3)
|
||||
throw "Error: expected o.x to be 2 but is " + result;
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Check that the function is properly allocated on OSR exit
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
if (f.x != 3)
|
||||
throw "Error: expected o.x to be 3 but is " + result;
|
|
@ -1,40 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
function shouldBeAsync(expected, promise, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
promise.then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (hadError)
|
||||
throw actual;
|
||||
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function sink (p, q) {
|
||||
var g = async function (x) { return x; };
|
||||
if (p) { if (q) OSRExit(); return g; }
|
||||
return async function (x) { return x; };
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var f = sink(true, false);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Check that the function is properly allocated on OSR exit
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
|
@ -1,46 +0,0 @@
|
|||
function shouldBe(expected, actual, msg = "") {
|
||||
if (msg)
|
||||
msg = " for " + msg;
|
||||
if (actual !== expected)
|
||||
throw new Error("bad value" + msg + ": " + actual + ". Expected " + expected);
|
||||
}
|
||||
|
||||
function shouldBeAsync(expected, promise, msg) {
|
||||
let actual;
|
||||
var hadError = false;
|
||||
promise.then(function(value) { actual = value; },
|
||||
function(error) { hadError = true; actual = error; });
|
||||
drainMicrotasks();
|
||||
|
||||
if (hadError)
|
||||
throw actual;
|
||||
|
||||
shouldBe(expected, actual, msg);
|
||||
}
|
||||
|
||||
var AsyncFunctionPrototype = async function(){}.__proto__;
|
||||
|
||||
function sink (p, q) {
|
||||
var g = async function(x) { return x; };
|
||||
if (p) { if (q) g.inner = 42; return g; }
|
||||
return async function(x) { return x; };
|
||||
}
|
||||
noInline(sink);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(42, f(42));
|
||||
}
|
||||
|
||||
// At this point, the function should be compiled down to the FTL
|
||||
|
||||
// Test the allocation on the implicit inner else branch
|
||||
var f = sink(true, false);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBeAsync(12, f(12));
|
||||
|
||||
// Check that the allocation did not sink beyond the property assignment
|
||||
var f = sink(true, true);
|
||||
shouldBe(f.__proto__, AsyncFunctionPrototype);
|
||||
shouldBe(42, f.inner);
|
|
@ -1,36 +0,0 @@
|
|||
function assert(b) {
|
||||
if (!b)
|
||||
throw new Error("Bad");
|
||||
}
|
||||
|
||||
async function* asyncIterator() {
|
||||
yield 42;
|
||||
}
|
||||
|
||||
function test1() {
|
||||
let p = asyncIterator();
|
||||
p.next().then((x) => {
|
||||
assert(x.value === 42);
|
||||
assert(x.done === false);
|
||||
});
|
||||
p.__proto__ = {};
|
||||
assert(p.next === undefined);
|
||||
}
|
||||
test1();
|
||||
|
||||
let error = null;
|
||||
async function test2() {
|
||||
let p2 = asyncIterator();
|
||||
p2.__proto__ = {};
|
||||
try {
|
||||
for await (let x of p2) {
|
||||
throw new Error("Bad!");
|
||||
}
|
||||
}
|
||||
catch(e) {
|
||||
error = e;
|
||||
}
|
||||
}
|
||||
test2();
|
||||
assert(error instanceof TypeError);
|
||||
assert(error.message === "undefined is not a function (near '...x of p2...')");
|
|
@ -1,468 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
const Logger = function () {
|
||||
var log = [];
|
||||
|
||||
this.logEvent = (type, value, done) => {
|
||||
log.push({ type, value, done});
|
||||
};
|
||||
this.logFulfilledEvent = (value, done) => {
|
||||
this.logEvent('fulfilled', value, done);
|
||||
};
|
||||
this.logRejectEvent = error => {
|
||||
this.logEvent('reject', error.toString(), true);
|
||||
};
|
||||
this.logCatchEvent = value => {
|
||||
this.logEvent('catch', value, true);
|
||||
};
|
||||
this.getLogger = () => log;
|
||||
|
||||
this.clear = () => {
|
||||
log = [];
|
||||
}
|
||||
};
|
||||
|
||||
const fulfillSpy = logger => result => logger.logFulfilledEvent(result.value, result.done);
|
||||
const rejectSpy = logger => error => logger.logRejectEvent(error);
|
||||
const catchSpy = logger => error => logger.logCatchEvent(error);
|
||||
|
||||
const assertLogger = function (loggerObject) {
|
||||
const logger = loggerObject.getLogger();
|
||||
|
||||
var _assertLogger = function () {
|
||||
let index = 0;
|
||||
|
||||
const isNotOutOfLength = () => {
|
||||
assert(index < logger.length, true, `Index is greater then log length`);
|
||||
}
|
||||
|
||||
this.fullfilled = function (expectedValue, message = 'on fulfill') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.fullfilledDone = function (expectedValue, message = 'on fulfill with done true') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.rejected = function (error, message = 'on reject') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'reject', msg);
|
||||
assert(step.value, error.toString(), msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.catched = function (expectedError, message = 'on catch') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'catch', msg);
|
||||
assert(step.value, expectedError, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.isFinal = function (message = '') {
|
||||
assert(index, logger.length, `expected final step: ${message}`);
|
||||
};
|
||||
};
|
||||
|
||||
return new _assertLogger();
|
||||
};
|
||||
|
||||
const getPromise = promiseHolder => {
|
||||
return new Promise((resolve, reject) => {
|
||||
promiseHolder.resolve = resolve;
|
||||
promiseHolder.reject = reject;
|
||||
});
|
||||
};
|
||||
|
||||
var logger = new Logger();
|
||||
const someValue = 'some-value';
|
||||
const errorMessage = 'error-message';
|
||||
|
||||
function * foo(value) {
|
||||
let re = yield '1:' + value;
|
||||
re = yield '2:' + re;
|
||||
re = yield '3:' + re;
|
||||
return 'end foo:' + re;
|
||||
}
|
||||
|
||||
async function * boo(value) {
|
||||
let reply = yield '0:' + value;
|
||||
reply = yield* foo(reply);
|
||||
yield '4:' + reply;
|
||||
}
|
||||
|
||||
var b = boo('init');
|
||||
const errorprint = error => print(error);
|
||||
b.next('0').then(fulfillSpy(logger), errorprint);
|
||||
b.next('1').then(fulfillSpy(logger), errorprint);
|
||||
b.next('2').then(fulfillSpy(logger), errorprint);
|
||||
b.next('3').then(fulfillSpy(logger), errorprint);
|
||||
b.next('4').then(fulfillSpy(logger), errorprint);
|
||||
b.next('5').then(fulfillSpy(logger), errorprint);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.fullfilled('1:1')
|
||||
.fullfilled('2:2')
|
||||
.fullfilled('3:3')
|
||||
.fullfilled('4:end foo:4')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var b2 = boo(':value');
|
||||
|
||||
b2.next(':0').then(fulfillSpy(logger));
|
||||
b2.next(':1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next(':2').then(fulfillSpy(logger));
|
||||
b2.next(':3').then(fulfillSpy(logger));
|
||||
b2.next(':4').then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0::value')
|
||||
.fullfilled('1::1')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var b2 = boo('#value');
|
||||
|
||||
b2.next('#0').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#2').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#3').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#4').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:#value')
|
||||
.fullfilled('1:#1')
|
||||
.fullfilled('2:#2')
|
||||
.rejected(new Error(errorMessage)) // TODO: Check if this correct
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
function * bar() {
|
||||
yield '1';
|
||||
yield '2';
|
||||
throw new Error(errorMessage);
|
||||
yield '3';
|
||||
return 'end foo';
|
||||
}
|
||||
|
||||
async function * baz() {
|
||||
yield '0';
|
||||
yield* bar();
|
||||
yield '4';
|
||||
}
|
||||
|
||||
logger.clear();
|
||||
var bz1 = baz();
|
||||
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
let promiseHolder = {};
|
||||
|
||||
function *joo() {
|
||||
yield '1';
|
||||
yield getPromise(promiseHolder);
|
||||
}
|
||||
|
||||
async function *goo () {
|
||||
yield '0';
|
||||
yield* joo();
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('2')
|
||||
.fullfilled('3')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
|
@ -1,955 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
const Logger = function () {
|
||||
var log = [];
|
||||
|
||||
this.logEvent = (type, value, done) => {
|
||||
log.push({ type, value, done});
|
||||
};
|
||||
this.logFulfilledEvent = (value, done) => {
|
||||
this.logEvent('fulfilled', value, done);
|
||||
};
|
||||
this.logRejectEvent = error => {
|
||||
this.logEvent('reject', error.toString(), true);
|
||||
};
|
||||
this.logCatchEvent = value => {
|
||||
this.logEvent('catch', value, true);
|
||||
};
|
||||
this.getLogger = () => log;
|
||||
|
||||
this.clear = () => {
|
||||
log = [];
|
||||
}
|
||||
};
|
||||
|
||||
const fulfillSpy = logger => result => logger.logFulfilledEvent(result.value, result.done);
|
||||
const rejectSpy = logger => error => logger.logRejectEvent(error);
|
||||
const catchSpy = logger => error => logger.logCatchEvent(error);
|
||||
|
||||
const assertLogger = function (loggerObject) {
|
||||
const logger = loggerObject.getLogger();
|
||||
|
||||
var _assertLogger = function () {
|
||||
let index = 0;
|
||||
|
||||
const isNotOutOfLength = () => {
|
||||
assert(index < logger.length, true, `Index is greater then log length`);
|
||||
}
|
||||
|
||||
this.fullfilled = function (expectedValue, message = 'on fulfill') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.fullfilledDone = function (expectedValue, message = 'on fulfill with done true') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.rejected = function (error, message = 'on reject') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'reject', msg);
|
||||
assert(step.value, error.toString(), msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.catched = function (expectedError, message = 'on catch') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'catch', msg);
|
||||
assert(step.value, expectedError, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.isFinal = function (message = '') {
|
||||
assert(index, logger.length, `expected final step: ${message}`);
|
||||
};
|
||||
};
|
||||
|
||||
return new _assertLogger();
|
||||
};
|
||||
|
||||
const getPromise = promiseHolder => {
|
||||
return new Promise((resolve, reject) => {
|
||||
promiseHolder.resolve = resolve;
|
||||
promiseHolder.reject = reject;
|
||||
});
|
||||
};
|
||||
|
||||
async function*generator () {
|
||||
yield 42;
|
||||
}
|
||||
|
||||
var passed = false;
|
||||
var iterator = generator();
|
||||
|
||||
iterator.next().then(function(step) {
|
||||
if(iterator[Symbol.asyncIterator]() === iterator && step.done === false && step.value === 42) passed = true;
|
||||
});
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(passed, true, '');
|
||||
|
||||
assert(generator[Symbol.toStringTag], 'AsyncGeneratorFunction');
|
||||
assert(Object.prototype.toString.call(generator), '[object AsyncGeneratorFunction]');
|
||||
|
||||
|
||||
var holder = {};
|
||||
|
||||
var promise = getPromise(holder);
|
||||
var logger = new Logger();
|
||||
|
||||
async function * baz(value) {
|
||||
var t = await promise;
|
||||
let reply = yield t + ' data-0' + value;
|
||||
let last = yield t + ' data-1' + reply;
|
||||
return 'data-result' + last;
|
||||
}
|
||||
|
||||
var bz1 = baz(':init');
|
||||
|
||||
bz1.next(':0').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next(':1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next(':2').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.isFinal();
|
||||
|
||||
holder.resolve('abcd');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-0:init')
|
||||
.fullfilled('abcd data-1:1')
|
||||
.fullfilledDone('data-result:2')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
promise = getPromise(holder);
|
||||
|
||||
var bz2 = baz(':init');
|
||||
|
||||
bz2.next(':0').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz2.next(':1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz2.next(':2').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.isFinal();
|
||||
|
||||
holder.reject('abcd');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected('abcd')
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
var holder1 = {}; holder2 = {};
|
||||
var promise1 = getPromise(holder1);
|
||||
var promise2 = getPromise(holder2);
|
||||
|
||||
logger.clear();
|
||||
|
||||
async function *boo() {
|
||||
var t1 = await promise1;
|
||||
yield t1 + ' data-1';
|
||||
yield t1 + ' data-2';
|
||||
var t2 = await promise2;
|
||||
yield t2 + ' data-3';
|
||||
yield t2 + ' data-4';
|
||||
return 'data-5';
|
||||
};
|
||||
|
||||
var b = boo();
|
||||
|
||||
b.next().then(fulfillSpy(logger));
|
||||
b.next().then(fulfillSpy(logger));
|
||||
b.next().then(fulfillSpy(logger));
|
||||
b.next().then(fulfillSpy(logger));
|
||||
b.next().then(fulfillSpy(logger));
|
||||
b.next().then(fulfillSpy(logger));
|
||||
|
||||
assertLogger(logger)
|
||||
.isFinal('async generator should not resolve any promise until await is no resolved');
|
||||
|
||||
holder1.resolve('abcd');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-1')
|
||||
.fullfilled('abcd data-2')
|
||||
.isFinal();
|
||||
|
||||
holder2.resolve('efgh');
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-1')
|
||||
.fullfilled('abcd data-2')
|
||||
.fullfilled('efgh data-3')
|
||||
.fullfilled('efgh data-4')
|
||||
.fullfilledDone('data-5')
|
||||
.fullfilledDone(undefined, error => print(error))
|
||||
.isFinal();
|
||||
|
||||
holder = {};
|
||||
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *foo() {
|
||||
var t = await 'abcd';
|
||||
yield t + ' data-5';
|
||||
var t2 = await promise;
|
||||
yield t2 + ' data-6';
|
||||
};
|
||||
|
||||
logger.clear();
|
||||
|
||||
var f = foo();
|
||||
|
||||
f.next().then(fulfillSpy(logger));
|
||||
f.next().then(fulfillSpy(logger));
|
||||
f.next().then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-5')
|
||||
.isFinal('async generator should not resolve any promise until await is not resolved');
|
||||
|
||||
holder.resolve('xyz');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-5')
|
||||
.fullfilled('xyz data-6')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal('async generator should not resolve any promise until await is not resolved');
|
||||
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *goo() {
|
||||
yield 'data-5';
|
||||
var t = await promise;
|
||||
yield t + ' data-6';
|
||||
yield t + ' data-7';
|
||||
};
|
||||
|
||||
logger.clear();
|
||||
var errorText = 'error-reject';
|
||||
var g = goo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('data-5')
|
||||
.isFinal('async generator should not resolve any promise until await is not resolved');
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('data-5')
|
||||
.isFinal('async generator should not resolve any promise until await is not resolved');
|
||||
|
||||
holder.reject(errorText);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('data-5')
|
||||
.rejected(errorText)
|
||||
.fullfilledDone(undefined, 'After reject all resolved value have undefined')
|
||||
.isFinal();
|
||||
|
||||
/* Method in class */
|
||||
|
||||
const someText = 'foo';
|
||||
const resolveText = 'bar';
|
||||
|
||||
logger.clear();
|
||||
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
class A {
|
||||
async * foo() { yield someText; }
|
||||
async * boo() { var text = await promise; yield text + someText; }
|
||||
}
|
||||
var a = new A;
|
||||
var gf = a.foo();
|
||||
|
||||
gf.next().then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(someText)
|
||||
.isFinal();
|
||||
|
||||
const gb = a.boo();
|
||||
|
||||
gb.next().then(fulfillSpy(logger));
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(someText)
|
||||
.isFinal();
|
||||
|
||||
holder.resolve(resolveText);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(someText)
|
||||
.fullfilled(resolveText + someText)
|
||||
.isFinal();
|
||||
|
||||
/* Async generator function expression */
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
var asyncGenExp = async function *() { var t = await promise; yield t + someText + someText; };
|
||||
|
||||
var gAsyncGenExp = asyncGenExp();
|
||||
|
||||
gAsyncGenExp.next().then(fulfillSpy(logger));
|
||||
|
||||
holder.resolve(resolveText);
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(resolveText + someText + someText)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
/*Test throw*/
|
||||
async function *joo() {
|
||||
yield 'data-1';
|
||||
yield 'data-2';
|
||||
yield 'data-3';
|
||||
};
|
||||
|
||||
var j = joo();
|
||||
var errorTextInFunction = "ErrorInFunction";
|
||||
const errorWrongGenerator = "|this| should be an async generator";
|
||||
let errorRaised = false;
|
||||
|
||||
j.next()
|
||||
.then(function(value) {
|
||||
fulfillSpy(logger)(value);
|
||||
return j.throw(new Error(errorTextInFunction));
|
||||
})
|
||||
.then(function (value) {
|
||||
fulfillSpy(logger)(value);
|
||||
}, function (error) {
|
||||
rejectSpy(logger)(error);
|
||||
return j.next();
|
||||
})
|
||||
.then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('data-1')
|
||||
.rejected((new Error(errorTextInFunction)).toString())
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var j1 = joo();
|
||||
|
||||
j1.throw(new Error(errorText)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var j2 = joo();
|
||||
|
||||
const returnValue = 'return-value';
|
||||
|
||||
j1.return(returnValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j1.throw(new Error(errorText)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilledDone(returnValue)
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
let j3 = joo();
|
||||
|
||||
j3.next.call(undefined).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
|
||||
j3.next.call(undefined).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
.fullfilled('data-1')
|
||||
.fullfilled('data-2')
|
||||
.fullfilled('data-3')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next.call(undefined).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
.fullfilled('data-1')
|
||||
.fullfilled('data-2')
|
||||
.fullfilled('data-3')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next.call(undefined).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
j3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
.fullfilled('data-1')
|
||||
.fullfilled('data-2')
|
||||
.fullfilled('data-3')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
|
||||
j3.next.call({}).catch(rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
j3.next.call('string').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new TypeError(errorWrongGenerator))
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
j3 = joo();
|
||||
let gen = generator();
|
||||
j3.next.call(gen).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(42)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *koo() {
|
||||
var t = await 'abcd';
|
||||
yield t + ' data-first';
|
||||
var t2 = await promise;
|
||||
yield t2 + ' data-second';
|
||||
yield t2 + ' data-third';
|
||||
};
|
||||
|
||||
const k1 = koo();
|
||||
|
||||
k1.next().then(fulfillSpy(logger));
|
||||
k1.next().then(fulfillSpy(logger));
|
||||
|
||||
k1.throw(new Error(errorText)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
k1.next().then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-first')
|
||||
.isFinal();
|
||||
|
||||
holder.resolve('xyz');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-first')
|
||||
.fullfilled('xyz data-second')
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
const k2 = koo();
|
||||
|
||||
k2.next().then(fulfillSpy(logger));
|
||||
k2.next().then(fulfillSpy(logger));
|
||||
|
||||
k2.return(returnValue).then(fulfillSpy(logger));
|
||||
|
||||
holder.resolve('xyz');
|
||||
|
||||
k2.throw(new Error(errorText)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
k2.next().then(fulfillSpy(logger));
|
||||
k2.next().then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abcd data-first')
|
||||
.fullfilled('xyz data-second')
|
||||
.fullfilledDone(returnValue)
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *loo() {
|
||||
var t = await promise;
|
||||
throw new Error(errorText);
|
||||
yield t + 'data-first';
|
||||
yield t + 'data-second';
|
||||
};
|
||||
|
||||
const l1 = loo();
|
||||
|
||||
l1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.isFinal();
|
||||
|
||||
holder.resolve('qwe');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
const l2 = loo();
|
||||
|
||||
l2.throw(new Error('another error')).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
holder.resolve('abc');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new Error('another error'))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
holder = {};
|
||||
promise = getPromise(holder);
|
||||
|
||||
const l3 = loo();
|
||||
|
||||
l3.return(someText).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
holder.resolve(resolveText);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilledDone(someText)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
async function *moo() {
|
||||
throw new Error(errorText);
|
||||
yield t + 'data-first';
|
||||
yield t + 'data-second';
|
||||
};
|
||||
|
||||
const m1 = moo();
|
||||
|
||||
m1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new Error(errorText))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const m2 = moo();
|
||||
|
||||
m2.throw(new Error('another error')).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(new Error('another error'))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const m3 = moo();
|
||||
|
||||
m3.return(someText).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilledDone(someText)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
async function* noo() {
|
||||
const x = Promise.resolve(1);
|
||||
const y = Promise.resolve(2);
|
||||
|
||||
const fromX = yield x;
|
||||
return y;
|
||||
}
|
||||
|
||||
const n1 = noo();
|
||||
|
||||
let value1 = Promise.resolve("a");
|
||||
let value2 = Promise.resolve("b");
|
||||
|
||||
n1.next(value1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n1.next(value2).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone(2)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const n2 = noo();
|
||||
|
||||
value1 = Promise.resolve("a");
|
||||
value2 = Promise.resolve("b");
|
||||
|
||||
n2.return(value1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n2.next(value2).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilledDone('a')
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *ooo() {
|
||||
yield promise;
|
||||
}
|
||||
|
||||
const o1 = ooo();
|
||||
|
||||
o1.next(value1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
holder.reject("a");
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected('a')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
promise = getPromise(holder);
|
||||
|
||||
async function *roo() {
|
||||
try {
|
||||
yield promise;
|
||||
} catch (e) {
|
||||
yield e;
|
||||
}
|
||||
}
|
||||
|
||||
const r1 = roo();
|
||||
value1 = 'value-1';
|
||||
value2 = 'value-2';
|
||||
value3 = 'value-3';
|
||||
|
||||
r1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
r1.next(value1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
r1.next(value1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
holder.reject("abc");
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('abc')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
holder1 = {};
|
||||
holder2 = {};
|
||||
promise1 = getPromise(holder1);
|
||||
promise2 = getPromise(holder2);
|
||||
|
||||
async function *poo() {
|
||||
try {
|
||||
yield promise1;
|
||||
} catch (e) {
|
||||
yield promise2;
|
||||
}
|
||||
}
|
||||
|
||||
const p1 = poo();
|
||||
|
||||
p1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
p1.next(value3).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
p1.next(value3).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
holder1.reject(value1);
|
||||
holder2.reject(value2);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.rejected(value2)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
async function *soo() {
|
||||
yield 1;
|
||||
}
|
||||
|
||||
const endValue = 'end-value-1';
|
||||
|
||||
logger.clear();
|
||||
|
||||
const s1 = soo();
|
||||
|
||||
s1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s1.return(Promise.resolve(endValue)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone()
|
||||
.fullfilledDone(endValue)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const s2 = soo();
|
||||
|
||||
s2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s2.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s2.return(Promise.reject(endValue)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone()
|
||||
.rejected(endValue)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
const s3 = soo();
|
||||
|
||||
s3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s3.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s3.return(endValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone()
|
||||
.fullfilledDone(endValue)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const s4 = soo();
|
||||
promise1 = Promise.resolve(endValue);
|
||||
|
||||
s4.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s4.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s4.throw(promise1).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone()
|
||||
.rejected(promise1)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const s5 = soo();
|
||||
|
||||
s5.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s5.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
s5.throw(endValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilledDone()
|
||||
.rejected(endValue)
|
||||
.isFinal();
|
||||
|
||||
async function *too() {
|
||||
return Promise.resolve('abcd');
|
||||
}
|
||||
|
||||
logger.clear();
|
||||
|
||||
const t = too();
|
||||
|
||||
t.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilledDone('abcd')
|
||||
.isFinal();
|
|
@ -1,20 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
let error = false;
|
||||
|
||||
async function * foo(value = obj) {
|
||||
yield '1';
|
||||
return 'end';
|
||||
}
|
||||
|
||||
try {
|
||||
var f = foo();
|
||||
} catch(e) {
|
||||
error = e instanceof ReferenceError;
|
||||
}
|
||||
|
||||
assert(error, true, 'Error in arguments declaration should to error during evaluation of async generator.');
|
|
@ -1,118 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
function checkSyntax(src) {
|
||||
try {
|
||||
eval(src);
|
||||
} catch (error) {
|
||||
if (error instanceof SyntaxError)
|
||||
throw new Error("Syntax Error: " + String(error) + "\n script: `" + src + "`");
|
||||
}
|
||||
}
|
||||
|
||||
function checkSyntaxError(src, message) {
|
||||
var bError = false;
|
||||
try {
|
||||
eval(src);
|
||||
} catch (error) {
|
||||
bError = error instanceof SyntaxError && (String(error) === message || typeof message === 'undefined');
|
||||
}
|
||||
if (!bError) {
|
||||
throw new Error("Expected syntax Error: " + message + "\n in script: `" + src + "`");
|
||||
}
|
||||
}
|
||||
|
||||
(function checkSimpleAsyncGeneratorSloppyMode() {
|
||||
checkSyntax('var a1 = async function*asyncGenWithName1(){ for await(const value of foo()) {} }');
|
||||
checkSyntax('var a1 = async function asyncWithName1(){ for await(const value of foo()) {} }');
|
||||
checkSyntax('var a1 = async function*asyncGenWithName1(){ for await(let value of foo()) {} }');
|
||||
checkSyntax('var a1 = async function asyncWithName1(){ for await(let value of foo()) {} }');
|
||||
})();
|
||||
|
||||
(function checkSimpleAsyncGeneratorStrictMode() {
|
||||
checkSyntax('"use strict"; var a1 = async function*asyncGenWithName1(){ for await(const value of foo()) {} }');
|
||||
checkSyntax('"use strict"; var a1 = async function asyncWithName1(){ for await(const value of foo()) {} }');
|
||||
checkSyntax('"use strict"; var a1 = async function*asyncGenWithName1(){ for await(let value of foo()) {} }');
|
||||
checkSyntax('"use strict"; var a1 = async function asyncWithName1(){ for await(let value of foo()) {} }');
|
||||
})();
|
||||
|
||||
|
||||
(function checkNestedAsyncGenerators() {
|
||||
var wrappers = [
|
||||
{start: 'var a1 = async function*asyncGenWithName1(){', finish: '}'},
|
||||
{start: 'async function*asyncGenWithName2(){ ', finish: '}'},
|
||||
{start: 'async function asyncWithName2(){ ', finish: '}'},
|
||||
{start: 'class A { async * method() { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => {', finish: '}'},
|
||||
{start: 'var a1 = async () => { try { ', finish: ' } catch (e) {} }'},
|
||||
{start: 'var a1 = async () => { { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => { if (true) { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => { if (true) ', finish: ' }'},
|
||||
{start: 'var a1 = async () => { if (true) foo(); else { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => { while (true) { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => { for(;;) { ', finish: ' } }'},
|
||||
{start: 'var a1 = async () => { switch(e) { case \'1\' : ', finish: ' } }'},
|
||||
];
|
||||
|
||||
expressions = [
|
||||
'for await(const value of foo()) {}',
|
||||
'for await(let value of foo()) {}',
|
||||
'for await(var value of foo()) {}',
|
||||
'for await(var [a, b] of foo()) {}',
|
||||
'for await(let {a, b} of foo()) {}',
|
||||
'for await(const [... b] of foo()) {}',
|
||||
'for await(const [,,, b] of foo()) {}',
|
||||
'for await(const value of boo) {}',
|
||||
'for await(let value of boo) {}',
|
||||
'for await(const value of foo().boo()) {}',
|
||||
'for await(let value of foo.boo()) {}',
|
||||
'for await(let value of foo.boo(value)) {}',
|
||||
'for await(let value of [1,2,3]) {}',
|
||||
'for await(value of [1,2,3]) {}',
|
||||
'for await(value of x + x) {}',
|
||||
'for await(value of f()) {}',
|
||||
'for await(value of (x + x)) {}',
|
||||
];
|
||||
|
||||
wrappers.forEach(wrapper => {
|
||||
expressions.forEach(exp => {
|
||||
checkSyntax(wrapper.start + exp + wrapper.finish);
|
||||
});
|
||||
})
|
||||
})();
|
||||
|
||||
|
||||
(function checkSimpleAsyncGeneratorSyntaxErrorInSloppyMode() {
|
||||
checkSyntaxError("var asyncGenFn = function () { for await(const value of foo()) {} }");
|
||||
checkSyntaxError("var asyncGenFn = async function () { var arr = () => { for await(const value of foo()) {} } }");
|
||||
checkSyntaxError("var asyncGenFn = function* () { for await(const value of foo()) {} }");
|
||||
checkSyntaxError("var asyncGenFn = async function* () { var arr = () => { for await(const value of foo()) {} } }");
|
||||
checkSyntaxError('var a1 = async function*asyncGenWithName1(){ for await(const value in foo()) {} }');
|
||||
checkSyntaxError('var a1 = async function asyncWithName1(){ for await(const value in foo()) {} }');
|
||||
checkSyntaxError('var a1 = async function asyncWithName1(){ for await (;;) {} }');
|
||||
checkSyntaxError("var a1 = async function asyncWithName1(){ for await (let v = 4;;) {} }");
|
||||
checkSyntaxError("var a1 = async function asyncWithName1(){ for await (let v of f();;) {} }");
|
||||
checkSyntaxError("var a1 = async function asyncWithName1(){ for await (let v of boo;;) {} }");
|
||||
checkSyntaxError("var a1 = async function asyncWithName1(){ for await (let v of boo of) {} }");
|
||||
checkSyntaxError("async function asyncWithName1(){ for await (let v of boo in) {} }");
|
||||
checkSyntaxError("async function asyncWithName1(){ for await (v in x + x ) {} }");
|
||||
})();
|
||||
|
||||
(function checkSimpleAsyncGeneratorSyntaxErrorInStrictMode() {
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = function () { for await(const value of foo()) {} }");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = async function () { var arr = () => { for await(const value of foo()) {} } }");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = function* () { for await(const value of foo()) {} }");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = async function* () { var arr = () => { for await(const value of foo()) {} } }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function*asyncGenWithName1(){ for await(const value in foo()) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await(const value in foo()) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await (;;) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await (let v = 4;;) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await (let v of f();;) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await (let v of boo;;) {} }");
|
||||
checkSyntaxError("'use strict'; var a1 = async function asyncWithName1(){ for await (let v of boo of) {} }");
|
||||
checkSyntaxError("'use strict'; async function asyncWithName1(){ for await (let v of boo in) {} }");
|
||||
checkSyntaxError("'use strict'; async function asyncWithName1(){ for await (v in x + x ) {} }");
|
||||
})();
|
|
@ -1,280 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
var result = 0;
|
||||
async function * foo() { yield 1; yield Promise.resolve(2); return 3;};
|
||||
|
||||
async function boo () {
|
||||
for await (const value of foo()) {
|
||||
result = result + value;
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(result, 3);
|
||||
|
||||
result = 0;
|
||||
|
||||
{
|
||||
const boo = async function () {
|
||||
for await (const val of foo()) {
|
||||
result = result + val;
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(result, 3);
|
||||
}
|
||||
|
||||
{
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
for await (const val of [1, 2, 3]) {
|
||||
result = result + val;
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(result, 6);
|
||||
}
|
||||
|
||||
{
|
||||
let error = false;
|
||||
|
||||
const boo = async function () {
|
||||
for await (const val of 1) {
|
||||
result = result + val;
|
||||
}
|
||||
}
|
||||
|
||||
boo().catch(raisedError => error = raisedError instanceof TypeError);
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(error, true);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
const o = {
|
||||
[Symbol.asyncIterator]() {
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
return { done: true }
|
||||
}
|
||||
}
|
||||
|
||||
result = -1;
|
||||
const boo = async function () {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result = val;
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 0);
|
||||
assert(result, -1);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
const o = {
|
||||
[Symbol.asyncIterator]() {
|
||||
this.index = 0;
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
this.index++;
|
||||
if (this.index <= 10)
|
||||
return { done: false, value: this.index };
|
||||
else
|
||||
return { done: true, value: this.index };
|
||||
}
|
||||
}
|
||||
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result += val;
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 10);
|
||||
assert(result, 55);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
let error = false;
|
||||
|
||||
const o = {
|
||||
[Symbol.asyncIterator]() {
|
||||
this.index = 0;
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
this.index++;
|
||||
if (this.index <= 10)
|
||||
return { done: false, value: this.index };
|
||||
else
|
||||
throw new Error('some error');
|
||||
}
|
||||
}
|
||||
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result += val;
|
||||
}
|
||||
}
|
||||
|
||||
boo().catch(e => { error = e instanceof Error && e.message === 'some error'; });
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 10);
|
||||
assert(result, 55);
|
||||
assert(error, true);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
let error = false;
|
||||
let emptyParam = false;
|
||||
|
||||
const o = {
|
||||
[Symbol.asyncIterator]() {
|
||||
this.index = 0;
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
emptyParam = args === undefined;
|
||||
this.index++;
|
||||
if (this.index <= 10)
|
||||
return { done: false, value: this.index };
|
||||
else
|
||||
throw new Error('some error');
|
||||
}
|
||||
}
|
||||
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
try {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result += val;
|
||||
}
|
||||
} catch (e) {
|
||||
error = e instanceof Error && e.message === 'some error';
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 10);
|
||||
assert(result, 55);
|
||||
assert(error, true);
|
||||
assert(emptyParam, true);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
let error = false;
|
||||
|
||||
const o = {
|
||||
[Symbol.asyncIterator]() {
|
||||
this.index = 0;
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
this.index++;
|
||||
if (this.index <= 10)
|
||||
return { done: false, value: this.index };
|
||||
else
|
||||
return { done: true, value: this.index };
|
||||
}
|
||||
}
|
||||
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
if (true) {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result += val;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 10);
|
||||
assert(result, 55);
|
||||
}
|
||||
|
||||
{
|
||||
let conter = 0;
|
||||
let error = false;
|
||||
let emptyParam = false;
|
||||
|
||||
const o = {
|
||||
[Symbol.iterator]() {
|
||||
this.index = 0;
|
||||
return this
|
||||
},
|
||||
next(args) {
|
||||
emptyParam = args === undefined;
|
||||
this.index++;
|
||||
if (this.index <= 10)
|
||||
return { done: false, value: this.index };
|
||||
else
|
||||
throw new Error('some error');
|
||||
}
|
||||
}
|
||||
|
||||
result = 0;
|
||||
const boo = async function () {
|
||||
try {
|
||||
for await (const val of o) {
|
||||
conter++;
|
||||
result += val;
|
||||
}
|
||||
} catch (e) {
|
||||
error = e instanceof Error && e.message === 'some error';
|
||||
}
|
||||
}
|
||||
|
||||
boo();
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assert(conter, 10);
|
||||
assert(result, 55);
|
||||
assert(error, true);
|
||||
assert(emptyParam, true);
|
||||
}
|
|
@ -1,122 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
function checkSyntax(src) {
|
||||
try {
|
||||
eval(src);
|
||||
} catch (error) {
|
||||
if (error instanceof SyntaxError)
|
||||
throw new Error("Syntax Error: " + String(error) + "\n script: `" + src + "`");
|
||||
}
|
||||
}
|
||||
|
||||
function checkSyntaxError(src, message) {
|
||||
var bError = false;
|
||||
try {
|
||||
eval(src);
|
||||
} catch (error) {
|
||||
bError = error instanceof SyntaxError && (String(error) === message || typeof message === 'undefined');
|
||||
}
|
||||
if (!bError) {
|
||||
throw new Error("Expected syntax Error: " + message + "\n in script: `" + src + "`");
|
||||
}
|
||||
}
|
||||
|
||||
(function checkSimpleAsyncGeneratorSloppyMode() {
|
||||
checkSyntax('var a1 = async function*asyncGenWithName1(){}');
|
||||
checkSyntax('var a2 = async function *asyncGenWithName2(){ yield 11; }');
|
||||
checkSyntax('var a3 = async function * asyncGenWithName2(){ await p; yield 11; }');
|
||||
checkSyntax('var d1 = async function*(){}');
|
||||
checkSyntax('var d2 = async function* (){ yield 11; }');
|
||||
checkSyntax('var d3 = async function * (){ await p; yield 11; }');
|
||||
checkSyntax('async function* withName1(){ }');
|
||||
checkSyntax('async function *withName2(){ yield 11; }');
|
||||
checkSyntax('async function * withName3(){ await p; yield 11; }');
|
||||
checkSyntax('class A { async * method() { } }');
|
||||
checkSyntax('class B { async * method() {yield 11;} }');
|
||||
checkSyntax('class C { async * method() {yield 11; await p;} }');
|
||||
checkSyntax('class D { async * "method"() {yield 11; await p;} }');
|
||||
checkSyntax('class F { async * 0() {yield 11; await p;} }');
|
||||
checkSyntax('var obj = { async * method() {yield 11; await p;} }');
|
||||
checkSyntax('({ async foo() {} })');
|
||||
checkSyntax('({ async : 1 })');
|
||||
})();
|
||||
|
||||
(function checkSimpleAsyncGeneratorStrictMode() {
|
||||
checkSyntax('"use strict"; var a1 = async function*asyncGenWithName1(){}');
|
||||
checkSyntax('"use strict"; var a2 = async function *asyncGenWithName2(){ yield 11; }');
|
||||
checkSyntax('"use strict"; var a3 = async function * asyncGenWithName2(){ await p; yield 11; }');
|
||||
checkSyntax('"use strict"; var d1 = async function*(){}');
|
||||
checkSyntax('"use strict"; var d2 = async function* (){ yield 11; }');
|
||||
checkSyntax('"use strict"; var d3 = async function * (){ await p; yield 11; }');
|
||||
checkSyntax('"use strict"; async function* withName1(){ }');
|
||||
checkSyntax('"use strict"; async function *withName2(){ yield 11; }');
|
||||
checkSyntax('"use strict"; async function * withName3(){ await p; yield 11; }');
|
||||
checkSyntax('"use strict"; class A { async * method() { } }');
|
||||
checkSyntax('"use strict"; class B { async * method() {yield 11;} }');
|
||||
checkSyntax('"use strict"; class C { async * method() {yield 11; await p;} }');
|
||||
checkSyntax('"use strict"; class D { async * "method"() {yield 11; await p;} }');
|
||||
checkSyntax('"use strict"; class E { async * ["calc" + "ulate"]() {yield 11; await p;} }');
|
||||
checkSyntax('"use strict"; class F { async * 0() {yield 11; await p;} }');
|
||||
checkSyntax('"use strict"; var obj = { async * method() {yield 11; await p;} }');
|
||||
checkSyntax('"use strict"; ({ async foo() {} })');
|
||||
checkSyntax('"use strict"; ({ async : 1 })');
|
||||
})();
|
||||
|
||||
|
||||
(function checkNestedAsyncGenerators() {
|
||||
var wrappers = [
|
||||
{start: 'var a1 = async function*asyncGenWithName1(){', finish: '}'},
|
||||
{start: 'async function*asyncGenWithName2(){ ', finish: '}'},
|
||||
{start: 'class A { async * method() { ', finish: ' } }'}
|
||||
];
|
||||
|
||||
expressions = [
|
||||
'await 10; yield 11; return 12;',
|
||||
'var async = 10; yield async;',
|
||||
'var async = 10; await async;',
|
||||
'var async = 10; return async;',
|
||||
'var async = function() {}; return async;',
|
||||
'var async = function() {}; return async();',
|
||||
];
|
||||
|
||||
wrappers.forEach(wrapper => {
|
||||
expressions.forEach(exp => {
|
||||
checkSyntax(wrapper.start + exp + wrapper.finish);
|
||||
});
|
||||
})
|
||||
})();
|
||||
|
||||
|
||||
(function checkSimpleAsyncGeneratorSyntaxErrorInSloppyMode() {
|
||||
checkSyntaxError("var asyncGenFn = async function *await() {}");
|
||||
checkSyntaxError("var asyncGenFn = async function*(await) {}");
|
||||
checkSyntaxError("var asyncGenFn = async function *withName(await) {}");
|
||||
checkSyntaxError("async function *asyncGeneratorFunctionDeclaration(await) {}");
|
||||
checkSyntaxError("var asyncGenFn = *async function () {}");
|
||||
checkSyntaxError("var asyncGenFn = *async function withName() {}");
|
||||
checkSyntaxError("*async function asyncGeneratorFunctionDeclaration(await) {}");
|
||||
checkSyntaxError("var obj = { *async asyncGeneratorMethod() {} };");
|
||||
checkSyntaxError("var obj = { async asyncGeneratorMethod*() {} };");
|
||||
checkSyntaxError("class A { get async* ttt() {} }");
|
||||
checkSyntaxError("class B { get *async ttt() {} }");
|
||||
checkSyntaxError('({ async = 1 })');
|
||||
})();
|
||||
|
||||
(function checkSimpleAsyncGeneratorSyntaxErrorInStrictMode() {
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = async function *await() {}");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = async function*(await) {}");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = async function *withName(await) {}");
|
||||
checkSyntaxError("'use strict'; async function *asyncGeneratorFunctionDeclaration(await) {}");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = *async function () {}");
|
||||
checkSyntaxError("'use strict'; var asyncGenFn = *async function withName() {}");
|
||||
checkSyntaxError("'use strict'; *async function asyncGeneratorFunctionDeclaration(await) {}");
|
||||
checkSyntaxError("'use strict'; var obj = { *async asyncGeneratorMethod() {} };");
|
||||
checkSyntaxError("'use strict'; var obj = { async asyncGeneratorMethod*() {} };");
|
||||
checkSyntaxError("'use strict'; class A { get async* ttt() {} }");
|
||||
checkSyntaxError("'use strict'; class B { get *async ttt() {} }");
|
||||
checkSyntaxError("'use strict'; ({ async = 1 })");
|
||||
})();
|
|
@ -1,470 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
const Logger = function () {
|
||||
var log = [];
|
||||
|
||||
this.logEvent = (type, value, done) => {
|
||||
log.push({ type, value, done});
|
||||
};
|
||||
this.logFulfilledEvent = (value, done) => {
|
||||
this.logEvent('fulfilled', value, done);
|
||||
};
|
||||
this.logRejectEvent = error => {
|
||||
this.logEvent('reject', error.toString(), true);
|
||||
};
|
||||
this.logCatchEvent = value => {
|
||||
this.logEvent('catch', value, true);
|
||||
};
|
||||
this.getLogger = () => log;
|
||||
|
||||
this.clear = () => {
|
||||
log = [];
|
||||
}
|
||||
};
|
||||
|
||||
const fulfillSpy = logger => result => logger.logFulfilledEvent(result.value, result.done);
|
||||
const rejectSpy = logger => error => logger.logRejectEvent(error);
|
||||
const catchSpy = logger => error => logger.logCatchEvent(error);
|
||||
|
||||
const assertLogger = function (loggerObject) {
|
||||
const logger = loggerObject.getLogger();
|
||||
|
||||
var _assertLogger = function () {
|
||||
let index = 0;
|
||||
|
||||
const isNotOutOfLength = () => {
|
||||
assert(index < logger.length, true, `Index is greater then log length`);
|
||||
}
|
||||
|
||||
this.fullfilled = function (expectedValue, message = 'on fulfill') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.fullfilledDone = function (expectedValue, message = 'on fulfill with done true') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.rejected = function (error, message = 'on reject') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'reject', msg);
|
||||
assert(step.value, error.toString(), msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.catched = function (expectedError, message = 'on catch') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'catch', msg);
|
||||
assert(step.value, expectedError, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.isFinal = function (message = '') {
|
||||
assert(index, logger.length, `expected final step: ${message}`);
|
||||
};
|
||||
};
|
||||
|
||||
return new _assertLogger();
|
||||
};
|
||||
|
||||
const getPromise = promiseHolder => {
|
||||
return new Promise((resolve, reject) => {
|
||||
promiseHolder.resolve = resolve;
|
||||
promiseHolder.reject = reject;
|
||||
});
|
||||
};
|
||||
|
||||
var logger = new Logger();
|
||||
const someValue = 'some-value';
|
||||
const errorMessage = 'error-message';
|
||||
const promiseHolder = {};
|
||||
|
||||
async function * foo(val) {
|
||||
let reply = yield '0:' + val;
|
||||
reply = yield getPromise(promiseHolder);
|
||||
return 'end foo:' + reply;
|
||||
}
|
||||
|
||||
let f = foo(someValue);
|
||||
|
||||
f.next(someValue + ':0').then(fulfillSpy(logger));
|
||||
f.next(someValue + ':1').then(fulfillSpy(logger));
|
||||
f.next(someValue + ':2').then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:' + someValue)
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:' + someValue)
|
||||
.fullfilled('1')
|
||||
.fullfilledDone('end foo:' + someValue + ':2')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
f = foo('init');
|
||||
|
||||
f.next('0').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('2').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.rejected('1')
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
f = foo('init');
|
||||
|
||||
f.next('0').then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
f.next('1').then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
f.next('2').then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.fullfilledDone(undefined)
|
||||
.catched('1')
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
const promiseHolder1 = {};
|
||||
const promiseHolder2 = {};
|
||||
|
||||
async function * boo() {
|
||||
const promise1 = getPromise(promiseHolder1);
|
||||
const promise2 = getPromise(promiseHolder2);
|
||||
yield '0';
|
||||
yield promise1;
|
||||
yield '1';
|
||||
yield promise2;
|
||||
return 'end foo';
|
||||
}
|
||||
|
||||
let b = boo();
|
||||
|
||||
b.next().then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
b.next().then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
b.next().then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
b.next().then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
b.next().then(fulfillSpy(logger)).catch(catchSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.resolve('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('#1')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder2.resolve('#2');
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('#1')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.fullfilledDone('end foo')
|
||||
.isFinal();
|
||||
|
||||
|
||||
logger.clear();
|
||||
|
||||
b = boo();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.resolve('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('#1')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder2.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('#1')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
b = boo();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.reject('#1');
|
||||
promiseHolder2.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.rejected('#1')
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
b = boo();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder2.reject('#2');
|
||||
promiseHolder1.reject('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.rejected('#1')
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
async function * bar() {
|
||||
yield '0';
|
||||
yield getPromise(promiseHolder1);
|
||||
throw new Error(errorMessage);
|
||||
yield getPromise(promiseHolder2);
|
||||
return 'end foo';
|
||||
}
|
||||
|
||||
logger.clear();
|
||||
b = bar();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
|
||||
promiseHolder1.resolve('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('#1')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
b = bar();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.reject('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.rejected('#1')
|
||||
.fullfilledDone()
|
||||
.fullfilledDone()
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
b = bar();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.resolve('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
b = bar();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.resolve('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled("#1")
|
||||
.fullfilledDone(someValue)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
b = bar();
|
||||
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder1.reject('#1');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.rejected("#1")
|
||||
.fullfilledDone(someValue)
|
||||
.isFinal();
|
|
@ -1,490 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
const getPromise = promiseHolder => {
|
||||
return new Promise((resolve, reject) => {
|
||||
promiseHolder.resolve = resolve;
|
||||
promiseHolder.reject = reject;
|
||||
});
|
||||
};
|
||||
|
||||
const Logger = function () {
|
||||
var log = [];
|
||||
|
||||
this.logEvent = (type, value, done) => {
|
||||
log.push({ type, value, done});
|
||||
};
|
||||
this.logFulfilledEvent = (value, done) => {
|
||||
this.logEvent('fulfilled', value, done);
|
||||
};
|
||||
this.logRejectEvent = error => {
|
||||
this.logEvent('reject', error.toString(), true);
|
||||
};
|
||||
this.logCatchEvent = value => {
|
||||
this.logEvent('catch', value, true);
|
||||
};
|
||||
this.logCustomEvent = event => {
|
||||
this.logEvent('custom', event, false);
|
||||
};
|
||||
this.getLogger = () => log;
|
||||
|
||||
this.clear = () => {
|
||||
log = [];
|
||||
}
|
||||
};
|
||||
|
||||
const fulfillSpy = logger => result => logger.logFulfilledEvent(result.value, result.done);
|
||||
const rejectSpy = logger => error => logger.logRejectEvent(error);
|
||||
const catchSpy = logger => error => logger.logCatchEvent(error);
|
||||
const customSpy = logger => event => logger.logCustomEvent(event);
|
||||
|
||||
const assertLogger = function (loggerObject) {
|
||||
const logger = loggerObject.getLogger();
|
||||
|
||||
var _assertLogger = function () {
|
||||
let index = 0;
|
||||
|
||||
const isNotOutOfLength = () => {
|
||||
assert(index < logger.length, true, `Index is greater then log length`);
|
||||
}
|
||||
|
||||
this.fullfilled = function (expectedValue, message = 'on fulfill') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.fullfilledDone = function (expectedValue, message = 'on fulfill with done true') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.rejected = function (error, message = 'on reject') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'reject', msg);
|
||||
assert(step.value, error.toString(), msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.catched = function (expectedError, message = 'on catch') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'catch', msg);
|
||||
assert(step.value, expectedError, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.custom = function (expectedValue, message = 'on custom event') {
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'custom', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.isFinal = function (message = '') {
|
||||
assert(index, logger.length, `expected final step: ${message}`);
|
||||
};
|
||||
};
|
||||
|
||||
return new _assertLogger();
|
||||
};
|
||||
|
||||
var logger = new Logger();
|
||||
const someValue = 'some-value';
|
||||
const errorMessage = 'error-message';
|
||||
|
||||
let asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
},
|
||||
throw (error) {
|
||||
customSpy(logger)('throw:' + error);
|
||||
return error;
|
||||
},
|
||||
return(value) {
|
||||
customSpy(logger)('return:' + value);
|
||||
return { value: value, done: true };
|
||||
}
|
||||
};
|
||||
|
||||
async function *foo () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let f = foo('Init');
|
||||
|
||||
f.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('C').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('next:C')
|
||||
.fullfilled(undefined)
|
||||
.custom('next:D')
|
||||
.fullfilled("C")
|
||||
.custom('next:E')
|
||||
.fullfilled("D")
|
||||
.custom('next:iter:Finish')
|
||||
.fullfilled("E")
|
||||
.fullfilled("3")
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
f = foo('Init');
|
||||
|
||||
f.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.return('C').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:C')
|
||||
.fullfilled(undefined)
|
||||
.fullfilledDone("C")
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
f = foo('Init');
|
||||
|
||||
f.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
f.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('throw:' + new Error(errorMessage))
|
||||
.fullfilled(undefined)
|
||||
.custom('next:D')
|
||||
.fullfilled(undefined)
|
||||
.custom('next:E')
|
||||
.fullfilled('D')
|
||||
.custom('next:iter:Finish')
|
||||
.fullfilled('E')
|
||||
.fullfilled('3')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
}
|
||||
};
|
||||
|
||||
async function *boo () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let b = boo('Init');
|
||||
|
||||
logger.clear();
|
||||
|
||||
b.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('C').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('next:C')
|
||||
.fullfilled(undefined)
|
||||
.custom('next:D')
|
||||
.fullfilled("C")
|
||||
.custom("next:iter:Finish")
|
||||
.fullfilled("D")
|
||||
.fullfilled("3")
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
b = boo('Init');
|
||||
|
||||
b.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Delegated generator does not have a \'throw\' method.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
},
|
||||
return (value) {
|
||||
customSpy(logger)('return:' + value);
|
||||
return { value: value, done: true };
|
||||
}
|
||||
};
|
||||
|
||||
async function *bar () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
b = bar('Init');
|
||||
|
||||
logger.clear();
|
||||
|
||||
b.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Delegated generator does not have a \'throw\' method.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
let ph = {};
|
||||
|
||||
asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
},
|
||||
return (value) {
|
||||
customSpy(logger)('return:' + value);
|
||||
return { value: getPromise(ph), done: true };
|
||||
}
|
||||
};
|
||||
|
||||
async function *baz () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
b = baz('Init');
|
||||
|
||||
logger.clear();
|
||||
|
||||
b.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Delegated generator does not have a \'throw\' method.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
ph.resolve('accept');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Delegated generator does not have a \'throw\' method.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
ph = {};
|
||||
|
||||
asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
},
|
||||
return (value) {
|
||||
customSpy(logger)('return:' + value);
|
||||
return getPromise(ph);
|
||||
}
|
||||
};
|
||||
|
||||
async function *foobar () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
fb = foobar('Init');
|
||||
|
||||
logger.clear();
|
||||
|
||||
fb.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.isFinal();
|
||||
|
||||
ph.resolve({ value: 'value', done: true });
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Delegated generator does not have a \'throw\' method.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
fb = foobar('Init');
|
||||
|
||||
logger.clear();
|
||||
|
||||
fb.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
fb.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.isFinal();
|
||||
|
||||
ph.resolve('X');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:undefined')
|
||||
.fullfilled(undefined)
|
||||
.rejected('TypeError: Iterator result interface is not an object.')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
|
@ -1,684 +0,0 @@
|
|||
var assert = function (result, expected, message = "") {
|
||||
if (result !== expected) {
|
||||
throw new Error('Error in assert. Expected "' + expected + '" but was "' + result + '":' + message );
|
||||
}
|
||||
};
|
||||
|
||||
const Logger = function () {
|
||||
var log = [];
|
||||
|
||||
this.logEvent = (type, value, done) => {
|
||||
log.push({ type, value, done});
|
||||
};
|
||||
this.logFulfilledEvent = (value, done) => {
|
||||
this.logEvent('fulfilled', value, done);
|
||||
};
|
||||
this.logRejectEvent = error => {
|
||||
this.logEvent('reject', error.toString(), true);
|
||||
};
|
||||
this.logCatchEvent = value => {
|
||||
this.logEvent('catch', value, true);
|
||||
};
|
||||
this.logCustomEvent = event => {
|
||||
this.logEvent('custom', event, false);
|
||||
};
|
||||
this.getLogger = () => log;
|
||||
|
||||
this.clear = () => {
|
||||
log = [];
|
||||
}
|
||||
};
|
||||
|
||||
const fulfillSpy = logger => result => logger.logFulfilledEvent(result.value, result.done);
|
||||
const rejectSpy = logger => error => logger.logRejectEvent(error);
|
||||
const catchSpy = logger => error => logger.logCatchEvent(error);
|
||||
const customSpy = logger => event => logger.logCustomEvent(event);
|
||||
|
||||
const assertLogger = function (loggerObject) {
|
||||
const logger = loggerObject.getLogger();
|
||||
|
||||
var _assertLogger = function () {
|
||||
let index = 0;
|
||||
|
||||
const isNotOutOfLength = () => {
|
||||
assert(index < logger.length, true, `Index is greater then log length`);
|
||||
}
|
||||
|
||||
this.fullfilled = function (expectedValue, message = 'on fulfill') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.fullfilledDone = function (expectedValue, message = 'on fulfill with done true') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'fulfilled', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.rejected = function (error, message = 'on reject') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'reject', msg);
|
||||
assert(step.value, error.toString(), msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.catched = function (expectedError, message = 'on catch') {
|
||||
isNotOutOfLength();
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'catch', msg);
|
||||
assert(step.value, expectedError, msg);
|
||||
assert(step.done, true, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.custom = function (expectedValue, message = 'on custom event') {
|
||||
|
||||
const msg = `step: ${index} - ${message}`;
|
||||
let step = logger[index];
|
||||
|
||||
assert(step.type, 'custom', msg);
|
||||
assert(step.value, expectedValue, msg);
|
||||
assert(step.done, false, msg);
|
||||
|
||||
index++;
|
||||
return this;
|
||||
};
|
||||
|
||||
this.isFinal = function (message = '') {
|
||||
assert(index, logger.length, `expected final step: ${message}`);
|
||||
};
|
||||
};
|
||||
|
||||
return new _assertLogger();
|
||||
};
|
||||
|
||||
const getPromise = promiseHolder => {
|
||||
return new Promise((resolve, reject) => {
|
||||
promiseHolder.resolve = resolve;
|
||||
promiseHolder.reject = reject;
|
||||
});
|
||||
};
|
||||
|
||||
var logger = new Logger();
|
||||
const someValue = 'some-value';
|
||||
const errorMessage = 'error-message';
|
||||
|
||||
async function * foo(value) {
|
||||
let re = yield '1:' + value;
|
||||
re = yield '2:' + re;
|
||||
re = yield '3:' + re;
|
||||
return 'end foo:' + re;
|
||||
}
|
||||
|
||||
async function * boo(value) {
|
||||
let reply = yield '0:' + value;
|
||||
reply = yield* foo(reply);
|
||||
yield '4:' + reply;
|
||||
}
|
||||
|
||||
var b = boo('init');
|
||||
|
||||
b.next('0').then(fulfillSpy(logger));
|
||||
b.next('1').then(fulfillSpy(logger));
|
||||
b.next('2').then(fulfillSpy(logger));
|
||||
b.next('3').then(fulfillSpy(logger));
|
||||
b.next('4').then(fulfillSpy(logger));
|
||||
b.next('5').then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:init')
|
||||
.fullfilled('1:1')
|
||||
.fullfilled('2:2')
|
||||
.fullfilled('3:3')
|
||||
.fullfilled('4:end foo:4')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var b2 = boo(':value');
|
||||
|
||||
b2.next(':0').then(fulfillSpy(logger));
|
||||
b2.next(':1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next(':2').then(fulfillSpy(logger));
|
||||
b2.next(':3').then(fulfillSpy(logger));
|
||||
b2.next(':4').then(fulfillSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0::value')
|
||||
.fullfilled('1::1')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
var b2 = boo('#value');
|
||||
|
||||
b2.next('#0').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#1').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#2').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#3').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
b2.next('#4').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0:#value')
|
||||
.fullfilled('1:#1')
|
||||
.fullfilled('2:#2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
async function * bar() {
|
||||
yield '1';
|
||||
yield '2';
|
||||
throw new Error(errorMessage);
|
||||
yield '3';
|
||||
return 'end foo';
|
||||
}
|
||||
|
||||
async function * baz() {
|
||||
yield '0';
|
||||
yield* bar();
|
||||
yield '4';
|
||||
}
|
||||
|
||||
logger.clear();
|
||||
var bz1 = baz();
|
||||
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
bz1.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
let promiseHolder = {};
|
||||
|
||||
async function *joo() {
|
||||
yield '1';
|
||||
yield getPromise(promiseHolder);
|
||||
}
|
||||
|
||||
async function *goo () {
|
||||
yield '0';
|
||||
yield* joo();
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('2')
|
||||
.fullfilled('3')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.return(someValue).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone(someValue)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
g = goo();
|
||||
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
g.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.rejected(new Error(errorMessage))
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
async function *koo() {
|
||||
yield '1';
|
||||
await getPromise(promiseHolder);
|
||||
}
|
||||
|
||||
async function *loo () {
|
||||
yield '0';
|
||||
yield* joo();
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let l = loo();
|
||||
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled("0")
|
||||
.fullfilled("1")
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.resolve('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.fullfilled('#2')
|
||||
.fullfilled("3")
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
l = loo();
|
||||
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
l.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled("0")
|
||||
.fullfilled("1")
|
||||
.isFinal();
|
||||
|
||||
promiseHolder.reject('#2');
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled('0')
|
||||
.fullfilled('1')
|
||||
.rejected('#2')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
let asyncIter = {
|
||||
[Symbol.asyncIterator]() { return this; },
|
||||
next (value) {
|
||||
customSpy(logger)('next:' + value);
|
||||
return { value: value, done: 'iter:Finish' === value };
|
||||
},
|
||||
throw (error) {
|
||||
customSpy(logger)('throw:' + error);
|
||||
return error;
|
||||
},
|
||||
return(value) {
|
||||
customSpy(logger)('return:' + value);
|
||||
return { value: value, done: true };
|
||||
}
|
||||
};
|
||||
|
||||
async function *moo () {
|
||||
yield '0';
|
||||
yield* asyncIter;
|
||||
yield '3';
|
||||
}
|
||||
|
||||
let m = moo('Init');
|
||||
|
||||
m.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('C').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('next:C')
|
||||
.fullfilled(undefined)
|
||||
.custom('next:D')
|
||||
.fullfilled('C')
|
||||
.custom('next:E')
|
||||
.fullfilled('D')
|
||||
.custom('next:iter:Finish')
|
||||
.fullfilled('E')
|
||||
.fullfilled('3')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
m = moo('Init');
|
||||
|
||||
m.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.return('C').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('return:C')
|
||||
.fullfilled(undefined)
|
||||
.fullfilledDone('C')
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
m = moo('Init');
|
||||
|
||||
m.next('A').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('B').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.throw(new Error(errorMessage)).then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('D').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('E').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('iter:Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
m.next('Finish').then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.custom('next:undefined')
|
||||
.fullfilled('0')
|
||||
.custom('throw:' + new Error(errorMessage))
|
||||
.fullfilled(undefined)
|
||||
.custom('next:D')
|
||||
.fullfilled(undefined)
|
||||
.custom('next:E')
|
||||
.fullfilled('D')
|
||||
.custom('next:iter:Finish')
|
||||
.fullfilled('E')
|
||||
.fullfilled('3')
|
||||
.fullfilledDone(undefined)
|
||||
.isFinal();
|
||||
|
||||
logger.clear();
|
||||
|
||||
async function* noo() {
|
||||
try {
|
||||
await Promise.reject("doop");
|
||||
} finally {
|
||||
yield* [1, 2, 3]; // Is this line reachable in this implementation?
|
||||
}
|
||||
}
|
||||
|
||||
const n = noo();
|
||||
|
||||
n.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
n.next().then(fulfillSpy(logger), rejectSpy(logger));
|
||||
|
||||
drainMicrotasks();
|
||||
|
||||
assertLogger(logger)
|
||||
.fullfilled(1)
|
||||
.fullfilled(2)
|
||||
.fullfilled(3)
|
||||
.rejected('doop');
|
|
@ -1,22 +0,0 @@
|
|||
var result = 0;
|
||||
function test1(arr) {
|
||||
return Array.of(...arr);
|
||||
}
|
||||
function test2() {
|
||||
return Array(...arguments);
|
||||
}
|
||||
|
||||
var result = 0;
|
||||
if (this.noInline) {
|
||||
noInline(test1)
|
||||
noInline(test2)
|
||||
}
|
||||
|
||||
var array = [1,2,3,4,5];
|
||||
|
||||
for (var i = 0; i < 10000; i++) {
|
||||
result ^= test2(1,2,3,4,5,6,7).length;
|
||||
}
|
||||
|
||||
if (result != 0)
|
||||
throw "Error: bad result: " + result;
|
|
@ -1,14 +0,0 @@
|
|||
function shouldThrowSyntaxError(str, message) {
|
||||
try {
|
||||
eval(str);
|
||||
throw new Error("Expected `" + str + "` to throw a SyntaxError, but did not throw.")
|
||||
} catch (e) {
|
||||
if (e.constructor !== SyntaxError)
|
||||
throw new Error("Expected `" + str + "` to throw a SyntaxError, but threw '" + e + "'");
|
||||
if (message !== void 0 && e.message !== message)
|
||||
throw new Error("Expected `" + str + "` to throw SyntaxError: '" + message + "', but threw '" + e + "'");
|
||||
}
|
||||
}
|
||||
|
||||
shouldThrowSyntaxError("0/-async J", "Unexpected identifier 'J'")
|
||||
|
|
@ -1,15 +0,0 @@
|
|||
|
||||
function foo(i, x) {
|
||||
return String.prototype.big.call(x);
|
||||
}
|
||||
noInline(foo);
|
||||
|
||||
for (var i = 0; i < 1000; i++) {
|
||||
try {
|
||||
if (i < 200)
|
||||
foo(i, "hello");
|
||||
else
|
||||
foo(i, undefined);
|
||||
} catch(e) {
|
||||
}
|
||||
}
|
|
@ -1,12 +0,0 @@
|
|||
// This test passes if it does not crash.
|
||||
|
||||
function foo() {
|
||||
if (1 < 2);
|
||||
while (true) {
|
||||
if (1 < 2) break;
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < 10000; i++)
|
||||
foo();
|
||||
|
|
@ -1,18 +0,0 @@
|
|||
function shouldThrow(func, message) {
|
||||
var error = null;
|
||||
try {
|
||||
func();
|
||||
} catch (e) {
|
||||
error = e;
|
||||
}
|
||||
if (!error)
|
||||
throw new Error("not thrown.");
|
||||
if (String(error) !== message)
|
||||
throw new Error("bad error: " + String(error));
|
||||
}
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
shouldThrow(function () {
|
||||
new Array.prototype.forEach(function () { });
|
||||
}, "TypeError: function is not a constructor (evaluating 'new Array.prototype.forEach(function () { })')");
|
||||
}
|
|
@ -1,3 +1,8 @@
|
|||
/**
|
||||
* Reviewed
|
||||
* Add length property check for function forms in the language folder
|
||||
*/
|
||||
|
||||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
|
|
|
@ -1,45 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
}
|
||||
|
||||
function shouldThrow(func, errorMessage) {
|
||||
var errorThrown = false;
|
||||
var error = null;
|
||||
try {
|
||||
func();
|
||||
} catch (e) {
|
||||
errorThrown = true;
|
||||
error = e;
|
||||
}
|
||||
if (!errorThrown)
|
||||
throw new Error('not thrown');
|
||||
if (String(error) !== errorMessage)
|
||||
throw new Error(`bad error: ${String(error)}`);
|
||||
}
|
||||
|
||||
{
|
||||
shouldBe(JSON.stringify(Object.getOwnPropertyNames(Array.prototype.filter).sort()), `["length","name"]`);
|
||||
shouldBe(Array.prototype.filter.name, "filter");
|
||||
shouldBe(JSON.stringify(Object.getOwnPropertyDescriptor(Array.prototype.filter, 'name')), `{"value":"filter","writable":false,"enumerable":false,"configurable":true}`);
|
||||
shouldBe(delete Array.prototype.filter.name, true);
|
||||
shouldBe(JSON.stringify(Object.getOwnPropertyNames(Array.prototype.filter).sort()), `["length"]`);
|
||||
}
|
||||
|
||||
{
|
||||
shouldThrow(function () {
|
||||
"use strict";
|
||||
Array.prototype.forEach.name = 42;
|
||||
}, `TypeError: Attempted to assign to readonly property.`);
|
||||
}
|
||||
|
||||
{
|
||||
var resolve = null;
|
||||
var reject = null;
|
||||
new Promise(function (arg0, arg1) {
|
||||
resolve = arg0;
|
||||
reject = arg1;
|
||||
});
|
||||
shouldBe(Object.getOwnPropertyDescriptor(resolve, 'name'), undefined);
|
||||
shouldBe(Object.getOwnPropertyDescriptor(reject, 'name'), undefined);
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
//@ runDefault("--jitPolicyScale=0", "--gcMaxHeapSize=2000")
|
||||
|
||||
// This test happened to catch a case where we failed to zero the space in the butterfly before m_lastOffset when reallocating.
|
||||
|
||||
var array = Array(1000);
|
||||
for (var i = 0; i < 100000; ++i) {
|
||||
array[i - array.length] = '';
|
||||
array[i ^ array.length] = '';
|
||||
}
|
|
@ -1,38 +0,0 @@
|
|||
//@ runDefault("--useFTLJIT=0", "--useConcurrentJIT=false")
|
||||
|
||||
let num = 150;
|
||||
|
||||
function foo(comp, o, b) {
|
||||
let sum = o.f;
|
||||
if (b)
|
||||
OSRExit();
|
||||
for (let i = 0; i < comp; ++i) {
|
||||
sum += o.f;
|
||||
}
|
||||
return sum;
|
||||
}
|
||||
noInline(foo);
|
||||
|
||||
let o = {f:25};
|
||||
let o2 = {f:25, g:44};
|
||||
o2.f = 45;
|
||||
o2.f = 45;
|
||||
o2.f = 45;
|
||||
o2.f = 45;
|
||||
let comp = {
|
||||
valueOf() { return num; }
|
||||
}
|
||||
|
||||
foo(comp, o2, true);
|
||||
foo(comp, o2, true);
|
||||
for (let i = 0; i < 500; ++i) {
|
||||
foo(comp, o2, false);
|
||||
}
|
||||
|
||||
let o3 = {g:24, f:73};
|
||||
num = 10000000;
|
||||
let result = foo(comp, o3, false);
|
||||
|
||||
if (result !== (num + 1)*73) {
|
||||
throw new Error("Bad: " + result);
|
||||
}
|
|
@ -1,77 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
}
|
||||
|
||||
function testH(string) {
|
||||
return string.match(/h/i);
|
||||
}
|
||||
noInline(testH);
|
||||
|
||||
function testHe(string) {
|
||||
return string.match(/he/i);
|
||||
}
|
||||
noInline(testHe);
|
||||
|
||||
function testHel(string) {
|
||||
return string.match(/hel/i);
|
||||
}
|
||||
noInline(testHel);
|
||||
|
||||
function testHell(string) {
|
||||
return string.match(/hell/i);
|
||||
}
|
||||
noInline(testHell);
|
||||
|
||||
function testHello(string) {
|
||||
return string.match(/hello/i);
|
||||
}
|
||||
noInline(testHello);
|
||||
|
||||
function testHelloW(string) {
|
||||
return string.match(/hellow/i);
|
||||
}
|
||||
noInline(testHelloW);
|
||||
|
||||
function testHelloWo(string) {
|
||||
return string.match(/hellowo/i);
|
||||
}
|
||||
noInline(testHelloWo);
|
||||
|
||||
function testHelloWor(string) {
|
||||
return string.match(/hellowor/i);
|
||||
}
|
||||
noInline(testHelloWor);
|
||||
|
||||
function testHelloWorl(string) {
|
||||
return string.match(/helloworl/i);
|
||||
}
|
||||
noInline(testHelloWorl);
|
||||
|
||||
function testHelloWorld(string) {
|
||||
return string.match(/helloworld/i);
|
||||
}
|
||||
noInline(testHelloWorld);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
shouldBe(testH("HelloWorld")[0], `H`);
|
||||
shouldBe(testHe("HelloWorld")[0], `He`);
|
||||
shouldBe(testHel("HelloWorld")[0], `Hel`);
|
||||
shouldBe(testHell("HelloWorld")[0], `Hell`);
|
||||
shouldBe(testHello("HelloWorld")[0], `Hello`);
|
||||
shouldBe(testHelloW("HelloWorld")[0], `HelloW`);
|
||||
shouldBe(testHelloWo("HelloWorld")[0], `HelloWo`);
|
||||
shouldBe(testHelloWor("HelloWorld")[0], `HelloWor`);
|
||||
shouldBe(testHelloWorl("HelloWorld")[0], `HelloWorl`);
|
||||
shouldBe(testHelloWorld("HelloWorld")[0], `HelloWorld`);
|
||||
shouldBe(testH("HelloWorldこんにちは")[0], `H`);
|
||||
shouldBe(testHe("HelloWorldこんにちは")[0], `He`);
|
||||
shouldBe(testHel("HelloWorldこんにちは")[0], `Hel`);
|
||||
shouldBe(testHell("HelloWorldこんにちは")[0], `Hell`);
|
||||
shouldBe(testHello("HelloWorldこんにちは")[0], `Hello`);
|
||||
shouldBe(testHelloW("HelloWorldこんにちは")[0], `HelloW`);
|
||||
shouldBe(testHelloWo("HelloWorldこんにちは")[0], `HelloWo`);
|
||||
shouldBe(testHelloWor("HelloWorldこんにちは")[0], `HelloWor`);
|
||||
shouldBe(testHelloWorl("HelloWorldこんにちは")[0], `HelloWorl`);
|
||||
shouldBe(testHelloWorld("HelloWorldこんにちは")[0], `HelloWorld`);
|
||||
}
|
|
@ -1,77 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
}
|
||||
|
||||
function testH(string) {
|
||||
return string.match(/H/);
|
||||
}
|
||||
noInline(testH);
|
||||
|
||||
function testHe(string) {
|
||||
return string.match(/He/);
|
||||
}
|
||||
noInline(testHe);
|
||||
|
||||
function testHel(string) {
|
||||
return string.match(/Hel/);
|
||||
}
|
||||
noInline(testHel);
|
||||
|
||||
function testHell(string) {
|
||||
return string.match(/Hell/);
|
||||
}
|
||||
noInline(testHell);
|
||||
|
||||
function testHello(string) {
|
||||
return string.match(/Hello/);
|
||||
}
|
||||
noInline(testHello);
|
||||
|
||||
function testHelloW(string) {
|
||||
return string.match(/HelloW/);
|
||||
}
|
||||
noInline(testHelloW);
|
||||
|
||||
function testHelloWo(string) {
|
||||
return string.match(/HelloWo/);
|
||||
}
|
||||
noInline(testHelloWo);
|
||||
|
||||
function testHelloWor(string) {
|
||||
return string.match(/HelloWor/);
|
||||
}
|
||||
noInline(testHelloWor);
|
||||
|
||||
function testHelloWorl(string) {
|
||||
return string.match(/HelloWorl/);
|
||||
}
|
||||
noInline(testHelloWorl);
|
||||
|
||||
function testHelloWorld(string) {
|
||||
return string.match(/HelloWorld/);
|
||||
}
|
||||
noInline(testHelloWorld);
|
||||
|
||||
for (var i = 0; i < 1e4; ++i) {
|
||||
shouldBe(testH("HelloWorld")[0], `H`);
|
||||
shouldBe(testHe("HelloWorld")[0], `He`);
|
||||
shouldBe(testHel("HelloWorld")[0], `Hel`);
|
||||
shouldBe(testHell("HelloWorld")[0], `Hell`);
|
||||
shouldBe(testHello("HelloWorld")[0], `Hello`);
|
||||
shouldBe(testHelloW("HelloWorld")[0], `HelloW`);
|
||||
shouldBe(testHelloWo("HelloWorld")[0], `HelloWo`);
|
||||
shouldBe(testHelloWor("HelloWorld")[0], `HelloWor`);
|
||||
shouldBe(testHelloWorl("HelloWorld")[0], `HelloWorl`);
|
||||
shouldBe(testHelloWorld("HelloWorld")[0], `HelloWorld`);
|
||||
shouldBe(testH("HelloWorldこんにちは")[0], `H`);
|
||||
shouldBe(testHe("HelloWorldこんにちは")[0], `He`);
|
||||
shouldBe(testHel("HelloWorldこんにちは")[0], `Hel`);
|
||||
shouldBe(testHell("HelloWorldこんにちは")[0], `Hell`);
|
||||
shouldBe(testHello("HelloWorldこんにちは")[0], `Hello`);
|
||||
shouldBe(testHelloW("HelloWorldこんにちは")[0], `HelloW`);
|
||||
shouldBe(testHelloWo("HelloWorldこんにちは")[0], `HelloWo`);
|
||||
shouldBe(testHelloWor("HelloWorldこんにちは")[0], `HelloWor`);
|
||||
shouldBe(testHelloWorl("HelloWorldこんにちは")[0], `HelloWorl`);
|
||||
shouldBe(testHelloWorld("HelloWorldこんにちは")[0], `HelloWorld`);
|
||||
}
|
|
@ -1,23 +0,0 @@
|
|||
function assert(b) {
|
||||
if (!b)
|
||||
throw new Error;
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
function foo(a, b) {
|
||||
let r1 = b[0];
|
||||
let x = [...a];
|
||||
let r2 = b[0];
|
||||
assert(r1 + r2 === 43);
|
||||
}
|
||||
noInline(foo);
|
||||
|
||||
let b = [42];
|
||||
let a = [];
|
||||
a[Symbol.iterator] = function* () {
|
||||
b[0] = 1;
|
||||
};
|
||||
for (let i = 0; i < 10000; ++i) {
|
||||
b[0] = 42;
|
||||
foo(a, b);
|
||||
}
|
|
@ -1,15 +0,0 @@
|
|||
function foo(o, i) {
|
||||
return o[i]();
|
||||
}
|
||||
|
||||
noInline(foo);
|
||||
|
||||
for (var i = 0; i < 10000; ++i) {
|
||||
var result = foo([function() { return 42; }], 0);
|
||||
if (result != 42)
|
||||
throw "Error: bad result: " + result;
|
||||
}
|
||||
|
||||
var result = foo([function() { return 43; }], 0);
|
||||
if (result != 43)
|
||||
throw "Error: bad result at end: " + result;
|
|
@ -1,29 +0,0 @@
|
|||
"use strict";
|
||||
|
||||
function Foo() {}
|
||||
|
||||
function second(next, cp) {
|
||||
return 100;
|
||||
}
|
||||
|
||||
function first(next, cp) {
|
||||
return cp < 60 ? new Foo() : next(cp);
|
||||
}
|
||||
|
||||
function createClosure(next, strategy) {
|
||||
return function closure(cp) {
|
||||
return strategy(next, cp);
|
||||
};
|
||||
}
|
||||
|
||||
var tmp = createClosure(null, second);
|
||||
var bar = createClosure(tmp, first);
|
||||
|
||||
noInline(bar);
|
||||
|
||||
for (var i=0; i<50000; i++) {
|
||||
bar(32);
|
||||
bar(32);
|
||||
bar(32);
|
||||
bar(100);
|
||||
}
|
|
@ -1,21 +0,0 @@
|
|||
"use strict";
|
||||
|
||||
function makeClosure(x)
|
||||
{
|
||||
return (f) => {
|
||||
if (x == 42) {
|
||||
x = 0;
|
||||
return f(f);
|
||||
}
|
||||
else
|
||||
return x;
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < 1000; ++i) {
|
||||
var g = makeClosure(42);
|
||||
var h = makeClosure(41);
|
||||
var value = g(h);
|
||||
if (value != 41)
|
||||
throw "Wrong result, got: " + value;
|
||||
}
|
|
@ -1,92 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error(`bad value: ${String(actual)}`);
|
||||
}
|
||||
|
||||
function shouldThrow(func, errorMessage) {
|
||||
var errorThrown = false;
|
||||
var error = null;
|
||||
try {
|
||||
func();
|
||||
} catch (e) {
|
||||
errorThrown = true;
|
||||
error = e;
|
||||
}
|
||||
if (!errorThrown)
|
||||
throw new Error('not thrown');
|
||||
if (String(error) !== errorMessage)
|
||||
throw new Error(`bad error: ${String(error)}`);
|
||||
}
|
||||
var globalEval = eval;
|
||||
var global = this;
|
||||
|
||||
// EvalContextType.
|
||||
{
|
||||
function hello()
|
||||
{
|
||||
return eval('new.target');
|
||||
}
|
||||
shouldBe(hello(), undefined);
|
||||
shouldBe(hello(), undefined);
|
||||
shouldBe(hello(), undefined);
|
||||
globalEval(`
|
||||
var thrown = false;
|
||||
try {
|
||||
eval('new.target');
|
||||
} catch (e) {
|
||||
thrown = true;
|
||||
shouldBe(String(e), "SyntaxError: new.target is only valid inside functions.");
|
||||
}
|
||||
shouldBe(thrown, true);
|
||||
`);
|
||||
|
||||
var thrown = false;
|
||||
try {
|
||||
globalEval('new.target');
|
||||
} catch (e) {
|
||||
thrown = true;
|
||||
shouldBe(String(e), "SyntaxError: new.target is only valid inside functions.");
|
||||
}
|
||||
shouldBe(thrown, true);
|
||||
}
|
||||
|
||||
// DerivedContextType.
|
||||
{
|
||||
var object = {
|
||||
hello()
|
||||
{
|
||||
return eval('super.ok');
|
||||
}
|
||||
};
|
||||
object.__proto__ = { ok: 42 };
|
||||
shouldBe(object.hello(), 42);
|
||||
|
||||
var test = {
|
||||
hello: function () {
|
||||
return eval('super.ok');
|
||||
}
|
||||
};
|
||||
test.__proto__ = { ok: 42 };
|
||||
shouldThrow(function () {
|
||||
test.hello();
|
||||
}, `SyntaxError: super is not valid in this context.`);
|
||||
}
|
||||
|
||||
// isArrowFunctionContext.
|
||||
{
|
||||
globalEval(`
|
||||
function ok()
|
||||
{
|
||||
return eval('this');
|
||||
}
|
||||
shouldBe(ok(), global)
|
||||
var hello = {
|
||||
hello()
|
||||
{
|
||||
var arrow = () => eval('this');
|
||||
shouldBe(arrow(), hello);
|
||||
}
|
||||
};
|
||||
hello.hello();
|
||||
`);
|
||||
}
|
|
@ -1,24 +0,0 @@
|
|||
//@ runDefault("--jitPolicyScale=0", "--validateFTLOSRExitLiveness=1", "--useConcurrentJIT=0")
|
||||
|
||||
// This should not crash in liveness validation.
|
||||
|
||||
function baz() { }
|
||||
noInline(baz);
|
||||
|
||||
function foo() {
|
||||
let i, j;
|
||||
let a0 = [0, 1];
|
||||
let a1 = [];
|
||||
for (i = 0; i < a0.length; i++) {
|
||||
a1.push();
|
||||
for (j = 0; j < 6; j++) {
|
||||
}
|
||||
for (j = 0; j < 4; j++) {
|
||||
baz();
|
||||
}
|
||||
}
|
||||
throw new Error();
|
||||
}
|
||||
try {
|
||||
new foo();
|
||||
} catch { }
|
|
@ -1,65 +0,0 @@
|
|||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + String(actual) + ' ' + String(expected));
|
||||
}
|
||||
noInline(shouldBe);
|
||||
|
||||
function zero()
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
noInline(zero);
|
||||
|
||||
function negativeZero()
|
||||
{
|
||||
return -0;
|
||||
}
|
||||
noInline(negativeZero);
|
||||
|
||||
var object = {
|
||||
valueOf()
|
||||
{
|
||||
return -0;
|
||||
}
|
||||
};
|
||||
|
||||
function test()
|
||||
{
|
||||
shouldBe(0 < zero(), false);
|
||||
shouldBe(0 < (-zero()), false);
|
||||
shouldBe(0 <= zero(), true);
|
||||
shouldBe(0 <= (-zero()), true);
|
||||
shouldBe(0 > zero(), false);
|
||||
shouldBe(0 > (-zero()), false);
|
||||
shouldBe(0 >= zero(), true);
|
||||
shouldBe(0 >= (-zero()), true);
|
||||
shouldBe(0 == zero(), true);
|
||||
shouldBe(0 == (-zero()), true);
|
||||
shouldBe(0 === zero(), true);
|
||||
shouldBe(0 === (-zero()), true);
|
||||
shouldBe(0 != zero(), false);
|
||||
shouldBe(0 != (-zero()), false);
|
||||
shouldBe(0 !== zero(), false);
|
||||
shouldBe(0 !== (-zero()), false);
|
||||
|
||||
shouldBe(0 < object, false);
|
||||
shouldBe(0 < -object, false);
|
||||
shouldBe(0 <= object, true);
|
||||
shouldBe(0 <= -object, true);
|
||||
shouldBe(0 > object, false);
|
||||
shouldBe(0 > -object, false);
|
||||
shouldBe(0 >= object, true);
|
||||
shouldBe(0 >= -object, true);
|
||||
shouldBe(0 == object, true);
|
||||
shouldBe(0 == -object, true);
|
||||
shouldBe(0 === object, false);
|
||||
shouldBe(0 === -object, true);
|
||||
shouldBe(0 != object, false);
|
||||
shouldBe(0 != -object, false);
|
||||
shouldBe(0 !== object, true);
|
||||
shouldBe(0 !== -object, false);
|
||||
}
|
||||
noInline(test);
|
||||
|
||||
for (var i = 0; i < 1e5; ++i)
|
||||
test();
|
|
@ -1,3 +1,8 @@
|
|||
/**
|
||||
* reviewed
|
||||
* v8: add tests for completion value resolution in try/catch/finally blocks (anything returning -2)
|
||||
*/
|
||||
|
||||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error('bad value: ' + actual);
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
function shouldBe(actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw new Error(`bad value: ${String(actual)}`);
|
||||
|
|
|
@ -1,68 +0,0 @@
|
|||
let nullSymbol = Symbol();
|
||||
|
||||
let propKeys = [
|
||||
"foo", "", undefined, null, true, false, 0, 10, 1234.567,
|
||||
Symbol("foo"), Symbol(""), nullSymbol,
|
||||
];
|
||||
|
||||
function toKeyString(x) {
|
||||
if (typeof x === "string")
|
||||
return '"' + x + '"';
|
||||
if (typeof x === "symbol")
|
||||
return x.toString();
|
||||
return "" + x;
|
||||
}
|
||||
|
||||
function toFuncName(x) {
|
||||
if (typeof x === "symbol") {
|
||||
if (x !== nullSymbol) {
|
||||
let str = x.toString();
|
||||
let key = str.slice(7, str.length - 1);
|
||||
return "[" + key + "]";
|
||||
}
|
||||
return "";
|
||||
}
|
||||
return "" + x;
|
||||
}
|
||||
|
||||
function shouldBe(title, actual, expected) {
|
||||
if (actual !== expected)
|
||||
throw Error(title + ": actual:" + actual + " expected:" + expected);
|
||||
}
|
||||
|
||||
function makeObj(propKey, classMethodName) {
|
||||
return {
|
||||
[propKey]: class { static [classMethodName](){} },
|
||||
};
|
||||
}
|
||||
noInline(makeObj);
|
||||
|
||||
for (var i = 0; i < 1000; i++) {
|
||||
for (var k = 0; k < propKeys.length; k++) {
|
||||
let key = propKeys[k];
|
||||
let o = makeObj(key, "prop");
|
||||
shouldBe("typeof o[" + toKeyString(key) + "].name", typeof o[key].name, "string");
|
||||
shouldBe("o[" + toKeyString(key) + "].name", o[key].name, toFuncName(key));
|
||||
}
|
||||
|
||||
for (var k = 0; k < propKeys.length; k++) {
|
||||
let key = propKeys[k];
|
||||
let o = makeObj(key, "name");
|
||||
shouldBe("typeof o[" + toKeyString(key) + "].name", typeof o[key], "function");
|
||||
}
|
||||
|
||||
for (var k = 0; k < propKeys.length; k++) {
|
||||
let key = propKeys[k];
|
||||
let prop = { toString() { return "prop" } };
|
||||
let o = makeObj(key, prop);
|
||||
shouldBe("typeof o[" + toKeyString(key) + "].name", typeof o[key].name, "string");
|
||||
shouldBe("o[" + toKeyString(key) + "].name", o[key].name, toFuncName(key));
|
||||
}
|
||||
|
||||
for (var k = 0; k < propKeys.length; k++) {
|
||||
let key = propKeys[k];
|
||||
let prop = { toString() { return "name" } };
|
||||
let o = makeObj(key, prop);
|
||||
shouldBe("typeof o[" + toKeyString(key) + "].name", typeof o[key], "function");
|
||||
}
|
||||
}
|
|
@ -1,3 +1,4 @@
|
|||
// Reviewed
|
||||
//@ skip
|
||||
|
||||
let length = 0x10001000;
|
||||
|
|
|
@ -1,19 +0,0 @@
|
|||
Array.prototype[1] = 5;
|
||||
|
||||
function arrayEq(a, b) {
|
||||
if (a.length !== b.length)
|
||||
throw new Error([a, "\n\n", b]);
|
||||
|
||||
for (let i = 0; i < a.length; i++) {
|
||||
if (a[i] !== b[i])
|
||||
throw new Error([a, "\n\n", b]);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
let obj = {};
|
||||
arrayEq([1,2,3].concat(4), [1,2,3,4]);
|
||||
arrayEq([1,2,3].concat(1.34), [1,2,3,1.34]);
|
||||
arrayEq([1.35,2,3].concat(1.34), [1.35,2,3,1.34]);
|
||||
arrayEq([1.35,2,3].concat(obj), [1.35,2,3,obj]);
|
||||
arrayEq([1,2,3].concat(obj), [1,2,3,obj]);
|
|
@ -1,115 +0,0 @@
|
|||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
function shouldThrowInvalidConstAssignment(f) {
|
||||
var threw = false;
|
||||
try {
|
||||
f();
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("TypeError") !== -1 && e.message.indexOf("readonly") !== -1)
|
||||
threw = true;
|
||||
}
|
||||
assert(threw);
|
||||
}
|
||||
noInline(shouldThrowInvalidConstAssignment);
|
||||
|
||||
|
||||
// Tests
|
||||
|
||||
const NUM_LOOPS = 100;
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 40;
|
||||
with ({x : 100}) {
|
||||
assert(x === 100);
|
||||
}
|
||||
with ({y : 100}) {
|
||||
assert(x === 40);
|
||||
}
|
||||
}
|
||||
noInline(foo);
|
||||
|
||||
function bar() {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
with ({x : 100}) {
|
||||
if (truth()) {
|
||||
const x = 50;
|
||||
const capX = function() { return x; }
|
||||
assert(x === 50);
|
||||
assert(capX() === x);
|
||||
}
|
||||
assert(x === 100);
|
||||
assert(capX() === 40);
|
||||
}
|
||||
with ({y : 100}) {
|
||||
if (truth()) {
|
||||
const x = 50;
|
||||
const capX = function() { return x; }
|
||||
assert(x === 50);
|
||||
assert(capX() === x);
|
||||
}
|
||||
assert(x === 40);
|
||||
assert(capX() === 40);
|
||||
}
|
||||
}
|
||||
noInline(bar);
|
||||
|
||||
function baz() {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
with ({x : 100}) {
|
||||
if (truth()) {
|
||||
const x = 50;
|
||||
assert(x === 50);
|
||||
}
|
||||
assert(x === 100);
|
||||
assert(capX() === 40);
|
||||
}
|
||||
with ({y : 100}) {
|
||||
if (truth()) {
|
||||
const x = 50;
|
||||
assert(x === 50);
|
||||
}
|
||||
assert(x === 40);
|
||||
assert(capX() === 40);
|
||||
}
|
||||
}
|
||||
noInline(baz);
|
||||
|
||||
for (let i = 0; i < NUM_LOOPS; i++) {
|
||||
foo();
|
||||
bar();
|
||||
baz();
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 40;
|
||||
with ({x : 100}) {
|
||||
assert(x === 100);
|
||||
x = 20;
|
||||
assert(x === 20);
|
||||
}
|
||||
assert(x === 40);
|
||||
with ({y : 100}) {
|
||||
assert(x === 40);
|
||||
x = 100;
|
||||
}
|
||||
}
|
||||
for (let i = 0; i < NUM_LOOPS; ++i) {
|
||||
shouldThrowInvalidConstAssignment(foo);
|
||||
}
|
||||
|
||||
})();
|
|
@ -1,204 +0,0 @@
|
|||
"use strict";
|
||||
|
||||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
// Tests
|
||||
|
||||
|
||||
const NUM_LOOPS = 100;
|
||||
|
||||
;(function () {
|
||||
function foo() {
|
||||
const x = 20;
|
||||
const y = "y";
|
||||
try {
|
||||
assert(x === 20);
|
||||
assert(y === "y");
|
||||
throw "error";
|
||||
} catch(e) {
|
||||
assert(x === 20);
|
||||
} finally {
|
||||
assert(x === 20);
|
||||
assert(y === "y");
|
||||
}
|
||||
|
||||
for (let i = 0; i < 1; i++) {
|
||||
let numFinally = 0;
|
||||
try {
|
||||
let a = 40;
|
||||
let capA = function() { return a; }
|
||||
assert(capA() === 40);
|
||||
try {
|
||||
const b = 41;
|
||||
const capB = function() { return b; }
|
||||
assert(capB() === 41);
|
||||
assert(capA() === 40);
|
||||
try {
|
||||
return 20;
|
||||
} catch(e){
|
||||
} finally {
|
||||
const c = 42;
|
||||
const capC = function() { return c; }
|
||||
assert(capC() === 42);
|
||||
assert(capB() === 41);
|
||||
assert(capA() === 40);
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
return 22;
|
||||
}
|
||||
} catch(e) {
|
||||
} finally {
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
return 23;
|
||||
}
|
||||
} catch(e) {
|
||||
} finally {
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
assert(numFinally === 3);
|
||||
return 24;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
assert(foo() === 24);
|
||||
}
|
||||
|
||||
})();
|
||||
|
||||
|
||||
;(function () {
|
||||
function foo() {
|
||||
for (let i = 0; i < 1; i++) {
|
||||
let numFinally = 0;
|
||||
let numErrors = 0;
|
||||
try {
|
||||
let a = 40;
|
||||
let capA = function() { return a; }
|
||||
assert(capA() === 40);
|
||||
try {
|
||||
const b = 41;
|
||||
const capB = function() { return b; }
|
||||
assert(capB() === 41);
|
||||
assert(capA() === 40);
|
||||
try {
|
||||
throw "e";
|
||||
} catch(e) {
|
||||
assert(i === 0);
|
||||
assert(capB() === 41);
|
||||
assert(capA() === 40);
|
||||
numErrors++;
|
||||
throw e;
|
||||
} finally {
|
||||
const c = 42;
|
||||
const capC = function() { return c; }
|
||||
const local = "local";
|
||||
assert(local === "local");
|
||||
assert(capC() === 42);
|
||||
assert(capB() === 41);
|
||||
assert(capA() === 40);
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
}
|
||||
} catch(e) {
|
||||
assert(i === 0);
|
||||
assert(capA() === 40);
|
||||
numErrors++;
|
||||
const local = "local";
|
||||
assert(local === "local");
|
||||
} finally {
|
||||
assert(capA() === 40);
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
const local = "local";
|
||||
assert(local === "local");
|
||||
return 23;
|
||||
}
|
||||
} catch(e) {
|
||||
//assert(i === 0);
|
||||
} finally {
|
||||
if (i === 0) {
|
||||
numFinally++;
|
||||
}
|
||||
|
||||
assert(numFinally === 3);
|
||||
assert(numErrors === 2);
|
||||
return 24;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
assert(foo() === 24);
|
||||
}
|
||||
|
||||
})();
|
||||
|
||||
|
||||
var d = 100;
|
||||
;(function (){
|
||||
function foo() {
|
||||
assert(d === 100);
|
||||
for (let i = 0; i < 1; i++) {
|
||||
let numFinally = 0;
|
||||
let numErrors = 0;
|
||||
const c = 44;
|
||||
assert(d === 100);
|
||||
try {
|
||||
const d = 45;
|
||||
if (truth()) {
|
||||
const a = 20;
|
||||
const capA = function() { return a; }
|
||||
assert(capA() === 20);
|
||||
if (truth()) {
|
||||
const b = 21;
|
||||
const e = 48;
|
||||
const capB = function() { return b; }
|
||||
assert(capB() === 21);
|
||||
assert(d === 45);
|
||||
try {
|
||||
throw "e";
|
||||
} catch(e) {
|
||||
assert(capA() === 20);
|
||||
assert(a === 20);
|
||||
numErrors++;
|
||||
} finally {
|
||||
assert(capA() === 20);
|
||||
assert(e === 48);
|
||||
numFinally++;
|
||||
return 30;
|
||||
}
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
assert(c === 44);
|
||||
assert(d === 100);
|
||||
numFinally++;
|
||||
assert(numFinally === 2);
|
||||
assert(numErrors === 1);
|
||||
return 40;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
assert(foo() === 40);
|
||||
}
|
||||
|
||||
})();
|
|
@ -1,130 +0,0 @@
|
|||
"use strict";
|
||||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
function shouldThrowInvalidConstAssignment(f) {
|
||||
var threw = false;
|
||||
try {
|
||||
f();
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("TypeError") !== -1 && e.message.indexOf("readonly") !== -1)
|
||||
threw = true;
|
||||
}
|
||||
assert(threw);
|
||||
}
|
||||
noInline(shouldThrowInvalidConstAssignment);
|
||||
|
||||
|
||||
// ========== tests below ===========
|
||||
|
||||
const NUM_LOOPS = 1000;
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const obj = {a: 20, b: 40, c: 50};
|
||||
const props = [];
|
||||
for (const p in obj)
|
||||
props.push(p);
|
||||
assert(props.length === 3);
|
||||
for (const p of props)
|
||||
assert(!!obj[p]);
|
||||
assert(props.indexOf("a") !== -1);
|
||||
assert(props.indexOf("b") !== -1);
|
||||
assert(props.indexOf("c") !== -1);
|
||||
}
|
||||
|
||||
function bar() {
|
||||
const funcs = [];
|
||||
for (const item of [1, 2, 3])
|
||||
funcs.push(function() { return item; });
|
||||
let i = 1;
|
||||
assert(funcs.length === 3);
|
||||
for (const f of funcs) {
|
||||
assert(f() === i);
|
||||
i++;
|
||||
}
|
||||
}
|
||||
function baz() {
|
||||
const funcs = [];
|
||||
const obj = {a:1, b:2, c:3};
|
||||
for (const p in obj)
|
||||
funcs.push(function() { return p; });
|
||||
let i = 1;
|
||||
assert(funcs.length === 3);
|
||||
for (const f of funcs) {
|
||||
assert(obj[f()] === i);
|
||||
i++;
|
||||
}
|
||||
}
|
||||
function taz() {
|
||||
for (const item of [1,2,3]) {
|
||||
const item = 20;
|
||||
assert(item === 20);
|
||||
}
|
||||
}
|
||||
function jaz() {
|
||||
let i = 0;
|
||||
for (const x = 40; i < 10; i++) {
|
||||
assert(x === 40);
|
||||
}
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
foo();
|
||||
bar();
|
||||
baz();
|
||||
taz();
|
||||
jaz();
|
||||
}
|
||||
})();
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
for (const item of [1,2,3])
|
||||
item = 20;
|
||||
}
|
||||
function bar() {
|
||||
for (const item of [1,2,3])
|
||||
eval("item = 20")
|
||||
}
|
||||
function baz() {
|
||||
for (const p in {a: 20, b: 40})
|
||||
p = 20;
|
||||
}
|
||||
function taz() {
|
||||
for (const p in {a: 20, b: 40})
|
||||
eval("p = 20")
|
||||
}
|
||||
function jaz() {
|
||||
for (const x = 0; x < 10; x++) { }
|
||||
}
|
||||
function raz() {
|
||||
for (const x = 0; x < 10; ++x) { }
|
||||
}
|
||||
function paz() {
|
||||
for (const x = 0; x < 10; x++) {
|
||||
let f = function() { return x; }
|
||||
}
|
||||
}
|
||||
function maz() {
|
||||
for (const x = 0; x < 10; ++x) {
|
||||
let f = function() { return x; }
|
||||
}
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowInvalidConstAssignment(foo);
|
||||
shouldThrowInvalidConstAssignment(bar);
|
||||
shouldThrowInvalidConstAssignment(baz);
|
||||
shouldThrowInvalidConstAssignment(taz);
|
||||
shouldThrowInvalidConstAssignment(jaz);
|
||||
shouldThrowInvalidConstAssignment(raz);
|
||||
shouldThrowInvalidConstAssignment(paz);
|
||||
shouldThrowInvalidConstAssignment(maz);
|
||||
}
|
||||
})();
|
|
@ -1,147 +0,0 @@
|
|||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
function shouldThrowTDZ(func) {
|
||||
var hasThrown = false;
|
||||
try {
|
||||
func();
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("ReferenceError") !== -1)
|
||||
hasThrown = true;
|
||||
}
|
||||
assert(hasThrown);
|
||||
}
|
||||
noInline(shouldThrowTDZ);
|
||||
|
||||
|
||||
// Tests
|
||||
|
||||
|
||||
const NUM_LOOPS = 1000;
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
delete x;
|
||||
const x = 20;
|
||||
}
|
||||
function bar() {
|
||||
delete x;
|
||||
const x = 20;
|
||||
function capX() { return x; }
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
}
|
||||
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
|
||||
function foo() {
|
||||
var hadError = false;
|
||||
try {
|
||||
x;
|
||||
} catch(e) {
|
||||
hadError = true;
|
||||
}
|
||||
assert(hadError);
|
||||
|
||||
if (truth()) {
|
||||
// This eval is enterpreted as follows:
|
||||
// eval("var x; x = 20");
|
||||
// We first assign undefined to the "var x".
|
||||
// Then, we interperet an assignment expression
|
||||
// into the resolved variable x. x resolves to the lexical "const x;"
|
||||
// Look at ECMA section 13.3.2.4 of the ES6 spec:
|
||||
// http://www.ecma-international.org/ecma-262/6.0/index.html#sec-variable-statement-runtime-semantics-evaluation
|
||||
// And also look at section 8.3.1 ResolveBinding:
|
||||
// http://www.ecma-international.org/ecma-262/6.0/index.html#sec-resolvebinding
|
||||
const x = 40;
|
||||
let threw = false;
|
||||
try {
|
||||
eval("var x = 20;");
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("TypeError") !== -1 && e.message.indexOf("readonly") !== -1)
|
||||
threw = true;
|
||||
}
|
||||
assert(threw);
|
||||
assert(x === 40);
|
||||
}
|
||||
assert(x === undefined);
|
||||
}
|
||||
|
||||
function bar() {
|
||||
var hadError = false;
|
||||
try {
|
||||
x;
|
||||
} catch(e) {
|
||||
hadError = true;
|
||||
}
|
||||
assert(hadError);
|
||||
|
||||
if (truth()) {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
let threw = false;
|
||||
try {
|
||||
eval("var x = 20;");
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("TypeError") !== -1 && e.message.indexOf("readonly") !== -1)
|
||||
threw = true;
|
||||
}
|
||||
assert(threw);
|
||||
assert(x === 40);
|
||||
}
|
||||
assert(x === undefined);
|
||||
}
|
||||
|
||||
function baz() {
|
||||
if (truth()) {
|
||||
const x = 40;
|
||||
eval("const x = 20; assert(x === 20);");
|
||||
assert(x === 40);
|
||||
}
|
||||
if (truth()) {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
eval("const x = 20; assert(x === 20);");
|
||||
assert(x === 40);
|
||||
}
|
||||
}
|
||||
|
||||
function baz() {
|
||||
// Test eval() caching.
|
||||
const x = 20;
|
||||
const evalString = "x;";
|
||||
|
||||
assert(eval(evalString) === 20);
|
||||
if (truth()) {
|
||||
const y = 60;
|
||||
assert(eval(evalString) === 20);
|
||||
assert(y === 60);
|
||||
if (truth()) {
|
||||
const y = 50, z = 70, x = 40;
|
||||
assert(eval(evalString) === 40);
|
||||
assert(y === 50 && z === 70);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
foo();
|
||||
bar();
|
||||
baz();
|
||||
}
|
||||
|
||||
})();
|
|
@ -1,274 +0,0 @@
|
|||
"use strict";
|
||||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
function shouldThrowInvalidConstAssignment(f) {
|
||||
var threw = false;
|
||||
try {
|
||||
f();
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("TypeError") !== -1 && e.message.indexOf("readonly") !== -1)
|
||||
threw = true;
|
||||
}
|
||||
assert(threw);
|
||||
}
|
||||
noInline(shouldThrowInvalidConstAssignment);
|
||||
|
||||
|
||||
// ========== tests below ===========
|
||||
|
||||
const NUM_LOOPS = 1000;
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 40;
|
||||
const {y} = {y: 50}, [z] = [60];
|
||||
assert(x === 40);
|
||||
assert(y === 50);
|
||||
assert(z === 60);
|
||||
}
|
||||
function bar() {
|
||||
const x = 40;
|
||||
const {y} = {y: 50}, [z] = [60];
|
||||
function capture() { return x + y + z; }
|
||||
assert(x === 40);
|
||||
assert(y === 50);
|
||||
assert(z === 60);
|
||||
assert(capture() === 150);
|
||||
if (truth()) {
|
||||
const x = "x";
|
||||
assert(x === "x");
|
||||
if (truth()) {
|
||||
let x = 100;
|
||||
const y = 200;
|
||||
assert(x === 100);
|
||||
assert(y === 200);
|
||||
|
||||
x = 10;
|
||||
assert(x === 10);
|
||||
}
|
||||
assert(x === "x");
|
||||
}
|
||||
assert(x === 40);
|
||||
}
|
||||
function baz() {
|
||||
let y = 10;
|
||||
function sideEffects() {
|
||||
y = 20;
|
||||
}
|
||||
|
||||
const x = 10;
|
||||
try {
|
||||
x = sideEffects();
|
||||
} catch(e) {}
|
||||
assert(y === 20);
|
||||
assert(x === 10);
|
||||
|
||||
try {
|
||||
x = y = 30;
|
||||
} catch(e) {}
|
||||
assert(y === 30);
|
||||
assert(x === 10);
|
||||
}
|
||||
function taz() {
|
||||
let y = 10;
|
||||
let z;
|
||||
function sideEffects() {
|
||||
y = 20;
|
||||
return ["hello", "world"];
|
||||
}
|
||||
|
||||
const x = 10;
|
||||
try {
|
||||
[z, x] = sideEffects();
|
||||
} catch(e) {}
|
||||
assert(y === 20);
|
||||
assert(x === 10);
|
||||
assert(z === "hello");
|
||||
}
|
||||
function jaz() {
|
||||
const x = "x";
|
||||
function capX() { return x; }
|
||||
assert(x === "x");
|
||||
assert(capX() === "x");
|
||||
if (truth()) {
|
||||
let y = 40;
|
||||
let capY = function() { return y; }
|
||||
assert(x === "x");
|
||||
assert(capX() === "x");
|
||||
}
|
||||
assert(x === "x");
|
||||
assert(capX() === "x");
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
foo();
|
||||
bar();
|
||||
baz();
|
||||
jaz();
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 40;
|
||||
x = 30;
|
||||
}
|
||||
function bar() {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
x = 30;
|
||||
}
|
||||
function baz() {
|
||||
const x = 40;
|
||||
assert(x === 40);
|
||||
function bad() { x = 10; }
|
||||
bad();
|
||||
}
|
||||
function jaz() {
|
||||
const x = 40;
|
||||
assert(x === 40);
|
||||
function bad() { eval("x = 10"); }
|
||||
bad();
|
||||
}
|
||||
function faz() {
|
||||
const x = 40;
|
||||
assert(x === 40);
|
||||
eval("x = 10");
|
||||
}
|
||||
function raz() {
|
||||
const x = 40;
|
||||
assert(x === 40);
|
||||
;({x} = {x: 20});
|
||||
}
|
||||
function taz() {
|
||||
const x = 40;
|
||||
function capX() { return x; }
|
||||
assert(capX() === 40);
|
||||
;({x} = {x: 20});
|
||||
}
|
||||
function paz() {
|
||||
const x = 20;
|
||||
const y = x = 20;
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowInvalidConstAssignment(foo);
|
||||
shouldThrowInvalidConstAssignment(bar);
|
||||
shouldThrowInvalidConstAssignment(baz);
|
||||
shouldThrowInvalidConstAssignment(jaz);
|
||||
shouldThrowInvalidConstAssignment(faz);
|
||||
shouldThrowInvalidConstAssignment(raz);
|
||||
shouldThrowInvalidConstAssignment(taz);
|
||||
shouldThrowInvalidConstAssignment(paz);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 40;
|
||||
eval("x = 30;");
|
||||
}
|
||||
function bar() {
|
||||
const x = 20;
|
||||
x += 20;
|
||||
}
|
||||
function baz() {
|
||||
const x = 20;
|
||||
x -= 20;
|
||||
}
|
||||
function taz() {
|
||||
const x = 20;
|
||||
shouldThrowInvalidConstAssignment(function() { x = 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x += 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x -= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x *= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x /= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x >>= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x <<= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x ^= 20; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x++; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { x--; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { ++x; });
|
||||
assert(x === 20);
|
||||
shouldThrowInvalidConstAssignment(function() { --x; });
|
||||
assert(x === 20);
|
||||
}
|
||||
function jaz() {
|
||||
const x = 20;
|
||||
let threw = false;
|
||||
try { x = 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x += 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x -= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x *= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x /= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x >>= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x <<= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x ^= 20; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { x++; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
|
||||
threw = false;
|
||||
try { x--; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
|
||||
threw = false;
|
||||
try { ++x; } catch(e) { threw = true}
|
||||
assert(threw);
|
||||
|
||||
threw = false;
|
||||
try { --x; } catch(e) { threw = true};
|
||||
assert(threw);
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowInvalidConstAssignment(foo);
|
||||
shouldThrowInvalidConstAssignment(bar);
|
||||
shouldThrowInvalidConstAssignment(baz);
|
||||
taz();
|
||||
jaz();
|
||||
}
|
||||
})();
|
|
@ -1,450 +0,0 @@
|
|||
"use strict";
|
||||
|
||||
function truth() {
|
||||
return true;
|
||||
}
|
||||
noInline(truth);
|
||||
|
||||
function assert(cond) {
|
||||
if (!cond)
|
||||
throw new Error("broke assertion");
|
||||
}
|
||||
noInline(assert);
|
||||
|
||||
function shouldThrowTDZ(func) {
|
||||
var hasThrown = false;
|
||||
try {
|
||||
func();
|
||||
} catch(e) {
|
||||
if (e.name.indexOf("ReferenceError") !== -1)
|
||||
hasThrown = true;
|
||||
}
|
||||
assert(hasThrown);
|
||||
}
|
||||
noInline(shouldThrowTDZ);
|
||||
|
||||
// Test cases
|
||||
|
||||
const NUM_LOOPS = 1000;
|
||||
const SHORT_LOOPS = 100;
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
x;
|
||||
const x = 20;
|
||||
}
|
||||
function bar() {
|
||||
const x = x;
|
||||
}
|
||||
function baz() {
|
||||
const {x: prop, y: prop2} = {x: prop};
|
||||
}
|
||||
function jaz() {
|
||||
const {x: prop, y: prop2} = {x: 20, y: prop};
|
||||
}
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
shouldThrowTDZ(baz);
|
||||
shouldThrowTDZ(jaz);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
x;
|
||||
const x = 20;
|
||||
function captureX() { return x; }
|
||||
}
|
||||
function bar() {
|
||||
captureX();
|
||||
const x = 20;
|
||||
function captureX() { return x; }
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
if (truth()) {
|
||||
const x = 20;
|
||||
assert(x === 20);
|
||||
}
|
||||
x;
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
if (truth()) {
|
||||
const y = 20;
|
||||
const captureY = function() { return y; }
|
||||
assert(y === 20);
|
||||
x;
|
||||
}
|
||||
const x = 20;
|
||||
const y = 40;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
if (truth()) {
|
||||
const y = 20;
|
||||
const x = 40;
|
||||
const captureAll = function() { return x + y; }
|
||||
assert(y === 20);
|
||||
assert(x === 40);
|
||||
assert(captureAll() === 60);
|
||||
}
|
||||
tdz;
|
||||
const tdz = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
if (truth()) {
|
||||
const y = 20;
|
||||
const x = 40;
|
||||
const captureAll = function() { return x + y + tdz; }
|
||||
assert(y === 20);
|
||||
assert(x === 40);
|
||||
}
|
||||
tdz;
|
||||
const tdz = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
x = 10;
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
x = 10;
|
||||
const x = 20;
|
||||
function cap() { return x; }
|
||||
}
|
||||
function bar() {
|
||||
captureX();
|
||||
const x = 20;
|
||||
function captureX() { return x; }
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
if (!truth()) {
|
||||
y;
|
||||
assert(false);
|
||||
}
|
||||
const y = undefined;
|
||||
assert(y === undefined);
|
||||
if (truth()) {
|
||||
x;
|
||||
}
|
||||
const x = undefined;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
eval("x;");
|
||||
const x = 20;
|
||||
}
|
||||
function bar() {
|
||||
function captureX() { return x; }
|
||||
eval("captureX();");
|
||||
const x = 20;
|
||||
}
|
||||
function baz() {
|
||||
function captureX() { return x; }
|
||||
function other() { return captureX; }
|
||||
other()();
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < SHORT_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
shouldThrowTDZ(baz);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const y = 40;
|
||||
eval("y; x;");
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
for (var i = 0; i < 1; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
const x = 20;
|
||||
const y = 40;
|
||||
assert(eval("x;") === 20);
|
||||
if (truth()) {
|
||||
assert(eval("eval('y');") === 40);
|
||||
eval("eval('x');");
|
||||
const x = 40;
|
||||
}
|
||||
}
|
||||
|
||||
for (var i = 0; i < SHORT_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
// FunctionResolveNode
|
||||
;(function() {
|
||||
function foo() {
|
||||
function captureX() { return x; }
|
||||
x();
|
||||
const x = function() { return 20; };
|
||||
}
|
||||
|
||||
function bar() {
|
||||
x();
|
||||
let x = function() { return 20; };
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
// TypeofResolveNode
|
||||
;(function() {
|
||||
function foo() {
|
||||
function captureX() { return x; }
|
||||
typeof x;
|
||||
const x = function() { return 20; };
|
||||
}
|
||||
|
||||
function bar() {
|
||||
typeof x;
|
||||
const x = function() { return 20; };
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
// ReadModifyResolveNode
|
||||
;(function() {
|
||||
function foo() {
|
||||
function captureX() { return x; }
|
||||
x++;
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
function bar() {
|
||||
x--;
|
||||
const x = 30;
|
||||
}
|
||||
|
||||
function baz() {
|
||||
x *= 40;
|
||||
const x = 30;
|
||||
}
|
||||
|
||||
function kaz() {
|
||||
function captureX() { return x; }
|
||||
x /= 20;
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
function haz() {
|
||||
function captureX() { return x; }
|
||||
--x;
|
||||
const x = 20;
|
||||
}
|
||||
|
||||
function jaz() {
|
||||
--x;
|
||||
const x = 30;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(bar);
|
||||
shouldThrowTDZ(baz);
|
||||
shouldThrowTDZ(kaz);
|
||||
shouldThrowTDZ(haz);
|
||||
shouldThrowTDZ(jaz);
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo(x) {
|
||||
const y = 50;
|
||||
let result = null;
|
||||
switch(x) {
|
||||
case 10:
|
||||
const y = 40;
|
||||
assert(y === 40);
|
||||
case 20:
|
||||
y += 1;
|
||||
assert(y === 41);
|
||||
result = y;
|
||||
break;
|
||||
default:
|
||||
result = x;
|
||||
break;
|
||||
}
|
||||
assert(y === 50);
|
||||
return result;
|
||||
}
|
||||
function bar(x) {
|
||||
const y = 50;
|
||||
let result = null;
|
||||
switch(x) {
|
||||
case 10:
|
||||
const y = 40;
|
||||
assert(y === 40);
|
||||
case 20:
|
||||
const capY = function() { return y; }
|
||||
y += 1;
|
||||
assert(y === 41);
|
||||
result = y;
|
||||
break;
|
||||
default:
|
||||
result = x;
|
||||
break;
|
||||
}
|
||||
assert(y === 50);
|
||||
return result;
|
||||
}
|
||||
function baz(x) {
|
||||
const y = 50;
|
||||
let result = null;
|
||||
switch(x) {
|
||||
case 10:
|
||||
const y = 40;
|
||||
assert(y === 40);
|
||||
case 20:
|
||||
let inc = function() { y += 1; }
|
||||
inc();
|
||||
assert(y === 41);
|
||||
result = y;
|
||||
break;
|
||||
default:
|
||||
result = x;
|
||||
break;
|
||||
}
|
||||
assert(y === 50);
|
||||
return result;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(function() { foo(20); });
|
||||
shouldThrowTDZ(function() { bar(20); });
|
||||
shouldThrowTDZ(function() { baz(20); });
|
||||
}
|
||||
})();
|
||||
|
||||
|
||||
;(function() {
|
||||
function foo() {
|
||||
[x] = [1];
|
||||
const x = null;
|
||||
}
|
||||
|
||||
function boo() {
|
||||
[x] = [1];
|
||||
const x = 20;
|
||||
function capX() { return x; }
|
||||
}
|
||||
|
||||
function bar() {
|
||||
({a, p: y} = {a: 100, p: 40});
|
||||
const y = 40;
|
||||
}
|
||||
|
||||
function zar() {
|
||||
({a, p: y} = {a: 100, p: 40});
|
||||
const y = 10;
|
||||
function capY() { return y; }
|
||||
}
|
||||
|
||||
function baz() {
|
||||
({a, p: {y}} = {a: 100, p: {p: {y: 40}}});
|
||||
const y = 100;
|
||||
}
|
||||
|
||||
function jaz() {
|
||||
({y} = {});
|
||||
const y = null;
|
||||
}
|
||||
|
||||
for (var i = 0; i < NUM_LOOPS; i++) {
|
||||
shouldThrowTDZ(foo);
|
||||
shouldThrowTDZ(boo);
|
||||
shouldThrowTDZ(bar);
|
||||
shouldThrowTDZ(zar);
|
||||
shouldThrowTDZ(baz);
|
||||
shouldThrowTDZ(jaz);
|
||||
}
|
||||
})();
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue