nashorn/test/script/basic/JDK-8035712.js
author hannesw
Fri, 03 Feb 2017 13:28:00 +0100
changeset 43555 1bb10bccf057
parent 31925 85c87bc2f1f3
permissions -rw-r--r--
8173888: Test for JDK-8169481 causes stack overflows in parser tests Reviewed-by: jlaskey, sundar

/*
 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

/**
 * JDK-8035712: Restore some of the RuntimeCallSite specializations
 *
 * @test
 * @run
 */

if ((typeof Assert) == "undefined") {
    Assert = { 
        assertTrue: function(x) { if(!x) { throw "expected true" } },
        assertFalse: function(x) { if(x) { throw "expected false" } },
    };
}

function nop() {}

function EQ(x, y) {
    // Exercise normal evaluation
    Assert.assertTrue (x == y);
    Assert.assertTrue (y == x);
    Assert.assertFalse(x != y);
    Assert.assertFalse(y != x);
    // Exercise the branch optimizer
    if (x == y) { nop(); } else { Assert.fail(); }
    if (y == x) { nop(); } else { Assert.fail(); }
    if (x != y) { Assert.fail(); } else { nop(); }
    if (y != x) { Assert.fail(); } else { nop(); }
}

function NE(x, y) {
    // Exercise normal evaluation
    Assert.assertTrue (x != y);
    Assert.assertTrue (y != x);
    Assert.assertFalse(x == y);
    Assert.assertFalse(y == x);
    // Exercise the branch optimizer
    if (x != y) { nop(); } else { Assert.fail(); }
    if (y != x) { nop(); } else { Assert.fail(); }
    if (x == y) { Assert.fail(); } else { nop(); }
    if (y == x) { Assert.fail(); } else { nop(); }
}

function STRICT_EQ(x, y) {
    // Exercise normal evaluation
    Assert.assertTrue (x === y);
    Assert.assertTrue (y === x);
    Assert.assertFalse(x !== y);
    Assert.assertFalse(y !== x);
    // Exercise the branch optimizer
    if (x === y) { nop(); } else { Assert.fail(); }
    if (y === x) { nop(); } else { Assert.fail(); }
    if (x !== y) { Assert.fail(); } else { nop(); }
    if (y !== x) { Assert.fail(); } else { nop(); }
}

function STRICT_NE(x, y) {
    // Exercise normal evaluation
    Assert.assertTrue (x !== y);
    Assert.assertTrue (y !== x);
    Assert.assertFalse(x === y);
    Assert.assertFalse(y === x);
    // Exercise the branch optimizer
    if (x !== y) { nop(); } else { Assert.fail(); }
    if (y !== x) { nop(); } else { Assert.fail(); }
    if (x === y) { Assert.fail(); } else { nop(); }
    if (y === x) { Assert.fail(); } else { nop(); }
}

function cmpToAnyNumber(cmp, value) {
    cmp(1, value);
    cmp(4294967296, value);
    cmp(1.2, value);
    cmp(Infinity, value);
    cmp(-Infinity, value);
    cmp(1/Infinity, value);
    cmp(0, value);
    cmp(-0, value);
    cmp(true, value);
    cmp(false, value);
}

function notEqualToAnyNumber(value) {
    cmpToAnyNumber(NE, value);
    cmpToAnyNumber(STRICT_NE, value);
}

notEqualToAnyNumber(null);
notEqualToAnyNumber(void 0);
notEqualToAnyNumber("abc");
notEqualToAnyNumber({});
notEqualToAnyNumber(["xyz"]);

function objectWithPrimitiveFunctionNotEqualToAnyNumber(fnName) {
    var obj = {
        count: 0
    };
    obj[fnName] = function() { this.count++; return "foo"; };
    notEqualToAnyNumber(obj);
    // Every NE will invoke it 8 times; cmpToAnyNumber has 10 comparisons
    // STRICT_NE doesn't invoke toString.
    Assert.assertTrue(80 === obj.count);
}
objectWithPrimitiveFunctionNotEqualToAnyNumber("valueOf");
objectWithPrimitiveFunctionNotEqualToAnyNumber("toString");

function objectEqualButNotStrictlyEqual(val, obj) {
    EQ(val, obj);
    STRICT_NE(val, obj);
}

function numberEqualButNotStrictlyEqualToObject(num, obj) {
    objectEqualButNotStrictlyEqual(num, obj);
    objectEqualButNotStrictlyEqual(num, [obj]);
    objectEqualButNotStrictlyEqual(num, [[obj]]);
}

function numberEqualButNotStrictlyEqualToZeroObjects(num) {
    numberEqualButNotStrictlyEqualToObject(num, [0]);
    numberEqualButNotStrictlyEqualToObject(num, "");
    numberEqualButNotStrictlyEqualToObject(num, []);
    numberEqualButNotStrictlyEqualToObject(num, "0");
}

numberEqualButNotStrictlyEqualToZeroObjects(0);
numberEqualButNotStrictlyEqualToZeroObjects(1/Infinity);
numberEqualButNotStrictlyEqualToZeroObjects(false);

function numberEqualButNotStrictlyEqualToObjectEquivalent(num) {
    var str = String(num);
    objectEqualButNotStrictlyEqual(num, str);
    objectEqualButNotStrictlyEqual(num, { valueOf:  function() { return str }});
    objectEqualButNotStrictlyEqual(num, { toString: function() { return str }});
    objectEqualButNotStrictlyEqual(num, { valueOf:  function() { return num }});
    objectEqualButNotStrictlyEqual(num, { toString: function() { return num }});
}

