1 // Copyright 2011 the V8 project authors. All rights reserved. 2 // Redistribution and use in source and binary forms, with or without 3 // modification, are permitted provided that the following conditions are 4 // met: 5 // 6 // * Redistributions of source code must retain the above copyright 7 // notice, this list of conditions and the following disclaimer. 8 // * Redistributions in binary form must reproduce the above 9 // copyright notice, this list of conditions and the following 10 // disclaimer in the documentation and/or other materials provided 11 // with the distribution. 12 // * Neither the name of Google Inc. nor the names of its 13 // contributors may be used to endorse or promote products derived 14 // from this software without specific prior written permission. 15 // 16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 17 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 18 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 19 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 20 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 27 28 // Flags: --allow-natives-syntax --max-opt-count=100 29 30 function A() { 31 } 32 33 A.prototype.X = function (a, b, c) { 34 assertTrue(this instanceof A); 35 assertEquals(1, a); 36 assertEquals(2, b); 37 assertEquals(3, c); 38 }; 39 40 A.prototype.Y = function () { 41 this.X.apply(this, arguments); 42 }; 43 44 A.prototype.Z = function () { 45 this.Y(1,2,3); 46 }; 47 48 var a = new A(); 49 a.Z(4,5,6); 50 a.Z(4,5,6); 51 %OptimizeFunctionOnNextCall(a.Z); 52 a.Z(4,5,6); 53 A.prototype.X.apply = function (receiver, args) { 54 return Function.prototype.apply.call(this, receiver, args); 55 }; 56 a.Z(4,5,6); 57 58 59 // Ensure that HArgumentsObject is inserted in a correct place 60 // and dominates all uses. 61 function F1() { } 62 function F2() { F1.apply(this, arguments); } 63 function F3(x, y) { 64 if (x) { 65 F2(y); 66 } 67 } 68 69 function F31() { 70 return F1.apply(this, arguments); 71 } 72 73 function F4() { 74 F3(true, false); 75 return F31(1); 76 } 77 78 F4(1); 79 F4(1); 80 F4(1); 81 %OptimizeFunctionOnNextCall(F4); 82 F4(1); 83 84 85 // Test correct adapation of arguments. 86 // Strict mode prevents arguments object from shadowing parameters. 87 (function () { 88 "use strict"; 89 90 function G2() { 91 assertArrayEquals([1,2], arguments); 92 } 93 94 function G4() { 95 assertArrayEquals([1,2,3,4], arguments); 96 } 97 98 function adapt2to4(a, b, c, d) { 99 G2.apply(this, arguments); 100 } 101 102 function adapt4to2(a, b) { 103 G4.apply(this, arguments); 104 } 105 106 function test_adaptation() { 107 adapt2to4(1, 2); 108 adapt4to2(1, 2, 3, 4); 109 } 110 111 test_adaptation(); 112 test_adaptation(); 113 %OptimizeFunctionOnNextCall(test_adaptation); 114 test_adaptation(); 115 })(); 116 117 // Test arguments access from the inlined function. 118 %NeverOptimizeFunction(uninlinable); 119 function uninlinable(v) { 120 assertEquals(0, v); 121 return 0; 122 } 123 124 function toarr_inner() { 125 var a = arguments; 126 var marker = a[0]; 127 uninlinable(uninlinable(0, 0), marker.x); 128 129 var r = new Array(); 130 for (var i = a.length - 1; i >= 1; i--) { 131 r.push(a[i]); 132 } 133 134 return r; 135 } 136 137 function toarr1(marker, a, b, c) { 138 return toarr_inner(marker, a / 2, b / 2, c / 2); 139 } 140 141 function toarr2(marker, a, b, c) { 142 var x = 0; 143 return uninlinable(uninlinable(0, 0), 144 x = toarr_inner(marker, a / 2, b / 2, c / 2)), x; 145 } 146 147 function test_toarr(toarr) { 148 var marker = { x: 0 }; 149 assertArrayEquals([3, 2, 1], toarr(marker, 2, 4, 6)); 150 assertArrayEquals([3, 2, 1], toarr(marker, 2, 4, 6)); 151 %OptimizeFunctionOnNextCall(toarr); 152 assertArrayEquals([3, 2, 1], toarr(marker, 2, 4, 6)); 153 delete marker.x; 154 assertArrayEquals([3, 2, 1], toarr(marker, 2, 4, 6)); 155 } 156 157 test_toarr(toarr1); 158 test_toarr(toarr2); 159 160 161 // Test that arguments access from inlined function uses correct values. 162 (function () { 163 function inner(x, y) { 164 "use strict"; 165 x = 10; 166 y = 20; 167 for (var i = 0; i < 1; i++) { 168 for (var j = 1; j <= arguments.length; j++) { 169 return arguments[arguments.length - j]; 170 } 171 } 172 } 173 174 function outer(x, y) { 175 return inner(x, y); 176 } 177 178 %OptimizeFunctionOnNextCall(outer); 179 %OptimizeFunctionOnNextCall(inner); 180 assertEquals(2, outer(1, 2)); 181 })(); 182 183 184 (function () { 185 function inner(x, y) { 186 "use strict"; 187 x = 10; 188 y = 20; 189 for (var i = 0; i < 1; i++) { 190 for (var j = 1; j <= arguments.length; j++) { 191 return arguments[arguments.length - j]; 192 } 193 } 194 } 195 196 function outer(x, y) { 197 return inner(x, y); 198 } 199 200 assertEquals(2, outer(1, 2)); 201 assertEquals(2, outer(1, 2)); 202 assertEquals(2, outer(1, 2)); 203 %OptimizeFunctionOnNextCall(outer); 204 assertEquals(2, outer(1, 2)); 205 })(); 206 207 208 // Test inlining and deoptimization of functions accessing and modifying 209 // the arguments object in strict mode with mismatched arguments count. 210 (function () { 211 "use strict"; 212 function test(outerCount, middleCount, innerCount) { 213 var forceDeopt = { deopt:false }; 214 function inner(x,y) { 215 x = 0; y = 0; 216 forceDeopt.deopt; 217 assertSame(innerCount, arguments.length); 218 for (var i = 0; i < arguments.length; i++) { 219 assertSame(30 + i, arguments[i]); 220 } 221 } 222 223 function middle(x,y) { 224 x = 0; y = 0; 225 if (innerCount == 1) inner(30); 226 if (innerCount == 2) inner(30, 31); 227 if (innerCount == 3) inner(30, 31, 32); 228 assertSame(middleCount, arguments.length); 229 for (var i = 0; i < arguments.length; i++) { 230 assertSame(20 + i, arguments[i]); 231 } 232 } 233 234 function outer(x,y) { 235 x = 0; y = 0; 236 if (middleCount == 1) middle(20); 237 if (middleCount == 2) middle(20, 21); 238 if (middleCount == 3) middle(20, 21, 22); 239 assertSame(outerCount, arguments.length); 240 for (var i = 0; i < arguments.length; i++) { 241 assertSame(10 + i, arguments[i]); 242 } 243 } 244 245 for (var step = 0; step < 4; step++) { 246 if (outerCount == 1) outer(10); 247 if (outerCount == 2) outer(10, 11); 248 if (outerCount == 3) outer(10, 11, 12); 249 if (step == 1) %OptimizeFunctionOnNextCall(outer); 250 if (step == 2) delete forceDeopt.deopt; 251 } 252 253 %DeoptimizeFunction(outer); 254 %DeoptimizeFunction(middle); 255 %DeoptimizeFunction(inner); 256 %ClearFunctionTypeFeedback(outer); 257 %ClearFunctionTypeFeedback(middle); 258 %ClearFunctionTypeFeedback(inner); 259 } 260 261 for (var a = 1; a <= 3; a++) { 262 for (var b = 1; b <= 3; b++) { 263 for (var c = 1; c <= 3; c++) { 264 test(a,b,c); 265 } 266 } 267 } 268 })(); 269 270 271 // Test materialization of arguments object with values in registers. 272 (function () { 273 "use strict"; 274 var forceDeopt = { deopt:false }; 275 function inner(a,b,c,d,e,f,g,h,i,j) { 276 var args = arguments; 277 forceDeopt.deopt; 278 assertSame(10, args.length); 279 assertSame(a, args[0]); 280 assertSame(b, args[1]); 281 assertSame(c, args[2]); 282 assertSame(d, args[3]); 283 assertSame(e, args[4]); 284 assertSame(f, args[5]); 285 assertSame(g, args[6]); 286 assertSame(h, args[7]); 287 assertSame(i, args[8]); 288 assertSame(j, args[9]); 289 } 290 291 var a = 0.5; 292 var b = 1.7; 293 var c = 123; 294 function outer() { 295 inner( 296 a - 0.3, // double in double register 297 b + 2.3, // integer in double register 298 c + 321, // integer in general register 299 c - 456, // integer in stack slot 300 a + 0.1, a + 0.2, a + 0.3, a + 0.4, a + 0.5, 301 a + 0.6 // double in stack slot 302 ); 303 } 304 305 outer(); 306 outer(); 307 %OptimizeFunctionOnNextCall(outer); 308 outer(); 309 delete forceDeopt.deopt; 310 outer(); 311 })(); 312