b436635ac4
When == is invoked on a Symbol or SIMD vector and an object, the object should be converted to a primitive with ToPrimitive and then compared again. This means, for example, that for a Symbol or SIMD vector s, s == Object(s). This patch makes that change in the implementation of ==. Only the runtime function needed to be changed, as the code stubs and compiler specializations don't operate on Symbols or SIMD vectors, and on these types, a fallback to the runtime function is always used. BUG=v8:3593 LOG=Y R=adamk Review URL: https://codereview.chromium.org/1421413002 Cr-Commit-Position: refs/heads/master@{#31614}
621 lines
17 KiB
JavaScript
621 lines
17 KiB
JavaScript
// Copyright 2015 the V8 project authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
// Flags: --harmony-simd --harmony-tostring --harmony-reflect
|
|
// Flags: --allow-natives-syntax --expose-natives-as natives --noalways-opt
|
|
|
|
function lanesForType(typeName) {
|
|
// The lane count follows the first 'x' in the type name, which begins with
|
|
// 'float', 'int', or 'bool'.
|
|
return Number.parseInt(typeName.substr(typeName.indexOf('x') + 1));
|
|
}
|
|
|
|
|
|
// Creates an instance that has been zeroed, so it can be used for equality
|
|
// testing.
|
|
function createInstance(type) {
|
|
// Provide enough parameters for the longest type (currently 16). It's
|
|
// important that instances be consistent to better test that different SIMD
|
|
// types can't be compared and are never equal or the same in any sense.
|
|
return SIMD[type](0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15);
|
|
}
|
|
|
|
|
|
function isValidSimdString(string, value, type, lanes) {
|
|
var simdFn = SIMD[type],
|
|
parseFn =
|
|
type.indexOf('Float') === 0 ? Number.parseFloat : Number.parseInt,
|
|
indexOfOpenParen = string.indexOf('(');
|
|
// Check prefix (e.g. SIMD.Float32x4.)
|
|
if (string.substr(0, indexOfOpenParen) !== 'SIMD.' + type)
|
|
return false;
|
|
// Remove type name (e.g. SIMD.Float32x4) and open parenthesis.
|
|
string = string.substr(indexOfOpenParen + 1);
|
|
var laneStrings = string.split(',');
|
|
if (laneStrings.length !== lanes)
|
|
return false;
|
|
for (var i = 0; i < lanes; i++) {
|
|
var fromString = parseFn(laneStrings[i]),
|
|
fromValue = simdFn.extractLane(value, i);
|
|
if (Math.abs(fromString - fromValue) > Number.EPSILON)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
var simdTypeNames = ['Float32x4', 'Int32x4', 'Uint32x4', 'Bool32x4',
|
|
'Int16x8', 'Uint16x8', 'Bool16x8',
|
|
'Int8x16', 'Uint8x16', 'Bool8x16'];
|
|
|
|
var nonSimdValues = [347, 1.275, NaN, "string", null, undefined, {},
|
|
function() {}];
|
|
|
|
function checkTypeMatrix(type, fn) {
|
|
// Check against non-SIMD types.
|
|
nonSimdValues.forEach(fn);
|
|
// Check against SIMD values of a different type.
|
|
for (var i = 0; i < simdTypeNames.length; i++) {
|
|
var otherType = simdTypeNames[i];
|
|
if (type != otherType) fn(createInstance(otherType));
|
|
}
|
|
}
|
|
|
|
|
|
// Test different forms of constructor calls.
|
|
function TestConstructor(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertFalse(Object === simdFn.prototype.constructor)
|
|
assertFalse(simdFn === Object.prototype.constructor)
|
|
assertSame(simdFn, simdFn.prototype.constructor)
|
|
|
|
assertSame(simdFn, instance.__proto__.constructor)
|
|
assertSame(simdFn, Object(instance).__proto__.constructor)
|
|
assertSame(simdFn.prototype, instance.__proto__)
|
|
assertSame(simdFn.prototype, Object(instance).__proto__)
|
|
}
|
|
|
|
|
|
function TestType(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
var typeofString = type.charAt(0).toLowerCase() + type.slice(1);
|
|
|
|
assertEquals(typeofString, typeof instance)
|
|
assertTrue(typeof instance === typeofString)
|
|
assertTrue(typeof Object(instance) === 'object')
|
|
assertEquals(null, %_ClassOf(instance))
|
|
assertEquals(type, %_ClassOf(Object(instance)))
|
|
}
|
|
|
|
|
|
function TestPrototype(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertSame(Object.prototype, simdFn.prototype.__proto__)
|
|
assertSame(simdFn.prototype, instance.__proto__)
|
|
assertSame(simdFn.prototype, Object(instance).__proto__)
|
|
}
|
|
|
|
|
|
function TestValueOf(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertTrue(instance === Object(instance).valueOf())
|
|
assertTrue(instance === instance.valueOf())
|
|
assertTrue(simdFn.prototype.valueOf.call(Object(instance)) === instance)
|
|
assertTrue(simdFn.prototype.valueOf.call(instance) === instance)
|
|
}
|
|
|
|
|
|
function TestGet(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertEquals(undefined, instance.a)
|
|
assertEquals(undefined, instance["a" + "b"])
|
|
assertEquals(undefined, instance["" + "1"])
|
|
assertEquals(undefined, instance[42])
|
|
}
|
|
|
|
|
|
function TestToBoolean(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertTrue(Boolean(Object(instance)))
|
|
assertFalse(!Object(instance))
|
|
assertTrue(Boolean(instance).valueOf())
|
|
assertFalse(!instance)
|
|
assertTrue(!!instance)
|
|
assertTrue(instance && true)
|
|
assertFalse(!instance && false)
|
|
assertTrue(!instance || true)
|
|
assertEquals(1, instance ? 1 : 2)
|
|
assertEquals(2, !instance ? 1 : 2)
|
|
if (!instance) assertUnreachable();
|
|
if (instance) {} else assertUnreachable();
|
|
}
|
|
|
|
|
|
function TestToString(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertEquals(instance.toString(), String(instance))
|
|
assertTrue(isValidSimdString(instance.toString(), instance, type, lanes))
|
|
assertTrue(
|
|
isValidSimdString(Object(instance).toString(), instance, type, lanes))
|
|
assertTrue(isValidSimdString(
|
|
simdFn.prototype.toString.call(instance), instance, type, lanes))
|
|
}
|
|
|
|
|
|
function TestToNumber(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
assertThrows(function() { Number(Object(instance)) }, TypeError)
|
|
assertThrows(function() { +Object(instance) }, TypeError)
|
|
assertThrows(function() { Number(instance) }, TypeError)
|
|
assertThrows(function() { instance + 0 }, TypeError)
|
|
}
|
|
|
|
|
|
function TestCoercions(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
// Test that setting a lane to value 'a' results in a lane with value 'b'.
|
|
function test(a, b) {
|
|
for (var i = 0; i < lanes; i++) {
|
|
var ainstance = simdFn.replaceLane(instance, i, a);
|
|
var lane_value = simdFn.extractLane(ainstance, i);
|
|
assertSame(b, lane_value);
|
|
}
|
|
}
|
|
|
|
switch (type) {
|
|
case 'Float32x4':
|
|
test(0, 0);
|
|
test(-0, -0);
|
|
test(NaN, NaN);
|
|
test(null, 0);
|
|
test(undefined, NaN);
|
|
test("5.25", 5.25);
|
|
test(Number.MAX_VALUE, Infinity);
|
|
test(-Number.MAX_VALUE, -Infinity);
|
|
test(Number.MIN_VALUE, 0);
|
|
break;
|
|
case 'Int32x4':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, -1);
|
|
test(-1.6, -1);
|
|
test(2147483647, 2147483647);
|
|
test(2147483648, -2147483648);
|
|
test(2147483649, -2147483647);
|
|
test(4294967295, -1);
|
|
test(4294967296, 0);
|
|
test(4294967297, 1);
|
|
break;
|
|
case 'Uint32x4':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, 4294967295);
|
|
test(-1.6, 4294967295);
|
|
test(4294967295, 4294967295);
|
|
test(4294967296, 0);
|
|
test(4294967297, 1);
|
|
break;
|
|
case 'Int16x8':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, -1);
|
|
test(-1.6, -1);
|
|
test(32767, 32767);
|
|
test(32768, -32768);
|
|
test(32769, -32767);
|
|
test(65535, -1);
|
|
test(65536, 0);
|
|
test(65537, 1);
|
|
break;
|
|
case 'Uint16x8':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, 65535);
|
|
test(-1.6, 65535);
|
|
test(65535, 65535);
|
|
test(65536, 0);
|
|
test(65537, 1);
|
|
break;
|
|
case 'Int8x16':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, -1);
|
|
test(-1.6, -1);
|
|
test(127, 127);
|
|
test(128, -128);
|
|
test(129, -127);
|
|
test(255, -1);
|
|
test(256, 0);
|
|
test(257, 1);
|
|
break;
|
|
case 'Uint8x16':
|
|
test(Infinity, 0);
|
|
test(-Infinity, 0);
|
|
test(NaN, 0);
|
|
test(0, 0);
|
|
test(-0, 0);
|
|
test(Number.MIN_VALUE, 0);
|
|
test(-Number.MIN_VALUE, 0);
|
|
test(0.1, 0);
|
|
test(-0.1, 0);
|
|
test(1, 1);
|
|
test(1.1, 1);
|
|
test(-1, 255);
|
|
test(-1.6, 255);
|
|
test(255, 255);
|
|
test(256, 0);
|
|
test(257, 1);
|
|
break;
|
|
case 'Bool32x4':
|
|
case 'Bool16x8':
|
|
case 'Bool8x16':
|
|
test(true, true);
|
|
test(false, false);
|
|
test(0, false);
|
|
test(1, true);
|
|
test(0.1, true);
|
|
test(NaN, false);
|
|
test(null, false);
|
|
test("", false);
|
|
test("false", true);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
function TestEquality(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
// Every SIMD value should equal itself, and non-strictly equal its wrapper.
|
|
assertSame(instance, instance)
|
|
assertEquals(instance, instance)
|
|
assertTrue(Object.is(instance, instance))
|
|
assertTrue(instance === instance)
|
|
assertTrue(instance == instance)
|
|
assertFalse(instance === Object(instance))
|
|
assertFalse(Object(instance) === instance)
|
|
assertTrue(instance == Object(instance))
|
|
assertTrue(Object(instance) == instance)
|
|
assertTrue(instance === instance.valueOf())
|
|
assertTrue(instance.valueOf() === instance)
|
|
assertTrue(instance == instance.valueOf())
|
|
assertTrue(instance.valueOf() == instance)
|
|
assertFalse(Object(instance) === Object(instance))
|
|
assertEquals(Object(instance).valueOf(), Object(instance).valueOf())
|
|
|
|
function notEqual(other) {
|
|
assertFalse(instance === other)
|
|
assertFalse(other === instance)
|
|
assertFalse(instance == other)
|
|
assertFalse(other == instance)
|
|
}
|
|
|
|
// SIMD values should not be equal to instances of different types.
|
|
checkTypeMatrix(type, function(other) {
|
|
assertFalse(instance === other)
|
|
assertFalse(other === instance)
|
|
assertFalse(instance == other)
|
|
assertFalse(other == instance)
|
|
});
|
|
|
|
// Test that f(a, b) is the same as f(SIMD(a), SIMD(b)) for equality and
|
|
// strict equality, at every lane.
|
|
function test(a, b) {
|
|
for (var i = 0; i < lanes; i++) {
|
|
var aval = simdFn.replaceLane(instance, i, a);
|
|
var bval = simdFn.replaceLane(instance, i, b);
|
|
assertSame(a == b, aval == bval);
|
|
assertSame(a === b, aval === bval);
|
|
}
|
|
}
|
|
|
|
switch (type) {
|
|
case 'Float32x4':
|
|
test(1, 2.5);
|
|
test(1, 1);
|
|
test(0, 0);
|
|
test(-0, +0);
|
|
test(+0, -0);
|
|
test(-0, -0);
|
|
test(0, NaN);
|
|
test(NaN, NaN);
|
|
break;
|
|
case 'Int32x4':
|
|
case 'Uint32x4':
|
|
case 'Int16x8':
|
|
case 'Uint16x8':
|
|
case 'Int8x16':
|
|
case 'Uint8x16':
|
|
test(1, 2);
|
|
test(1, 1);
|
|
test(1, -1);
|
|
break;
|
|
case 'Bool32x4':
|
|
case 'Bool16x8':
|
|
case 'Bool8x16':
|
|
test(true, false);
|
|
test(false, true);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
function TestSameValue(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
var sameValue = Object.is
|
|
var sameValueZero = natives.ImportNow("SameValueZero");
|
|
|
|
// SIMD values should not be the same as instances of different types.
|
|
checkTypeMatrix(type, function(other) {
|
|
assertFalse(sameValue(instance, other));
|
|
assertFalse(sameValueZero(instance, other));
|
|
});
|
|
|
|
// Test that f(a, b) is the same as f(SIMD(a), SIMD(b)) for sameValue and
|
|
// sameValueZero, at every lane.
|
|
function test(a, b) {
|
|
for (var i = 0; i < lanes; i++) {
|
|
var aval = simdFn.replaceLane(instance, i, a);
|
|
var bval = simdFn.replaceLane(instance, i, b);
|
|
assertSame(sameValue(a, b), sameValue(aval, bval));
|
|
assertSame(sameValueZero(a, b), sameValueZero(aval, bval));
|
|
}
|
|
}
|
|
|
|
switch (type) {
|
|
case 'Float32x4':
|
|
test(1, 2.5);
|
|
test(1, 1);
|
|
test(0, 0);
|
|
test(-0, +0);
|
|
test(+0, -0);
|
|
test(-0, -0);
|
|
test(0, NaN);
|
|
test(NaN, NaN);
|
|
break;
|
|
case 'Int32x4':
|
|
case 'Uint32x4':
|
|
case 'Int16x8':
|
|
case 'Uint16x8':
|
|
case 'Int8x16':
|
|
case 'Uint8x16':
|
|
test(1, 2);
|
|
test(1, 1);
|
|
test(1, -1);
|
|
break;
|
|
case 'Bool32x4':
|
|
case 'Bool16x8':
|
|
case 'Bool8x16':
|
|
test(true, false);
|
|
test(false, true);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
function TestComparison(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var a = createInstance(type), b = createInstance(type);
|
|
|
|
function compare(other) {
|
|
var throwFuncs = [
|
|
function lt() { a < b; },
|
|
function gt() { a > b; },
|
|
function le() { a <= b; },
|
|
function ge() { a >= b; },
|
|
function lt_same() { a < a; },
|
|
function gt_same() { a > a; },
|
|
function le_same() { a <= a; },
|
|
function ge_same() { a >= a; },
|
|
];
|
|
|
|
for (var f of throwFuncs) {
|
|
assertThrows(f, TypeError);
|
|
%OptimizeFunctionOnNextCall(f);
|
|
assertThrows(f, TypeError);
|
|
assertThrows(f, TypeError);
|
|
}
|
|
}
|
|
|
|
// Test comparison against the same SIMD type.
|
|
compare(b);
|
|
// Test comparison against other types.
|
|
checkTypeMatrix(type, compare);
|
|
}
|
|
|
|
|
|
// Test SIMD value wrapping/boxing over non-builtins.
|
|
function TestCall(type, lanes) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
simdFn.prototype.getThisProto = function () {
|
|
return Object.getPrototypeOf(this);
|
|
}
|
|
assertTrue(instance.getThisProto() === simdFn.prototype)
|
|
}
|
|
|
|
|
|
function TestAsSetKey(type, lanes, set) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
function test(set, key) {
|
|
assertFalse(set.has(key));
|
|
assertFalse(set.delete(key));
|
|
if (!(set instanceof WeakSet)) {
|
|
assertSame(set, set.add(key));
|
|
assertTrue(set.has(key));
|
|
assertTrue(set.delete(key));
|
|
} else {
|
|
// SIMD values can't be used as keys in WeakSets.
|
|
assertThrows(function() { set.add(key) });
|
|
}
|
|
assertFalse(set.has(key));
|
|
assertFalse(set.delete(key));
|
|
assertFalse(set.has(key));
|
|
}
|
|
|
|
test(set, instance);
|
|
}
|
|
|
|
|
|
function TestAsMapKey(type, lanes, map) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
function test(map, key, value) {
|
|
assertFalse(map.has(key));
|
|
assertSame(undefined, map.get(key));
|
|
assertFalse(map.delete(key));
|
|
if (!(map instanceof WeakMap)) {
|
|
assertSame(map, map.set(key, value));
|
|
assertSame(value, map.get(key));
|
|
assertTrue(map.has(key));
|
|
assertTrue(map.delete(key));
|
|
} else {
|
|
// SIMD values can't be used as keys in WeakMaps.
|
|
assertThrows(function() { map.set(key, value) });
|
|
}
|
|
assertFalse(map.has(key));
|
|
assertSame(undefined, map.get(key));
|
|
assertFalse(map.delete(key));
|
|
assertFalse(map.has(key));
|
|
assertSame(undefined, map.get(key));
|
|
}
|
|
|
|
test(map, instance, {});
|
|
}
|
|
|
|
|
|
// Test SIMD type with Harmony reflect-apply.
|
|
function TestReflectApply(type) {
|
|
var simdFn = SIMD[type];
|
|
var instance = createInstance(type);
|
|
|
|
function returnThis() { return this; }
|
|
function returnThisStrict() { 'use strict'; return this; }
|
|
function noop() {}
|
|
function noopStrict() { 'use strict'; }
|
|
var R = void 0;
|
|
|
|
assertSame(SIMD[type].prototype,
|
|
Object.getPrototypeOf(
|
|
Reflect.apply(returnThis, instance, [])));
|
|
assertSame(instance, Reflect.apply(returnThisStrict, instance, []));
|
|
|
|
assertThrows(
|
|
function() { 'use strict'; Reflect.apply(instance); }, TypeError);
|
|
assertThrows(
|
|
function() { Reflect.apply(instance); }, TypeError);
|
|
assertThrows(
|
|
function() { Reflect.apply(noopStrict, R, instance); }, TypeError);
|
|
assertThrows(
|
|
function() { Reflect.apply(noop, R, instance); }, TypeError);
|
|
}
|
|
|
|
|
|
function TestSIMDTypes() {
|
|
for (var i = 0; i < simdTypeNames.length; ++i) {
|
|
var type = simdTypeNames[i],
|
|
lanes = lanesForType(type);
|
|
TestConstructor(type, lanes);
|
|
TestType(type, lanes);
|
|
TestPrototype(type, lanes);
|
|
TestValueOf(type, lanes);
|
|
TestGet(type, lanes);
|
|
TestToBoolean(type, lanes);
|
|
TestToString(type, lanes);
|
|
TestToNumber(type, lanes);
|
|
TestCoercions(type, lanes);
|
|
TestEquality(type, lanes);
|
|
TestSameValue(type, lanes);
|
|
TestComparison(type, lanes);
|
|
TestCall(type, lanes);
|
|
TestAsSetKey(type, lanes, new Set);
|
|
TestAsSetKey(type, lanes, new WeakSet);
|
|
TestAsMapKey(type, lanes, new Map);
|
|
TestAsMapKey(type, lanes, new WeakMap);
|
|
TestReflectApply(type);
|
|
}
|
|
}
|
|
TestSIMDTypes();
|
|
|
|
// Tests for the global SIMD object.
|
|
function TestSIMDObject() {
|
|
assertSame(typeof SIMD, 'object');
|
|
assertSame(SIMD.constructor, Object);
|
|
assertSame(Object.getPrototypeOf(SIMD), Object.prototype);
|
|
assertSame(SIMD + "", "[object SIMD]");
|
|
// The SIMD object is mutable.
|
|
SIMD.foo = "foo";
|
|
assertSame(SIMD.foo, "foo");
|
|
delete SIMD.foo;
|
|
delete SIMD.Bool8x16;
|
|
assertSame(SIMD.Bool8x16, undefined);
|
|
}
|
|
TestSIMDObject()
|