OLD | NEW |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 part of ssa; | 5 part of ssa; |
6 | 6 |
7 /** | 7 /** |
8 * [InvokeDynamicSpecializer] and its subclasses are helpers to | 8 * [InvokeDynamicSpecializer] and its subclasses are helpers to |
9 * optimize intercepted dynamic calls. It knows what input types | 9 * optimize intercepted dynamic calls. It knows what input types |
10 * would be beneficial for performance, and how to change a invoke | 10 * would be beneficial for performance, and how to change a invoke |
(...skipping 117 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
128 UnaryOperation operation(ConstantSystem constantSystem) { | 128 UnaryOperation operation(ConstantSystem constantSystem) { |
129 return constantSystem.bitNot; | 129 return constantSystem.bitNot; |
130 } | 130 } |
131 | 131 |
132 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, | 132 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
133 Compiler compiler) { | 133 Compiler compiler) { |
134 // All bitwise operations on primitive types either produce an | 134 // All bitwise operations on primitive types either produce an |
135 // integer or throw an error. | 135 // integer or throw an error. |
136 JavaScriptBackend backend = compiler.backend; | 136 JavaScriptBackend backend = compiler.backend; |
137 if (instruction.inputs[1].isPrimitiveOrNull(compiler)) { | 137 if (instruction.inputs[1].isPrimitiveOrNull(compiler)) { |
138 return backend.intType; | 138 return backend.uint32Type; |
139 } | 139 } |
140 return super.computeTypeFromInputTypes(instruction, compiler); | 140 return super.computeTypeFromInputTypes(instruction, compiler); |
141 } | 141 } |
142 | 142 |
143 HInstruction tryConvertToBuiltin(HInvokeDynamic instruction, | 143 HInstruction tryConvertToBuiltin(HInvokeDynamic instruction, |
144 Compiler compiler) { | 144 Compiler compiler) { |
145 JavaScriptBackend backend = compiler.backend; | 145 JavaScriptBackend backend = compiler.backend; |
146 HInstruction input = instruction.inputs[1]; | 146 HInstruction input = instruction.inputs[1]; |
147 if (input.isNumber(compiler)) { | 147 if (input.isNumber(compiler)) { |
148 return new HBitNot(input, instruction.selector, backend.intType); | 148 return new HBitNot(input, instruction.selector, |
| 149 computeTypeFromInputTypes(instruction, compiler)); |
149 } | 150 } |
150 return null; | 151 return null; |
151 } | 152 } |
152 } | 153 } |
153 | 154 |
154 class UnaryNegateSpecializer extends InvokeDynamicSpecializer { | 155 class UnaryNegateSpecializer extends InvokeDynamicSpecializer { |
155 const UnaryNegateSpecializer(); | 156 const UnaryNegateSpecializer(); |
156 | 157 |
157 UnaryOperation operation(ConstantSystem constantSystem) { | 158 UnaryOperation operation(ConstantSystem constantSystem) { |
158 return constantSystem.negate; | 159 return constantSystem.negate; |
(...skipping 165 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
324 abstract class BinaryBitOpSpecializer extends BinaryArithmeticSpecializer { | 325 abstract class BinaryBitOpSpecializer extends BinaryArithmeticSpecializer { |
325 const BinaryBitOpSpecializer(); | 326 const BinaryBitOpSpecializer(); |
326 | 327 |
327 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, | 328 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
328 Compiler compiler) { | 329 Compiler compiler) { |
329 // All bitwise operations on primitive types either produce an | 330 // All bitwise operations on primitive types either produce an |
330 // integer or throw an error. | 331 // integer or throw an error. |
331 HInstruction left = instruction.inputs[1]; | 332 HInstruction left = instruction.inputs[1]; |
332 JavaScriptBackend backend = compiler.backend; | 333 JavaScriptBackend backend = compiler.backend; |
333 if (left.isPrimitiveOrNull(compiler)) { | 334 if (left.isPrimitiveOrNull(compiler)) { |
334 return backend.intType; | 335 return backend.uint32Type; |
335 } | 336 } |
336 return super.computeTypeFromInputTypes(instruction, compiler); | 337 return super.computeTypeFromInputTypes(instruction, compiler); |
337 } | 338 } |
338 | 339 |
339 bool argumentLessThan32(HInstruction instruction) { | 340 bool argumentLessThan32(HInstruction instruction) { |
340 if (!instruction.isConstantInteger()) return false; | 341 if (!instruction.isConstantInteger()) return false; |
341 HConstant rightConstant = instruction; | 342 HConstant rightConstant = instruction; |
342 IntConstant intConstant = rightConstant.constant; | 343 IntConstant intConstant = rightConstant.constant; |
343 int count = intConstant.value; | 344 int count = intConstant.value; |
344 return count >= 0 && count <= 31; | 345 return count >= 0 && count <= 31; |
345 } | 346 } |
346 | 347 |
347 bool isPositive(HInstruction instruction) { | 348 bool isPositive(HInstruction instruction, Compiler compiler) { |
348 // TODO: We should use the value range analysis. Currently, ranges | 349 // TODO: We should use the value range analysis. Currently, ranges |
349 // are discarded just after the analysis. | 350 // are discarded just after the analysis. |
350 return instruction is HBinaryBitOp; | 351 return instruction.isUInt32(compiler); |
351 } | 352 } |
352 } | 353 } |
353 | 354 |
354 class ShiftLeftSpecializer extends BinaryBitOpSpecializer { | 355 class ShiftLeftSpecializer extends BinaryBitOpSpecializer { |
355 const ShiftLeftSpecializer(); | 356 const ShiftLeftSpecializer(); |
356 | 357 |
357 BinaryOperation operation(ConstantSystem constantSystem) { | 358 BinaryOperation operation(ConstantSystem constantSystem) { |
358 return constantSystem.shiftLeft; | 359 return constantSystem.shiftLeft; |
359 } | 360 } |
360 | 361 |
(...skipping 11 matching lines...) Expand all Loading... |
372 clearAllSideEffects(instruction); | 373 clearAllSideEffects(instruction); |
373 } | 374 } |
374 return null; | 375 return null; |
375 } | 376 } |
376 | 377 |
377 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 378 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
378 Compiler compiler) { | 379 Compiler compiler) { |
379 JavaScriptBackend backend = compiler.backend; | 380 JavaScriptBackend backend = compiler.backend; |
380 return new HShiftLeft( | 381 return new HShiftLeft( |
381 instruction.inputs[1], instruction.inputs[2], | 382 instruction.inputs[1], instruction.inputs[2], |
382 instruction.selector, backend.intType); | 383 instruction.selector, computeTypeFromInputTypes(instruction, compiler)); |
383 } | 384 } |
384 } | 385 } |
385 | 386 |
386 class ShiftRightSpecializer extends BinaryBitOpSpecializer { | 387 class ShiftRightSpecializer extends BinaryBitOpSpecializer { |
387 const ShiftRightSpecializer(); | 388 const ShiftRightSpecializer(); |
388 | 389 |
| 390 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
| 391 Compiler compiler) { |
| 392 HInstruction left = instruction.inputs[1]; |
| 393 HInstruction right = instruction.inputs[2]; |
| 394 JavaScriptBackend backend = compiler.backend; |
| 395 if (left.isUInt32(compiler)) return left.instructionType; |
| 396 return super.computeTypeFromInputTypes(instruction, compiler); |
| 397 } |
| 398 |
389 HInstruction tryConvertToBuiltin(HInvokeDynamic instruction, | 399 HInstruction tryConvertToBuiltin(HInvokeDynamic instruction, |
390 Compiler compiler) { | 400 Compiler compiler) { |
391 HInstruction left = instruction.inputs[1]; | 401 HInstruction left = instruction.inputs[1]; |
392 HInstruction right = instruction.inputs[2]; | 402 HInstruction right = instruction.inputs[2]; |
393 if (left.isNumber(compiler)) { | 403 if (left.isNumber(compiler)) { |
394 if (argumentLessThan32(right) && isPositive(left)) { | 404 if (argumentLessThan32(right) && isPositive(left, compiler)) { |
395 return newBuiltinVariant(instruction, compiler); | 405 return newBuiltinVariant(instruction, compiler); |
396 } | 406 } |
397 // Even if there is no builtin equivalent instruction, we know | 407 // Even if there is no builtin equivalent instruction, we know |
398 // the instruction does not have any side effect, and that it | 408 // the instruction does not have any side effect, and that it |
399 // can be GVN'ed. | 409 // can be GVN'ed. |
400 clearAllSideEffects(instruction); | 410 clearAllSideEffects(instruction); |
401 } | 411 } |
402 return null; | 412 return null; |
403 } | 413 } |
404 | 414 |
405 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 415 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
406 Compiler compiler) { | 416 Compiler compiler) { |
407 JavaScriptBackend backend = compiler.backend; | 417 JavaScriptBackend backend = compiler.backend; |
408 return new HShiftRight( | 418 return new HShiftRight( |
409 instruction.inputs[1], instruction.inputs[2], | 419 instruction.inputs[1], instruction.inputs[2], |
410 instruction.selector, backend.intType); | 420 instruction.selector, computeTypeFromInputTypes(instruction, compiler)); |
411 } | 421 } |
412 | 422 |
413 BinaryOperation operation(ConstantSystem constantSystem) { | 423 BinaryOperation operation(ConstantSystem constantSystem) { |
414 return constantSystem.shiftRight; | 424 return constantSystem.shiftRight; |
415 } | 425 } |
416 } | 426 } |
417 | 427 |
418 class BitOrSpecializer extends BinaryBitOpSpecializer { | 428 class BitOrSpecializer extends BinaryBitOpSpecializer { |
419 const BitOrSpecializer(); | 429 const BitOrSpecializer(); |
420 | 430 |
421 BinaryOperation operation(ConstantSystem constantSystem) { | 431 BinaryOperation operation(ConstantSystem constantSystem) { |
422 return constantSystem.bitOr; | 432 return constantSystem.bitOr; |
423 } | 433 } |
424 | 434 |
| 435 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
| 436 Compiler compiler) { |
| 437 HInstruction left = instruction.inputs[1]; |
| 438 HInstruction right = instruction.inputs[2]; |
| 439 JavaScriptBackend backend = compiler.backend; |
| 440 if (left.isUInt31(compiler) && right.isUInt31(compiler)) { |
| 441 return backend.uint31Type; |
| 442 } |
| 443 return super.computeTypeFromInputTypes(instruction, compiler); |
| 444 } |
| 445 |
425 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 446 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
426 Compiler compiler) { | 447 Compiler compiler) { |
427 JavaScriptBackend backend = compiler.backend; | 448 JavaScriptBackend backend = compiler.backend; |
428 return new HBitOr( | 449 return new HBitOr( |
429 instruction.inputs[1], instruction.inputs[2], | 450 instruction.inputs[1], instruction.inputs[2], |
430 instruction.selector, backend.intType); | 451 instruction.selector, computeTypeFromInputTypes(instruction, compiler)); |
431 } | 452 } |
432 } | 453 } |
433 | 454 |
434 class BitAndSpecializer extends BinaryBitOpSpecializer { | 455 class BitAndSpecializer extends BinaryBitOpSpecializer { |
435 const BitAndSpecializer(); | 456 const BitAndSpecializer(); |
436 | 457 |
437 BinaryOperation operation(ConstantSystem constantSystem) { | 458 BinaryOperation operation(ConstantSystem constantSystem) { |
438 return constantSystem.bitAnd; | 459 return constantSystem.bitAnd; |
439 } | 460 } |
440 | 461 |
| 462 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
| 463 Compiler compiler) { |
| 464 HInstruction left = instruction.inputs[1]; |
| 465 HInstruction right = instruction.inputs[2]; |
| 466 JavaScriptBackend backend = compiler.backend; |
| 467 if (left.isUInt31(compiler) || right.isUInt31(compiler)) { |
| 468 return backend.uint31Type; |
| 469 } |
| 470 return super.computeTypeFromInputTypes(instruction, compiler); |
| 471 } |
| 472 |
441 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 473 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
442 Compiler compiler) { | 474 Compiler compiler) { |
443 JavaScriptBackend backend = compiler.backend; | 475 JavaScriptBackend backend = compiler.backend; |
444 return new HBitAnd( | 476 return new HBitAnd( |
445 instruction.inputs[1], instruction.inputs[2], | 477 instruction.inputs[1], instruction.inputs[2], |
446 instruction.selector, backend.intType); | 478 instruction.selector, computeTypeFromInputTypes(instruction, compiler)); |
447 } | 479 } |
448 } | 480 } |
449 | 481 |
450 class BitXorSpecializer extends BinaryBitOpSpecializer { | 482 class BitXorSpecializer extends BinaryBitOpSpecializer { |
451 const BitXorSpecializer(); | 483 const BitXorSpecializer(); |
452 | 484 |
453 BinaryOperation operation(ConstantSystem constantSystem) { | 485 BinaryOperation operation(ConstantSystem constantSystem) { |
454 return constantSystem.bitXor; | 486 return constantSystem.bitXor; |
455 } | 487 } |
456 | 488 |
| 489 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
| 490 Compiler compiler) { |
| 491 HInstruction left = instruction.inputs[1]; |
| 492 HInstruction right = instruction.inputs[2]; |
| 493 JavaScriptBackend backend = compiler.backend; |
| 494 if (left.isUInt31(compiler) && right.isUInt31(compiler)) { |
| 495 return backend.uint31Type; |
| 496 } |
| 497 return super.computeTypeFromInputTypes(instruction, compiler); |
| 498 } |
| 499 |
457 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 500 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
458 Compiler compiler) { | 501 Compiler compiler) { |
459 JavaScriptBackend backend = compiler.backend; | 502 JavaScriptBackend backend = compiler.backend; |
460 return new HBitXor( | 503 return new HBitXor( |
461 instruction.inputs[1], instruction.inputs[2], | 504 instruction.inputs[1], instruction.inputs[2], |
462 instruction.selector, backend.intType); | 505 instruction.selector, computeTypeFromInputTypes(instruction, compiler)); |
463 } | 506 } |
464 } | 507 } |
465 | 508 |
466 abstract class RelationalSpecializer extends InvokeDynamicSpecializer { | 509 abstract class RelationalSpecializer extends InvokeDynamicSpecializer { |
467 const RelationalSpecializer(); | 510 const RelationalSpecializer(); |
468 | 511 |
469 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, | 512 TypeMask computeTypeFromInputTypes(HInvokeDynamic instruction, |
470 Compiler compiler) { | 513 Compiler compiler) { |
471 JavaScriptBackend backend = compiler.backend; | 514 JavaScriptBackend backend = compiler.backend; |
472 if (instruction.inputs[1].isPrimitiveOrNull(compiler)) { | 515 if (instruction.inputs[1].isPrimitiveOrNull(compiler)) { |
(...skipping 109 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
582 } | 625 } |
583 | 626 |
584 HInstruction newBuiltinVariant(HInvokeDynamic instruction, | 627 HInstruction newBuiltinVariant(HInvokeDynamic instruction, |
585 Compiler compiler) { | 628 Compiler compiler) { |
586 JavaScriptBackend backend = compiler.backend; | 629 JavaScriptBackend backend = compiler.backend; |
587 return new HLessEqual( | 630 return new HLessEqual( |
588 instruction.inputs[1], instruction.inputs[2], | 631 instruction.inputs[1], instruction.inputs[2], |
589 instruction.selector, backend.boolType); | 632 instruction.selector, backend.boolType); |
590 } | 633 } |
591 } | 634 } |
OLD | NEW |