// Copyright 2013 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // Flags: --allow-natives-syntax var imul_func = Math.imul; function imul_polyfill(a, b) { var ah = a >>> 16 & 0xffff; var al = a & 0xffff; var bh = b >>> 16 & 0xffff; var bl = b & 0xffff; return al * bl + (ah * bl + al * bh << 16 >>> 0) | 0; } function TestMathImul(expected, a, b) { function imul_meth_closure(a, b) { return Math.imul(a, b); }; %PrepareFunctionForOptimization(imul_meth_closure); function imul_func_closure(a, b) { return imul_func(a, b); } // Test reference implementation. ; %PrepareFunctionForOptimization(imul_func_closure); assertEquals(expected, imul_polyfill(a, b)); // Test direct method call. assertEquals(expected, Math.imul(a, b)); // Test direct function call. assertEquals(expected, imul_func(a, b)); // Test optimized method call inside closure. assertEquals(expected, imul_meth_closure(a, b)); assertEquals(expected, imul_meth_closure(a, b)); %OptimizeFunctionOnNextCall(imul_meth_closure); assertEquals(expected, imul_meth_closure(a, b)); // Test optimized function call inside closure. assertEquals(expected, imul_func_closure(a, b)); assertEquals(expected, imul_func_closure(a, b)); %OptimizeFunctionOnNextCall(imul_func_closure); assertEquals(expected, imul_func_closure(a, b)); // Deoptimize closures and forget type feedback. %DeoptimizeFunction(imul_meth_closure); %DeoptimizeFunction(imul_func_closure); %ClearFunctionFeedback(imul_meth_closure); %ClearFunctionFeedback(imul_func_closure); } TestMathImul(8, 2, 4); TestMathImul(-8, -1, 8); TestMathImul(4, -2, -2); TestMathImul(-5, 0xffffffff, 5); TestMathImul(-10, 0xfffffffe, 5); TestMathImul(0, false, 7); TestMathImul(0, 7, false); TestMathImul(0, false, false); TestMathImul(7, true, 7); TestMathImul(7, 7, true); TestMathImul(1, true, true); TestMathImul(0, false, true); TestMathImul(0, true, false); TestMathImul(0, undefined, 7); TestMathImul(0, 7, undefined); TestMathImul(0, undefined, undefined); TestMathImul(0, -0, 7); TestMathImul(0, 7, -0); TestMathImul(0, 0.1, 7); TestMathImul(0, 7, 0.1); TestMathImul(0, 0.9, 7); TestMathImul(0, 7, 0.9); TestMathImul(7, 1.1, 7); TestMathImul(7, 7, 1.1); TestMathImul(7, 1.9, 7); TestMathImul(7, 7, 1.9); TestMathImul(0, "str", 7); TestMathImul(0, 7, "str"); TestMathImul(0, {}, 7); TestMathImul(0, 7, {}); TestMathImul(0, [], 7); TestMathImul(0, 7, []); // 2^30 is a smi boundary on arm and ia32. var two_30 = 1 << 30; TestMathImul(-two_30, two_30, 7); TestMathImul(-two_30, 7, two_30); TestMathImul(0, two_30, two_30); TestMathImul(two_30, -two_30, 7); TestMathImul(two_30, 7, -two_30); TestMathImul(0, -two_30, -two_30); // 2^31 is a smi boundary on x64. var two_31 = 2 * two_30; TestMathImul(-two_31, two_31, 7); TestMathImul(-two_31, 7, two_31); TestMathImul(0, two_31, two_31); TestMathImul(-two_31, -two_31, 7); TestMathImul(-two_31, 7, -two_31); TestMathImul(0, -two_31, -two_31); // 2^31 - 1 is the largest int32 value. var max_val = two_31 - 1; TestMathImul(two_31 - 7, max_val, 7); TestMathImul(two_31 - 7, 7, max_val); TestMathImul(1, max_val, max_val); // 2^16 is a boundary value that overflows when squared. var two_16 = 1 << 16; TestMathImul(0, two_16, two_16); TestMathImul(-two_16, two_16 - 1, two_16); TestMathImul(-two_16, two_16, two_16 - 1); TestMathImul(-2 * two_16 + 1, two_16 - 1, two_16 - 1);