blob: f589ffe254aaa037103580e4c7c92f204099d459 [file] [log] [blame]
let testCases = [
// Numbers
['1', NaN, NaN, 2, 2],
['1.5', NaN, NaN, 1 + 1.5, 1 + 1.5],
[NaN, NaN, NaN, NaN, NaN],
// Strings.
['""', NaN, NaN, 1, 1],
['new String()', NaN, NaN, 1, 1],
['"WebKit!"', NaN, NaN, NaN, NaN],
// Objects.
['{ }', NaN, NaN, NaN, NaN],
['{ foo: 1 }', NaN, NaN, NaN, NaN],
['{ toString: function() { return ""; } }', NaN, NaN, 1, 1],
['{ toString: function() { return "WebKit"; } }', NaN, NaN, NaN, NaN],
// Others.
['null', NaN, NaN, 1, 1],
['undefined', NaN, NaN, NaN, NaN]
];
for (let testCase of testCases) {
let otherOperand = testCase[0];
let expectedLeftValueWithHole = testCase[1];
let expectedRightValueWithHole = testCase[2];
let expectedLeftValue = testCase[3];
let expectedRightValue = testCase[4];
eval(
`// Those holes are not observable by arithmetic operation.
// The return value is always going to be NaN.
function nonObservableHoleOnLhs(array, otherValue) {
return Math.min(array[0]) + Math.min(otherValue);
}
noInline(nonObservableHoleOnLhs);
function observableHoleOnLhs(array, otherValue) {
let value = array[0];
return [Math.min(value) + Math.min(otherValue), value];
}
noInline(observableHoleOnLhs);
function nonObservableHoleOnRhs(array, otherValue) {
return Math.min(otherValue) + Math.min(array[0]);
}
noInline(nonObservableHoleOnRhs);
function observableHoleOnRhs(array, otherValue) {
let value = array[0];
return [Math.min(otherValue) + Math.min(value), value];
}
noInline(observableHoleOnRhs);
let testArray = new Array;
for (let i = 1; i < 3; ++i) {
testArray[i] = i + 0.5
}
let isEqual = function(a, b) {
if (a === a) {
return a === b;
}
return b !== b;
}
for (let i = 0; i < 1e4; ++i) {
let lhsResult1 = nonObservableHoleOnLhs(testArray, ${otherOperand});
if (!isEqual(lhsResult1, ${expectedLeftValueWithHole}))
throw "Error on nonObservableHoleOnLhs at i = " + i + " with operand " + ${otherOperand} + " expected " + ${expectedLeftValueWithHole} + " got " + lhsResult1;
let lhsResult2 = observableHoleOnLhs(testArray, ${otherOperand});
if (!isEqual(lhsResult2[0], ${expectedLeftValueWithHole}) || lhsResult2[1] !== undefined)
throw "Error on observableHoleOnLhs at i = " + i;
let rhsResult1 = nonObservableHoleOnRhs(testArray, ${otherOperand});
if (!isEqual(rhsResult1, ${expectedRightValueWithHole}))
throw "Error on nonObservableHoleOnRhs at i = " + i + " with operand " + ${otherOperand} + " expected " + ${expectedRightValueWithHole} + " got " + rhsResult1;
let rhsResult2 = observableHoleOnRhs(testArray, ${otherOperand});
if (!isEqual(rhsResult2[0], ${expectedRightValueWithHole}) || rhsResult2[1] !== undefined)
throw "Error on observableHoleOnRhs at i = " + i;
}
// Fill the hole, make sure everything still work correctly.
testArray[0] = 1.;
for (let i = 0; i < 1e4; ++i) {
let lhsResult1 = nonObservableHoleOnLhs(testArray, ${otherOperand});
if (!isEqual(lhsResult1, ${expectedLeftValue}))
throw "Error on non hole nonObservableHoleOnLhs at i = " + i + " expected " + ${expectedLeftValue} + " got " + lhsResult1;
let lhsResult2 = observableHoleOnLhs(testArray, ${otherOperand});
if (!isEqual(lhsResult2[0], ${expectedLeftValue}) || lhsResult2[1] !== 1)
throw "Error on non hole observableHoleOnLhs at i = " + i + " expected " + ${expectedLeftValue} + " got " + lhsResult2[0];
let rhsResult1 = nonObservableHoleOnRhs(testArray, ${otherOperand});
if (!isEqual(rhsResult1, ${expectedRightValue}))
throw "Error on non hole nonObservableHoleOnRhs at i = " + i + " with operand " + ${otherOperand} + " expected " + ${expectedRightValue} + " got " + rhsResult1;
let rhsResult2 = observableHoleOnRhs(testArray, ${otherOperand});
if (!isEqual(rhsResult2[0], ${expectedRightValue}) || rhsResult2[1] !== 1)
throw "Error on non hole observableHoleOnRhs at i = " + i;
}`
);
}