numberEqualButNotStrictlyEqualToObjectEquivalent(1);
numberEqualButNotStrictlyEqualToObjectEquivalent(4294967296);
numberEqualButNotStrictlyEqualToObjectEquivalent(1.2);
numberEqualButNotStrictlyEqualToObjectEquivalent(Infinity);
numberEqualButNotStrictlyEqualToObjectEquivalent(-Infinity);
numberEqualButNotStrictlyEqualToObjectEquivalent(1/Infinity);
numberEqualButNotStrictlyEqualToObjectEquivalent(0);
numberEqualButNotStrictlyEqualToObjectEquivalent(-0);

STRICT_EQ(1, new java.lang.Integer(1));
STRICT_EQ(1, new java.lang.Double(1));
STRICT_EQ(1.2, new java.lang.Double(1.2));

function LE(x, y) {
    // Exercise normal evaluation
    Assert.assertTrue(x <= y);
    Assert.assertTrue(y >= x);
    Assert.assertFalse(x > y);
    Assert.assertFalse(x < y);
    // Exercise the branch optimizer
    if (x <= y) { nop(); } else { Assert.fail(); }
    if (y >= x) { nop(); } else { Assert.fail(); }
    if (x > y) { Assert.fail(); } else { nop(); }
    if (y < x) { Assert.fail(); } else { nop(); }
}

function mutuallyLessThanOrEqual(x, y) {
    LE(x, y);
    LE(y, x);
}

mutuallyLessThanOrEqual(0, null);
mutuallyLessThanOrEqual(false, null);
mutuallyLessThanOrEqual(1/Infinity, null);

function mutuallyLessThanEqualToObjectWithValue(num, val) {
    mutuallyLessThanOrEqual(num, { valueOf: function() { return val } });
    mutuallyLessThanOrEqual(num, { toString: function() { return val } });
}

mutuallyLessThanEqualToObjectWithValue(false, 0);
mutuallyLessThanEqualToObjectWithValue(false, "");

mutuallyLessThanEqualToObjectWithValue(true, 1);
mutuallyLessThanEqualToObjectWithValue(true, "1");

function lessThanEqualToObjectEquivalent(num) {
    var str = String(num);
    mutuallyLessThanOrEqual(num, str);
    mutuallyLessThanEqualToObjectWithValue(num, num);
    mutuallyLessThanEqualToObjectWithValue(num, str);
}

lessThanEqualToObjectEquivalent(1);
lessThanEqualToObjectEquivalent(4294967296);
lessThanEqualToObjectEquivalent(1.2);
lessThanEqualToObjectEquivalent(Infinity);
lessThanEqualToObjectEquivalent(-Infinity);
lessThanEqualToObjectEquivalent(1/Infinity);
lessThanEqualToObjectEquivalent(0);
lessThanEqualToObjectEquivalent(-0);

function INCOMPARABLE(x, y) {
    // Exercise normal evaluation
    Assert.assertFalse(x < y);
    Assert.assertFalse(x > y);
    Assert.assertFalse(x <= y);
    Assert.assertFalse(x >= y);
    Assert.assertFalse(y < x);
    Assert.assertFalse(y > x);
    Assert.assertFalse(y <= x);
    Assert.assertFalse(y >= x);
    // Exercise the branch optimizer
    if (x < y) { Assert.fail(); } else { nop(); }
    if (x > y) { Assert.fail(); } else { nop(); }
    if (x <= y) { Assert.fail(); } else { nop(); }
    if (x >= y) { Assert.fail(); } else { nop(); }
    if (y < x) { Assert.fail(); } else { nop(); }
    if (y > x) { Assert.fail(); } else { nop(); }
    if (y <= x) { Assert.fail(); } else { nop(); }
    if (y >= x) { Assert.fail(); } else { nop(); }
}

function isIncomparable(value) {
    cmpToAnyNumber(INCOMPARABLE, value);
}

isIncomparable(void 0);
isIncomparable({ valueOf: function() { return NaN }});
isIncomparable({ toString: function() { return NaN }});

// Force ScriptRuntime.LT(Object, Object) etc. comparisons
function cmpObj(fn, x, y) {
    fn({valueOf: function() { return x }}, {valueOf: function() { return y }});
}

function LT(x, y) {
    Assert.assertTrue(x < y);
    Assert.assertTrue(y > x);
    Assert.assertFalse(x >= y);
    Assert.assertFalse(y <= x);
}

cmpObj(LT, 1, 2);
cmpObj(LT, 1, "2");
cmpObj(LT, "1", 2);
cmpObj(LT, "a", "b");
cmpObj(LT, -Infinity, 0);
cmpObj(LT, 0, Infinity);
cmpObj(LT, -Infinity, Infinity);
cmpObj(INCOMPARABLE, 1, NaN);
cmpObj(INCOMPARABLE, NaN, NaN);
cmpObj(INCOMPARABLE, "boo", NaN);
cmpObj(INCOMPARABLE, 1, "boo"); // boo number value will be NaN

// Test that a comparison call site can deoptimize from (int, int) to (object, object)
(function(){
    var x = [1,  2,  "a"];
    var y = [2, "3", "b"];
    for(var i = 0; i < 3; ++i) {
        Assert.assertTrue(x[i] < y[i]);
    }
})();