| OLD | NEW |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, 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 import '../common.dart'; | 5 import '../common.dart'; |
| 6 import '../common/codegen.dart' show CodegenRegistry, CodegenWorkItem; | 6 import '../common/codegen.dart' show CodegenRegistry, CodegenWorkItem; |
| 7 import '../common/tasks.dart' show CompilerTask; | 7 import '../common/tasks.dart' show CompilerTask; |
| 8 import '../compiler.dart' show Compiler; | 8 import '../compiler.dart' show Compiler; |
| 9 import '../constants/constant_system.dart'; | 9 import '../constants/constant_system.dart'; |
| 10 import '../constants/values.dart'; | 10 import '../constants/values.dart'; |
| 11 import '../core_types.dart' show CoreClasses; | 11 import '../core_types.dart' show CoreClasses; |
| 12 import '../dart_types.dart'; | 12 import '../dart_types.dart'; |
| 13 import '../elements/elements.dart'; | 13 import '../elements/elements.dart'; |
| 14 import '../io/source_information.dart'; | 14 import '../io/source_information.dart'; |
| 15 import '../js/js.dart' as js; | 15 import '../js/js.dart' as js; |
| 16 import '../js_backend/backend_helpers.dart' show BackendHelpers; | 16 import '../js_backend/backend_helpers.dart' show BackendHelpers; |
| 17 import '../js_backend/js_backend.dart'; | 17 import '../js_backend/js_backend.dart'; |
| 18 import '../js_emitter/js_emitter.dart' show NativeEmitter; | 18 import '../js_emitter/js_emitter.dart' show NativeEmitter; |
| 19 import '../native/native.dart' as native; | 19 import '../native/native.dart' as native; |
| 20 import '../types/types.dart'; | 20 import '../types/types.dart'; |
| 21 import '../universe/call_structure.dart' show CallStructure; | 21 import '../universe/call_structure.dart' show CallStructure; |
| 22 import '../universe/selector.dart' show Selector; | 22 import '../universe/selector.dart' show Selector; |
| 23 import '../universe/use.dart' show DynamicUse, StaticUse, TypeUse; | 23 import '../universe/use.dart' show DynamicUse, StaticUse, TypeUse; |
| 24 import '../util/util.dart'; | 24 import '../util/util.dart'; |
| 25 import '../world.dart' show ClassWorld; | 25 import '../world.dart' show ClosedWorld; |
| 26 import 'codegen_helpers.dart'; | 26 import 'codegen_helpers.dart'; |
| 27 import 'nodes.dart'; | 27 import 'nodes.dart'; |
| 28 import 'variable_allocator.dart'; | 28 import 'variable_allocator.dart'; |
| 29 | 29 |
| 30 class SsaCodeGeneratorTask extends CompilerTask { | 30 class SsaCodeGeneratorTask extends CompilerTask { |
| 31 final JavaScriptBackend backend; | 31 final JavaScriptBackend backend; |
| 32 final Compiler compiler; | 32 final Compiler compiler; |
| 33 final SourceInformationStrategy sourceInformationFactory; | 33 final SourceInformationStrategy sourceInformationFactory; |
| 34 | 34 |
| 35 SsaCodeGeneratorTask(JavaScriptBackend backend, this.sourceInformationFactory) | 35 SsaCodeGeneratorTask(JavaScriptBackend backend, this.sourceInformationFactory) |
| (...skipping 2661 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2697 void emitIsViaInterceptor( | 2697 void emitIsViaInterceptor( |
| 2698 HIsViaInterceptor node, SourceInformation sourceInformation, | 2698 HIsViaInterceptor node, SourceInformation sourceInformation, |
| 2699 {bool negative: false}) { | 2699 {bool negative: false}) { |
| 2700 checkTypeViaProperty( | 2700 checkTypeViaProperty( |
| 2701 node.interceptor, node.typeExpression, sourceInformation, | 2701 node.interceptor, node.typeExpression, sourceInformation, |
| 2702 negative: negative); | 2702 negative: negative); |
| 2703 } | 2703 } |
| 2704 | 2704 |
| 2705 js.Expression generateReceiverOrArgumentTypeTest( | 2705 js.Expression generateReceiverOrArgumentTypeTest( |
| 2706 HInstruction input, TypeMask checkedType) { | 2706 HInstruction input, TypeMask checkedType) { |
| 2707 ClassWorld classWorld = compiler.closedWorld; | 2707 ClosedWorld closedWorld = compiler.closedWorld; |
| 2708 TypeMask inputType = input.instructionType; | 2708 TypeMask inputType = input.instructionType; |
| 2709 // Figure out if it is beneficial to turn this into a null check. | 2709 // Figure out if it is beneficial to turn this into a null check. |
| 2710 // V8 generally prefers 'typeof' checks, but for integers and | 2710 // V8 generally prefers 'typeof' checks, but for integers and |
| 2711 // indexable primitives we cannot compile this test into a single | 2711 // indexable primitives we cannot compile this test into a single |
| 2712 // typeof check so the null check is cheaper. | 2712 // typeof check so the null check is cheaper. |
| 2713 bool isIntCheck = checkedType.containsOnlyInt(classWorld); | 2713 bool isIntCheck = checkedType.containsOnlyInt(closedWorld); |
| 2714 bool turnIntoNumCheck = isIntCheck && input.isIntegerOrNull(compiler); | 2714 bool turnIntoNumCheck = isIntCheck && input.isIntegerOrNull(compiler); |
| 2715 bool turnIntoNullCheck = !turnIntoNumCheck && | 2715 bool turnIntoNullCheck = !turnIntoNumCheck && |
| 2716 (checkedType.nullable() == inputType) && | 2716 (checkedType.nullable() == inputType) && |
| 2717 (isIntCheck || | 2717 (isIntCheck || |
| 2718 checkedType.satisfies(helpers.jsIndexableClass, classWorld)); | 2718 checkedType.satisfies(helpers.jsIndexableClass, closedWorld)); |
| 2719 | 2719 |
| 2720 if (turnIntoNullCheck) { | 2720 if (turnIntoNullCheck) { |
| 2721 use(input); | 2721 use(input); |
| 2722 return new js.Binary("==", pop(), new js.LiteralNull()) | 2722 return new js.Binary("==", pop(), new js.LiteralNull()) |
| 2723 .withSourceInformation(input.sourceInformation); | 2723 .withSourceInformation(input.sourceInformation); |
| 2724 } else if (isIntCheck && !turnIntoNumCheck) { | 2724 } else if (isIntCheck && !turnIntoNumCheck) { |
| 2725 // input is !int | 2725 // input is !int |
| 2726 checkBigInt(input, '!==', input.sourceInformation); | 2726 checkBigInt(input, '!==', input.sourceInformation); |
| 2727 return pop(); | 2727 return pop(); |
| 2728 } else if (turnIntoNumCheck || checkedType.containsOnlyNum(classWorld)) { | 2728 } else if (turnIntoNumCheck || checkedType.containsOnlyNum(closedWorld)) { |
| 2729 // input is !num | 2729 // input is !num |
| 2730 checkNum(input, '!==', input.sourceInformation); | 2730 checkNum(input, '!==', input.sourceInformation); |
| 2731 return pop(); | 2731 return pop(); |
| 2732 } else if (checkedType.containsOnlyBool(classWorld)) { | 2732 } else if (checkedType.containsOnlyBool(closedWorld)) { |
| 2733 // input is !bool | 2733 // input is !bool |
| 2734 checkBool(input, '!==', input.sourceInformation); | 2734 checkBool(input, '!==', input.sourceInformation); |
| 2735 return pop(); | 2735 return pop(); |
| 2736 } else if (checkedType.containsOnlyString(classWorld)) { | 2736 } else if (checkedType.containsOnlyString(closedWorld)) { |
| 2737 // input is !string | 2737 // input is !string |
| 2738 checkString(input, '!==', input.sourceInformation); | 2738 checkString(input, '!==', input.sourceInformation); |
| 2739 return pop(); | 2739 return pop(); |
| 2740 } | 2740 } |
| 2741 reporter.internalError(input, 'Unexpected check: $checkedType.'); | 2741 reporter.internalError(input, 'Unexpected check: $checkedType.'); |
| 2742 return null; | 2742 return null; |
| 2743 } | 2743 } |
| 2744 | 2744 |
| 2745 void visitTypeConversion(HTypeConversion node) { | 2745 void visitTypeConversion(HTypeConversion node) { |
| 2746 if (node.isArgumentTypeCheck || node.isReceiverTypeCheck) { | 2746 if (node.isArgumentTypeCheck || node.isReceiverTypeCheck) { |
| 2747 ClassWorld classWorld = compiler.closedWorld; | 2747 ClosedWorld closedWorld = compiler.closedWorld; |
| 2748 // An int check if the input is not int or null, is not | 2748 // An int check if the input is not int or null, is not |
| 2749 // sufficient for doing an argument or receiver check. | 2749 // sufficient for doing an argument or receiver check. |
| 2750 assert(compiler.options.trustTypeAnnotations || | 2750 assert(compiler.options.trustTypeAnnotations || |
| 2751 !node.checkedType.containsOnlyInt(classWorld) || | 2751 !node.checkedType.containsOnlyInt(closedWorld) || |
| 2752 node.checkedInput.isIntegerOrNull(compiler)); | 2752 node.checkedInput.isIntegerOrNull(compiler)); |
| 2753 js.Expression test = generateReceiverOrArgumentTypeTest( | 2753 js.Expression test = generateReceiverOrArgumentTypeTest( |
| 2754 node.checkedInput, node.checkedType); | 2754 node.checkedInput, node.checkedType); |
| 2755 js.Block oldContainer = currentContainer; | 2755 js.Block oldContainer = currentContainer; |
| 2756 js.Statement body = new js.Block.empty(); | 2756 js.Statement body = new js.Block.empty(); |
| 2757 currentContainer = body; | 2757 currentContainer = body; |
| 2758 if (node.isArgumentTypeCheck) { | 2758 if (node.isArgumentTypeCheck) { |
| 2759 generateThrowWithHelper( | 2759 generateThrowWithHelper( |
| 2760 helpers.throwIllegalArgumentException, node.checkedInput, | 2760 helpers.throwIllegalArgumentException, node.checkedInput, |
| 2761 sourceInformation: node.sourceInformation); | 2761 sourceInformation: node.sourceInformation); |
| (...skipping 144 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2906 node.dartType == (node.dartType.element as ClassElement).thisType); | 2906 node.dartType == (node.dartType.element as ClassElement).thisType); |
| 2907 registry.registerInstantiatedClass(coreClasses.listClass); | 2907 registry.registerInstantiatedClass(coreClasses.listClass); |
| 2908 push(new js.ArrayInitializer(arguments) | 2908 push(new js.ArrayInitializer(arguments) |
| 2909 .withSourceInformation(node.sourceInformation)); | 2909 .withSourceInformation(node.sourceInformation)); |
| 2910 } | 2910 } |
| 2911 } | 2911 } |
| 2912 | 2912 |
| 2913 bool typeVariableAccessNeedsSubstitution( | 2913 bool typeVariableAccessNeedsSubstitution( |
| 2914 TypeVariableElement element, TypeMask receiverMask) { | 2914 TypeVariableElement element, TypeMask receiverMask) { |
| 2915 ClassElement cls = element.enclosingClass; | 2915 ClassElement cls = element.enclosingClass; |
| 2916 ClassWorld classWorld = compiler.closedWorld; | 2916 ClosedWorld closedWorld = compiler.closedWorld; |
| 2917 | 2917 |
| 2918 // See if the receiver type narrows the set of classes to ones that can be | 2918 // See if the receiver type narrows the set of classes to ones that can be |
| 2919 // indexed. | 2919 // indexed. |
| 2920 // TODO(sra): Currently the only convenient query is [singleClass]. We | 2920 // TODO(sra): Currently the only convenient query is [singleClass]. We |
| 2921 // should iterate over all the concrete classes in [receiverMask]. | 2921 // should iterate over all the concrete classes in [receiverMask]. |
| 2922 ClassElement receiverClass = receiverMask.singleClass(classWorld); | 2922 ClassElement receiverClass = receiverMask.singleClass(closedWorld); |
| 2923 if (receiverClass != null) { | 2923 if (receiverClass != null) { |
| 2924 if (backend.rti.isTrivialSubstitution(receiverClass, cls)) { | 2924 if (backend.rti.isTrivialSubstitution(receiverClass, cls)) { |
| 2925 return false; | 2925 return false; |
| 2926 } | 2926 } |
| 2927 } | 2927 } |
| 2928 | 2928 |
| 2929 if (classWorld.isUsedAsMixin(cls)) return true; | 2929 if (closedWorld.isUsedAsMixin(cls)) return true; |
| 2930 | 2930 |
| 2931 return classWorld.anyStrictSubclassOf(cls, (ClassElement subclass) { | 2931 return closedWorld.anyStrictSubclassOf(cls, (ClassElement subclass) { |
| 2932 return !backend.rti.isTrivialSubstitution(subclass, cls); | 2932 return !backend.rti.isTrivialSubstitution(subclass, cls); |
| 2933 }); | 2933 }); |
| 2934 } | 2934 } |
| 2935 | 2935 |
| 2936 void visitReadTypeVariable(HReadTypeVariable node) { | 2936 void visitReadTypeVariable(HReadTypeVariable node) { |
| 2937 TypeVariableElement element = node.dartType.element; | 2937 TypeVariableElement element = node.dartType.element; |
| 2938 Element helperElement = helpers.convertRtiToRuntimeType; | 2938 Element helperElement = helpers.convertRtiToRuntimeType; |
| 2939 registry.registerStaticUse( | 2939 registry.registerStaticUse( |
| 2940 new StaticUse.staticInvoke(helperElement, CallStructure.ONE_ARG)); | 2940 new StaticUse.staticInvoke(helperElement, CallStructure.ONE_ARG)); |
| 2941 | 2941 |
| (...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2999 registry.registerStaticUse(new StaticUse.staticInvoke( | 2999 registry.registerStaticUse(new StaticUse.staticInvoke( |
| 3000 helper, new CallStructure.unnamed(argumentCount))); | 3000 helper, new CallStructure.unnamed(argumentCount))); |
| 3001 return backend.emitter.staticFunctionAccess(helper); | 3001 return backend.emitter.staticFunctionAccess(helper); |
| 3002 } | 3002 } |
| 3003 | 3003 |
| 3004 @override | 3004 @override |
| 3005 void visitRef(HRef node) { | 3005 void visitRef(HRef node) { |
| 3006 visit(node.value); | 3006 visit(node.value); |
| 3007 } | 3007 } |
| 3008 } | 3008 } |
| OLD | NEW |